2025 Autor: John Day | [email protected]. Ostatnio zmodyfikowany: 2025-01-13 06:58
Przegląd
Do zbudowania tego urządzenia zainspirowało mnie zadanie domowe na kursie on-line Digital Signal Processing. Jest to dekoder DTMF zaimplementowany w Arduino UNO, wykrywa cyfrę naciśniętą na klawiaturze telefonu w trybie tonowym na podstawie wytwarzanego dźwięku.
Krok 1: Zrozumienie algorytmu
W DTMF każdy symbol kodowany jest dwiema częstotliwościami zgodnie z tabelą na rysunku.
Urządzenie przechwytuje sygnał wejściowy z mikrofonu i oblicza amplitudy ośmiu częstotliwości. Dwie częstotliwości o maksymalnych amplitudach dają rząd i kolumnę zakodowanego symbolu.
Pozyskiwanie danych
W celu przeprowadzenia analizy widma próbki powinny być zbierane z określoną przewidywalną częstotliwością. Aby to osiągnąć wykorzystałem tryb free-run ADC z maksymalną precyzją (preskaler 128) dający częstotliwość próbkowania 9615Hz. Poniższy kod pokazuje, jak skonfigurować ADC Arduino.
nieważne initADC() {
// Rozpocznij ADC; f = (16 MHz/preskaler) / 13 cykli/konwersja ADMUX = 0; // Wybór kanału, prawy-adj, użyj pinu AREF ADCSRA = _BV(ADEN) | // ADC włącz _BV(ADSC) | // ADC start _BV(ADATE) | // Automatyczne wyzwalanie _BV(ADIE) | // Przerwanie włączone _BV(ADPS2) | _BV(ADPS1) | _BV(ADPS0); // 128:1 / 13 = 9615 Hz ADCSRB = 0; // Tryb swobodnej pracy DIDR0 = _BV(0); // Wyłącz wejście cyfrowe dla ADC pin TIMSK0 = 0; // Timer0 off } Obsługa przerwań wygląda tak ISR(ADC_vect) { uint16_t sample = ADC;samples[samplePos++] = sample - 400; if(PrzykładPoz >= N) { ADCSRA &= ~_BV(ADIE); // Bufor pełny, przerwanie } }
Analiza widma
Po zebraniu próbek obliczam amplitudy symboli kodowania 8 częstotliwości. Nie muszę do tego uruchamiać pełnego FFT, więc użyłem algorytmu Goertzela.
void goertzel(uint8_t *samples, float *spectrum) {
pływak v_0, v_1, v_2; pływać re, im, amp; for (uint8_t k = 0; k < IX_LEN; k++) { float c = pgm_read_float(&(cos_t[k])); float s = pgm_read_float(&(sin_t[k])); pływak a = 2. * c; v_0 = v_1 = v_2 = 0; dla (uint16_t i = 0; i < N; i++) { v_0 = v_1; v_1 = v_2; v_2 = (zmiennoprzecinkowa)(próbki) + a * v_1 - v_0; } re = c * v_2 - v_1; im = s * v_2; amp = sqrt(re * re + im * im); widmo[k] = amp; } }
Krok 2: Kodeks
Powyższy rysunek pokazuje przykład kodowania cyfry 3, gdzie maksymalna amplituda odpowiada częstotliwościom 697Hz i 1477Hz.
Cały szkic wygląda następująco
/** * Połączenia: * [Mic to Arduino] * - Out -> A0 * - Vcc -> 3.3V * - Gnd -> Gnd * - Arduino: AREF -> 3.3V * [Display to Arduino] * - Vcc - > 5V * - Gnd -> Gnd * - DIN -> D11 * - CLK -> D13 * - CS -> D9 */ #zawiera #zawiera
#włączać
#zdefiniuj CS_PIN 9
#zdefiniuj N 256
#define IX_LEN 8 #define PRÓG 20
Sterownik LEDMatrix lmd(1, CS_PIN);
próbki uint8_t[N];
lotny uint16_t samplePos = 0;
widmo zmiennoprzecinkowe[IX_LEN];
// Częstotliwości [697.0, 770.0, 852.0, 941.0, 1209.0, 1336,0, 1477,0, 1633.0]
// Obliczono dla 9615Hz 256 próbek const float cos_t[IX_LEN] PROGMEM = { 0.8932243011955153, 0.8700869911087115, 0.8448535652497071, 0.8032075314806449, 0.6895405447370669, 0.6343932841636456, 0.555.05709930130823 } const float sin_t[IX_LEN] PROGMEM = { 0.44961132965460654, 0.49289819222978404, 0.5349976198870972, 0.5956993044924334, 0.7242470829514669, 0.773010453627369, 0.8314696123025451, 0.8819212643483549
struktura typedef {
cyfra znaku; indeks uint8_t; } cyfra_t;
digit_t wykryta_cyfra;
const char table[4][4] PROGMEM = {
{'1', '2', '3', 'A'}, {'4', '5', '6', 'B'}, {'7', '8', '9', ' C'}, {'*', '0', '#', 'D'} };
const uint8_t char_indexes[4][4] PROGMEM = {
{1, 2, 3, 10}, {4, 5, 6, 11}, {7, 8, 9, 12}, {15, 0, 14, 13} };
czcionka bajtowa[16][8] = {
{0x00, 0x38, 0x44, 0x4c, 0x54, 0x64, 0x44, 0x38}, // 0 {0x04, 0x0c, 0x14, 0x24, 0x04, 0x04, 0x04, 0x04}, // 1 {0x00, 0x30, 0x48, 0x04, 0x04, 0x38, 0x40, 0x7c}, // 2 {0x00, 0x38, 0x04, 0x04, 0x18, 0x04, 0x44, 0x38}, // 3 {0x00, 0x04, 0x0c, 0x14, 0x24, 0x7e, 0x04, 0x04 }, // 4 {0x00, 0x7c, 0x40, 0x40, 0x78, 0x04, 0x04, 0x38}, // 5 {0x00, 0x38, 0x40, 0x40, 0x78, 0x44, 0x44, 0x38}, // 6 {0x00, 0x7c, 0x04, 0x04, 0x08, 0x08, 0x10, 0x10}, // 7 {0x00, 0x3c, 0x44, 0x44, 0x38, 0x44, 0x44, 0x78}, // 8 {0x00, 0x38, 0x44, 0x44, 0x3c, 0x04, 0x04, 0x78}, // 9 {0x00, 0x1c, 0x22, 0x42, 0x42, 0x7e, 0x42, 0x42}, // A {0x00, 0x78, 0x44, 0x44, 0x78, 0x44, 0x44, 0x7c}, / / B {0x00, 0x3c, 0x44, 0x40, 0x40, 0x40, 0x44, 0x7c}, // C {0x00, 0x7c, 0x42, 0x42, 0x42, 0x42, 0x44, 0x78}, // D {0x00, 0x0a, 0x7f, 0x14, 0x28, 0xfe, 0x50, 0x00}, // # {0x00, 0x10, 0x54, 0x38, 0x10, 0x38, 0x54, 0x10} // * };
nieważne initADC() {
// Rozpocznij ADC; f = (16 MHz/preskaler) / 13 cykli/konwersja ADMUX = 0; // Wybór kanału, prawy-adj, użyj pinu AREF ADCSRA = _BV(ADEN) | // ADC włącz _BV(ADSC) | // ADC start _BV(ADATE) | // Automatyczne wyzwalanie _BV(ADIE) | // Przerwanie włączone _BV(ADPS2) | _BV(ADPS1) | _BV(ADPS0); // 128:1 / 13 = 9615 Hz ADCSRB = 0; // Tryb swobodnej pracy DIDR0 = _BV(0); // Wyłącz wejście cyfrowe dla ADC pin TIMSK0 = 0; // Timer0 wyłączony }
void goertzel(uint8_t *samples, float *spectrum) {
pływak v_0, v_1, v_2; pływać re, im, amp; for (uint8_t k = 0; k < IX_LEN; k++) { float c = pgm_read_float(&(cos_t[k])); float s = pgm_read_float(&(sin_t[k])); pływak a = 2. * c; v_0 = v_1 = v_2 = 0; dla (uint16_t i = 0; i < N; i++) { v_0 = v_1; v_1 = v_2; v_2 = (zmiennoprzecinkowa)(próbki) + a * v_1 - v_0; } re = c * v_2 - v_1; im = s * v_2; amp = sqrt(re * re + im * im); widmo[k] = amp; } }
float avg(float *a, uint16_t len) {
wynik zmiennoprzecinkowy = 0,0; for (uint16_t i = 0; i < len; i++) { wynik += a; } zwraca wynik / len; }
int8_t get_single_index_above_threshold(float *a, uint16_t len, float próg) {
if (próg < PRÓG) { return -1; } int8_t ix = -1; for (uint16_t i = 0; i próg) { if (ix == -1) { ix = i; } else { return -1; } } } return ix; }
void detect_digit(zmiennoprzecinkowa *widmo) {
float avg_row = avg(widmo, 4); float śr_kol = śr(&widmo[4], 4); int8_t wiersz = get_single_index_above_threshold(widmo, 4, avg_row); int8_t col = get_single_index_above_threshold(&spectrum[4], 4, avg_col); if (wiersz != -1 && col != -1 && avg_col > 200) { wykryta_cyfra.cyfra = pgm_read_byte(&(tabela[wiersz][kol])); wykryty_cyfra.index = pgm_read_byte(&(char_indexes[wiersz][kol])); } else { wykryta_cyfra.cyfra = 0; } }
void drawSprite(bajt* sprite) {
// Maska służy do pobrania bitu kolumny z maski bajtowej wiersza sprite = B10000000; for(int iy = 0; iy < 8; iy++) { for(int ix = 0; ix < 8; ix++) { lmd.setPixel(7 - iy, ix, (bool)(sprite[iy] i maska));
// przesuń maskę o jeden piksel w prawo
maska = maska >> 1; }
// zresetuj maskę kolumny
maska = B10000000; } }
pusta konfiguracja () {
cli(); initADC(); sei();
Serial.początek(115200);
lmd.setEnabled(prawda); lmd.setIntensity(2); lmd.wyczyść(); lmd.wyświetlacz();
wykryta_cyfra.cyfra = 0;
}
bez znaku długie z = 0;
pusta pętla () {
while(ADCSRA i _BV(ADIE)); // Czekaj na zakończenie próbkowania dźwięku goertzel(samples, spectrum); wykryj_cyfra(widmo);
if (wykryta_cyfra.cyfra != 0) {
drawSprite(font[wykryta_cyfra.indeks]); lmd.wyświetlacz(); } if (z % 5 == 0) { for (int i = 0; i < IX_LEN; i++) { Serial.print(spectrum); Serial.print("\t"); } Serial.println(); Serial.println((int)detected_digit.digit); } z++;
próbkaPos = 0;
ADCSRA |= _BV(ADIE); // Wznów przerwanie próbkowania
}
ISR(ADC_wektor) {
próbka uint16_t = ADC;
próbki[przykładPoz++] = próbka - 400;
if(PrzykładPoz >= N) { ADCSRA &= ~_BV(ADIE); // Bufor pełny, przerwanie } }
Krok 3: Schematy
Należy wykonać następujące połączenia:
Mikrofon do Arduino
Out -> A0
Vcc -> 3.3V Gnd -> Gnd
Ważne jest, aby podłączyć AREF do 3,3V
Wyświetlacz do Arduino
Vcc -> 5V
Masa -> Masa DIN -> D11 CLK -> D13 CS -> D9
Krok 4: Wniosek
Co można tutaj poprawić? Użyłem N=256 próbek przy częstotliwości 9615Hz, która ma pewien wyciek widma, jeśli N=205 i częstotliwość 8000Hz to żądane częstotliwości pokrywają się z siatką dyskretyzacji. W tym celu ADC powinien być używany w trybie przepełnienia timera.