Advertisement
AntonioVillanueva

Frecuencimetro

Feb 9th, 2025
140
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
  1. /*
  2. Antonio Villanueva Segura F4LEC
  3. Frecuencimetro Chino
  4. Utiliza la CPU Atmel48P y normalmentes el preescaler MB501
  5. con un ratio 1:64 , en mi caso el fabricante Chino habia utilizado
  6. el MB505 con un ratio 1:128 por lo que la frecuencia leida era la mitad
  7. es por eso que me decidi a escribir este codigo
  8. PC0(ADC0)23 ->RS
  9. PC1(ADC1)24 ->EN
  10. PC2(ADC2)25 ->D4
  11. PC3(ADC3)26 ->D6
  12. PC4(ADC4/SDA) 27 ->D7
  13. PC5(ADC5/SCL) 28 ->D5
  14. LCD, 2 lignes * 8, 0802A
  15.  
  16. Estos pines los utilizo para ajustar el error de medidas
  17. cuando los cortocircuitamos espera que se le introduzca la frecuencia
  18. de referencia en mi caso ajusto con estas frecuencias
  19.  
  20. PD1(3) -> JP1 7Mhz
  21. PD2(4) -> JP2 14Mhz
  22. PD3(5) -> JP3 28Mhz
  23.  
  24.  
  25. */
  26.  
  27. #include <EEPROM.h>
  28. #include <Arduino.h>
  29. #include <LiquidCrystal.h>
  30.  
  31. #define F_CPU 20000000UL  // Frecuencia del cristal: 20 MHz
  32. #define PRESCALER 1024
  33. #define OCR1A_VALUE 19531  // (F_CPU / PRESCALER) - 1
  34.  
  35. #define MB505_PIN PD5      // Pin donde llega la señal del MB505 (Pin 11)
  36. #define MB505_PORT PIND    // Puerto donde está el pin del MB505
  37. #define MB505_BIT  5       // Bit del pin del MB505 en el puerto
  38.  
  39. #define JP1
  40. #define JP2
  41. #define JP3
  42.  
  43. // LCD 802A paralelo  pins
  44. #define LCD_RS_PIN   A0  // PC0
  45. #define LCD_EN_PIN   A1  // PC1
  46. #define LCD_D4_PIN   A2  // PC2
  47. #define LCD_D5_PIN   A5  // PC5
  48. #define LCD_D6_PIN   A3  // PC3
  49. #define LCD_D7_PIN   A4  // PC4
  50.  
  51. // LiquidCrystal object
  52. LiquidCrystal lcd(LCD_RS_PIN, LCD_EN_PIN, LCD_D4_PIN, LCD_D5_PIN, LCD_D6_PIN, LCD_D7_PIN);
  53.  
  54. volatile uint32_t pulse_count = 0;
  55. volatile bool timer_flag = false;
  56. float frequency_MHz = 0.0; //Frecuencia Mhz
  57. float offset_freq =1.0 ;//Correccion de error de Medida
  58.  
  59. // Interrupción por cambio de estado en el pin PD5
  60. ISR(PCINT2_vect) {
  61.   if (PIND & (1 << PD5)) {
  62.     // Flanco ascendente
  63.     pulse_count++;
  64.   }
  65. }
  66.  
  67. // Interrupción del Timer1 Compare Match A
  68. ISR(TIMER1_COMPA_vect) {
  69.   timer_flag = true;
  70. }
  71.  
  72.  
  73. float correccion_frecuencia(float fref ,float freq){
  74.     /*factor_correccion = frecuencia_real / frecuencia_medida
  75.     frecuencia_compensada = frecuencia_medida * factor_correccion
  76.     factor_correccion =7.000000 /6.931456=1.009888831
  77.     frecuencia compensada =frecuencia medida *factor correccion=
  78.     */
  79.     return fref/freq;
  80. }
  81. /*
  82. // Función para escribir un float en la EEPROM
  83. void writeFloatToEEPROM(int address, float value) {
  84.   if (value <=0 || value>2.1){value =1.0;}
  85.   byte* p = (byte*)(void*)&value;
  86.   for (int i = 0; i < sizeof(float); i++) {
  87.     EEPROM.write(address + i, *p++);
  88.   }
  89. }
  90.  
  91. // Función para leer un float de la EEPROM
  92. float readFloatFromEEPROM(int address) {
  93.   float value;
  94.   byte* p = (byte*)(void*)&value;
  95.   for (int i = 0; i < sizeof(float); i++) {
  96.     *p++ = EEPROM.read(address + i);
  97.   }
  98.   if (value <=0 || value>2.1){return 1.0;}
  99.   return value;
  100. }
  101. */
  102. void setup() {
  103.  
  104.   // Configurar JP1(PD1),JP2(PD2),JP3(PD3)   como entradas
  105.   DDRD &= ~((1 << DDD1) | (1 << DDD2) | (1 << DDD3));
  106.   // Habilitar resistencias pull-up internas
  107.   PORTD |= ((1 << PORTD1) | (1 << PORTD2) | (1 << PORTD3));
  108.  
  109.   //Leer EPROM factor offset
  110.   //offset_freq=readFloatFromEEPROM(0 );
  111.  
  112.   // Configurar LCD
  113.   lcd.begin(8, 2);
  114.   lcd.clear();
  115.   lcd.setCursor(0,0);
  116.   lcd.print(" F4LEC");
  117.   delay(3000);
  118.   lcd.clear();
  119.  
  120.  
  121.   // Configurar PD5 (MB505_PIN) como entrada
  122.   DDRD &= ~(1 << MB505_PIN);
  123.  
  124.   // Habilitar interrupciones por cambio de estado en el grupo de pines donde está PD5
  125.   PCICR |= (1 << PCIE2);   // Habilitar PCIE2 (para PD0-PD7)
  126.   PCMSK2 |= (1 << PCINT5); // Habilitar interrupción para el pin PD5
  127.  
  128.   // Configurar Timer1
  129.   cli();
  130.   TCCR1A = 0;
  131.   TCCR1B = 0;
  132.   TCCR1B |= (1 << WGM12);
  133.   OCR1A = OCR1A_VALUE;
  134.   TCCR1B |= (1 << CS10) | (1 << CS12);
  135.   TIMSK1 |= (1 << OCIE1A);
  136.   sei();
  137. }
  138.  
  139. void loop() {
  140.  
  141.   // Leer JP1 ,JP2,JP3 para ajustar frecuencia
  142.   uint8_t jp1 = (PIND & (1 << PIND1)) >> PIND1;
  143.   //uint8_t jp2 = (PIND & (1 << PIND2)) >> PIND2;
  144.   //uint8_t jp3 = (PIND & (1 << PIND3)) >> PIND3;
  145.  
  146.  
  147.   if (timer_flag) {
  148.     timer_flag = false;
  149.  
  150.     // Calcular la frecuencia en MHz
  151.     frequency_MHz = (float)pulse_count * 128.0 / 1000000.0; // * 128 por el divisor del MB505, / 1000000 para MHz
  152.  
  153.     //Si el JP de correccion esta activado crea la correccion
  154.     if (jp1 == 0) { offset_freq = correccion_frecuencia(10.0 ,frequency_MHz);}
  155.  
  156.     if (jp1==0){
  157.      
  158.          lcd.clear();
  159.          lcd.setCursor(0, 0);
  160.          lcd.print("Adjust !");
  161.          delay(1000);
  162.        
  163.          //writeFloatToEEPROM(0,offset_freq );                
  164.      
  165.     }
  166.    
  167.     frequency_MHz *=offset_freq;//Correccion frecuencia
  168.          
  169.    
  170.     //frequency_MHz *=1.009888831;
  171.     // Mostrar la frecuencia en el LCD 802a
  172.     lcd.clear();
  173.     lcd.setCursor(0, 0);
  174.     lcd.print(frequency_MHz, 8); // Mostrar hasta 8 decimales
  175.     lcd.setCursor(0, 1);    
  176.     lcd.print("MHz");
  177.  
  178.     pulse_count = 0; // Reiniciar el contador de pulsos
  179.   }
  180. }
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement