TI中文支持网
TI专业的中文技术问题搜集分享网站

有关STM32的USART+DMA连续通信问题

 大神们帮着看一下这个程序吧,为什么进不了中断呢?
#include "sys.h"
#include "usart.h"
//加入以下代码,支持printf函数,而不需要选择use MicroLIB
#if 1
#pragma import(__use_no_semihosting)             
//标准库需要的支持函数                 
struct __FILE 

int handle; 

}; 

FILE __stdout;       
//定义_sys_exit()以避免使用半主机模式    
_sys_exit(int x) 

x = x; 

//重定义fputc函数 
int fputc(int ch, FILE *f)
{      
while((USART1->SR&0X40)==0);//循环发送,直到发送完毕   
    USART1->DR = (u8) ch;      
return ch;
}
#endif 
DMA_InitTypeDef   DMA_InitStructure1;  
u8 USART_RX_BUF[3];     //接收缓冲,最大3个字节.
u8 USART_RX_STA=0;       //接收状态标记
char m;
void uart_init(u32 bound)
{
    //GPIO端口设置
    GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure; 
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE); 
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_AFIO, ENABLE);    
USART_DeInit(USART1);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_Init(GPIOA, &GPIO_InitStructure);                     
   //USART 初始化设置     
USART_InitStructure.USART_BaudRate = bound;//一般设置为9600;
USART_InitStructure.USART_WordLength = USART_WordLength_8b;
USART_InitStructure.USART_StopBits = USART_StopBits_1;
USART_InitStructure.USART_Parity = USART_Parity_No;   //无奇偶校验
USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;  //接收和发送使能
    USART_Init(USART1, &USART_InitStructure);    
    NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel5_IRQn; 
    NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; // 
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
NVIC_Init(&NVIC_InitStructure); //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器USART1
     
    USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启中断   
    USART_Cmd(USART1, ENABLE);                  //使能串口 
USART_DMACmd(USART1, USART_DMAReq_Rx | USART_DMAReq_Tx, ENABLE);//使能USART1的DMA发送和接收功能

}  
void MYDMA_Config_USART(DMA_Channel_TypeDef* DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{  
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_AFIO|RCC_APB2Periph_USART1, ENABLE);
  RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA1传输
__nop();                    //等待 DMA1 时钟稳定,DMA开启需要时间稳定
  __nop();                    //经测试最少 2 个 nop
  __nop();   
    DMA_DeInit(DMA_CHx);   //将DMA的通道1寄存器重设为缺省值
DMA_InitStructure1.DMA_PeripheralBaseAddr = cpar;  //DMA外设ADC基地址
  DMA_InitStructure1.DMA_MemoryBaseAddr = cmar;  //DMA内存基地址
DMA_InitStructure1.DMA_DIR = DMA_DIR_PeripheralSRC;  //数据传输方向,从外设读取数据
  DMA_InitStructure1.DMA_BufferSize = cndtr;  //DMA通道的DMA缓存的大小 
DMA_InitStructure1.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  //外设地址寄存器不变
DMA_InitStructure1.DMA_MemoryInc = DMA_MemoryInc_Enable;  //内存地址寄存器变化
DMA_InitStructure1.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte ;  //外设数据宽度为8位
DMA_InitStructure1.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte ; //内存数据宽度为8位
  DMA_InitStructure1.DMA_Mode = DMA_Mode_Circular;  //工作在循环模式
  DMA_InitStructure1.DMA_Priority = DMA_Priority_High; //DMA通道优先
  DMA_InitStructure1.DMA_M2M = DMA_M2M_Disable;  //DMA通道x没有设置为内存到内存传输
DMA_Init(DMA_CHx, &DMA_InitStructure1);  //根据DMA_InitStruct中指定的参数初始化DMA的通道USART1_Tx_DMA_Channel所标识的寄存器

DMA_ITConfig(DMA1_Channel5, DMA_IT_TC, ENABLE);   //DMA5传输完成中断   
USART_Cmd(USART1, ENABLE);  
USART_DMACmd(USART1, USART_DMAReq_Rx | USART_DMAReq_Tx, ENABLE);  //使能USART1的接收DMA请求
DMA_Cmd(DMA1_Channel5, ENABLE);


 
void DMA1_Channel5_IRQHandler(void)
{  
    if(DMA_GetITStatus(DMA1_IT_TC5)) //通道5传输完成中断TC 还有传输 过半中断HT 错误中断TE 全局中断GL
     {
   DMA_Cmd(DMA1_Channel5, DISABLE); 
       //DataCounter = DMA_GetCurrDataCounter(DMA1_Channel5);//获取剩余长度,一般都为0,调试用
        USART_ClearFlag(USART1,USART_FLAG_TC);
DMA_ClearITPendingBit(DMA1_IT_TC5);    //清除全部中断标志
        
  for(m=0;m<3;m++)
{
USART1_Send_Byte(USART_RX_BUF[m]); 
}  
DMA_Cmd(DMA1_Channel5, ENABLE);  
     }
}

void USART1_Send_Byte(unsigned char byte)   //串口发送一个字节
{
        USART_SendData(USART1, byte);        //通过库函数  发送数据
        while( USART_GetFlagStatus(USART1,USART_FLAG_TC)!= SET);  
        //等待发送完成。   检测 USART_FLAG_TC 是否置1;    //见库函数 P359 介绍
 }
///////////////////////////////////////////主函数///////////////////////////////////////////
 int main(void)
{    
SystemInit();
RCC_Configuration();
I2C_Configuration();
NVIC_Configuration();
delay_init(72);
LCD_Init();
   MYDMA_Config_USART(DMA1_Channel5,(u32)&USART1->DR,(u8)USART_RX_BUF,3);
uart_init(9600);//串口初始化为9600
while(1)
{
Confige1115(3);
read1115();
USART_ClearFlag(USART1,USART_FLAG_TC);
delay_ms(300);
printf("\n\n");//插入换行
}
}

说明: 
Confige1115(3);//配置ADS1115(一个12位的AD) 
read1115();//读取AD采集到得数据 
存在问题: 
1.不明白下面QQ截图中的那短话,来自《STM32中文参考手册V10》535页,不理解用DMA的USART串口通信其中有关中断标志处理的先后顺序。 
2.不清楚DMA传输时数据的流程 
3.不清楚数据的传送过程

xyz549040622:

大哥,STM32的问题跑到TI的论坛。。。

kqian0327:

你好,

STM32是非TI MCU,请到ST相关论坛找到合适的答案 : )

赞(0)
未经允许不得转载:TI中文支持网 » 有关STM32的USART+DMA连续通信问题
分享到: 更多 (0)