реклама на сайте
подробности

 
 
 
Reply to this topicStart new topic
> STM32F103 DMA half transfer
uriy
сообщение Dec 27 2014, 17:53
Сообщение #1


Гуру
******

Группа: Свой
Сообщений: 2 429
Регистрация: 30-11-05
Из: Ижевск
Пользователь №: 11 606



Использую ADC+TIM3+DMA
Хочу получить прерывание после завершения преобразования половины буфера.
Но когда попадаю в прерывание всегда оказываются взведены все три бита DMA1_IT_HT1, DMA1_IT_TC1 и DMA1_IT_GL1
В регистре DMA1_ISR = 0x07 и буфер заполнен полностью.

Обработчик прерывания:
Код
void DMA1_Channel1_IRQHandler(void)
{
    if (DMA_GetITStatus(DMA1_IT_HT1))
    {
        DMA_ClearITPendingBit(DMA1_IT_HT1);
        test = 1;
    }
    if (DMA_GetITStatus(DMA1_IT_TC1))
    {
        DMA_ClearITPendingBit(DMA1_IT_TC1);
        test = 2;
    }
#warning remove it
    memset(ADCBuffer, 0, sizeof(ADCBuffer));
    DMA_ClearITPendingBit(DMA1_IT_GL1);
}


Ниже инициализация таймера и ADC
CODE
#define ADC1_DR_Address ((uint32_t)0x4001244C)
uint16_t ADCBuffer[1000];

static void InitTimer3(void)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;

/* TIM3 clock enable */
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);

/* Time base configuration */
TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);
TIM_TimeBaseStructure.TIM_Period = 0x19;
TIM_TimeBaseStructure.TIM_Prescaler = 0x0;
TIM_TimeBaseStructure.TIM_ClockDivision = 0x0;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);

/* TIM3 TRGO selection */
TIM_SelectOutputTrigger(TIM3, TIM_TRGOSource_Update);

/* TIM3 enable counter */
//TIM_Cmd(TIM3, ENABLE);
}


CODE
void InitADC(void)
{
ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
DMA_InitTypeDef DMA_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_2;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOC, &GPIO_InitStructure);

/* DMA1 channel1 configuration ----------------------------------------------*/
DMA_DeInit(DMA1_Channel1);
DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;
DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&ADCBuffer;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize = sizeof(ADCBuffer);
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_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
//DMA_ITConfig(DMA1_Channel1, (DMA_IT_TC | DMA_IT_HT), ENABLE);
DMA_ClearITPendingBit(DMA1_IT_HT1);
DMA_ClearITPendingBit(DMA1_IT_TC1);
DMA_ITConfig(DMA1_Channel1, DMA1_IT_HT1, ENABLE);
/* Enable DMA1 channel1 */
DMA_Cmd(DMA1_Channel1, ENABLE);

ADC_ExternalTrigConvCmd(ADC1, ENABLE);

NVIC_InitTypeDef NVIC_InitStructure;
NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
NVIC_EnableIRQ(DMA1_Channel1_IRQn);

/* ADC1 configuration ------------------------------------------------------*/
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T3_TRGO;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfChannel = 2;
ADC_Init(ADC1, &ADC_InitStructure);

/* ADC1 regular channel Configuration */
ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 1, ADC_SampleTime_239Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 2, ADC_SampleTime_239Cycles5);

/* Enable ADC1 DMA */
ADC_DMACmd(ADC1, ENABLE);

/* Enable ADC1 */
ADC_Cmd(ADC1, ENABLE);

/* Enable ADC1 reset calibaration register */
ADC_ResetCalibration(ADC1);
/* Check the end of ADC1 reset calibration register */
while (ADC_GetResetCalibrationStatus(ADC1));

/* Start ADC1 calibaration */
ADC_StartCalibration(ADC1);
/* Check the end of ADC1 calibration */
while (ADC_GetCalibrationStatus(ADC1));

}


Вызываю их в таком порядке:
Код
    InitTimer3();
    InitADC();
    TIM_Cmd(TIM3, ENABLE);
    TIM_DMACmd(TIM3, TIM_DMA_Update, ENABLE);


Повторю что АЦП работает. Данные в буфер складыаются из обоих каналов как надо.
Но когда попадаю в прерыание уже заполнены все 1000 элементов массива.
Вместо ожидаемых 500.
И установлены все три бита прерывания: заполнена половина массива, заполнен весь массив и бит глобального прерывания

Сообщение отредактировал IgorKossak - Dec 28 2014, 19:13
Причина редактирования: [codebox] для длинного кода, [code] - для короткого!!!
Go to the top of the page
 
+Quote Post
Golikov A.
сообщение Dec 28 2014, 09:50
Сообщение #2


Гуру
******

Группа: Свой
Сообщений: 4 256
Регистрация: 17-02-06
Пользователь №: 14 454



а как массив очищается? когда перезапускаеться ДМА?
потому что может вы пропустили тот момент когда была половина, и уже попадаете когда все и половина в том числе?
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Dec 28 2014, 10:04
Сообщение #3


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Может я совсем не понимаю этих библиотек, но мне кажется что вот тут вы разрешаете только одно из прерываний:
Цитата(uriy @ Dec 27 2014, 19:53) *
Код
    //DMA_ITConfig(DMA1_Channel1, (DMA_IT_TC | DMA_IT_HT), ENABLE);
    DMA_ClearITPendingBit(DMA1_IT_HT1);
    DMA_ClearITPendingBit(DMA1_IT_TC1);
    DMA_ITConfig(DMA1_Channel1, DMA1_IT_HT1, ENABLE);


Поэтому по остальным вы в обработчик не попадаете.


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
uriy
сообщение Dec 29 2014, 07:47
Сообщение #4


Гуру
******

Группа: Свой
Сообщений: 2 429
Регистрация: 30-11-05
Из: Ижевск
Пользователь №: 11 606



Цитата
потому что может вы пропустили тот момент когда была половина, и уже попадаете когда все и половина в том числе?

Как это? Прерывание half-transfer должно возникнуть когда через DMA прошла половина отсчетов. А именно половина вот этого
DMA_InitStructure.DMA_BufferSize = sizeof(ADCBuffer);
При старте массив пустой. При первом входе в прерывание я хочу увидеть наполовину заполеннный массив.
Разве не так?
Ну и для тестов я еще вот это сделал в прерывании memset(ADCBuffer, 0, sizeof(ADCBuffer));

У меня используется два входа ADC. Один подцеплен на землю, на другом половина питания.
DMA складвает в буффер поочередно значения близкие к 0 и к 2048.

Цитата
а как массив очищается? когда перезапускаеться ДМА?
А зачем его очищать. Данные просто будут писаться поверх старых.
Но сейчас все-таки для наглядности он очищается внутри прерывания, выше упомянутой memset(ADCBuffer, 0, sizeof(ADCBuffer));

Следующая строка действительно включает прерывания тольо по half transfer DMA1_IT_HT1
DMA_ITConfig(DMA1_Channel1, DMA1_IT_HT1, ENABLE);
Но ведь при первом входе в прерывание должна быть заполнена только половина буфера.
А на самом деле буфер заполнен полность.
Но вашу мысль все равно проверю. Спасибо.

Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Dec 29 2014, 08:13
Сообщение #5


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(uriy @ Dec 29 2014, 09:47) *
А на самом деле буфер заполнен полностью.
Так пока отладчик остановил все, пока вычитал регистры, память - таймер уже успел намолотить новых отсчетов. Думаю, что если вы сделаете один шаг, то увидите, что весь массив снова обновился. Надо где-то выставить битик, чтобы таймера останавливались при входе в отладку и прекращали запускать АЦП снова и снова.


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
Golikov A.
сообщение Dec 29 2014, 12:00
Сообщение #6


Гуру
******

Группа: Свой
Сообщений: 4 256
Регистрация: 17-02-06
Пользователь №: 14 454



ну или регистр флагов сразу сохранить и остановиться на след строчке, посмотреть какой он был при входе
Go to the top of the page
 
+Quote Post

Reply to this topicStart new topic
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 


RSS Текстовая версия Сейчас: 23rd July 2025 - 03:55
Рейтинг@Mail.ru


Страница сгенерированна за 0.02868 секунд с 7
ELECTRONIX ©2004-2016