当前位置:   article > 正文

关于STM32的SPI使用(中断方式)的探讨_stm32 spi中断接收

stm32 spi中断接收

还没有用过STM32的SPI功能, 所以想尝试着做做看.

以前做串口通信都是用中断方式做的, 所以做SPI通信, 首先想到的就是用中断方式做, 网上有一些例程, 但是好像也有没有解释的很清楚的, 至少我没有理解.

以下我将从自己的认知来写一下, 大神绕过, 小白可以看看, 来看看是否有你自己的盲点, 本文也只是描述我自己碰到的问题的点, 不包含整个代码.

1. 关于GPIO口模式的设置
输出口没有什么可说的, 都是GPIO_Mode_AF_PP, 但是输入口模式, 需要注意, 网上有的教程是让设置成GPIO_Mode_IN_FLOATING, 实测这个模式不行, 需要使用GPIO_Mode_IPU模式, 使用floating模式时, 我监控接收到的数跟主机发送过来的数有出入, 有时候不一样, 改成IPU模式, 没有异常. 不要轻信网上的一些帖子, 别拿他们的当教条!! 要持怀疑态度.


以下是我初始化SPI的代码

  1. //因为SPI1的输出要靠PA5&PA6&PA7来完成, 所以须先初始化PA5&PA6&PA7的工作模式
  2. //注意SPI引脚是否和JATG口或者SWD口复用.
  3. //先初始化引脚的模式
  4. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //初始化管脚的工作模式之前不写这一句也可以
  5. //初始化管脚
  6. //PA5_SPI1_CLK
  7. PAx.GPIO_Pin = GPIO_Pin_5;
  8. PAx.GPIO_Mode = GPIO_Mode_IPU;//GPIO_Mode_IN_FLOATING;
  9. //PAx.GPIO_Mode = GPIO_Mode_IN_FLOATING;//GPIO_Mode_IN_FLOATING;
  10. PAx.GPIO_Speed = GPIO_Speed_50MHz;
  11. GPIO_Init(GPIOA, &PAx);
  12. //PA6_SPI1_MISO
  13. PAx.GPIO_Pin = GPIO_Pin_6;
  14. PAx.GPIO_Mode = GPIO_Mode_AF_PP;//GPIO_Mode_Out_PP;
  15. PAx.GPIO_Speed = GPIO_Speed_50MHz;
  16. GPIO_Init(GPIOA, &PAx);
  17. //PA7_SPI1_MOSI
  18. PAx.GPIO_Pin = GPIO_Pin_7;
  19. PAx.GPIO_Mode = GPIO_Mode_IPU;//GPIO_Mode_IN_FLOATING;
  20. //PAx.GPIO_Mode = GPIO_Mode_IN_FLOATING;//GPIO_Mode_IN_FLOATING;
  21. PAx.GPIO_Speed = GPIO_Speed_50MHz;
  22. GPIO_Init(GPIOA, &PAx);
  23. //打开外设时钟
  24. RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);
  25. //初始化SPI相关寄存器
  26. SPIxInit.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
  27. SPIxInit.SPI_Mode = SPI_Mode_Slave;
  28. SPIxInit.SPI_DataSize = SPI_DataSize_8b;
  29. SPIxInit.SPI_CPOL = SPI_CPOL_Low;
  30. SPIxInit.SPI_CPHA = SPI_CPHA_1Edge;
  31. SPIxInit.SPI_NSS = SPI_NSS_Soft;
  32. SPIxInit.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_16;
  33. SPIxInit.SPI_FirstBit = SPI_FirstBit_MSB;
  34. SPI_Init(SPI1, &SPIxInit);

2. 关于SPI的发送中断
这里说下SPI的发送中断, 主要是主机的SPI, 这个中断很有意思, SPI_I2S_IT_TXE, 这个中断叫做"发送缓冲区空中断", 仔细理解这个名字, 当发送缓冲区为空时, 就会产生这个中断, 所以, 如果当在初始化SPI时, 直接打开这个中断, 程序会直接跑到SPI的中断服务程序中, 一直跳不出来, 原因就是此时SPI发送寄存器一直是空的, 所以, 感觉这个中断只有在要发送数据以后打开, 并且在发送完一帧数据后, 在中断服务程序中把该中断关闭. 至于那个标志 清不清的无所谓了, 反正也要自己做判断

3. 关于SPI的接收中断
也不知道是我自己理解不对, 还是作者就这么设计的这个中断, 这个中断的名称为:SPI_I2S_IT_RXNE, 叫做"接收缓冲区非空中断", 也需要自己理解这个中断, 个人实际应该叫接收缓冲区填满中断, 这个不是重点, 重点是如何清除这个标志, 其它一些外设清中断标志位是使用XXXX_ClearITPendingBit()这个函数, 但是这里不能这么用, 这里的用法是使用SPI_I2S_ReceiveData()读取缓冲区里面的数据, 使用完该函数后, 该中断标志自动清除, 经测试SPI_I2S_ClearITPendingBit(SPI1, SPI_I2S_IT_RXNE);这个是无效的.

总结: 我认为值得分享的就这三条, 其它都是常规操作, 以上内容来源于STM32的使用手册和实践

有感兴趣的可以留言交流

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/酷酷是懒虫/article/detail/843918
推荐阅读
相关标签
  

闽ICP备14008679号