新手上路
- 积分
- 22
- 金钱
- 22
- 注册时间
- 2020-5-18
- 在线时间
- 35 小时
|
15金钱
刚学习一个月,尝试编写ADC通过 DMA传输并且在串口上显示的程序,但是每次检测到的数据都是0。不知道是不是哪里配置错误了,检查了好久,水平有限,看不出来。
希望有大佬帮忙看一下,万分感谢! 程序上传到附件里了。
ADC的程序:
#define N 4
ADC_HandleTypeDef ADC1_Handler;//ADC句柄
u16 value[N];
//初始化ADC
//ch: ADC_channels
//通道值 0~16取值范围为:ADC_CHANNEL_0~ADC_CHANNEL_16
void Adc_Init(void)
{
GPIO_InitTypeDef GPIO_Initure;
__HAL_RCC_ADC1_CLK_ENABLE(); //使能ADC1时钟
__HAL_RCC_GPIOB_CLK_ENABLE(); //开启GPIOB时钟
GPIO_Initure.Pin=GPIO_PIN_1; //PB1
GPIO_Initure.Mode=GPIO_MODE_ANALOG; //模拟
GPIO_Initure.Pull=GPIO_NOPULL; //不带上下拉
HAL_GPIO_Init(GPIOB,&GPIO_Initure);
ADC1_Handler.Instance=ADC1;
ADC1_Handler.Init.ClockPrescaler=ADC_CLOCK_SYNC_PCLK_DIV4; //4分频,ADCCLK=PCLK2/4=96/4=24MHZ
ADC1_Handler.Init.Resolution=ADC_RESOLUTION_12B; //12位模式
ADC1_Handler.Init.DataAlign=ADC_DATAALIGN_RIGHT; //右对齐
ADC1_Handler.Init.ScanConvMode=ENABLE; //非扫描模式
ADC1_Handler.Init.EOCSelection=DISABLE; //关闭EOC中断
ADC1_Handler.Init.ContinuousConvMode=DISABLE; //关闭连续转换
ADC1_Handler.Init.NbrOfConversion=1; //1个转换在规则序列中 也就是只转换规则序列1
ADC1_Handler.Init.DiscontinuousConvMode=DISABLE; //禁止不连续采样模式
ADC1_Handler.Init.NbrOfDiscConversion=0; //不连续采样通道数为0
ADC1_Handler.Init.ExternalTrigConv=ADC_SOFTWARE_START; //软件触发
ADC1_Handler.Init.ExternalTrigConvEdge=ADC_EXTERNALTRIGCONVEDGE_NONE;//使用软件触发
ADC1_Handler.Init.DMAContinuousRequests=ENABLE; //开启DMA请求
ADC1_Handler.DMA_Handle=&ADC1TxDMA_Handler ;
HAL_ADC_Init(&ADC1_Handler); //初始化
ADC_ChannelConfTypeDef ADC1_ChanConf;
ADC1_ChanConf.Channel=ADC_CHANNEL_9; //通道9
ADC1_ChanConf.Rank=1; //第1个序列,序列1
ADC1_ChanConf.SamplingTime=ADC_SAMPLETIME_480CYCLES; //采样时间
ADC1_ChanConf.Offset=0;
HAL_ADC_ConfigChannel(&ADC1_Handler,&ADC1_ChanConf); //通道配置
// HAL_ADC_Start(&ADC1_Handler); //开启ADC
}
DMA的程序:
uint32_t ADC_ConvertedValue[4];
DMA_HandleTypeDef ADC1TxDMA_Handler; //DMA句柄
void MYDMA_Config(DMA_Stream_TypeDef *DMA_Streamx,u32 chx)
{
__HAL_RCC_DMA2_CLK_ENABLE();//DMA2时钟使能
// DMA配置
ADC1TxDMA_Handler.Instance=DMA_Streamx; //数据流选择
ADC1TxDMA_Handler.Init.Channel=chx; //通道选择
ADC1TxDMA_Handler.Init.Direction=DMA_MEMORY_TO_PERIPH; //存储器到外设
ADC1TxDMA_Handler.Init.PeriphInc=DMA_PINC_DISABLE; //外设非增量模式
ADC1TxDMA_Handler.Init.MemInc=DMA_MINC_ENABLE; //存储器增量模式
ADC1TxDMA_Handler.Init.PeriphDataAlignment=DMA_PDATAALIGN_BYTE; //外设数据长度:8位
ADC1TxDMA_Handler.Init.MemDataAlignment=DMA_MDATAALIGN_BYTE; //存储器数据长度:8位
ADC1TxDMA_Handler.Init.Mode=DMA_CIRCULAR; //循环模式
ADC1TxDMA_Handler.Init.Priority=DMA_PRIORITY_MEDIUM; //中等优先级
ADC1TxDMA_Handler.Init.FIFOMode=DMA_FIFOMODE_DISABLE;
ADC1TxDMA_Handler.Init.FIFOThreshold=DMA_FIFO_THRESHOLD_FULL;
ADC1TxDMA_Handler.Init.MemBurst=DMA_MBURST_SINGLE; //存储器突发单次传输
ADC1TxDMA_Handler.Init.PeriphBurst=DMA_PBURST_SINGLE; //外设突发单次传输
ADC1TxDMA_Handler.XferCpltCallback = HAL_DMA_IRQHandler;
HAL_NVIC_SetPriority(DMA2_Stream0_IRQn,0,0); //抢占优先级为2,子优先级为2
HAL_NVIC_EnableIRQ(DMA2_Stream0_IRQn); // 使能中断线 0
HAL_ADC_Start_DMA(&ADC1_Handler,&ADC_ConvertedValue[0],4) ;
__HAL_LINKDMA(&ADC1_Handler,DMA_Handle,ADC1TxDMA_Handler); //将DMA与ADC1联系起来(发送DMA)
HAL_DMA_DeInit(&ADC1TxDMA_Handler);
HAL_DMA_Init(&ADC1TxDMA_Handler);
}
void DMA2_Stream0_IRQHandler(void)
{
HAL_DMA_IRQHandler(&ADC1TxDMA_Handler);
}
主程序:#define N 4
extern u16 value[N];
extern uint32_t ADC_ConvertedValue[4];
int main(void)
{
u16 adcx;
float temp[6];
int k,j;
HAL_Init(); //初始化HAL库
Stm32_Clock_Init(96,4,2,4); //设置时钟,96Mhz
delay_init(96); //初始化延时函数
uart_init(115200); //初始化串口115200
LED_Init(); //初始化LED
// LED_SMG_Init(); //数码管初始化
Adc_Init(); //ADC初始化
MYDMA_Config(DMA2_Stream0,DMA_CHANNEL_0); //初始化DMA 数据流0,通道0
// TIM3_Init(20-1,9600-1); //数码管2ms定时显示
while(1)
{
for(j=0;j<4;j++)
{
value[j] = (u16)(ADC_ConvertedValue[j]);
printf("DMA DATA%d\t:%d \r\n",j,value[j]);//显示ADCC采样后的原始值
}
for(k=0; k<4; k++)
{
temp[k]=(float)value[k]*(3.3/4096); //获取计算后的带小数的实际电压值,比如3.1111
adcx=temp[k]; //赋值整数部分给adcx变量,因为adcx为u16整形
printf("Potentiometer voltage=%d", adcx);
temp[k]-=adcx; //把已经显示的整数部分去掉,留下小数部分,比如3.1111-3=0.1111
temp[k]*=1000;
//小数部分乘以1000,例如:0.1111就转换为111.1,相当于保留三位小数。
printf(".%d\r\n",value[k]);
}
delay_ms(500);
}
}
|
|