21.1 文件系統(tǒng)概述
21.1.1 簡介
FATFS是一個(gè)完全免費(fèi)開源的FAT文件系統(tǒng)模塊,專門為小型的嵌入式系統(tǒng)而設(shè)計(jì)。它完全用標(biāo)準(zhǔn)C語言編寫,所以具有良好的硬件平臺(tái)獨(dú)立性,甚至可以移植到8位的單片機(jī)上而只需做簡單的修改。它支持FAT12、FAT16和FAT32,支持多個(gè)存儲(chǔ)媒介;有獨(dú)立的緩沖區(qū),可以對多個(gè)文件進(jìn)行讀/寫,并特別對8位單片機(jī)和16位單片機(jī)做了優(yōu)化。
FATFS的特點(diǎn)有:
(1)Windows兼容的FAT文件系統(tǒng)(支持FAT12/FAT16/FAT32)
(2)與平臺(tái)無關(guān),移植簡單
(3)代碼量少、效率高
(4)多種配置選項(xiàng)
(5)支持多卷(物理驅(qū)動(dòng)器或分區(qū),最多10個(gè)卷)
(6)多個(gè)ANSI/OEM代碼頁包括DBCS
(7)支持長文件名、ANSI/OEM或Unicode
(8)支持RTOS
(9)支持多種扇區(qū)大小
(10)只讀、最小化的API和I/O緩沖區(qū)等
FATFS的這些特點(diǎn),加上免費(fèi)、開源的原則,使得FATFS應(yīng)用非常廣泛。FATFS模塊的層次結(jié)構(gòu)如下圖所示。
最頂層是應(yīng)用層,使用者無需理會(huì)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模塊,實(shí)現(xiàn)了FAT文件讀寫協(xié)議。FATFS模塊提供的是ff.c和ff.h。除非有必要,使用者一般不用修改,使用時(shí)將頭文件直接包含進(jìn)去即可。
需要我們編寫移植代碼的是FATFS模塊提供的底層接口,它包括存儲(chǔ)媒介讀寫接口和供給文件創(chuàng)建修改時(shí)間的實(shí)時(shí)時(shí)鐘。FATFS的源代碼用戶可以通過官網(wǎng):http://elm-chan.org/fsw/ff/00index_e.html下載到。目前最新的版本是R0.14,這里我們采用最新版本的FATFS為例來講解如何將文件系統(tǒng)移植到STM32中。
源代碼下載之后,進(jìn)行解壓可以發(fā)現(xiàn)里面一共有兩個(gè)文件夾,doc和src,其中doc是對文件系統(tǒng)的描述,源碼都在src里面,其中,與平臺(tái)無關(guān)的是:
ffconf.h FATFS模塊配置文件
ff.h FATFS和應(yīng)用模塊公用的包含文件
ff.c FATFS模塊
diskio.h FATFS和diskI/O模塊公用的包含文件
interger.h 數(shù)據(jù)類型定義
option 可選的外部功能(比如支持中文等)
與平臺(tái)相關(guān)的代碼是:
diskio.c FATFS和diskI/O模塊接口層文件
FATFS模塊在移植的時(shí)候,我們一般只需要修改2個(gè)文件,即ffconf.h和diskio.c。FATFS模塊的所有配置項(xiàng)都是存放在ffconf.h里面,我們可以通過配置里面的一些選項(xiàng),來滿足自己的需求。接下來我們介紹幾個(gè)重要的配置選項(xiàng)。
21.1.2 文件系統(tǒng)配置
(1)_FS_TINY:這個(gè)選項(xiàng)在R0.07版本中開始出現(xiàn),之前的版本都是以獨(dú)立的C文件出現(xiàn)(FATFS和TinyFATFS),有了這個(gè)選項(xiàng)之后,兩者整合在一起了,使用起來更方便。我們使用FATFS,所以把這個(gè)選項(xiàng)定義為0即可
(2)_FS_READONLY:這個(gè)用來配置是不是只讀,本章我們需要讀寫都用,所以這里設(shè)置為0即可
(3)_USE_STRFUNC:這個(gè)用來設(shè)置是否支持字符串類操作,比如f_putc,f_puts等,我們需要用到,故設(shè)置這里為1
(4)_USE_MKFS:這個(gè)用來定時(shí)是否使能格式化,本章需要用到,所以設(shè)置這里為1
(5)_USE_FASTSEEK:這個(gè)用來使能快速定位,我們設(shè)置為1,使能快速定位
(6)_USE_LABEL:這個(gè)用來設(shè)置是否支持磁盤盤符讀取與設(shè)置。設(shè)置為1,使能,就可以通過相關(guān)函數(shù)讀取或者設(shè)置磁盤的名字了
(7)_CODE_PAGE:這個(gè)用于設(shè)置語言類型,包括很多選項(xiàng),我們這里設(shè)置為936,即簡體中文(GBK碼,需要c936.c文件支持,該文件在option文件夾)
(8)_USE_LFN:該選項(xiàng)用于設(shè)置是否支持長文件名,取值范圍為03。0,表示不支持長文件名,13是支持長文件名,但是存儲(chǔ)地方不一樣,這里使用3,通過ff_memalloc函數(shù)來動(dòng)態(tài)分配長文件名的存儲(chǔ)區(qū)域
(9)_VOLUMES:用于設(shè)置FATFS支持的邏輯設(shè)備數(shù)目,我們設(shè)置為2,即支持2個(gè)設(shè)備
(10)_MAX_SS:扇區(qū)緩沖的最大值,一般設(shè)置為512
21.1.3 接口移植
(1)磁盤初始化
函數(shù)名稱 | disk_initialize |
---|---|
函數(shù)原型 | DSTATUS disk_initialize (BYTE pdrv) |
功能描述 | 初始化磁盤驅(qū)動(dòng)器 |
函數(shù)參數(shù) | pdrv:指定要初始化的邏輯驅(qū)動(dòng)器編號,即盤符,取值范圍0~9 |
返回值 | 返回一個(gè)磁盤狀態(tài)作為結(jié)果 |
所在文件 | diskio.c |
備注 | 該函數(shù)用于初始化一個(gè)邏輯驅(qū)動(dòng)器為讀寫數(shù)據(jù)做準(zhǔn)備 |
(2)檢查磁盤狀態(tài)
函數(shù)名稱 | disk_status |
---|---|
函數(shù)原型 | DSTATUS disk_ status (BYTE pdrv) |
功能描述 | 查詢磁盤驅(qū)動(dòng)器狀態(tài) |
函數(shù)參數(shù) | pdrv:指定要初始化的邏輯驅(qū)動(dòng)器編號,即盤符,取值范圍0~9 |
返回值 | 返回下面標(biāo)志的組合STA_NOINIT:表明磁盤沒有初始化STA_NODISK:表示驅(qū)動(dòng)器中沒有設(shè)備STA_PROTECTED:表示設(shè)備被寫保護(hù) |
所在文件 | 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ū)動(dòng)器編號,即盤符,取值范圍0 |
返回值 | RES_OK:成功RES_ERROR:讀操作期間產(chǎn)生了錯(cuò)誤且無法恢復(fù)RES_PARERR:非法參數(shù)RES_NOTRDY:磁盤驅(qū)動(dòng)器沒有初始化 |
所在文件 | 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ū)動(dòng)器編號,即盤符,取值范圍0 |
返回值 | RES_OK:成功RES_ERROR:讀操作期間產(chǎn)生了錯(cuò)誤且無法恢復(fù)RES_WRPRT:媒體被寫保護(hù)RES_PARERR:非法參數(shù)RES_NOTRDY:磁盤驅(qū)動(dòng)器沒有初始化 |
所在文件 | diskio.c |
(5)磁盤雜項(xiàng)功能
函數(shù)名稱 | disk_ioctl |
---|---|
函數(shù)原型 | DRESULT disk_ioctl (BYTE pdrv, BYTE cmd, void* buff) |
功能描述 | 從磁盤中寫入數(shù)據(jù) |
函數(shù)參數(shù) | pdrv:指定要初始化的邏輯驅(qū)動(dòng)器編號,即盤符,取值范圍0~9cmd:命令代碼*buff:指向參數(shù)緩沖區(qū)指針 |
返回值 | RES_OK:成功RES_ERROR:讀操作期間產(chǎn)生了錯(cuò)誤且無法恢復(fù)RES_WRPRT:媒體被寫保護(hù)RES_PARERR:非法參數(shù)RES_NOTRDY:磁盤驅(qū)動(dòng)器沒有初始化 |
所在文件 | 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 ) ;
//讀出錯(cuò)
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 ) ;
//寫出錯(cuò)
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ù),所以這一個(gè)函數(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
{
//為磁盤i工作區(qū)申請內(nèi)存
fs[ i ] = ( FATFS* )mymalloc( SRAMIN, sizeof( FATFS ) ) ;
if( !fs[ i ] )
break ;
}
file = ( FIL* )mymalloc( SRAMIN, sizeof( FIL ) ) ; //為file申請內(nèi)存
ftemp = ( FIL* )mymalloc( SRAMIN, sizeof( FIL ) ) ; //為ftemp申請內(nèi)存
fatbuf = ( u8* )mymalloc( SRAMIN, 512 ) ; //為fatbuf申請內(nèi)存
//申請有一個(gè)失敗,即失敗
if( ( i==FF_VOLUMES )&&file&&ftemp&&fatbuf )
return 0 ;
else
return 1 ;
}
u8 exf_getfree( u8 *drv, u32 *total, u32 *free )
{
FATFS *fs1;
u8 res;
u32 fre_clust=0, fre_sect=0, tot_sect=0;
//得到磁盤信息及空閑簇?cái)?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ū)動(dòng)文件中的兩個(gè)參數(shù),如下圖所示。
21.3 內(nèi)存管理
21.3.1 內(nèi)存管理簡介
內(nèi)存管理,是指軟件運(yùn)行時(shí)對計(jì)算機(jī)內(nèi)存資源的分配和使用的技術(shù)。其最主要的目的是如何高效,快速的分配,并且在適當(dāng)?shù)臅r(shí)候釋放和回收內(nèi)存資源。內(nèi)存管理的實(shí)現(xiàn)方法有很多種,他們其實(shí)最終都是要實(shí)現(xiàn)2個(gè)函數(shù):malloc和free;malloc函數(shù)用于內(nèi)存申請,free函數(shù)用于內(nèi)存釋放。
這一部分我們使用了一種比較簡單的辦法來實(shí)現(xiàn):分塊式內(nèi)存管理。下面我們介紹一下該方法的實(shí)現(xiàn)原理,如下圖所示。
從上圖可以看出,分塊式內(nèi)存管理由內(nèi)存池和內(nèi)存管理表兩部分組成。內(nèi)存池被等分為n塊,對應(yīng)的內(nèi)存管理表,大小也為n,內(nèi)存管理表的每一個(gè)項(xiàng)對應(yīng)內(nèi)存池的一塊內(nèi)存。內(nèi)存管理表的項(xiàng)值代表的意義為:當(dāng)該項(xiàng)值為0的時(shí)候,代表對應(yīng)的內(nèi)存塊未被占用,當(dāng)該項(xiàng)值非零的時(shí)候,代表該項(xiàng)對應(yīng)的內(nèi)存塊已經(jīng)被占用,其數(shù)值則代表被連續(xù)占用的內(nèi)存塊數(shù)。比如某項(xiàng)值為10,那么說明包括本項(xiàng)對應(yīng)的內(nèi)存塊在內(nèi),總共分配了10個(gè)內(nèi)存塊給外部的某個(gè)指針。
內(nèi)存分配方向如圖所示,是從頂?shù)降椎姆峙浞较颉<词紫葟淖钅┒碎_始找空內(nèi)存。當(dāng)內(nèi)存管理剛初始化的時(shí)候,內(nèi)存表全部清零,表示沒有任何內(nèi)存塊被占用。
21.3.2 分配原理
當(dāng)指針p調(diào)用malloc申請內(nèi)存的時(shí)候,先判斷p要分配的內(nèi)存塊數(shù)m,然后從第n項(xiàng)開始,向下查找,直到找到m塊連續(xù)的空內(nèi)存塊(即對應(yīng)內(nèi)存管理表項(xiàng)為0),然后將這m個(gè)內(nèi)存管理表項(xiàng)的值都設(shè)置為m(標(biāo)記被占用),最后,把最后的這個(gè)空內(nèi)存塊的地址返回指針p,完成一次分配。注意,如果當(dāng)內(nèi)存不夠的時(shí)候(找到最后也沒找到連續(xù)的m塊空閑內(nèi)存),則返回NULL給p,表示分配失敗。
21.3.3 釋放原理
當(dāng)p申請的內(nèi)存用完,需要釋放的時(shí)候,調(diào)用free函數(shù)實(shí)現(xiàn)。free函數(shù)先判斷p指向的內(nèi)存地址所對應(yīng)的內(nèi)存塊,然后找到對應(yīng)的內(nèi)存管理表項(xiàng)目,得到p所占用的內(nèi)存塊數(shù)目m(內(nèi)存管理表項(xiàng)目的值就是所分配內(nèi)存塊的數(shù)目),將這m個(gè)內(nèi)存管理表項(xiàng)目的值都清零,標(biāo)記釋放,完成一次內(nèi)存釋放。
21.3.4 源代碼實(shí)現(xiàn)
(1)創(chuàng)建malloc.h文件,并輸入以下代碼。
/*********************************************************************************************************
內(nèi) 存 管 理 文 件
*********************************************************************************************************/
#ifndef _MALLOC_H_
#define _MALLOC_H_
#include "sys.h"
/*********************************************************************************************************
數(shù) 據(jù) 結(jié) 構(gòu) 定 義
*********************************************************************************************************/
//定義兩個(gè)內(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個(gè)區(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:代表錯(cuò)誤
其他:內(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 ++ ;
//搜索整個(gè)內(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個(gè)空內(nèi)存塊
if( cmemb==nmemb )
{
//標(biāo)注內(nèi)存塊非空
for( i=0; i
21.4 實(shí)驗(yàn)例程
實(shí)驗(yàn):利用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í)鐘設(shè)置
SysTick_Init( 72 ) ; //延時(shí)初始化
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)錯(cuò)誤,重新格式化 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)
{
}
}
-
單片機(jī)
+關(guān)注
關(guān)注
6023文章
44377瀏覽量
628533 -
嵌入式系統(tǒng)
+關(guān)注
關(guān)注
40文章
3520瀏覽量
128810 -
C語言
+關(guān)注
關(guān)注
180文章
7575瀏覽量
134184 -
文件系統(tǒng)
+關(guān)注
關(guān)注
0文章
280瀏覽量
19831 -
FATFS
+關(guān)注
關(guān)注
0文章
43瀏覽量
18211
發(fā)布評論請先 登錄
相關(guān)推薦
評論