[mw_shl_code=c,true]volatile u8 ADCfinishsign = 0;
u16 sensor[3]={1,1,1};
u16 ADCval[adccount][adcchcount];
/***********************************************************************************
单通道,连续转换,得到10组数据后停止,启动下一通道
ADC_Channel_12,ADC_Channel_13,ADC_Channel_14
***********************************************************************************/
void Adc_Init(void)
{
ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOC|RCC_APB2Periph_ADC1, ENABLE ); //使能ADC1通道时钟
RCC_ADCCLKConfig(RCC_PCLK2_Div6); //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M
//PA1 作为模拟通道输入引脚
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚
GPIO_Init(GPIOC, &GPIO_InitStructure);
ADC_DeInit(ADC1); //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在独立模式
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; //模数转换工作在连续转换模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //转换由软件而不是外部触发启动
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = 3; //顺序进行规则转换的ADC通道的数目
ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器
ADC_RegularChannelConfig(ADC1, ADC_Channel_12, 1, ADC_SampleTime_71Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 2, ADC_SampleTime_71Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 3, ADC_SampleTime_71Cycles5 );
ADC_DMACmd(ADC1, ENABLE);
ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1
ADC_ResetCalibration(ADC1); //使能复位校准
while(ADC_GetResetCalibrationStatus(ADC1))
; //等待复位校准结束
ADC_ResetCalibration(ADC1); //使能复位校准
while(ADC_GetResetCalibrationStatus(ADC1))
; //等待复位校准结束
}
/***********************************************************************************
***********************************************************************************/
void ADC_Start(void)
{
DMA_Cmd( DMA1_Channel1, DISABLE );
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能
DMA1_Channel1->CNDTR = adccount*adcchcount;
DMA_Cmd(DMA1_Channel1, ENABLE);
}
/***********************************************************************************
***********************************************************************************/[/mw_shl_code]
[mw_shl_code=c,true]#define adcchcount 3[/mw_shl_code]
[mw_shl_code=c,true] #define adccount 10[/mw_shl_code]
[mw_shl_code=c,true]void GetADCAverage(void)
{
u8 i,j;
u16 temmin=0,temmax=0,temsum=0;
if(ADCfinishsign)
{
ADCfinishsign=0;
for( j=0; j<adcchcount; j++ )[/mw_shl_code]
[mw_shl_code=c,true] {
temmin = ADCval[0][j];
temmax = ADCval[0][j];
temsum=0;
for( i=0; i<adccount; i++ )
{
temsum += ADCval[j];
if( ADCval[j]<temmin )
{
temmin = ADCval[j];
}
else
if( ADCval[j]>temmax )
{
temmax = ADCval[j];
}
}
temsum -= (temmin+temmax);
sensor[j] = temsum/(adccount-2);
}
}
}
[/mw_shl_code]
[mw_shl_code=c,true]void DMA1_Channel1Config(void)
{
DMA_InitTypeDef DMA_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA传输
DMA_DeInit(DMA1_Channel1);
DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&ADC1->DR;
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADCval;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize = adccount*adcchcount;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //数据宽度为16位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //数据宽度为16位
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //工作在正常缓存模式
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //DMA通道 x拥有中优先级
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
DMA_ITConfig(DMA1_Channel1,DMA_IT_TC,ENABLE); //发送完成中断
NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
}
/***********************************************************************************
***********************************************************************************/
void DMA1_Channel1_IRQHandler(void)
{
if(DMA1->ISR&(1<<1))
{
ADC_SoftwareStartConvCmd(ADC1, DISABLE);
DMA1->IFCR|=(1<<1);
ADCfinishsign=1;
}
}
[/mw_shl_code]
我是这样设计的:ADC扫描模式,PC2、3、4。DMA只传输一次(非循环模式),启动ADC后设置DMA传输的数量,DMA传输完成中断关闭ADC。并将标志置位。
目的是:每次打开ADC后转换10次然后关闭。
结果:14通道接到3.3V,其他接地。可以看到14通道的结果乱跑。
|