Friday, December 16, 2016

Measuring Wind Speed with an Anemometer and Arduino

In this video we look at how to measure wind speed using an anemometer and Arduino. This approach will work on both ARM and AVR based Arduinos.


//*****************Arduino anemometer sketch******************************
const byte interruptPin = 3; //anemomter input to digital pin
volatile unsigned long sTime = 0; //stores start time for wind speed calculation
unsigned long dataTimer = 0; //used to track how often to communicate data
volatile float pulseTime = 0; //stores time between one anemomter relay closing and the next
volatile float culPulseTime = 0; //stores cumulative pulsetimes for averaging
volatile bool start = true; //tracks when a new anemometer measurement starts
volatile unsigned int avgWindCount = 0; //stores anemometer relay counts for doing average wind speed
float aSetting = 60.0; //wind speed setting to signal alarm

void setup() {
  pinMode(13, OUTPUT); //setup LED pin to signal high wind alarm condition
  pinMode(interruptPin, INPUT_PULLUP); //set interrupt pin to input pullup
  attachInterrupt(interruptPin, anemometerISR, RISING); //setup interrupt on anemometer input pin, interrupt will occur whenever falling edge is detected
  dataTimer = millis(); //reset loop timer
}

void loop() {
 
  unsigned long rTime = millis();
  if((rTime - sTime) > 2500) pulseTime = 0; //if the wind speed has dropped below 1MPH than set it to zero
     
  if((rTime - dataTimer) > 1800){ //See if it is time to transmit
   
    detachInterrupt(interruptPin); //shut off wind speed measurement interrupt until done communication
    float aWSpeed = getAvgWindSpeed(culPulseTime,avgWindCount); //calculate average wind speed
    if(aWSpeed >= aSetting) digitalWrite(13, HIGH);   // high speed wind detected so turn the LED on
    else digitalWrite(13, LOW);   //no alarm so ensure LED is off
    culPulseTime = 0; //reset cumulative pulse counter
    avgWindCount = 0; //reset average wind count

    float aFreq = 0; //set to zero initially
    if(pulseTime > 0.0) aFreq = getAnemometerFreq(pulseTime); //calculate frequency in Hz of anemometer, only if pulsetime is non-zero
    float wSpeedMPH = getWindMPH(aFreq); //calculate wind speed in MPH, note that the 2.5 comes from anemometer data sheet
   
    Serial.begin(57600); //start serial monitor to communicate wind data
    Serial.println();
    Serial.println("...................................");
    Serial.print("Anemometer speed in Hz ");
    Serial.println(aFreq);
    Serial.print("Current wind speed is ");
    Serial.println(wSpeedMPH);
    Serial.print("Current average wind speed is ");
    Serial.println(aWSpeed);
    Serial.end(); //serial uses interrupts so we want to turn it off before we turn the wind measurement interrupts back on
   
    start = true; //reset start variable in case we missed wind data while communicating current data out
    attachInterrupt(digitalPinToInterrupt(interruptPin), anemometerISR, RISING); //turn interrupt back on
    dataTimer = millis(); //reset loop timer
  }
}

//using time between anemometer pulses calculate frequency of anemometer
float getAnemometerFreq(float pTime) { return (1/pTime); }
//Use anemometer frequency to calculate wind speed in MPH, note 2.5 comes from anemometer data sheet
float getWindMPH(float freq) { return (freq*2.5); }
//uses wind MPH value to calculate KPH
float getWindKPH(float wMPH) { return (wMPH*1.61); }
//Calculates average wind speed over given time period
float getAvgWindSpeed(float cPulse,int per) {
  if(per) return getWindMPH(getAnemometerFreq((float)(cPulse/per)));
  else return 0; //average wind speed is zero and we can't divide by zero
  }

//This is the interrupt service routine (ISR) for the anemometer input pin
//it is called whenever a falling edge is detected
void anemometerISR() {
  unsigned long cTime = millis(); //get current time
  if(!start) { //This is not the first pulse and we are not at 0 MPH so calculate time between pulses
   // test = cTime - sTime;
    pulseTime = (float)(cTime - sTime)/1000;
    culPulseTime += pulseTime; //add up pulse time measurements for averaging
    avgWindCount++; //anemomter went around so record for calculating average wind speed
  }
  sTime = cTime; //store current time for next pulse time calculation
  start = false; //we have our starting point for a wind speed measurement
}

13 comments:

  1. Hi Force Tronics, I tried to use this code on an ESP8266, Wemos D1 R2 but it didn't work. Do you have a version of this application for this type o MC? Thanks!

    ReplyDelete
  2. Thank you! I used your code in my program...

    ReplyDelete
  3. Fantastic video to see wind speed on Arduino. I also have a vane anemometer for wind direction. It is on the same mast as the wind speed device. Do you have info or a video for wind direction?

    ReplyDelete
  4. This comment has been removed by the author.

    ReplyDelete
  5. Do you have the code for the digital display on a LED of that arduino's output?

    ReplyDelete
  6. Hello, I wonder how to increase the average calculation time window..
    My anemometer uses 14 pulses per rotation so , variable wind speeds doesnt show an average very healthy..
    I am asking how can we get the last 10 secs for average for example?

    Regards

    Meanwhile, I edited the code for additional one-wire temperature sensor , LCD and web output. If interested I can share

    ReplyDelete
    Replies
    1. Yes I would like to have the code for the lcd display if you wouldnt care..
      Tnx

      Delete
  7. Nice effort. But how come it shows 0.0Hz ? Eventhough anemometer is still spinning?

    ReplyDelete
  8. This comment has been removed by the author.

    ReplyDelete
  9. The above Video and caluclation gives a perfect idea about anemometer. It gives Air Velocity and either Relative Humidity, Dew Point, Temperature or Windchill. I think MINI THERMO-ANEMOMETER will help to calcute this .

    ReplyDelete
  10. Would it be okay if I change the Vcc to 5v? Are there any changes to be made to the code if the Vcc is 5v?

    ReplyDelete
  11. Hey Admin! Thanks for sharing this informative & excellent post with us . I really like what you've posted here and wish you the best of luck with this blog and thanks again for sharing. Keep posting regularly! I would like to suggest you the best Hardware Tools suppliers in Dubai, UAE

    ReplyDelete
  12. Your blog is nice, Thank you for sharing this valuable content

    anemometer singapore

    ReplyDelete