热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

关于c:STM32发现F3SPI环回RXFIFO收不到数据

STM32discoveryF3SPIloopbackRXFIFOreceivesnodata


STM32 discovery F3 SPI loopback RXFIFO receives no data


我正在使用 STM32 F3 发现套件并开始弄乱 SPI 外围设备。我从一个简单的环回系统开始:我检查 TXFIFOLVL 状态,如果它未满,我将数据发送到 DR 寄存器,然后应该环回我的 RxBuffer(当 RXFIFOLVL 不为空时,我从 DR 读取数据),但我遇到了一个问题——我的接收缓冲区没有得到任何东西,我似乎不明白为什么。我不使用 HAL 或标准外设库,所以我配置 SPI 并通过如下寄存器值使用它:

SPI 代码的头文件:










1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24



#define GPIOA_ENABLE                    0b1<<17             // Enable GPIO port A clock in AHBENR register
#define SPI1_CLOCK_ENABLE               0b1<<12             // Enable SPI1 clock in APB2ENR register
#define SPI1_PIN_ALT_FNC                0b1010<<4           // Sets PA5,PA6 & PA7 to Alternative function
#define SPI1_OUTPUT_TYPE                ~(0b111<<5)         // Sets PA5, PA6 & PA7 to push-pull
#define SPI1_PIN_SPEED                  0b1111<<4           // Sets pins from 4 to 7 to work on 50 MHz output speed
#define SPI1_PIN_ALT_FNC_LOW            0b0101<<4           // Sets the Alternative function to AF5 in alternative function low register
#define SPI1_PIN_ALT_FNC_HIGH           0b0101<<4           // Sets the Alternative function to AF5 in alternative function high register
#define SPI1_BAUDRATE_PRESCALER_2       0b000<<3            // F_PCLK/2
#define SPI1_BAUDRATE_PRESCALER_128     0b110<<3            // F_PCLK/128
#define SPI1_MASTER_MODE                0b1<<2              // Sets the SPI1 to master mode
#define SPI1_PERI_ENABLE                0b1<<6              // Enable the SPI peripheral
#define SPI1_SSM_ENABLE                 0b1<<9              // Enable SPI software slave management
#define SPI1_SSI_ENABLE                 0b1<<8              // SPI1 internal slave select
#define SPI1_NSSP_ENABLE                0b1<<3              // Enable NSS pulse management
#define SPI1_FRXTH_8BIT                 0b1<<12             //Set the FIFO reception threshold to 8 bits
#define SPI1_DATA_SIZE                  0b0111<<8           // SPI1 DATA size
#define SPI1_TXFIFO_FULL_FLAG           0b11<<11            // SPI1 Tx FIFO transmission flag
#define SPI1_RXFIFO_EMPTY_FLAG          0b00<<9             // SPI1 Rx FIFO reception flag

#include"main.h"
#include"stm32f3xx_hal.h"

void spi_init();
void spi_WriteRead(uint8_t *rxBuffer, uint8_t *txBuffer, uint8_t bufferSize);



SPI 代码的代码文件:










1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48



#include"SPI_toSD.h"

/* SPI1 configuration

 * PA5 - SCK

 * PA6 - MISO

 * PA7 - MOSI

 */

void spi_init(){

// Start the GPIO and peripheral clocks in Reset and Clock Control register

RCC->AHBENR |= GPIOA_ENABLE;

RCC->APB2ENR |= SPI1_CLOCK_ENABLE;

// Configure the GPIOs for SPI communication

GPIOA->MODER |= SPI1_PIN_ALT_FNC;

GPIOA->OTYPER &= SPI1_OUTPUT_TYPE;

GPIOA->OSPEEDR |= SPI1_PIN_SPEED;

GPIOA->AFR[0] |= SPI1_PIN_ALT_FNC_LOW;

GPIOA->AFR[1] |= SPI1_PIN_ALT_FNC_HIGH;

// Configure the SPI peripheral

SPI1->CR1 |= SPI1_BAUDRATE_PRESCALER_2;

SPI1->CR1 |= SPI1_SSM_ENABLE;

SPI1->CR1 |= SPI1_MASTER_MODE;

SPI1->CR1 |= SPI1_SSI_ENABLE;

SPI1->CR2 |= SPI1_DATA_SIZE;

SPI1->CR2 |= SPI1_FRXTH_8BIT;

SPI1->CR2 |= SPI1_NSSP_ENABLE;

SPI1->CR1 |= SPI1_PERI_ENABLE;

SPI1->CR1 &= ~SPI1_SSI_ENABLE;

}

void spi_WriteRead(uint8_t *rxBuffer, uint8_t *txBuffer, uint8_t bufferSize){
int i;
while((SPI1->SR & 0b11<<11)==SPI1_TXFIFO_FULL_FLAG);
for(i=0;i<bufferSize;i++){

        SPI1->DR |= *txBuffer;  // send *txBuffer++

        txBuffer++;

    while((SPI1->SR & 0b11<<9)!=SPI1_RXFIFO_EMPTY_FLAG){

        *rxBuffer = SPI1->DR;

        rxBuffer++;

    }
}

}



在 main 中,我简单地定义我的缓冲区并像这样初始化它们:










1
2



uint8_t rx_buff[SIZE] = {0,0,0,0,0,0,0,0,0,0};
uint8_t tx_buff[SIZE] = {1,2,3,4,5,6,7,8,9,10};


所以在调用我的 spi_WriteRead() 函数之后,我自然希望这些缓冲区具有相同的值。

我调用我的 spi_init() 函数并在我的 while 循环中调用 spi_WriteRead() 函数:










1


  spi_WriteRead(rx_buff,tx_buff,SIZE);


SIZE 在我的 main.c 中定义为:










1


#define SIZE  10


我使用 SW4STM32 环境进行编码和调试,所以在我的调试器中我可以看到所有的寄存器值。我的 SPI 就像我定义的那样被初始化,我的数据被发送到 TXFIFO 缓冲区,但没有任何东西进入 RXFIFO 缓冲区。如果我检查 SPI SR 寄存器,我可以看到我的 TXFIFO 已满,但 RXFIFO 标志说它是空的。

有没有人知道我可能做错了什么?我是否严重误解了关于 SPI 的一些简单信息?感谢您的输入!



相关讨论




  • 您是否使用示波器/逻辑分析仪验证了 SPI 的物理线路?


  • 是的,我刚刚用 PicoScope 进行了检查,它似乎发送了乱码,这可以解释 TXFIFO 填充并保持满,而 RXFIFO 保持空。


  • 如果有什么理由不使用好的库?但是,这里有来自 keil keil.com/dd2/pack 的非常方便的 io 驱动程序。






编辑:

好好看看这里:










1
2
3
4



#define SPI1_SSI_ENABLE                 0b1<<8

...
SPI1->CR1 |= SPI1_PERI_ENABLE;

SPI1->CR1 &= ~SPI1_SSI_ENABLE;


现在您可能知道为什么 #define 宏通常被认为是一个坏主意。如果您使用 stm32f3xxx.h 标头中的 #define 值,则不会遇到此问题,因为所有带有操作的值都在那里有括号。你没有它们。这就是为什么您的代码在编译器中看起来像这样:










1
2


SPI1->CR1 |= SPI1_PERI_ENABLE;

SPI1->CR1 &= ~0b1<<8;


相当于:










1
2


SPI1->CR1 |= SPI1_PERI_ENABLE;

SPI1->CR1 &= (~0b1)<<8;


更进一步:










1
2


SPI1->CR1 |= SPI1_PERI_ENABLE;

SPI1->CR1 &= 0xffffff00;


可能不是你想要的。

您还应该知道,如果您的设备是主设备,那么 SSI 和 SSM 位都应该设置。 https://stackoverflow.com/a/42169600/157344

原件:

请注意,在这些设备中,当您直接访问 SPI1->DR 时,您会一次发送/接收两个字节。那是因为这个寄存器被定义为 uint16_t 并且 SPI 支持所谓的"数据打包"(在参考手册中搜索它)。如果您想一次发送/接收一个字节,那么您需要像这样将寄存器转换为写入和读取:










1
2


readByte = (volatile uint8_t*)SPI1->DR;
(volatile uint8_t*)SPI1->DR = writeByte;


顺便说一句 - 为什么不使用 CMSIS 标头提供的#defines?您不必定义诸如 SPI1_MASTER_MODE...

之类的东西



相关讨论




  • 谢谢你的提示,我不知道。必须在参考手册中浏览过它。因此,如果我真的不在乎数据是作为 WORD 还是作为 BYTE 发送的,我也可以将指针增加 2 而不是 1,对吗?同样在调试器中,我注意到 spi_init() 完成后 CR1 中的 SPI 启用字段设置为 0,这是正常行为还是应该调查它?


  • @saukijan - 您可以一次发送 8 位或 16 位数据,但要增加指针,您还必须从缓冲区读取/写入 2 个字节。当前,您读取一个字节并将其作为 16 位 0x00?? 写入 DR 寄存器。一般来说,新的 STM32 设备中的 SPI 可能会很棘手......


  • @saukijan - 看看编辑后的答案。它现在应该可以解决您的(主要)问题。


  • 感谢您的详细解释,现在它是有道理的。我将对其进行测试并稍后报告结果。


  • 是的,括号是问题所在。下次我会知道使用 CMSIS 定义。谢谢你的帮助!










推荐阅读
  • 微软头条实习生分享深度学习自学指南
    本文介绍了一位微软头条实习生自学深度学习的经验分享,包括学习资源推荐、重要基础知识的学习要点等。作者强调了学好Python和数学基础的重要性,并提供了一些建议。 ... [详细]
  • 如何自行分析定位SAP BSP错误
    The“BSPtag”Imentionedintheblogtitlemeansforexamplethetagchtmlb:configCelleratorbelowwhichi ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 本文介绍了P1651题目的描述和要求,以及计算能搭建的塔的最大高度的方法。通过动态规划和状压技术,将问题转化为求解差值的问题,并定义了相应的状态。最终得出了计算最大高度的解法。 ... [详细]
  • Java String与StringBuffer的区别及其应用场景
    本文主要介绍了Java中String和StringBuffer的区别,String是不可变的,而StringBuffer是可变的。StringBuffer在进行字符串处理时不生成新的对象,内存使用上要优于String类。因此,在需要频繁对字符串进行修改的情况下,使用StringBuffer更加适合。同时,文章还介绍了String和StringBuffer的应用场景。 ... [详细]
  • Android源码深入理解JNI技术的概述和应用
    本文介绍了Android源码中的JNI技术,包括概述和应用。JNI是Java Native Interface的缩写,是一种技术,可以实现Java程序调用Native语言写的函数,以及Native程序调用Java层的函数。在Android平台上,JNI充当了连接Java世界和Native世界的桥梁。本文通过分析Android源码中的相关文件和位置,深入探讨了JNI技术在Android开发中的重要性和应用场景。 ... [详细]
  • 本文讲述了CodeForces1016C题目的解法。文章首先介绍了一种错误的理解,然后给出了正确的解法。其中,当位于一个角上时,有两种选择,一种是先一直走一行再返回来走,另一种是走到这一列的另一行上然后再往右走一列。作者给出了两种解法,一种是直接计算,一种是动态规划。最后,取两种解法的最优解作为答案。文章附上了源代码。 ... [详细]
  • 本文介绍了数据库的存储结构及其重要性,强调了关系数据库范例中将逻辑存储与物理存储分开的必要性。通过逻辑结构和物理结构的分离,可以实现对物理存储的重新组织和数据库的迁移,而应用程序不会察觉到任何更改。文章还展示了Oracle数据库的逻辑结构和物理结构,并介绍了表空间的概念和作用。 ... [详细]
  • C++字符字符串处理及字符集编码方案
    本文介绍了C++中字符字符串处理的问题,并详细解释了字符集编码方案,包括UNICODE、Windows apps采用的UTF-16编码、ASCII、SBCS和DBCS编码方案。同时说明了ANSI C标准和Windows中的字符/字符串数据类型实现。文章还提到了在编译时需要定义UNICODE宏以支持unicode编码,否则将使用windows code page编译。最后,给出了相关的头文件和数据类型定义。 ... [详细]
  • 开发笔记:实验7的文件读写操作
    本文介绍了使用C++的ofstream和ifstream类进行文件读写操作的方法,包括创建文件、写入文件和读取文件的过程。同时还介绍了如何判断文件是否成功打开和关闭文件的方法。通过本文的学习,读者可以了解如何在C++中进行文件读写操作。 ... [详细]
  • 本文介绍了在CentOS上安装Python2.7.2的详细步骤,包括下载、解压、编译和安装等操作。同时提供了一些注意事项,以及测试安装是否成功的方法。 ... [详细]
  • This article discusses the efficiency of using char str[] and char *str and whether there is any reason to prefer one over the other. It explains the difference between the two and provides an example to illustrate their usage. ... [详细]
  • 如何用JNI技术调用Java接口以及提高Java性能的详解
    本文介绍了如何使用JNI技术调用Java接口,并详细解析了如何通过JNI技术提高Java的性能。同时还讨论了JNI调用Java的private方法、Java开发中使用JNI技术的情况以及使用Java的JNI技术调用C++时的运行效率问题。文章还介绍了JNIEnv类型的使用方法,包括创建Java对象、调用Java对象的方法、获取Java对象的属性等操作。 ... [详细]
  • 本文讨论了在VMWARE5.1的虚拟服务器Windows Server 2008R2上安装oracle 10g客户端时出现的问题,并提供了解决方法。错误日志显示了异常访问违例,通过分析日志中的问题帧,找到了解决问题的线索。文章详细介绍了解决方法,帮助读者顺利安装oracle 10g客户端。 ... [详细]
  • 本文详细介绍了Android中的坐标系以及与View相关的方法。首先介绍了Android坐标系和视图坐标系的概念,并通过图示进行了解释。接着提到了View的大小可以超过手机屏幕,并且只有在手机屏幕内才能看到。最后,作者表示将在后续文章中继续探讨与View相关的内容。 ... [详细]
author-avatar
可爱的天使keven_464
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有