OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 2751|回复: 8

SPI DMA 写入Flash时,受SPI分频影响

[复制链接]

4

主题

13

帖子

0

精华

初级会员

Rank: 2

积分
57
金钱
57
注册时间
2016-9-22
在线时间
17 小时
发表于 2020-2-16 23:11:47 | 显示全部楼层 |阅读模式
30金钱
本帖最后由 xiaomu 于 2020-2-16 23:14 编辑

最近在做SPI3处理FLASH的时候,发现一共问题,不知道什么原因,能不被解决。MCU: STM32F103RCT6     FLASH:W25Q16
为了增加FLASH的写入速率,使用了DMA。读取数据没有使用DMA。很顺利的是,SPI3通过DMA很顺利的将数据发到了FLASH中,当时使用的SPI3的分频是2分频,也就是SPI_BaudRatePrescaler_2。但是当我增加分频时,写入变发生了错误。具体如下图

图1

图1

图1 在SPI_BaudRatePrescaler_2下通过DMA写入到FLASH的数据,再读出来:STM32 SPI TEST!。这时候一切正常。

图2

图2

图2 当SPI的分频改为SPI_BaudRatePrescaler_4的时候,通过DMA写入到FLASH的数据,再读出来就是???????????了。通过测试,确定是写入时的问题。
尝试了SPI_BaudRatePrescaler_4--SPI_BaudRatePrescaler_256,均出现各种问题,而不用DMA时,像原子哥例程中的例程那样,则都没有问题。
希望有大神能够指教。

下面是代码
SPI和DMA配置void SPI3_Init(void)
{
        GPIO_InitTypeDef GPIO_InitStructure;
  SPI_InitTypeDef  SPI_InitStructure;

        RCC_APB2PeriphClockCmd(        RCC_APB2Periph_GPIOB, ENABLE );//PORTB时钟使能
        RCC_APB1PeriphClockCmd(        RCC_APB1Periph_SPI3,  ENABLE );//SPI2时钟使能        

        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3 | GPIO_Pin_4 | GPIO_Pin_5;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  //PB3/4/5复用推挽输出
        GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
        GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIOB

        GPIO_SetBits(GPIOB,GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5);  //PB3/4/5上拉

        SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;  //设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工
        SPI_InitStructure.SPI_Mode = SPI_Mode_Master;                //设置SPI工作模式:设置为主SPI
        SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;                //设置SPI的数据大小:SPI发送接收8位帧结构
        SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;                //串行同步时钟的空闲状态为高电平
        SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;        //串行同步时钟的第二个跳变沿(上升或下降)数据被采样
        SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;                //NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制
        SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_64;                //定义波特率预分频的值:波特率预分频值为256
        SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;        //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
        SPI_InitStructure.SPI_CRCPolynomial = 7;        //CRC值计算的多项式
        SPI_Init(SPI3, &SPI_InitStructure);  //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器

        SPI_Cmd(SPI3, ENABLE); //使能SPI外设
       
        SPI3_ReadWriteByte(0xff);//启动传输                 


}   
//SPI 速度设置函数
//SpeedSet:
//SPI_BaudRatePrescaler_2   2分频   
//SPI_BaudRatePrescaler_8   8分频   
//SPI_BaudRatePrescaler_16  16分频  
//SPI_BaudRatePrescaler_256 256分频

void SPI3_SetSpeed(u8 SPI_BaudRatePrescaler)
{
          assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));
        SPI3->CR1&=0XFFC7;
        SPI3->CR1|=SPI_BaudRatePrescaler;        //设置SPI3速度
        SPI_Cmd(SPI3,ENABLE);

}

//SPIx 读写一个字节
//TxData:要写入的字节
//返回值:读取到的字节
u8 SPI3_ReadWriteByte(u8 TxData)
{               
        u8 retry=0;                                        
        while (SPI_I2S_GetFlagStatus(SPI3, SPI_I2S_FLAG_TXE) == RESET) //检查指定的SPI标志位设置与否:发送缓存空标志位
                {
                retry++;
                if(retry>200)return 0;
                }                          
        SPI_I2S_SendData(SPI3, TxData); //通过外设SPIx发送一个数据
        retry=0;

        while (SPI_I2S_GetFlagStatus(SPI3, SPI_I2S_FLAG_RXNE) == RESET) //检查指定的SPI标志位设置与否:接受缓存非空标志位
                {
                retry++;
                if(retry>200)return 0;
                }                                                              
        return SPI_I2S_ReceiveData(SPI3); //返回通过SPIx最近接收的数据                                            
}


void SPI3_DMA_Configuration(DMA_Channel_TypeDef* DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{
        DMA_InitTypeDef DMA_InitStructure;
       
        RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA2, ENABLE);        //使能DMA传输

        DMA_DeInit(DMA_CHx);   //将DMA的通道1寄存器重设为缺省值

        DMA_InitStructure.DMA_PeripheralBaseAddr =cpar; //DMA外设ADC基地址
        DMA_InitStructure.DMA_MemoryBaseAddr = cmar;  //DMA内存基地址
        DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;  //数据传输方向,从内存读取发送到外设
        DMA_InitStructure.DMA_BufferSize = cndtr;  //DMA通道的DMA缓存的大小
        DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;        //外设地址寄存器不变
        DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;        //内存地址寄存器递增
        DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;  //数据宽度为8位
        DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //数据宽度为8位
        DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  //工作在正常缓存模式
        DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //DMA通道 x拥有中优先级
        DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  //DMA通道x没有设置为内存到内存传输
        DMA_Init(DMA_CHx, &DMA_InitStructure);  //根据DMA_InitStruct中指定的参数初始化DMA的通道USART1_Tx_DMA_Channel所标识的寄存器
        DMA_ITConfig(DMA_CHx, DMA_IT_TC, ENABLE);

       
}

void SPI3_DMA_Enable(u16 DataLen)
{
        //SPI_Cmd(SPI3, ENABLE);


        SPI_I2S_DMACmd(SPI3,SPI_I2S_DMAReq_Tx,ENABLE);//使能SPI3的DMA发送

        DMA_Cmd(SPI3_TX_DMA_Channel, DISABLE);//关闭通道
        DMA_SetCurrDataCounter(SPI3_TX_DMA_Channel, DataLen);//设置数据传输长度
       
       
        DMA_Cmd(SPI3_TX_DMA_Channel, ENABLE);//打开通道       
       
       
        while(DMA_GetFlagStatus(DMA2_FLAG_TC2)==RESET);//等待传输完成
        DMA_ClearFlag(DMA2_FLAG_TC2);//清除通道4传输完成标志
       
}





正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

16

主题

426

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2765
金钱
2765
注册时间
2018-11-8
在线时间
438 小时
发表于 2020-2-17 20:25:29 | 显示全部楼层
楼主 请问SPI_Init函数里 最后为什么要发个0xff来启动传输???SPI不是拉低片选以后直接发指令就行了嘛?
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165540
金钱
165540
注册时间
2010-12-1
在线时间
2117 小时
发表于 2020-2-18 00:48:12 | 显示全部楼层
帮顶
回复

使用道具 举报

4

主题

13

帖子

0

精华

初级会员

Rank: 2

积分
57
金钱
57
注册时间
2016-9-22
在线时间
17 小时
 楼主| 发表于 2020-2-18 15:42:22 | 显示全部楼层
babyrabbit 发表于 2020-2-17 20:25
楼主 请问SPI_Init函数里 最后为什么要发个0xff来启动传输???SPI不是拉低片选以后直接发指令就行了嘛?

这个不发也是可以的,起初是为了在仿真的时候看看SPI的寄存器里有没有值
回复

使用道具 举报

4

主题

13

帖子

0

精华

初级会员

Rank: 2

积分
57
金钱
57
注册时间
2016-9-22
在线时间
17 小时
 楼主| 发表于 2020-2-18 15:45:09 | 显示全部楼层
回复

使用道具 举报

3

主题

1155

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
7464
金钱
7464
注册时间
2015-1-15
在线时间
1368 小时
发表于 2020-2-19 21:40:10 来自手机 | 显示全部楼层
SPI+DMA是可以实现的
回复

使用道具 举报

4

主题

13

帖子

0

精华

初级会员

Rank: 2

积分
57
金钱
57
注册时间
2016-9-22
在线时间
17 小时
 楼主| 发表于 2020-2-22 14:20:01 | 显示全部楼层
再顶一下
回复

使用道具 举报

4

主题

13

帖子

0

精华

初级会员

Rank: 2

积分
57
金钱
57
注册时间
2016-9-22
在线时间
17 小时
 楼主| 发表于 2020-2-22 14:21:39 | 显示全部楼层
yklstudent 发表于 2020-2-19 21:40
SPI+DMA是可以实现的

是的,目前的问题是spi的速度降低后,传输出现错误了。
回复

使用道具 举报

3

主题

1155

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
7464
金钱
7464
注册时间
2015-1-15
在线时间
1368 小时
发表于 2020-2-23 13:21:54 | 显示全部楼层
xiaomu 发表于 2020-2-22 14:21
是的,目前的问题是spi的速度降低后,传输出现错误了。

应该还是你的驱动代码有问题
一分耕耘一分收获。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-26 05:15

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表