基于stm32f103zet6的FAT16文件系統(tǒng)學(xué)習(xí)3(初步分析ff9a)
文件系統(tǒng)分析了兩天,自己都不知道入沒入門,現(xiàn)在就把這兩天分析的結(jié)果總結(jié)一下吧!
一、res = f_mount(0,&fs);
首先是掛接根文件系統(tǒng),為什么要掛接根文件系統(tǒng)內(nèi)容?因為根文件系統(tǒng)里面會對我們的SD卡進(jìn)行初始化,除此之外
f_mount函數(shù)可以實現(xiàn)在FatFs模塊上注冊/ 注銷一個工作區(qū)。 在使用任何其他文件函數(shù)之前,必須使用該函數(shù)為每個
卷注冊一個工作區(qū)。要注銷一個工作區(qū),只要指定FileSystemObject 為NULL即可,然后該工作區(qū)可以被丟
棄。 該函數(shù)只初始化給定的工作區(qū),以及將該工作區(qū)的地址注冊到內(nèi)部表中,不訪問磁盤I/O 層。卷裝入過程是在
f_mount函數(shù)后或存儲介質(zhì)改變后的第一次文件訪問時完成的。
接著分析這個函數(shù)是怎么定義的:
/********************************************************************************
*函數(shù)名稱:f_mount
*函數(shù)說明:用于掛載/卸載磁盤驅(qū)動器
*輸入?yún)?shù):vol:Logicaldrivenumbertobemounted/unmounted
*fs:Pointertonewfilesystemobject(NULLforunmount)
*返回參數(shù):掛載標(biāo)識FR_OK表示成功!
*注意事項:無
*********************************************************************************/
FRESULTf_mount(
BYTEvol,/*Logicaldrivenumbertobemounted/unmounted*/
FATFS*fs/**/
)
{
FATFS*rfs;
if(vol>=_VOLUMES)/*Checkifthedrivenumberisvalid*///如果
returnFR_INVALID_DRIVE;
rfs=FatFs[vol];/*Getcurrentfsobject*/
if(rfs){
#if_FS_LOCK
clear_lock(rfs);
#endif
#if_FS_REENTRANT/*Discardsyncobjectofthecurrentvolume*/
if(!ff_del_syncobj(rfs->sobj))returnFR_INT_ERR;
#endif
rfs->fs_type=0;/*Clearoldfsobject*/
}
if(fs){
fs->fs_type=0;/*Clearnewfsobject*/
#if_FS_REENTRANT/*Createsyncobjectforthenewvolume*/
if(!ff_cre_syncobj(vol,&fs->sobj))returnFR_INT_ERR;
#endif
}
FatFs[vol]=fs;/*Registernewfsobject*/
returnFR_OK;
}
1、vol = 0,fs = &fs 首先看這個rfs = FatFs[vol];/* Get current fs object */
2、接著我們看到這個 FATFS *FatFs[_VOLUMES],從而知道了FatFs[vol]指向的是個 FATFS類型的結(jié)構(gòu)體
那么到底FATFS的原型是什么呢?看下面!這是去除宏定義精簡之后的結(jié)構(gòu)體成員。
typedefstruct{
BYTEfs_type;/*FATsub-type(0:Notmounted)*/
BYTEdrv;/*Physicaldrivenumber*/
BYTEcsize;/*Sectorspercluster(1,2,4...128)*/
BYTEn_fats;/*NumberofFATcopies(1,2)*/
BYTEwflag;/*win[]dirtyflag(1:mustbewrittenback)*/
BYTEfsi_flag;/*fsinfodirtyflag(1:mustbewrittenback)*/
WORDid;/*FilesystemmountID*/
WORDn_rootdir;/*Numberofrootdirectoryentries(FAT12/16)*/
DWORDlast_clust;/*Lastallocatedcluster*/
DWORDfree_clust;/*Numberoffreeclusters*/
DWORDfsi_sector;/*fsinfosector(FAT32)*/
DWORDn_fatent;/*NumberofFATentries(=numberofclusters+2)*/
DWORDfsize;/*SectorsperFAT*/
DWORDfatbase;/*FATstartsector*/
DWORDdirbase;/*Rootdirectorystartsector(FAT32:Cluster#)*/
DWORDdatabase;/*Datastartsector*/
DWORDwinsect;/*Currentsectorappearinginthewin[]*/
BYTEwin[_MAX_SS];/*DiskaccesswindowforDirectory,FAT(andDataontinycfg)*///存放的就是我們讀SD卡第一扇區(qū)的內(nèi)容
}FATFS;
到這里我們就可以推斷出rfs也應(yīng)該是FATFS*類型的,所以可以通過rfs ->drv的方法去訪問文件系統(tǒng)的各項參數(shù)了。下面驗證一下:
3、確實在ff.c文件的f_mount函數(shù)里面定義了一個這樣的變量
FATFS *rfs;
從而驗證了我們的猜想。總結(jié)一下這幾句話的作用
if (vol >= _VOLUMES)/* Check if the drive number is valid */
return FR_INVALID_DRIVE;
rfs = FatFs[vol];/* Get current fs object */
我的理解:掛載的目的就是為了給rfs分配地址,假如沒有使用到內(nèi)存管理malloc的話,使用這種方法也可以的。但是這里的rfs還是NULL的。
4、接著我們可以看看下面的代碼,這也是去除了相關(guān)宏定義之后的代碼
if(rfs){
rfs->fs_type=0;/*Clearoldfsobject*/
}
if(fs){
fs->fs_type=0;/*Clearnewfsobject*/
}
FatFs[vol]=fs;/*Registernewfsobject*/
returnFR_OK;
分析:
a、非零為真,那么在這里rfs這個地址肯定是空的,因為給rfs賦值就是NULL。
b、那么fs又是什么東西呢?看傳入?yún)?shù)可以知道:fs = &fs ,因為之前這個是定義過了的:FATFS *rfs;。
明顯的知道我們傳入的fs不為NULL的,所以自然執(zhí)行了fs->fs_type = 0;將類型清零,只不過,這個類型是
我在主函數(shù)里面使用的(新的),那么清零到底表示什么意思呢?其實是有說明的FAT sub-type (0:Not mounted)
這就是表示卸載的意思!
c、好的,卸載掉舊的,那么就可以注冊新的了, FatFs[vol] = fs;/* Register new fs object */
總結(jié)一下這上面幾行代碼的作用:
我的理解:就是實現(xiàn)了掛載文件系統(tǒng)的目標(biāo)文件,也就是給它分配了一個地址空間。而FatFs[vol]這個指針會被其他的ff.c里面的函數(shù)調(diào)用的
二、接著我們分析一下如何實現(xiàn)打開一個文件的
1、首先看一看main里面的調(diào)用
res = f_open(&fdst, "0:/test.txt", FA_CREATE_ALWAYS | FA_WRITE);
然后查看函數(shù)定義:這是去除無關(guān)宏定義之后的代碼: