• STM32cube库配置双ADC的同步规则采样


    http://www.stmcu.org/module/forum/forum.php?mod=viewthread&tid=605203&extra=page%3D&page=1

    ADC1与ADC2工作在规则同步扫描模式,TIM4CC4触发,DMA读取ADC结果;设计目标是每周波(20毫秒)采样32点或64点或128点等数据,TIM4的初始化程序自动计算采样间隔,例如如果每周期采样32个点的数据,那么TIM4的触发间隔是20*1000/32=625微秒,也就是每625微秒触发一次ADC,由于使能了ADC的扫描模式,所以一次触发转换6个ADC通道产生6个32位的数据(在F103上采用的是间断模式,而F407的间断模式没有使用起来,只好用扫描模式),每个通道转换结束后由DMA读取转换结果保存到内存缓冲区等待计算,TIM4一次触发ADC后DMA读取6个32位数据,DMA计数到32*6=192个数据后产生中断,DMA的实际中断间隔是20毫秒,在DMA中断内重新设置DMA参数同时设置转换结束标志供计算使用。
    void TIMx_Configuration(void)
    {
            Uint32 Temp;
            RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);        
            TIM_InternalClockConfig(TIM4);
            TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);

            if(((RCC->CFGR >> 10) & 0x04)==0)
            {
                    Temp=1;
            }
            else
            {
                    Temp=2;
            }
            TIM_TimeBaseStructure.TIM_Period = TIM4_ClkFre/AdcPrNum/50;
            TIM_TimeBaseStructure.TIM_Prescaler = APB1CLK*Temp/TIM4_ClkFre-1;
            TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
            TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Down;
            TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);        

            TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
            TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;            
            TIM_OCInitStructure.TIM_Pulse=TIM_TimeBaseStructure.TIM_Period/10;        
            TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
            TIM_OC4Init(TIM4, &TIM_OCInitStructure);

            TIM_OC4PreloadConfig(TIM4, TIM_OCPreload_Enable);        
             TIM_Cmd(TIM4, ENABLE);
    }
    void ADC_Configuration(void)
    {
            RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC,ENABLE);
            RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);        
            
            ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;
            ADC_InitStructure.ADC_ScanConvMode = ENABLE;
            ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
            ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_Rising;
            ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T4_CC4;
            ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
            ADC_InitStructure.ADC_NbrOfConversion = AdcChNum;
            ADC_Init(ADC1,&ADC_InitStructure);
            
            ADC_CommonInitStructure.ADC_Mode = ADC_DualMode_RegSimult;
            ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div6;
            ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_2;
            ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_10Cycles;
            ADC_CommonInit(&ADC_CommonInitStructure);
            
            ADC_DiscModeChannelCountConfig(ADC1,AdcChNum);        
            ADC_DMARequestAfterLastTransferCmd(ADC1, ENABLE);
            ADC_DMACmd(ADC1, ENABLE);
            ADC_Cmd(ADC1, ENABLE);
            
            ADC1->SMPR2 |= ADC_SMPR_13_5 << 0;
            ADC1->SMPR2 |= ADC_SMPR_13_5 << 3;
            ADC1->SMPR2 |= ADC_SMPR_13_5 << 6;
            ADC1->SMPR2 |= ADC_SMPR_13_5 << 9;
            ADC1->SMPR2 |= ADC_SMPR_13_5 << 12;
            ADC1->SMPR2 |= ADC_SMPR_13_5 << 15;
            ADC1->SMPR2 |= ADC_SMPR_13_5 << 18;
            ADC1->SMPR2 |= ADC_SMPR_13_5 << 21;
            ADC1->SMPR2 |= ADC_SMPR_13_5 << 24;
            ADC1->SMPR2 |= ADC_SMPR_13_5 << 27;        

            ADC1->SMPR1 |= ADC_SMPR_13_5 << 0;
            ADC1->SMPR1 |= ADC_SMPR_13_5 << 3;
            ADC1->SMPR1 |= ADC_SMPR_13_5 << 6;
            ADC1->SMPR1 |= ADC_SMPR_13_5 << 9;
            ADC1->SMPR1 |= ADC_SMPR_13_5 << 12;
            ADC1->SMPR1 |= ADC_SMPR_13_5 << 15;
            ADC1->SMPR1 |= ADC_SMPR_13_5 << 18;
            ADC1->SMPR1 |= ADC_SMPR_13_5 << 21;        
                   
            ADC1->SQR3 |= UL1AinCH << 0;
            ADC1->SQR3 |= UczAinCH << 5;
            ADC1->SQR3 |= UaAinCH << 10;
            ADC1->SQR3 |= UbAinCH << 15;
            ADC1->SQR3 |= UcAinCH << 20;
            ADC1->SQR3 |= U0AinCH << 25;
            ADC1->SQR2 |= 7 << 0;               
            ADC1->SQR2 |= 8 << 5;
            ADC1->SQR2 |= 9 << 10;
            ADC1->SQR2 |= 10 << 15;
            ADC1->SQR2 |= 11 << 20;
            ADC1->SQR2 |= 12 << 25;
            ADC1->SQR1 |= 13 << 0;
            ADC1->SQR1 |= 14 << 5;
            ADC1->SQR1 |= 15 << 10;
            ADC1->SQR1 |= 16 << 15;
            ADC1->SQR1 |= (AdcChNum-1) << 20;        


            RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC2,ENABLE);        
            
            ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;
            ADC_InitStructure.ADC_ScanConvMode = ENABLE;
            ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
            ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;
            ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T4_CC4;
            ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
            ADC_InitStructure.ADC_NbrOfConversion = AdcChNum;
            ADC_Init(ADC2,&ADC_InitStructure);
            
            ADC_CommonInitStructure.ADC_Mode = ADC_DualMode_RegSimult;
            ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div6;
            ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_2;
            ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_10Cycles;
            ADC_CommonInit(&ADC_CommonInitStructure);
            
            ADC_DiscModeChannelCountConfig(ADC2,AdcChNum);        
            ADC_DMARequestAfterLastTransferCmd(ADC2, ENABLE);
            ADC_DMACmd(ADC2, ENABLE);
            ADC_Cmd(ADC2, ENABLE);
            
            ADC2->SMPR2 |= ADC_SMPR_13_5 << 0;
            ADC2->SMPR2 |= ADC_SMPR_13_5 << 3;
            ADC2->SMPR2 |= ADC_SMPR_13_5 << 6;
            ADC2->SMPR2 |= ADC_SMPR_13_5 << 9;
            ADC2->SMPR2 |= ADC_SMPR_13_5 << 12;
            ADC2->SMPR2 |= ADC_SMPR_13_5 << 15;
            ADC2->SMPR2 |= ADC_SMPR_13_5 << 18;
            ADC2->SMPR2 |= ADC_SMPR_13_5 << 21;
            ADC2->SMPR2 |= ADC_SMPR_13_5 << 24;
            ADC2->SMPR2 |= ADC_SMPR_13_5 << 27;        

            ADC2->SMPR1 |= ADC_SMPR_13_5 << 0;
            ADC2->SMPR1 |= ADC_SMPR_13_5 << 3;
            ADC2->SMPR1 |= ADC_SMPR_13_5 << 6;
            ADC2->SMPR1 |= ADC_SMPR_13_5 << 9;
            ADC2->SMPR1 |= ADC_SMPR_13_5 << 12;
            ADC2->SMPR1 |= ADC_SMPR_13_5 << 15;
            ADC2->SMPR1 |= ADC_SMPR_13_5 << 18;
            ADC2->SMPR1 |= ADC_SMPR_13_5 << 21;
            
            ADC2->SQR3 |= UL2AinCH << 0;
            ADC2->SQR3 |= UdcAinCH << 5;
            ADC2->SQR3 |= IaAinCH << 10;
            ADC2->SQR3 |= IbAinCH << 15;
            ADC2->SQR3 |= IcAinCH << 20;
            ADC2->SQR3 |= I0AinCH << 25;
            ADC2->SQR2 |= 7 << 0;        
            ADC2->SQR2 |= 8 << 5;
            ADC2->SQR2 |= 9 << 10;
            ADC2->SQR2 |= 10 << 15;
            ADC2->SQR2 |= 11 << 20;
            ADC2->SQR2 |= 12 << 25;
            ADC2->SQR1 |= 13 << 0;
            ADC2->SQR1 |= 14 << 5;
            ADC2->SQR1 |= 15 << 10;
            ADC2->SQR1 |= 16 << 15;
            ADC2->SQR1 |= (AdcChNum-1) << 20;
    }
    void DMA2_Stream0_IRQHandler(void)
    {
            #if(DMA2_STREAM0)
            if(DMA2->LISR & (1 << 5))
            {
                    DMA2->LIFCR |= (1<<5);
                    DMA_Cmd(DMA2_Stream0, DISABLE);
                    DMA2_Stream0->NDTR = AdcPrNum*AdcChNum;
                    DMA2_Stream0->M0AR = (u32)&ADC1_Buf;
                    DMA_Cmd(DMA2_Stream0, ENABLE);
            }
            #endif
    }
    AdcPrNum 为每周期采样点数;AdcChNum 为通道个数,例如需要采样UA、UB、UC、IA、IB、IC6路模拟信号,则AdcChNum =6/2=3;
    void DMA_Configuration(void)
    {
            RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2,ENABLE);
            DMA_DeInit(DMA2_Stream0);

            NVIC_InitStructure.NVIC_IRQChannel = DMA2_Stream0_IRQChannel;
            NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
            NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;
            NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
            NVIC_Init(&NVIC_InitStructure);

            DMA_InitStructure.DMA_Channel=DMA2_STREAM0_CH0_ADC1;
            DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&ADC->CDR;
            DMA_InitStructure.DMA_Memory0BaseAddr = (u32)&ADC1_Buf;
            DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
            DMA_InitStructure.DMA_BufferSize = AdcPrNum*AdcChNum;
            
            DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
            DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;               
            DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;        
            DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;
            DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
            DMA_InitStructure.DMA_Priority = DMA_Priority_High;
            DMA_InitStructure.DMA_FIFOMode=DMA_FIFOMode_Disable;
            DMA_InitStructure.DMA_FIFOThreshold=DMA_FIFOThreshold_1QuarterFull;
            DMA_InitStructure.DMA_MemoryBurst=DMA_MemoryBurst_Single;
            DMA_InitStructure.DMA_PeripheralBurst=DMA_PeripheralBurst_Single;               
                            
            DMA_Init(DMA2_Stream0, &DMA_InitStructure);

            DMA_ITConfig(DMA2_Stream0, DMA_IT_TC , ENABLE);
            DMA_Cmd(DMA2_Stream0, ENABLE);
    }

  • 相关阅读:
    hdu1151 二分图(无回路有向图)的最小路径覆盖 Air Raid
    二分图多重匹配问题
    二分图最大匹配问题及其扩展
    ZOJ3741 状压DP Eternal Reality
    POJ2699:The Maximum Number of Strong Kings(枚举+贪心+最大流)
    POJ2396:Budget(带下界的网络流)
    POJ2391:Ombrophobic Bovines(最大流+Floyd+二分)
    POJ1637:Sightseeing tour(混合图的欧拉回路)
    URAL1277 Cops and Thieves(最小割)
    Leetcode 44. Wildcard Matching
  • 原文地址:https://www.cnblogs.com/zym0805/p/7047753.html
Copyright © 2020-2023  润新知