0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

文件系統(tǒng)FatFs的移植

汽車電子技術(shù) ? 來源:滑小稽筆記 ? 作者:電子技術(shù)園地 ? 2023-03-01 14:38 ? 次閱讀

21.1 文件系統(tǒng)概述

21.1.1 簡介

FATFS是一個完全免費開源的FAT文件系統(tǒng)模塊,專門為小型的嵌入式系統(tǒng)而設(shè)計。它完全用標(biāo)準(zhǔn)C語言編寫,所以具有良好的硬件平臺獨立性,甚至可以移植到8位的單片機上而只需做簡單的修改。它支持FAT12、FAT16和FAT32,支持多個存儲媒介;有獨立的緩沖區(qū),可以對多個文件進行讀/寫,并特別對8位單片機和16位單片機做了優(yōu)化。

FATFS的特點有:

(1)Windows兼容的FAT文件系統(tǒng)(支持FAT12/FAT16/FAT32)

(2)與平臺無關(guān),移植簡單

(3)代碼量少、效率高

(4)多種配置選項

(5)支持多卷(物理驅(qū)動器或分區(qū),最多10個卷)

(6)多個ANSI/OEM代碼頁包括DBCS

(7)支持長文件名、ANSI/OEM或Unicode

(8)支持RTOS

(9)支持多種扇區(qū)大小

(10)只讀、最小化的API和I/O緩沖區(qū)等

FATFS的這些特點,加上免費、開源的原則,使得FATFS應(yīng)用非常廣泛。FATFS模塊的層次結(jié)構(gòu)如下圖所示。

圖片

最頂層是應(yīng)用層,使用者無需理會FATFS的內(nèi)部結(jié)構(gòu)和復(fù)雜的FAT協(xié)議,只需要調(diào)用FATFS模塊提供給用戶的一系列應(yīng)用接口函數(shù),如f_open,f_read,f_write和f_close等,就可以像在PC上讀寫文件那樣簡單。

中間層FATFS模塊,實現(xiàn)了FAT文件讀寫協(xié)議。FATFS模塊提供的是ff.c和ff.h。除非有必要,使用者一般不用修改,使用時將頭文件直接包含進去即可。

需要我們編寫移植代碼的是FATFS模塊提供的底層接口,它包括存儲媒介讀寫接口和供給文件創(chuàng)建修改時間的實時時鐘。FATFS的源代碼用戶可以通過官網(wǎng):http://elm-chan.org/fsw/ff/00index_e.html下載到。目前最新的版本是R0.14,這里我們采用最新版本的FATFS為例來講解如何將文件系統(tǒng)移植到STM32中。

源代碼下載之后,進行解壓可以發(fā)現(xiàn)里面一共有兩個文件夾,doc和src,其中doc是對文件系統(tǒng)的描述,源碼都在src里面,其中,與平臺無關(guān)的是:

ffconf.h FATFS模塊配置文件

ff.h FATFS和應(yīng)用模塊公用的包含文件

ff.c FATFS模塊

diskio.h FATFS和diskI/O模塊公用的包含文件

interger.h 數(shù)據(jù)類型定義

option 可選的外部功能(比如支持中文等)

與平臺相關(guān)的代碼是:

diskio.c FATFS和diskI/O模塊接口層文件

FATFS模塊在移植的時候,我們一般只需要修改2個文件,即ffconf.h和diskio.c。FATFS模塊的所有配置項都是存放在ffconf.h里面,我們可以通過配置里面的一些選項,來滿足自己的需求。接下來我們介紹幾個重要的配置選項。

21.1.2 文件系統(tǒng)配置

(1)_FS_TINY:這個選項在R0.07版本中開始出現(xiàn),之前的版本都是以獨立的C文件出現(xiàn)(FATFS和TinyFATFS),有了這個選項之后,兩者整合在一起了,使用起來更方便。我們使用FATFS,所以把這個選項定義為0即可

(2)_FS_READONLY:這個用來配置是不是只讀,本章我們需要讀寫都用,所以這里設(shè)置為0即可

(3)_USE_STRFUNC:這個用來設(shè)置是否支持字符串類操作,比如f_putc,f_puts等,我們需要用到,故設(shè)置這里為1

(4)_USE_MKFS:這個用來定時是否使能格式化,本章需要用到,所以設(shè)置這里為1

(5)_USE_FASTSEEK:這個用來使能快速定位,我們設(shè)置為1,使能快速定位

(6)_USE_LABEL:這個用來設(shè)置是否支持磁盤盤符讀取與設(shè)置。設(shè)置為1,使能,就可以通過相關(guān)函數(shù)讀取或者設(shè)置磁盤的名字了

(7)_CODE_PAGE:這個用于設(shè)置語言類型,包括很多選項,我們這里設(shè)置為936,即簡體中文(GBK碼,需要c936.c文件支持,該文件在option文件夾)

(8)_USE_LFN:該選項用于設(shè)置是否支持長文件名,取值范圍為03。0,表示不支持長文件名,13是支持長文件名,但是存儲地方不一樣,這里使用3,通過ff_memalloc函數(shù)來動態(tài)分配長文件名的存儲區(qū)域

(9)_VOLUMES:用于設(shè)置FATFS支持的邏輯設(shè)備數(shù)目,我們設(shè)置為2,即支持2個設(shè)備

(10)_MAX_SS:扇區(qū)緩沖的最大值,一般設(shè)置為512

21.1.3 接口移植

(1)磁盤初始化

函數(shù)名稱 disk_initialize
函數(shù)原型 DSTATUS disk_initialize (BYTE pdrv)
功能描述 初始化磁盤驅(qū)動器
函數(shù)參數(shù) pdrv:指定要初始化的邏輯驅(qū)動器編號,即盤符,取值范圍0~9
返回值 返回一個磁盤狀態(tài)作為結(jié)果
所在文件 diskio.c
備注 該函數(shù)用于初始化一個邏輯驅(qū)動器為讀寫數(shù)據(jù)做準(zhǔn)備

(2)檢查磁盤狀態(tài)

函數(shù)名稱 disk_status
函數(shù)原型 DSTATUS disk_ status (BYTE pdrv)
功能描述 查詢磁盤驅(qū)動器狀態(tài)
函數(shù)參數(shù) pdrv:指定要初始化的邏輯驅(qū)動器編號,即盤符,取值范圍0~9
返回值 返回下面標(biāo)志的組合STA_NOINIT:表明磁盤沒有初始化STA_NODISK:表示驅(qū)動器中沒有設(shè)備STA_PROTECTED:表示設(shè)備被寫保護
所在文件 diskio.c

(3)磁盤讀數(shù)據(jù)

函數(shù)名稱 disk_read
函數(shù)原型 DRESULT disk_read (BYTE pdrv, BYTE* buff, LBA_t sector, UINT count)
功能描述 從磁盤中讀取數(shù)據(jù)
函數(shù)參數(shù) pdrv:指定要初始化的邏輯驅(qū)動器編號,即盤符,取值范圍09*buff:指向存儲讀取數(shù)據(jù)字節(jié)數(shù)組的指針sector:指定開始扇區(qū)的邏輯塊上的地址count:指定要讀取的扇區(qū)數(shù),取值范圍1128
返回值 RES_OK:成功RES_ERROR:讀操作期間產(chǎn)生了錯誤且無法恢復(fù)RES_PARERR:非法參數(shù)RES_NOTRDY:磁盤驅(qū)動器沒有初始化
所在文件 diskio.c

(4)磁盤寫數(shù)據(jù)

函數(shù)名稱 disk_write
函數(shù)原型 DRESULT disk_write(BYTE pdrv, BYTE* buff, LBA_t sector, UINT count)
功能描述 從磁盤中寫入數(shù)據(jù)
函數(shù)參數(shù) pdrv:指定要初始化的邏輯驅(qū)動器編號,即盤符,取值范圍09*buff:指向存儲讀取數(shù)據(jù)字節(jié)數(shù)組的指針sector:指定開始扇區(qū)的邏輯塊上的地址count:指定要讀取的扇區(qū)數(shù),取值范圍1128
返回值 RES_OK:成功RES_ERROR:讀操作期間產(chǎn)生了錯誤且無法恢復(fù)RES_WRPRT:媒體被寫保護RES_PARERR:非法參數(shù)RES_NOTRDY:磁盤驅(qū)動器沒有初始化
所在文件 diskio.c

(5)磁盤雜項功能

函數(shù)名稱 disk_ioctl
函數(shù)原型 DRESULT disk_ioctl (BYTE pdrv, BYTE cmd, void* buff)
功能描述 從磁盤中寫入數(shù)據(jù)
函數(shù)參數(shù) pdrv:指定要初始化的邏輯驅(qū)動器編號,即盤符,取值范圍0~9cmd:命令代碼*buff:指向參數(shù)緩沖區(qū)指針
返回值 RES_OK:成功RES_ERROR:讀操作期間產(chǎn)生了錯誤且無法恢復(fù)RES_WRPRT:媒體被寫保護RES_PARERR:非法參數(shù)RES_NOTRDY:磁盤驅(qū)動器沒有初始化
所在文件 diskio.c

21.2 FATFS代碼移植

21.2.1 diskio.c文件修改

(1)修改宏定義如下圖所示。

圖片

將14,15行代碼修改為

#define SD_CARD //SD卡,卷標(biāo)為0

#define EX_FLASH //外部flash,卷標(biāo)為1

(2)修改disk_status函數(shù)如下所示。

DSTATUS disk_status( BYTE pdrv )

{

return RES_OK;

}

(3)修改disk_initialize函數(shù)如下所示。

DSTATUS disk_initialize( BYTE pdrv )

{

int res ;

   switch( pdrv )

   {

          case SD_CARD     :      res = SD_Init() ;     break;      //初始化SD卡

          case EX_FLASH     :      W25QXX_Init();      break;//初始化外部FLASH

          case DEV_USB       :      break;

   }

   if( res )

          return STA_NOINIT ;

   else

          return 0 ;

}

(4)修改disk_read函數(shù)如下所示。

DRESULT disk_read( BYTE pdrv, BYTE *buff, LBA_t sector, UINT count )

{

int result;

   switch( pdrv )

   {

          //SD卡

          case SD_CARD      :

                 result = SD_ReadDisk( buff, sector, count ) ;

                 //讀出錯

                 while( result )

                 {

                        SD_Init() ;                                      //重新初始化SD卡

                        result = SD_ReadDisk( buff, sector, count ) ;

                 }

          break;

          //外部FLASH讀寫

          case EX_FLASH     :

                 for( ; count>0; count-- )

                 {

                        W25QXX_Read( buff, sector*512, 512 ) ;

                        sector++;

                        buff+=512;

                 }

          break;

          case DEV_USB       :      break;

   }

   if( result )

          return RES_ERROR ;

   else

          return RES_OK ;

}

(5)修改disk_write函數(shù)如下所示。

#if FF_FS_READONLY == 0

DRESULT disk_write( BYTE pdrv, const BYTE *buff, LBA_t sector, UINT count )

{

int result;

   switch ( pdrv )

   {

          //SD卡

          case SD_CARD :

                 result = SD_WriteDisk( ( u8* )buff, sector, count ) ;

                 //寫出錯

                 while( result )

                 {

                        SD_Init() ;                                 //重新初始化SD卡

                        result = SD_WriteDisk( ( u8* )buff, sector, count ) ;

                 }

                 break ;

          //外部FLASH

          case EX_FLASH :

                 for( ; count>0; count-- )

                 {

                        W25QXX_Write( ( u8* )buff, sector*512, 512 ) ;

                        sector ++ ;

                        buff += 512 ;

                 }

                 break ;

          //其他

          case DEV_USB :

                 break ;

   }

   if( result )

          return RES_ERROR ;

   else

          return RES_OK ;

}

#endif

(6)修改disk_ioctl函數(shù)如下所示。

DRESULT disk_ioctl( BYTE pdrv, BYTE cmd, void *buff )

{

DRESULT res;

   //SD卡

   if( pdrv==SD_CARD )

   {

          switch( cmd )

          {

                 case CTRL_SYNC :

                        res = RES_OK ;

                        break ;

                 case GET_SECTOR_SIZE :

                        *( DWORD* )buff = 512 ;

                        res = RES_OK;

                        break ;

                 case GET_BLOCK_SIZE :

                        *( WORD* )buff = SDCardInfo.CardBlockSize ;

                        res = RES_OK ;

                        break ;

                 case GET_SECTOR_COUNT :

                        *( DWORD* )buff = SDCardInfo.CardCapacity/512 ;

                        res = RES_OK ;

                        break;

                 default :

                        res = RES_PARERR ;

                        break ;

          }

   }

   //外部FLASH

   else if( pdrv==EX_FLASH )

   {

          switch( cmd )

          {

                 case CTRL_SYNC :

                        res = RES_OK ;

                        break ;

                 case GET_SECTOR_SIZE :

                        *( WORD* )buff = 512 ;

                        res = RES_OK ;

                        break ;

                 case GET_BLOCK_SIZE :

                        *( WORD* )buff = 8 ;

                        res = RES_OK ;

                        break ;

                 case GET_SECTOR_COUNT :

                        *( DWORD* )buff = 2048*12 ;

                        res = RES_OK ;

                        break ;

                 default :

                        res = RES_PARERR ;

                        break ;

          }

   }

   else

          res = RES_ERROR ;   //其他的不支持

   return res;

}

(7)由于新版的文件系統(tǒng)去掉了獲取時間函數(shù),所以這一個函數(shù)需要我們自己添加。

//獲取時間

DWORD get_fattime()

{

return 0 ;

}

21.2.2 ffsystem.c文件的修改

(1)內(nèi)存分配ff_memalloc

void* ff_memalloc( UINT msize )

{

return ( void* )mymalloc( SRAMIN, msize ) ;

}

(2)內(nèi)存釋放ff_memfree

void ff_memfree( void* mblock )

{

myfree( SRAMIN, mblock ) ;

}

21.2.3 exfuns.c與exfuns.h文件的創(chuàng)建

(1)創(chuàng)建exfuns.h文件,并輸入以下代碼。

#ifndef _EXFUNS_H

#define _EXFUNS_H

#include "sys.h"

#include "ff.h"

extern FATFS *fs[ FF_VOLUMES ] ;

extern FIL *file;

extern FIL *ftemp;

extern UINT br,bw;

extern FILINFO fileinfo;

extern DIR dir;

u8 exfuns_init( void ) ; //為exfuns申請內(nèi)存

u8 exf_getfree( u8 *drv, u32 *total, u32 *free ) ; //得到磁盤總?cè)萘亢褪S嗳萘?/p>

#endif

(2)創(chuàng)建exfuns.c文件,并輸入以下代碼。

#include "exfuns.h"

#include "malloc.h"

FATFS *fs[ FF_VOLUMES ] ; //邏輯磁盤工作區(qū)

FIL *file ; //文件1

FIL *ftemp ; //文件2

UINT br, bw ; //讀寫變量

FILINFO fileinfo ; //文件信息

DIR dir ; //目錄

u8 *fatbuf ; //SD卡數(shù)據(jù)緩存區(qū)

u8 exfuns_init()

{

u8 i;

   for( i=0; i

}

u8 exf_getfree( u8 *drv, u32 *total, u32 *free )

{

FATFS *fs1;

   u8 res;

   u32 fre_clust=0, fre_sect=0, tot_sect=0;

   //得到磁盤信息及空閑簇數(shù)量

   res = ( u32 )f_getfree( ( const TCHAR* )drv, ( DWORD* )&fre_clust, &fs1 ) ;

   if( res==0 )

   {                                                                              

          tot_sect =( fs1->n_fatent-2 )*fs1->csize ;          //得到總扇區(qū)數(shù)

          fre_sect = fre_clust*fs1->csize ;                         //得到空閑扇區(qū)數(shù)

          //扇區(qū)大小不是512字節(jié),則轉(zhuǎn)換為512字節(jié)

          #if FF_MAX_SS!=512

                 tot_sect*=fs1->ssize/512;

                 fre_sect*=fs1->ssize/512;

          #endif

          *total=tot_sect>>1 ;                                                                    //單位為KB

          *free=fre_sect>>1 ;                                            //單位為KB

}

return res;

}

注:如果SD卡文件系統(tǒng)不能正確掛載則需要修改SD卡驅(qū)動文件中的兩個參數(shù),如下圖所示。

圖片

21.3 內(nèi)存管理

21.3.1 內(nèi)存管理簡介

內(nèi)存管理,是指軟件運行時對計算機內(nèi)存資源的分配和使用的技術(shù)。其最主要的目的是如何高效,快速的分配,并且在適當(dāng)?shù)臅r候釋放和回收內(nèi)存資源。內(nèi)存管理的實現(xiàn)方法有很多種,他們其實最終都是要實現(xiàn)2個函數(shù):malloc和free;malloc函數(shù)用于內(nèi)存申請,free函數(shù)用于內(nèi)存釋放。

這一部分我們使用了一種比較簡單的辦法來實現(xiàn):分塊式內(nèi)存管理。下面我們介紹一下該方法的實現(xiàn)原理,如下圖所示。

圖片

從上圖可以看出,分塊式內(nèi)存管理由內(nèi)存池和內(nèi)存管理表兩部分組成。內(nèi)存池被等分為n塊,對應(yīng)的內(nèi)存管理表,大小也為n,內(nèi)存管理表的每一個項對應(yīng)內(nèi)存池的一塊內(nèi)存。內(nèi)存管理表的項值代表的意義為:當(dāng)該項值為0的時候,代表對應(yīng)的內(nèi)存塊未被占用,當(dāng)該項值非零的時候,代表該項對應(yīng)的內(nèi)存塊已經(jīng)被占用,其數(shù)值則代表被連續(xù)占用的內(nèi)存塊數(shù)。比如某項值為10,那么說明包括本項對應(yīng)的內(nèi)存塊在內(nèi),總共分配了10個內(nèi)存塊給外部的某個指針。

內(nèi)存分配方向如圖所示,是從頂?shù)降椎姆峙浞较颉<词紫葟淖钅┒碎_始找空內(nèi)存。當(dāng)內(nèi)存管理剛初始化的時候,內(nèi)存表全部清零,表示沒有任何內(nèi)存塊被占用。

21.3.2 分配原理

當(dāng)指針p調(diào)用malloc申請內(nèi)存的時候,先判斷p要分配的內(nèi)存塊數(shù)m,然后從第n項開始,向下查找,直到找到m塊連續(xù)的空內(nèi)存塊(即對應(yīng)內(nèi)存管理表項為0),然后將這m個內(nèi)存管理表項的值都設(shè)置為m(標(biāo)記被占用),最后,把最后的這個空內(nèi)存塊的地址返回指針p,完成一次分配。注意,如果當(dāng)內(nèi)存不夠的時候(找到最后也沒找到連續(xù)的m塊空閑內(nèi)存),則返回NULL給p,表示分配失敗。

21.3.3 釋放原理

當(dāng)p申請的內(nèi)存用完,需要釋放的時候,調(diào)用free函數(shù)實現(xiàn)。free函數(shù)先判斷p指向的內(nèi)存地址所對應(yīng)的內(nèi)存塊,然后找到對應(yīng)的內(nèi)存管理表項目,得到p所占用的內(nèi)存塊數(shù)目m(內(nèi)存管理表項目的值就是所分配內(nèi)存塊的數(shù)目),將這m個內(nèi)存管理表項目的值都清零,標(biāo)記釋放,完成一次內(nèi)存釋放。

21.3.4 源代碼實現(xiàn)

(1)創(chuàng)建malloc.h文件,并輸入以下代碼。

/*********************************************************************************************************
                內(nèi)    存    管    理    文    件
*********************************************************************************************************/
#ifndef _MALLOC_H_
#define _MALLOC_H_


#include "sys.h"
/*********************************************************************************************************
                數(shù)    據(jù)    結(jié)    構(gòu)    定    義
*********************************************************************************************************/
//定義兩個內(nèi)存池
#define SRAMIN   0    //內(nèi)部內(nèi)存池
#define SRAMBANK   1  //定義支持的SRAM塊數(shù)
//mem1內(nèi)存參數(shù)設(shè)定
#define MEM1_BLOCK_SIZE      32                                    //內(nèi)存塊大小為32字節(jié)
#define MEM1_MAX_SIZE      40*1024                                  //最大管理內(nèi)存40K
#define MEM1_ALLOC_TABLE_SIZE  MEM1_MAX_SIZE/MEM1_BLOCK_SIZE                      //內(nèi)存表大小
//內(nèi)存管理控制器
struct _m_mallco_dev
{
  void ( *init )( u8 ) ;              //初始化
  u8 ( *perused )( u8 ) ;              //內(nèi)存使用率
  u8   *membase[ SRAMBANK ] ;            //內(nèi)存池 管理SRAMBANK個區(qū)域的內(nèi)存
  u16 *memmap[ SRAMBANK ] ;            //內(nèi)存管理狀態(tài)表
  u8  memrdy[ SRAMBANK ] ;            //內(nèi)存管理是否就緒
};
extern struct _m_mallco_dev mallco_dev;                                  //在mallco.c里面定義
/*********************************************************************************************************
                  函    數(shù)    列    表
*********************************************************************************************************/
void my_mem_init( u8 memx ) ;                                      //內(nèi)存管理初始化函數(shù)
u8 my_mem_perused( u8 memx ) ;                                      //獲得內(nèi)存使用率
void myfree( u8 memx, void *ptr ) ;                                    //內(nèi)存釋放
void *mymalloc( u8 memx, u32 size ) ;                                  //內(nèi)存分配
void *myrealloc( u8 memx, void *ptr, u32 size ) ;                            //重新分配內(nèi)存


#endif

(2)創(chuàng)建malloc.c文件,并輸入以下代碼。

/*********************************************************************************************************
                內(nèi)    存    管    理    程    序
*********************************************************************************************************/
#include "malloc.h"      


__align(32) u8 mem1base[ MEM1_MAX_SIZE ] ;                                //內(nèi)部SRAM內(nèi)存池
u16 mem1mapbase[ MEM1_ALLOC_TABLE_SIZE ] ;                                //內(nèi)部SRAM內(nèi)存池MAP
const u32 memtblsize[ SRAMBANK ]={ MEM1_ALLOC_TABLE_SIZE } ;                      //內(nèi)存表大小
const u32 memblksize[ SRAMBANK ]={ MEM1_BLOCK_SIZE } ;                          //內(nèi)存分塊大小
const u32 memsize[ SRAMBANK ]={ MEM1_MAX_SIZE } ;                            //內(nèi)存總大小
//內(nèi)存管理控制器
struct _m_mallco_dev mallco_dev=
{
  my_mem_init,                  //內(nèi)存初始化
  my_mem_perused,                  //內(nèi)存使用率
  mem1base,                    //內(nèi)存池
  mem1mapbase,                  //內(nèi)存管理狀態(tài)表
  0,                        //內(nèi)存管理未就緒
};
/***************************************************
Name    :mymemcpy
Function  :復(fù)制內(nèi)存
Paramater  :
      *des:目的地址
      *src:源地址
      n:需要復(fù)制的內(nèi)存長度(字節(jié)為單位)
Return    :None
***************************************************/
void mymemcpy( void *des, void *src, u32 n )
{
    u8 *xdes=des ;
  u8 *xsrc=src ;
    while( n-- )
    *xdes++ =*xsrc++ ;
}
/***************************************************
Name    :mymemset
Function  :設(shè)置內(nèi)存
Paramater  :
      *s:內(nèi)存首地址
      c:要設(shè)置的值
      count:需要設(shè)置的內(nèi)存大小(字節(jié)為單位)
Return    :None
***************************************************/
void mymemset( void *s, u8 c, u32 count )
{
    u8 *xs=s ;
    while( count-- )
    *xs++ = c ;
}
/***************************************************
Name    :my_mem_perused
Function  :內(nèi)存分配
Paramater  :
      memx:所屬內(nèi)存塊
      size:要分配的內(nèi)存大小(字節(jié))
Return    :
      0xFFFFFFFF:代表錯誤
      其他:內(nèi)存偏移地址
***************************************************/
u32 my_mem_malloc( u8 memx, u32 size )
{  
    signed long offset=0 ;
  u32 i, nmemb, cmemb=0 ;                                        //需要的內(nèi)存塊數(shù)+連續(xù)空內(nèi)存塊數(shù)
  //未初始化,先執(zhí)行初始化
    if( !mallco_dev.memrdy[ memx ] )
    mallco_dev.init( memx ) ;
  //不需要分配
    if( size==0 )
    return 0xFFFFFFFF ;
    nmemb = size/memblksize[ memx ] ;                                  //獲取需要分配的連續(xù)內(nèi)存塊數(shù)
    if( size%memblksize[ memx ] )
    nmemb ++ ;
  //搜索整個內(nèi)存控制區(qū)
    for( offset=memtblsize[ memx ]-1; offset>=0; offset-- )
    {
    //連續(xù)空內(nèi)存塊數(shù)增加
    if( !mallco_dev.memmap[ memx ][ offset ] )
      cmemb ++ ;
    else
      cmemb = 0 ;                                          //連續(xù)內(nèi)存塊清零
    //找到了連續(xù)nmemb個空內(nèi)存塊
    if( cmemb==nmemb )
    {
      //標(biāo)注內(nèi)存塊非空
            for( i=0; i

21.4 實驗例程

實驗:利用FATFS R0.14掛載SD卡與W25Q128,并在LCD上顯示SD卡的已用容量與剩余容量。

#include "sys.h"
#include "delay.h"
#include "usart1.h"
#include "lcd.h"
#include "sdio_sdcard.h"
#include "w25q128.h"
#include "ff.h"
#include "exfuns.h"
#include "malloc.h"


int main()
{
  BYTE work[ FF_MAX_SS ] ;
  u32 total, free ;
  u8 res=0;
  u8 Str[ 30 ] ;
   STM32_Clock_Init( 9 ) ;                                        //系統(tǒng)時鐘設(shè)置
  SysTick_Init( 72 ) ;                                        //延時初始化
  USART1_Init( 72, 115200 ) ;                                      //串口初始化為115200
  LCD_Init() ;                                            //初始化LCD
  W25QXX_Init() ;                                            //初始化W25Q128
  my_mem_init( SRAMIN ) ;                                        //初始化內(nèi)部內(nèi)存池
  while( SD_Init() ) ;                                        //初始化SD卡
  exfuns_init() ;
  f_mount( fs[0], "0:", 1 ) ;                                      //掛載SD卡
  res = f_mount( fs[1], "1:", 1 ) ;                                  //掛載FLASH
  //FLASH 磁盤,FAT 文件系統(tǒng)錯誤,重新格式化 FLASH
  if( res==0x0D )
  {
    LCD_ShowString( 0, 0, "Flash Disk Formatting..." ) ;                      //格式化FLASH
    res = f_mkfs( "1:", 0, work, sizeof work ) ;                          //格式化 FLASH,1,盤符
    if(res==0)
    {
      f_setlabel( ( const TCHAR * )"1:FLASH" ) ;                          //設(shè)置Flash磁盤名:FLASH
      LCD_ShowString( 0, 0, "Flash Disk Format Finish ") ;                    //格式化完成
    }
    else
      LCD_ShowString( 0, 0, "Flash Disk Format Error " ) ;                    //格式化失敗
    delay_ms( 1000 ) ;
  }
  //得到SD卡的總?cè)萘亢褪S嗳萘?/span>
  while( exf_getfree( "0", &total, &free ) ) ;
  sprintf( ( char* )Str, "total=%03d MB, free=%03d MB", total>>10, free>>10 ) ;
  LCD_ShowString( 0, 40, Str ) ;
   while(1)
  {

  }
}
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 嵌入式系統(tǒng)
    +關(guān)注

    關(guān)注

    41

    文章

    3551

    瀏覽量

    129104
  • C語言
    +關(guān)注

    關(guān)注

    180

    文章

    7594

    瀏覽量

    135856
  • FATFS
    +關(guān)注

    關(guān)注

    0

    文章

    44

    瀏覽量

    18253
收藏 人收藏

    評論

    相關(guān)推薦

    基于STM32完成FATFS文件系統(tǒng)移植與運用

    這篇文章主要演示FATFS文件系統(tǒng)如何移植到自己的工程,并完成文件的讀寫。 因為SD卡采用的是SPI模擬時序,所以,其他單片機一樣可以照著移植
    的頭像 發(fā)表于 03-08 09:17 ?5847次閱讀
    基于STM32完成<b class='flag-5'>FATFS</b><b class='flag-5'>文件系統(tǒng)</b><b class='flag-5'>移植</b>與運用

    FATFS文件系統(tǒng)移植

    FatFs 是用于小型嵌入式系統(tǒng)的通用 FAT/exFAT 文件系統(tǒng)模塊。FatFs 模塊是按照 ANSI C (C89) 編寫的,與磁盤 I/O 層完全分離。因此它獨立于平臺。它可以
    的頭像 發(fā)表于 06-06 09:28 ?4494次閱讀
    <b class='flag-5'>FATFS</b><b class='flag-5'>文件系統(tǒng)</b><b class='flag-5'>移植</b>

    FATFS文件系統(tǒng)移植

    這是本人,整理的FATFS文件系統(tǒng)移植資料,很少的一部分,希望大家把自己搜到的比較好的資料,分享一下,本人正在學(xué)習(xí)fatfs文件系統(tǒng),希望和
    發(fā)表于 09-11 16:05

    介紹移植fatfs文件系統(tǒng)步驟

    就可以派上用場,fatfs是一小型的可裁剪免費文件系統(tǒng)移植起來比較方便,移植文件系統(tǒng)大大縮短開發(fā)周期,可以很方便的操作內(nèi)存,廢話不再多說
    發(fā)表于 11-23 07:19

    FatFs文件系統(tǒng)的原理是什么?如何對FATFS進行移植

    FatFs文件系統(tǒng)的原理是什么?FatFs文件系統(tǒng)移植方法是什么?如何實現(xiàn)eMMC卡中文件的讀
    發(fā)表于 11-25 07:52

    FatFs文件系統(tǒng)使用

    STM系列FatFs文件系統(tǒng)使用文件,希望對大家有幫助。
    發(fā)表于 11-06 18:10 ?8次下載

    Fatfs文件系統(tǒng)移植

    Fatfs文件系統(tǒng)移植)一、文件系統(tǒng)介紹二、移植條件、說明1、FatFs模塊在可
    發(fā)表于 11-15 18:51 ?22次下載
    <b class='flag-5'>Fatfs</b>(<b class='flag-5'>文件系統(tǒng)</b>的<b class='flag-5'>移植</b>)

    FATFS文件系統(tǒng)詳解

    采用的獨特的文件系統(tǒng)結(jié)構(gòu)CDFS:CDFS是大部分的光盤的文件系統(tǒng)exFATFATFS文件系統(tǒng)FATFS是一個完全免費開源的FAT 文件系統(tǒng)
    發(fā)表于 11-29 09:51 ?29次下載
    <b class='flag-5'>FATFS</b><b class='flag-5'>文件系統(tǒng)</b>詳解

    文件系統(tǒng)FatFs文件系統(tǒng)在嵌入式芯片LPC18XX上的移植

    文件系統(tǒng)FatFs文件系統(tǒng)在嵌入式芯片LPC18XX上的移植
    發(fā)表于 12-04 10:51 ?12次下載
    【<b class='flag-5'>文件系統(tǒng)</b>】<b class='flag-5'>FatFs</b><b class='flag-5'>文件系統(tǒng)</b>在嵌入式芯片LPC18XX上的<b class='flag-5'>移植</b>

    手把手教你在flash上移植fatfs文件系統(tǒng)(含實時操作系統(tǒng))

    文章目錄基于stm32f103系列MCU,在spi_flash上移植建立fatfs文件系統(tǒng)移植準(zhǔn)備FATFS
    發(fā)表于 12-09 12:51 ?30次下載
    手把手教你在flash上<b class='flag-5'>移植</b><b class='flag-5'>fatfs</b><b class='flag-5'>文件系統(tǒng)</b>(含實時操作<b class='flag-5'>系統(tǒng)</b>)

    FATFS文件系統(tǒng)原版文件下載

    FATFS文件系統(tǒng)原版文件下載
    發(fā)表于 06-25 09:02 ?0次下載

    STM32入門學(xué)習(xí)筆記之文件系統(tǒng)FatFs移植1

    FATFS是一個完全免費開源的FAT文件系統(tǒng)模塊,專門為小型的嵌入式系統(tǒng)而設(shè)計。它完全用標(biāo)準(zhǔn)C語言編寫,所以具有良好的硬件平臺獨立性,甚至可以移植到8位的單片機上而只需做簡單的修改。它
    的頭像 發(fā)表于 02-16 15:26 ?936次閱讀
    STM32入門學(xué)習(xí)筆記之<b class='flag-5'>文件系統(tǒng)</b><b class='flag-5'>FatFs</b>的<b class='flag-5'>移植</b>1

    STM32入門學(xué)習(xí)筆記之文件系統(tǒng)FatFs移植2

    FATFS是一個完全免費開源的FAT文件系統(tǒng)模塊,專門為小型的嵌入式系統(tǒng)而設(shè)計。它完全用標(biāo)準(zhǔn)C語言編寫,所以具有良好的硬件平臺獨立性,甚至可以移植到8位的單片機上而只需做簡單的修改。它
    的頭像 發(fā)表于 02-16 15:26 ?733次閱讀

    STM32入門學(xué)習(xí)筆記之文件系統(tǒng)FatFs移植3

    FATFS是一個完全免費開源的FAT文件系統(tǒng)模塊,專門為小型的嵌入式系統(tǒng)而設(shè)計。它完全用標(biāo)準(zhǔn)C語言編寫,所以具有良好的硬件平臺獨立性,甚至可以移植到8位的單片機上而只需做簡單的修改。它
    的頭像 發(fā)表于 02-16 15:27 ?1687次閱讀
    STM32入門學(xué)習(xí)筆記之<b class='flag-5'>文件系統(tǒng)</b><b class='flag-5'>FatFs</b>的<b class='flag-5'>移植</b>3

    STM32入門學(xué)習(xí)筆記之文件系統(tǒng)FatFs移植4

    FATFS是一個完全免費開源的FAT文件系統(tǒng)模塊,專門為小型的嵌入式系統(tǒng)而設(shè)計。它完全用標(biāo)準(zhǔn)C語言編寫,所以具有良好的硬件平臺獨立性,甚至可以移植到8位的單片機上而只需做簡單的修改。它
    的頭像 發(fā)表于 02-16 15:27 ?804次閱讀