<!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml" lang="" xml:lang=""> <head> <meta charset="utf-8" /> <meta name="generator" content="pandoc" /> <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes" /> <meta name="author" content="M.T. Konstapel" /> <meta name="dcterms.date" content="2025-01-14" /> <title>Weather station</title> <link rel="stylesheet" href="./css/mvp.css" /> <style type="text/css"> :root { --width-content: 1080px; } nav { justify-content: space-around; } </style> </head> <body> <header id="title-block-header"> <nav id="TOC"> <ul> <li> <a href="#">Index</a> <ul> <li><a href="#why-do-you-need-a-weather-station" id="toc-why-do-you-need-a-weather-station">Why do you need a weather station?</a></li> <li><a href="#what-should-a-weather-station-measure" id="toc-what-should-a-weather-station-measure">What should a weather station measure?</a></li> <li><a href="#what-sensors-do-we-need" id="toc-what-sensors-do-we-need">What sensors do we need?</a></li> <li><a href="#what-to-use-for-communication-with-the-outside-world" id="toc-what-to-use-for-communication-with-the-outside-world">What to use for communication with the outside world?</a></li> <li><a href="#what-else" id="toc-what-else">What else?</a></li> <li><a href="#theory-of-operation---hardware" id="toc-theory-of-operation---hardware">Theory of operation - Hardware</a></li> <li><a href="#theory-of-operation---software" id="toc-theory-of-operation---software">Theory of operation - Software</a></li> <li><a href="#prototype" id="toc-prototype">Prototype</a></li> <li><a href="#specifications" id="toc-specifications">Specifications</a></li> <li><a href="#schematic" id="toc-schematic">Schematic</a></li> <li><a href="#problems-i-encounter-after-four-months-of-use" id="toc-problems-i-encounter-after-four-months-of-use">Problems I encounter after four months of use</a></li> <li><a href="#additions" id="toc-additions">Additions</a></li> <li><a href="#software-dependencies" id="toc-software-dependencies">Software dependencies</a></li> <li><a href="#license" id="toc-license">License</a></li> </ul> </li> <li> <a href="./weather_station.pdf">PDF version</a> </li> <li> <a href="https://git.meezenest.nl/marcel/weather_station">Git repo</a> </li> <li> <a href="https://meezenest.nl/mees/weather_station.html">Back</a> </li> </ul> <a href="https://www.meezenest.nl/mees/"><img alt="Logo" src="./images/mees_logo.svg" height="70"></a> </nav> <h1 class="title">Weather station</h1> <p class="subtitle">with ModBus RTU interface</p> <p class="author">M.T. Konstapel</p> <p class="date">2025-01-14</p> <p><a href="./weather_station.pdf"><i>PDF version</i></a></p> </header> <main> <article> <p><b>Abstract </b><p>A weather station build around a SparkFun Weather Meter Kit (SEN-15901). The temperature, humidity and pressure are measured with I2C sensors housed in an RS1 Passive Radiation Shield from Garni. The data can be read via an RS485 ModBus RTU interface. The main processor is an Arduino Pro Mini (ATmega328P 5V@16MHz)</p></p> <h1 id="why-do-you-need-a-weather-station">Why do you need a weather station?</h1> <p>Well, you don’t…because if you want to know the weather, you look on your phone. So why bother than? Because since the beginning of time, people are obsessed with the weather. When I was a child, my grandmother was measuring the temperature and rainfall on a daily basis. My grandfather had an allotment, so he also was very interested in the weather. The first thing my father read in the newspaper was the weather report and the last thing he watched on the television was… the weather report. And every hour he listed to the weather report on the radio. If he talked to someone he always started the conversation by talking about the weather. And when I open a new browser window, it automatically opens the weather page.</p> <p>So the weather is fascinating and taking your own measurements is a lot of fun.</p> <h1 id="what-should-a-weather-station-measure">What should a weather station measure?</h1> <p>As my grandmother already measured temperature and rainfall, these ones are mandatory. And for the rest I looked at the website of the Dutch meteorological institute. They measure wind direction, average wind speed of the last 10 minutes, maximum wind gust of the last 10 minutes, rainfall of the last hour as well as the last 24 hours, temperature, humidity and atmospheric pressure.</p> <h3 id="measurements">Measurements</h3> <ul> <li>Wind direction</li> <li>Wind speed (average of last 10 minutes)</li> <li>Wind gust (last 10 minutes)</li> <li>Rain fall (last hour)</li> <li>Rain fall (last 24 hours)</li> <li>Temperature</li> <li>Humidity</li> <li>Atmospheric pressure</li> </ul> <h1 id="what-sensors-do-we-need">What sensors do we need?</h1> <h2 id="wind-and-rain">Wind and rain</h2> <p>Measuring wind and rain is difficult. Well, not if you want to do it by hand: place a beaker on the ground and wait a day. Than measure the amount of water in it. Empty the beaker and start again. And for the wind, you can stick a pole in the ground and attach a ribbon to it. The direction of the wind can than be made visible. And even the wind speed can be determent by measuring the angle between the ribbon and the ground.</p> <p>But how to do this automatically? Of course you can buy a fancy commercial weather station. These are surprisingly cheap these days. But that’s not a challenge. Besides, than you buy into a proprietary ecosystem. And it probably only works when connected to the cloud. No thanks!</p> <p>Building from scratch is an option, but I am an electronic engineer, not a mechanical one. I can imagine that won’t be a success. Besides going the professional route, which is ridiculously expensive, there is really only one option left: the SparkFun SEN-15901 Weather Meter.</p> <figure> <img src="./images/SparkFun-Weather_Meter.jpg" title="SparkFun Weather Meter" alt="SparkFun Weather Meter" /> <figcaption aria-hidden="true">SparkFun Weather Meter</figcaption> </figure> <p>But this contraption does not come with any signal conditioning. We have to make some kind of interface. Luckily, Sparkfun provides an Arduino library, so we only have to connect the SEN-15901 to an Arduino and run the code.</p> <h2 id="temperature-humidity-and-air-pressure">Temperature, humidity and air pressure</h2> <p>These three are easy: there are a lot of I2C chips capable of measuring these parameters. I choose the Silicon Labs Si7021 for humidity and temperature and the Bosch BMP280 for pressure. Just hook them up to the Arduino’s I2C bus, load the available libraries and Bob’s your uncle.</p> <p>To mount these sensors on the same mast as the SparkFun weather meter I use the RS1 passive radiation shield from Garni.</p> <figure> <img src="./images/garni_rs1.jpg" title="Garni RS1 Passive Radiation Shield" alt="Garni RS1 Passive Radiation Shield" /> <figcaption aria-hidden="true">Garni RS1 Passive Radiation Shield</figcaption> </figure> <h3 id="sensors">Sensors</h3> <ul> <li>SparkFun SEN-15901 Weather Station</li> <li>Silicon Labs Si7021</li> <li>Bosch BMP280</li> </ul> <h1 id="what-to-use-for-communication-with-the-outside-world">What to use for communication with the outside world?</h1> <p>Most consumer grade weather stations (and almost all other consumer grade goods for that matter) use proprietary interfaces and protocols. Probably to annoy the more technical skilled customer as you are not able to interface these devices with other brands or self build systems. I really hate that practice, so I won’t do that. Instead I will implement a ModBus RTU interface. Dating back to 1979, this is the industrial standard for communication between devices. And if the professionals all use it, why not use it for this weather station?</p> <h2 id="modbus">ModBus</h2> <p>ModBus is a client/server data communications protocol in the application layer of the OSI model. ModBus can work over several different physical interfaces. For this application I will use an RS-485 interface. This interface is easy to implement and cables can be very long, making it easy to locate the weather station. ModBus is a lightweight protocol which can comfortably fit inside an under-powered micro-controller like an Atmel ATmega328P. A simple RS-485 to USB dongle connected to a PC is all you need to read the values from the weather station.</p> <h1 id="what-else">What else?</h1> <p>Not much to be honest. Almost everything can be done in software. Of course we need a power supply. And preferably a reverse polarity protection. An input voltage of 12 Volt is convenient. 12 Volt power bricks can be found in every charity shop and you can also use a 12 Volt lead acid or lithium battery to power the weather station.</p> <h1 id="theory-of-operation---hardware">Theory of operation - Hardware</h1> <figure> <img src="./images/block_diagram.svg" title="Block diagram of weather station" alt="Block diagram of weather station" /> <figcaption aria-hidden="true">Block diagram of weather station</figcaption> </figure> <h2 id="wind-speed">Wind speed</h2> <p>Measuring the wind speed is done by a cup anemometer. It consisted of three or four hemispherical cups on horizontal arms mounted on a vertical shaft. The air flow past the cups in any horizontal direction turned the shaft at a rate roughly proportional to the wind’s speed. Every rotation, a magnet passes alongside a reed switch. The rate at which the reed switch opens en closes is a measure of the wind speed.</p> <figure> <img src="./images/sparkfun_cup_anemometer.jpg" title="Cup anemometer" alt="Cup anemometer" /> <figcaption aria-hidden="true">Cup anemometer</figcaption> </figure> <p>By connecting one side of the reed switch to ground and the other via a pull-up resistor to the supply voltage the mechanical switching action is translated to an electrical pulse signal. This pulse can be read by a micro-controller.</p> <figure> <img src="./images/diagram_cup_anemometer.svg" title="Cup anemometer: theory of operation" alt="Cup anemometer: theory of operation" /> <figcaption aria-hidden="true">Cup anemometer: theory of operation</figcaption> </figure> <h2 id="wind-direction">Wind direction</h2> <p>Measuring the wind direction is done by a wind vane. It consists of a vertical blade mounted on a vertical shaft. Because the blade can turn, it will always find the position of the least air resistance. The shape of the blade is chosen so that it will always points directly to the wind. A magnet mounted on the shaft rotates past several reed switches. The switch that is closest to the magnet will close. If the magnet is precisely between two reed switches both switches will close increasing the resolution of the wind vane.</p> <figure> <img src="./images/sparkfun_wind_vane.jpg" title="Wind vane" alt="Wind vane" /> <figcaption aria-hidden="true">Wind vane</figcaption> </figure> <p>Each reed switch is connected to a resistor and every resister has a different value. The total resistance of the network will change according to the wind direction. By connecting one side of the network to ground and the other side via a resistor to VCC, a resisive divider in made. This resistive divider converts the variable resistance to an analog voltage which can be sampled by the A/D converter of a micro-controller.</p> <figure> <img src="./images/diagram_wind_vane.svg" title="Wind vane: theory of operation" alt="Wind vane: theory of operation" /> <figcaption aria-hidden="true">Wind vane: theory of operation</figcaption> </figure> <table> <thead> <tr class="header"> <th>Direction</th> <th>Resistance</th> </tr> </thead> <tbody> <tr class="odd"> <td>0°</td> <td>33kΩ</td> </tr> <tr class="even"> <td>22.5°</td> <td>6.57kΩ</td> </tr> <tr class="odd"> <td>45°</td> <td>8.2kΩ</td> </tr> <tr class="even"> <td>67.5°</td> <td>891Ω</td> </tr> <tr class="odd"> <td>90°</td> <td>1kΩ</td> </tr> <tr class="even"> <td>112.5°</td> <td>688Ω</td> </tr> <tr class="odd"> <td>135°</td> <td>2.2kΩ</td> </tr> <tr class="even"> <td>157.5°</td> <td>1.41kΩ</td> </tr> <tr class="odd"> <td>180°</td> <td>3.9kΩ</td> </tr> <tr class="even"> <td>202.5°</td> <td>3.14kΩ</td> </tr> <tr class="odd"> <td>225°</td> <td>16kΩ</td> </tr> <tr class="even"> <td>247.5°</td> <td>14.12kΩ</td> </tr> <tr class="odd"> <td>270°</td> <td>120kΩ</td> </tr> <tr class="even"> <td>292.5°</td> <td>42.12kΩ</td> </tr> <tr class="odd"> <td>315°</td> <td>64.9kΩ</td> </tr> <tr class="even"> <td>337.5°</td> <td>21.88kΩ</td> </tr> </tbody> </table> <h2 id="rain-fall">Rain fall</h2> <p>Measuring the amount of rain fall is done by a self-emptying tipping bucket. Rainwater is collected and funneled to a tipping bucket. The bucket tips over when a certain amount of water is collected. The bucket drains and a second bucket is automatically placed under the funnel. When a certain amount of water is collected in this second bucket it will tip over and the first bucket is raised again.</p> <figure> <img src="./images/sparkfun_rain_meter_inside.jpg" title="Rain meter" alt="Rain meter" /> <figcaption aria-hidden="true">Rain meter</figcaption> </figure> <p>Every time the bucket tips over a magnet passes by a reed switch, which closes and opens again. As with the cup anemometer this mechanical movement can be translated to an electrical pulse by connecting one side of the switch to ground and the other side via a pull-up resistor to VCC. This pulse can than be read by a micro-controller.</p> <p><sup>NOTE</sup> The rain meter is very sensitive: even a small amount of movement and the bucket tips over. Mounting the rain meter in the mast together with the wind meters can cause false triggers from the rocking motion of the mast.</p> <figure> <img src="./images/diagram_rain_meter.svg" title="Rain meter: theory of operation" alt="Rain meter: theory of operation" /> <figcaption aria-hidden="true">Rain meter: theory of operation</figcaption> </figure> <h2 id="humidity">Humidity</h2> <p>Measuring the relative humidity is done by an electronic sensor based on capacitive sensing using polymeric dielectrics. The humidity sensor is a small capacitor consisting of a hygroscopic dielectric material placed between a pair of electrodes. Absorption of moisture by the sensor results in an increase in sensor capacitance. The opposite is also true: when the moisture disappears, sensor capacitane decreases. There is a direct relationship between relative humidity, the amount of moisture present in the sensor, and the sensor capacitance. The relative humidity is defined as the ratio of the amount of water vapor in the air at a specific temperature to the maximum amount that the air could hold at that temperature, expressed as a percentage. As the humidity sensor has a build in temperature sensor, it can calculate the relative humidity.</p> <figure> <img src="./images/capacitive_humidity_sensor.png" title="Humidity sensor" alt="Humidity sensor" /> <figcaption aria-hidden="true">Humidity sensor</figcaption> </figure> <p>The Si7021 humidity sensor has an I²C bus for communication with a micro-controller.</p> <figure> <img src="./images/Si7021_block_diagram.png" title="Humidity sensor: block diagram" alt="Humidity sensor: block diagram" /> <figcaption aria-hidden="true">Humidity sensor: block diagram</figcaption> </figure> <h2 id="temperature">Temperature</h2> <p>Measuring the temperature is done by the build in temperature sensor of the humidity sensor. This sensor is used by the humidity sensor to calculate the relative humidity. But as this sensor is very accurate it can be used for ambient temperature measurments.</p> <h2 id="atmospheric-pressure">Atmospheric pressure</h2> <p>Measuring the atmospheric pressure is done by an electronic sensor based on a piezo-resistive pressure sensing element. The piezo-resistive effect is a change in the electrical resistivity of a semiconductor or metal when mechanical strain is applied. In this case the strain comes from the atmospheric pressure. The sensor measures the resistance which is proportional to the atmospheric pressure.</p> <figure> <img src="./images/piezo_resistive_pressure_sensor.png" title="Piezo-resistive pressure sensor" alt="Piezo-resistive pressure sensor" /> <figcaption aria-hidden="true">Piezo-resistive pressure sensor</figcaption> </figure> <p>The BMP280 pressure sensor has an on board temperature sensor which can also be used to measure the ambient temperature. As this sensor is less accurate compared to the sensor of the Si7021 humidity sensor, the sensor is only used as a backup sensor. The BMP280 has an I²C bus for communication with a micro-controller.</p> <figure> <img src="./images/BMP280_block_diagram.png" title="Pressure sensor: block diagram" alt="Pressure sensor: block diagram" /> <figcaption aria-hidden="true">Pressure sensor: block diagram</figcaption> </figure> <h2 id="illumination">Illumination</h2> <p>This sensor is still under development.</p> <h2 id="modbus-interface">ModBus interface</h2> <p>The RS-485 interface is build with a MAX485E driver chip from Maxim Integrated. Nothing much to say as the implementation is pretty much following the typical application from the datasheet.</p> <figure> <img src="./images/rs-485.svg" title="RS-485 interface" alt="RS-485 interface" /> <figcaption aria-hidden="true">RS-485 interface</figcaption> </figure> <p>If the device is the first or last device on the RS-485 bus, a 120 Ohm termination resistor can be enabled by placing a jumper on header J7.</p> <p>The Arduino micro-controller can be programmed via an in circuit programmer, which shares the serial port with the MAX485E. Resistor R5 isolates the output of the MAX485 from the signal of the programmer.</p> <h2 id="i²c-bus">I²C bus</h2> <p>The I²C bus is integrated in the micro-controller. But because the micro-controller uses a power supply of 5 Volt and the I²C sensors use 3.3 Volt a bidirectional level shifter is needed. This way the sensors can be used on the 5V I²C bus without the risk of damaging the sensors.</p> <figure> <img src="./images/i2c_bus.svg" title="I²C bus level shifter" alt="I²C bus level shifter" /> <figcaption aria-hidden="true">I²C bus level shifter</figcaption> </figure> <p>Let’s assume the I²C signal lines on either end of the MOSFETs are either outputting a logic high or is configured as an input. Effectively this means there is nothing pulling the signal levels down.</p> <p>The voltage between the gate and source of both MOSFETs is at 0V (both are at 3.3V) so the MOSFET is switched off. Therefore both sides of the MOSFETs are logic high.</p> <p>When either of the 3.3 Volt signal lines outputs a logic low the corresponding drain is pulled to ground. Now the voltage between the gate and the source is 3.3V and the MOSFET turns on causing the 5 Volt side to go low as well.</p> <p>When either of the 5 Volt signal lines outputs a logic low the body diode of the corresponding MOSFET start conducting, causing the source voltage to drop below the gate voltage. The MOSFET switches on and the 3.3 volt side goes low.</p> <h2 id="power-supply">Power supply</h2> <p>Typical, a 12 Volt power supply is used to power the device, but it can be powered from a wide range of voltages, from 6.5 to 36 Volt. A switching regulator (U3) supplies the 5 Volt power rail and a linear low drop regulator (U4) supplies the 3.3 volt power rail.</p> <figure> <img src="./images/power_supply.svg" title="Power supply" alt="Power supply" /> <figcaption aria-hidden="true">Power supply</figcaption> </figure> <h3 id="input-protection">Input protection</h3> <p>C1 and C5 short out high frequency signals, protecting the input from ESD. Bidirectional transient-voltage-suppression diodes D1 end D2 clamp transient voltages, again protecting the input from ESD.</p> <p>And than Q3 and its surrounding components: this is the reverse polarity protection. Usually, a series diode is used, but due to the voltage drop across such a diode it dissipates energy which is wasteful. The circuit with Q3 on the other hand has a very low voltage drop resulting in an almost zero loss solution.</p> <p>If VCC is applied in the correct polarity, the source will immediately rise to the about VCC because of the body diode conducting.</p> <p>The gate will charge towards -VCC with respect to the source through R1. When the gate reaches the threshold voltage the MOSFET channel will begin to conduct, and by the time the gate-source voltage reaches a few volts the MOSFET channel will be conducting almost all the current, the output voltage will be close to VCC. It continues to charge until it reaches about -10V at which point the zener diode begins to shunt significant current away from the gate.</p> <p>In steady state with VCC on the drain the gate sits at -10V with respect to the source, and the MOSFET happily conducts in the reverse direction.</p> <p>When VCC is applied in the reverse polarity, the body diode of the MOSFET cannot conduct. Only a small leakage current can flow from the source to the drain via resistor R1 and zener diode D3, which now acts as a normal diode. The gate and the source are now at almost the same potential and the MOSFET cannot conduct, protecting the device from reverse polarity.</p> <figure> <img src="./images/reverse_polarity_protection.svg" title="Reverse polarity protection" alt="Reverse polarity protection" /> <figcaption aria-hidden="true">Reverse polarity protection</figcaption> </figure> <h2 id="microcontroller">Microcontroller</h2> <p>The heart of the circuit is an Arduino Pro Mini, which is basically an Atmel ATmega328P with a special Arduino bootloader, making it an easy platform for developing software.</p> <figure> <img src="./images/micro-controller.svg" title="Microcontroller" alt="Microcontroller" /> <figcaption aria-hidden="true">Microcontroller</figcaption> </figure> <p>Both the signals from the rain meter and the cup anemometer are connected to interrupt pins of the micro-controller. The signal from the rain meter is lightly filtered by C8. The output of the wind vane is connectd to an analog input of the micro-controller.</p> <p>The ModBus address can be set by DIP switch J9.</p> <h1 id="theory-of-operation---software">Theory of operation - Software</h1> <h2 id="wind-speed-1">Wind speed</h2> <p>The pulse from the cup anemometer is connected to an interrupt input of the micro-controller. Every time its logic level changes an interrupt routine is called. This routine increments a counter and checks how many time has passed since the previous interrupt call. If the previous call was more than 6 seconds ago, the wind speed is (almost) zero. If the previous call was just over 3 second ago the interrupt counter now holds the amount of pulses in three seconds. This value is stored and from that value the wind speed can be calculated. If the previous call was less than 3 seconds ago the measurement is still in progress and no further action is taken. A measurment period of three seconds is choosen because it is the standard as used by the Dutch meteorological institute (KNMI).</p> <figure> <img src="./images/wind_speed_diagram.svg" title="Wind speed interrupt" alt="Wind speed interrupt" /> <figcaption aria-hidden="true">Wind speed interrupt</figcaption> </figure> <h2 id="wind-direction-1">Wind direction</h2> <p>The analog signal from the wind vane is fed into the analog to digital converter of the micro-controller. The software samples this signal and determines which value from a lookup table is closest to the value from the ADC. The lookup table now gives the wind direction in degrees.</p> <figure> <img src="./images/wind_direction_diagram.svg" title="Getting the wind direction" alt="Getting the wind direction" /> <figcaption aria-hidden="true">Getting the wind direction</figcaption> </figure> <p>As the tolerances between micro-controllers can be high, the wind vane has to be calibrated in order to get a correct lookup table.</p> <h2 id="rain-fall-1">Rain fall</h2> <p>The pulse from the rain meter is connected to an interrupt input of the micro-controller. Every time a rising edge is detected an interrupt routine is called. This routine debounces the signal and increments the rain counter. This counter can be used to calculate the rain fall.</p> <h2 id="humidity-1">Humidity</h2> <p>Via the I²C bus, the humidity value of the sensor is read. As the sensor can become saturated with moisture it can get stuck at 100%. This happens in particular with fog or other high humidity and condensing weather types. The sensor has a build in heater to drive of moisture and thus preventing this problem. Because the temperature of the sensor rises when the heater is turned on, accurate ambient temperature and humidity readings are no longer possible. But with a smart algorithm it is possible to get the benefits of the build in heater while still being able to use the sensor as an ambient thermometer.</p> <p>When the humidity rises above 95% for more than an hour the current temperature and humidity are stored and the heater is switched on for 5 minutes. Than the heater is switched off again. If after 15 minutes the humidity is still above 95% the heater is turned on again for another 5 minutes. But not before the temperature and humidity are measured and stored, as the sensor is now cooled off to ambient temperature. If the humidity is below 95% the sensor is free from moisture and the process is not repeated for another hour.</p> <p>When the heater algorithm is active, the temperature and humidity values are updated every 20 minutes instead of every 2 seconds. Statis bit 0 (ModBus register 30014) indicated if the heater is on or off and status bit 1 gives the update rate of the temperature and humidity values.</p> <p>This algorithm can be enabled by setting the HeaterCoil (see ModBus secion).</p> <figure> <img src="./images/smart_heater.svg" title="Heater algorithm" alt="Heater algorithm" /> <figcaption aria-hidden="true">Heater algorithm</figcaption> </figure> <h2 id="temperature-1">Temperature</h2> <p>The temperature is read from the humidity sensor as this sensor gives the most accurate temperature readings. When the heater is on (see section humidity above) the temperature readings are temporary stopped and only updated every 20 minutes. As a backup, the slightly less accurate temperature readings from the pressure sensor can be used.</p> <h2 id="atmospheric-pressure-1">Atmospheric pressure</h2> <p>Via the I²C bus, the atmospheric pressure value of the sensor is read. There is nothing further to say about this sensor: it is rather boring.</p> <h2 id="illumination-1">Illumination</h2> <p>This sensor is still under development.</p> <h2 id="modbus-interface-1">ModBus interface</h2> <p>The weather station uses ModBus RTU over a simplex RS-485 line. For now, the ModBus address is hard coded as 14 in the software. The values are available in the input registers and can be read via function code 04.</p> <p>Below an example of how to read the wind direction and enable the heater algorithm in Python using the minimalmodbus library.</p> <pre><code>#!/usr/bin/env python3 import minimalmodbus # port name, slave address (in decimal) instrument = minimalmodbus.Instrument('/dev/ttyUSB1', 14) # register number, number of decimals, function code wind_direction = instrument.read_register(1, 1, 4) print(wind_direction) # register address, value, function code instrument.write_bit(0, 1, 5)</code></pre> <h3 id="input-registers-read-only">Input registers (read only)</h3> <p>Input registers are numbered 30001 to 39999 but have data addresses 0x000 to 0x270E. The measurements and order of the measurements are the same as for APRS weather reports. But of course we use SI units.</p> <table> <colgroup> <col style="width: 13%" /> <col style="width: 53%" /> <col style="width: 33%" /> </colgroup> <thead> <tr class="header"> <th>Address</th> <th>Description</th> <th>Units</th> </tr> </thead> <tbody> <tr class="odd"> <td>00</td> <td>Device ID (0x5758)</td> <td>NO UNIT</td> </tr> <tr class="even"> <td>01</td> <td>Wind direction</td> <td>degrees * 10</td> </tr> <tr class="odd"> <td>02</td> <td>Wind speed (average of 10 minutes)</td> <td>m/s * 100</td> </tr> <tr class="even"> <td>03</td> <td>Wind gust (peak of last 10 minutes)</td> <td>m/s * 100</td> </tr> <tr class="odd"> <td>04</td> <td>Temperature (two’s complement)</td> <td>degrees Celcius * 100</td> </tr> <tr class="even"> <td>05</td> <td>Rain last hour</td> <td>l/m2 * 100</td> </tr> <tr class="odd"> <td>06</td> <td>Rain last 24 hours</td> <td>l/m2 * 100</td> </tr> <tr class="even"> <td>07</td> <td>Rain since midnight</td> <td>NOT IMPLEMENTED</td> </tr> <tr class="odd"> <td>08</td> <td>Humidity</td> <td>percent * 100</td> </tr> <tr class="even"> <td>09</td> <td>Barometric pressure</td> <td>hPa * 10</td> </tr> <tr class="odd"> <td>10</td> <td>Luminosity</td> <td>W/m2</td> </tr> <tr class="even"> <td>11</td> <td>Snow fall</td> <td>NOT IMPLEMENTED</td> </tr> <tr class="odd"> <td>12</td> <td>Raw rain counter</td> <td>l/m2 * 100</td> </tr> <tr class="even"> <td>13</td> <td>Temperature (two’s complement)</td> <td>degrees Celcius * 100</td> </tr> <tr class="odd"> <td>14</td> <td>Status bits</td> <td>see table below</td> </tr> </tbody> </table> <p><sup>NOTE</sup> Register 13 holds the backup temperature reading from the pressure sensor.</p> <table> <colgroup> <col style="width: 18%" /> <col style="width: 31%" /> <col style="width: 25%" /> <col style="width: 24%" /> </colgroup> <thead> <tr class="header"> <th>Status bits</th> <th>Description</th> <th>logic 0</th> <th>logic 1</th> </tr> </thead> <tbody> <tr class="odd"> <td>0</td> <td>Heater status</td> <td>heater off</td> <td>heater on</td> </tr> <tr class="even"> <td>1</td> <td>Temp/humidity update</td> <td>every 20 minutes</td> <td>every 2 seconds</td> </tr> <tr class="odd"> <td>2</td> <td>Heater algorithm</td> <td>disabled</td> <td>enabled</td> </tr> </tbody> </table> <p>The ModBus registers are 16 bit wide. For better precision, some units are scaled by a factor of 10 or 100. This way, values with up to two decimal points can be stored as 16 bit integer values. Just divide by 10 or 100 to get the floating point values.</p> <h3 id="output-coils-write-only">Output coils (write only)</h3> <p>Output coils registers are numbered 1 to 9999 but have data addresses 0x000 to 0x270E. The default value of a register is 0.</p> <table> <thead> <tr class="header"> <th>Address</th> <th>Description</th> <th>logic 0</th> <th>logic 1</th> </tr> </thead> <tbody> <tr class="odd"> <td>0</td> <td>Heater algorithm</td> <td>disabled</td> <td>enabled</td> </tr> </tbody> </table> <h1 id="prototype">Prototype</h1> <p>I wanted to locate the weather station at about 100 meters from the house. That meant that interfacing the weather station was not just a matter of connecting a wire to it. And 100 meters is also a bit much for a wifi connection. As I already had experience with LoRa I opted for that. But not LoRaWAN, but LoRa APRS. This is a ham radio network that I often use. I even run my own digipeater. So LoRa APRS it is.</p> <figure> <img src="./images/prototype_block_diagram.svg" title="Block diagram of the prototype" alt="Block diagram of the prototype" /> <figcaption aria-hidden="true">Block diagram of the prototype</figcaption> </figure> <p>The weather station’s RS-485 interface is connected to a Raspberry Pi Zero 2W running the aprx digipeater software, as well as some specially written Python programs to interface the build in LoRa transceiver, the GPS module and the weather station itself. Every 10 minutes the digipeater will read the weather station’s registers and sends the data as PE1RXF telemetry messages (see <a href="https://www.meezenest.nl/mees-elektronica/projects/aprs_telemetry/APRS_protocol_nodes_PE1RXF.pdf">https://www.meezenest.nl/mees-elektronica/projects/aprs_telemetry/APRS_protocol_nodes_PE1RXF.pdf</a>) over the APRS network to a server, which presents the data in a Grafana dashboard. The digipeater can also send standardized APRS weather reports over the APRS network. But more about this project can be found here: <a href="https://www.meezenest.nl/mees-elektronica/RPi_LoRa_shield.html">https://www.meezenest.nl/mees-elektronica/RPi_LoRa_shield.html</a></p> <figure> <img src="./images/prototype_overview_small.jpg" title="The prototype in the garden" alt="The prototype in the garden" /> <figcaption aria-hidden="true">The prototype in the garden</figcaption> </figure> <p>As a housing for the prototype, I used an old beehive. These are weatherproof and I had one laying around.</p> <figure> <img src="./images/prototype_sensors_small.jpg" title="Closeup of the sensors" alt="Closeup of the sensors" /> <figcaption aria-hidden="true">Closeup of the sensors</figcaption> </figure> <figure> <img src="./images/prototype_inside_annotations_small.jpg" title="Inside the beehive" alt="Inside the beehive" /> <figcaption aria-hidden="true">Inside the beehive</figcaption> </figure> <h1 id="specifications">Specifications</h1> <h2 id="wind">Wind</h2> <ul> <li><p>Wind speed is measured by taking 3 second averages from the cup anemometer and using these samples to calculate the average over a 10 minute periode.</p></li> <li><p>Wind gust is measured by taking 3 second averages from the cup anemometer.</p></li> <li><p>Wind vane has 8 main directions and another 8 directions in between. But these last do not have the same weight, eg. these positions are not as likely to be measured as the main directions. This is due to the construction of the wind vane: it has eight reed switches for the main directions and if the wind direction happens to sit exactly in between two reed switches, both switches are closed giving the extra 8 sub directions. Not great, but it is what it is…</p></li> </ul> <h2 id="rain">Rain</h2> <pre><code>Resolution: 0.2794 mm/impulse</code></pre> <h2 id="humidity-2">Humidity</h2> <pre><code>Operating range : 0 - 100 % RH Recommended range : 20 - 80 % RH Accuracy : +/- 3 % RH (0-80 % RH) +/- 4.5 % (max when > 80 % RH) Heater to drive of moisture (can be enabled via ModBus)</code></pre> <h2 id="pressure">Pressure</h2> <pre><code>Operating range : 300 - 1100 hPa Accuracy : +/- 1.0 hPa (0 - 65 °C) +/- 1.7 hPa (-20 - 0 °C)</code></pre> <h2 id="temperature-2">Temperature</h2> <pre><code>Main sensor ----------- Operating range : -10 - 85 °C (typ) -40 - 85 °C (max) Accuracy : +/- 0.3 °C (typ) +/- 0.4 °C (max) +/- 0.5 °C (max when < -10°C) Backup sensor ------------- Operating range : 0 - 65 °C (typ) -40 - 85 °C (max) Accuracy : +/- 0.5 °C (25 °C) +/- 1.0 °C (0 - 65 °C)</code></pre> <h2 id="modbus-1">ModBus</h2> <pre><code>Physical : RS-485 simplex RTU Settings : 9600 bd 8N1 Address : 14</code></pre> <h1 id="schematic">Schematic</h1> <p><a href="./images/weather_station_schematic.pdf"><img src="./images/weather_station_schematic.svg" alt="Schematic" /></a></p> <h1 id="problems-i-encounter-after-four-months-of-use">Problems I encounter after four months of use</h1> <h2 id="humidity-sensor">Humidity sensor</h2> <p>The Si7021 humidity sensor is not made for outdoor use. The datasheet is clear about that. But a lot of people use this cheap sensor for weather stations anyway. So I choose this sensor for my design. But that was not smart. After an initial time without any problems, the sensor started to saturate. This happened during a very wet and mild winter we had. My first solution was to utilize the build in heater to drive of the moisture. That worked, but only for a short period. The heater blew up and the sensor started to report humidity levels above 100% and because of a bug in the firmware of the sensor, the humidity register wrapped around and the sensor reported humidity levels of 0-30%. I tried to find a software solution, which worked for a while, but the sensor deteriorated even more. To the point of being totally useless. So I searched for another, better sensor. And I found the HYT-221 from Innovative Sensor Technology. This sensor is designed to work outdoors and can even be used in saunas, where the humidity levels are always high and the air is condensating. The datasheet specifically mentions outdoor weather stations as an application. The only downside is its price: it is ten times more expensive than the Si7021.</p> <p>The sensor can be controlled via the I2C bus, so implementing the new sensor in the firmware was very easy. From version 0.3.0 onward, this sensor is used. The Si7021 is removed from the code.</p> <h2 id="pressure-sensor">Pressure sensor</h2> <p>After a year the BMP280 pressure sensor started to give odd pressure values. The pressure was around 700hPa and followed the temperature curve. The temperature sensor still worked fine. After replacing the BMP280 the barometric values returned to normal. I suspect that it malfunctioned because of the perpetual fog we had for about two months. The sensor was mounted in the Garni RS1 Passive Radiation Shield. Rain could not enter the shield, but fog could.</p> <h1 id="additions">Additions</h1> <h2 id="luminosity-sensor">Luminosity sensor</h2> <p>From version 0.3.1 onward, the weather station has an ambient light sensor. It is an SEN0562 from DFRobot and it outputs the light intensity in Lux. It is a 5 Volt only device, so it should be connected to the 5 Volt I2C bus.</p> <figure> <img src="./images/dfrobot-gravity-ip68-waterproof-ambient-light-sensor-1-65535lx-i2c.png" alt="The SEN0562 ambient light sensor" /> <figcaption aria-hidden="true">The SEN0562 ambient light sensor</figcaption> </figure> <figure> <img src="./images/dfrobot-gravity-ip68-waterproof-ambient-light-sensor-wiring.png" alt="Connection of the I2C bus" /> <figcaption aria-hidden="true">Connection of the I2C bus</figcaption> </figure> <p>The sensor has the following specifications:</p> <ul> <li>Supply Voltage: 5V</li> <li>Operating Current: 1µA</li> <li>Detection Range: 1 - 65535lx</li> <li>Accuracy: 1.2lx</li> <li>Communication Mode: I2C</li> <li>Operating Temperature: -40 to 85°C/-40 to 185℉</li> <li>Waterproof Rating: IP68</li> <li>Thread Length: 10mm</li> <li>Cutout Size: 26mm</li> <li>Wrench Size: 31mm</li> <li>Cable Diameter: 3mm</li> <li>Wire Length: 1m</li> </ul> <p>Sample code:</p> <pre><code>#include "Wire.h" #define address 0x23 //I2C address 0x23 void setup() { Serial.begin(9600); Wire.begin(); } uint8_t buf[4] = {0}; uint16_t data, data1; float Lux; void loop() { readReg(0x10, buf, 2); //Register address 0x10 data = buf[0] << 8 | buf[1]; Lux = (((float)data )/1.2); Serial.print("LUX:"); Serial.print(Lux); Serial.print("lx"); Serial.print("\n"); delay(500); } uint8_t readReg(uint8_t reg, const void* pBuf, size_t size) { if (pBuf == NULL) { Serial.println("pBuf ERROR!! : null pointer"); } uint8_t * _pBuf = (uint8_t *)pBuf; Wire.beginTransmission(address); Wire.write(&reg, 1); if ( Wire.endTransmission() != 0) { return 0; } delay(20); Wire.requestFrom(address, (uint8_t) size); for (uint16_t i = 0; i < size; i++) { _pBuf[i] = Wire.read(); } return size; }</code></pre> <!--- # Bill of materials # Component placement # Cables and pinouts --> <h1 id="software-dependencies">Software dependencies</h1> <ul> <li>Arduino IDE</li> </ul> <h2 id="arduino-libraries">Arduino libraries</h2> <ul> <li>https://github.com/sparkfun/SparkFun_Weather_Meter_Kit_Arduino_Library</li> <li>https://github.com/orgua/iLib</li> <li>https://github.com/epsilonrt/modbus-arduino</li> <li>https://github.com/epsilonrt/modbus-serial</li> </ul> <p>Libraries are included with the source code of this project</p> <h1 id="license">License</h1> <p>Copyright (C) 2023-2025 M.T. Konstapel</p> <p><a href="https://meezenest.nl/mees/">https://meezenest.nl/mees/</a></p> <p>The software is published as open-source software (GPL). The hardware is published as open-source hardware (OSH).</p> <h2 id="software">Software</h2> <p>This program 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.</p> <h2 id="hardware-and-documentation">Hardware and documentation</h2> <p>This work is licensed under a Creative Commons Attribution-ShareAlike 4.0 International License.</p> <hr> </article> </main> <footer> <p>© 2025-01-14 M.T. Konstapel <a href="https://meezenest.nl/mees/">https://meezenest.nl/mees/</a> </p><p>This work is licensed under a <a rel="license" href="http://creativecommons.org/licenses/by-sa/4.0/">Creative Commons Attribution-ShareAlike 4.0 International License</a>. </p> </footer> </body> </html>