need help in read rpm of the fan using external interrupt in a given program by arduino uno

Status
Not open for further replies.

mobinmk

Full Member level 2
Joined
May 27, 2010
Messages
140
Helped
4
Reputation
8
Reaction score
4
Trophy points
1,298
Location
Kollam (Quilon),kerala, India
Visit site
Activity points
2,484
need help in read rpm of fan using external interrupt in given program by arduino uno

Hi Frndz,

Am new bie here.

i wana to make VFD of 3 phase induction motor.
succesfully i generate 3 pwm pulses by DDS technique used in aurdino uno.

the program is given below

Code C - [expand]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
/****************************************************************** 
 * DDS-sinewave - 3phase 
  *   
   ******************************************************************/ 
 
 
#include "avr/pgmspace.h" 
#include "avr/io.h" 
PROGMEM  prog_uchar sine256[] = 
{
  127,130,133,136,139,143,146,149,152,155,158,161,164,167,170,173,176,178,181,184,187,190,192,195,198,200,203,205,208,210,
 212,215,217,219,221,223,225,227,229,231,233,234,236,238,239,240,242,243,244,245,247,248,249,249,250,251,252,252,253,
253,253,254,254,254,254,254,254,254,253,253,253,252,252,251,250,249,249,248,247,245,244,243,242,240,239,238,236,234,
233,231,229,227,225,223,221,219,217,215,212,210,208,205,203,200,198,195,192,190,187,184,181,178,176,173,170,167,164,
161,158,155,152,149,146,143,139,136,133,130,127,124,121,118,115,111,108,105,102,99,96,93,90,87,84,81,78,76,73,70,67,64,
62,59,56,54,51,49,46,44,42,39,37,35,33,31,29,27,25,23,21,20,18,16,15,14,12,11,10,9,7,6,5,5,4,3,2,2,1,1,1,0,0,0,0,0,0,0,1,1,1,2,
2,3,4,5,5,6,7,9,10,11,12,14,15,16,18,20,21,23,25,27,29,31,33,35,37,39,42,44,46,49,51,54,56,59,62,64,67,70,73,76,78,81,84,87,
90,93,96,99,102,105,108,111,115,118,121,124
 
};
 
 
#define cbi(sfr, bit) (_SFR_BYTE(sfr) &= ~_BV(bit))
#define sbi(sfr, bit) (_SFR_BYTE(sfr) |= _BV(bit)) 
#define PWM_OUT_1 11  // PWM output on pin 11 
#define PWM_OUT_2 10  // PWM output on pin 10 
#define PWM_OUT_3 9  // PWM output on pin 9 
#define LED_PIN  13  // LED status on pin 13 
#define TEST_PIN 7  // Scope trigger on pin 7 
#define POTEN_IN 0  // Potentiometer on pin 0 
#define OFFSET_1 85  // Offset for second-phase
#define OFFSET_2 170  // Offset for third-phase  
 
      double dfreq; 
      const double refclk = 31376.6;  // measured 
      const uint64_t twoTo32 = pow(2, 32); // compute value at startup and use as constant 
      
      
      // variables used inside interrupt service declared as voilatile 
      volatile uint8_t icnt;   // var inside interrupt 
      volatile uint8_t icnt1;   // var inside interrupt 
      volatile uint8_t c4ms;   // counter incremented every 4ms 
      volatile uint32_t phase_accum;  // pahse accumulator 
      volatile uint32_t tword_m;  // dds tuning word m 
      
      
      
   void setup()
   {  
     pinMode(LED_PIN, OUTPUT); 
     // sets the digital pin as output  
     Serial.begin(115200);   // connect to the serial port   
     Serial.println("DDS Test");    
     
      pinMode(TEST_PIN, OUTPUT);         // sets the digital pin as output  
      pinMode(PWM_OUT_1, OUTPUT);       // PWM output / frequency output   
      pinMode(PWM_OUT_2, OUTPUT);       // PWM output / frequency output   
      pinMode(PWM_OUT_3, OUTPUT);       // PWM output / frequency output 
      
     // Setup the timers   
     setup_timer1();   
     setup_timer2(); 
     
       // disable interrupts to avoid timing distortion   
       cbi (TIMSK0, TOIE0);             // disable Timer0 !!! delay() is now not available   
       sbi (TIMSK2, TOIE2);             // enable Timer2 Interrupt  
    dfreq = 1000.0;                // initial output frequency = 1000.0 Hz   
    tword_m = twoTo32 * dfreq / refclk;       // calulate DDS new tuning word  
  } 
  
  
  void loop() 
  {   
    if (c4ms > 250)                // timer / wait for a full second   
    {     
      c4ms = 0;    
      dfreq = analogRead(POTEN_IN);        // read Poti on analog pin 0 to adjust output frequency from 0..1023 Hz  
      cbi (TIMSK2, TOIE2);            // disble Timer2 Interrupt    
      tword_m = twoTo32 * dfreq / refclk;       // calulate DDS new tuning word     
      sbi (TIMSK2, TOIE2);            // enable Timer2 Interrupt   
      Serial.print(dfreq);     
      Serial.print("  ");     
      Serial.println(tword_m);  
    } 
  } 
  
  
   // timer1 setup 
 // set prscaler to 1, PWM mode to phase correct PWM,  16000000/512 = 31.25kHz clock  
    void setup_timer1(void) 
    {   
      // Timer1 Clock Prescaler to : 1   
      sbi (TCCR1B, CS10);   
      cbi (TCCR1B, CS11);   
      cbi (TCCR1B, CS12);  
      // Timer0 PWM Mode set to Phase Correct PWM   
      cbi (TCCR1A, COM1A0);           // clear Compare Match   
      sbi (TCCR1A, COM1A1);   
      cbi (TCCR1A, COM1B0);           // clear Compare Match   
      sbi (TCCR1A, COM1B1);  
      
      sbi (TCCR1A, WGM10);            // Mode 1  / Phase Correct PWM   
      cbi (TCCR1A, WGM11);   
      cbi (TCCR1B, WGM12);   
      cbi (TCCR1B, WGM13); 
    }  
    
    
    // timer2 setup
    // set prscaler to 1, PWM mode to phase correct PWM,  16000000/512 = 31.25kHz clock 
    void setup_timer2(void) 
    {   
      // Timer2 Clock Prescaler to : 1   
      sbi (TCCR2B, CS20);   
      cbi (TCCR2B, CS21);   
      cbi (TCCR2B, CS22);  
  // Timer2 PWM Mode set to Phase Correct PWM   
    cbi (TCCR2A, COM2A0);           // clear Compare Match   
    sbi (TCCR2A, COM2A1);   
    sbi (TCCR2A, WGM20);            // Mode 1  / Phase Correct PWM   
    cbi (TCCR2A, WGM21);   
    cbi (TCCR2B, WGM22); 
    
    }   
    
//****************************************************************** 
// Timer2 Interrupt Service at 31.25kHz = 32us 
// this is the timebase REFCLOCK for the DDS generator 
// FOUT = (M (REFCLK)) / (2 exp 32) 
// runtime : 8 microseconds ( inclusive push and pop) 
ISR(TIMER2_OVF_vect) 
{   
  sbi(PORTD, TEST_PIN);           // Test / set PORTD,TEST_PIN high to observe timing with a oscope  
  phase_accum += tword_m;          // soft DDS, phase accu with 32 bits   
  icnt = phase_accum >> 24;          // use upper 8 bits for phase accu as frequency information  
  OCR2A = pgm_read_byte_near(sine256 + icnt);  // read value fron ROM sine table and send to PWM DAC  
  OCR1A = pgm_read_byte_near(sine256 + (uint8_t)(icnt + OFFSET_1));  
  OCR1B = pgm_read_byte_near(sine256 + (uint8_t)(icnt + OFFSET_2));  
  if (icnt1++ == 125)             // increment variable c4ms every 4 milliseconds   
      {     
        c4ms++;     
      icnt1 = 0;   
      }   
  cbi(PORTD, TEST_PIN);           // reset PORTD,TEST_PIN 
}



nw i wana to read rpm of motor using hall sensor connected to pin 2 or 3, external interrupt.
also i wana to display the rotor speed in lcd display
pls help me to add prgrm in above sketch

regards
mobin
 
Last edited by a moderator:

Re: need help in read rpm of fan using external interrupt in given program by arduino

If your admisible error is shorter than a full revolution time, you need a polling (sampling) timer.

If the error could be greater than few RPMs, you could use an external interrupt to count the rotor revolution during a predefined time interval.
 

Re: need help in read rpm of fan using external interrupt in given program by arduino

thnks sir,

this is my final program

Code:
/****************************************************************** 
 * DDS-sinewave - 3phase 
  *   
   ******************************************************************/ 


#include "avr/pgmspace.h" 
#include "avr/io.h" 
PROGMEM  prog_uchar sine256[] = 
{
  127,130,133,136,139,143,146,149,152,155,158,161,164,167,170,173,176,178,181,184,187,190,192,195,198,200,203,205,208,210,
 212,215,217,219,221,223,225,227,229,231,233,234,236,238,239,240,242,243,244,245,247,248,249,249,250,251,252,252,253,
253,253,254,254,254,254,254,254,254,253,253,253,252,252,251,250,249,249,248,247,245,244,243,242,240,239,238,236,234,
233,231,229,227,225,223,221,219,217,215,212,210,208,205,203,200,198,195,192,190,187,184,181,178,176,173,170,167,164,
161,158,155,152,149,146,143,139,136,133,130,127,124,121,118,115,111,108,105,102,99,96,93,90,87,84,81,78,76,73,70,67,64,
62,59,56,54,51,49,46,44,42,39,37,35,33,31,29,27,25,23,21,20,18,16,15,14,12,11,10,9,7,6,5,5,4,3,2,2,1,1,1,0,0,0,0,0,0,0,1,1,1,2,
2,3,4,5,5,6,7,9,10,11,12,14,15,16,18,20,21,23,25,27,29,31,33,35,37,39,42,44,46,49,51,54,56,59,62,64,67,70,73,76,78,81,84,87,
90,93,96,99,102,105,108,111,115,118,121,124

};


#define cbi(sfr, bit) (_SFR_BYTE(sfr) &= ~_BV(bit))
#define sbi(sfr, bit) (_SFR_BYTE(sfr) |= _BV(bit)) 
#define PWM_OUT_1 11  // PWM output on pin 11 
#define PWM_OUT_2 10  // PWM output on pin 10 
#define PWM_OUT_3 9  // PWM output on pin 9 
#define LED_PIN  13  // LED status on pin 13 
#define TEST_PIN 7  // Scope trigger on pin 7 
#define POTEN_IN 0  // Potentiometer on pin 0 
#define OFFSET_1 85  // Offset for second-phase
#define OFFSET_2 170  // Offset for third-phase  

      double dfreq; 
      const double refclk = 31376.6;  // measured 
      const uint64_t twoTo32 = pow(2, 32); // compute value at startup and use as constant 
      
      
      // variables used inside interrupt service declared as voilatile 
      volatile uint8_t icnt;   // var inside interrupt 
      volatile uint8_t icnt1;   // var inside interrupt 
      volatile uint8_t c4ms;   // counter incremented every 4ms 
      volatile uint32_t phase_accum;  // pahse accumulator 
      volatile uint32_t tword_m;  // dds tuning word m 
      volatile uint8_t iAtt;  // signal attenuation (percent): 0 .. 100
      volatile uint8_t iSine1; // sine table value (phase 1): 0 .. 255 
      volatile uint8_t iSine2; // sine table value (phase 2): 0 .. 255 
      volatile uint8_t iSine3; // sine table value (phase 3): 0 .. 255 
      
      
   void setup()
   {  
     pinMode(LED_PIN, OUTPUT); 
     // sets the digital pin as output  
     Serial.begin(115200);   // connect to the serial port   
     Serial.println("DDS Test");    
     
      pinMode(TEST_PIN, OUTPUT);         // sets the digital pin as output  
      pinMode(PWM_OUT_1, OUTPUT);       // PWM output / frequency output   
      pinMode(PWM_OUT_2, OUTPUT);       // PWM output / frequency output   
      pinMode(PWM_OUT_3, OUTPUT);       // PWM output / frequency output 
      
     // Setup the timers   
     setup_timer1();   
     setup_timer2(); 
     
       // disable interrupts to avoid timing distortion   
       cbi (TIMSK0, TOIE0);             // disable Timer0 !!! delay() is now not available   
       sbi (TIMSK2, TOIE2);             // enable Timer2 Interrupt  
    dfreq = 1000.0;                // initial output frequency = 1000.0 Hz   
    tword_m = twoTo32 * dfreq / refclk;       // calulate DDS new tuning word  
  } 
  
  
  void loop() 
  {   
    if (c4ms > 250)                // timer / wait for a full second   
    {     
      c4ms = 0;    
      dfreq = analogRead(POTEN_IN);        // read Poti on analog pin 0 to adjust output frequency from 0..1023 Hz  
      cbi (TIMSK2, TOIE2);            // disble Timer2 Interrupt    
      tword_m = twoTo32 * dfreq / refclk;       // calulate DDS new tuning word     
      iAtt = 50;
      sbi (TIMSK2, TOIE2);            // enable Timer2 Interrupt   
      Serial.print(dfreq);     
      Serial.print("  ");     
      Serial.println(tword_m);  
    } 
  } 
  
  
   // timer1 setup 
 // set prscaler to 1, PWM mode to phase correct PWM,  16000000/512 = 31.25kHz clock  
    void setup_timer1(void) 
    {   
      // Timer1 Clock Prescaler to : 1   
      sbi (TCCR1B, CS10);   
      cbi (TCCR1B, CS11);   
      cbi (TCCR1B, CS12);  
      // Timer0 PWM Mode set to Phase Correct PWM   
      cbi (TCCR1A, COM1A0);           // clear Compare Match   
      sbi (TCCR1A, COM1A1);   
      cbi (TCCR1A, COM1B0);           // clear Compare Match   
      sbi (TCCR1A, COM1B1);  
      
      sbi (TCCR1A, WGM10);            // Mode 1  / Phase Correct PWM   
      cbi (TCCR1A, WGM11);   
      cbi (TCCR1B, WGM12);   
      cbi (TCCR1B, WGM13); 
    }  
    
    
    // timer2 setup
    // set prscaler to 1, PWM mode to phase correct PWM,  16000000/512 = 31.25kHz clock 
    void setup_timer2(void) 
    {   
      // Timer2 Clock Prescaler to : 1   
      sbi (TCCR2B, CS20);   
      cbi (TCCR2B, CS21);   
      cbi (TCCR2B, CS22);  
  // Timer2 PWM Mode set to Phase Correct PWM   
    cbi (TCCR2A, COM2A0);           // clear Compare Match   
    sbi (TCCR2A, COM2A1);   
    sbi (TCCR2A, WGM20);            // Mode 1  / Phase Correct PWM   
    cbi (TCCR2A, WGM21);   
    cbi (TCCR2B, WGM22); 
    
    }   
    
//****************************************************************** 
// Timer2 Interrupt Service at 31.25kHz = 32us 
// this is the timebase REFCLOCK for the DDS generator 
// FOUT = (M (REFCLK)) / (2 exp 32) 
// runtime : 8 microseconds ( inclusive push and pop) 
ISR(TIMER2_OVF_vect) 
{   
  sbi(PORTD, TEST_PIN);           // Test / set PORTD,TEST_PIN high to observe timing with a oscope  
  phase_accum += tword_m;          // soft DDS, phase accu with 32 bits   
  icnt = phase_accum >> 24;          // use upper 8 bits for phase accu as frequency information  
  // Perform the attenuation:

  iSine1 = int(pgm_read_byte_near(sine256 + icnt));  // read value fron ROM sine table
  iSine1 = round(iSine1*iAtt/100); // get the attenuated sine value

  iSine2 = int(pgm_read_byte_near(sine256 + (uint8_t)(icnt + OFFSET_1)));
  iSine2 = round(iSine2*iAtt/100);

  iSine3 = int(pgm_read_byte_near(sine256 + (uint8_t)(icnt + OFFSET_2)));
  iSine3 = round(iSine3*iAtt/100);

// Now set the PWM DAC  	

  OCR2A = byte(iSine1);  
  OCR1A = byte(iSine2);  
  OCR1B = byte(iSine3);  
  
  
//********************************************
//********************************************
//********************************************
  if (icnt1++ == 125)             // increment variable c4ms every 4 milliseconds   
      {     
        c4ms++;     
      icnt1 = 0;   
      }   
  cbi(PORTD, TEST_PIN);           // reset PORTD,TEST_PIN 
}

i wana to measure the rpm of 3 phase induction motor using hall sensor ,by external interrupt in arduino uno

pls check the program to measure rpm

2 programs there.
1 is read rpm
2nd is read rpm and avg

i wana to add these code in my final program
Code:
// read RPM

volatile int rpmcount = 0;//see **broken link removed**
int rpm = 0;
unsigned long lastmillis = 0;

void setup(){
 Serial.begin(9600); 
 attachInterrupt(0, rpm_fan, FALLING);//interrupt cero (0) is on pin two(2).
}

void loop(){
 
 if (millis() - lastmillis == 1000){  /*Uptade every one second, this will be equal to reading frecuency (Hz).*/
 
 detachInterrupt(0);    //Disable interrupt when calculating
 
 
 rpm = rpmcount * 60;  /* Convert frecuency to RPM, note: this works for one interruption per full rotation. For two interrups per full rotation use rpmcount * 30.*/
 
 Serial.print("RPM =\t"); //print the word "RPM" and tab.
 Serial.print(rpm); // print the rpm value.
 Serial.print("\t Hz=\t"); //print the word "Hz".
 Serial.println(rpmcount); /*print revolutions per second or Hz. And print new line or enter.*/
 
 rpmcount = 0; // Restart the RPM counter
 lastmillis = millis(); // Uptade lasmillis
 attachInterrupt(0, rpm_fan, FALLING); //enable interrupt
  }
}


void rpm_fan(){ /* this code will be executed every time the interrupt 0 (pin2) gets low.*/
  rpmcount++;
}


// Elimelec Lopez - April 25th 2013

Code v2 (Calculate Average)
Download RPM_V2.ino

// read RPM and calculate average every then readings.
const int numreadings = 10;
int readings[numreadings];
unsigned long average = 0;
int index = 0;
unsigned long total; 

volatile int rpmcount = 0;//see **broken link removed** 
unsigned long rpm = 0;
unsigned long lastmillis = 0;

void setup(){
 Serial.begin(9600); 
 attachInterrupt(0, rpm_fan, FALLING);
}

void loop(){
 
  
 if (millis() - lastmillis >= 1000){  /*Uptade every one second, this will be equal to reading frecuency (Hz).*/
 
 detachInterrupt(0);    //Disable interrupt when calculating
 total = 0;  
 readings[index] = rpmcount * 60;  /* Convert frecuency to RPM, note: this works for one interruption per full rotation. For two interrups per full rotation use rpmcount * 30.*/
 
 for (int x=0; x<=9; x++){
   total = total + readings[x];
 }
 
 average = total / numreadings;
 rpm = average;
 
 rpmcount = 0; // Restart the RPM counter
 index++;
 if(index >= numreadings){
  index=0; 
 } 
 
 
if (millis() > 11000){  // wait for RPMs average to get stable

 Serial.print(" RPM = ");
 Serial.println(rpm);
}
 
 lastmillis = millis(); // Uptade lasmillis
  attachInterrupt(0, rpm_fan, FALLING); //enable interrupt
  }
}


void rpm_fan(){ /* this code will be executed every time the interrupt 0 (pin2) gets low.*/
  rpmcount++;
}
 
Last edited by a moderator:

Re: need help in read rpm of fan using external interrupt in given program by arduino

The second block of code is enough (it calculates both last read and the average RPM). Use the last read value for error correction and the average one for displaying on LCD.
 

Re: need help in read rpm of fan using external interrupt in given program by arduino

thnks sir,

i wana to implement pid controller .

how to designing the valuesof kp,ki,kd for different refernce speed??
 

Re: need help in read rpm of fan using external interrupt in given program by arduino

Why do you want to implement a speed controller? Induction motors have only a small, well predictable torque proportional slip. And in case of a fan, there will be effectively no load variations.

Controller parameters can be either empirically determined (e.g. using Ziegler-Nichols method) or estimated based on a control process model.
 

    V

    Points: 2
    Helpful Answer Positive Rating
Re: need help in read rpm of fan using external interrupt in given program by arduino

I don't know the Arduino library and if it can be easily used to tune a speed controller. Generally it's suggested to know the basic control process properties before attempting to autotune it.
 

Status
Not open for further replies.

Similar threads

Cookies are required to use this site. You must accept them to continue using the site. Learn more…