Weather station with ModBus over RS-485
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

588 lines
20 KiB

/*********************************************************************************
*
* 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 <https://www.gnu.org/licenses/>.
*
* 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 <ModbusSerial.h>
#include "SparkFun_Weather_Meter_Kit_Arduino_Library.h"
//I2C
#include <Wire.h>
#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(&reg, 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
}
}