我要做的是以DMA接收1024筆ADC的資料然後處理,接著再接收另外1024筆ADC的資料然後處理的循環,所以必須自行控制ADC和DMA的開始時間。
目前遇到的問題是,重新啟動ADC和DMA一定次數就會產生overrun。
因為接收的timing很重要,所以希望能夠直接避免overrun的產生。
以下是我程式的相關部份:
uint16_t AD1_VALUE[1024] = {0};
uint16_t AD2_VALUE[1024] = {0};
void initADC(void){
ADC_InitTypeDef ADC_InitStructure;
ADC_CommonInitTypeDef ADC_CommonInitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2 | RCC_AHB1Periph_GPIOA, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC3, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ;
GPIO_Init(GPIOA, &GPIO_InitStructure);
ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div2;
ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled;
ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles;
ADC_CommonInit(&ADC_CommonInitStructure);
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_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfConversion = 1;
ADC_Init(ADC3, &ADC_InitStructure);
ADC_RegularChannelConfig(ADC3, ADC_Channel_3, 1, ADC_SampleTime_3Cycles);
ADC_DMARequestAfterLastTransferCmd(ADC3, DISABLE);
ADC_EOCOnEachRegularChannelCmd(ADC3, DISABLE);
ADC_DMACmd(ADC3, ENABLE);
ADC_Cmd(ADC3, ENABLE);
}
void initADC_DMA()
{
DMA_InitTypeDef DMA_InitStructure;
DMA_StructInit(&DMA_InitStructure);
DMA_InitStructure.DMA_Channel = DMA_Channel_2;
DMA_InitStructure.DMA_PeripheralBaseAddr = (unsigned int)(ADC3_BASE + 0x4c);
DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)AD1_VALUE;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
DMA_InitStructure.DMA_BufferSize = 1024;
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_Priority = DMA_Priority_VeryHigh;
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
DMA_DeInit(DMA2_Stream0);
DMA_Init(DMA2_Stream0, &DMA_InitStructure);
DMA_ITConfig(DMA2_Stream0, DMA_IT_TC, DISABLE);
DMA_Cmd(DMA2_Stream0, ENABLE);
}
main()
{
initADC();
initADC_DMA();
ADC_SoftwareStartConv(ADC3);
while(1) {
//do something
if (DMA_GetCmdStatus(DMA2_Stream0) == DISABLE) { //準備接收新的1024筆資料
ADC_DMACmd(ADC3, DISABLE);
while (ADC_GetFlagStatus(ADC3, ADC_FLAG_EOC) == RESET) ;
while (ADC3->SR & ADC_FLAG_EOC)
ADC_ClearFlag(ADC3, ADC_FLAG_EOC);
while (DMA_GetITStatus(DMA2_Stream0, DMA_IT_TCIF0) == SET)
DMA_ClearITPendingBit(DMA2_Stream0, DMA_IT_TCIF0);
while (DMA_GetFlagStatus(DMA2_Stream0, DMA_FLAG_TCIF0) == SET)
DMA_ClearFlag(DMA2_Stream0, DMA_FLAG_TCIF0);
DMA_SetCurrDataCounter(DMA2_Stream0, 1024);
DMA2_Stream0->M0AR = (i & 0x01 ? (uint32_t)AD1_VALUE: (uint32_t)AD2_VALUE);
DMA_Cmd(DMA2_Stream0, ENABLE);
ADC_DMACmd(ADC3, ENABLE);
ADC_SoftwareStartConv(ADC3);
//overrun在這裡被發現
}
}
}
謝謝。