利用SPI协议读写SD卡


一、HAL库配置

配置USART1

在这里插入图片描述
配置SPI1

在这里插入图片描述
配置FATFS

在这里插入图片描述
GPIO配置

在这里插入图片描述
时钟配置

在这里插入图片描述

Serial Wire配置

在这里插入图片描述
生成工程

在这里插入图片描述

二、移植并添加工程

(一)移植驱动文件

在这里插入图片描述
SDdriver.cSDdriver.h文件添加到工程中
在这里插入图片描述
复制到与工程同一路径后后,在工程里面添加SDdriver.c文件,并添加包含该文件的路径。

(二)修改user_diskio.c文件

①添加包含的头文件

#include "diskio.h"		/* Declarations of disk functions */
#include "SDdriver.h"

在这里插入图片描述
接下来分别对USER_initialize、USER_status、USER_read、USE_WRITE、USE_IOCTL函数进行修改,将其内容替换为下面对应函数相关内容

②修改USER_initialize函数

	uint8_t res;
	res = SD_init();//SD_Initialize() 
		 	if(res)//STM32 SPI的bug,在sd卡操作失败的时候如果不执行下面的语句,可能导致SPI读写异常
			{
				SPI_setspeed(SPI_BAUDRATEPRESCALER_256);
				spi_readwrite(0xff);//提供额外的8个时钟
				SPI_setspeed(SPI_BAUDRATEPRESCALER_2);
			}
	if(res)
		return  STA_NOINIT;
	else 
		return RES_OK; //初始化成功

在这里插入图片描述
③修改USER_status函数

 switch (pdrv)
	{
		case 0 :
			return RES_OK;
		case 1 :
			return RES_OK;
		case 2 :
			return RES_OK;
		default:
			return STA_NOINIT;
	}

④修改USER_read函数

uint8_t res;
	if( !count )
	{    
		return RES_PARERR;  /* count不能等于0,否则返回参数错误 */
	}
	switch (pdrv)
	{
		case 0:
		    res=SD_ReadDisk(buff,sector,count);	 
				if(res == 0){
					return RES_OK;
				}else{
					return RES_ERROR;
				}                                               
		default:
			return RES_ERROR;
	}

⑤修改USE_WRITE函数

uint8_t  res;
	if( !count )
	{    
		return RES_PARERR;  /* count不能等于0,否则返回参数错误 */
	}
	switch (pdrv)
	{
		case 0:
		    res=SD_WriteDisk((uint8_t *)buff,sector,count);
				if(res == 0){
					return RES_OK;
				}else{
					return RES_ERROR;
				}                                                
		default:return RES_ERROR;
	}

⑥修改USE_IOCTL函数

  DRESULT res;
	 switch(cmd)
	    {
		    case CTRL_SYNC:
						SD_CS(1);
						do{
							HAL_Delay(20);
						}while(spi_readwrite(0xFF)!=0xFF);
						res=RES_OK;
						SD_CS(0);
		        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 = SD_GetSectorCount();
		        res = RES_OK;
		        break;
		    default:
		        res = RES_PARERR;
		        break;
	    }
		return res;

(三)main文件配置

包含头文件

#include "SDdriver.h"

在main函数前添加变量与函数定义

uint16_t uart_value[3];
uint8_t aRxBuffer1;	//uart rx buff 

void WritetoSD(BYTE write_buff[],uint8_t bufSize);
char SD_FileName[] = "test.txt";
uint8_t WriteBuffer[] = "山有木兮\r\n";

//uint8_t test_sd =0;	//用于测试格式化
uint8_t write_cnt =0;	//写SD卡次数

void WritetoSD(BYTE write_buff[],uint8_t bufSize)
{
	FATFS fs;
	FIL file;
	uint8_t res=0;
	UINT Bw;	
	
	res = SD_init();		//SD卡初始化
	
	if(res == 1)
	{
		printf("SD卡初始化失败! \r\n");		
	}
	
	res=f_mount(&fs,"0:",1);		//挂载
	
//	if(test_sd == 0)		//用于测试格式化
	if(res == FR_NO_FILESYSTEM)		//没有文件系统,格式化
	{
//		test_sd =1;				//用于测试格式化
		printf("没有文件系统! \r\n");		
		res = f_mkfs("", 0, 0);		//格式化sd卡
		if(res == FR_OK)
		{
			//printf("格式化成功! \r\n");		
			res = f_mount(NULL,"0:",1); 		//格式化后先取消挂载
			res = f_mount(&fs,"0:",1);			//重新挂载	
			if(res == FR_OK)
			{
				//printf("SD卡已经成功挂载,可以进进行文件写入测试!\r\n");
			}	
		}
		else
		{
			printf("格式化失败! \r\n");		
		}
	}
	else if(res == FR_OK)
	{
		//printf("挂载成功! \r\n");		
	}
	else
	{
		printf("挂载失败! \r\n");
	}	
	
	res = f_open(&file,SD_FileName,FA_OPEN_ALWAYS |FA_WRITE);
	if((res & FR_DENIED) == FR_DENIED)
	{
		printf("卡存储已满,写入失败!\r\n");		
	}
	
	f_lseek(&file, f_size(&file));//确保写词写入不会覆盖之前的数据
	if(res == FR_OK)
	{
		//printf("打开成功/创建文件成功! \r\n");		
		res = f_write(&file,write_buff,bufSize,&Bw);		//写数据到SD卡
		if(res == FR_OK)
		{
			printf("文件写入成功! \r\n");			
		}
		else
		{
			printf("文件写入失败! \r\n");
		}		
	}
	else
	{
		printf("打开文件失败!\r\n");
	}	
	
	f_close(&file);						//关闭文件		
	f_mount(NULL,"0:",1);		 //取消挂载
	
}


void Get_SDCard_Capacity(void)
{
	FRESULT result;
	FATFS FS;
	FATFS *fs;
	DWORD fre_clust,AvailableSize,UsedSize;  
	uint16_t TotalSpace;
	uint8_t res;
	
	res = SD_init();		//SD卡初始化
	if(res == 1)
	{
		printf("SD卡初始化失败! \r\n");		
	}
	else
	{
		printf("SD卡初始化成功! \r\n");		
	}
	
	/* 挂载 */
	res=f_mount(&FS,"0:",1);		//挂载
	if (res != FR_OK)
	{
		printf("FileSystem Mounted Failed (%d)\r\n", result);
	}

	res = f_getfree("0:", &fre_clust, &fs);  /* 根目录 */
	if ( res == FR_OK ) 
	{
		TotalSpace=(uint16_t)(((fs->n_fatent - 2) * fs->csize ) / 2 /1024);
		AvailableSize=(uint16_t)((fre_clust * fs->csize) / 2 /1024);
		UsedSize=TotalSpace-AvailableSize;              
		/* Print free space in unit of MB (assuming 512 bytes/sector) */
		printf("\r\n%d MB total drive space.\r\n""%d MB available.\r\n""%d MB  used.\r\n",TotalSpace, AvailableSize,UsedSize);
	}
	else 
	{
		printf("Get SDCard Capacity Failed (%d)\r\n", result);
	}		
} 

将main函数替代为以下内容

int main(void)
{
  /* USER CODE BEGIN 1 */
  /* USER CODE END 1 */
  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_SPI1_Init();
  MX_FATFS_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
	HAL_UART_Receive_IT(&huart1,&aRxBuffer1,1); 	//enable uart	

	printf(" mian \r\n");
	Get_SDCard_Capacity();	//得到使用内存并选择格式化
  /* USER CODE END 2 */
  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
		WritetoSD(WriteBuffer,sizeof(WriteBuffer));		

		printf("writing\r\n");
    /* USER CODE END WHILE */
    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

(四)其他配置及接线

在usart.c中重定向printf()函数

int fputc(int ch, FILE *f)    
{
    HAL_UART_Transmit(&huart1, (unsigned char *)&ch, 1, 0xFFFF);    //修改【4】,串口重定向
    return ch;
}

在usart.h声明并包含头文件stdio.h
打开Micro LIB
编译完成即可
本实验是基于源码自己重新新建了一个工程,如果怕麻烦,可直接下载源码

源码: 提取码27r2

三、实例演示

接线方式

SD卡stm32系统板
CSPA4
SCKPA5
MISOPA6
MOSIPA7
VCC5V
GNDGND

上位机显示

初始化成功,正在写入数据

在这里插入图片描述

写数据成功,查看已写好的数据

在这里插入图片描述


总结

驱动SD卡的时候,最好将供电电压提高到5V,提高更强的驱动能力。我之前默认使用3.3v由于驱动能力不够,所以一直初始化失败,卡顿了好久。所以基于硬件,稍微有一点有错就会导致整体失败,要特别注意细节。

STM32用cube配置FATFS模式下SPI读写SD卡

  • 2
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 在STM32F4系列中,使用SPI和DMA读写SD卡是相对较简单的操作。 首先,我们需要配置SPI和DMA。对于SPI,我们需要设置SPI的工作模式、数据大小、极性和相位等参数。我们还需要配置DMA来传输数据。DMA可以在SPISD卡之间进行数据传输,提高速度和效率。 然后,我们需要初始化SD卡。这包括发送一系列的命令和参数来配置SD卡,使其能够与MCU进行通信。这些命令可以通过SPI进行传输,并且可以使用DMA进行数据传输。 接下来,我们可以使用SPI和DMA来读写SD卡。通过发送读写命令和地址来指定要读写的数据块。通过SPI和DMA进行数据传输,以读取或写入数据。 同时要注意,对于读取数据,我们需要等待SD卡的响应,并确保数据正确接收到MCU。对于写入数据,我们也需要等待SD卡的响应,并检查是否成功写入。 最后,在使用完SD卡后,我们需要进行关闭和释放相关资源的操作。这包括关闭SPI和DMA的功能,并确保SD卡处于适当的状态。 综上所述,通过配置SPI和DMA,初始化SD卡并使用SPI和DMA进行数据传输,我们可以在STM32F4上实现SD卡读写操作。这种方法能够提高效率和速度,并简化了操作过程。 ### 回答2: STM32F4 SPI+DMA方式读写SD卡的操作步骤如下: 1. 初始化SPI和DMA:首先需要初始化SPI和DMA模块,设置SPI相关参数,如数据位数、时钟分频等。同时,配置DMA的通道和相关参数,以便实现SPI数据的直接传输。 2. 初始化SD卡:根据SD卡规范,使用SPI发送命令和参数来初始化SD卡。初始化过程中,需要发送CMD0命令以及CMD8命令等,以及在响应中获取SD卡的OCR、CSD等信息。 3. 读写SD卡:使用SPI+DMA方式进行SD卡读写操作。对于读操作,先发送CMD17命令来指定要读取的块地址,然后启动DMA传输,将从SD卡读取的数据保存到目标内存中。对于写操作,先发送CMD24命令来指定要写入的块地址,然后启动DMA传输,将数据从源内存传输到SD卡。 4. 数据检验与校验:在读写操作完成后,需要进行数据的检验与校验。对于读操作,可以使用CRC校验码进行数据的完整性验证;对于写操作,可以使用CRC校验码来确保写入的数据正确。 5. 错误处理与重试:在读写操作中,可能会遇到SPI和DMA传输错误、SD卡响应错误等情况。在这种情况下,可以根据具体的错误类型进行相应的处理和重试操作,例如重新初始化SPI和DMA,重新发送命令等。 总结:通过SPI+DMA方式进行SD卡读写操作,可以提高数据传输的效率和速度,并且能够充分利用STM32F4的硬件资源。在实际操作过程中,需要注意配置SPI和DMA的相关参数,正确发送SD卡的命令和参数,以及处理可能出现的错误情况。 ### 回答3: STM32F4系列微控制器支持SPI(串行外设接口)和DMA(直接内存访问)功能,可以实现对SD卡读写操作。 首先,配置SPI硬件资源。选择一个可用的SPI外设和对应的引脚,并配置SPI的时钟频率、数据位、极性、相位等参数。 其次,配置DMA传输。选择一个可用的DMA通道,并设置传输方向、数据宽度、传输数量等参数,以使得DMA能够自动地在SPI和内存之间传输数据。 然后,初始化SD卡。通过发送SD卡命令和接收响应来识别和初始化SD卡,包括设置SPI的工作模式、速度、起始块地址等。 接下来,进行数据传输操作。如果要读取SD卡上的数据,首先发送读数据命令和相应的地址;然后通过DMA启动数据传输,将SD卡中的数据读取到指定的内存地址;最后等待DMA传输完成,并检查传输数据的正确性。 如果要写入数据到SD卡,首先发送写数据命令和相应的地址;然后通过DMA启动数据传输,将指定内存地址的数据写入到SD卡中;最后等待DMA传输完成,并检查写入数据的正确性。 最后,进行数据处理和错误处理。对于读取操作,可以对传输的数据进行解析和处理,以满足应用的需求。对于写入操作,可以检查写入数据是否正确,并处理写入数据过程中可能出现的错误,如超时、电压不稳定等。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值