Кто любит RISC в жизни, заходим, не стесняемся.
Ответить

Помогите запустить ADC DMA

Пн дек 06, 2021 23:13:55

Недавно перешел на CubeIDE, запустил ADC, LCD. таймеры, UART, а DMA не получается.
Уже пробую самый простой случай. Один канал загружаю в массив по DMA, а там ничего нет.
В кубеMX все настроил, объявил массив,
uint16_t AD[20];
старт ADC DMA
HAL_ADC_Start_DMA(&hadc1, (uint32_t*)AD, 20);
Потом в цикле каждую секунду вывожу на LCD поочередно элементы массива, но там ничего нет.
В дебуге тоже вижу одни нули.
Что ещё нужно настроить? Проц stm32f103.
Вложения
ADC_DMA103.txt
(9.68 KiB) Скачиваний: 100

Re: Помогите запустить ADC DMA

Пн дек 06, 2021 23:32:57

Могу дать ссылку на работающий код. Естественно, без калокуба.
P.S. Вряд ли кто рискнет лезть в то ублюдство, что "куб" нагенерировал. Оно не предназначено для чтения человеком.

Re: Помогите запустить ADC DMA

Пн дек 06, 2021 23:56:39

Eddy_Em, не откажусь посмотреть, что люди генерируют.:) и сравню, может чего у куба не хватает.

Re: Помогите запустить ADC DMA

Вт дек 07, 2021 00:26:35

Я в основном с F072 работаю (т.к. у них одновременно можно запустить CAN и USB), но на F103 тоже немного железяк делал. вот, например. Настройка ADC with DMA. Каждое значение заносится в массив из 9 величин, затем, когда нужно определенный канал опросить, делаю медианную фильтрацию.

Re: Помогите запустить ADC DMA

Вт дек 07, 2021 04:13:13

linkov1959, поменяйте местами MX_ADC1_Init() и MX_DMA_Init(). В последней версии CubeIDE это вроде исправили.
Код:
  MX_GPIO_Init();
  MX_DMA_Init(); // инициализируем dma перед adc
  MX_ADC1_Init();
  MX_USART1_UART_Init();

Достаточно часто встречается
1) неправильный порядок инициализации, нужно поменять местами (можно в редакторе периферии на вкладке Generated Function Calls стрелочками поменять этот порядок, тогда при генерации кода не придётся каждый раз вручную редактировать)
2) не включены прерывания, на которых основана логика работы HAL, нужно включить через редактор параметров периферии
3) DMA не имеет доступа к адресному пространству выделенной памяти, нужно через ld скрипт и документацию принудительно перенести массивы по другим адресам

Re: Помогите запустить ADC DMA

Вт дек 07, 2021 11:31:06

rezistor2000, вечером попробую. У меня элементы массива читаются в бесконечном цикле и когда индекс превышает размер массива после нулей читаются какие-то большие числа. Откуда они берутся?
Последний раз редактировалось linkov1959 Вт дек 07, 2021 13:53:04, всего редактировалось 1 раз.

Re: Помогите запустить ADC DMA

Вт дек 07, 2021 11:33:46

А чего вдруг должны читаться нули когда индекс превышает размер массива?

Re: Помогите запустить ADC DMA

Вт дек 07, 2021 13:55:54

Reflector, после 20 нулей должна быть ошибка или ничего, пустой LCD.

Re: Помогите запустить ADC DMA

Вт дек 07, 2021 14:08:09

Допустим есть такой код:
Код:
uint16_t arr[] = { 0x1111, 0x2222 };
uint16_t arr2[] = { 0xABCD };

rtt.println(arr, arr2);

for (int i = 0; i < 3; i++)
   rtt.printxln(arr[i]);

Если его запустить, то получим:
Код:
0x20001520, 0x20001524
1111
2222
ABCD

Как видно компилятор разместил второй массив сразу за первым и при выходе за границы первого прочитал содержимое второго, т.е. прочитано может быть что угодно.

Re: Помогите запустить ADC DMA

Вт дек 07, 2021 16:01:13

Reflector, Понял, за массивом в памяти была другая переменная и ни к ADC, ни к DMA отношения не имеет.
DMA берет данные из регистра ADC и пишет в пределах указанного размера памяти при вызове
HAL_ADC_Start_DMA(&hadc1, (uint32_t*)AD, 20); Размер массива DMA не волнует, а только лишь адрес начала массива. В свою очередь при чтении массива компилятор не волнует его размер и будет читать, пока индекс растет.

Re: Помогите запустить ADC DMA

Вт дек 07, 2021 16:21:36

Не то чтобы размер массива компилятор не волновал, если он видит выход за пределы массива, то предупреждение выдаст, а чтобы была именно ошибка нужно использовать какие-то классы с соответствующими проверками. Например, если бы я написал так с включенными assert(), то получил бы рантаймовую ошибку потому что доступ к оператору [] проверяется:
Код:
Deque<uint16_t, 2> arr = { 0x1111, 0x2222 };

for (int i = 0; i < 3; i++)
   rtt.printxln(arr[i]);

// 1111
// 2222
// Error: re/deque.h, 60

Re: Помогите запустить ADC DMA

Вт дек 07, 2021 22:35:28

linkov1959, поменяйте местами MX_ADC1_Init() и MX_DMA_Init(). В последней версии CubeIDE это вроде исправили.
Код:
  MX_GPIO_Init();
  MX_DMA_Init(); // инициализируем dma перед adc
  MX_ADC1_Init();
  MX_USART1_UART_Init();

Достаточно часто встречается
1) неправильный порядок инициализации, нужно поменять местами (можно в редакторе периферии на вкладке Generated Function Calls стрелочками поменять этот порядок, тогда при генерации кода не придётся каждый раз вручную редактировать)
2) не включены прерывания, на которых основана логика работы HAL, нужно включить через редактор параметров периферии
3) DMA не имеет доступа к адресному пространству выделенной памяти, нужно через ld скрипт и документацию принудительно перенести массивы по другим адресам

Поменял местами Init и все изменилось! LCD стал мельтишить и глючить, выводить всякую фигню, но я понял, что на правильном пути.
После старта DMA прописал стоп. DMA записал массив и остановился, а в цикле неспешно вывелись все элементы по очереди!
Спасибо за помощь!

Добавлено after 2 hours 16 minutes 27 seconds:
Вот и выходит, что не я дурак, а в кубе ошибка, на который все молились! Теперь и с Mikroe пришло письмо с извинениями, типа все исправим! А сколько времени и сил потрачено зря!

Re: Помогите запустить ADC DMA

Ср дек 08, 2021 20:53:37

Лежит у меня TFT 240х320, хотел на микробейсике запустить, но не получилось. В кубе должно получиться!

Добавлено after 4 hours 21 minute 48 seconds:
Получилось!

Re: Помогите запустить ADC DMA

Чт авг 03, 2023 23:10:31

linkov1959, поменяйте местами MX_ADC1_Init() и MX_DMA_Init(). В последней версии CubeIDE это вроде исправили.
Код:
  MX_GPIO_Init();
  MX_DMA_Init(); // инициализируем dma перед adc
  MX_ADC1_Init();
  MX_USART1_UART_Init();

Достаточно часто встречается
1) неправильный порядок инициализации, нужно поменять местами (можно в редакторе периферии на вкладке Generated Function Calls стрелочками поменять этот порядок, тогда при генерации кода не придётся каждый раз вручную редактировать)
2) не включены прерывания, на которых основана логика работы HAL, нужно включить через редактор параметров периферии
3) DMA не имеет доступа к адресному пространству выделенной памяти, нужно через ld скрипт и документацию принудительно перенести массивы по другим адресам

Спасибо несколько часов тупил проверял а вот оно что
Ответить