STM32F10XX中SPI的DMA发送数据

2024-09-02 23:32

本文主要是介绍STM32F10XX中SPI的DMA发送数据,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

参考资料:

http://blog.csdn.net/jdh99/article/details/7603029

http://www.openedv.com/posts/list/3159.htm

    上面提到的两篇博文比较详细深刻的说明了DMA的工作方式以及SPI的DMA传输方式的特点。结合对Stm32F103VET6中SPI的DMA传输方式的配置和学习谈谈感受,在看下面的内容之前请先看上面的两篇参考博文,这里就不在说明。

    要使用SPI的DMA功能,首先配置好SPI外设,这里以SPI1为例子。下面的代码初始化了SPI1对应的GPIO以及SPI1工作的模式。

void SpiCC3000Init(void)
{
     SPI_InitTypeDef   SPI_InitStructure;
    GPIO_InitTypeDef  GPIO_InitStructure;
  /*!< Disable SPI */
  SPI_Cmd(SPI_USED, DISABLE);
  
  /*!< DeInitializes the SPI */
  SPI_I2S_DeInit(SPI_USED);
  
  /*!< SPI Periph clock disable */
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2 , DISABLE); 
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1 , DISABLE);   
    /*Enable SPI2 Clock */
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2,ENABLE);
  RCC_APB2PeriphClockCmd(    RCC_APB2Periph_GPIOB,    ENABLE);
    /*Enable SPI1 Clock */
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1 |RCC_APB2Periph_GPIOA,ENABLE);
    /*Config  SCLK and MOSI */
    GPIO_InitStructure.GPIO_Pin = SPI_CLK_PIN| SPI_MOSI_PIN;// 
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;//AF_PP
    GPIO_Init(SPI_PORT, &GPIO_InitStructure);
  /*!< Configure SPI pins: MISO */
  GPIO_InitStructure.GPIO_Pin = SPI_MISO_PIN;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
  GPIO_Init(SPI_PORT, &GPIO_InitStructure);
    /*!< Configure SPI pins: CS output high */
    GPIO_InitStructure.GPIO_Pin = SPI_CS_PIN;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_Init(SPI_PORT, &GPIO_InitStructure);
    GPIO_SetBits(SPI_PORT,SPI_CS_PIN);
    
  /*!< CC3000 SPI Init */
  SPI_StructInit(&SPI_InitStructure);
  SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
  SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
  SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
  SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low;  
  SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;
  SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
  SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_8;/* The buadrate is a fraction of the 72MHz clock*/
  SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
    SPI_InitStructure.SPI_CRCPolynomial = 7;
  SPI_Init(SPI_USED, &SPI_InitStructure);        
  SPI_SSOutputCmd(SPI_USED,ENABLE);//NSS(CS) 
  SPI_Cmd(SPI_USED, ENABLE);
  
  /*Configure SPI for DMA Operation*/
#if defined(ENABLE_SPI_DMA)
  SpiCC3000DMAInit(); 
#else
  SpiRxInterruptClkInit();
#endif
}
    在配置好SPI1以后,配置DMA1功能(选择SPI1的Tx连接到DMA上),因为SPI1对应的DMA功能由DMA1来实现。  下面函数中的DMA时钟将在调用它的函数中开启。
void C3000_DMA_Config(SPI_DMADirection_TypeDef Direction, uint8_t* buffer, uint16_t NumData)
{
    
#if defined(ENABLE_SPI_DMA)
  DMA_InitTypeDef DMA_InitStructure;
  /* Initialize the DMA_PeripheralBaseAddr member */
  DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t )&SPI1->DR; //SPI_DR_BASE; // 
    //DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t )&SPI2->DR; //SPI_DR_BASE; // 
  /* Initialize the DMA_MemoryBaseAddr member */
  DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)buffer;
   /* Initialize the DMA_PeripheralInc member */
  DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  /* Initialize the DMA_MemoryInc member */
  DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
  /* Initialize the DMA_PeripheralDataSize member */
  DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
  /* Initialize the DMA_MemoryDataSize member */
  DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
  /* Initialize the DMA_Mode member */
    //yichuan
  DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
//DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
  /* Initialize the DMA_Priority member */
  DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh;
  /* Initialize the DMA_M2M member */
  DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
  
  /* If using DMA for Reception */
  if (Direction == SPI_DMA_RX)
  {
    /* Initialize the DMA_DIR member */
    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
    
    /* Initialize the DMA_BufferSize member */
    DMA_InitStructure.DMA_BufferSize = NumData;
    
    DMA_DeInit(SPI_DMA_RX_CHANNEL);
    
    DMA_Init(SPI_DMA_RX_CHANNEL, &DMA_InitStructure);
  }
   /* If using DMA for Transmission */
  else if (Direction == SPI_DMA_TX)
  { 
    /* Initialize the DMA_DIR member */
    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;
    
    /* Initialize the DMA_BufferSize member */
    DMA_InitStructure.DMA_BufferSize = NumData;
    
    DMA_DeInit(SPI_DMA_TX_CHANNEL);
 
    DMA_Init(SPI_DMA_TX_CHANNEL, &DMA_InitStructure);
  }
#endif
}
   配置好DMA1和SPI1后,要做的事情就是把二者联合起来,且配置DMA1的发送完成中断。
void SpiCC3000DMAInit(void)

    
  RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
 
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1); 
  DMA_DeInit(SPI_DMA_RX_CHANNEL);
  DMA_DeInit(SPI_DMA_TX_CHANNEL);
 
  /* Configure and enable SPI DMA TX Channel interrupt */ 
 // NVIC_RxInt_InitStructure.NVIC_IRQChannel = DMA1_Channel5_IRQn; //SPI2 TX
  NVIC_RxInt_InitStructure.NVIC_IRQChannel = DMA1_Channel3_IRQn;//SPI1 TX // not config the Rx channel
  NVIC_RxInt_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
  NVIC_RxInt_InitStructure.NVIC_IRQChannelSubPriority = 0;//0
  NVIC_RxInt_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  NVIC_Init(&NVIC_RxInt_InitStructure);
 
 
  /* Configure DMA Peripheral but don't send data*/
  C3000_DMA_Config(SPI_DMA_RX, (uint8_t*)wlan_rx_buffer,0); 
 //C3000_DMA_Config(SPI_DMA_TX, (uint8_t*)wlan_tx_buffer,0);    
  C3000_DMA_Config(SPI_DMA_TX, (uint8_t*)wlan_tx_buffer,1700);  //buffer is 1700
            
 
 
    /* Enable SPI DMA request */
  SPI_I2S_DMACmd(SPI_USED,SPI_I2S_DMAReq_Tx, ENABLE);
  SPI_I2S_DMACmd(SPI_USED,SPI_I2S_DMAReq_Rx, ENABLE);
  /* Enable the DMA Channels Interrupts */
  /*It should be put after the DMA config  */
  DMA_ITConfig(SPI_DMA_TX_CHANNEL, DMA_IT_TC, ENABLE); 
 
  /* Enable DMA RX Channel */
  DMA_Cmd(SPI_DMA_RX_CHANNEL, ENABLE);  
    /*Note: Enable the SPI_DMA Channel,it begin SPI translate*/
  /* Enable DMA TX Channel ,begin the dma translate*/  
  DMA_Cmd(SPI_DMA_TX_CHANNEL, ENABLE); 
}
  这里要强调的是:使能DMA中断标志的语句DMA_ITConfig(SPI_DMA_TX_CHANNEL, DMA_IT_TC, ENABLE); 应当放到DMA1配置完以后,不然发送数据完成以后,不会进入中断处理函数,就是相当于发送完成后产生中断的这个功能并没有配置成功,如果放到DMA1配置前面。下面的图片是参考手册中关于DMA的配置过程,其中中断的使能在第六步,所以应当在DMA初始化完成后,再使能中断功能。

 

    接下来就是中断处理函数了,响应传输完成中断的产生

void DMA1_Channel3_IRQHandler(void)
{
    if(DMA_GetITStatus(DMA1_IT_TC3)==SET)
    {
            GPIO_InitTypeDef GPIO_InitStructure;
            RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);
            GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
            GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
            GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
            GPIO_Init(GPIOC, &GPIO_InitStructure);
            GPIO_ResetBits(GPIOC,GPIO_Pin_6);
 
    }
   DMA_ClearITPendingBit(DMA1_IT_TC3);
 
}
最后启动DMA传输是通过DMA_Cmd(SPI_DMA_TX_CHANNEL, ENABLE) 这句话实现,当DMA通道被使能以后,它就自动开始传输数据,而不影响CPU其干其它事情,直到DMA传输完数据产生中断时,CPU才会去处理中断函数。

 

这篇关于STM32F10XX中SPI的DMA发送数据的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1131381

相关文章

【STM32】SPI通信-软件与硬件读写SPI

SPI通信-软件与硬件读写SPI 软件SPI一、SPI通信协议1、SPI通信2、硬件电路3、移位示意图4、SPI时序基本单元(1)开始通信和结束通信(2)模式0---用的最多(3)模式1(4)模式2(5)模式3 5、SPI时序(1)写使能(2)指定地址写(3)指定地址读 二、W25Q64模块介绍1、W25Q64简介2、硬件电路3、W25Q64框图4、Flash操作注意事项软件SPI读写W2

STM32 ADC+DMA导致写FLASH失败

最近用STM32G070系列的ADC+DMA采样时,遇到了一些小坑记录一下; 一、ADC+DMA采样时进入死循环; 解决方法:ADC-dma死循环问题_stm32 adc dma死机-CSDN博客 将ADC的DMA中断调整为最高,且增大ADCHAL_ADC_Start_DMA(&hadc1, (uint32_t*)adc_buffer, ADC_Buffer_Size); 的ADC_Bu

学习硬件测试05:NTC(ADC)+正弦波(DAC)+DMA(ADC+DAC)(P73、P76、P78)

文章以下内容全部为硬件相关知识,鲜有软件知识,并且记的是自己需要的部分,大家可能看不明白。 一、NTC(ADC) 1.1实验现象 本实验用 NTC 采集温度,数码管实时显示温度数据(整数),左下角 USB 小串口每隔 1S 打印温度信息。 1.2硬件电路 NTC 电阻是一个模拟温度传感器,随着温度的升高,电阻值逐渐减小。电路简单介绍如下: 电源滤波电容在 25℃ 室温下 NTC 电

物联网——DMA+AD多通道

DMA简介 存储器映像 某些数据在运行时不会发生变化,则设置为常量,存在Flash存储器中,节省运行内存的空间 DMA结构图 DMA访问权限高于cpu 结构要素 软件触发源:存储器到存储器传输完成后,计数器清零 硬件触发源:ADC、定时器、串口 重写计数器时,需关闭DMA DMA请求 数据宽度与对齐 目标宽度小于传输带宽:高位补零,反之,舍弃高位 数据转运与D

stm32之软件SPI读写W25Q64存储器应用案例

系列文章目录 1. stm32之SPI通信协议 文章目录 系列文章目录前言一、电路接线图二、应用案例代码三、应用案例分析3.1 SPI通信模块3.2 W25Q64模块3.3 主程序 前言 提示:本文主要用作在学习江科大自化协STM32入门教程后做的归纳总结笔记,旨在学习记录,如有侵权请联系作者 本案例使用软件SPI通信的方式实现了STM32与W25Q64 Flas

STM32F103调试DMA+PWM 实现占空比逐渐增加的软启效果

实现效果:DMA+PWM 实现PWM输出时,从低电平到输出占空比逐渐增加再到保持高电平的效果,达到控制 MOS 功率开关软启的效果。 1.配置时钟 2.TIM 的 PWM 功能配置 选择、配置 TIM 注意:选择 TIM 支持 DMA 控制输出 PWM 功能的通道,有的TIM通道支持PWM 但不支持PWM注意选择。 PWM参数设置 Counter Period :

Flink1.10基于工厂模式的任务提交与SPI机制

《2021年最新版大数据面试题全面开启更新》 Flink任务执行模式包含了yarn-session、standalone、per-job、local, 在1.10中又增加k8s的执行模式,那么在任务提交过程中如何根据不同的执行模式进行任务提交呢?主要通过两个接口来实现:PipelineExecutorFactory 与 PipelineExecutor。PipelineExecutorF

DMA引起数组越界

今遇到了内存越界问题,很隐蔽   EXTERN  __IO uint16_t RegularConvData_Tab[2*3]; 定义的DMA搬运工的buffer大小为6   实际上当时红线标注,改为了12,导致后续的变量被赋值 只要此值小于定义的buffer的大小就可以了   这个越界是非常不注意的问题,编译器不报错

OpenGL DMA接口

Opengl的DMA版本接口主要作用是解决以前访问opengl对象, 必须先将对象绑定到当前状态机下才能访问的问题,这会导致驱动层需要去频繁的进行对象的引用查找。  比如以前非DMA版本的接口操作顶点数据  glGenBuffers(1, &vbo);glBindBuffer(GL_ARRAY_BUFFER, vbo);glBufferData(GL_ARRAY_BUFFER, sizeof

转载 SPI的比喻理解

SPI 传输是一个虚拟的移位寄存器方式。 你这么理解就可以: 主机和从机之间有一条 16 格的传送带。主机一格一格拨动它转动(相当于发送时钟)。 如果是主机发送,它就把一个个的东西放在传送带上,转动 8 次,就传到从机一侧了。这时,从机可以从传送带上将东西取下。如果从机没有取东西,这些东西再转 8 次又回到主机一侧。 如果是主机接收,从机就要把 8 个东西一次放上传送带。当主机转动 8 次,东西就