这里会显示出您选择的修订版和当前版本之间的差别。
| 两侧同时换到之前的修订记录 前一修订版 后一修订版 | 前一修订版 | ||
| icore3_arm_hal_17 [2020/03/31 15:59] zgf [二、 实验设备及平台] | icore3_arm_hal_17 [2022/03/18 15:07] (当前版本) sean | ||
|---|---|---|---|
| 行 2: | 行 2: | ||
| |技术支持电话|**0379-69926675-801**||| | |技术支持电话|**0379-69926675-801**||| | ||
| |技术支持邮件|Gingko@vip.163.com||| | |技术支持邮件|Gingko@vip.163.com||| | ||
| - | |技术论坛|http://www.eeschool.org||| | ||
| ^ 版本  ^ 日期  ^ 作者  ^ 修改内容  ^ | ^ 版本  ^ 日期  ^ 作者  ^ 修改内容  ^ | ||
| | V1.0 | 2020-03-31  | gingko  | 初次建立  | | | V1.0 | 2020-03-31  | gingko  | 初次建立  | | ||
| 行 89: | 行 88: | ||
| http://elm-chan.org/fsw/ff/00index_e.html | http://elm-chan.org/fsw/ff/00index_e.html | ||
| === 4.FATFS模块的层次结构: === | === 4.FATFS模块的层次结构: === | ||
| - | {{ :icore3:icore3_arm_hal_17_1.png?direct&400 |}} | + | {{ :icore3:icore3_arm_hal_17_1.png?direct&300 |}} | 
| * **最顶层是应用层:** | * **最顶层是应用层:** | ||
| * 使用者无需理会FATFS的内部结构和复杂的FAT协议,只需要调用FATFS模块提供给用户的一系列应用接口即可,如f_open,f_read, f_write和f_close等,就可以像在PC上读/写文件那样简单。 | * 使用者无需理会FATFS的内部结构和复杂的FAT协议,只需要调用FATFS模块提供给用户的一系列应用接口即可,如f_open,f_read, f_write和f_close等,就可以像在PC上读/写文件那样简单。 | ||
| 行 103: | 行 102: | ||
| * FATFS的这些特点,加上免费、开源的原则,使得FATFS应用非常广泛。 | * FATFS的这些特点,加上免费、开源的原则,使得FATFS应用非常广泛。 | ||
| * STM32F407上带有SDIO控制器,iCore3核心板上将SDIO连接到TF卡座上。本实验将Micro SD卡插入TF卡座上即可。通过FATFS创建“SD卡测试.txt”文件,并且写入数据0-511,然后读出并打印到终端上。硬件连接示意图如下: | * STM32F407上带有SDIO控制器,iCore3核心板上将SDIO连接到TF卡座上。本实验将Micro SD卡插入TF卡座上即可。通过FATFS创建“SD卡测试.txt”文件,并且写入数据0-511,然后读出并打印到终端上。硬件连接示意图如下: | ||
| - | {{ :icore3:icore3_arm_hal_17_1.png?direct&400 |}} | + | {{ :icore3:icore3_arm_hal_17_2.png?direct&400 |}} | 
| * 硬件电路图如下图所示: | * 硬件电路图如下图所示: | ||
| - | {{ :icore3:icore3_arm_hal_17_1.png?direct&750 |}} | + | {{ :icore3:icore3_arm_hal_17_3.png?direct&750 |}} | 
| ==== 四、 实验程序 ==== | ==== 四、 实验程序 ==== | ||
| === 1. 主函数 === | === 1. 主函数 === | ||
| <code c> | <code c> | ||
| - | + | int main(void) | |
| + | { | ||
| + | int i,j; | ||
| + | static FRESULT res; | ||
| + | unsigned char write_buffer[512];  | ||
| + | unsigned char read_buffer[512];  | ||
| + | unsigned int counter; | ||
| + | HAL_Init(); | ||
| + | SystemClock_Config(); | ||
| + | MX_GPIO_Init(); | ||
| + | MX_DMA_Init(); | ||
| + | MX_SDIO_SD_Init(); | ||
| + | MX_UART4_Init(); | ||
| + | MX_FATFS_Init(); | ||
| + | uart4.initialize(115200);//设置波特率 | ||
| + | uart4.printf("\x0c");  //清屏  | ||
| + | uart4.printf("\033[1;32;40m");  //设置字体终端为绿色  | ||
| + | uart4.printf("\r\nHello, I am iCore3!\r\n\r\n");//串口信息输出 | ||
| + | HAL_Delay(500); | ||
| + | res = f_mount(&SDFatFS,"0:",1); //判断f_mount是否成功 | ||
| + | if(res != RES_OK){ | ||
| + | uart4.printf("f_mount error!\r\n"); | ||
| + | while(1){ | ||
| + | LED_RED_ON; | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_OFF; | ||
| + | HAL_Delay(500); | ||
| + | } | ||
| + | }else{ | ||
| + | uart4.printf("f_mount successful!\r\n"); | ||
| + | } | ||
| + | res = f_open(&SDFile,"0:/SD卡测试.txt",FA_READ | FA_WRITE | FA_OPEN_ALWAYS); //打开源文件 | ||
| + | if(res != RES_OK){ | ||
| + | uart4.printf("f_open error!\r\n"); | ||
| + | while(1){ | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_ON; | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_OFF; | ||
| + | } | ||
| + | }else{ | ||
| + | uart4.printf("f_open successful!\r\n"); | ||
| + | } | ||
| + | res = f_lseek(&SDFile,0); //移动写指针到文件首  | ||
| + | if(res != RES_OK){ | ||
| + | uart4.printf("f_lseek error!\r\n"); | ||
| + | while(1){ | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_ON; | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_OFF; | ||
| + | } | ||
| + | }else{ | ||
| + | uart4.printf("f_lseek successful!\r\n"); | ||
| + | } | ||
| + | for(i = 0;i < 512;i++){ | ||
| + | write_buffer[i] = i % 256; | ||
| + | } | ||
| + | //向文件中写入数据  | ||
| + | res = f_write(&SDFile,write_buffer,512,&counter); | ||
| + | if(res != RES_OK || counter != 512){ | ||
| + | uart4.printf("f_write error!\r\n"); | ||
| + | while(1){ | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_ON; | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_OFF; | ||
| + | } | ||
| + | }else{ | ||
| + | uart4.printf("f_write successful!\r\n"); | ||
| + | } | ||
| + | //移动读指针到文件首  | ||
| + | res = f_lseek(&SDFile,0);  | ||
| + | if(res != RES_OK){ | ||
| + | uart4.printf("f_lseek error!\r\n"); | ||
| + | while(1){ | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_ON; | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_OFF; | ||
| + | } | ||
| + | }else{ | ||
| + | uart4.printf("f_lseek successful!\r\n"); | ||
| + | } | ||
| + | //读取文件数据  | ||
| + | res = f_read(&SDFile,read_buffer,512,&counter); | ||
| + | if(res != RES_OK || counter != 512){ | ||
| + | uart4.printf("f_read error!\r\n"); | ||
| + | while(1){ | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_ON; | ||
| + | HAL_Delay(500); | ||
| + | LED_RED_OFF; | ||
| + | } | ||
| + | }else{ | ||
| + | uart4.printf("f_read successful!\r\n"); | ||
| + | } | ||
| + | f_close(&SDFile); | ||
| + | //打印读取到的数据  | ||
| + | uart4.printf("read data:\r\n"); | ||
| + | for(i = 0;i < 32;i++){ | ||
| + | for(j = 0; j < 16; j ++) | ||
| + | uart4.printf(" %02X",read_buffer[i*16+j]); | ||
| + | uart4.printf("\r\n"); | ||
| + | } | ||
| + | while (1) | ||
| + | { | ||
| + | LED_GREEN_ON; | ||
| + | HAL_Delay(500); | ||
| + | LED_GREEN_OFF; | ||
| + | HAL_Delay(500); | ||
| + | } | ||
| + | } | ||
| </code> | </code> | ||
| === 2 FATFS函数介绍 === | === 2 FATFS函数介绍 === | ||
| <code c> | <code c> | ||
| - | + | FRESULT f_mount ( //挂载/卸载逻辑驱动器 | |
| + | FATFS* fs, /* 指向文件系统对象的指针*/ | ||
| + | const TCHAR* path, /* 要安装/卸载的逻辑驱动器号 */ | ||
| + | BYTE opt /* 模式选项0:不安装(延迟安装),1:立即安装*/ | ||
| + | ) | ||
| + | FRESULT f_open ( //打开或创建文件 | ||
| + | FIL* fp, /* 指向空白文件对象的指针 */ | ||
| + | const TCHAR* path, /* 指向文件名的指针 */ | ||
| + | BYTE mode /* 访问模式和文件打开模式标志 */ | ||
| + | ) | ||
| + | FRESULT f_read ( //读文件 | ||
| + | FIL* fp, /* 指向文件对象的指针 */ | ||
| + | void* buff,  /* 指向数据缓冲区的指针 */ | ||
| + | UINT btr, /* 读取的字节数 */ | ||
| + | UINT* br /* 指向读取的字节数的指针 */ | ||
| + | ) | ||
| + | FRESULT f_write ( //写文件 | ||
| + | FIL* fp, /* 指向文件对象的指针 */ | ||
| + | const void* buff, /* 指向要写入的数据的指针 */ | ||
| + | UINT btw, /* 要写入的字节数 */ | ||
| + | UINT* bw /* 指向写入字节数的指针 */ | ||
| + | ) | ||
| + | FRESULT f_sync ( //冲洗一个写文件的缓存信息 | ||
| + | FIL* fp /* 指向文件对象的指针 */ | ||
| + | ) | ||
| + | FRESULT f_close ( //关闭一个文件 | ||
| + | FIL* fp /* 指向要关闭的文件对象的指针 */ | ||
| + | ) | ||
| + | FRESULT f_lseek ( //移动文件读/写指针 | ||
| + | FIL* fp, /* 指向文件对象的指针 */ | ||
| + | FSIZE_t ofs /* 指向文件头的指针 */ | ||
| + | ) | ||
| + | FRESULT f_opendir ( //创建目录对象 | ||
| + | DIR* dp, /* 指向要创建的目录对象的指针 */ | ||
| + | const TCHAR* path /* 指向目录路径的指针 */ | ||
| + | ) | ||
| + | FRESULT f_closedir ( //  关闭目录 | ||
| + | DIR *dp /* 指向要关闭的目录对象的指针 */ | ||
| + | ) | ||
| </code> | </code> | ||
| === 3. SD结构体定义 === | === 3. SD结构体定义 === | ||
| <code c> | <code c> | ||
| - | + | FRESULT f_readdir ( //顺序读取目录条目 | |
| + | DIR* dp, /* 指向打开目录对象的指针 */ | ||
| + | FILINFO* fno /* 指向要返回的文件信息的指针 */ | ||
| + | ) | ||
| + | FRESULT f_stat ( //获取文件状态 | ||
| + | const TCHAR* path, /* 指向文件路径的指针 */ | ||
| + | FILINFO* fno /* 指向要返回的文件信息的指针 */ | ||
| + | ) | ||
| + | FRESULT f_getfree ( //获取空闲簇数 | ||
| + | const TCHAR* path, /* 逻辑驱动器号的路径名 */ | ||
| + | DWORD* nclst,  /* 指向变量的指针以返回空闲簇的数量*/ | ||
| + | FATFS** fatfs  /* 返回指向相应文件系统对象的指针的指针 */ | ||
| + | ) | ||
| + | FRESULT f_truncate ( //截断文件 | ||
| + | FIL* fp /* 指向文件对象的指针 */ | ||
| + | ) | ||
| + | FRESULT f_unlink ( //删除一个文件或目录 | ||
| + | const TCHAR* path /* 指向文件或目录路径的指针 */ | ||
| + | ) | ||
| + | FRESULT f_mkdir ( //创建一个目录 | ||
| + | const TCHAR* path /* 指向目录路径的指针 */ | ||
| + | ) | ||
| + | FRESULT f_rename (  //重命名文件/目录 | ||
| + | const TCHAR* path_old,  /* 指向要重命名的对象名称的指针 */ | ||
| + | const TCHAR* path_new /* 指向新名称的指针 */ | ||
| + | ) | ||
| </code> | </code> | ||
| ==== 五、 实验步骤 ==== | ==== 五、 实验步骤 ==== | ||
| 行 129: | 行 306: | ||
| ==== 六、 实验现象 ==== | ==== 六、 实验现象 ==== | ||
| 在终端显示屏上可以看到FATFS操作的信息,如下图: | 在终端显示屏上可以看到FATFS操作的信息,如下图: | ||
| - | {{ :icore3:icore3_arm_hal_17_4.png?direct&700 |}} | + | {{ :icore3:icore3_arm_hal_17_4.png?direct&550 |}} | 
| 附录: | 附录: | ||