当前位置:首页 > 单片机 > 单片机
[导读]这次讲的是将程序、图片或其他文件下载到SPI Flash中。我使用的是W25X16的SPI Flash,他共有2MB空间,2个Block,512ge Sector,8096个Page。由于SPI Flash不能直接跑程序,我们从接口就知道了。接下去我们就来讲讲怎么

这次讲的是将程序、图片或其他文件下载到SPI Flash中。我使用的是W25X16的SPI Flash,他共有2MB空间,2个Block,512ge Sector,8096个Page。由于SPI Flash不能直接跑程序,我们从接口就知道了。

接下去我们就来讲讲怎么编写SPI flash的升级功能。这次的工程是基于之前的Internal Flash修改而来的。修改的部分主要在USB_User组里:

我只将改改的部分。

hw_config.c、usb_istr.c、usb_prop.c、usb_pwr.c这介个文件没有什么还修改的。usb_desc.c文件需要修改下接口字符串描述符,由于我们的SPI Flash空间2M,所以我们将SPI Flash的2M空间全部设置成可读可写可擦除。


/*接口字符串描述符*/

uint8_t DFU_StringInterface0[DFU_SIZ_STRING_INTERFACE0] =

{

DFU_SIZ_STRING_INTERFACE0,

0x03,

//Interface 1: "@ SPI Flash: W25X16 /0x00000000/1*2048kg

'@', 0, 'S', 0, 'P', 0, 'I', 0, ' ', 0, 'F', 0, 'l', 0, 'a', 0, 's', 0,/*18*/

'h', 0, ' ', 0, ':', 0, ' ', 0, 'W', 0, '2', 0, '5', 0, 'X', 0, '1', 0, '6', 0,/*20*/

'/', 0, '0', 0, 'x', 0, '0', 0, '0', 0, '0', 0, '0', 0, '0', 0, '0', 0, '0', 0, '0', 0,/*22*/

'/', 0, '1', 0, '*', 0, '2', 0, '0', 0, '4', 0, '8', 0, 'K', 0, 'g', 0/*18*/

};


接下去,添加我们的W25X16 SPI Flash的驱动代码spi_flash.c,这个代码可以网上下载。接下去将上个工程里的flash—_if.c文件修改成spi_if.c,并修改里面的函数:

/*******************************************************************************

* Function Name : SPI_If_Init

* Description : Initializes the Media on the STM32

* Input : None

* Output : None

* Return : None

*******************************************************************************/

uint16_t SPI_If_Init(void)

{

SPI_Flash_Init();

return MAL_OK;

}


/*******************************************************************************

* Function Name : SPI_If_Erase

* Description : Erase sector

* Input : None

* Output : None

* Return : None

*******************************************************************************/

uint16_t SPI_If_Erase(uint32_t SectorAddress)

{

printf("正在擦除SPI Flash...rn");

SPI_Flash_Erase_Chip();

//SPI_Flash_Erase_Sector(SectorAddress);

printf("擦除成功!rn");

return MAL_OK;

}


/*******************************************************************************

* Function Name : SPI_If_Write

* Description : Write sectors

* Input : None

* Output : None

* Return : None

*******************************************************************************/

uint16_t SPI_If_Write(uint32_t SectorAddress, uint32_t DataLength)

{

uint32_t idx, pages;

printf("SPI_IFWrite写入数据长度为%drn",DataLength);

pages = (((DataLength & 0xFF00)) >> 8);


if (DataLength & 0xFF) /* Not a 256 aligned data */

{

for ( idx = DataLength; idx < ((DataLength & 0xFF00) + 0x100) ; idx++) // idx = DataLength; idx < ((DataLength & 0xFF00) + 0x100) ; idx++

{

MAL_Buffer[idx] = 0xFF;

}

pages = (((DataLength & 0xFF00)) >> 8 ) + 1;

}


for (idx = 0; idx < pages; idx++)

{

printf("正在向0x%x地址写数据rn",SectorAddress);

SPI_Flash_Write(&MAL_Buffer[idx*256], SectorAddress, 256);

SectorAddress += 0x100;

}

return MAL_OK;

}


/*******************************************************************************

* Function Name : SPI_If_Read

* Description : Read sectors

* Input : None

* Output : None

* Return : buffer address pointer

*******************************************************************************/

uint8_t *SPI_If_Read(uint32_t SectorAddress, uint32_t DataLength)

{

printf("正在读取地址0x%x处开始的%d个数据rn",SectorAddress,DataLength);

SPI_Flash_Read(MAL_Buffer, SectorAddress, (uint16_t)DataLength);

return MAL_Buffer;

}


然后要修改的是dfu_mal.c这个文件。修改成如下就可以了:

/*******************************************************************************

* Function Name : MAL_Init

* Description : STM32初始化的媒体初始化

* Input : None

* Output : None

* Return : None

*******************************************************************************/

uint16_t MAL_Init(void)

{

SPI_If_Init(); /* SPI Flash */

return MAL_OK;

}


/*******************************************************************************

* Function Name : MAL_Erase

* Description : 擦除扇区

* Input : None

* Output : None

* Return : None

*******************************************************************************/

uint16_t MAL_Erase(uint32_t SectorAddress)

{


switch (SectorAddress & MAL_MASK)//参看地址

{

case SPI_FLASH_BASE:

pMAL_Erase = SPI_If_Erase;

break;

default:

return MAL_FAIL;

}

return pMAL_Erase(SectorAddress);//指向擦除函数

}


/*******************************************************************************

* Function Name : MAL_Write

* Description : 写扇区

* Input : None

* Output : None

* Return : None

*******************************************************************************/

uint16_t MAL_Write (uint32_t SectorAddress, uint32_t DataLength)

{


switch (SectorAddress & MAL_MASK)//查看地址

{

case SPI_FLASH_BASE:

pMAL_Write = SPI_If_Write;

break;

default:

return MAL_FAIL;

}

return pMAL_Write(SectorAddress, DataLength);//调用写扇区函数

}


/*******************************************************************************

* Function Name : MAL_Read

* Description : 度扇区

* Input : None

* Output : None

* Return : Buffer pointer

*******************************************************************************/

uint8_t *MAL_Read (uint32_t SectorAddress, uint32_t DataLength)

{

switch (SectorAddress & MAL_MASK)//查看地址

{

case SPI_FLASH_BASE:

pMAL_Read = SPI_If_Read;

break;

default:

return 0;

}

return pMAL_Read (SectorAddress, DataLength);//调用如扇区函数

}


/*******************************************************************************

* Function Name : MAL_GetStatus

* Description : 获取状态

* Input : None

* Output : None

* Return : MAL_OK

*******************************************************************************/

uint16_t MAL_GetStatus(uint32_t SectorAddress , uint8_t Cmd, uint8_t *buffer)

{ //更具地址查找定时表的对应的选项

uint8_t x = (SectorAddress >> 26) & 0x03 ;

/* 0x000000000 --> 0 */

/* 0x640000000 --> 1 */

/* 0x080000000 --> 2 */


uint8_t y = Cmd & 0x01;


SET_POLLING_TIMING(TimingTable[x][y]); /* x: 擦除/写 定时 */

/* y: Media */

return MAL_OK;

}


最后的话,就是我们的main函数了,这里的main函数当然没有程序跳转了,我在这里用到了4个按键,

WAKEUP按键(PA0)按下表示向spi flash的0地址写入一组数据

TAMPER按键(PC13)按下表示读取0地址开始的数据

USER1按键(PA8)按下表示擦写0地址开始的那个扇区数据

USER2按键(PD3)按下表示向spi flash的0地址写入另一组数据

这样的话,就可以试试检测spi flash 读写是否正确了。

uint8_t DeviceState;

uint8_t DeviceStatus[6];


u8 WRITE_Buffer[]="神舟III号 SPI 读写访问程序";//spi flash写入数据缓存

u8 WRITE_Buffer1[]="神舟I号 SPI 读写访问程序";//spi flash写入数据缓存

u8 READ_Buffer[sizeof(WRITE_Buffer)];//spi flash读出数据缓存


/********************************************************

函数:main()

描述:程序入口地址

参数:无

返回:无

********************************************************/

int main(void)

{

BSP_Init();

printf(" |===============================================|rn");

printf(" STM32 DFU 程序开始 rn");

printf("|===============================================|rn");

SPI_Flash_Init();

{

u32 i;

i=SPI_Flash_ReadID(); //读取spi flash的芯片ID,一定要读,否则读写会出错

printf("ID:%xrn",i);

}

/* Enter DFU mode */

DeviceState = STATE_dfuERROR;

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

在微控制器领域,MSP430与STM32无疑是两颗璀璨的明星。它们各自凭借其独特的技术特点和广泛的应用领域,在市场上占据了重要的位置。本文将深入解析MSP430与STM32之间的区别,探讨它们在不同应用场景下的优势和局限...

关键字: MSP430 STM32 单片机

STM32是由意法半导体公司(STMicroelectronics)推出的基于ARM Cortex-M内核的32位微控制器系列,以其高性能、低功耗、丰富的外设接口和强大的生态系统深受广大嵌入式开发者喜爱。本文将详细介绍S...

关键字: STM32 单片机

STM32与51单片机之间有什么差异呢?两者可以说是一场科技与性能的较量了。在科技飞速发展的今天,微控制器(MCU)已广泛应用于各类电子设备和系统中,发挥着举足轻重的作用。其中,STM32和51单片机作为两种常见的微控制...

关键字: STM32 51单片机 MCU

西班牙塞维利亚,2024年3月12日 — Teledyne Technologies旗下公司、全球成像解决方案创新者Teledyne e2v宣布扩展其Flash™ CMOS图像传感器系列,推出Flash 2K LSA,该...

关键字: 图像传感器 Flash 摄像机

电磁铁是一种利用电流产生磁场的装置,具有快速响应、易于控制等特点,在工业自动化、电子设备、科学实验等领域有着广泛的应用。STM32是一款功能强大的微控制器,具有高性能、低功耗、易于编程等优点,是控制电磁铁的理想选择。本文...

关键字: 电磁铁 微控制器 STM32

边缘人工智能的实现涉及到三个基本 要素:安全性,连接性、自主性,而其中自主性是AI能力的体现,也是边缘AI有别于其他传统的物联网的关键。而通过ST Edge AI套件,就可以帮助各种不同类型的开发者实现覆盖全硬件平台的全...

关键字: 边缘人工智能 AI STM32

今天,小编将在这篇文章中为大家带来STM32单片机最小系统的有关报道,通过阅读这篇文章,大家可以对它具备清晰的认识,主要内容如下。

关键字: 单片机 单片机最小系统 STM32

STM32是一款由STMicroelectronics生产的微控制器系列,具有高性能、低功耗和丰富的外设资源。其中,串口通信是一种常用的通信方式,可以实现与其他设备之间的数据传输。

关键字: STM32 串口通信 微控制器

在这篇文章中,小编将为大家带来JTAG接口与Flash的相关报道。如果你对本文即将要讲解的内容存在一定兴趣,不妨继续往下阅读哦。

关键字: JTAG Flash

STM32是一种广泛使用的微控制器,具有丰富的通信接口。其中,串口通信是STM32与其他设备或系统进行数据交换的重要方式之一。本文将详细介绍STM32串口通信的原理、应用及常见故障。

关键字: STM32 串口通信
关闭
关闭