Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: STM32F205, не могу подружить DMA и GPIO
Форум разработчиков электроники ELECTRONIX.ru > Сайт и форум > В помощь начинающему > ARM, 32bit
A.Lex
Простая задача: принять данные с параллельного порта (внешнее АЦП) и разместить в буфере.

Тактирование с TIM8.

Это не коммерческий проект, попробовать. И не работает. Если в обработчике прерывания раскомментировать строку

//Parallel_Data_Buffer[i] = GPIOB->IDR;

буфер заполняется. А вот DMA в буфер ничего не пишет. Чувствую, что накосячил, но чтение мануала и апнот не помогло.

Код
#define DMA_STREAM DMA2_Stream1
#define DMA_CHANNEL    DMA_Channel_7
#define BUFFER_SIZE    127

void dma_init(void)
{

RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2, ENABLE);    
    

  /* GPIOA and GPIOD clock enable */
      RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);  

  /* GPIO Configuration ------------------------------------------------------*/
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_3
                                | GPIO_Pin_4 | GPIO_Pin_5
                              | GPIO_Pin_6 | GPIO_Pin_7
                              | GPIO_Pin_8 | GPIO_Pin_9
                              | GPIO_Pin_10;    
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
  GPIO_Init(GPIOB, &GPIO_InitStructure);


    DMA_Cmd(DMA_STREAM, DISABLE);
    DMA_DeInit(DMA_STREAM);

  /* DMA Channel5 Configuration ----------------------------------------------*/
  /* Configure DMA Stream      */
  DMA_InitStructure.DMA_Channel = DMA_CHANNEL;  
  DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&GPIOB->IDR;
  DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t) Parallel_Data_Buffer[0];
  DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
  DMA_InitStructure.DMA_BufferSize = (uint32_t)BUFFER_SIZE;
  DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
  DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //_Word;
  DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //_Word;
  DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //  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_Init(DMA_STREAM, &DMA_InitStructure);                      
    
  /* Enable DMA Stream Transfer Complete interrupt      */
  DMA_ITConfig(DMA_STREAM, DMA_IT_TC, ENABLE);           

  /* DMA Stream enable          */
  DMA_Cmd(DMA_STREAM, ENABLE);     

}

//------------------------------------------------------------
void TIM8_UP_TIM13_IRQHandler(void)
{
    if(TIM_GetITStatus(TIM8,TIM_IT_Update)!=RESET)
    {
        TIM8->SR = (uint16_t)~TIM_IT_Update;
    }
    GPIOA->BSRRH = GPIO_Pin_2; //  Тактируем АЦП
    __NOP();
    GPIOA->BSRRL = GPIO_Pin_2;

//Parallel_Data_Buffer[i] = GPIOB->IDR;
    
if(i < 127)    i++;
else i = 0;
}


Falkon_99
А как в вашей программе DMA узнает о приходе очередного байта?
Насколько я знаю для работы DMA необходимы регулярные запросы от переферии.
A.Lex
Я надеялся, что задание
DMA_InitStructure.DMA_Channel = DMA_CHANNEL;


DMA_Init(DMA_STREAM, &DMA_InitStructure);

именно и определит источник (и момент) события, вызывающего считывание данных. Это не так?

Falkon_99
Да. и в настройке таймера тоже нужно включить DMA, в регистре
TIM8 DMA/interrupt enable register (TIM8_DIER), походу бит UDE: Update DMA request enable


я правильно понял, что в обработчике прерывания таймера еще вырабатывается сигнал для тактирования АЦП? если да, то зачем? микросхема АЦП требует тактирования?
A.Lex
Цитата(Falkon_99 @ Dec 9 2013, 14:34) *
Да. и в настройке таймера тоже нужно включить DMA, в регистре
TIM8 DMA/interrupt enable register (TIM8_DIER), походу бит UDE: Update DMA request enable


я правильно понял, что в обработчике прерывания таймера еще вырабатывается сигнал для тактирования АЦП? если да, то зачем? микросхема АЦП требует тактирования?

Спасибо, посмотрю таймер.

Да, АЦП требует внешнего тактирования (AD9280).

Добавил TIM_DMACmd(TIM8, TIM_DMA_Update, ENABLE); Не помогло sad.gif
A.Lex
CODE
#include "stm32f2xx.h"



int main(void)
{

/* System Clocks Configuration ---------------------------------------------*/
/* Enable TIM3 clock */
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);

/* Enable DMA clock */
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);

/* GPIOA and GPIOD clock enable */
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA | RCC_AHB1Periph_GPIOB
| RCC_AHB1Periph_GPIOC, ENABLE);

/* GPIO Configuration ------------------------------------------------------*/
GPIO_Configuration();

/* DMA Channel6 Configuration ----------------------------------------------*/
DMA_InitStructure.DMA_Channel = DMA_Channel_5;

DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&GPIOB->IDR;
DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t) Parallel_Data_Buffer;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;

DMA_InitStructure.DMA_BufferSize = 512;
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_VeryHigh;
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_Init(DMA1_Stream4, &DMA_InitStructure);

/* Enable DMA Channel5 */
DMA_Cmd(DMA1_Stream4, ENABLE);


/* TIM3 Configuration ------------------------------------------------------*/

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

/* Input Capture Mode configuration: Channel1 */
TIM_ICInitStructure.TIM_Channel = TIM_Channel_1;
TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Falling;
TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;
TIM_ICInitStructure.TIM_ICPrescaler = 0;
TIM_ICInitStructure.TIM_ICFilter = 0;
TIM_ICInit(TIM3, &TIM_ICInitStructure);

/* Enable TIM3 DMA */
TIM_DMACmd(TIM3, TIM_DMA_CC1, ENABLE);

/* Enable TIM3 counter */
TIM_Cmd(TIM3, ENABLE);


while (1)
{
/* Trigger TIM3 IC event => DMA request by toggling PA.02 */
GPIO_ResetBits(GPIOA, GPIO_Pin_2);
GPIO_SetBits(GPIOA, GPIO_Pin_2);
}
}

void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;

/* Enable GPIOA, GPIOB, GPIOC, GPIOD, GPIOE and AFIO clocks */
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA | RCC_AHB1Periph_GPIOB
| RCC_AHB1Periph_GPIOC, ENABLE);

/* GPIOA Configuration: PA2 GPIO Output -> TIM3 Channel1 in Input */
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 ;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init(GPIOA, &GPIO_InitStructure);


GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_3
| GPIO_Pin_4 | GPIO_Pin_5
| GPIO_Pin_6 | GPIO_Pin_7
| GPIO_Pin_8 | GPIO_Pin_9
| GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
GPIO_Init(GPIOB, &GPIO_InitStructure);
}
Взял пример из AN2548 (stsw-stm32007), подкорректировал для STM32F2xx - не работает sad.gif


Falkon_99
не понятно, гляньте здесь
A.Lex
Цитата(Falkon_99 @ Dec 10 2013, 21:09) *
не понятно, гляньте здесь

Спасибо, посмотрю.

PS: в моем коде ошибка, д.б. TIM2 CH3, DMA1_Stream1 DMA_Channel_3, исправил, но все равно не работает.

Нашел работаюший пример на форуме, теперь буду разбираться в чем проблема.
kan35
Действительно, если выбираете периферия2память, то нужен сигнал триггера, в вашем случае можно попробовать выбрать типа память2память на stream #2.
Но на мой взгляд лучше сделать так: настроить триггирование по таймеру через соответствующий stream и channel, а в качестве регистра периферии можно выбрать любой регистр, в том числе и GPIOx->IDR.
A.Lex
Цитата(kan35 @ Dec 12 2013, 06:46) *
Действительно, если выбираете периферия2память, то нужен сигнал триггера, в вашем случае можно попробовать выбрать типа память2память на stream #2.
Но на мой взгляд лучше сделать так: настроить триггирование по таймеру через соответствующий stream и channel, а в качестве регистра периферии можно выбрать любой регистр, в том числе и GPIOx->IDR.
Спасибо, но, похоже, баг в библиотеке.


Код, приведенный ниже, РАБОТАЕТ, а то же через библиотечные функции - нет.

CODE


#define DMA_STREAM DMA2_Stream5


#define DMA_CHANNEL DMA_Channel_6
#define BUFFER_SIZE 127
void dma_init(void)
{

/* System Clocks Configuration ---------------------------------------------*/

/* GPIOA and GPIOD clock enable */
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);

/* GPIO Configuration ------------------------------------------------------*/
// GPIO_Configuration();
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_3
| GPIO_Pin_4 | GPIO_Pin_5
| GPIO_Pin_6 | GPIO_Pin_7
| GPIO_Pin_8 | GPIO_Pin_9
| GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
GPIO_Init(GPIOB, &GPIO_InitStructure);



/* Enable DMA clock */
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2, ENABLE);

DMA_STREAM->PAR = (uint32_t)&GPIOB->IDR;

DMA_STREAM->M0AR = (uint32_t)& Parallel_Data_Buffer[0];

DMA_STREAM->NDTR = (uint32_t)BUFFER_SIZE;

DMA_STREAM->CR = 0x00;

DMA_STREAM->CR |= DMA_Channel_6;

DMA_STREAM->CR |= DMA_SxCR_PL_1; //high

DMA_STREAM->FCR &=~(DMA_SxFCR_FTH | DMA_SxFCR_DMDIS);
DMA_STREAM->FCR &=~ DMA_SxFCR_DMDIS;//Direct mode enable

DMA_STREAM->CR |= DMA_DIR_PeripheralToMemory; //memory to memory

DMA_STREAM->CR &=~ DMA_SxCR_PINC;//nie inkrementuje

DMA_STREAM->CR |= DMA_SxCR_MINC;//inkrementuje

DMA_STREAM->CR |= DMA_SxCR_PSIZE_0;//Half Word (16 bitow)

DMA_STREAM->CR |= DMA_SxCR_MSIZE_0;//half Word (16 bity)
//Tryb:
DMA_STREAM->CR |= DMA_SxCR_CIRC;//Circular mode


DMA_STREAM->CR &=~(DMA_SxCR_PBURST_1 | DMA_SxCR_PBURST_0);//single
DMA_STREAM->CR &=~(DMA_SxCR_MBURST_1 | DMA_SxCR_MBURST_0);//single

DMA_STREAM->CR |= DMA_SxCR_EN;
while(!(DMA_STREAM->CR & DMA_SxCR_EN));
} void tim_init(void)
{
//======================== Timer ================================================
/* TIM1 clock enable */
RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1 , ENABLE);

TIM1->PSC = 100;

TIM1->ARR = 6000;

TIM1->CR1 |= TIM_CR1_ARPE //bit 7, Auto reload preload enable
| TIM_CR1_URS //Update request source
| TIM_CR1_CEN; // bit 0, counter enable

TIM1->DIER |= TIM_DIER_UDE
| TIM_DIER_UIE;//Update DMA request enable
}




Может быть, кому-то пригодится. И спасибо drzasiek с польского форума sm.gif

Falkon_99
Иногда регистры настроить проще, чем библиотеки SPL.

Правда немного приходится REFManual покурить, но это даже полезно!

А по коду, скорее всего какойто бит пропустили. Легко проверить в режиме отладки, посмотреть какие данные записаны в регистры (DMA, TIM) после их инициализации, и сравнить 2 варианта...
Это будет полезно, чтоб в следующий раз не наступить на эти же грабли
A.Lex
Цитата(Falkon_99 @ Dec 12 2013, 12:13) *
А по коду, скорее всего какойто бит пропустили. Легко проверить в режиме отладки, посмотреть какие данные записаны в регистры (DMA, TIM) после их инициализации, и сравнить 2 варианта...
Это будет полезно, чтоб в следующий раз не наступить на эти же грабли


Когда будет свободное время - займусь. А в библиотеках баги присутствуют, уже сталкивался. Дебагер - незаменимя вещь sm.gif


Еще раз спасибо всем откликнувшимся за помощь!

kan35
Цитата(A.Lex @ Dec 12 2013, 17:35) *
Когда будет свободное время - займусь. А в библиотеках баги присутствуют, уже сталкивался. Дебагер - незаменимя вещь sm.gif


Еще раз спасибо всем откликнувшимся за помощь!

Не считая того, вы в вашем коде канал таймера выбран не верно, зачем настраиваете таймер в режим input capture?? надо бы в output compare...
Кроме того, параметры вводите не по правилам, например: TIM_TimeBaseStructure.TIM_ClockDivision = 0;
а надо TIM_TimeBaseInitStruct->TIM_ClockDivision = TIM_CKD_DIV1;
Где то повезет, а где то - нет, зато потом будете навсегда считать, что в библиотеках баги. Вся проблема во внимательности и в системном походе (которого нет).
A.Lex
Цитата(kan35 @ Dec 14 2013, 10:32) *
Не считая того, вы в вашем коде канал таймера выбран не верно, зачем настраиваете таймер в режим input capture?? надо бы в output compare...
Кроме того, параметры вводите не по правилам, например: TIM_TimeBaseStructure.TIM_ClockDivision = 0;
а надо TIM_TimeBaseInitStruct->TIM_ClockDivision = TIM_CKD_DIV1;
Где то повезет, а где то - нет, зато потом будете навсегда считать, что в библиотеках баги. Вся проблема во внимательности и в системном походе (которого нет).


В первом варианте был как раз input capture. Остальное - ИМХО дело вкуса, ибо

#define TIM_CKD_DIV1 ((uint16_t)0x0000)





kan35
Цитата(A.Lex @ Dec 14 2013, 15:47) *
В первом варианте был как раз input capture. Остальное - ИМХО дело вкуса, ибо

#define TIM_CKD_DIV1 ((uint16_t)0x0000)

Нет, это не дело вкуса, это везение. Вряд ли вы заглядывали в этот дефайн раньше (а там могло быть что угодно, если вы посмотрите в функцию инициализации то поймете почему). В ваших кусках кодов то одни косяки, то другие, а вы пишите про ошибки в библиотеке. Многие кстати про ошибки говорят, но ни один не может привести пример. В своем коде просто сложнее искать ошибки, проще списать на другого. Кроме того, SPL пишут те же люди, что и CMSIS, так что лишний раз ставить под сомнение их код стоит ли?.. Извините, просто накипело.
Напишите еще раз код, проверьте чтобы прерывание срабатывало, перенаправьте триггер вместо прерывания в соотвествующий канал и поток DMA и оно не сможет не работать.
A.Lex
Цитата(kan35 @ Dec 14 2013, 15:19) *
Нет, это не дело вкуса, это везение. Вряд ли вы заглядывали в этот дефайн раньше (а там могло быть что угодно, если вы посмотрите в функцию инициализации то поймете почему). В ваших кусках кодов то одни косяки, то другие, а вы пишите про ошибки в библиотеке. Многие кстати про ошибки говорят, но ни один не может привести пример. В своем коде просто сложнее искать ошибки, проще списать на другого. Кроме того, SPL пишут те же люди, что и CMSIS, так что лишний раз ставить под сомнение их код стоит ли?.. Извините, просто накипело.
Напишите еще раз код, проверьте чтобы прерывание срабатывало, перенаправьте триггер вместо прерывания в соотвествующий канал и поток DMA и оно не сможет не работать.

Соглашусь с Вами, если Вы соизволите привести рабочие коды по данной теме.
kan35
:-)
Могу взяться за эту задачу за деньги. А если серьезно, почему бы вам самому это не сделать? Еще ни одного корректного варианта в ваших исходниках не было, вот вам и мотив - исправиться в конце концов, и выложить код без косяков. И если не заработает, мы поищем что в нем еще не так.
A.Lex
Может быть я "туп как дерево", но и до меня люди сталкивались с данной проблемой - польский форум.

В моем случае - это задача одноразовая, для "проекта выходного дня". Возникнет снова необходимость - займусь.


Выложенный мной код работает, можете сделать лучше - сделайте и выложите, кто-нибудь будет Вам благодарен.

kan35
Вы меня польским форумом не напугаете))) Поляки они ведь такие же как и мы, тоже любят поумничать.
У меня под рукой нет F2/F4 что бы быстро попробовать. Могу накидать код "от руки" так сказать, а вы проверите, ок?

PS: в вашем случае можно сделать скриншот регистров таймера и дма в обоих вариантах и увидите где вы были не правы.
A.Lex
PS:
Цитата(kan35 @ Dec 26 2013, 12:36) *
Вы меня польским форумом не напугаете))) Поляки они ведь такие же как и мы, тоже любят поумничать.
У меня под рукой нет F2/F4 что бы быстро попробовать. Могу накидать код "от руки" так сказать, а вы проверите, ок?

PS: в вашем случае можно сделать скриншот регистров таймера и дма в обоих вариантах и увидите где вы были не правы.


Выкладывайте, проверю.


PS:Если Вы внимательно прочитали мой пост, для меня проблема не актуальна
.
andrey74
Я все-таки добился чтобы дергались ноги порта С через ДМА, через библиотеки от ST.
A.Lex
Цитата(andrey74 @ Jan 15 2014, 09:21) *
Я все-таки добился чтобы дергались ноги порта С через ДМА, через библиотеки от ST.
Поздравляю!


Значит, я был неправ.
Для просмотра полной версии этой страницы, пожалуйста, пройдите по ссылке.
Invision Power Board © 2001-2025 Invision Power Services, Inc.