Вот, нашёл такое на одном из форумов:
CODE
// адрес регистра данных, откуда будем брать результат конвертации
#define ADC_CDR_ADDRESS ((uint32_t)0x40012308)
// переменные, описывающие свойства периферии
ADC_InitTypeDef ADC_InitStructure;
ADC_CommonInitTypeDef ADC_CommonInitStructure;
DMA_InitTypeDef DMA_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
u32 ADCDualConvertedValue;
void main(void)
{
ADCDualConvertedValue=0;
// разрешаем тактирование используемых периферий
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2 | RCC_AHB1Periph_GPIOC, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_ADC2, ENABLE);
// описываем свойства порта ввода-вывода по которому будем производить оцифровку
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ;
GPIO_Init(GPIOC, &GPIO_InitStructure);
// описываем свойства DMA через который будет вестись обмен
DMA_InitStructure.DMA_Channel = DMA_Channel_0;
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)ADC_CDR_ADDRESS; // адрес регистра данных АЦП
DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)&ADCDualConvertedValue;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
DMA_InitStructure.DMA_BufferSize = 1;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;
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_HalfFull;
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
DMA_Init(DMA2_Stream0, &DMA_InitStructure);
// запуск DMA
DMA_Cmd(DMA2_Stream0, ENABLE);
// отключаем АЦП1 2
ADC_Cmd(ADC1, DISABLE);
ADC_Cmd(ADC2, DISABLE);
// описываем АЦП
// общие параметры работы АЦП
ADC_CommonInitStructure.ADC_Mode = ADC_DualMode_RegSimult;
ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_6Cycles;
ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_2;
ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div2;
ADC_CommonInit(&ADC_CommonInitStructure);
ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;
// настраиваем АЦП1
ADC_InitStructure.ADC_ScanConvMode = DISABLE;
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_Rising;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T8_TRGO;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfConversion = 1;
ADC_Init(ADC1, &ADC_InitStructure);
// указывем канал, подлежащий оцифровке (температура)
ADC_RegularChannelConfig(ADC1, ADC_Channel_16, 1, ADC_SampleTime_3Cycles);
// разрешаем работу датчика температуры
ADC_TempSensorVrefintCmd (ENABLE);
// настраиваем АЦП2
ADC_Init(ADC2, &ADC_InitStructure);
// указывем канал, подлежащий оцифровке
ADC_RegularChannelConfig(ADC2, ADC_Channel_12, 1, ADC_SampleTime_3Cycles);
// разрешаем выполнять запросы DMA
ADC_MultiModeDMARequestAfterLastTransferCmd(ENABLE);
// запускаем АЦП1
ADC_Cmd(ADC1, ENABLE);
// запускаем АЦП2
ADC_Cmd(ADC2, ENABLE);
// настраиваем таймер
TIM_Cmd(TIM8, DISABLE);
// подключаем тактирование
RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM8, ENABLE);
// описываем таймер
// TIM_TimeBaseStructure.TIM_Period = (560000 /125);
// TIM_TimeBaseStructure.TIM_Prescaler = 0x00;
// делитель 33600 - частота прерываний = 84MHz/33600= 2500 = 2.5 kHz
TIM_TimeBaseStructure.TIM_Prescaler = 33600-1;
// Период загружается в ARR - значение перезагрузки. Счётчик считает 0..ARR
TIM_TimeBaseStructure.TIM_Period = 2; //период 2 импульсов - частота 1250 гц
TIM_TimeBaseStructure.TIM_ClockDivision = 0x0;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;
// записываем описание в регистры
TIM_TimeBaseInit(TIM8, &TIM_TimeBaseStructure);
// сбрасываем, указываем выходной триггер (для запуска АЦП)
TIM_SetCounter(TIM8, 0);
TIM_SelectOutputTrigger(TIM8, TIM_TRGOSource_Update);
// запуcкаем таймер
TIM_Cmd(TIM8, ENABLE);
while (1)
{
} // while
} // main
ADC1 цифрует свой канал 12 (PC2), ADC2 внутренний датчик температуры. Запуск по TIM8, результат в ADCDualConvertedValue.
Оно работает вроде нормально, наблюдал переменную под отладчиком. Осталось только понять, как повесить на DMA обработчик по заполнению буфера.
И непонятно, откуда взялся адрес ADC_CDR_ADDRESS. Из него DMA результаты читает.
У меня тут написано
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(ADC1->DR);
Т.е. регистр данных ADC1.
Мой код зависал из-за отсутствия этого самого обработчика прерываний DMA.
Убрал у себя вот этот кусок:
Код
// Enable the DMA Stream IRQ Channel
NVIC_InitStructure.NVIC_IRQChannel = DMA2_Stream0_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init ( &NVIC_InitStructure );
DMA_ITConfig ( DMA2_Stream0, DMA_IT_HT | DMA_IT_TC, ENABLE );
Виснуть перестало, но в буфере нули...
Может потому, что я с разгону написал тот же ADC_CDR_ADDRESS, что и в примере выше. У меня ведь один АЦП и один канал.
В итоге пока что хрень

.
Кто-нибудь знает, как написать обработчик прерываний DMA?
Обработчик прерываний DMA для F4:
// dma2 stream 0 irq handler
void DMA2_Stream0_IRQHandler ( void )
{
}
Только что в нём писать ?...
Чтобы возить такого пассажира, необходим лимузин другого класса.
(с) Мария Эдуарда