Tip
I recommend these steps to optimize reception if you are using a RX470C RF receiver module. Other receivers were not tested.
This programm allows the reception of temperature data from a ThermoPro TP65s outdoor temperature sensor using MicroPython. The TP sensor transmitts it's data via the 433 MHz ISM Band which can be aquired using a RX470C-V01 receiver. The RX470C-V01 module is a low cost (~2 €) 433 MHz receiver for use with microcontrollers (MCUs). The program was tested on a Raspberry Pi Pico and on an Esp32-WROVER-B running MicroPython v1.23.0.
Changes in V2:
- Changing source address of the transmitter during reset is taken into account. I think it is generated by a LFSR.
- Channel is detected (Ch. 1 - 3).
- Data from a second temperature transmitter (unknown brand) with a similar protocol can be received.
To Do:
- Identify low-power indicator bit of the TP transmitter. A battery charge lasts over a year so I couldn't identify it yet.
The RX470C-V01 module (RX) only needs one GPIO pin for communication with the MCU.
It can be supplied with either 5V or 3.3V.
RX has two identical digital output (DO) pins, either of them may be used.
Exemplary connections:
- RX_GND --> MCU_GND
- RX_VCC --> MCU_3V3
- RX_DO --> MCU_GPIO14
A HackRF and UniversalRadioHacker were used to analyze the RF protocol of the ThermoPro TP65s outdoor temperature sensor (refferred to as TP sensor in the following). The TP sensor periodically transmitts temperature data to the indoor base station using on-off-keying (OOK) on the 433 MHz band. OOK is the simplest form of Amplitude Shift Keying (ASK) modulation. Basically, the carrier signal is turned on and off to transmitt high and low states.
The data is transmitted similar to a morse encoding with alternating high and low pulses. Each bit is encoded by a high pulse and a low pulse. High pulses have fixed durations of ~500 µs followed by low pulses with durations of either 2000 µs or 4000 µs, encoding 0 and 1, respectively. Each transmission consits of six similar messages containing the data. The individual messages are seperated by a long low pulse (gap) with a duration of approx. 8800 µs. Data is transmitted periodically every 50s.
OOK Protocol:
- Bit 0: HIGH 500 µs + LOW 2000 µs
- Bit 1: HIGH 500 µs + LOW 4000 µs
- Gap : LOW 8800 µs
Structure of the Transmissions:
- Bit 01 - 14: Transmitter Address (probably LFSR generated)
- Bit 15 - 16: Channel No (1 - 3)
- Bit 17 - 28: Data = 12 Bit
- Bit 29 - 37: End Sequence = '000000011'
- Gap
Decoding the Data:
- Data is encoded in the two's complement format consisting of 12 bit.
- The decoded integer must be divided by ten to obtain the temperature.
Examples:
- Temperature above 0 °C: Convert binary data to decimal number and divide by 10.
- E. g.
'000010100110' --> 166 --> 16,6 °C
- E. g.
- Temperature below 0 °C: Convert binary data to decimal number, subtract 4096 (2^12) and divide by 10.
- E. g.
'111110111100' --> 4028 --> 4028 - 4096 = -68 --> -6,8°C
- E. g.
The figure shows measured signals of the outoor sensor at different temperatures and the received binary data.
I bought this receiver as part of a low-cost temperature station at a discounter some years ago (probably from Aldi or Lidl). I could not identify the brand. It was placed outside for years and the battery contacts where oxidized. Therefore, I attached an external battery holder.
Interestingly, this temperature sensor uses the same OOK encoding pattern as the TP sensor. It was easy to integrate in the receiver programm.
The main difference compared to the TP sensor is the length of a single message which is 32 frames. Furthermore, the data is transmitted periodically every 30 seconds.
Structure of the Transmissions:
- Bit 01 - 12: Transmitter Address (probably LFSR generated)
- Bit 13 - 24: Data = 12 Bit
- Bit 25 - 32: End Sequence
- Gap
I built a Touch Control Panel for my bedside table which also displays the temperature data from the TP sensor and from the no name sensor. I used a Cheap Yellow Display and LVGL to create a simple UI.
Parts of the signal decoding function are based on Peter Hinch's micropython_remote RX class provided under MIT license.