STM32F407的ADC DMA采样双缓冲实现

 

volatile UINT8 nSendFlag = 0xFF;

#define ADC_BUFFER_LEN_HALF     256

UINT16 g_ADC_Buffer[ADC_BUFFER_LEN_HALF] = {0};
UINT16 g_ADC_Buffer_2[ADC_BUFFER_LEN_HALF] = {0};

 

void ADC1_DMA_Config()
{
    DMA_InitTypeDef DMA_InitStructure;
    NVIC_InitTypeDef NVIC_InitStructure;
    
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2, ENABLE);
    DMA_DeInit(DMA2_Stream0);
    while(DMA_GetCmdStatus(DMA2_Stream0) != DISABLE){}
    // DMA SET
    DMA_InitStructure.DMA_Channel                 = DMA_Channel_0;
    DMA_InitStructure.DMA_PeripheralBaseAddr     = (uint32_t)&ADC1->DR,
    DMA_InitStructure.DMA_Memory0BaseAddr         = (uint32_t)g_ADC_Buffer;
    DMA_InitStructure.DMA_DIR                    = DMA_DIR_PeripheralToMemory;
    DMA_InitStructure.DMA_BufferSize            = ADC_BUFFER_LEN;
    DMA_InitStructure.DMA_PeripheralInc            = DMA_PeripheralInc_Disable;
    DMA_InitStructure.DMA_MemoryInc             = DMA_MemoryInc_Enable;
    DMA_InitStructure.DMA_PeripheralDataSize    = DMA_PeripheralDataSize_HalfWord;
    DMA_InitStructure.DMA_MemoryDataSize        = DMA_MemoryDataSize_HalfWord;
    DMA_InitStructure.DMA_Mode                    = DMA_Mode_Circular;
    DMA_InitStructure.DMA_Priority                = DMA_Priority_High;
    DMA_InitStructure.DMA_FIFOMode                = DMA_FIFOMode_Disable;
    DMA_InitStructure.DMA_FIFOThreshold            = DMA_FIFOThreshold_Full;
    DMA_InitStructure.DMA_MemoryBurst            = DMA_MemoryBurst_Single;
    DMA_InitStructure.DMA_PeripheralBurst         = DMA_PeripheralBurst_Single;
    
    DMA_DoubleBufferModeConfig(DMA2_Stream0,(uint32_t)&g_ADC_Buffer_2,DMA_Memory_0);//DMA_Memory_0?????
    DMA_DoubleBufferModeCmd(DMA2_Stream0,ENABLE);
    DMA_Init(DMA2_Stream0, &DMA_InitStructure); 
        
    // NVIC
    //DMA_ClearITPendingBit(DMA2_Stream0,DMA_IT_HTIF0);
    //DMA_ClearITPendingBit(DMA2_Stream0,DMA_IT_TCIF0);
    //DMA_ITConfig(DMA2_Stream0,DMA_IT_TC,ENABLE);
    //DMA_ITConfig(DMA2_Stream0,DMA_IT_HT,ENABLE);    
        
    NVIC_InitStructure.NVIC_IRQChannel=DMA2_Stream0_IRQn; 
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0x02;   
    NVIC_InitStructure.NVIC_IRQChannelSubPriority=0x02;                      
    NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;
    NVIC_Init(&NVIC_InitStructure);
    DMA_ClearITPendingBit(DMA2_Stream0,DMA_IT_TCIF0);
    DMA_ITConfig(DMA2_Stream0,DMA_IT_TC,ENABLE);
    
    //while (DMA_GetCmdStatus(DMA2_Stream0) != DISABLE){}
    DMA_Cmd(DMA2_Stream0, ENABLE);
    
}

void ADC1_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
     ADC_CommonInitTypeDef ADC_CommonInitStructure;
    ADC_InitTypeDef ADC_InitStructure;
    
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
    // PA5
    GPIO_InitStructure.GPIO_Pin        = GPIO_Pin_5;//GPIO_Pin_5, PA5, PA6
    GPIO_InitStructure.GPIO_Mode     = GPIO_Mode_AN;
    GPIO_InitStructure.GPIO_PuPd    = GPIO_PuPd_NOPULL;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    
    //ADC1
    RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC1, ENABLE);
    RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC1, DISABLE);
    // Common Set
    ADC_CommonInitStructure.ADC_Mode    = ADC_Mode_Independent;
    ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles;
    ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4;
    ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_1;
    // ADC Set
    ADC_InitStructure.ADC_Resolution    = ADC_Resolution_12b;
    ADC_InitStructure.ADC_ScanConvMode  = DISABLE;
    ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
    ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;
    ADC_InitStructure.ADC_ExternalTrigConv  = ADC_ExternalTrigConv_T1_CC1;
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//ADC_DataAlign_Left;//ADC_DataAlign_Right;
    ADC_InitStructure.ADC_NbrOfConversion = 1;
    ADC_Init(ADC1, &ADC_InitStructure);    
    ADC_Cmd(ADC1, ENABLE);
    
    ADC1_DMA_Config();
    ADC_DMACmd(ADC1, ENABLE);
    
    ADC_RegularChannelConfig(ADC1, ADC_Channel_5, 1, ADC_SampleTime_15Cycles); //ADC_SampleTime_3Cycles
    ADC_DMARequestAfterLastTransferCmd(ADC1, ENABLE);
    
}


void DMA2_Stream0_IRQHandler(void) 
{

    if (DMA_GetITStatus(DMA2_Stream0, DMA_IT_TCIF0) == SET)  
    {
        DMA_ClearITPendingBit(DMA2_Stream0, DMA_IT_TCIF0);
        if(DMA_GetCurrentMemoryTarget(DMA2_Stream0) == 1) //
        {
            nSendFlag = 1;
        }else{
            nSendFlag = 2;
        }
    }    
}

void Net_Send(....)

{

     .....

}

void Send_ADC_Data(void){

          if(ADC_Status.nSFlag == 1){
                            
                     Net_Send(CMD_DATA_TEST_1 ADC_Status.nID, g_ADC_Buffer);
                      ADC_Status.nSFlag = 0xFF;
                      memset(g_ADC_Buffer, 0, ADC_BUFFER_LEN_HALF);
            }else if(ADC_Status.nSFlag == 2){{
                       Net_Send(CMD_DATA_TEST_2, ADC_Status.nID, g_ADC_Buffer_2);
                       ADC_Status.nSFlag = 0xFF;
                       memset(g_ADC_Buffer_2, 0, ADC_BUFFER_LEN_HALF); 
             }

 }

 

int main(void)

{

          ........

          ADC1_Init();

         ......

     ADC_SoftwareStartConv(ADC1);

       while(1)

     {

            Send_ADC_Data();

     }

 

}

STM32F4定时器触发ADC DMA双缓冲的方法如下: 首先,配置定时器,设置计数器的自动重装载值和预分频器,以确定定时器的计数周期和触发频率。 然后,配置ADC,选择需要转换的通道和采样速率,使其准备好接收转换请求。 接下来,配置DMA,设置双缓冲模式,分配两个数据缓冲区,一个用于DMA传输期间,一个用于处理数据。设置DMA传输长度和目的地址,以便将ADC数据直接传输到缓冲区。 在启动定时器之前,启动DMA传输,并将DMA请求与定时器的触发事件相关联。这样,当定时器满足触发条件时,ADC将自动进行转换,并且转换完成的数据将通过DMA传输到缓冲区。 在主循环中,检测DMA传输完成事件,并根据需要处理接收到的数据。此时,可以开始对缓冲区中的数据进行处理,例如计算平均值、滤波或其他处理方式。 同时,在DMA传输完成后,需要交换两个缓冲区的角色,使之成为当前处理数据的缓冲区。这样,当下一次DMA传输完成时,可以将数据传输到另一个缓冲区,以保证数据的连续传输。 最后,根据需要,可以选择暂停或停止定时器和DMA传输,以便在不需要时节省功耗或进行其他操作。 总结:通过配置STM32F4的定时器、ADCDMA,可以实现定时触发ADC转换并通过DMA进行双缓冲传输的功能。这种方式可以提高数据处理的效率和精度,并且减少CPU的负载。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值