本帖最后由 qjp1988113 于 2021-1-30 16:13 编辑
做好了LCD的显示,接下来移植一下SD的FATFS系统,并让他显示根目录下的图片:
SD用SPI驱动,与LCD分时复用:
初始化SD_CS引脚,挂在SPI1总线上的还有XPT2406触摸芯片,我们这里也一并定义,并将触摸芯片的CS拉高,不选用:
void SD_port_init(void)
{
GPIO_InitType GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2PERIPH_GPIOB, ENABLE);
GPIO_StructInit(&GPIO_InitStructure);
//PB4 PB5
GPIO_InitStructure.GPIO_Pins = SD_CS_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT_PP;
GPIO_InitStructure.GPIO_MaxSpeed = GPIO_MaxSpeed_50MHz;
GPIO_Init(GPIOB, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pins = XPT2046_CS_PIN ;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT_PP;
//GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_InitStructure.GPIO_MaxSpeed = GPIO_MaxSpeed_50MHz;
GPIO_Init(GPIOB, &GPIO_InitStructure);
__SD_CS_SET();
__XPT2046_CS_SET();
}
SD卡对应SPI操作函数:
uint8_t SD_SPI_ReadWriteByte(uint8_t data)
{
return __SD_WRITE_BYTE(data);
}
//set spi in low speed mode.
void SD_SPI_SpeedLow(void)
{
SPI1->CTRL1&=0XFFC7;
SPI1->CTRL1|=SPI_MCLKP_256;
SPI_Enable(SPI1, ENABLE);
}
//set spi in high speed mode.
void SD_SPI_SpeedHigh(void)
{
SPI1->CTRL1&=0XFFC7;
SPI1->CTRL1|=SPI_MCLKP_32;
SPI_Enable(SPI1, ENABLE);
}
其中__SD_WRITE_BYTE(data) 对应为:SPI1ReadWriteByte(__DATA)
就是还是之前的SPI读写一字节函数。
其他SD操作函数函数:
uint8_t SD_WaitReady(void);
uint8_t SD_GetResponse(uint8_t Response);
uint8_t SD_Initialize(void);
uint8_t SD_ReadDisk(uint8_t*buf,uint32_t sector,uint8_t cnt);
uint8_t SD_WriteDisk(uint8_t*buf,uint32_t sector,uint8_t cnt);
uint32_t SD_GetSectorCount(void);
uint8_t SD_GetCID(uint8_t *cid_data);
uint8_t SD_GetCSD(uint8_t *csd_data);
FATFS移植,这里把配置文件提到一个新建的FATFS_CUSTEM的文件夹里面,以后直接使用。
FATFS关键就自在于diskio和ff两个C文件里面的读写函数,替换成我们现有硬件的即可。
如下面的,初始化和读写,实际就是调用的SD的初始化和读写:
//初始化磁盘
DSTATUS disk_initialize (
BYTE drv /* Physical drive nmuber (0..) */
)
{
uint8_t res=0;
switch(drv)
{
case SD_CARD://SD卡
res = SD_Initialize();//SD_Initialize()
if(res)//STM32 SPI的bug,在sd卡操作失败的时候如果不执行下面的语句,可能导致SPI读写异常
{
SD_SPI_SpeedLow();
SD_SPI_ReadWriteByte(0xff);//提供额外的8个时钟
SD_SPI_SpeedHigh();
}
break;
default:
res=1;
}
if(res)return STA_NOINIT;
else return 0; //初始化成功
}
//获得磁盘状态
DSTATUS disk_status (
BYTE drv /* Physical drive nmuber (0..) */
)
{
return 0;
}
//读扇区
//drv:磁盘编号0~9
//*buff:数据接收缓冲首地址
//sector:扇区地址
//count:需要读取的扇区数
DRESULT disk_read (
BYTE drv, /* Physical drive nmuber (0..) */
BYTE *buff, /* Data buffer to store read data */
DWORD sector, /* Sector address (LBA) */
BYTE count /* Number of sectors to read (1..255) */
)
{
uint8_t res=0;
if (!count)return RES_PARERR;//count不能等于0,否则返回参数错误
switch(drv)
{
case SD_CARD://SD卡
res=SD_ReadDisk(buff,sector,count);
if(res)//STM32 SPI的bug,在sd卡操作失败的时候如果不执行下面的语句,可能导致SPI读写异常
{
SD_SPI_SpeedLow();
SD_SPI_ReadWriteByte(0xff);//提供额外的8个时钟
SD_SPI_SpeedHigh();
}
break;
default:
res=1;
}
//处理返回值,将SPI_SD_driver.c的返回值转成ff.c的返回值
if(res==0x00)return RES_OK;
else return RES_ERROR;
}
//写扇区
//drv:磁盘编号0~9
//*buff:发送数据首地址
//sector:扇区地址
//count:需要写入的扇区数
#if _READONLY == 0
DRESULT disk_write (
BYTE drv, /* Physical drive nmuber (0..) */
const BYTE *buff, /* Data to be written */
DWORD sector, /* Sector address (LBA) */
BYTE count /* Number of sectors to write (1..255) */
)
{
uint8_t res=0;
if (!count)return RES_PARERR;//count不能等于0,否则返回参数错误
switch(drv)
{
case SD_CARD://SD卡
res=SD_WriteDisk((uint8_t*)buff,sector,count);
break;
default:
res=1;
}
//处理返回值,将SPI_SD_driver.c的返回值转成ff.c的返回值
if(res == 0x00)return RES_OK;
else return RES_ERROR;
}
其中fatfs_storage.C 里面封装了一些常见的比如复制,比较、显示BMP等。
整个程序目录:
下载调试:
代码:
SPI_LCD _FATFS_BMP.rar
(5.74 MB)
|
cool 。。。。。。