/********************************************************************************* * * weather_station is a weatherstation build around the SparkFun weather meter * It can measure wind speed, wind gust , wind direction, rain fall, temperature, * humidity and air pressure and has an RS-485 ModBus interface for your convenience. * * LED on Arduino gives status: * * ON : Booting * BLINK : I2C ERROR * FLASH : Heartbeat * * Copyright (C) 2023-2025 M.T. Konstapel https://meezenest.nl/mees * * This file is part of weather_station * * weather_station is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * weather_station is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with weather_station. If not, see . * * 2023-01-21: - Buffer overflow when calculating average wind speed in AverageOfArray() * Fix: use 32 bit register for average_value. * - Changed some variables to the propper standard (uint8_t, uint16_t, etc.) * - SparkFun wind interrupt now calculates over 3 seconds in stead of 1 second (KNMI standard) * * 2024-05-02: - Removed code for si7021 * - Added code for HYT221 humidity sensor * * 2025-01-13: - Cleanup of code * - Debounce rainfall meter from 100ms to 250mms * - Debug messages can be switched on and off with the _DEBUG_ variable * - Main temperature and backup temperature registers switched: the sensor on the HYT221 heats up by * the circuitry and is therefore about 1.5 degrees above ambient temperature. The BMP280 does not heat up. * - Added SEN0562 luminosity sensor * * See CHANGELOG.md * **********************************************************************************/ // When set to 1 debug messages are send to the serial port, set it to 0 for production code! #define _DEBUG_ 0 #include #include "SparkFun_Weather_Meter_Kit_Arduino_Library.h" //I2C #include #include "i2c.h" //Temperature and humidity sensor #define HYT_ADDR 0x28 // I2C address of the HYT 221, 271, 371 and most likely the rest of the family // Light meter #define SEN0562_ADDR 0x23 //I2C address of Gravity SEN0562 light meter // Pressure sensor #include "i2c_BMP280.h" BMP280 bmp280; float PRESSURE_OFFSET = -100; // Calibration of BMP280: offset in Pascal /**************************/ /* Configurable variables */ /**************************/ // Sparkfun weather station int windDirectionPin = A0; int windSpeedPin = 2; int rainfallPin = 3; // RS485 driver #define RS485_RE 11 // Tight to RS485_DE and must be configured as an input to prevent a short circuit #define RS485_DE 12 // Used Pins const int TxenPin = RS485_DE; // -1 disables the feature, change that if you are using an RS485 driver, this pin would be connected to the DE and /RE pins of the driver. // ModBus address const byte SlaveId = 14; /* Modbus Registers Offsets (0-9999) * * 30000: Weater station ID (0x5758) * 30001: Wind direction (degrees) * 30002: Wind speed (average over 10 minutes in km/h) * 30003: Wind gust (peak wind speed in the last 10 minutes in km/h) * 30004: Temperature (degrees Celcius) * 30005: Rain last hour (l/m2) * 30006: Rain last 24 hours (l/m2) * 30007: Rain since midnight (l/m2) [NOT IMPLEMENTED, always 0] * 30008: Humidity (percent) * 30009: Barometric pressure (hPa) * 30010: Luminosity (W/m2) * 30011: Snow fall [NOT IMPLEMENTED, always 0] * 30012: Raw rainfall counter (mm) * 30013: Temperature pressure sensor (degrees Celsius) * 30014: Status bits 0=heater, 1-15: reserved * */ const int SensorIDIreg = 0; const int SensorWindDirectionIreg = 1; const int SensorWindSpeedIreg = 2; const int SensorWindGustIreg = 3; const int SensorTemperatureIreg = 4; const int SensorRainIreg = 5; const int SensorRainLast24Ireg = 6; const int SensorRainSinceMidnightIreg = 7; const int SensorHumidityIreg = 8; const int SensorPressureIreg = 9; const int SensorLuminosityIreg = 10; const int SensorSnowFallIreg = 11; const int SensorRainfallRawIreg = 12; const int SensorTemperatureBackupIreg = 13; const int SensorStatusBitsIreg = 14; /* Modbus Registers Offsets (0-9999) * Coils * 0 = Heater algorithm (0 = disable, 1 = enable) */ const int HeaterCoil = 0; // Legacy register, not used anymore. // RS-485 serial port #define MySerial Serial // define serial port used, Serial most of the time, or Serial1, Serial2 ... if available const unsigned long Baudrate = 9600; /******************************/ /* END Configurable variables */ /******************************/ // Create an instance of the weather meter kit SFEWeatherMeterKit weatherMeterKit(windDirectionPin, windSpeedPin, rainfallPin); // ModbusSerial object ModbusSerial mb (MySerial, SlaveId, TxenPin); unsigned long ts; unsigned long HourTimer; uint16_t WindGustData1[30]; uint8_t WindGustData1Counter=0; uint16_t WindGustData2[10]; uint8_t WindGustData2Counter=0; uint16_t WindAverageData1[30]; uint8_t WindAverageData1Counter=0; uint16_t WindAverageData2[10]; uint8_t WindAverageData2Counter=0; uint16_t RainPerHour[24]; uint8_t RainPerHourCounter=0; struct MeasuredData { uint16_t WindDirection; uint16_t WindSpeed; uint16_t WindGust; uint16_t Rain; uint16_t RainLast24; uint16_t SensorRainSinceMidnight; uint16_t Pressure; uint16_t StatusBits = 0; uint16_t RainfallCounter = 0; float TemperatureHygrometer; float Humidity; float TemperatureBarometer; float Luminosity; bool HeaterStatus = 0; } MeasuredData; void ReadHYT221 (void) { double humidity; double temperature; Wire.beginTransmission(HYT_ADDR); // Begin transmission with given device on I2C bus Wire.requestFrom(HYT_ADDR, 4); // Request 4 bytes // Read the bytes if they are available // The first two bytes are humidity the last two are temperature if(Wire.available() == 4) { int b1 = Wire.read(); int b2 = Wire.read(); int b3 = Wire.read(); int b4 = Wire.read(); Wire.endTransmission(); // End transmission and release I2C bus // combine humidity bytes and calculate humidity int rawHumidity = b1 << 8 | b2; // compound bitwise to get 14 bit measurement first two bits // are status/stall bit (see intro text) rawHumidity = (rawHumidity &= 0x3FFF); humidity = 100.0 / pow(2,14) * rawHumidity; // Scale for more decimal positions when converted to integer value for ModBus MeasuredData.Humidity = 100 * humidity; // combine temperature bytes and calculate temperature b4 = (b4 >> 2); // Mask away 2 least significant bits see HYT 221 doc int rawTemperature = b3 << 6 | b4; temperature = 165.0 / pow(2,14) * rawTemperature - 40; // Scale for more decimal positions when converted to integer value for ModBus MeasuredData.TemperatureHygrometer = 100 * temperature; if (_DEBUG_) { Serial.print("HYT221 humidity: "); Serial.print(MeasuredData.Humidity/100); Serial.print("% - Temperature: "); Serial.println(MeasuredData.TemperatureHygrometer/100); } } else { Serial.println("Not enough bytes available on wire."); } } uint8_t ReadSEN0562_register(uint8_t reg, const void* pBuf) { if (pBuf == NULL) { Serial.println("pBuf ERROR!! : null pointer"); } uint8_t * _pBuf = (uint8_t *)pBuf; Wire.beginTransmission(SEN0562_ADDR); Wire.write(®, 1); if ( Wire.endTransmission() != 0) { return 0; } delay(20); Wire.requestFrom(SEN0562_ADDR, 2); for (uint16_t i = 0; i < 2; i++) { _pBuf[i] = Wire.read(); } return 2; } void ReadSEN0562() { uint8_t buf[4] = {0}; uint16_t data, data1; ReadSEN0562_register(0x10, buf); //Register address 0x10 data = buf[0] << 8 | buf[1]; MeasuredData.Luminosity = (((float)data )/1.2)*100; if (_DEBUG_) { Serial.print("SEN0562 light intensity: "); Serial.print(MeasuredData.Luminosity/100); Serial.println(" Lux"); } } // Read BMP280 void ReadBMP280 (void) { bmp280.awaitMeasurement(); float pascal; bmp280.getPressure(pascal); pascal = (pascal - PRESSURE_OFFSET) / 10; // Convert to hPa MeasuredData.Pressure = pascal; bmp280.getTemperature(MeasuredData.TemperatureBarometer); // Scale for more decimal positions when converted to integer value for ModBus MeasuredData.TemperatureBarometer *= 100; bmp280.triggerMeasurement(); if (_DEBUG_) { Serial.print("BMP280 pressure: "); Serial.print(MeasuredData.Pressure); Serial.print("hPa - Temperature: "); Serial.println(MeasuredData.TemperatureBarometer/100); } } int MaxOfArray (int array[], uint16_t length) { int maximum_value = 0; while (length) { // decrement length, because 0/1 problem: if lenght = n, than last position off array is n-1 length--; if (array[length] > maximum_value) maximum_value = array[length]; } return maximum_value; } uint16_t AverageOfArray (uint16_t array[], uint16_t length) { uint32_t tmp_value = 0; uint8_t tmp_length = length; uint16_t average_value = 0; while (length) { // decrement length, because 0/1 problem: if lenght = n, than last position off array is n-1 length--; tmp_value += array[length]; } average_value = tmp_value/tmp_length; return average_value; } // Call this function every 2 seconds void ReadSparkfunWeatherStation (void) { unsigned char cnt=0; float tmpRegister; tmpRegister = 10*weatherMeterKit.getWindDirection(); // Use float for conversion to degrees times 10, than put it in integer register for ModBus MeasuredData.WindDirection = tmpRegister; tmpRegister = 100*(weatherMeterKit.getWindSpeed())/3.6; // Use float for conversion to m/s times 100, than put it in integer register for ModBus MeasuredData.WindSpeed = tmpRegister; tmpRegister = 100*weatherMeterKit.getTotalRainfall(); // Use float for conversion to l/m2 times 100, than put it in integer register for ModBus MeasuredData.Rain = tmpRegister; // FIFO for calculating wind gust of last 10 minutes // to preserve valuable RAM we cannot store all measurements of the last 10 minutes. // So we use a hack: store the last 30 values in a FIFO and every minute we store the maximum value from this FIFO in another FIFO. // This second FIFO is 10 deep: it stores the maximum values of the last 10 minutes. // The maximum value from this FIFO is the maximum wind gust of the last 10 minutes. if ( WindGustData1Counter < 29 ) { WindGustData1Counter++; } else { if ( WindGustData2Counter < 9 ) { WindGustData2Counter++; } else { WindGustData2Counter=0; } WindGustData2[WindGustData2Counter] = MaxOfArray(WindGustData1, 30); WindGustData1Counter=0; } WindGustData1[WindGustData1Counter] = MeasuredData.WindSpeed; MeasuredData.WindGust= MaxOfArray(WindGustData2, 10); // Smart FIFO, same as for Wind Gust, but now for average wind speed over 10 minutes if ( WindAverageData1Counter < 29 ) { WindAverageData1Counter++; } else { if ( WindAverageData2Counter < 9 ) { WindAverageData2Counter++; } else { WindAverageData2Counter=0; } WindAverageData2[WindAverageData2Counter] = AverageOfArray(WindAverageData1, 30); WindAverageData1Counter=0; WindAverageData1[WindAverageData1Counter] = MeasuredData.WindSpeed; } WindAverageData1[WindAverageData1Counter] = MeasuredData.WindSpeed; MeasuredData.WindSpeed = AverageOfArray(WindAverageData2, 10); // Record rainfall in one hour, save last 24 readings in FIFO if ( ( millis() - HourTimer) >= 3.6e+6) { HourTimer = millis(); if ( RainPerHourCounter < 23 ) { RainPerHourCounter++; } else { RainPerHourCounter=0; } RainPerHour[RainPerHourCounter] = MeasuredData.Rain; // Every time before we reset the TotalRainCounter we add the amount to the RawRainCounter. // This 16 bit register will eventually overflow, but 655.35mm of rain fall is a lot! MeasuredData.RainfallCounter += MeasuredData.Rain; // We don't care about the rounding error due to the convertion from float to int weatherMeterKit.resetTotalRainfall(); // Calculate rain fall in the last 24 hours MeasuredData.RainLast24=0; for (cnt=0; cnt<24;cnt++) { MeasuredData.RainLast24 += RainPerHour[cnt]; } } MeasuredData.Rain = RainPerHour[RainPerHourCounter]; } void setup() { MySerial.begin (Baudrate); // works on all boards but the configuration is 8N1 which is incompatible with the MODBUS standard // prefer the line below instead if possible // MySerial.begin (Baudrate, MB_PARITY_EVEN); // initialize digital pin LED_BUILTIN as an output and turn it on. pinMode(LED_BUILTIN, OUTPUT); digitalWrite(LED_BUILTIN, HIGH); //Setup control lines for RS485 driver pinMode(RS485_RE,INPUT); // In hardware connected to RS485_DE. Should be input to prevent a short circuit! pinMode(RS485_DE,OUTPUT); digitalWrite(RS485_DE,LOW); mb.config (Baudrate); mb.setAdditionalServerData ("TEMP_SENSOR"); // for Report Server ID function (0x11) // Add SensorIreg registers - Use addIreg() for analog Inputs mb.addIreg (SensorIDIreg); mb.addIreg (SensorWindDirectionIreg); mb.addIreg (SensorWindSpeedIreg); mb.addIreg (SensorWindGustIreg); mb.addIreg (SensorTemperatureIreg); mb.addIreg (SensorRainIreg); mb.addIreg (SensorRainLast24Ireg); mb.addIreg (SensorRainSinceMidnightIreg); mb.addIreg (SensorHumidityIreg); mb.addIreg (SensorPressureIreg); mb.addIreg (SensorLuminosityIreg); mb.addIreg (SensorSnowFallIreg); mb.addIreg (SensorRainfallRawIreg); mb.addIreg (SensorTemperatureBackupIreg); mb.addIreg (SensorStatusBitsIreg); // Add HeaterCoil register mb.addCoil (HeaterCoil); // Set Weather station ID mb.Ireg (SensorIDIreg, 0x5758); // Set unused register to zero mb.Ireg (SensorRainSinceMidnightIreg, 0); mb.Ireg (SensorSnowFallIreg, 0); Serial.println(F("Weather station v0.3.1")); Serial.println(F("(C)2024-2025 M.T. Konstapel")); Serial.println(F("This project is free and open source")); Serial.println(F("More details: https://meezenest.nl/mees/")); // Initialize BMP280 pressure sensor Serial.print(F("Pressure sensor BMP280 ")); if (bmp280.initialize()) Serial.println(F("found")); else { Serial.println(F("missing")); while(1) { digitalWrite(LED_BUILTIN, HIGH); // turn the LED on (HIGH is the voltage level) delay(500); // wait for half a second digitalWrite(LED_BUILTIN, LOW); // turn the LED off by making the voltage LOW delay(500); } } // onetime-measure: bmp280.setEnabled(0); bmp280.triggerMeasurement(); // Expected ADC values have been defined for various platforms in the // library, however your platform may not be included. This code will check // if that's the case #ifdef SFE_WMK_PLAFTORM_UNKNOWN // The platform you're using hasn't been added to the library, so the // expected ADC values have been calculated assuming a 10k pullup resistor // and a perfectly linear 16-bit ADC. Your ADC likely has a different // resolution, so you'll need to specify it here: weatherMeterKit.setADCResolutionBits(10); #endif // Here we create a struct to hold all the calibration parameters SFEWeatherMeterKitCalibrationParams calibrationParams = weatherMeterKit.getCalibrationParams(); // The wind vane has 8 switches, but 2 could close at the same time, which // results in 16 possible positions. Each position has a resistor connected // to GND, so this library assumes a voltage divider is created by adding // another resistor to VCC. Some of the wind vane resistor values are // fairly close to each other, meaning an accurate ADC is required. However // some ADCs have a non-linear behavior that causes this measurement to be // inaccurate. To account for this, the vane resistor values can be manually // changed here to compensate for the non-linear behavior of the ADC calibrationParams.vaneADCValues[WMK_ANGLE_0_0] = 943; calibrationParams.vaneADCValues[WMK_ANGLE_22_5] = 828; calibrationParams.vaneADCValues[WMK_ANGLE_45_0] = 885; calibrationParams.vaneADCValues[WMK_ANGLE_67_5] = 702; calibrationParams.vaneADCValues[WMK_ANGLE_90_0] = 785; calibrationParams.vaneADCValues[WMK_ANGLE_112_5] = 404; calibrationParams.vaneADCValues[WMK_ANGLE_135_0] = 460; calibrationParams.vaneADCValues[WMK_ANGLE_157_5] = 82; calibrationParams.vaneADCValues[WMK_ANGLE_180_0] = 91; calibrationParams.vaneADCValues[WMK_ANGLE_202_5] = 64; calibrationParams.vaneADCValues[WMK_ANGLE_225_0] = 185; calibrationParams.vaneADCValues[WMK_ANGLE_247_5] = 125; calibrationParams.vaneADCValues[WMK_ANGLE_270_0] = 285; calibrationParams.vaneADCValues[WMK_ANGLE_292_5] = 242; calibrationParams.vaneADCValues[WMK_ANGLE_315_0] = 628; calibrationParams.vaneADCValues[WMK_ANGLE_337_5] = 598; // The rainfall detector contains a small cup that collects rain water. When // the cup fills, the water is dumped and the total rainfall is incremented // by some value. This value defaults to 0.2794mm of rain per count, as // specified by the datasheet calibrationParams.mmPerRainfallCount = 0.2794; // The rainfall detector switch can sometimes bounce, causing multiple extra // triggers. This input is debounced by ignoring extra triggers within a // time window, which defaults to 100ms calibrationParams.minMillisPerRainfall = 250; // The anemometer contains a switch that opens and closes as it spins. The // rate at which the switch closes depends on the wind speed. The datasheet // states that a wind of 2.4kph causes the switch to close once per second calibrationParams.kphPerCountPerSec = 2.4; // Because the anemometer generates discrete pulses as it rotates, it's not // possible to measure the wind speed exactly at any point in time. A filter // is implemented in the library that averages the wind speed over a certain // time period, which defaults to 1 second. Longer intervals result in more // accurate measurements, but cause delay in the measurement // Dutch metrology institute (KNMI) defines that the windspeed and gust should // be calculated from 3 seconds measurements. calibrationParams.windSpeedMeasurementPeriodMillis = 3000; // Now we can set all the calibration parameters at once weatherMeterKit.setCalibrationParams(calibrationParams); // Begin weather meter kit weatherMeterKit.begin(); ts = millis(); RainPerHourCounter = ts; } void loop() { // Call once inside loop() - all magic here mb.task(); // Read each two seconds if ( ( millis() - ts) >= 2000) { ts = millis(); digitalWrite(LED_BUILTIN, HIGH); // LED as heartbeat // Read temperature and humidity ReadHYT221(); ReadSEN0562(); // Read pressure and temperature ReadBMP280(); // Read Wind and rain ReadSparkfunWeatherStation(); // Setting Sparkfun weather station registers mb.Ireg (SensorWindDirectionIreg, MeasuredData.WindDirection); mb.Ireg (SensorWindSpeedIreg, MeasuredData.WindSpeed); mb.Ireg (SensorWindGustIreg, MeasuredData.WindGust); mb.Ireg (SensorRainIreg, MeasuredData.Rain); mb.Ireg (SensorRainLast24Ireg, MeasuredData.RainLast24); mb.Ireg (SensorTemperatureIreg, MeasuredData.TemperatureBarometer); mb.Ireg (SensorHumidityIreg, MeasuredData.Humidity); mb.Ireg (SensorPressureIreg, MeasuredData.Pressure); mb.Ireg (SensorTemperatureBackupIreg, MeasuredData.TemperatureHygrometer); mb.Ireg (SensorLuminosityIreg, MeasuredData.Luminosity); mb.Ireg (SensorRainfallRawIreg, MeasuredData.RainfallCounter); mb.Ireg (SensorStatusBitsIreg, MeasuredData.StatusBits); // enable or disable smart heater (legacy code, does not do anything except setting the status bit) if (mb.Coil (HeaterCoil)) { MeasuredData.StatusBits |= 0x04; // Set bit } else { MeasuredData.StatusBits &= 0x0B; // Reset bit } digitalWrite(LED_BUILTIN, LOW); // LED as heartbeat } }