Микроконтроллеры компании STMicrielectronics широко применяются в различных сферах профессиональной разработки и технического творчества. Невысокая стоимость, доступность демонстрационных плат, развитый набор периферии, достаточно высокая производительность, низкое энергопотребление и развитая экосистема разработки делает создание различных устройств на основе данных контроллеров простым и увлекательным делом.
С выходом в серию семейства STM32G я наконец не выдержал и решил попробовать реализовать одну давнюю идею: собрать простое радиоприемное устройство диапазона LW/MW с минимальным количеством внешних электронных компонентов. Конструкция и программное обеспечение должны быть достаточно простыми, чтобы можно было на пальцах объяснить новичку основы цифровой обработки сигналов, не углубляясь в теорию.
Для реализации проекта была выбрана доступная демонстрационная плата NUCLEO-G431KB, содержащая контроллер STM32G431KBT6 (flash 128KB, ram 32KB, тактовая частота процессора до 170 MHz), минимально необходимую обвязку и интегрированный отладчик/программатор STLINK-V3E. Особенно ценным является наличие в контроллере 12 разрядного аналого-цифрового преобразователя ADC с частотой дискретизации до 4 MHz, что теоретически позволяет обрабатывать сигналы с частотами до 2 MHz.
Излагать материал я буду от простого к сложному в виде описания последовательных экспериментов, постепенно усложняя программное обеспечение. Компиляция программ выполнялась в Windows 10 с использованием бесплатной интегрированной среды разработки STM32CubeIDE от компании STMicrielectronics (https://www.st.com/en/development-tools/stm32cubeide.html). Исходные коды экспериментальных проектов находятся по адресу https://github.com/OlegDyakov/simple-radio.
Для разработки программ использовались следующие официальные документы от компании STMicrielectronics:
«RM0440, Reference manual STM32G4 Series advanced Arm ® -based 32-bit MCUs» - подробное описание функциональных блоков контроллеров серии STM32G4.
«DS12589 Rev 2 - Datasheet STM32G431x6, STM32G431x8, STM32G431xB» - описание контроллеров линейки STM32G431x6, STM32G431x8, STM32G431xB.
«PM0214 Rev 10 - STM32 Cortex ® -M4 MCUs and MPUs programming manual» - руководство программиста.
Для выполнения экспериментов был создан исследовательский стенд (рис. 1), который в последствии позволил изготовить реально работающий радиоприемник начального уровня. В принципе, эксперименты можно не повторять, они предназначены для объяснения принципов работы программного обеспечения и получения характеристик с реальной платы.
1. Структура радиоприемника
Радиоприемник построен по классической схеме приемника прямого усиления и предназначен для приема АМ радиостанций, вещающих в диапазонах длинных и средних волн (LW и MW). Сигнал с ферритовой магнитной антенны поступает на усилитель высокой частоты (УВЧ). Далее сигнал оцифровывается в аналого-цифровом преобразователе (ADC) и подается на перестраиваемый цифровой узкополосной полосовой фильтр, обеспечивающий настройку на нужную несущую частоту радиостанции. Выделенный сигнал передается на АМ демодулятор, реализованный по принципу диодного детектора. Для уменьшения эффекта «замирания сигнала» после демодулятора стоит система автоматической регулировки усиления (АРУ). Полученный цифровой звуковой сигнал поступает на цифро-аналоговый преобразователь (DAC), далее - на усилитель низкой частоты (УНЧ) и динамик (наушники).
2. Базовый проект
Для разработки программного обеспечения для контроллеров STM32 существует большое количество очень хороших готовых профессиональных программных пакетов. К сожалению, новичкам достаточно сложно сразу осмыслить сложный мир профессиональных средств разработки. К тому же стандартные библиотеки (CMSIS, HAL) закрывают от программиста механизмы работы с функциональными модулями контроллера.
У разработчика имеется довольно подробное описание функциональных модулей контроллера, например: “RM0440 Reference manual STM32G4 Series advanced Arm ® -based 32-bit MCUs“, содержащее более 2000 страниц. Как правило, программисты применяют стандартные библиотеки от производителей контроллера, которые, мягко говоря, довольно сильно изолируют разработчика от управления аппаратными блоками. Если что-то работает не так, или нам нужно “выжать” из железки максимум, приходиться «лезть» в код библиотеки, который написан на профессиональном C, предназначен для обеспечения переносимости на разные контроллеры и системы разработки и довольно сложен для понимания людям, которые делают первые шаги в embedded программировании. Хотелось быть «поближе к транзисторам», поэтому был разработан простой базовый проект на C, который является основой для реализации всех экспериментов, описанных в данной статье.
Экспериментальные проекты носят исследовательский характер и не претендуют на демонстрацию профессиональных подходов при разработке встроенного программного обеспечения.
Рассмотрим простейшую программу, написанную на языке С, которая выполняет переключение светодиода LD2 и логических уровней на выводе CN4-7 (PB7) демонстрационной платы NUCLEO-G431KB. Вывод PB7 мы будем использовать для измерения с помощью осциллографа времени выполнения различных функций в экспериментальных проектах. Проект простейшей программы находится в папке 01_Minimal.
Проект состоит из следующих файлов:
system_init.c |
инициализация регистров контроллера и оперативной памяти при старте, |
vectors.c |
таблица прерываний, |
gpio.c |
настройка ножек контроллера, |
main.c |
основной цикл программы, |
stm32g431.h |
описание регистров контроллера, используемых в проекте, |
stm32f4.ld |
скрипт программы компоновщика. |
Начнем анализ проекта с файла скрипта компоновщика «stm32f4.ld». Этот файл предназначен для предоставления информации компоновщику, каким образом организована основная память контроллера (FLASH, RAM), как необходимо размещать секции (области данных, с которыми работает компилятор С) в физической памяти контроллера, а также какую функцию программы необходимо вызывать сразу после рестарта.
Компилятор языка С размещает различные типы данных в разных стандартных секциях памяти:
.text - исполняемый код программы,
.rodata - константы,
.data - глобальные переменные, имеющие начальные константные значения (например int a=5;),
.bss - неинициализированные глобальные переменные (например int b; ) .
Схематично распределение памяти контроллера, а также размещение секций кода и данных программы представлены на рисунке 3. Здесь указаны только те области адресного пространства контроллера, которые имеют отношение к нашему проекту.
Кроме описания памяти и определения схемы расположения секций, в файле скрипта задаются значения констант, отображающих начальные и конечные адреса секций с данными, соответственно:
_sdata_ram, _edata_ram для секции .data, располагающейся в RAM;
_sdata_flash, _edata_flash для секции .data, располагающейся во FLASH;
_sbss, _ebss для секции .bss;
_svector, _evector для секции .vectors;
_estack начальный адрес стека.
Данные константы используются при инициализации памяти в коде программы, расположенном в файле system_init.c.
Контроллер STM32G431KBT6 является достаточно развитой системой на кристалле (SoC), содержащей около 50 функциональных модулей, которые управляются через регистры. Для программного обеспечения регистры доступны как 32 битные ячейки памяти, расположенные в разделе адресного пространства «Периферия» (рисунок 3). Описание всех регистров контроллера (названия регистров, адреса регистров, название и расположение битовых полей внутри регистра) занимает более 13 тысяч строк (стандартный файл stm32g431xx.h из пакета CMSIS).
К счастью, все устроено таким образом, что после аппаратного перезапуска в контроллере активизируется минимально необходимое количество блоков. Если неиспользуемые модули специально не включать, то они не будут оказывать никакого влияния на процесс работы программы. В проектах будем использовать усеченный вариант файла описания - stm32g431.h, который содержит определения управляющих регистров тех функциональных модулей, которые необходимы для реализации программ - экспериментов.
Функциональные модули в адресном пространстве контроллера представлены в виде последовательности 32 битных регистров (ячеек памяти). Согласно спецификации, для каждого модуля определен базовый адрес (адрес первого регистра в блоке) и смещение. Например, у модуля GPIOB базовый адрес GPIOB_BASE = 0x48000400. Смещение управляющего регистра GPIOB_ODR (output data register) равен 0x14, т.е. абсолютный адрес равен значению GPIOB_ODR = GPIOB_BASE + 0x14 = 0x48000414.
/*********************** GPIO (General-purpose I/Os) *********************************/
#define GPIOB_BASE (0x48000400)
#define GPIOB_MODER (*(volatile uint32_t *) (GPIOB_BASE + 0x00))
#define GPIOB_ODR (*(volatile uint32_t *) (GPIOB_BASE + 0x14))
В файле stm32g431.h определены макросы, которые позволяют сделать более наглядной работу с регистрами контроллера.
//Макросы, упрощающие работу с регистрами, где
//REG - адрес регистра
//BIT - 32 битное слово - биты для которых будет выполнена данная функция.
//Например: INVERT(GPIOB_ODR,0x6) проинвертирует биты 1 и 2 в регистре GPIOB_ODR
#define SET_BIT(REG, BIT) ((REG) |= (BIT))
#define CLEAR_BIT(REG, BIT) ((REG) &= ~(BIT))
#define INVERT_BIT(REG, BIT) ((REG) ^= (BIT))
#define READ_BIT(REG, BIT) ((REG) & (BIT))
#define CLEAR_REG(REG) ((REG) = (0x0))
#define READ_REG(REG) ((REG))
#define WRITE_REG(REG, VAL) ((REG) = (VAL))
#define MODIFY_REG(REG, CLEARMASK, SETMASK) WRITE_REG((REG), (((READ_REG(REG)) & (~(CLEARMASK))) | (SETMASK)))
Теперь перейдем к обсуждению последовательности действий контроллера при отработке аппаратного рестарта.
Контроллер имеет внутренний RC тактовый генератор на 16MHz (HSI RC 16). Сразу после рестарта выходной сигнал с данного генератора используется в качестве тактового сигнала для вычислительного ядра и периферийных модулей.
При отработке аппаратного рестарта, после выполнения ряда настроечных процедур, ядро контроллера загружает содержимое ячейки 0x00000000 в регистр SP (указатель стека), а содержимое ячейки 0x00000004 в регистр PC (счетчик команд). Другими словами, ячейка 0x00000000 должна содержать адрес начала стека, а ячейка 0x00000004 адрес, с которого начинается выполнение кода программы при рестарте (Reset handler’s address).
Область памяти в начале адресного пространства является виртуальной (0x00000000 - 0x00080000), т.е. в зависимости от настройки метода загрузки микроконтроллера, в данную область могут отображаться различные модули памяти: внутренний FLASH, RAM, системный ROM, внешний FLASH. Метод загрузки определяется распайкой вывода PB8 контроллера. Для платы NUCLEO-G431KB по умолчанию загрузка выполняется из внутреннего FLASH. Это означает, что ячейки FLASH памяти, расположенные по адресам 0x08000000 – 0x08020000, также доступны по адресам 0x00000000 – 0x00020000. Таким образом, для данного режима загрузки можно сказать, что микроконтроллер стартует с первого адреса FLASH памяти т.е. с адреса 0x08000000.
Как мы видели из файла «stm32f4.ld», первые 256 слов во внутреннем флэше занимает таблица векторов прерывания (секция. vectors). Элементы таблицы векторов прерывания содержат адреса специальных функций (обработчиков), которые автоматически вызываются при возникновении прерываний от различных модулей микроконтроллера или в случае аппаратных ошибок. За исключением начального элемента таблицы, который содержит адрес вершины стека.
Определение таблицы векторов прерывания находится в файле «vectors.c».
__attribute__ ((section(".vectors"),used)) //Таблицу размещаем в секции .vectors
uint32_t * vectors[] = {
(uint32_t *) &_estack, //Вершина стека
(uint32_t *) start_up, //Обработчик reset (точка входа)
(uint32_t *) nmi_handler, //Обработчик немаскируемого прерывания
(uint32_t *) hardfault_handler //Обработчик прерывания аппаратной ошибки
};
Как видно из кода таблицы, в базовом проекте определены только первые три обработчика, необходимые для работы проекта. Остальные элементы таблицы мы будем заполнять в следующих проектах по мере необходимости.
Функции nmi_handler и hardfault_handler обрабатывают аварийные ситуации. Они переводят вычислительное ядро в глухой цикл.
Функция start_up находится в файле «system_init.c» и выполняет действия по инициализации RAM, необходимые для работы программы, скомпилированной с языка С. Сначала выполняется копирование данных секции .data из FLASH в RAM, затем выполняется запись нулей в секцию .bss.
Для настройки линий PB7 и PB8 на режим логического выхода используется функция GpioInit, расположенная в файле gpio.c.
void GpioInit(void)
{
// Подать тактовый сигнал на модуль GPIOA
SET_BIT(RCC_AHB2ENR, RCC_AHB2ENR_GPIOBEN);
//Включить PIN8 на вывод (светодиод LD2)
MODIFY_REG(GPIOB_MODER, GPIO_MODER_MODE8, 1<<GPIO_MODER_MODE8_Pos);
//Включить PIN7 на вывод
MODIFY_REG(GPIOB_MODER, GPIO_MODER_MODE7, 1<<GPIO_MODER_MODE7_Pos);
}
Основные минимальные настройки (по умолчанию) загружаются в управляющие регистры на аппаратном уровне при рестарте контроллера. Для реализации функции переключения светодиода необходимо настроить модуль GPIO (General-purpose I/O). Для этого первым делом нужно подать тактовый сигнал на данный модуль. Эта операция выполняется через модуль RCC (Reset and clock control).
Для того чтобы точно определить период цикла переключения светодиода с помощью осциллографа, дополнительно к PB8 также настроим на вывод линию PB7 и будем изменять значения на данных выводах одновременно.
Основной цикл программы находится в файле «main.c».
Измеренное осциллографом время между двумя переключениями на линии PB7 составляет 686 мс. Это время в основном расходуется на выполнение цикла:
while( i ) i--;
Ассемблерный код, сгенерированный компилятором для данной строки, имеет следующий вид:
a: ldr r3, [r7, #4] // Загрузить переменную i в регистр r3. (2 такта)
subs r3, #1 // Вычесть 1 из регистра r3. (1 такт)
str r3, [r7, #4] // Сохранить регистр r3 в переменную i. (2 такта)
ldr r3, [r7, #4] // Загрузить переменную i в регистр r3. (2 такта)
cmp r3, #0 // Сравнить регистр r3 с 0 (1 такт)
bne.n a // Если не равно перейти на метку a: (1 такт +
// 2 такта перестройка конвейера команд)
Данный код можно увидеть в файле «01_Minimal.list» в папке 01_Minimal\Debug\ проекта.
В комментариях для каждой ассемблерной инструкции указаны выполняемые действия и время выполнения в тактах. Общее время одного цикла равно 11 тактам. При частоте внутреннего RC генератора 16 MHz период одного такта составляет 62,5 нс. Время выполнения одной итерации цикла составляет 62,5 х 11 = 687,5 нс. На миллион итераций будет затрачено приблизительно 687 мс, что практически совпадает с результатом, полученным с помощью осциллографа при измерении сигнала на ножке PB7 (рис. 4). Данная техника оценки времени выполнения функций будет использоваться в следующих экспериментах.
3. Подключение внешнего кварцевого резонатора
Стабильность внутреннего RC генератора микроконтроллера составляет приблизительно 1%, что недостаточно для поддержания точной настройки радиоприемника. Демонстрационная плата NUCLEO-G431KB содержит кварцевый резонатор на 24MHz, который обеспечивает стабильность порядка 0,01%. Для физического подключения кварцевого резонатора к микроконтроллеру необходимо запаять две перемычки под номером 9 и 10, как показано на рисунке 5.
Также неплохо сразу удалить перемычки 2 и 3, которые нам будут мешать в дальнейшем при подключении энкодера. Удаление перемычек лучше выполнять с помощью паяльного фена.
Тактовая частота вычислительного ядра микроконтроллера STM32G431KBT6 может достигать 170 MHz. Для подключения кварцевого резонатора к внутренним схемам микроконтроллера и увеличения тактовой частоты до 170 MHz необходимо выполнить настройку следующих модулей:
Power control (PWR),
Embedded Flash memory (FLASH),
Reset and clock control (RCC).
Выполняются перечисленные операции в функции SystemClock_Config() из файла «system_init.c»
void SystemClock_Config(void)
{
//Включить режим повышения выходного напряжения главного регулятора до 1,28 вольт
CLEAR_BIT(PWR_CR5, PWR_CR5_R1MODE);
//Установка задержки чтения FLASH памяти до 4-х тактов
FLASH_ACR=FLASH_ACR_DBG_SWEN | FLASH_ACR_DCEN | FLASH_ACR_ICEN | FLASH_ACR_LATENCY_4WS ;
while((FLASH_ACR & 0xf) != FLASH_ACR_LATENCY_4WS );
//Подключение внешнего кварцевого резонатора (HSE ON)
SET_BIT(RCC_CR, RCC_CR_HSEON);
while(READ_BIT(RCC_CR, RCC_CR_HSERDY) != (RCC_CR_HSERDY));
//Настройка PLL, используемого для SYSCLK домена
//PLLM=6 (значение поля 5), PLLN=85 (значение поля 85), PLLR=2 (значение поля 0)
MODIFY_REG(RCC_PLLCFGR, RCC_PLLCFGR_PLLSRC | RCC_PLLCFGR_PLLM| RCC_PLLCFGR_PLLN | RCC_PLLCFGR_PLLR, RCC_PLLCFGR_PLLSRC_HSE | 5 <<RCC_PLLCFGR_PLLM_Pos | (85 << RCC_PLLCFGR_PLLN_Pos) | 0<<RCC_PLLCFGR_PLLR_Pos);
//Включение PLL
SET_BIT(RCC_CR, RCC_CR_PLLON);
while(READ_BIT(RCC_CR, RCC_CR_PLLRDY) != (RCC_CR_PLLRDY));
//Включение выхода PLL, используемого для SYSCLK
SET_BIT(RCC_PLLCFGR, RCC_PLLCFGR_PLLREN);
//Подключение выхода PLL в качестве источника SYSCLK
MODIFY_REG(RCC_CFGR, RCC_CFGR_SW, RCC_CFGR_SW_PLL);
}
Повышение тактовой частоты до максимального значения 170 MHz требует принятия специальных мер по настройке внутренних блоков.
Во-первых, необходимо увеличить напряжение питание ядра микроконтроллера с 1,2 вольт до 1,28 вольт (boost mode). Управление питанием внутренних блоков микроконтроллера осуществляет специальный модуль Power control (PWR).
Во-вторых, следует увеличить количество тактов ожидания чтения данных из FLASH памяти. FLASH память является относительно медленным устройством и позволяет считывать информацию без дополнительных тактов ожидания со скоростью до 30 MHz. Если тактовая частота ядра повышается до 170 MHz, то необходимо увеличить количество тактов ожидания до 4.
Для повышения тактовой частоты c 24MHz от кварцевого резонатора до 170 MHz используется блок PLL (phase-locked loop). Настройка PLL выполняется с помощью регистра RCC_PLLCFGR (PLL configuration register). Для настройки частоты используется три поля: PLLM, PLLN, PLLR:
Fclk = ( ( Finput / PLLM ) * PLLN ) / PLLR .
В нашем случае PLLM=6, PLLN=85, PLLR=2 и тактовая частота равна 170 = ((24/6)*85)/2.
После внесения всех перечисленных выше изменений, измеренное осциллографом на линии PB7 время между двумя переключениями составляет 64,7 мс, что соответствует 64,7 нс на одну итерацию цикла, или 64,7/11 = 5,88 нс на такт (что приблизительно соответствует 170 MHz тактовой частоты). То есть все настройки выполнены правильно.
Базовый проект с подключенным внешним кварцевым генератором и увеличенной тактовой частотой процессора до 170 MHz содержится в папке 02_HSE.
4. Таймер TIM2
Для работы радиоприемника нам необходимо подать на ADC внутренний тактовый сигнал дискретизации, частота и стабильность которого в конечном счете будут определять точность настройки на радиостанцию. Обычно генератор тактовой частоты реализуется с помощью одного из таймеров контроллера. Будем использовать таймер общего назначения TIM2.
Для демонстрации работы настоим таймер на частоту 2 Гц (период 500 мс). Для этого нам необходимо записать в регистр TIM2_ARR (Auto-reload register) число K, которое определяет количество периодов входной тактовой частоты таймера (170 MHz), помещающихся в одном периоде выходной частоты таймера. Более точно: Fвых = Fтакт/(K+1). Если нам необходима частота 2 Гц, то коэффициент K должен быть равен 85000000 – 1. По умолчанию таймер работает в режиме прямого циклического счета (считает от 0 до K). При достижении значения K, счетчик таймера сбрасывается в 0, генерируется аппаратный сигнал “Counter overflow” и счет вновь повторяется от 0 до K. В нашем случае сигнал “Counter overflow” будет вырабатываться каждые 500 мс.
Настроим таймер TIM2 таким образом, чтобы при генерации сигнала “Counter overflow” вызывался обработчик прерывания - функция TIM2_IRQHandler, в которой будем одновременно переключать значение сигналов на выводах PB7 (Test) и PB8 (LD2) контроллера.
Все описанные действия выполняются в функциях, расположенных в файле «tim2.c» (проект 03_TIM2).
void TIM2_Init(void)
{
SET_BIT(RCC_APB1ENR1, RCC_APB1ENR1_TIM2EN); // Подаем на TIM2 тактовую частоту
TIM2_ARR = 85000000 - 1; // Загружаем Auto-reload register
// f = 170 000 000/(K+1), 2Hz -> (85000000 - 1)
SET_BIT(NVIC_ISER0, (1 << 28)); // Разрешить в NVIC прерывание #28 (TIM2)
SET_BIT(TIM2_DIER, TIM_DIER_UIE); // Разрешить прерывание по переполнению таймера
SET_BIT(TIM2_CR1, TIM_CR1_CEN); // Включить таймер
}
void TIM2_IRQHandler(void)
{
CLEAR_BIT(TIM2_SR, TIM_SR_UIF); // Сброс флага переполнения
INVERT_BIT(GPIOB_ODR, GPIO_ODR_OD7); // Инвертировать (PB7)
INVERT_BIT(GPIOB_ODR, GPIO_ODR_OD8); // Инвертировать LD2 (PB8)
}
Особо следует отметить процедуру подключения обработчика прерывания для таймера TIM2. По таблице 97 (STM32G4 Series vector table) документа “RM0440 Reference manual STM32G4 Series advanced Arm-based 32-bit MCUs” находим, что прерывание для устройства TIM2 имеет номер 28. Для настройки прерывания необходимо выполнить следующие действия:
1. Разрешить генерацию прерывания от устройства TIM2 (#28) в функциональном модуле NVIC (Nested vectored interrupt controller).
2. Поместить адрес функции обработчика прерывания в таблицу векторов прерывания (массив uint32_t *vectors[] в файле «vectors.c») в элемент массива под номером 28+16.
3. Разрешить прерывание по сигналу “Counter overflow” в регистре TIM2_DIER (DMA/Interrupt enable register).
После компиляции и загрузки программы в контроллер сигнал на выводе PB7 должен иметь следующий вид:
5. DAC
В следующем примере подключим к проекту цифро-аналоговый преобразователь (DAC) и будем выводить в него данные по прерыванию от TIM2.
Проект программы находится в папке 04_TIM2_DAC. В данный проект добавлен файл «dac.c»
Файл содержит одну короткую функцию инициализации модуля DAC1. В данном случае нам даже не нужно настраивать вывод PA4, так как он настроен на аналоговый режим по умолчанию.
Будем выводить информацию в DAC по прерыванию от TIM2. Для этого смодифицируем файл «tim2.c».
void TIM2_Init(void)
{
SET_BIT(RCC_APB1ENR1, RCC_APB1ENR1_TIM2EN); // Подаем на TIM2 тактовую частоту
// Загружаем Auto-reload register f = 170 000 000/(TIM2_ARR + 1) (1 MHz -> 170 - 1)
TIM2_ARR = 170 - 1;
SET_BIT(NVIC_ISER0, (1 << 28)); // Разрешить в NVIC прерывание #28 (TIM2)
SET_BIT(TIM2_DIER, TIM_DIER_UIE); // Разрешить прерывание по переполнению таймера
SET_BIT(TIM2_CR1, TIM_CR1_CEN); // Включить таймер
}
void TIM2_IRQHandler(void)
{
// Переменная I увеличивается на 1 при каждом вызове обработчика прерывания
static int I=0;
CLEAR_BIT(TIM2_SR, TIM_SR_UIF); // Сброс флага переполнения
DAC1_DHR12R1 = I++; // Запись в регистр данных DAC 12-ти младших разрядов переменной I
}
Устанавливаем период генерации таймера TIM2 равным 1 мкс. Модифицируем обработчик прерывания TIM2_IRQHandler. Добавляем в файл «main.c» вызов функции инициализации DAC1 - DAC1_Init(). После компиляции и загрузки программы 04_TIM2_DAC получаем на выходе DAC (вывод PA4) следующий сигнал (рисунок 8).
Частота полученного пилообразного сигнала равна 244.1 Hz (колонка Average внизу картинки). Разрядность DAC равна 12 битам, то есть устройство может вывести 4096 уровней напряжения в диапазоне от 0 до 3,3 вольт. При записи в 32-х разрядный регистр данных DAC1_DHR12R1, старшие разряды с 12 по 31 отбрасываются. Таким образом в обработчике прерывания циклически перебираются значения DAC1_DHR12R1 от 0 до 4095 с интервалом одного шага в 1 мкс. Период полного перебора равен 4096 мкс, то есть частота равна 1/0.004096 = 244.14 Hz, что согласуется с измерением.
6. Генератор синусоидального сигнала
Теперь попробуем создать генератор синусоидального сигнала.
Различные колебательные системы: механические, электрические, тепловые, гидравлические, и т.д., описываются идентичными математическими выражениями. Как писали классики: «Единство природы обнаруживается в поразительной аналогичности дифференциальных уравнений, относящихся к разным областям явлений».
Поэтому для создания генератора будем использовать модель пружинного маятника, как наиболее наглядную.
Из курса физики средней школы известно, что ускорение равно:
a = F/m (1).
По закону Гука:
F = - kX (2).
Подставив второе выражение в первое получим:
a = - (k/m)*X.
Обозначим k/m, как K. В этом случае предыдущее выражение примет вид:
a = - K*X (3).
Ускорение - это вторая производная координаты по времени, в связи с чем выражение (3) можно записать:
X”(t) = - K*X(t), или X”(t) + K*X(t) = 0 (4).
Уравнение (4) является дифференциальным уравнением второго порядка, частным решением которого является функция:
X(t) = A*Sin(2π*f*t) (5),
где A – амплитуда, f – частота. Действительно, если взять вторую производную от выражения (5), получим:
X”(t) = - A*(2π*f)2*Sin(2π*f*t) (6).
Подставив (5) и (6) в (4) получим:
- A*(2π*f)2*Sin(2π*f*t) = -K* A*Sin(2π*f*t).
Сократив справа и слева одинаковые сомножители получим:
(2π*f)2 = K (7),
То есть, при K = (2π*f)2 выражение (7) является тождеством. Таким образом пружинный маятник колеблется по синусоидальному закону, при этом частота колебаний на основании выражения (7) равна:
Для моделирования пружинного маятника, заменим в выражении (4) вторую производную X”(t) дискретным эквивалентом. Сделаем это следующим образом.
По определению производной:
Если взять достаточно малое ∆t = (t+ – t), то
где t+ - это следующий отсчет времени. То есть для выражения X”(t) = -K*X(t) можно записать :
((X(t+) – X(t))/ds) - (X(t) – X(t-))/ds))/ds ≈ -K*X(t) (9),
где t - текущий отсчет времени, t+ - следующий отсчет времени, t- - предыдущий отсчет времени, ds - шаг дискретизации. Преобразовав выражение (9), можно записать:
(X(t+) – X(t)) - (X(t) – X(t-)) = -K*ds2*X(t) (10),
Обозначим V1 = (X(t+) – X(t)) и V0 = (X(t) – X(t-)). При этом выражение (10) примет вид:
V1 = V0 - K*ds2*X(t) (11).
Из формулы V1 = (X(t+) – X(t)) следует, что:
X(t+) = X(t) + V1 (12).
Выражения (11) и (12) можно использовать для генерации дискретного эквивалента функции A*Sin(2π*f*t) «налету», итерационно. Если обозначить R = K*ds2 , то эквивалентом выражений (11) и (12) на языке C будет запись:
V -= X*R;
X += V;
При этом на основании выражения (7) R можно вычислить по формуле:
где f – частота генерируемого сигнала в герцах, ds – период дискретизации в секундах.
Изменим код обработчика прерывания в файле «tim2.c»:
#define ds 0.000001 // Период дискретизации в секундах
#define PI 3.1415926 // Число ПИ
#define PW 0.031415926 // PI*f*ds при f = 10000 Hz
void TIM2_IRQHandler(void)
{
static float R = 4 * PW * PW; //R=(2*PW)^2
static float V=0;
static float X = 4096/3.3 * 0.5; //Начальное значение - амплитуда сигнала 0.5 V
static float S=1500; //Смещение DAC
CLEAR_BIT(TIM2_SR, TIM_SR_UIF); //Сброс флага прерывания по переполнению таймера
//Осциллятор
V -= X*R;
X += V;
DAC1_DHR12R1 = X + S;
}
Здесь мы используем математику с плавающей запятой, поэтому нам необходимо включить сопроцессор FPU. Это делается добавлением одной строчки в функцию start_up() файла «system_init.c»:
/* Запуск FPU */
SCB_CPACR |= ((3 << (10*2))|(3 << (11*2))); /* set CP10 and CP11 Full Access */
DAC поддерживает только положительные целые значения от 0 до 4095. Переменная X принимает значения приблизительно от -620.0 до +620.0. Для того чтобы перенести значения X в положительную область, перед выводом в DAC будем прибавлять к X смещение S, равное 1500.
После компиляции и загрузки программы на выходе DAC можно наблюдать следующий сигнал (рисунок 10).
К сожалению, на частотах, приближающихся к частоте дискретизации, формула (13) перестает давать точный результат для вычисления R. Это объясняется тем, что замена производной X”(t) разностным уравнением (9) становиться достаточно грубым приближением. Например, на рисунке 11 приведена осциллограмма выходного сигнала при значении f в формуле (13) равной 200 KHz. Как видно из рисунка, реальный сигнал имеет частоту 216 KHz.
Более точная формула для вычисления величины R в зависимости от частоты имеет вид:
R = (2*sin(PI*W))2 (14),
где W - относительная частота равная f*ds. При этом W должна быть меньше 0.5 (частота Найквиста).
Проект программы по генерации синусоидального сигнала находится в папке 05_TIM2_DAC_SIN.
7. Подключение ADC
Блок-схема экспериментальной программы работы с АЦП достаточно проста: активируется ADC1 и DAC1, таймер TIM2 используется в качестве генератора частоты дискретизации для ADC, по прерыванию от ADC (окончание преобразования) сигнал с выхода ADC записывается в DAC.
Проект программы находится в папке 04_ADC_DAC. В проект добавлен файл «adc.c» , который состоит из двух функций: функции инициализации, и функции обработчика прерывания от ADC.
Как и в случае с DAС, специально настраивать вывод PA0 нет необходимости, так как аналоговый режим выбирается по умолчанию после сброса контроллера. Настройка прерывания выполняется таким же образом, как для таймера TIM2 в предыдущем примере. Прерывание происходит по окончании процесса преобразования текущего отсчета в ADC.
Некоторым изменениям подвергся файл «tim2.c». Вместо генерации прерывания теперь таймер работает в режиме мастера, то есть выдает сигнал “Counter overflow” на внутренние линии контроллера для использования в качестве триггера старта преобразования в ADC. Таким образом событие “Counter overflow” задает частоту дискретизации (Fd) входного сигнала.
Если собрать исследовательский стенд в соответствие с рисунком 1 и подать переменное напряжение от генератора на вход ADC, то на выходе DAC можно увидеть сигнал, представленный на рисунке 13. На всех осциллограммах данного раздела вход осциллографа работал в режиме AC.
При увеличении частоты генератора картинка будет меняться, на частоте 200 KHz будут отчетливо видны интервалы дискретизации сигнала (рисунок 14).
При увеличении частоты входного сигнала до значения Fn = Fd/2 = 500 KHz (частота Найквиста/Котельникова) сигнал на выходе DAC примет следующий вид (рисунок 15).
Это максимальная частота входного синусоидального сигнала, для которой выходной сигнал будет сохранять точную информацию о частоте. При дальнейшем увеличении частоты входного сигнала Fin частота выходного сигнала Fout будет изменяться в соответствии с графиком, представленным на рисунке 16.
График соответствует следующим выражениям:
Если Fin % Fd < Fn, то Fout = Fin % Fn; (15)
Если Fin % Fd >= Fn, то Fout = Fn – ( Fin % Fn); (16)
где % - остаток от деления; Fin - частота входного сигнала, подаваемого на ADC; Fout - частота выходного сигнала с DAC; Fd - частота дискретизации; Fn - частота Найквиста, равная Fd/2.
В качестве иллюстрации можно привести осциллограмму, изображенную на рисунке 17. В данном случае частота входного сигнала равна 3900 KHz, частота дискретизации -1000 KHz и соответственно частота Найквиста - 500 KHz. Осциллограф показывает частоту на выходе приблизительно 100 KHz. Проверяем условие Fin % Fd = 3900 % 1000 = 900. Полученный результат больше частоты Найквиста, поэтому выбираем выражение (16). Подставив значения в формулу (16), мы получим Fout = 500 – (3900 % 500) = 500 – 400 = 100 KHz.
Максимальное значение частоты входного сигнала определяется частотными характеристиками блока выборки и хранения ADC. Измерения показывают, что схема выборки и хранения контроллера STM32G431KBT6 работает вполне удовлетворительно вплоть до частот порядка 100 MHz. Например, так выглядит осциллограмма сигнала c частотой 65.1 MHz (рисунок 18).
Данное обстоятельство определяет два следствия: 1) Входные цепи приемника, подключенные к ADC, нужно тщательно оберегать от просачивания высокочастотных внедиапазонных сигналов. 2) При соответствующем проектировании входных цепей радиоприемника можно обеспечить прием сигналов радиостанций SW и даже FM диапазонов (!).
8. Перестраиваемый полосовой цифровой фильтр
Структурная схема проекта для исследования фильтра представлена на рисунке 19.
Мы будем использовать простейший цифровой фильтр, построенный на основе синусоидального генератора, описанного в разделе 6. Для того чтобы математическую модель идеального осциллятора, реализованного на основе пружинного маятника, превратить в фильтр, необходимо в модель осциллятора ввести потери. Поступим следующим образом - на каждом шаге вычисления нового значения X будем уменьшать это значение на некоторую небольшую величину, которая зависит от предыдущего значения X:
X += ADC;
V-=X*R; (17)
X+=V – X*L.
Потери определяются величиной L. Чем меньше значение L, тем меньше потери.
Перед выполнением очередного вычисления новых величин V и X необходимо прибавить значение выходного сигнала от ADC к X.
Интересно отметить, что для полученного фильтра полоса пропускания dF практически не зависит от частоты fr (частоты резонанса фильтра) и по уровню 0.7 приблизительно равна величине:
dF ≈ 0.18 * fd * L,
где fd – частота дискретизации. Это очень ценное свойство для перестраиваемых по частоте полосовых фильтров.
Фильтр обеспечивает «усиление» сигнала, т.е. ведет себя подобно обычному колебательному контуру. Выходной сигнал можно определить по формуле:
Uвых ≈ Uвх/L.
В данном случае величина 1/L аналогична добротности Q обычного LC колебательного контура.
Обозначим Wr = fr*ds. Более точная формула при Wr > 0.1:
Uвых ≈ Uвх(1 + Sin(PI*Wr)^2)/L
Обобщая сказанное, можно записать:
R ≈ (2*Sin(PI*Wr))2 (настройка частоты фильтра, погрешность не больше 0.5%);
dF ≈ 0.18 * fd * L (настройка полосы пропускания фильтра);
Uвых/Uвх ≈ (1 + Sin(PI*Wr)^2)/L (определение коэффициента передачи);
Проект с примером фильтра находится в папке 07_ADC_DAC_FIL. Параметры фильтра следующие:
Частота дискретизации fd = 1 MHz (ds = 1 мкс);
Центральная частота полосы пропускания фильтра fr = 100 KHz (Wr = 0.1);
Потери L = 0.01.
Рассчитываем по формулам:
R для центральной частоты полосы пропускания Wr = 0.1, R = 0,3819660;
полоса пропускания dF = 0.18*1000000*0.01 = 1800 Hz;
коэффициент передачи Uвых/Uвх = (1 + Sin(PI*0.1)^2)/0.01 = 110.
Код фильтра находится в функции-обработчике прерывания ADC1_IRQHandler файла «adc.c».
void ADC1_IRQHandler(void)
{
//Параметры фильтра на частоту 100 KHz
//R ≈ (2*Sin(PI*Wr))^2, где Wr = Fr/Fd, Fr = 100 KHz, Fd = 1000 KHz
static float R = 0.381965999;
static float L = 0.01;
//Начальные значения
static float X = 0;
static float V = 0;
//Смещение ADC и DAC
static float S = 1900;
SET_BIT(ADC1_ISR, (1 << 3)); // Сброс флага переполнения в ADC1
//Фильтр
X += (int)ADC1_DR - S;
V -= X * R;
X += V - X * L;
//Запись в DAC
DAC1_DHR12R1 = X + S;
}
Функция ADC1_IRQHandler вызывается с интервалом 1 мкс, т.е. в данном случае частота дискретизации fd равна 1 MHz.
После загрузки программы в контроллер измеряем характеристики сигнала на выходе DAC.
Как видно из диаграмм, измеренные характеристики фильтра совпадают с расчетными величинами с достаточной для практического применения точностью. Заметим, что данный фильтр сохраняет работоспособность до частот, достигающих значений Wr = 0,47, а для частот Wr <0,3 обеспечивает приемлемый уровень шумов.
Если потери L установить равным 0.001, мы получим результат, приведенный ниже.
Мы получили впечатляющий результат: фильтр в связке с 12-ти разрядным АЦП сохраняет работоспособность при амплитуде входного сигнала 1 mV. Данное значение сопоставимо с величиной одного интервала квантования АЦП (3.3V / 4096 = 0.8 mV). При этом фильтр эквивалентен LC контуру с добротность порядка Q=500.
Улучшить параметры фильтрации вне полосы пропускания возможно за счет каскадного соединения нескольких фильтров (выход одного на вход следующего). Для выбранного контроллера STM32G431KBT6 за время интервала дискретизации 1 мкс программе, написанной на C, удается обсчитать два каскада фильтра.
Для двухкаскадного фильтра при расстройке частоты входного сигнала на 10KHz происходит уменьшение амплитуды выходного сигнала где-то в 50 раз (33dB), что вполне достаточно для создания радиоприемника среднего уровня.
9. Детектор
Структурная схема проекта для исследования детектора представлена на следующем рисунке.
Модель детектора содержит функцию вычисления абсолютного значения величины (модуля) и простейшего фильтра низкой частоты.
В качестве модели фильтра низкой частоты будем использовать модель интегрирующей RC цепи.
На языке C реализация интегрирующей цепочки будет иметь вид:
U += (I – U) * K;
где U – выходной сигнал, I - входной сигнал, K рассчитывается на основании формулы (22).
Тестовый проект находится в папке 08_ADC_DAC_FIL_DET.
Код фильтра и детектора находится в обработчике прерывания ADC1_IRQHandler файла «adc.c».
#define FD 1000 //Частота дискретизации KHz
#define FR 100 //Частота фильтра KHz
#define FLPF 1 //Частота перегиба ФНЧ KHz
static float R; //Параметр фильтра, зависящий от частоты
void ADC1_IRQHandler(void)
{
//Параметр, определяющий полосу пропускания фильтра dF
//L ≈ 5.6 * Fd/FD
static float L = 0.004;
//Начальные значения переменных фильтра
static float X = 0;
static float V = 0;
//Параметр интегрирующей цепочки
static float K = 2*M_PI*FLPF/FD; //K = 2PI*f/fd
//Начальные значения переменных детектора
static float U = 0;
float D;
int out;
//Смещение для ADC и DAC
static int S_ADC = 2000;
static int S_DAC = 100;
// Сброс флага переполнения в ADC1
SET_BIT(ADC1_ISR, (1 << 3));
//Фильтр
X += (int)ADC1_DR- S_ADC;
V -= X * R;
X += V - X * L;
//Детектор
D = fabs(X);
U += (D - U) * K; // Интегрирующая цепь
out = U + S_DAC;
// Ограничитель, предотвращает переполнение DAC при сильном сигнале
if( out > 4000) out = 4000;
//Запись в DAC
DAC1_DHR12R1 = out;
}
После загрузки программы в контроллер измеряем характеристики сигнала на выходе DAC.
Полученный макет является, по сути, приемником амплитудно-модулированных сигналов (АМ) для диапазона частот 10 KHz – 400 KHz. Чувствительность устройства при отношении сигнал/шум 10дб составляет порядка 2 mV, селективность 26дб при расстройке 5 KHz, полоса пропускания 600 Hz по уровню 0.7. Это довольно неплохие характеристики.
К недостаткам устройства можно отнести относительно высокий уровень шума звукового сигнала в следствии ограниченного разрешения ADC и DAC (12 разрядов), относительно низкой частоты дискретизации (фильтр используется до частот Wr = 0.4, или всего 2.5 такта частоты дискретизации на период входного сигнала) и ограниченной разрядной сетки формата float32.
10. Настройка частоты – энкодер
Для изменения частоты настройки фильтра будем использовать специальное устройство - инкрементальный энкодер. Устроен он просто: две контактные группы замыкаются в нужном порядке в зависимости от направления вращения вала. Используемый энкодер EC11 содержит дополнительную кнопку (BUT), которая замыкается при нажатии на вал.
При вращении энкодера будем увеличивать/уменьшать частоту на 1 KHz на каждый «щелчок». При нажатии на кнопку энкодера (кнопка BUT) будем увеличивать частоту до ближайшего большего значения сотен. Например, если частота была 123 KHz, то после нажатия на кнопку BUT значение будет 200 KHz.
Работа с инкрементальным энкодером является стандартной функцией для таймеров контроллера.
Выберем таймер TIM3 для обслуживания энкодера. Отрабатывать нажатие кнопки будем через контроллер внешних прерываний (EXTI).
Для обеспечения работы таймера TIM3 с энкодером нам нужно выполнить следующие шаги.
Настройка выводов контроллера PA6, PA7 на режим логического входа, с «подтяжкой» к положительной шине питания с помощью внутреннего резистора.
Настройка PA6 и PA7 на альтернативную функцию ввода ( см. datasheet DS12589 Rev.2 Table 13. Alternate function), а именно подключение данных ножек через схемы внутренней коммутации ко входам tim_ti1 и tim_ti2 таймера TIM3, отвечающим за обработку сигналов от энкодера.
Настройка управляющих регистров таймера TIM3 на режим работы с энкодером.
Настройка прерывания от TIM3 по изменению содержимого счетчика таймера.
Создание функции-обработчика прерывания от таймера TIM3, которая будет модифицировать глобальную переменную «frequency».
Для обработки нажатия кнопки «BUT» через контроллер внешних прерываний выполняем следующие шаги.
Настройка вывода контроллера PA5 на режим логического входа с «подтяжкой» к положительной шине питания с помощью внутреннего резистора.
Настройка прерывания от блока EXTI при возникновении положительного фронта на ножке PA5 (канал внешнего прерывания EXTI5).
Создание функции-обработчика прерывания от блока EXTI по каналу EXTI9_5, которая будет увеличивать глобальную переменную «frequency» на 100 при каждом событии прерывания.
Работа счетчика таймера (регистр TIM3_CNT) в режиме обслуживания энкодера демонстрируется следующим рисунком.
Необходимо заметить, что значения регистра TIM3_CNT изменяются в диапазоне от 0 до значения, записанного в регистр TIM3_ARR (Auto-reload register) минус 1.
На рисунке 34 представлена схема исследовательского стенда с подключенным энкодером. Для работы энкодера необходимо удалить перемычки 2 и 3, как указано на рисунке 5.
Проект работы с энкодером находится в папке 09_ADC_DAC_DET_FIL_DET_EN. В проект был добавлен файл «encoder.c», в котором находится функция инициализации энкодера и обработчики прерывания от TIM3 и EXTI5. Модификация глобальной переменной R, необходимой для задания центральной частоты полосы пропускания фильтра, выполняется каждый раз при вызове обработчика прерывания.
Для работы энкодера в проект введены следующие константы:
#define FREQ_MAX 400 // Максимальное значение частоты KHz
#define FREQ_MIN 100 // Минимальное значение частоты KHz
#define FREQ_START 100 // Начальное значение частоты KHz
Посмотреть результат работы программы можно с помощью осциллографа и генератора, включенного в режиме качания частоты.
11. OLED дисплей
Цифровая шкала настройки позволяет существенно повысить удобство использования радиоприемника. В проекте используется OLED графический индикатор на основе контроллера SSD1306. Индикатор имеет разрешение 128 х 32 точки и подключается к микроконтроллеру через интерфейс I2C.
Вывод алфавитно-символьной информации графического индикатора требует достаточно большого количества операций: генерация графического образа символа в памяти контроллера и побайтная передача образа символа в память дисплея по интерфейсу I2C.
Работу по передаче образа символа можно переложить на модуль прямого доступа к памяти (DMA) и таким образом разгрузить центральный процессор для задач обработки сигнала.
При вызове функции print() выполняется формирование в памяти контроллера многобайтной последовательности (команда), которую нужно передать по интерфейсу I2C в индикатор. Далее в модуль DMA записывается начальный адрес блока данных и количество байт для передачи, после чего в модуль I2C передается команда «Старт передачи». Далее процесс передачи блока данных по каналу I2C выполняется без участия основного процессора контроллера.
Тестовый проект с программой для стенда, схема которого представлена на рисунке 36, находится в папке 10_ADC_DAC_DET_FIL_DET_EN_OLED.
12. Практическая схема радиоприемника
Схема простого радиоприемного устройства на базе отладочной платы NUCLEO-G431KB представлена на рисунке 37. Радиоприемник принимает сигналы в диапазоне длинных волн (LW) 150 – 280 KHz. На транзисторах T1 – T3 собран усилитель высокой частоты (УВЧ) с общим коэффициентом усиления порядка 100. Элементы L1, R3 и C3 образуют низкодобротный колебательный контур магнитной антенны, настроенный на середину диапазона. Первый каскад УВЧ собран по схеме дифференциального усилителя, который в значительной степени позволяет подавить синфазную помеху по входу и по линии питания. Подавление синфазной помехи является критически важным во время приема в городских условиях, при наличии большого количества бытовых электроприборов. Транзисторы T1 и T2 должны иметь по возможности одинаковые коэффициенты усиления по току. На транзисторе T3 собран второй каскад УВЧ, с выхода которого через разделительный конденсатор C6 высокочастотный сигнал подается на вход АЦП контроллера. Резисторы R9 и R10 образуют делитель напряжения, который выводит «нулевую» точку АЦП в середину диапазона значений, где-то около 2000.
Низкочастотный сигнал с выхода ЦАП поступает на фильтр низкой частоты (R11, C7, L2, C8), с частотой среза порядка 4 KHz, который служит для подавления шумов, возникающих за счет цифровой обработки сигнала. При некотором снижении качества фильтрации дроссель L2 можно заменить на резистор номиналом 1 ком. Выход фильтра подключен к усилителю мощности, собранному на транзисторе T4 по схеме эмиттерного повторителя. Наличие усилителя мощности позволяет подключить к устройству низкоомные наушники. Особенностью каскада является то, что он по постоянному току подключен к выходу ЦАП и не имеет отдельных цепей смещения. Выходное напряжение с выхода ЦАП изменяется от 0.6 до 3 вольт, что автоматически задает оптимальный режим работы каскада. При необходимости громкоговорящего приема сигнал с переменного резистора R13 можно подать на вход внешнего усилителя мощности.
Радиоприемник питается от четырех аккумуляторов AA с номинальным напряжением 1.2V.
Проект с программой простейшего радиоприемника находится в каталоге 11_SIMPLE_RADIO.
Основное отличие от предыдущего проекта заключается в ведении автоматической регулировки уровня низкочастотного сигнала. Амплитуда выходного сигнала после детектора может меняться в диапазоне 90 дб. Динамический диапазон ЦАП составляет приблизительно 70 дб. Данное обстоятельство приводит к ограничению или полной потере выходного сигнала при приеме мощной радиостанции. Решением данной проблемы является автоматическое уменьшение уровня выходного сигнала при превышении среднего уровня некоторого порога, т.е. применение автоматической регулировки усиления (АРУ).
Программный код АРУ выглядит довольно просто:
OUT = U1;
//АРУ
U2 += (D - U2) * K2; // Интегрирующая цепь АРУ
if( U2 > AGCT) OUT = U1*(AGCT/U2);
Где U1 - это выход ФНЧ детектора. Мы используем еще одну интегрирующую цепь с частотой перегиба порядка 1 Hz. Если значение U2 меньше некоторого порога AGCT (порядка 1000), то сигнал OUT не меняется. В противном случае выходной сигнал уменьшается по формуле OUT = U1*(AGCT/U2). Дополнительным преимуществом использования АРУ является уменьшение эффекта замирания сигнала. Конечно, здесь речь идет о достаточно мощных радиостанциях.
К сожалению, в данной конфигурации программа работает на пределе производительности, в связи с чем пришлось уменьшить частоту дискретизации до 700 KHz. Для приема в длинноволновом диапазоне это не критично, и мне удавалось в Зеленограде (Московская область) в вечернее время на подоконнике окна многоэтажного дома достаточно уверенно принимать сигналы радиостанций на частотах 153 KHz (радио Румынии), 225 KHz (радио Варшава).
Необходимо заметить, что максимальная частота зависит не только от возможностей контроллера и разработанного программного кода, но и от используемого транслятора и его настроек. В проекте применялся Toolchain, поставляемый вместе с STM32CubeIDE (gnu-tools-for-stm32.9-2020-q2-update) при отключенной оптимизации (Optimization level: None –O0). При использовании компилятора и базовых библиотек от другого toolchain производительность может отличаться, иногда существенно.
По количеству мощных радиостанций диапазон средних волн 530 – 1600 KHz (MW) значительно более интереснее длинноволнового диапазона, но для достижения подобных «высот» нам необходимо увеличить производительность программы в 5 раз и поднять частоту дискретизации до 3500 KHz. Мы использовали далеко не все возможности микроконтроллера. В следующих главах рассмотрим возможности для «разгона» программы.
13. Захват сигнала с помощью DMA
Первое очевидное ускорение — это не тратить время на вызов обработчика прерывания при обработке каждого отсчета от ADC. Только на вызов прерывания требуется минимум 12 тактов процессора. К счастью, в арсенале контроллера имеется модуль прямого доступа к памяти (DMA). Записывать данные от ADC можно циклически в буфер памяти, расположенный в RAM, практически без участия процессного ядра. При заполнении буфера модуль DMA вызовет обработчик прерывания, который может запустить к исполнению код обработки сигнала сразу для большого количества отсчетов. Такая схема работы позволяет параллельно заполнять текущий буфер памяти отсчетами от ADC с помощью DMA и обрабатывать предыдущий буфер с помощью ядра процессора.
Проект программы, работающей по описанной схеме, находится в папке 12_SIMPLE_RADIO_DMA.
Изменений по сравнению с предыдущем проектом не так много, как может показаться на первый взгляд. В более ранних экспериментах мы просчитывали каждый отсчет ADC по прерыванию «Окончание преобразования в ADC» в обработчике ADC1_IRQHandler, в текущем эксперименте обрабатывается сразу 64 отсчета по прерываниям от DMA - «заполнена первая половина буфера» или «заполнена вторая половина буфера». Вся математика остается прежней и выполняется в обработчике DMA1_CH4_IRQHandler.
Подача запросов на передачу одного слова через DMA выполняется через специальный мультиплексор DMAMUX, который имеет 127 входов и может подключать запросы практически от всех устройств контроллера (рис. 40). Запрос на передачу отсчета от ADC подключен к линии номер 5.
Контроллер имеет два модуля DMA, каждый из которых содержит 6 независимых каналов. В нашем проекте мы используем DMA1 канал 4. Настройка мультиплексора DMAMUX1 и блока DMA1 выполняется в функции ADC1_init из файла «adc.c».
Еще одно изменение произошло в последовательности обработки отсчетов в обработчике DMA1_CH4_IRQHandler. Перестраиваемый полосовой фильтр работает внутри цикла перебора отсчетов из буфера, а ФНЧ и АРУ реализован вне цикла. Таким образом мы существенно разгружаем процессор в части обработки звукового сигнала.
Внесенные изменения позволили увеличить частоту дискретизации более чем в 2 раза с 700 KHz до 1600 KHz (константа FD в файле adc.c). Теперь имеется возможность настраивать фильтр на частоты 150 – 750 Khz, принимать сигналы диапазона LW и захватывать нижнюю часть MW диапазона.
14. Блок цифровой обработки сигнала (DSP) на ассемблере
Итак, для перекрытия всего MW диапазона необходимо увеличить скорость обработки сигнала минимум еще в 2 раза. Традиционным путем увеличения производительности программы является написание наиболее нагруженных частей кода на языке ассемблера. Такой критической секцией в нашем проекте является обработчик прерывания DMA1_CH4_IRQHandler.
Вычислительное ядро контроллера содержит два процессора: один - для обработки целых чисел, другой - для чисел с плавающей запятой (рисунок 41). Сопроцессор для работы с плавающей точкой содержит 32 регистра. На рисунке 42 представлено распределение регистров сопроцессора при использовании стандартных функций языка C.
Для работы радиоприемника необходимо использовать 12 глобальных переменных с плавающей точкой:
параметры фильтра - X, V, R, L, M;
параметры интегрирующей цепочки детектора - U1, K1;
параметры интегрирующей цепочки АРУ U2, K2;
порог АРУ - AGCT;
смещение ADC и DAC - S_ADC, S_DAC.
Для ускорения работы было бы не плохо хранить перечисленные переменные непосредственно в регистрах сопроцессора. Но имеется одно принципиальное препятствие – если мы используем в проекте какую-либо арифметику с плавающей запятой в функциях, написанных на языке C, то регистры сопроцессора могут быть произвольно переписаны и наши переменные будут изменены.
В текущем экспериментальном проекте мы полностью берем контроль над работой сопроцессора и отказываемся от использования плавающей точки в коде на C. Для этого нам нужно переписать две существующие функции:
DMA1_CH4_IRQHandler – обработчик прерывания от DMA для ADC.
KHZ_to_R - расчет параметра фильтра в зависимости от частоты R=(2*Sin(PI*W))^2.
Также необходимо добавить новую функцию dsp_init, которая записывает в регистры сопроцессора начальные значения при старте программы.
В предыдущем проекте в функции KHZ_to_R использовалась стандартная библиотечная функция языка C - sin(x). В новом проекте мы будем применять другой метод расчета R, а именно кусочно-линейную аппроксимацию, которая позволяет достичь даже несколько лучших результатов по точности настройки фильтра.
Проект находится в папке 13_SIMPLE_RADIO_DMA_ASM. Весь ассемблерный код помещен в файл «dsp.s». Не будем подробно останавливаться на программировании на ассемблере для ARM cortex M4 процессоров. Код достаточно подробно прокомментирован.
Использование ассемблера позволило увеличить скорость обработки практически в три раза. Теперь мы можем поднять частоту дискретизации до значения 3600 KHz и таким образом перекрыть весь MW диапазон.
Рассмотрим методику получения коэффициентов аппроксимации параметра R цифрового фильтра. Для определения корректирующей функции воспользуемся формулой расчета параметра фильтра:
Нам необходимо раскомментировать строку под номером 100 в функции KHZ_to_R файла «dsp.s». Будем подавать на вход контроллера амплитудно-модулированный сигнал от генератора. Частоту сигнала будем менять от 100 KHz до 1600 KHz с шагом 100 KHz (схема стенда рис. 36). Для каждого значения входной частоты будем настраивать фильтр с помощью энкодера в резонанс и считывать показание частоты на OLED индикаторе. Результаты можно записать в виде таблицы (первых три столбца).
№ |
Входная частота Fr KHz |
Выходная частота Frc KHz |
A[i] |
B[i] |
0 |
100 |
100 |
1,000 |
0,00 |
1 |
200 |
199 |
0,990 |
1,00 |
2 |
300 |
297 |
0,980 |
3,00 |
3 |
400 |
392 |
0,950 |
12,00 |
4 |
500 |
485 |
0,930 |
20,00 |
5 |
600 |
574 |
0,890 |
40,00 |
6 |
700 |
658 |
0,840 |
70,00 |
7 |
800 |
738 |
0,800 |
98,00 |
8 |
900 |
812 |
0,740 |
146,00 |
9 |
1000 |
880 |
0,680 |
200,00 |
10 |
1100 |
941 |
0,610 |
270,00 |
11 |
1200 |
995 |
0,540 |
347,00 |
12 |
1300 |
1042 |
0,470 |
431,00 |
13 |
1400 |
1081 |
0,390 |
535,00 |
14 |
1500 |
1111 |
0,300 |
661,00 |
15 |
1600 |
1134 |
0,230 |
766,00 |
Таблица 1. Результат расчета коэффициентов линейной аппроксимации
Таким образом, чтобы настроить приемник на частоту, например, 1000 KHz нам необходимо подставить в формулу (13) значение частоты 880 KHz.
Для расчета промежуточных значений будем использовать формулу:
где Frc - скорректированная частота, Fr – частота резонанса, i – индекс элемента массива коэффициентов линейной аппроксимации i = Fr % 100 (номер строки в таблице 1).
Значение A[i] и B[i] рассчитываются по формулам:
Пример расчета элементов массивов приведен в excel файле «Коррекция шкалы F I.xlsx», размещенном в каталоге текущего проекта. Полученные коэффициенты помещаем в массивы A[17] и B[17] файла «adc.c».
Теперь нужно закомментировать строчку номер 100 в функции KHZ_to_R файла «dsp.s» и проверить точность настройки. Измерения показали, что до частоты 1000 KHz погрешность настройки не превышает 1 KHz и на частотах от 1000 KHz до 1600 KHz не превышает 3 KHz.
Теперь можно проверить реальную скорость обработки блока в 64 отсчета ADC в функции DMA1_CH4_IRQHandler. Для этого раскомментируем строчки в данной функции, отвечающие за изменение сигнала на выводе PB7. Осциллограмма сигнала представлена на рисунке 44. Как видно из рисунка, время обсчета 64 отсчетов составляет приблизительно 10 мкс, а время записи сигнала в буфер (период между вызовами функции DMA1_CH4_IRQHandler) составляет приблизительно 17 мкс. Таким образом мы имеем достаточный запас по производительности и можем внести некоторые усовершенствования в программу.
Надо отметить, что в связи с расширением диапазона принимаемых сигналов схема приемника (рис. 37) претерпела небольшие изменения (рис. 45): 1) количество витков катушки L1 теперь должно быть 35 + 35 витков, 2) из схемы удалены элементы R3, C2.
15. Усовершенствованный радиоприемник
Увеличение скорости обработки позволило использовать еще один каскад фильтра в функции TIM4_IRQHandler и таким образом улучшить селективность радиоприемника.
Также был организован вывод относительной величины входного сигнала на OLED индикатор. В качестве уровня входного сигнала используется выходное значение интегрирующей цепочки блока АРУ (U2), деленное на 10. Вывод происходит раз в секунду по прерыванию от таймера TIM4. Величина «100» на индикаторе приблизительно соответствует 10 мВ на входе ADC1. Учитывая, что коэффициент усиления УВЧ составляет около 100, одна единица выводимого уровня сигнала соответствует 1 мкВ сигнала от магнитной антенны.
Еще одно усовершенствование связано с изменением коэффициента компенсации передачи фильтра (M) в зависимости от частоты настройки. При изменении частоты настройки от низкочастотного конца диапазона к высокочастотному коэффициент передачи двухкаскадного фильтра увеличивается приблизительно 50 раз. Изменение происходит по нелинейному закону. Компенсация коэффициента передачи фильтра позволяет снизить уровень шумов в высокочастотной части диапазона.
Финальная обобщенная схема радиоприемника представлена на рисунке 46. Проект находится в папке 14_SIMPLE_RADIO_DMA_ASM_II.
Хочется отметить, что весь проект содержит около 1000 строк кода с комментариями, занимает около 7KB флэша и 300 байт RAM.
Демонстрацию работы радиоприемника можно посмотреть по ссылке https://youtu.be/W8Ki2eaznIk.
Комментарии (50)
Saalur
11.11.2021 15:07+8Исключительно добротный материал. Проработанная и качественная подача материала, отличное оформление статьи. Номинируйтесь на Технотекст 2021, с удовольствием проголосую.
iShrimp
11.11.2021 19:19+2Чувствуется основательный подход к делу, автор - инженер старой школы, который получил в руки все современные технологии :)
Один вопрос: если уж упомянули гетеродинирование, то может быть стоит первым этапом после АЦП умножать сигнал на синус ВЧ, а затем не спеша обрабатывать его фильтрами высоких порядков?
OlegDyakov Автор
11.11.2021 22:38+2Спасибо. Задача была сделать учебный приемник "прямого усиления". Схема с гетеродинированием в плане для следующей статьи.
Make_Pic
14.11.2021 13:08Я опять выступлю как скептик, в статье слишком большая "разжеванность" подачи некоторых аспектов, выходящих за рамки статьи. Пример про пружину. А в общем тема DSP обработки сигналов на распространенных MCU, младших Cortex-ах, вполне неплохо представлена автором.
zxcvbv
11.11.2021 15:50+3А в AM-диапазоне еще есть жизнь?
Javian
11.11.2021 16:11+1Сам удивился. Возможно находится автор у границы Китая или Украины (https://worldradiomap.com/map/). По фразе "По количеству мощных радиостанций диапазон средних волн 530 – 1600 KHz (MW) значительно более интереснее длинноволнового диапазона" подозреваю Китай.
У нас более интереснее попробовать сделать "навигатор" для системы "Чайка". Что-то об этом попадалось на хабре.
muromdx
11.11.2021 20:54+2У автора типичный для Центральной России вечерний эфир - на ДВ принимается Румыния (153 кГц) и Польша (225 кГц), на СВ много мощной Румынии, громко слышна Украина на 549 кГц, Питер на 828 кГц, Эстония на русском на 1035 кГц, Радио Свобода на 1386 кГц, Вести ФМ на 1413 кГц. В ролике он сам сказал, что находится в Зеленограде. Рядом с Китаем эфир будет кардинально другой, там славянской речи вы не услышите. Можете попробовать для интереса включить вечером приемник с ДВ и СВ, желательно на улице, если есть такая возможность. Жизнь на СВ все еще есть.
Javian
11.11.2021 21:04+1Вероятно главное условие приема - большая магнитная антенна от советского радиоприемника типа ВЭФ.
muromdx
11.11.2021 21:12+1Да, антенна сильно влияет на прием. Но у автора еще и довольно чистый эфир. Сейчас в городской квартире, где куча помех от китайских зарядок, блоков питания компьютеров, телевизоров и всего прочего, бывает очень проблематично послушать АМ-станции.
sim2q
11.11.2021 21:52+1Сейчас в топе диодные светильники без фильтров вообще!
Они ещё и по частотам уже в СВ напрямую залезли.
OlegDyakov Автор
11.11.2021 22:43+2На самом деле, эфир типичный для городской квартиры, с обилием светодиодных светильников. Избавиться от части помех помог дифференциальный каскад на входе усилителя высокой частоты.
DanilinS
11.11.2021 16:28+3Вопрос по реализации:
1) Чип STM32G431 имеет математический ускоритель для расчета тригонометрии. Может имеет смысл его использовать для расчетов? А то такой инструмент простаивает, а вы на ассемблер уходите ...
2) На чипе есть аппаратный цифровой фильтр FMAC. Зачем тогда Вам нужен программый фильтр?
Все это вместе может сильно снять нагрузку с основного чипа.
mctMaks
11.11.2021 17:39цифровой фильтр FMAC
а вы его сами пробовали? интересно как оно в действии
edit: а разве FMAC не в новых stm32U5 появился?увидел. в это тоже есть, они его почему-то на диаграмме не показали как FMAC, а просто написали filtering. для U5 было было виднее)
iShrimp
11.11.2021 19:01+1Ещё можно ускорить программу, переписав вычисления на целочисленных SIMD-инструкциях. Хотя у процессоров Cortex-M4 нет 64-битных регистров, но они умеют обрабатывать 32-битное слово как 2 16-битных или 4 8-битных числа одной командой. При разрядности сигнала 12 бит часть вычислений можно проводить в 16-битном формате.
OlegDyakov Автор
11.11.2021 23:16+1Пробовал, получается не очень хорошо. Использование 16 бит не обеспечивает достаточный динамический диапазон для качественной работы фильтра. Использование float32 существенно улучшает работу без заметного уменьшения производительности. Когда познакомился с системой команд сопроцессора с плавающей точкой, очень удивился, насколько нерационально написана стандартная библиотека С работы с float (смотрел дизассемблерный код).
Многие команды сопроцессора, такие как сложение, вычитание, умножение и т.д. выполняются за один такт. А DSP команда VMLA.F32 (Floating-point Multiply
and Add Sd <- Sd + Sn × Sm) выполняется за три такта. То есть на ассемблере работа с int и float не сильно отличается по времени.iShrimp
12.11.2021 16:38Когда познакомился с системой команд сопроцессора с плавающей точкой, очень удивился, насколько нерационально написана стандартная библиотека С работы с float (смотрел дизассемблерный код).
Это может быть вызвано требованиями стандартов языка С, т.к. каждая функция должна корректно обрабатывать все возможные значения, предусмотренные IEEE-754 (0.0, -0.0, NaN, субнормальные числа и т.д.), с точностью до последнего бита. Недавно была статья об опции компилятора -ffast-math, которая может помочь ускорить вычисления за счёт уменьшения строгости проверок. Например, без ключа -fassociative-math компилятор не станет переставлять аргументы и упрощать многие математические выражения.
OlegDyakov Автор
11.11.2021 22:59+11) Генерация синуса в тексте статьи приводилась для описания модели элементарного осциллятора и демонстрации работы ЦАП. В макете приемника осциллятор не использовался. Для реализации гетеродинной схемы попробую использовать CORDIC сопроцессор.
2) Аппаратный ускоритель FMAC работает с 16-ти разрядными целыми числами. В реализации приемника используется 32-х битный float, а это уже совершенно другой уровень качества обработки сигнала.
VT100
11.11.2021 22:33+1Ничоси! Пасхалочка с маятником...
Например, на рисунке 11 приведена осциллограмма выходного сигнала при значении f в формуле (13) равной 200 KHz.
В online-симуляторе DDS'ок от Аналоговых Девиц — примерно такие-же буераки на выходе. А после фильтра — всё становится чинно-благородно.
Итак, для перекрытия всего MW диапазона необходимо увеличить скорость обработки сигнала минимум еще в 2 раза. Традиционным путем увеличения производительности программы является написание наиболее нагруженных частей кода на языке ассемблера.
НЯЗ — ещё можно перенести критичные секции кода в ОЗУ и избавиться от циклов ожидания ПЗУ.
OlegDyakov Автор
11.11.2021 23:341) В приемнике на выходе перестраиваемого фильтра стоит ФНЧ для сглаживания "буераков". На рисунке 11 - демонстрация "ужасов" промежуточных результатов цифровой обработки.
2) Хорошая идея - перенести критический код в RAM. Попробую.
zxcvbv
12.11.2021 08:59Хорошая идея - перенести критический код в RAM. Попробую.
Ускорятся только операции условных переходов. Сама внутрянка циклов и if'ов будет работать +/- на той же скорости.
VT100
12.11.2021 16:10Имеете в виду что-то типа "ART accelerator"? Но, если есть свободное ОЗУ, — можно получить гарантированные "0-wait state".
zxcvbv
12.11.2021 16:20Имеете в виду что-то типа "ART accelerator"?
Instruction prefetch есть даже на древнем STM32F0, который в отсутствие бранчей устраняет wait state.
А на STM32G* есть ещё Cache memory, при включении которой, если бранчей немного и они часто повторяются, простои минимизируются и на ветвлениях тоже.
OlegDyakov Автор
12.11.2021 19:57ART акселератор контроллера STM32G содержит очередь предварительной выборки инструкций и кеш ветвлений. Код DSP обработчика ( функция DMA1_CH4_IRQHandler из файла dsp.s) достаточно компактна - код основного цикла обработки занимает 62 байта и скорее всего полностью помещается в кэш.
В главе 2 (базовый проект) аналитически подсчитывается, что количество тактов ядра, затрачиваемых на выполнение цикла while( i ) i--; равно 11 (без тактов ожидания). В главе 3 (подключение внешнего кварцевого резонатора) проводится измерение с помощью осциллографа времени одной итерации цикла на частоте ядра 170 MHz. Оно равно 64,7 нс, или или 64,7/11 = 5,88 нс на такт, что соответствует 170 MHz тактовой частоты. То есть ART акселератор для коротких блоков кода довольно эффективно работает.
tri_botinka
11.11.2021 23:09-1Это фигня.. Народ умудрялся на голой ардуинке за счёт тонкого real-time выдавать с цап модулированный аналоговый телесигнал с видео, который можно было на антенну телика подать. Правда не высокого разрешения и чб, но все же..
Strijar
12.11.2021 14:36+2Модулировынный не получится - очень высокая частота. На AVR делали генератор NTSC сигнала через SPI. Но это не антенный вход, а композитный видео. При этом и цветной тоже (в палитре 16 цветов)
Eugeny5995
11.11.2021 23:19+2В этой серии контроллеров есть такой экземпляр: stm32g474. У него есть волшебный high resolution таймер, который в сочетании с возможностью конфигурации тактовой частоты способен творить чудеса для построения квадратурных детекторов или сэмплеров. Сам применяю именно эту серию и вам рекомендую :)
OlegDyakov Автор
11.11.2021 23:20+1Спасибо. Именно так и собирался сделать для квадратурного детектора. Таймер уже опробовал в работе.
IgorPie
12.11.2021 03:00Очень хорошая статья!
Мк серии G выбраны из каких соображений?
Есть же stm32h750, 16 бит АЦП, 400МГц, 64 бит double аппаратно и цена 7 долларов.
На борту заявлено пара ОУ, тоже можно пустить в дело
OlegDyakov Автор
12.11.2021 10:41+1Спасибо.
stm32G выбран из соображения цены и качества. NUCLEO-G431KB самая доступная плата, а статья прежде всего адресована старшеклассникам и "младшекурсникам". Да и хотелось сделать именно "простой" приемник прямого усиления (хотя , на мой взгляд, это DDR).
stm32h750 - лежит у меня на столе. Очень хочется попробовать.
Пытался использовать ОУ на борту в качестве УВЧ. Имеется два недостатка: 1) частота единичного усиления 15 MHz, то есть на 2 MHz усиление не более 7 (7*7 = 49, а нужно 100); 2) очень сильно шумят.
vsb
12.11.2021 10:23Потихоньку разбираюсь с STM32, ваша статья мне, похоже, пригодится, большое спасибо.
Единственное, что хотел бы попросить авторов подобных статей - когда вы пишете какую-то информацию, добавляйте ссылку к мануалам. Этого очень не хватает. Мануалы огромные и ориентироваться в них без опыта сложно. Как говорится, научите ловить рыбу, а не кушать.
OlegDyakov Автор
12.11.2021 10:49+1Добрый день.
Мануалы, которые я использовал для работы, указаны в начале статьи.
«RM0440, Reference manual STM32G4 Series advanced Arm ® -based 32-bit MCUs» - подробное описание функциональных блоков контроллеров серии STM32G4.
«DS12589 Rev 2 - Datasheet STM32G431x6, STM32G431x8, STM32G431xB» - описание контроллеров линейки STM32G431x6, STM32G431x8, STM32G431xB.
-
«PM0214 Rev 10 - STM32 Cortex ® -M4 MCUs and MPUs programming manual» - руководство программиста.
Не дал прямые ссылки. Просто не уверен, что это можно сделать ничего не нарушая.
Названия документов указаны точно, через googl вы сразу выйдите на ссылки.
VT100
12.11.2021 16:12Не дал прямые ссылки. Просто не уверен, что это можно сделать ничего не нарушая.
Так они на сайте производителя должны лежать. Или я что-то пропустил и ST micro. теперь огораживается?
zxcvbv
12.11.2021 16:25добавляйте ссылку к мануалам. Этого очень не хватает.
Ссылка к мануалам ищется в 1 клик, что через STM32MCUFinder, что сразу в STM32CubeIDE, в зависимости от выбранной железяки...
vsb
12.11.2021 16:29Я имел в виду ссылки на конкретные главы и тд. К примеру в статье описывается, что такие-то адреса маппятся на такие-то адреса. Но в каком месте это написано - я сходу не смог найти. Как я написал - мануалы огромные и ориентироваться в них сложно, когда не представляешь, куда смотреть. Если проследить за тем, как ориентируется в них опытный человек, можно для себя много полезного вынести.
Сами-то мануалы, конечно, найти не проблема.
zxcvbv
12.11.2021 16:39Если честно, не понимаю, в чем проблема. Все RM идут с оглавнением по блокам ядра и периферии. Тыкаем нужный блок, получаем в начале краткие fundamentals, в конце описание регистров и картой адресов. Некоторые старые RM еще идут с примерами кода в конце (в новых почему-то их зачастую уже нет).
А вообще их по-хорошему надо читать от корки до корки (хотя бы один раз, по самому жирному камню), или хотя бы на две трети, опустив совсем уж специфическую периферию. Не представляю, что можно накодить, ознакомившись лишь точечно с конкретными блоками, не представляя картины по железке в целом.
STMshchik
12.11.2021 13:45Очень крутая статья. Какую литературу по цифровой обработке сигналов можете посоветовать для новичка? Захотелось даже самому всё это испробовать и сделать фильтр для звуковых частот на МК
OlegDyakov Автор
12.11.2021 20:33+1Как было правильно сказано про меня в одном из комментариев "инженер старой школы, который получил в руки все современные технологии :)". То есть я изучал цифровую обработку еще в советское время. Но как-то мне попалась в руки одна интересная книга, которую могу порекомендовать новичкам: Юкио Сато "Без паники! Цифровая обработка сигналов". Книга была переведена на русский язык в 2010 году.
STMshchik
24.11.2021 08:16Стал читать эту книгу. Это бомба просто! Везде пичкают непонятные математические выкладки. А тут автор проводит целый экскурс в необходимые разделы математики и очень наглядно объясняет математику. Вспоминаю вузовский курс математики и офигеваю, как он наглядно, доступно с приведением аналогий объясняет все понятия.
skoptsev
12.11.2021 23:32Это просто бомба. Тянет на диплом. Спасибо, очень интересно, а главное доступно.
DrAndyHunter
15.11.2021 10:09Простите за возможно глупейший вопрос, я только интересуюсь ЦОС и радиоприемом.
Я вычитал у Ричарда Лайонса, что частоту дискретизации можно выбрать не >= 2f, а наоборот, еще и меньше чем срединная частота необходимой полосы приема?OlegDyakov Автор
15.11.2021 17:50+1Хороший вопрос. Возможность приема на "занайквистовых" частотах я постараюсь описать в следующей статье :)
McHummer1
22.11.2021 23:59+1Зачот!
Аффтар, пеши есчо!
Единственная просьба - не поднимать в одной и той же статье две разные сложные темы - здесь стоило бы по возможности использовать hal или на худой конец cmsis, а низкоуровневое программирование вынести в отдельную статью. Две трудные темы вместе затрудняют восприятие.
С нетерпением жду следующий статей на тему програмных приемников и обработки сигнала на stm32
balamutang
Надо же, SDR своими руками.
Следующий шаг - эту конструкцию стоит доработать до супергетеродина, частота ПЧ будет как раз по зубам этому контроллеру. А в качестве гетеродина взять какой-нить Si5351
Ну и модуляций добавить, не только АМ, но и SSB и FM можно докинуть
iliasam
Квадратур тут для SSB явно не хватает.
OlegDyakov Автор
Изначально статья задумывалась как учебный материал для новичков. В дальнейшем обязательно доработаю конструкцию и опубликую результаты. Кое-какие наработки уже имеются.