源:STM32F10X SPI操作flash MX25L64读写数据

  前一段时间在弄SPI,之前没接触过嵌入式外围应用,就是单片机也只接触过串口通信,且也是在学校的时候了。从离开手机硬件测试岗位后,自己一直想在嵌入式方面发展,在1月4号开始自己的第二份工作后,首先接触到的是为STM32F103写SPI控制flash读写操作,现记下曾经的脚印,希望以后能少走弯路!心得:细心活!

  简单的一种应用,ARM芯片作为master,flash为slaver,实现单对单通信。ARM主控芯片STM32F103,flash芯片为MACRONIX INTERNATIONAL的MX25L6465E,64Mbit。

  SPI应该是嵌入式外围中最简单的一种应用了吧!一般SPI应用有两种方法:软件仿真,手动模拟产生时序和应用主控芯片的SPI控制器。

  一般采用第二种方法比较好,比较稳定。应用主控芯片的SPI控制器,要点:正确的初始化SPI、操作SPI各寄存器和正确理解flash的时序。下面是过程,采用的是STM32F10X自带的库函数

1、初始化:void SpiFlashInitialzation(void);

  要知道硬件是怎么连接的,是SPI1还是SPI2连接到flash中去,通过连接图知道我们要操作的是SPI2。初始化大概3个部分,配置时钟;配置GPIO;配置SPI2。这里要注意的是,CS片选脚是作为普通的GPIO来使用,输出方式为“推挽式输出”,其他CLK,MISO,MOSI为“复用功能推挽式输出”;

代码:

void SpiFlashInitialzation(void)
{/*初始化的SPI,GPIO结构体*/SPI_InitTypeDef  SPI_InitStructure;GPIO_InitTypeDef GPIO_InitStructure;RCC_APB1PeriphClockCmd( RCC_APB1Periph_SPI2,  ENABLE); /*在RCC_APB1ENB中使能SPI2时钟(位14)*/RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOB,  ENABLE);/*因为与SPI2相关的4个引脚和GPIOB相*//*关,GPIOB时钟(位3),这句现在还不 *//*确定要不要,待调试时再确定              *//*上面这一句是必须的,因为CS脚是当做GPIO来使用的,2011-01-30调试*//*配置SPI_FLASH_CLK(PB13),SPI_FLASH_MISO(PB14),SPI_FLASH_MOSI(PB15)*/GPIO_InitStructure.GPIO_Pin=GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;                    /*复用功能推挽式输出*/GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init( GPIOB, &GPIO_InitStructure);/*配置输入SPI_FLASH_CS(PB12)*/GPIO_InitStructure.GPIO_Pin=GPIO_Pin_12;GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;   /*推挽式输出*/GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;GPIO_Init( GPIOB, &GPIO_InitStructure);SPI_FLASH_CS_SET;             /*不选flash*//* SPI2配置 增加于2010-01-13*//* 注意:  在SPI_NSS_Soft模式下,SSI位决定了NSS引脚上(PB12)的电平,*            而SSM=1时释放了NSS引脚,NSS引脚可以用作GPIO口*/SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;   /*双线双向全双工BIDI MODE=0*/SPI_InitStructure.SPI_Mode = SPI_Mode_Master;                                 /*SSI位为1,MSTR位为1*/SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;                              /*SPI发送接收8位帧结构*/SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;                                     /*CPOL=1,CPHA=1,模式3*/SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;                                          /*内部NSS信号由SSI位控制,SSM=1*/SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_4;    /*波特率预分频值为4*/SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;                                       /*数据传输从MSB位开始*/SPI_InitStructure.SPI_CRCPolynomial = 7;                                                      /*复位默认值*/SPI_Init(SPI_SELECT, &SPI_InitStructure);SPI_Cmd(SPI_SELECT,ENABLE);       /*使能SPI2*/}

2、正确的操作SPI控制器;

  这里需要注意的是理解SPI状态寄存器,特别是SPI_SR位7忙标志位BSY要小心,每次操作SPI要先读SPI_SR,BSY不忙才可下一步,然后就是操作缓冲器了。这里还有一个问题曾经困扰了我好久,SPI的时序问题,就是CLK怎么输出时序,最后我的理解是SPI每发送一个字节,CLK就自动会产生时序,如果没发送,CLK也就停止,这样节省了功耗。于是,如果SPI要接收字节,就必须先要发一个字节,例如发一个SPI_DUMMY_BYTE,Dummy byte有些flash有定义有些没有,没有的话自己随便定义一个,只要不和命令字相同就可以了。

u8 SpiFlashSendByte(u8 send_data);

u8 SpiFlashReceiveByte(void);

代码:

/*******************************2011-01-13******************************/
/*功能:       SPI发送一个字节*参数:       send_data:   待发送的字节*返回:       无*/
u8 SpiFlashSendByte(u8 send_data)
{/*检查Busy位,SPI的SR中的位7,SPI通信是否为忙,直到不忙跳出*///while( SET==SPI_I2S_GetFlagStatus(SPI_SELECT, SPI_I2S_FLAG_BSY));/*检查TXE位,SPI的SR中的位1,发送缓冲器是否为空,直到空跳出*/while( RESET==SPI_I2S_GetFlagStatus(SPI_SELECT,SPI_I2S_FLAG_TXE));SPI_I2S_SendData(SPI_SELECT, send_data);                        /*发送一个字节*//*发送数据后再接收一个字节*/while( RESET==SPI_I2S_GetFlagStatus(SPI_SELECT, SPI_I2S_FLAG_RXNE) );return( SPI_I2S_ReceiveData(SPI_SELECT) );}

/*******************************2011-01-13******************************/
/*功能:       SPI接收flash的一个字节*参数:       接收到的字节*返回:       无*/
u8 SpiFlashReceiveByte(void)
{/*检查RXNE位,SPI的SR中位0,确定接收缓冲器是有数据的*/return(SpiFlashSendByte(SPI_DUMMY_BYTE));
}

3、理解flash的读写操作

  首先,写数据之前必须要擦除,因为所有的flash只能从1变为0,擦除将flash全部置1,写的时候相应位置0。

  读写操作这部分,flash芯片手册详细的说明了操作步骤,需要注意的是:flash MX25L64的状态寄存器。对flash操作之前,先读flash_SR,确保WIP=0(flash空闲),对flash擦除、编程等操作确保WEL=1(flash能够接受擦出编程等操作)。

  在对flash进行写操作时,要理解一点:对flash写数据(也就是Page Program(PP),Command 02)是基于页(256bytes)为单位的,如果数据写到页的末尾,会从当前页的首地址继续开始写剩余的数据,这样就有可能造成成数据的丢失,注意就可以了!主要是理解手册中的这段话:The Page Program(PP) instruction is for programming the memory to be "0"......If the eight least significant address bits(A7-A0) are not all 0,all transmitted data going beyond the end of the current page are grogrammed from the start address of the same page(from the address A7-A0 are all 0).If more than 256 bytes are sent to the device,the data of the last 256-byte is programmed at the requtest page and previous data will be disregarded. If less than 256 bytes .......

代码:

/*********************************2011-01-29*****************************/
/*功能:    在指定地址处开始从flash读取数据参数:     pData_from_flash,读取到的数据存放指针address_to_read,  待读取的数据开始地址,地址格式有效位为:A23-A0返回:     指向读取到的数据指针pData_from_flash*/
void SpiFlashReadData( u8 *pData_from_flash, u32 address_to_read , u16 size_to_read)
{/*先检查flash设备是否为忙,然后检查SPI控制器是否处于忙状态*/while( FLASH_SR_WIP==(SpiReadFlash_SR() & FLASH_SR_WIP) );/*读flash_SR*/while( SET==SPI_I2S_GetFlagStatus(SPI_SELECT, SPI_I2S_FLAG_BSY));/*读SPI_SR*/SPI_FLASH_CS_RESET;                                                                /*失能设备*/SpiFlashSendByte(SPI_COMMAND_READ);         /*发送读命令*/SpiFlashSendByte( (u8)((address_to_read & 0xFF0000) >> 16) );/*发送A23~A16*/SpiFlashSendByte( (u8)((address_to_read & 0xFF00) >> 8) );      /*发送A15~A8  */SpiFlashSendByte( (u8)(address_to_read & 0xFF) );                         /*发送A7~A0   */while( size_to_read>0 ){*pData_from_flash=SpiFlashReceiveByte();  /*读取数据*/pData_from_flash++;size_to_read--;}SPI_FLASH_CS_SET;
}

/*******************************2011-01-29******************************/
/*功能:     往指定地址处开始写数据*参数:     pBuff_to_write:       指向待写入的数据指针*              address_to_write:   flash何处开始写数据的地址*              size_to_write:          写入的数据字节数*返回:     TRUE:    写入成功*               FALSE:  写入失败*注意:     size_to_write,必须小于FLASH_PAGE_SIZE的大小(256 bytes),如果数据写到页*              的末尾,会从当前页的首地址0x00继续写剩余的数据,这样就造成数据的丢失,*              所以调用此函数得确保这一情况不会发生*/
void SpiFlashWritePageData(u8 *pBuff_to_write,u32 address_to_write, u16 size_to_write)
{/*先检查flash设备是否为忙,然后检查SPI是否处于忙状态*/while( FLASH_SR_WIP==(SpiReadFlash_SR() & FLASH_SR_WIP) );/*flash_SR*/while( SET==SPI_I2S_GetFlagStatus(SPI_SELECT, SPI_I2S_FLAG_BSY));/*SPI_SR*//*获得对flash的写权限*/while( FLASH_SR_WEL != (SpiReadFlash_SR() &FLASH_SR_WEL) ){SpiFlashWriteEnable();                                                       /*如果WEL为复位,则置位*/}SPI_FLASH_CS_RESET;SpiFlashSendByte(SPI_COMMAND_PP);                                                  /*发送写PP命令*/SpiFlashSendByte( (u8)((address_to_write & 0xFF0000) >> 16) );   /*发送A23~A16*/SpiFlashSendByte( (u8)((address_to_write & 0xFF00) >> 8) );         /*发送A15~A8  */SpiFlashSendByte( (u8)(address_to_write  & 0xFF) );                          /*发送A7~A0   */while( size_to_write>0 ){SpiFlashSendByte(*pBuff_to_write);pBuff_to_write++;size_to_write--;}SPI_FLASH_CS_SET;/*2011-01-14*//*检查设备已经写完才退出*/while ( FLASH_SR_WIP==(SpiReadFlash_SR() & FLASH_SR_WIP) );/**/}

4、  读写操作完成了,大概也就完成了,其它的参考flash手册就OK啦,不在描述。

另外,还有一种方法,是用软件模拟时序,这方法用在没有SPI控制器的单片机上很实用。

void SpiSendOneByte(u8 send_byte)
{_nop_();_nop_();//SPI_SCLK_RESET;/*第一个上升沿*/for( __IO u8  i=8; i>0; i-- ){SPI_SCLK_RESET;if( 0X00 != (send_byte & 0x80) ){SPI_MOSI_SET;}else{SPI_MOSI_RESET;}send_byte<<=1;SPI_SCLK_SET;_nop_();_nop_();_nop_();}
}

/*******************************************************************/
/*Serial Modes Supported(for Normal Serial mode)*/
/*                                    CPOL  CPHASerial mode 0:           0          0Serial mode 3:           1          1*//*功能:  从高到低接收一个字节,高位先接收*//*输出:  接收到的数据*//*下降沿时,数据出现在SO,低电平的时候把数据读到*/u8 SpiGetOneByte(void)
{__IO u8 get_byte=0;for( __IO u8  i=0; i<8; i++ ){get_byte<<=1;SPI_SCLK_RESET;_nop_();_nop_();_nop_();_nop_();_nop_();if( 1==SPI_MISO ){get_byte |= SPI_MISO;}SPI_SCLK_SET;_nop_();_nop_();_nop_();}return(get_byte);
}

STM32F10X SPI操作flash MX25L64读写数据(转)相关推荐

  1. STM32F10X SPI操作flash MX25L64读写数据

    STM32F10X SPI操作flash MX25L64读写数据 前一段时间在弄SPI,之前没接触过嵌入式外围应用,就是单片机也只接触过串口通信,且也是在学校的时候了.从离开手机硬件测试岗位后,自己一 ...

  2. 学习STM32 Flash存储 W25Q64 SPI总线存储模块进行读写数据

    今天学习 Flash 存储芯片进行数据写入和读取方法.了解W25Q64 存储芯片的使用.能够用 STM32 单片机对 W25Q64 进行写入数据,擦除数据,读取数据. w25q64 是什么? W25Q ...

  3. spi flash通用读写软IP

    spi flash通用读写模块,有两个模块,分别为spiflash控制模块和spi控制模块 spiflash控制模块RTL代码如下: //功能描述 //这是一个spiflash的控制程序 //写选择( ...

  4. 软件模拟SPI时序实现25Q64读写操作

    软件模拟SPI时序实现25Q64读写操作 单片机采用SPI/IIC通讯协议访问外围电子模块如:显示屏.EEPROM.FLASH.各种电子传感器等等越来越多,掌握SPI/IIC通讯协议访问外设非常必要. ...

  5. NAND FLASH的读写操作(硬件原理图分析)

    转载:NAND FLASH的读操作及原理 硬件原理 上面是我使用的NAND FLASH的硬件原理图,面对这些引脚,很难明白他们是什么含义,下面直接引用韦东山老师的课程中的提问: NAND FLASH是 ...

  6. stm32使用SPI对W25Q64--8M字节FLASH的读写

     先W25Q64介绍: 板上有个带SPI的FLASH产品,W25Q64,华邦公司,大容量64Mb的,也就是8M字节, (W25Q64把这个8M的容量分为128块-Block,每块64K字节,分出1 ...

  7. U-Boot操作Flash (NOR, NAND SPI)

    U-Boot中操作三种flash的命令集:                  Action                                          NOR           ...

  8. QCC300x笔记(5) -- 外部Flash的读写操作

    哈喽大家好,这是该系列博文的第五篇~ 篇~ <<[系列博文索引]快速通道 > 1.    QCC300X 外部Flash的读写         QCC300x是使用外部Flash片子 ...

  9. 【Android 高性能音频】AAudio 音频流 读写操作 ( 音频流读写数据 | 阻塞时间设定 | 注意事项 | AAudioStream_read | AAudioStream_write )

    文章目录 I . AAudio 音频流 读写操作 简介 II . AAudio 音频流 读写操作 阻塞时间设定 III . AAudio 音频流 读取 固定帧数 操作 注意点 IV . AAudio ...

最新文章

  1. leetcode 30. Substring with Concatenation of All Words 与所有单词相关联的字串 滑动窗口法
  2. php中ini set,php ini_set更改php.ini配置功能
  3. html 判断整数,javascript如何判断值是否为整数?
  4. ​稳健、可靠全真即时通信网的架构与应用
  5. 【转载】我目瞪口呆地望着即时通讯
  6. lstm原文_LRCN: LSTM与CNN相结合模型
  7. 【渝粤教育】国家开放大学2019年春季 2727税收基础 参考试题
  8. 笔记本电脑插入耳机只能外放,耳机没声音
  9. 【数据分析】数据分析在电商精准营销及效果评估中的应用
  10. 568A与568B 线序
  11. 综合1:stm32F4,ATKESP8266wifi,DS1302,AT24C02,KEY状态机,LCD屏幕320*240
  12. 大疆精灵4rtk照片信息读取
  13. 如何设置添加页眉 计算机知识,怎么设置页眉页脚-轻松解决计算机二级之页眉页脚不再是问题...
  14. 电商项目:高仿小米商城(API文档)
  15. 计算机vb里代码里的双引号,在VB中使用字符串中的左双引号
  16. 在java中使用openCV遍历红色,像素遍历(仅限java)
  17. Linux技术--mysql数据库基础操作
  18. vue 右键菜单contextMenu
  19. dellT440和T620重装系统问题总结
  20. 详解 masm + textpad 编译环境的搭建

热门文章

  1. HTML5+CSS大作业——响应式个人简历介绍(5页)-模板下载
  2. “《正在爆发的互联网革命》——30余位互联网大腕一致推荐,3.5亿网民必看!”
  3. 电骡(eMule)中文版 - NAT原理简介及各种路由器的端口映射方法
  4. Python解析VOC数据标签文件
  5. python+diango写酒店管理系统_课内资源 - 基于Python的Django框架实现的中式快餐厅管理信息系统网站...
  6. 微软crm mscrm 区域城市联动 用 js+查找列 实现
  7. IIC总线协议,7位,8位,10位地址
  8. educoder——面向对象程序设计java——第二阶段Java面向对象——思政-1 : 科技-->人工智能
  9. 中国联通:2018年7月沃指数之移动应用APP排行榜
  10. discuz x3.5