大神们帮着看一下这个程序吧,为什么进不了中断呢?
#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相关论坛找到合适的答案 : )