当前位置: 首页 > news >正文

HC32F460 DMA的链式传输(SPI主机+DMA发送/接收)

1、SPI主机+DMA发送

SPI的发送操作在实现上与串口发送非常相似,尤其在使用DMA进行数据传输时,其配置流程和数据搬运机制是基本一致的,因此在此不再重复介绍DMA的具体配置细节。

唯一需要特别注意的是,在SPI发送过程中,主设备通过片选信号(CS)来控制通信时序。需要正确的输出片选信号。

发送部分代码如下

1. //SPI1主机 只发送 2. void SPI1_SendData(uint8_t *pu8Data, uint16_t u16Len) 3. { 4. uint32_t Timeout = 0xfffffff0; 5. while(SPI_GetStatus(CM_SPI1,SPI_FLAG_IDLE) == 0) //等待总线空闲 6. { 7. Timeout--; 8. if (Timeout-- == 0) 9. { 10. return; 11. } 12. }; 13. Wait_DMA_Ready(SPI1_TX_DMA_UNIT, SPI1_TX_DMA_CH, (uint32_t)(&CM_SPI1->DR)); // 等待DMA准备就绪 14. GPIO_ResetPins(SPI1_SS0_PORT, SPI1_SS0_PIN); //拉低片选 15. if (u16Len == 1) //如果只有一个数据,直接发送 16. { 17. SPI_WriteData(CM_SPI1, *pu8Data); 18. while(SPI_GetStatus(CM_SPI1,SPI_FLAG_IDLE) == 0) //等待发送完成 19. { 20. Timeout--; 21. if (Timeout-- == 0) 22. { 23. return; 24. } 25. }; 26. 27. GPIO_SetPins(SPI1_SS0_PORT, SPI1_SS0_PIN); //拉高片选 28. return; 29. } 30. 31. //多个数据传输 32. DMA_SetTransCount(SPI1_TX_DMA_UNIT, SPI1_TX_DMA_CH, u16Len - 1); // 设置DMA传输长度 33. DMA_SetSrcAddr(SPI1_TX_DMA_UNIT, SPI1_TX_DMA_CH, (uint32_t)(pu8Data + 1)); // 设置DMA源地址跳过第一个字节 34. DMA_SetDestAddr(SPI1_TX_DMA_UNIT, SPI1_TX_DMA_CH, (uint32_t)(&CM_SPI1->DR)); // 设置DMA目的地址 35. SPI_WriteData(CM_SPI1, *pu8Data); // 发送第一个字节 36. while(SPI_GetStatus(CM_SPI1,SPI_FLAG_IDLE) == 0) //等待总线空闲 37. { 38. Timeout--; 39. if (Timeout-- == 0) 40. { 41. return; 42. } 43. }; 44. GPIO_SetPins(SPI1_SS0_PORT, SPI1_SS0_PIN); //拉高片选 45. }

2、SPI主机+DMA接收

SPI主机的接收机制与串口接收存在一定差异。在SPI通信中,主机的接收操作总是伴随着发送操作进行的,也就是说,数据的接收是通过发送过程中的时钟同步完成的。因此,接收到的数据长度在通信开始前就是已知的,这一点为接收流程的简化提供了便利。

基于这一特点,可以在SPI主机发送逻辑的基础上,直接整合接收处理代码,实现发送与接收的同步控制。具体来说:

  • 在启动SPI传输之前,预先配置好接收DMA的起始地址和接收数据长度;
  • 随后启动SPI主机传输,整个接收过程将由DMA自动完成;
  • 由于接收长度已知,且接收与发送同步进行,因此在整个数据传输过程中,无需手动干预DMA通道的使能状态(CHEN),即可保证接收数据的完整性和正确性。

代码如下

//SPI1发送&接收数据 void SPI1_Send_Receive_Data(uint8_t *SendData, uint8_t *RevData,uint16_t u16Len) { uint32_t Timeout = 0xfffffff0; while(SPI_GetStatus(CM_SPI1,SPI_FLAG_IDLE) == 0) //等待总线空闲 { Timeout--; if (Timeout-- == 0) { return; } }; Timeout = 0; Wait_DMA_Ready(SPI1_TX_DMA_UNIT, SPI1_TX_DMA_CH, (uint32_t)(&CM_SPI1->DR));//等待DMA就绪 GPIO_ResetPins(SPI1_SS0_PORT, SPI1_SS0_PIN); //拉低片选 if (u16Len == 1) //如果只有一个数据,直接发送 { SPI_WriteData(CM_SPI1, *SendData); while(SPI_GetStatus(CM_SPI1,SPI_FLAG_IDLE) == 0) //等待发送完成 { Timeout--; if (Timeout-- == 0) { return; } }; //拉高片选 GPIO_SetPins(SPI1_SS0_PORT, SPI1_SS0_PIN); return; } //设置发送DMA参数 DMA_SetTransCount(SPI1_TX_DMA_UNIT, SPI1_TX_DMA_CH, u16Len - 1); // 设置DMA传输长度 DMA_SetSrcAddr(SPI1_TX_DMA_UNIT, SPI1_TX_DMA_CH, (uint32_t)(SendData + 1)); // 设置DMA源地址(跳过第一个字节) DMA_SetDestAddr(SPI1_TX_DMA_UNIT, SPI1_TX_DMA_CH, (uint32_t)(&CM_SPI1->DR)); // 设置DMA目的地址 //设置接收DMA参数 DMA_SetDestAddr(SPI1_RX_DMA_UNIT, SPI1_RX_DMA_CH, (uint32_t)RevData); // 设置DMA目的地址 DMA_SetTransCount(SPI1_RX_DMA_UNIT, SPI1_RX_DMA_CH, u16Len); // 设置DMA传输长度 SPI_WriteData(CM_SPI1, *SendData); // 发送第一个字节 while(SPI_GetStatus(CM_SPI1,SPI_FLAG_IDLE) == 0) // { Timeout--; if (Timeout-- == 0) { return; } }; GPIO_SetPins(SPI1_SS0_PORT, SPI1_SS0_PIN); }
http://www.jsqmd.com/news/100233/

相关文章:

  • 常见安全设备理解
  • Transformer模型完全指南:从零开始学习大模型架构【收藏必学】
  • Windows下部署EmotiVoice语音合成全指南
  • 电机控制器程序就是新能源汽车的“灵魂操纵师“,这玩意儿直接决定了车子加速时是窜得像猎豹还是肉得像乌龟。今天咱们扒开控制器的外壳,看看代码层到底在搞什么飞机
  • 23、网络基础:IP地址、子网掩码与FreeBSD网络配置
  • 3天内搭建可商用的开源AI
  • AI纪元2025终章:开源革命、监管铁幕与人类主体性的觉醒
  • LobeChat能否支持AR/VR交互?三维空间对话界面畅想
  • Excalidraw:开源手绘风白板绘图工具
  • 国产大模型横评:从Kimi到Qwen,哪款最适合程序员?
  • 腾讯混元HunyuanVideo-Foley:声画合一的视频音效革命
  • 自动紧急制动系统仿真实战手记
  • Gradle配置groovy增量编译
  • 腾讯云国际站代理商的TAPD有什么优势呢?
  • AI模型训练入门指南:手把手教你构建自己的智能模型
  • EmotiVoice开源TTS项目结构与配置详解
  • 数据访问:MyBatisMybatis-Plus
  • 中小企业的营销“暖心伙伴”——北京易美之尚,让增长不再难
  • openFuyao AI推理加速方案深度解析
  • 2025年安徽靠谱交通事故律师事务所排行榜,口碑好的交通事故 - mypinpai
  • 钢铁厂除氧供气 / 炉门驱动用工业级螺杆空压机​选型注意
  • PaddlePaddle模型服务化部署:配合HTML界面实现可视化推理
  • 如何提升银包铜的抗氧化性?
  • 博客管理系统测试报告
  • HC32F460 DMA的链式传输(SPI从机+DMA发送/接收)
  • 毅硕HPC | NVIDIA DGX Spark 万字硬核评测:将AI超级工厂带上桌面
  • 大模型Agent强化学习完全指南:从PPO到GRPO的工具使用技术解析
  • 新国标电动车爬坡困境:当限速25km/h遭遇安全危机,无责伤亡谁来买单?
  • 腾讯云国际站代理商的定制化技术支持服务的成功案例有哪些?
  • VonaJS是如何做到文件级别精确HMR(热更新)的?