2023-01-18 17:18:19 +00:00
|
|
|
/* Citroen TPMS. Usually 443.92 Mhz FSK.
|
|
|
|
*
|
|
|
|
* Preamble of ~14 high/low 52 us pulses
|
|
|
|
* Sync of high 100us pulse then 50us low
|
|
|
|
* Then Manchester bits, 10 bytes total.
|
|
|
|
* Simple XOR checksum. */
|
|
|
|
|
|
|
|
#include "../../app.h"
|
|
|
|
|
2023-01-18 19:25:39 +00:00
|
|
|
static bool decode(uint8_t* bits, uint32_t numbytes, uint32_t numbits, ProtoViewMsgInfo* info) {
|
2023-01-18 17:18:19 +00:00
|
|
|
/* We consider a preamble of 17 symbols. They are more, but the decoding
|
|
|
|
* is more likely to happen if we don't pretend to receive from the
|
|
|
|
* very start of the message. */
|
|
|
|
uint32_t sync_len = 17;
|
2023-01-18 19:25:39 +00:00
|
|
|
const char* sync_pattern = "10101010101010110";
|
|
|
|
if(numbits - sync_len < 8 * 10) return false; /* Expect 10 bytes. */
|
2023-01-18 17:18:19 +00:00
|
|
|
|
2023-01-18 19:25:39 +00:00
|
|
|
uint64_t off = bitmap_seek_bits(bits, numbytes, 0, numbits, sync_pattern);
|
|
|
|
if(off == BITMAP_SEEK_NOT_FOUND) return false;
|
2023-01-18 17:18:19 +00:00
|
|
|
FURI_LOG_E(TAG, "Renault TPMS preamble+sync found");
|
|
|
|
|
|
|
|
info->start_off = off;
|
|
|
|
off += sync_len; /* Skip preamble + sync. */
|
|
|
|
|
|
|
|
uint8_t raw[10];
|
2023-01-18 19:25:39 +00:00
|
|
|
uint32_t decoded = convert_from_line_code(
|
|
|
|
raw, sizeof(raw), bits, numbytes, off, "01", "10"); /* Manchester. */
|
2023-01-18 17:18:19 +00:00
|
|
|
FURI_LOG_E(TAG, "Citroen TPMS decoded bits: %lu", decoded);
|
|
|
|
|
2023-01-18 19:25:39 +00:00
|
|
|
if(decoded < 8 * 10) return false; /* Require the full 10 bytes. */
|
2023-01-18 17:18:19 +00:00
|
|
|
|
|
|
|
/* Check the CRC. It's a simple XOR of bytes 1-9, the first byte
|
|
|
|
* is not included. The meaning of the first byte is unknown and
|
|
|
|
* we don't display it. */
|
|
|
|
uint8_t crc = 0;
|
2023-01-18 19:25:39 +00:00
|
|
|
for(int j = 1; j < 10; j++) crc ^= raw[j];
|
|
|
|
if(crc != 0) return false; /* Require sane checksum. */
|
2023-01-18 17:18:19 +00:00
|
|
|
|
2023-01-18 19:25:39 +00:00
|
|
|
info->pulses_count = (off + 8 * 10 * 2) - info->start_off;
|
2023-01-18 17:18:19 +00:00
|
|
|
|
|
|
|
int repeat = raw[5] & 0xf;
|
2023-01-18 19:25:39 +00:00
|
|
|
float kpa = (float)raw[6] * 1.364;
|
|
|
|
int temp = raw[7] - 50;
|
2023-01-18 17:18:19 +00:00
|
|
|
int battery = raw[8]; /* This may be the battery. It's not clear. */
|
|
|
|
|
2023-01-18 19:25:39 +00:00
|
|
|
snprintf(info->name, sizeof(info->name), "%s", "Citroen TPMS");
|
|
|
|
snprintf(
|
|
|
|
info->raw,
|
|
|
|
sizeof(info->raw),
|
2023-01-18 17:18:19 +00:00
|
|
|
"%02X%02X%02X%02X%02X%02X%02X%02X%02X%02X",
|
2023-01-18 19:25:39 +00:00
|
|
|
raw[0],
|
|
|
|
raw[1],
|
|
|
|
raw[2],
|
|
|
|
raw[3],
|
|
|
|
raw[4],
|
|
|
|
raw[5],
|
|
|
|
raw[6],
|
|
|
|
raw[7],
|
|
|
|
raw[8],
|
|
|
|
raw[9]);
|
|
|
|
snprintf(
|
|
|
|
info->info1,
|
|
|
|
sizeof(info->info1),
|
|
|
|
"Tire ID %02X%02X%02X%02X",
|
|
|
|
raw[1],
|
|
|
|
raw[2],
|
|
|
|
raw[3],
|
|
|
|
raw[4]);
|
|
|
|
snprintf(info->info2, sizeof(info->info2), "Pressure %.2f kpa", (double)kpa);
|
|
|
|
snprintf(info->info3, sizeof(info->info3), "Temperature %d C", temp);
|
|
|
|
snprintf(info->info4, sizeof(info->info4), "Repeat %d, Bat %d", repeat, battery);
|
2023-01-18 17:18:19 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-01-18 19:25:39 +00:00
|
|
|
ProtoViewDecoder CitroenTPMSDecoder = {"Citroen TPMS", decode};
|