历史上的今天
返回首页

历史上的今天

今天是:2024年08月31日(星期六)

2021年08月31日 | 玩转STM32CubeMX | SPI总线

2021-08-31 来源:eefocus

1.SPI总线及W25QXX芯片

1.1 SPI总线简介

SPI全称Serial Peripheral Interface,即串行外围设备接口。是Motorola首先在其MC68HCXX系列处理器上定义的。SPI接口主要应用在EEPROM、FLASH、实时时钟、AD转换器,还有数字信号处理器和数字信号解码器之间。SPI是一种高速的、全双工、同步通讯总线,在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为PCB的布局节省空间提供了方便,正是这种简单易用的特性,如今越来越多的芯片集成了这种通讯协议。下图是SPI内部结构简易图

从上图可以看出,主设备和从设备都有一个串行移位寄存器,主设备通过向它的SPI串行寄存器写入一个字节来发起一次传输,寄存器通过MOSI信号线将字节传送给从设备,从设备也将自已的移位寄存器中的内容通过MISO信号线返回给主设备。这样两个移位寄存器中的内容就被交换。外设的写操作和读操作时同步完成的,如果只进行写操作,主设备只需要忽略接收到的字节,如果主设备要进行读操作,就必须发送一个空字节来引发从设备的传输。


SPI接口一般使用4条线通讯,单向传输时也可以使用3条线,其中3条线为SPI总线(MISO,MOSI,SCLK),1条为SPI片选信号线(CS),它们的作用如下:


*MISO:主设备数据输入,从设备数据输出


*MOSI:主设备数据输出,从设备数据输入


*SCLK:时钟信号,由主设备产生


*CS:从设备片选信号,由主设备控制


SPI使用MOSI/MISO信号线来传输数据,使用SCLK信号线进行数据同步。MOSI/MISO数据线在SCLK的每个时钟周期传输1位数据,且数据输入输出是同时进行的。数据传输时,MSB先行或LSB先行没有硬性规定,但是两个SPI通讯设备之间必须使用同样的协定,一般都会采用MSB先行模式。


当有多个SPI从设备与SPI主设备相连时,设备的MOSI/MISO/SCLK信号线并联到相同的SPI总线上,即无论有多少个从设备,都共同使用者3条总线;而每个从设备都有独立的1条CS信号线,该信号线独占主设备的一个引脚,即有多少个从设备就有多少条片选信号线。当主设备要选择从设备时,把该从设备的CS信号线设置为低电平,该从设备即被选中(片选有效),接着主设备开始与从设备进行SPI通讯。


SPI总线根据时钟极性(CPOL)和时钟相位(CPHA)的配置不同,可以有四种工作方式:

*MISO:主设备数据输入,从设备数据输出


*MOSI:主设备数据输出,从设备数据输入


*SCLK:时钟信号,由主设备产生


*CS:从设备片选信号,由主设备控制



1.2 W25QXX芯片简介

W25QXX芯片是华邦公司推出的大容量SPI FLASH产品,该系列有W25Q16/32/62/128等。本例程使用W25Q64,W25Q64容量为64Mbits(8M字节):8MB的容量分为128个块(Block)(块大小为64KB),每个块又分为16个扇区(Sector)(扇区大小为4KB);W25Q64的最小擦除单位为一个扇区即4KB,因此在选择芯片的时候必须要有4K以上的SRAM(可以开辟4K的缓冲区)。W25Q64的擦写周期多达10万次,具有20年的数据保存期限。下表是W25QXX的常用命令表

2.硬件设计

D1指示灯用来提示系统运行状态,K_UP按键用来控制W25Q64数据写入,K_DOWN按键用来控制W25Q64数据读取,串口1用来打印写入和读取的数据信息


*指示灯D1


*USART1串口


*W25Q64


*K_UP和K_DOWN按键


3.软件设计

3.1 STM32CubeMX设置

➡️ RCC设置外接HSE,时钟设置为72M

➡️ PC0设置为GPIO推挽输出模式、上拉、高速、默认输出电平为高电平

➡️ USART1选择为异步通讯方式,波特率设置为115200Bits/s,传输数据长度为8Bit,无奇偶校验,1位停止位

➡️ PA0设置为GPIO输入模式、下拉模式;PE3设置为GPIO输入模式、上拉模式

➡️ PG13设置为GPIO推挽输出模式、上拉、高速(片选引脚)

➡️ 激活SPI2,不开启NSS,数据长度8位,MSB先输出,分频因子256,CPOL为HIGH,CPHA为第二个边沿,不开启CRC检验,NSS为软件控制

➡️输入工程名,选择路径(不要有中文),选择MDK-ARM V5;勾选Generated periphera initialization as a pair of ‘.c/.h’ files per IP ;点击GENERATE CODE,生成工程代码


3.2 MDK-ARM软件编程


➡️ 在spi.c文件下可以看到SPI2的初始化函数,片选管脚的初始化在gpio.c中


void MX_SPI2_Init(void){

  hspi2.Instance = SPI2;

  hspi2.Init.Mode = SPI_MODE_MASTER;//设置为主模式

  hspi2.Init.Direction = SPI_DIRECTION_2LINES;//双线模式

  hspi2.Init.DataSize = SPI_DATASIZE_8BIT;//8位数据长度

  hspi2.Init.CLKPolarity = SPI_POLARITY_HIGH;//串行同步时钟空闲状态为高电平

  hspi2.Init.CLKPhase = SPI_PHASE_2EDGE;//第二个跳变沿采样

  hspi2.Init.NSS = SPI_NSS_SOFT;//NSS软件控制

  hspi2.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_256;//分配因子256

  hspi2.Init.FirstBit = SPI_FIRSTBIT_MSB;//MSB先行

  hspi2.Init.TIMode = SPI_TIMODE_DISABLE;//关闭TI模式

  hspi2.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE;//关闭硬件CRC校验

  hspi2.Init.CRCPolynomial = 10;

  if (HAL_SPI_Init(&hspi2) != HAL_OK){

    Error_Handler();

  }

}


void HAL_SPI_MspInit(SPI_HandleTypeDef* spiHandle){

  GPIO_InitTypeDef GPIO_InitStruct = {0};

  if(spiHandle->Instance==SPI2){

  __HAL_RCC_SPI2_CLK_ENABLE();  

  __HAL_RCC_GPIOB_CLK_ENABLE();

  /**SPI2 GPIO Configuration    

  PB13     ------> SPI2_SCK

  PB14     ------> SPI2_MISO

  PB15     ------> SPI2_MOSI */

  GPIO_InitStruct.Pin = GPIO_PIN_13|GPIO_PIN_15;

  GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;

  GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;

  HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);

  GPIO_InitStruct.Pin = GPIO_PIN_14;

  GPIO_InitStruct.Mode = GPIO_MODE_INPUT;

  GPIO_InitStruct.Pull = GPIO_NOPULL;

  HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);

  }

}


➡️ 创建按键驱动文件key.c 和相关头文件key.h,参考按键输入例程


➡️ 创建包含W25Q64芯片的相关操作函数及驱动函数的文件w25qxx.c和w25qxx.h,这里仅介绍几个重要的函数,源文件下载方式见文末介绍


//这里仅介绍几个重要的函数

void W25QXX_Init(void){

  W25Qx_Disable();

  W25QXX_TYPE = W25QXX_ReadID();//读取芯片ID

  printf("FLASH ID:%Xrn",W25QXX_TYPE);

  if(W25QXX_TYPE == 0xc816)

    printf("FLASH TYPE:W25Q64rn");

}


uint16_t W25QXX_ReadID(void){

  uint16_t ID;

  uint8_t id[2]={0};

  uint8_t cmd[4] = {W25X_ManufactDeviceID,0x00,0x00,0x00};//读取ID命令

  W25Qx_Enable();//使能器件

  HAL_SPI_Transmit(&hspi2,cmd,4,1000);

  HAL_SPI_Receive(&hspi2,id,2,1000);

  W25Qx_Disable();//取消片选

  ID = (((uint16_t)id[0])<<8)|id[1];

  return ID;

}


void W25QXX_Read(uint8_t* pBuffer,uint32_t ReadAddr,uint16_t NumByteToRead){

  uint8_t cmd[4] = {0};

  cmd[0] = W25X_ReadData;//读取命令

  cmd[1] = ((uint8_t)(ReadAddr>>16));

  cmd[2] = ((uint8_t)(ReadAddr>>8));

  cmd[3] = ((uint8_t)ReadAddr);


  W25Qx_Enable();//使能器件

  HAL_SPI_Transmit(&hspi2,cmd,4,1000);

  HAL_SPI_Receive(&hspi2,pBuffer,NumByteToRead,1000);

  W25Qx_Disable();//取消片选

}


void W25QXX_Write(uint8_t* pBuffer,uint32_t WriteAddr,uint16_t NumByteToWrite){

  uint32_t secpos;

  uint16_t secoff;

  uint16_t secremain;

  uint16_t i;

  uint8_t *W25QXX_BUF;

  W25QXX_BUF = W25QXX_BUFFER;

  secpos = WriteAddr/4096; //扇区地址

  secpos = WriteAddr%4096; //在扇区里的偏移

  secremain = 4096-secoff; //扇区剩余空间大小

  printf("WriteAddr:0x%X,NumByteToWrite:%drn",WriteAddr,NumByteToWrite);

  if(NumByteToWrite <= secremain)  //不大于4K字节

    secremain = NumByteToWrite;

  while(1){

    W25QXX_Read(W25QXX_BUF,secpos*4096,4096);//读取整个扇区内容

    for(i=0;i      if(W25QXX_BUF[secoff+i] != 0xff)//需要擦除

break;

    }

    if(i < secremain){//需要擦除

      W25QXX_Erase_Sector(secpos);//擦除扇区

      for(i=0;i W25QXX_BUF[i+secoff] = pBuffer[i];

      }

      W25QXX_Write_NoCheck(W25QXX_BUF,secpos*4096,4096);//写入整个扇区

    }

    else{

      W25QXX_Write_NoCheck(pBuffer,WriteAddr,secremain);//写入扇区剩余空间

    }

    if(NumByteToWrite == secremain){//写入结束了

      break;

    }

    else{ //写入未结束

      secpos++; //扇区地址增1

      secoff = 0; //偏移位置为0

      pBuffer += secremain; //指针偏移

      WriteAddr += secremain; //写地址偏移

      NumByteToWrite -= secremain;//字节数递减

      if(NumByteToWrite > 4096)

secremain = 4096; //下个扇区还没是写不完

      else

secremain = NumByteToWrite;//下个扇区可以写完了

    }

  }

}


➡️ 在main.c文件下编写SPI测试代码


/* USER CODE BEGIN PV */

uint8_t wData[0x100];

uint8_t rData[0x100];

uint32_t i;

/* USER CODE END PV */

int main(void){

  /* USER CODE BEGIN 1 */

  uint8_t key;

  /* USER CODE END 1 */

  HAL_Init();

  SystemClock_Config();

  MX_GPIO_Init();

  MX_SPI2_Init();

  MX_USART1_UART_Init();

  /* USER CODE BEGIN 2 */

  W25QXX_Init();

  for(i=0;i<0x100;i++){

    wData[i] = i;

    rData[i] = 0;

  }

  /* USER CODE END 2 */

  while (1){

    key = KEY_Scan(0);

    if(key == KEY_UP_PRES){

      printf("KEY_UP_PRES write data...rn");

      W25QXX_Erase_Sector(0);

      W25QXX_Write(wData,0,256);

    }

    if(key == KEY_DOWN_PRES){

      printf("KEY_DOWN_PRES read data...rn");

      W25QXX_Read(rData,0,256);

      for(i=0;i<256;i++){

printf("0x%02X ",rData[i]);

      }

    }

    HAL_GPIO_TogglePin(GPIOC,GPIO_PIN_0);

    HAL_Delay(200);

  }

}


4.下载验证

编译无误下载到开发板后,可以看到D1指示灯不断闪烁,当按下K_UP按键后数据写入到W25Q64芯片内,当按下K_DOWN按键后读取W25Q64芯片的值,同时串口打印出相应信息

推荐阅读

史海拾趣

DEVCON公司的发展小趣事

在1950年代,电子工业正处于蓬勃发展的初期。阿尔·克瑞顿,一位富有创新精神的化学家,在美国马萨诸塞州丹佛斯市的一个实验室里,经过无数次的试验,终于研发出了一种名为Devcon可塑钢修补剂的新型材料。这种材料经济实惠、不易燃,且能够作为焊接的替代品,迅速在电子制造和维修领域引起了广泛关注。这就是DEVCON公司的起点,一个由创新精神和市场需求共同点燃的火花。

EUDYNA公司的发展小趣事

随着技术的不断进步和市场需求的日益增长,EUDYNA开始实施全球化战略。公司不仅在全球范围内建立了多个研发中心和生产基地,还积极拓展海外市场,与众多国际知名企业建立了紧密的合作关系。EUDYNA的全球化战略不仅为公司带来了更广阔的市场空间,也进一步提升了其品牌影响力和竞争力。

Burr-Brown公司的发展小趣事

2000年,Burr-Brown被著名的美国德州仪器公司(Texas Instruments)收购,成为其高性能模拟器件部门的一部分。这一收购为Burr-Brown提供了更强大的资源和支持,使其能够继续专注于模拟器件(如ADC、DAC等)的生产和创新。在德州仪器的引领下,Burr-Brown持续保持技术领先,为全球电子行业的发展做出了重要贡献。

这五个故事共同展现了Burr-Brown公司从创立到发展壮大的历程,以及其在电子行业中的卓越成就和持续创新的精神。通过不断适应市场变化和技术进步,Burr-Brown成功地在竞争激烈的电子行业中脱颖而出,成为一家备受尊敬的企业。

General Instrument Corp公司的发展小趣事
验证输入信号是否在变换器的允许范围内,避免超出其最大或最小输入电压。
Churod Electronics Co Ltd公司的发展小趣事

中汇瑞德电子股份有限公司的创业之路并非一帆风顺。2006年,公司创始人周康平凭借对电子行业的深刻理解和前瞻视野,决定涉足继电器制造领域。当时,市场竞争激烈,技术门槛高,但周康平坚信,只要专注于技术研发和产品质量,就能在市场中脱颖而出。他带领团队日夜奋战,攻克了一个又一个技术难题,逐渐在继电器制造领域站稳了脚跟。

Equator Technologies公司的发展小趣事

随着智能手机市场的迅速崛起,Equator敏锐地捕捉到了这一市场机会。从XX年代后期开始,公司逐渐将业务重心转向手机图像处理领域。通过与手机厂商的合作,Equator成功地将其先进的视频处理技术应用于智能手机中,显著提升了手机的显示效果和用户体验。这一转型不仅为Equator带来了丰厚的回报,也进一步巩固了其在电子行业的地位。

问答坊 | AI 解惑

创意十足的[蜡烛]是不是你说了算

USB 精油蜡烛 蜡烛是营造气氛的好东西,但是乱滴的蜡油和引发火灾的可能性,让科科们敬而远之。没关系,现在有兼顾安全与便利的产品出现啦,这款 USB 烛光灯座是浪漫的好选择,采用 USB 或两颗3号电池供电,LED 模拟的烛火相当逼真,更妙的是要熄 ...…

查看全部问答>

【FPGA技术问题】关于区域约束(Area Constraints)

  nArea Constraints是Floorplanner最容易、最有效的应用 n大型设计首选布局工具- Floorplanner p在综合中,为了防止单独的component名称被改变,选择“Keep Hierarchy”参数选项 p设计的每个组成部分可以被约束限定到某一个区域 p ...…

查看全部问答>

求手机X01HT厂商开发包

我们打算在X01HT这款手机上进行软件开发,有谁知道哪里有厂商开发包提供(收费的也可以), 可以通知我,非常感谢!QQ:176609958   邮箱:mchsh1985@126.com …

查看全部问答>

s3c2410的一个地址映射问题!

在看一个步进电机的驱动程序,有个问题: 步进电机的物理地址是0x28000006,对应的虚拟地址是0xd3000006,不懂的是这个是怎么计算出来的? 32位有4G的虚拟地址,我只知道是怎么进行内存映射的,但是对于GPIO口怎么映射的,不怎么懂! 请指教,谢谢!…

查看全部问答>

免费风暴来袭 - 快来申请 MSP430 LaunchPad 开始您的设计之旅吧!

德仪社区搞的活动,今天开始了哦 http://www.deyisupport.com/question_answer/f/55/t/6227.aspx…

查看全部问答>

請問F28335的DMA可以接收UART嗎?

我用的是板子F28335 ezDSP 3組SCI都被使用了 因為工作環境震動很大  所有的電路板都會被震動到 板子已經固定不知道怎麼再固定了... 因為GPS的資料一直發生 收不到的情形 (震動很大的時候資料就斷了) 後來實驗結果發現 SCIC RX 的PI ...…

查看全部问答>

世界50强招聘汽车动力部件研发项目工程师人才

职位:项目工程师 地点:上海 职责:负责的项目为汽车动力总成零部件,电机、逆变器等;协助项目经理管理项目进程(公司设置的项目经理主要是面向客户的,所以项目工程师要在研发的技术等问题上协助项目经理)。 要求:1、有motor、inventor等类 ...…

查看全部问答>

拿到开发板后第一周的心得,提交

很开心能拿到Helper2416的开发板,从拿到开发板初,就准备全身心的投入进去进行相应的学习,要学习就要有个学习的方法不是,咱是摸着石头过河。OK,先进行资料的整理了,先从BBS的贴子入手吧,https://bbs.eeworld.com.cn/thread-441984-1-1.html从 ...…

查看全部问答>

发帖是回复可见好,还是完全公开好?

本人每天都会花点时间逛各种论坛,看帖子,学技术。本人比较低调,基本上不喜欢发帖子,最近在这个电子工程师论坛逛了几次,看了很多帖子,发现,这里面的大部分人都有一个通病,发帖子必须回复能看。本是一个技术交流的论坛,被你们这帮人搞得都是 ...…

查看全部问答>