renault.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. /* Renault tires TPMS. Usually 443.92 Mhz FSK.
  2. *
  3. * Preamble + sync + Manchester bits. ~48us short pulse.
  4. * 9 Bytes in total not counting the preamble. */
  5. #include "../../app.h"
  6. #define USE_TEST_VECTOR 0
  7. static const char *test_vector =
  8. "...01010101010101010110" // Preamble + sync
  9. /* The following is Marshal encoded, so each two characters are
  10. * actaully one bit. 01 = 0, 10 = 1. */
  11. "010110010110" // Flags.
  12. "10011001101010011001" // Pressure, multiply by 0.75 to obtain kpa.
  13. // 244 kpa here.
  14. "1010010110011010" // Temperature, subtract 30 to obtain celsius. 22C here.
  15. "1001010101101001"
  16. "0101100110010101"
  17. "1001010101100110" // Tire ID. 0x7AD779 here.
  18. "0101010101010101"
  19. "0101010101010101" // Two FF bytes (usually). Unknown.
  20. "0110010101010101"; // CRC8 with (poly 7, initialization 0).
  21. static bool decode(uint8_t *bits, uint32_t numbytes, uint32_t numbits, ProtoViewMsgInfo *info) {
  22. if (USE_TEST_VECTOR) { /* Test vector to check that decoding works. */
  23. bitmap_set_pattern(bits,numbytes,0,test_vector);
  24. numbits = strlen(test_vector);
  25. }
  26. if (numbits-12 < 9*8) return false;
  27. const char *sync_pattern = "01010101010101010110";
  28. uint64_t off = bitmap_seek_bits(bits,numbytes,0,numbits,sync_pattern);
  29. if (off == BITMAP_SEEK_NOT_FOUND) return false;
  30. FURI_LOG_E(TAG, "Renault TPMS preamble+sync found");
  31. info->start_off = off;
  32. off += 20; /* Skip preamble. */
  33. uint8_t raw[9];
  34. uint32_t decoded =
  35. convert_from_line_code(raw,sizeof(raw),bits,numbytes,off,
  36. "01","10"); /* Manchester. */
  37. FURI_LOG_E(TAG, "Renault TPMS decoded bits: %lu", decoded);
  38. if (decoded < 8*9) return false; /* Require the full 9 bytes. */
  39. if (crc8(raw,8,0,7) != raw[8]) return false; /* Require sane CRC. */
  40. info->pulses_count = (off+8*9*2) - info->start_off;
  41. uint8_t flags = raw[0]>>2;
  42. float kpa = 0.75 * ((uint32_t)((raw[0]&3)<<8) | raw[1]);
  43. int temp = raw[2]-30;
  44. fieldset_add_bytes(info->fieldset,"Tire ID",raw+3,3*2);
  45. fieldset_add_float(info->fieldset,"Pressure kpa",kpa,2);
  46. fieldset_add_int(info->fieldset,"Temperature C",temp,8);
  47. fieldset_add_hex(info->fieldset,"Flags",flags,6);
  48. fieldset_add_bytes(info->fieldset,"Unknown1",raw+6,2);
  49. fieldset_add_bytes(info->fieldset,"Unknown2",raw+7,2);
  50. return true;
  51. }
  52. /* Give fields and defaults for the signal creator. */
  53. static void get_fields(ProtoViewFieldSet *fieldset) {
  54. uint8_t default_id[3]= {0xAB, 0xCD, 0xEF};
  55. fieldset_add_bytes(fieldset,"Tire ID",default_id,3*2);
  56. fieldset_add_float(fieldset,"Pressure kpa",123,2);
  57. fieldset_add_int(fieldset,"Temperature C",20,8);
  58. // We don't know what flags are, but 1B is a common value.
  59. fieldset_add_hex(fieldset,"Flags",0x1b,6);
  60. fieldset_add_bytes(fieldset,"Unknown1",(uint8_t*)"\xff",2);
  61. fieldset_add_bytes(fieldset,"Unknown2",(uint8_t*)"\xff",2);
  62. }
  63. /* Create a Renault TPMS signal, according to the fields provided. */
  64. static void build_message(RawSamplesBuffer *samples, ProtoViewFieldSet *fieldset)
  65. {
  66. uint32_t te = 50; // Short pulse duration in microseconds.
  67. // Preamble + sync
  68. const char *psync = "01010101010101010101010101010110";
  69. const char *p = psync;
  70. while(*p) {
  71. raw_samples_add_or_update(samples,*p == '1',te);
  72. p++;
  73. }
  74. // Data, 9 bytes
  75. uint8_t data[9] = {0};
  76. unsigned int raw_pressure = fieldset->fields[1]->fvalue * 4 / 3;
  77. data[0] = fieldset->fields[3]->uvalue << 2; // Flags
  78. data[0] |= (raw_pressure >> 8) & 3; // Pressure kpa high 2 bits
  79. data[1] = raw_pressure & 0xff; // Pressure kpa low 8 bits
  80. data[2] = fieldset->fields[2]->value + 30; // Temperature C
  81. memcpy(data+3,fieldset->fields[0]->bytes,6); // ID, 24 bits.
  82. data[6] = fieldset->fields[4]->bytes[0]; // Unknown 1
  83. data[7] = fieldset->fields[5]->bytes[0]; // Unknown 2
  84. data[8] = crc8(data,8,0,7);
  85. // Generate Manchester code for each bit
  86. for (uint32_t j = 0; j < 9*8; j++) {
  87. if (bitmap_get(data,sizeof(data),j)) {
  88. raw_samples_add_or_update(samples,true,te);
  89. raw_samples_add_or_update(samples,false,te);
  90. } else {
  91. raw_samples_add_or_update(samples,false,te);
  92. raw_samples_add_or_update(samples,true,te);
  93. }
  94. }
  95. }
  96. ProtoViewDecoder RenaultTPMSDecoder = {
  97. .name = "Renault TPMS",
  98. .decode = decode,
  99. .get_fields = get_fields,
  100. .build_message = build_message
  101. };