热门标签 | 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 定义。谢谢你的帮助!










推荐阅读
  • 如何自行分析定位SAP BSP错误
    The“BSPtag”Imentionedintheblogtitlemeansforexamplethetagchtmlb:configCelleratorbelowwhichi ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • 本文介绍了Oracle数据库中tnsnames.ora文件的作用和配置方法。tnsnames.ora文件在数据库启动过程中会被读取,用于解析LOCAL_LISTENER,并且与侦听无关。文章还提供了配置LOCAL_LISTENER和1522端口的示例,并展示了listener.ora文件的内容。 ... [详细]
  • 安卓select模态框样式改变_微软Office风格的多端(Web、安卓、iOS)组件库——Fabric UI...
    介绍FabricUI是微软开源的一套Office风格的多端组件库,共有三套针对性的组件,分别适用于web、android以及iOS,Fab ... [详细]
  • 本文讨论了在Spring 3.1中,数据源未能自动连接到@Configuration类的错误原因,并提供了解决方法。作者发现了错误的原因,并在代码中手动定义了PersistenceAnnotationBeanPostProcessor。作者删除了该定义后,问题得到解决。此外,作者还指出了默认的PersistenceAnnotationBeanPostProcessor的注册方式,并提供了自定义该bean定义的方法。 ... [详细]
  • 本文介绍了多因子选股模型在实际中的构建步骤,包括风险源分析、因子筛选和体系构建,并进行了模拟实证回测。在风险源分析中,从宏观、行业、公司和特殊因素四个角度分析了影响资产价格的因素。具体包括宏观经济运行和宏经济政策对证券市场的影响,以及行业类型、行业生命周期和行业政策对股票价格的影响。 ... [详细]
  • 本文介绍了为什么要使用多进程处理TCP服务端,多进程的好处包括可靠性高和处理大量数据时速度快。然而,多进程不能共享进程空间,因此有一些变量不能共享。文章还提供了使用多进程实现TCP服务端的代码,并对代码进行了详细注释。 ... [详细]
  • Python正则表达式学习记录及常用方法
    本文记录了学习Python正则表达式的过程,介绍了re模块的常用方法re.search,并解释了rawstring的作用。正则表达式是一种方便检查字符串匹配模式的工具,通过本文的学习可以掌握Python中使用正则表达式的基本方法。 ... [详细]
  • 从零学Java(10)之方法详解,喷打野你真的没我6!
    本文介绍了从零学Java系列中的第10篇文章,详解了Java中的方法。同时讨论了打野过程中喷打野的影响,以及金色打野刀对经济的增加和线上队友经济的影响。指出喷打野会导致线上经济的消减和影响队伍的团结。 ... [详细]
  • 本文介绍了PE文件结构中的导出表的解析方法,包括获取区段头表、遍历查找所在的区段等步骤。通过该方法可以准确地解析PE文件中的导出表信息。 ... [详细]
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社区 版权所有