1

Тема: Анализатор спектра на матрице 8x8 MAX7219

Основная статья - http://rcl-radio.ru/?p=104916

http://forum.rcl-radio.ru/uploads/images/2021/08/34ff0611e67633d8327f067df2fad15f.png
http://forum.rcl-radio.ru/uploads/images/2021/08/09cc23c52bd645bbcf3fabeb786d52d3.png
http://forum.rcl-radio.ru/uploads/images/2021/08/04c8def3982275c21bf30bd84cd9b6c4.png
http://forum.rcl-radio.ru/uploads/images/2021/08/32dab712d533a8583010c3ff7b8537e9.png

////////// |PORT   |pin | Arduino 
// MAX7219
#define CLK PD2 // | 4  | D2
#define CS  PD3 // | 5  | D3
#define DIN PD4 // | 6  | D4

#define AUTO_GAIN 1        // автонастройка по громкости
#define LOW_PASS 40        // нижний порог чувствительности шумов (нет скачков при отсутствии звука)
#define DEF_GAIN 300       // максимальный порог по умолчанию 
#define FHT_N 256          // ширина спектра х2
#define LOG_OUT 1
#define INPUT_GAIN 1.0 
#define BR 0               // яркость

#include <FHT.h>  // http://forum.rcl-radio.ru/misc.php?action=pan_download&item=297&download=1
byte posOffset[32] = {4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,21,22,23,24,25,26,27,28,29,30,31,32,33,34};

int i1;
unsigned long data[32];
unsigned long data1,data2,data3,data4,data5,data6,data7,data8;
byte a[32];
int ur[32],urr[32];
unsigned long gainTimer;
int gain_sp = DEF_GAIN; 
int maxValue; 
int maxValue_f;
float k = 0.5; 


void setup(){
   ADMUX  = 0b11000000; ADCSRA = 0b11010110;
  DDRD |= (1 << DIN) | (1 << CS) | (1 << CLK);
  max7219_L(0x0F, 0);// тест выкл.
  max7219_L(0x0C, 0x01010101);// вкл. индик.
  max7219_L(0x0A, 0x01010101*BR);// яркость
  max7219_L(0x09, 0);// дешифраторы выкл.
  max7219_L(0x0B, 0x07070707);// кол-во разрядов 
  }

void loop(){

  analyzeAudio();

      for (int i = 0; i < FHT_N/2; i++) {                     
      if (fht_log_out[i] < LOW_PASS) fht_log_out[i] = 0;
      fht_log_out[i] = (float)fht_log_out[i] * INPUT_GAIN;}

       maxValue = 0;
      for (int pos = 0; pos < 32; pos++) {  
        if (fht_log_out[posOffset[pos]] > maxValue) maxValue = fht_log_out[posOffset[pos]];
        int posLevel = map(fht_log_out[posOffset[pos]], LOW_PASS, gain_sp, 0, 8);posLevel = constrain(posLevel, 0, 8);
        
        urr[pos] = posLevel;
        if(urr[pos]<ur[pos]){ur[pos]=ur[pos]-1;}
        else{ur[pos] = posLevel;}
        delayMicroseconds(10);
        

  switch(ur[pos]){
    case 0: a[pos]=0b00000000;break;
    case 1: a[pos]=0b00000001;break;
    case 2: a[pos]=0b00000011;break;
    case 3: a[pos]=0b00000111;break;
    case 4: a[pos]=0b00001111;break;
    case 5: a[pos]=0b00011111;break;
    case 6: a[pos]=0b00111111;break;
    case 7: a[pos]=0b01111111;break;
    case 8: a[pos]=0b11111111;break;
    }
     data1 = data1 +  ((uint32_t(a[pos] >> 0) & 1)<<pos);
     data2 = data2 +  ((uint32_t(a[pos] >> 1) & 1)<<pos);
     data3 = data3 +  ((uint32_t(a[pos] >> 2) & 1)<<pos);
     data4 = data4 +  ((uint32_t(a[pos] >> 3) & 1)<<pos);
     data5 = data5 +  ((uint32_t(a[pos] >> 4) & 1)<<pos);
     data6 = data6 +  ((uint32_t(a[pos] >> 5) & 1)<<pos);
     data7 = data7 +  ((uint32_t(a[pos] >> 6) & 1)<<pos);
     data8 = data8 +  ((uint32_t(a[pos] >> 7) & 1)<<pos);
      }
      
 max7219_L(8, data8 );
 max7219_L(7, data7  );
 max7219_L(6, data6  );
 max7219_L(5, data5  );
 max7219_L(4, data4  );
 max7219_L(3, data3  );
 max7219_L(2, data2  );
 max7219_L(1, data1  );
  data1=0;data2=0;data3=0;data4=0;data5=0;data6=0;data7=0;data8=0;
 

  if (AUTO_GAIN) {
    if (millis() - gainTimer > 10) {      
      maxValue_f = maxValue * k + maxValue_f * (1 - k);
      if (maxValue_f > LOW_PASS) gain_sp = maxValue_f;
       else gain_sp = DEF_GAIN;
       gainTimer = millis();}}
    }

void max7219_L(byte reg_n, unsigned long h){
  byte h1 = h >> 24;
  byte h2 = h >> 16;
  byte h3 = h >> 8;
  byte h4 = h;
  PORTD &=~(1 << CS);WriteBit16(reg_n,h1);WriteBit16(reg_n,h2);WriteBit16(reg_n,h3);WriteBit16(reg_n,h4);PORTD |=(1 << CS);
  }   

void WriteBit16(byte reg, byte data){  
     for(char i = 7; i >= 0; i--){
        PORTD &= ~(1 << CLK);
        if(((reg >> i) & 1) == 1){PORTD |= (1 << DIN);}else{PORTD &= ~(1 << DIN);}
        PORTD |=(1 << CLK);}
     for(char i = 7; i >= 0; i--){
        PORTD &= ~(1 << CLK);
        if(((data >> i) & 1) == 1){PORTD |= (1 << DIN);}else{PORTD &= ~(1 << DIN);}
        PORTD |=(1 << CLK);}
        PORTD &= ~(1 << CLK);PORTD |= (1 << DIN);
  } 

void analyzeAudio() { 
 while(i1 < FHT_N){i1++; 
    do{ADCSRA |= (1 << ADSC);} 
    while((ADCSRA & (1 << ADIF)) == 0);fht_input[i1] = (ADCL|ADCH << 8);}i1=0;
    fht_window(); // window the data for better frequency response
    fht_reorder(); // reorder the data before doing the fht
    fht_run(); // process the data in the fht
    fht_mag_log(); // take the output of the fht
}

2

Re: Анализатор спектра на матрице 8x8 MAX7219

3

Re: Анализатор спектра на матрице 8x8 MAX7219

Доброго дня. Возможно соединить два проекта в одной Ардуине Анализатор + часы?

4 (2025-02-14 11:24:53 отредактировано vanzan777)

Re: Анализатор спектра на матрице 8x8 MAX7219

Здраствуйте. Вот еще один допилинный проект. Он не мой. Но допилил до рабочего состояния.
Сылки прилагаю.

https://microkontroller.ru/arduino-proj … ino/?amp=1

https://microkontroller.ru/arduino-proj … 2x8/?amp=1

https://github.com/shajeebtm/Arduino-au … r-analyzer

https://github.com/kosme/arduinoFFT  тут исправления.

Видео работы

https://www.youtube.com/watch?v=GF_i0EnUEro

5 (2025-02-14 10:47:53 отредактировано vanzan777)

Re: Анализатор спектра на матрице 8x8 MAX7219

Код программы допиленный

/*
Copyright (c) 2019 Shajeeb TM

https://github.com/shajeebtm/Arduino-audio-spectrum-visualizer-analyzer/
https://create.arduino.cc/projecthub/Shajeeb/32-band-audio-spectrum-visualizer-analyzer-902f51

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
*/



#include "arduinoFFT.h"
#include <MD_MAX72xx.h>
#include <SPI.h>

#define SAMPLES 64            //Must be a power of 2
#define HARDWARE_TYPE MD_MAX72XX::FC16_HW   // Set display type  so that  MD_MAX72xx library treets it properly
#define MAX_DEVICES  4   // Total number display modules
#define CLK_PIN   13  // Clock pin to communicate with display
#define DATA_PIN  11  // Data pin to communicate with display
#define CS_PIN    10  // Control pin to communicate with display
#define  xres 32      // Total number of  columns in the display, must be <= SAMPLES/2
#define  yres 8       // Total number of  rows in the display


int MY_ARRAY[]={0, 128, 192, 224, 240, 248, 252, 254, 255}; // default = standard pattern
int MY_MODE_1[]={0, 128, 192, 224, 240, 248, 252, 254, 255}; // standard pattern
int MY_MODE_2[]={0, 128, 64, 32, 16, 8, 4, 2, 1}; // only peak pattern
int MY_MODE_3[]={0, 128, 192, 160, 144, 136, 132, 130, 129}; // only peak +  bottom point
int MY_MODE_4[]={0, 128, 192, 160, 208, 232, 244, 250, 253}; // one gap in the top , 3rd light onwards
int MY_MODE_5[]={0, 1, 3, 7, 15, 31, 63, 127, 255}; // standard pattern, mirrored vertically

 
double vReal[SAMPLES];
double vImag[SAMPLES];
char data_avgs[xres];

int yvalue;
int displaycolumn , displayvalue;
int peaks[xres];
const int buttonPin = 5;    // the number of the pushbutton pin
int state = HIGH;             // the current reading from the input pin
int previousState = LOW;   // the previous reading from the input pin
int displaymode = 1;
unsigned long lastDebounceTime = 0;  // the last time the output pin was toggled
unsigned long debounceDelay = 50;    // the debounce time; increase if the output flickers


MD_MAX72XX mx = MD_MAX72XX(HARDWARE_TYPE, CS_PIN, MAX_DEVICES);   // display object

const uint16_t samples = 64;
const double sampling = 40;
const uint8_t amplitude = 4;
const double startFrequency = 2;
const double stopFrequency = 16.4;
const double step_size = 0.1;

/*
These are the input and output vectors
Input vectors receive computed results from FFT
*/


/* Create FFT object */
ArduinoFFT<double> FFT = ArduinoFFT<double>(vReal, vImag, samples, sampling);

unsigned long startTime;

#define SCL_INDEX 0x00
#define SCL_TIME 0x01
#define SCL_FREQUENCY 0x02
#define SCL_PLOT 0x03 


void setup() {
    
    ADCSRA = 0b11100101;      // set ADC to free running mode and set pre-scalar to 32 (0xe5)
    ADMUX = 0b00000000;       // use pin A0 and external voltage reference
    pinMode(buttonPin, INPUT);
    mx.begin();           // initialize display
    delay(50);            // wait to get reference voltage stabilized
}
 
void loop() {
   // ++ Sampling
   for(int i=0; i<SAMPLES; i++)
    {
      while(!(ADCSRA & 0x10));        // wait for ADC to complete current conversion ie ADIF bit set
      ADCSRA = 0b11110101 ;               // clear ADIF bit so that ADC can do next operation (0xf5)
      int value = ADC - 512 ;                 // Read from ADC and subtract DC offset caused value
      vReal[i]= value/8;                      // Copy to bins after compressing
      vImag[i] = 0;                         
    }
    // -- Sampling

 
    // ++ FFT
    FFT.windowing(FFTWindow::Hamming, FFTDirection::Forward);	/* Weigh data */
    FFT.compute(FFTDirection::Forward); /* Compute FFT */
     FFT.complexToMagnitude(); /* Compute magnitudes */
    // -- FFT

    
    // ++ re-arrange FFT result to match with no. of columns on display ( xres )
    int step = (SAMPLES/2)/xres; 
    int c=0;
    for(int i=0; i<(SAMPLES/2); i+=step)  
    {
      data_avgs[c] = 0;
      for (int k=0 ; k< step ; k++) {
          data_avgs[c] = data_avgs[c] + vReal[i+k];
      }
      data_avgs[c] = data_avgs[c]/step; 
      c++;
    }
    // -- re-arrange FFT result to match with no. of columns on display ( xres )

    
    // ++ send to display according measured value 
    for(int i=0; i<xres; i++)
    {
      data_avgs[i] = constrain(data_avgs[i],0,80);            // set max & min values for buckets
      data_avgs[i] = map(data_avgs[i], 0, 80, 0, yres);        // remap averaged values to yres
      yvalue=data_avgs[i];

      peaks[i] = peaks[i]-1;    // decay by one light
      if (yvalue > peaks[i]) 
          peaks[i] = yvalue ;
      yvalue = peaks[i];    
      displayvalue=MY_ARRAY[yvalue];
      displaycolumn=31-i;
      mx.setColumn(displaycolumn, displayvalue);              // for left to right
     }
     // -- send to display according measured value 
     
    displayModeChange ();         // check if button pressed to change display mode
} 

void displayModeChange() {
  int reading = digitalRead(buttonPin);
  if (reading == HIGH && previousState == LOW && millis() - lastDebounceTime > debounceDelay) // works only when pressed
  
  {

   switch (displaymode) {
    case 1:    //       move from mode 1 to 2
      displaymode = 2;
      for (int i=0 ; i<=8 ; i++ ) {
        MY_ARRAY[i]=MY_MODE_2[i];
      }
      break;
    case 2:    //       move from mode 2 to 3
      displaymode = 3;
      for (int i=0 ; i<=8 ; i++ ) {
        MY_ARRAY[i]=MY_MODE_3[i];
      }
      break;
    case 3:    //     move from mode 3 to 4
      displaymode = 4;
      for (int i=0 ; i<=8 ; i++ ) {
        MY_ARRAY[i]=MY_MODE_4[i];
      }
      break;
    case 4:    //     move from mode 4 to 5
      displaymode = 5;
      for (int i=0 ; i<=8 ; i++ ) {
        MY_ARRAY[i]=MY_MODE_5[i];
      }
      break;
    case 5:    //      move from mode 5 to 1
      displaymode = 1;      
      for (int i=0 ; i<=8 ; i++ ) {
        MY_ARRAY[i]=MY_MODE_1[i];
      }
      break;
  }

    lastDebounceTime = millis();
  }
  previousState = reading;
}

Просидел пару ночей)

6 (2025-02-14 10:47:09 отредактировано vanzan777)

Re: Анализатор спектра на матрице 8x8 MAX7219

http://forum.rcl-radio.ru/uploads/images/2025/02/32a4aaf574d9874d87301af53944be54.png

Пять разных режима отображения.