Перейти к содержанию

STM32F103: ADC+DMA


Smith2007

Рекомендуемые сообщения

Была аналогичная проблема с STM32F373 и SDADC. Разбираться было некогда. Попробуйте обработать ошибку в прерывании DMA. Если DMA пропускает запрос от периферии он должен выставить флаг ошибки.

Ссылка на комментарий
Поделиться на другие сайты

Реклама: ООО ТД Промэлектроника, ИНН: 6659197470, Тел: 8 (800) 1000-321

2 минуты назад, snn_krs сказал:

Попробуйте обработать ошибку в прерывании DMA. Если DMA пропускает запрос от периферии он должен выставить флаг ошибки

Пробовал. DMA не выставляет флаг ошибки и в прерывание не попадаю.

Ссылка на комментарий
Поделиться на другие сайты

20% скидка на весь каталог электронных компонентов в ТМ Электроникс!

Акция "Лето ближе - цены ниже", успей сделать выгодные покупки!

Плюс весь апрель действует скидка 10% по промокоду APREL24 + 15% кэшбэк и бесплатная доставка!

Перейти на страницу акции

Реклама: ООО ТМ ЭЛЕКТРОНИКС, ИНН: 7806548420, info@tmelectronics.ru, +7(812)4094849

40 минут назад, Smith2007 сказал:

Я думаю, что ДМА счетчик не сдвинул как раз. Пропустил запрос от ADC. 

Так вы же написали, что 1 канал АЦП пишет во вторую ячейку. Выходит счетчик как раз сдвинут а данные с прошлого канала.

47 минут назад, Smith2007 сказал:

но данные поместил уже не во второй элемент массива, а в первый.

ну исходя из превого поста все на оборот) Но тут вам виднее, вы дебажили.

 

Ссылка на комментарий
Поделиться на другие сайты

Выбираем схему BMS для корректной работы литий-железофосфатных (LiFePO4) аккумуляторов

 Обязательным условием долгой и стабильной работы Li-FePO4-аккумуляторов, в том числе и производства EVE Energy, является применение специализированных BMS-микросхем. Литий-железофосфатные АКБ отличаются такими характеристиками, как высокая многократность циклов заряда-разряда, безопасность, возможность быстрой зарядки, устойчивость к буферному режиму работы и приемлемая стоимость. Но для этих АКБ, также как и для других, очень важен контроль процесса заряда и разряда, а специализированных микросхем для этого вида аккумуляторов не так много. Инженеры КОМПЭЛ подготовили список имеющихся микросхем и возможных решений от разных производителей. Подробнее>>

Реклама: АО КОМПЭЛ, ИНН: 7713005406, ОГРН: 1027700032161

Это не пропуск запроса от АЦП. Это приход лишних запросов.

Стоит очень внимательно проанализировать все возможные пути запуска преобразования у АЦП.

И весь код работы с таймерами №2 и №4, если они используются.

Ноги у проблемы отсюда растут. Где-то лишние запуски генерируются.

Ну и судя по этой строке
 

ADC1->CR2 |= ADC_CR2_EXTTRIG;		//ADC_CR2_SWSTART;

АЦП должно начинать новое преобразование сразу же по окончанию предыдущего.

Только вот настройки EXTSEL битов на работу с SWSTART событием я в коде не вижу.

Изменено пользователем Вуйко
Ссылка на комментарий
Поделиться на другие сайты

4 часа назад, Вуйко сказал:

Только вот настройки EXTSEL битов на работу с SWSTART событием я в коде не вижу.

Добавил настройку, но ни чего не поменялось.

void Init_ADC_DMA(void) {

	
	uint32_t pin;
	
	// Включаем тактирование GPIOA, AFIO, ADC1
	RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_AFIOEN | RCC_APB2ENR_ADC1EN;
		
	RCC->CFGR |= RCC_CFGR_ADCPRE_DIV8;	//Предделитель АЦП 72/6 = 12 МГц
	
	// Очистим настройки пинов. Нулевое значение соответствует Input Analog
	GPIOA->CRL &=  ~(	GPIO_CRL_CNF0 | GPIO_CRL_MODE0
									| GPIO_CRL_CNF1 | GPIO_CRL_MODE1
									| GPIO_CRL_CNF5 | GPIO_CRL_MODE5
									| GPIO_CRL_CNF6 | GPIO_CRL_MODE6
									| GPIO_CRL_CNF7 | GPIO_CRL_MODE7 );
	
	// Настройка DMA1
	RCC->AHBENR |= RCC_AHBENR_DMA1EN;					// Включаем тактирование DMA1
	
	// Deinit DMA1 Channel1
	DMA1_Channel1->CCR &= ~DMA_CCR1_EN;		// Отключаем DMA1 CH1
	//DMA1_Channel1->CCR = 0;								// Reset DMA1 Channel1 control register
	DMA1->IFCR |= DMA_ISR_GIF1 | DMA_ISR_TCIF1 | DMA_ISR_HTIF1 | DMA_ISR_TEIF1; // Reset interrupt pending bits for DMA1 Channel1

	// Настраиваем DMA
	DMA1_Channel1->CNDTR = ADC_COUNT;							// Количество передаваемых данных
	DMA1_Channel1->CPAR = (uint32_t) &(ADC1->DR );// Адрес перифирии
	DMA1_Channel1->CMAR = (uint32_t) arrSignal;		// Адрес памяти
	DMA1_Channel1->CCR 	|= DMA_CCR1_MINC					// Инкремент памяти
											| DMA_CCR1_PSIZE_0				// Режим периферии 16 бит
											| DMA_CCR1_MSIZE_0				// Режим памяти 16 бит
											| DMA_CCR1_PL_1						// Channel Priority level hight
											| DMA_CCR1_CIRC						// Circular mode
											| DMA_CCR1_TEIE						// Transfer error interrupt enable
											| DMA_CCR1_TCIE;					// Transfer complete interrupt enable
  DMA1_Channel1->CCR |= DMA_CCR1_EN;						// Включаем DMA1 Channel3

	// Очистим настройки ADC1
	ADC1->CR1 = 0;
	ADC1->CR2 = 0;
	ADC1->SMPR2 = 0;
	ADC1->SQR1 = 0;
	ADC1->SQR2 = 0;
	ADC1->SQR3 = 0;
	
	// Настраиваем ADC1
	ADC1->CR1 |= ADC_CR1_SCAN;		// Scan mode
	ADC1->CR2 |= ADC_CR2_CONT;		// Continuous Conversion
	ADC1->CR2 |= ADC_CR2_DMA;			// DMA mode
	ADC1->CR2 |= ADC_CR2_TSVREFE;	// Temperature Sensor and VREFINT Enable
	ADC1->CR2 |= ADC_CR2_EXTSEL;
	ADC1->CR2 &= ~ADC_CR2_ALIGN;	// Data Alignment
	ADC1->SQR1 |= ((uint32_t)(ADC_COUNT - 1) << 20);			// Сканируем 6 каналов (6-1 = 5)
	
	// Настройка семплирования. Номер канала совпадает с номером пина
	ADC1->SMPR2 |= ADC_SMPR2_SMP0;	// Channel 0 Sample time selection 111: 239.5 cycles
	ADC1->SMPR2 |= ADC_SMPR2_SMP1;	// Channel 1 Sample time selection 111: 239.5 cycles
	ADC1->SMPR2 |= ADC_SMPR2_SMP5;	// Channel 2 Sample time selection 111: 239.5 cycles
	ADC1->SMPR2 |= ADC_SMPR2_SMP6;	// Channel 3 Sample time selection 111: 239.5 cycles
	ADC1->SMPR2 |= ADC_SMPR2_SMP7;	// Channel 4 Sample time selection 111: 239.5 cycles
	ADC1->SMPR1 |= ADC_SMPR1_SMP16;	// Channel 5 Sample time selection 111: 239.5 cycles
	
	// Настройка пинов
	pin = 0x00;
	ADC1->SQR3 |= (pin << (5 * 0));			// Pin0
	pin = 0x01;
	ADC1->SQR3 |= (pin << (5 * 1));			// Pin1
	pin = 0x06;
	ADC1->SQR3 |= (pin << (5 * 2));			// Pin5
	pin = 0x07;
	ADC1->SQR3 |= (pin << (5 * 3));			// Pin6
	pin = 0x05;
	ADC1->SQR3 |= (pin << (5 * 4));			// Pin7
	pin = 0x10;
	ADC1->SQR3 |= (pin << (5 * 5));			// Pin16 - Температурный сенсор
	
	ADC1->CR2 |= ADC_CR2_ADON;		// A/D Converter ON / OFF

	// Обнуляем калибровку
  ADC1->CR2 |= ADC_CR2_RSTCAL; 
  while (ADC1->CR2 & ADC_CR2_RSTCAL);

  //запускаем калибровку и ждем ее завершение
  ADC1->CR2 |= ADC_CR2_CAL; 
  while (ADC1->CR2 & ADC_CR2_CAL);

	ADC1->CR2 |= ADC_CR2_EXTTRIG;		//ADC_CR2_SWSTART;
	//ADC1->CR2 |= ADC_CR2_DMA;				// DMA mode
	ADC1->CR2 |= ADC_CR2_SWSTART;
}

 

2019-11-16_04-49-43.png

Ссылка на комментарий
Поделиться на другие сайты

  • 5 месяцев спустя...

Столкнулся с такой же проблемой в своем проекте на F030F4P6, при передаче из ADC по DMA в массив из 3-х значений они записывались со смещением по кругу на единицу. В моём случае это было вызвано тем, что после инициализации  DMA, в момент установки бита EN в DMA, в нулевой элемент массива записывалось какое-то значение, а счетчик транзакций CNDTR1 уменьшался на 1. Для себя решил проблему так: DMA EN=0, установка корректного CNDTR1, DMA EN=1. После этого всё пишется правильно 

Ссылка на комментарий
Поделиться на другие сайты

Присоединяйтесь к обсуждению

Вы можете написать сейчас и зарегистрироваться позже. Если у вас есть аккаунт, авторизуйтесь, чтобы опубликовать от имени своего аккаунта.
Примечание: Ваш пост будет проверен модератором, прежде чем станет видимым.

Гость
Unfortunately, your content contains terms that we do not allow. Please edit your content to remove the highlighted words below.
Ответить в этой теме...

×   Вставлено с форматированием.   Восстановить форматирование

  Разрешено использовать не более 75 эмодзи.

×   Ваша ссылка была автоматически встроена.   Отображать как обычную ссылку

×   Ваш предыдущий контент был восстановлен.   Очистить редактор

×   Вы не можете вставлять изображения напрямую. Загружайте или вставляйте изображения по ссылке.

Загрузка...
  • Последние посетители   0 пользователей онлайн

    • Ни одного зарегистрированного пользователя не просматривает данную страницу
  • Сообщения

    • ИМХО, только с24, с25, с12 имеет смысл проверить/заменить.
    • Я сам найду, вы мне название напишите пожалуйста.
    • Добрый вечер..)) Нет в моем случае лучше найти готовый для моего Цифровой осциллограф Instrustar ISDS205A 2 канала х 20 МГц..
    • А как же вы квазар спаяли? Или ЛУТом не владеете? Можно даже маркером все прорисовать. А за осцил боятся не стоит, ему на вход будет приходить допустимое напряжение (указано в хар-ках осциллографа).
    • Релюхи не причём. При неисправности релюх симптомы были бы другие. И ИБП даже не клацает когда переходит в  "аварию".   Да. Аккум новый. Более того - 2 новых пробовал. И без АКБ тоже. Не влияет. Да мелкие особо не вздуваются ж. Хотя С21 под подозрением..   Спасибо. Буду иметь ввиду.
    • #include <Wire.h> #include <Oregon_TM.h> #include <BME280I2C.h> //////////////////////////////////////////////////////////////////////////////////////////////////////////// //Скетч для устройства, передающего данные датчика BME280 в формате Oregon Scientific THGN132N //Принципиальная схема прилагается. //Для работы необходима библиотека https://github.com/finitespace/BME280/ //Устройство работает от 3-ех пальчиковых батареек, для экономии электричества заливать скетч нужно через ISP //////////////////////////////////////////////////////////////////////////////////////////////////////////// //Также возможна передача данных в формате - THP (температура, влажность, давление, напряжение батареи) //Пример с приёмником поддерживает расшифоовку THP //////////////////////////////////////////////////////////////////////////////////////////////////////////// # define THGN_SEND 1 // Передавать ли данные в формате THGN132 # define THP_SEND 0 // Передавать ли данные в формате THP # define DEVICE_LOG 1 //Писать ли лог В Serial # define DONE_PIN 15 // вывод сигнала об окончании работы на таймер # define BME_WAIT 10 // Сколько мс ожидать датчик BME # define BATTERY_THR 3.5 // Порог напряжения для выставляения флага разряда батарейки (THGN) ///////////////////////////////////////////////////////////////////////////////////////////////// //Ниблы датчика THP //Во всех полях младшие ниблы идут вперёд!!! // 1-2 - тип (55) // 3 - канал (0-7) // 4-6 - (температура от -100С) * 10. Т.е. +25.1С = 1251 = 4E3h // 7-9 - Влажность *10 Т.е. 25.1% = 251 = 0FBh // 10-12 - (давление от 500ммртст) * 10. Т.е. 765мм = 2650 = A5Ah // 13-15 - данные с АЦП (A0) // 16-17 - CheckSUM // 18-19 - CRC8 (poly 0x07 start 0x00) ///////////////////////////////////////////////////////////////////////////////////////////////// Oregon_TM transmitter(4); BME280I2C bme; bool bme_present = false; float bme_temp(NAN), bme_hum(NAN), bme_pres(NAN); ///////////////////////////////////////////////////////////////////////////////////////////////// void setup() { digitalWrite(DONE_PIN, LOW); pinMode(DONE_PIN, OUTPUT); #ifdef DEVICE_LOG Serial.begin(115200); Serial.println("Waiting for BMEsensor..."); #endif //Обмен данными с BME////////////////////////////////// Wire.begin(); while(!bme.begin()) { if (millis() > BME_WAIT) break; } if (!bme.begin()) { #ifdef DEVICE_LOG Serial.println("No BME sensor found"); #endif bme_present = false; } else { switch(bme.chipModel()) { case BME280::ChipModel_BME280: bme_present = true; bme.read(bme_pres, bme_temp, bme_hum); #ifdef DEVICE_LOG Serial.println("Found BME280 sensor! Success."); Serial.print("Temperature = "); Serial.print(bme_temp, 1); Serial.println("C"); Serial.print("Humidity = "); Serial.print(bme_hum, 1); Serial.println("%"); Serial.print("Pressure = "); Serial.print(bme_pres * 0.75, 1); Serial.println("mmHg"); #endif break; default: #ifdef DEVICE_LOG Serial.println("Found UNKNOWN sensor! Error!"); #endif bme_present = false; } } //Напряжения батареи/////////////////////////////////////////// word battvotage = (word)(((float)(1.1 * 16368) / Vbg()) * 100); #ifdef DEVICE_LOG Serial.print("Battery voltage = "); Serial.println(battvotage,HEX); #endif //Подготовка и отправка данных THGN////////////////////////////////////// transmitter.protocol == 2; if (THGN_SEND) { transmitter.setType(THGN132); transmitter.setChannel(3); transmitter.setBatteryFlag(battvotage < BATTERY_THR); if (bme_present) { if (bme_hum > 98) bme_hum = 98; if (bme_hum < 2) bme_hum = 2; if (bme_temp > 70) bme_temp = 70; if (bme_temp < -50) bme_temp = -50; transmitter.setTemperature(bme_temp); transmitter.setHumidity(bme_hum); transmitter.setComfort(bme_temp, bme_hum); } else { transmitter.setTemperature(-49.9); transmitter.setHumidity(2); transmitter.setComfort(-49.9, 2); } transmitter.SendPacket(); } // Если отправляются оба формата пакетов, межу ними надо выдержать паузу if (THP_SEND && THGN_SEND) delay(100); //Подготовка и отправка данных THP////////////////////////////////////// if (THP_SEND) { transmitter.setType(THP); transmitter.setChannelTHP(1); transmitter.setBatteryTHP( battvotage); if (bme_present) { transmitter.setTemperatureTHP(bme_temp); transmitter.setHumidityTHP(bme_hum); transmitter.setPressureTHP(bme_pres * 0.75); // перевод Pa в mmHg } else { transmitter.setErrorTHP(); } transmitter.SendPacket(); } #ifdef DEVICE_LOG Serial.println(); Serial.print(millis()); Serial.println("ms"); Serial.println(); #endif //Команда на отключение питания digitalWrite(DONE_PIN, HIGH); } ///////////////////////////////////////////////////////////////////////////////////////////////// void loop(){} ///////////////////////////////////////////////////////////////////////////////////////////////// int Vbg() { ADMUX = (1<<REFS0)|(0<<REFS1)|(1<<MUX3)|(1<<MUX2)|(1<<MUX1)|(0<<MUX0); long buffersamp=0; for (int n=0x0; n<=0xff; n++ ) { ADCSRA = 0xc7; while (bit_is_set(ADCSRA,ADSC)); buffersamp += ADC; } buffersamp >>=4; //16368 full scale 14bit ADCSRA &= ~(1 << ADEN); // отключаем АЦП return buffersamp; } Вот код программы. Пробовал по разному и от 5в запитывал и всю систему от 3,3 в. Причем голая ардуинка с таймером работает в нормальном режиме некоторое время, но потом все равно слетает и начинает питать мк постоянно( Причем это происходит всегда через разный промежуток времени.
×
×
  • Создать...