打印

while(SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) == RESET);死循环

[复制链接]
10482|14
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
andiwxz|  楼主 | 2013-10-14 15:10 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
我使用stm32f103ZET6控制外部flash芯片at45db161d。在读取flash内保存的数据的时候,有一定几率会出现在while(SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) == RESET);死循环的状态。经测试好像数据量大的时候容易出现。我降低了通讯的速度,现象有所好转,但没有完全杜绝。请问还需要怎么处理。我的spi初始化如下:
           /* Configure SPI2 pins: NSS, SCK, MISO and MOSI ---------------------------------*/
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  GPIO_Init(GPIOB, &GPIO_InitStructure);


   /* flash at45db161d select pin */
        GPIO_InitStructure.GPIO_Pin= GPIO_Pin_14;
        GPIO_InitStructure.GPIO_Speed=GPIO_Speed_10MHz;
        GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;
        GPIO_Init(GPIOF, &GPIO_InitStructure);
        NOSelect_SD();
    /* flash SD select pin */
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_15 ;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  GPIO_Init(GPIOF, &GPIO_InitStructure);
  NotSelect_Flash();

  SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;//SPI_Direction_1Line_Tx;
  SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
  SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
  SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;//SPI_CPOL_Low;
  SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;//SPI_CPHA_1Edge;
  SPI_InitStructure.SPI_NSS  = SPI_NSS_Soft;
  SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_64;/* 72M/64=1.125M */
  SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
  SPI_InitStructure.SPI_CRCPolynomial = 7;
  //SPI_I2S_DeInit(SPI2);
  SPI_Init(SPI2, &SPI_InitStructure);
  /* Enable SPI_MASTER */
  SPI_Cmd(SPI2, ENABLE);
  SPI_CalculateCRC(SPI2, DISABLE);
沙发
chuangpu| | 2013-10-14 16:58 | 只看该作者
不是很了解这一块   楼主   帮你顶一个   看看别人的意见如何   顶你一个

使用特权

评论回复
板凳
huzi2099| | 2013-10-14 17:21 | 只看该作者
那个while循环是在等接收到的数据,一读DR寄存器即被清0.
你检查一下while前是否开启读的流程或者其它地方读过了,总之要保证while前正在读

使用特权

评论回复
地板
luojijian| | 2013-10-14 17:29 | 只看该作者
好像见到过,把循环判断那里直接用寄存器判断,不要调用函数

使用特权

评论回复
5
IJK| | 2013-10-14 17:48 | 只看该作者
    /* Configure SPI2 pins: NSS, SCK, MISO and MOSI ---------------------------------*/
   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;
   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz;

看上去 GPIO_Speed_2MHz 的速率低了,可以试着改为GPIO_Speed_50MHz

使用特权

评论回复
6
cy757| | 2013-10-15 08:51 | 只看该作者
/* Enable SPI2, GPIOA and GPIOBclocks */
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);  
        RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2, ENABLE);
你看看这两个时钟是否打开,并且没有设置错误。
我就

使用特权

评论回复
7
andiwxz|  楼主 | 2013-11-11 10:23 | 只看该作者
目前还没有解决。先谢谢大家

使用特权

评论回复
8
sinc_mark| | 2014-6-19 10:43 | 只看该作者
andiwxz 发表于 2013-11-11 10:23
目前还没有解决。先谢谢大家

楼主,后来怎么解决了?我现在的情况和你一样,是死在那个循环中了!

使用特权

评论回复
9
jiebin689| | 2014-7-2 00:13 | 只看该作者
sinc_mark 发表于 2014-6-19 10:43
楼主,后来怎么解决了?我现在的情况和你一样,是死在那个循环中了!

把while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) == RESET)
改成while((SPI2->SR &SPI_I2S_FLAG_RXNE) == RESET);
看看得不得。
或者干脆直接把SPI_I2S_FLAG_RXNE写成0x0001或0x0003

使用特权

评论回复
10
liurenyin| | 2015-9-9 16:17 | 只看该作者
最后到底怎么解决的啊?我现在也是这个情况。

使用特权

评论回复
11
s976228410| | 2015-11-11 15:23 | 只看该作者
楼主 我的SPI2时钟脚都没有输出,会是怎么情况呢?

使用特权

评论回复
12
天灵灵地灵灵| | 2015-11-11 16:38 | 只看该作者

判断前,就是While之前,先清理一下那个寄存器,那个应该可以清理的。

使用特权

评论回复
13
无帝老三| | 2016-12-13 14:26 | 只看该作者
sinc_mark 发表于 2014-6-19 10:43
楼主,后来怎么解决了?我现在的情况和你一样,是死在那个循环中了!

这么奇怪的问题,这么多人遇到,也没看到解决方案在哪里

使用特权

评论回复
14
无帝老三| | 2016-12-13 16:17 | 只看该作者
可能的原因之一:
调试过程中,如果打开了SPI监控窗口,如果调试器先读走了DR的内容,就会导致RXNE位被清除,导致死在while中。

使用特权

评论回复
15
wangtop55| | 2017-12-3 13:49 | 只看该作者
问题:STM32103RDT6 SPI2最大速率fclck/2分频下出现此问题,降速至 fclck/4及以下频率通讯正常

解决:查阅stm32f1xx_hal_spi.c 后发现在开头提到在Polling模式下 8bits数据接收发最高支持fPCLK/4,而在DMA模式下fPCLK/2是可以的。遂改用DMA方式后,通讯正常。

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

166

主题

258

帖子

3

粉丝