Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: Настройка ADC в STM32
Форум разработчиков электроники ELECTRONIX.ru > Микроконтроллеры (MCs) > ARM
Jenya7
Есть возможность использовать откоректированное значение опорного напряжения.
Цитата
Calculating the actual VDDA voltage using the internal reference voltage
The VDDA power supply voltage applied to the microcontroller may be subject to variation or
not precisely known. The embedded internal voltage reference (VREFINT) and its calibration
data acquired by the ADC during the manufacturing process at VDDA = 3.3 V can be used to
evaluate the actual VDDA voltage level.
The following formula gives the actual VDDA voltage supplying the device:
VDDA = 3.3 V ₓ VREFINT_CAL / VREFINT_DATA
Where:
• VREFINT_CAL is the VREFINT calibration value
• VREFINT_DATA is the actual VREFINT output value converted by ADC

VREFINT_CAL как я понимаю мы берем из ADC1->CALFACT
а VREFINT_DATA - откуда берем?
Эдди
Данные из канала АЦП, вестимо (ADC1_IN17).
P.S. Я так понял, VREFINT_CAL есть только у L-серий. А у обычных F нужно это значение измерить.
Т.е. если, скажем, мерить с точностью до сотых вольты (только зачем?), то VDDA = X / CH17 → X = VDDA * CH17 (считать лучше в uint32_t).
Скажем, дает нам вольтметр 3.25В, а внутреннее опорное — 1540ADU, тогда X = 325*1540 = 500500, и в любой момент просто делим 500500 на данные с CH17, получая VDD.
Jenya7
Цитата(Эдди @ Jul 25 2017, 11:11) *
Данные из канала АЦП, вестимо (ADC1_IN17).
P.S. Я так понял, VREFINT_CAL есть только у L-серий. А у обычных F нужно это значение измерить.
Т.е. если, скажем, мерить с точностью до сотых вольты (только зачем?), то VDDA = X / CH17 → X = VDDA * CH17 (считать лучше в uint32_t).
Скажем, дает нам вольтметр 3.25В, а внутреннее опорное — 1540ADU, тогда X = 325*1540 = 500500, и в любой момент просто делим 500500 на данные с CH17, получая VDD.

только для L-серий? не знаю, у меня в STM32F303VCT6 есть ADC1->CALFACT.
смотрим формулу
Цитата
VCHANNELx = 3.3 V * VREFINT_CAL * ADCx_DATA / VREFINT_DATA * FULL_SCALE

Where:
• VREFINT_CAL is the VREFINT calibration value
• ADCx_DATA is the value measured by the ADC on channel x (right-aligned)
• VREFINT_DATA is the actual VREFINT output value converted by the ADC
• FULL_SCALE is the maximum digital value of the ADC output. For example with 12-bit
resolution, it will be 212 - 1 = 4095 or with 8-bit resolution, 28 - 1 = 255.

"Данные из канала АЦП, вестимо" - это ADCx_DATA
а VREFINT_DATA откуда?
scifi
Цитата(Jenya7 @ Jul 25 2017, 09:50) *
VCHANNELx = 3.3 V * VREFINT_CAL * ADCx_DATA / VREFINT_DATA * FULL_SCALE

Очевидно же.
VREFINT_CAL - калибровка из ПЗУ.
VREFINT_DATA - измеренное на канале АЦП Vrefint
ADCx_DATA - измеренное на ином канале АЦП, результат именно этого измерения корректируется, приводится к вольтам и присваивается переменной VCHANNELx.
Jenya7
Цитата(scifi @ Jul 25 2017, 12:26) *
Очевидно же.
VREFINT_CAL - калибровка из ПЗУ.
VREFINT_DATA - измеренное на канале АЦП Vrefint
ADCx_DATA - измеренное на ином канале АЦП, результат именно этого измерения корректируется, приводится к вольтам и присваивается переменной VCHANNELx.

ааа понял. спасибо. то есть я должен измерить канал #define ADC_Channel_Vrefint ((uint8_t)ADC_Channel_18)
я могу передать его в функцию в качестве аргумента?
Код
uint32_t ADC_Read(uint32_t channel)
{
    uint32_t adc_val;
    
    ADC_RegularChannelConfig(ADCx, channel, 1, ADC_SampleTime_7Cycles5);
    
    /* Start ADC1 Software Conversion */
     ADCx->CR |= ADC_CR_ADSTART;
    
    /* Test EOC flag */
    while(ADC_GetFlagStatus(ADCx, ADC_FLAG_EOC) == RESET);
    
    /* Get ADC1 converted data */
    adc_val = (uint16_t)ADCx->DR;
    
    return adc_val;
}
scifi
Цитата(Jenya7 @ Jul 25 2017, 10:49) *
я могу передать его в функцию в качестве аргумента?

Это как хотите. Есть множество способов всё это организовать.
Кстати, что за нездоровое увлечение типом uint32_t? Есть отличный тип int - всего 3 буквы, и работает ничуть не хуже.
Код
int ADC_Read(int channel)
{
    ...    
    return ADCx->DR;
}
Jenya7
Цитата(scifi @ Jul 25 2017, 12:53) *
Это как хотите. Есть множество способов всё это организовать.
Кстати, что за нездоровое увлечение типом uint32_t? Есть отличный тип int - всего 3 буквы, и работает ничуть не хуже.
Код
int ADC_Read(int channel)
{
    ...    
    return ADCx->DR;
}

рекомендуют так - uint32_t, для портабилити. int он signed вообще то.
scifi
Цитата(Jenya7 @ Jul 25 2017, 10:58) *
рекомендуют так - uint32_t, для портабилити

Это глупости и танцы с бубном. Какая такая портабилити кода, написанного для STM32? На другой STM32 он перенесётся и так, и эдак.

Цитата(Jenya7 @ Jul 25 2017, 10:58) *
int он signed вообще то.

Ну и что? Почему-то это никому не мешает делать, к примеру, вот так:
for (int i = 0; i < 10000; i++)

С другой стороны, я не настаиваю. Просто int короче. Ну и красивше, но это субъективное.
Jenya7
Цитата(scifi @ Jul 25 2017, 13:02) *
Это глупости и танцы с бубном. Какая такая портабилити кода, написанного для STM32? На другой STM32 он перенесётся и так, и эдак.
С другой стороны, я не настаиваю. Просто int короче. Ну и красивше, но это субъективное.

но int он знаковый. а unsigned int это уже не три это... много букаф. sm.gif
Цитата(scifi @ Jul 25 2017, 13:02) *
Ну и что? Почему-то это никому не мешает делать, к примеру, вот так:
for (int i = 0; i < 10000; i++)

ну вообще да. чего я боюсь. что мне не хватит 2^32 / 2.

кстати. VREFINT_DATA он ведь общий для ADC1 и ADC2. то есть мне достаточно померять на ADC1?

а в ADCx_CCR есть бит VREFEN: VREFINT enable - его надо выставить?
Эдди
Цитата(Jenya7 @ Jul 25 2017, 09:50) *
только для L-серий? не знаю, у меня в STM32F303VCT6 есть ADC1->CALFACT.

А, у меня только F030, F042 и F103. Там такого нет, но несложно померить один раз — я серии железяк не делаю.
Вообще, при питании от кренки или чего понадежней нет такого размаха напряжения питания, чтобы прямо калибровать надо было. Во всяком случае, я резисторами калибровал — честные 12 бит было, никаких диких шумов…

Цитата(Jenya7 @ Jul 25 2017, 10:58) *
int он signed вообще то.

Лучше вообще не использовать типы вроде char/int/long и т.п. Мало ли — перенесете высокоуровневый код с STM32 на STM8, и получите жесть, т.к. типы данных будут разной длины!
scifi
Прошу прощения, что нечаянно набросил про int. Боюсь, снова начнётся хождение между трёх сосен, как это здесь часто бывает crying.gif

Цитата(Jenya7 @ Jul 25 2017, 11:07) *
кстати. VREFINT_DATA он ведь общий для ADC1 и ADC2. то есть мне достаточно померять на ADC1?

Так точно.
Jenya7
Цитата(scifi @ Jul 25 2017, 13:30) *
Прошу прощения, что нечаянно набросил про int. Боюсь, снова начнётся хождение между трёх сосен, как это здесь часто бывает crying.gif
Так точно.

и еще нужно включить канал
ADC1_2->CCR |= ADC12_CCR_VREFEN;

фигня получается однако
Код
vref_data = ADC_Read(ADC1, ADC_Channel_Vrefint);  //получаем 2838
vref_cal = (uint32_t)ADC1->CALFACT;      //получаем 58
    
vref = (3300 * vref_cal) / vref_data;           //получаем 67


vref = 67 ???
scifi
Здрассте, приехали.
ADC1->CALFACT - это совсем не то.
Заводская калибровка - это *(uint16_t*)0x1FF80078, если эта штука вообще присутствует в вашем МК.
Эдди
Цитата(Jenya7 @ Jul 25 2017, 11:45) *
ADC1->CALFACT; //получаем 58

Мне с выражением за вас даташит читать? Этот регистр не имеет никакого отношения к "заводскому" измерению напряжения на vref, это — внутреннее значение калибровок, которое вы можете изменять!
Кстати, насчет STM32F0 я был не прав: в них этот регистр есть (0x1FFF F7BA - 0x1FFF F7BB). А вот в F103 — нет.
Jenya7
Цитата(scifi @ Jul 25 2017, 14:55) *
Здрассте, приехали.
ADC1->CALFACT - это совсем не то.
Заводская калибровка - это *(uint16_t*)0x1FF80078, если эта штука вообще присутствует в вашем МК.

ой. что то я в даташит не нашел даже упоминания о регистре с заводской калибровкой. поиск по адресу 0x1FF80078 тоже не дал результатов. у STM32F303VCT6 такой навороченный ADC не может быть чтоб у него не было этой опции.
scifi
Цитата(Jenya7 @ Jul 25 2017, 13:16) *
ой. что то я в даташит не нашел даже упоминания о регистре с заводской калибровкой. поиск по адресу 0x1FF80078 тоже не дал результатов. у STM32F303VCT6 такой навороченный ADC не может быть чтоб у него не было этой опции.

0x1FFFF7BA
Учу читать даташиты. Дорого biggrin.gif
Jenya7
Цитата(scifi @ Jul 25 2017, 15:40) *
0x1FFFF7BA
Учу читать даташиты. Дорого biggrin.gif

нету. crying.gif весь даташит перерыл.
scifi
Цитата(Jenya7 @ Jul 25 2017, 14:00) *
нету. crying.gif весь даташит перерыл.

Фигасе laughing.gif
Эдди
Цитата(Jenya7 @ Jul 25 2017, 14:00) *
весь даташит перерыл.

Есть понятие RM - там описывается семейство МК, и нет сведений о конкретных моделях: распиновке, особенностях питания и "заводским" регистрам.
А есть понятие даташита — там описывается конкретная линейка с распиновкой и всякой всячиной.
Jenya7
Цитата(scifi @ Jul 25 2017, 16:07) *
Фигасе laughing.gif

спасибки. теперь вижу. у меня поиск почему то не находит.

а что значит 0x1FFF F7BA - 0x1FFF F7BB? данные лежат в двух регистрах? получается громадное знечение.
Obam
Цитата(Jenya7 @ Jul 25 2017, 15:46) *
спасибки. теперь вижу. у меня поиск почему то не находит.

а что значит 0x1FFF F7BA - 0x1FFF F7BB? данные лежат в двух регистрах? получается громадное знечение.

Аграмадное!!!! 16 бит (;
scifi
Цитата(Jenya7 @ Jul 25 2017, 14:46) *
спасибки. теперь вижу. у меня поиск почему то не находит.

Искать надо уметь laughing.gif Говорят, где-то даже проводят соревнования по поиску в интернете laughing.gif

Цитата(Jenya7 @ Jul 25 2017, 14:46) *
а что значит 0x1FFF F7BA - 0x1FFF F7BB? данные лежат в двух регистрах? получается громадное знечение.

Вы будете смеяться, но 16-разрядное число занимает в памяти аккурат 2 байта, то есть 0x1FFFF7BA - 0x1FFFF7BB

А почему у вас рядом с ником написано "профессионал"? smile3009.gif
Jenya7
Цитата(Obam @ Jul 25 2017, 17:15) *
Аграмадное!!!! 16 бит (;

как это у вас получается? регистр 32 бита.

ура. работает.
vref_data = ADC_Read(ADC1, ADC_Channel_Vrefint);
vref_cal = *(uint16_t*)0x1FFFF7BA;
vref = (3300 * vref_cal) / vref_data;

получаю 2.97 вольта. вольтметр меряет 2.94 но это китайский, флюка под рукой нет.
Obam
Цитата(Jenya7 @ Jul 25 2017, 16:21) *
как это у вас получается? регистр 32 бита.

ура. работает.
vref_data = ADC_Read(ADC1, ADC_Channel_Vrefint);
vref_cal = *(uint16_t*)0x1FFFF7BA;
vref = (3300 * vref_cal) / vref_data;

получаю 2.97 вольта. вольтметр меряет 2.94 но это китайский, флюка под рукой нет.


И вас не смутило, что этот якобы "регистр" расположен по некратному 4 адресу?

Цитата
А почему у вас рядом с ником написано "профессионал"?

На день ангела подарили… Ж8-/
Jenya7
Цитата(scifi @ Jul 25 2017, 17:16) *
А почему у вас рядом с ником написано "профессионал"? smile3009.gif

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