ford.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172
  1. /* Ford tires TPMS. Usually 443.92 Mhz FSK (in Europe).
  2. *
  3. * 52 us short pules
  4. * Preamble: 0101010101010101010101010101
  5. * Sync: 0110 (that is 52 us gap + 104 us pulse + 52 us gap)
  6. * Data: 8 bytes Manchester encoded
  7. * 01 = zero
  8. * 10 = one
  9. */
  10. #include "../../app.h"
  11. static bool decode(uint8_t *bits, uint32_t numbytes, uint32_t numbits, ProtoViewMsgInfo *info) {
  12. const char *sync_pattern = "010101010101" "0110";
  13. uint8_t sync_len = 12+4; /* We just use 12 preamble symbols + sync. */
  14. if (numbits-sync_len < 8*8) return false;
  15. uint64_t off = bitmap_seek_bits(bits,numbytes,0,numbits,sync_pattern);
  16. if (off == BITMAP_SEEK_NOT_FOUND) return false;
  17. FURI_LOG_E(TAG, "Fort TPMS preamble+sync found");
  18. info->start_off = off;
  19. off += sync_len; /* Skip preamble and sync. */
  20. uint8_t raw[8];
  21. uint32_t decoded =
  22. convert_from_line_code(raw,sizeof(raw),bits,numbytes,off,
  23. "01","10"); /* Manchester. */
  24. FURI_LOG_E(TAG, "Ford TPMS decoded bits: %lu", decoded);
  25. if (decoded < 8*8) return false; /* Require the full 8 bytes. */
  26. /* CRC is just the sum of the first 7 bytes MOD 256. */
  27. uint8_t crc = 0;
  28. for (int j = 0; j < 7; j++) crc += raw[j];
  29. if (crc != raw[7]) return false; /* Require sane CRC. */
  30. info->pulses_count = (off+8*8*2) - info->start_off;
  31. float psi = 0.25 * (((raw[6]&0x20)<<3)|raw[4]);
  32. /* Temperature apperas to be valid only if the most significant
  33. * bit of the value is not set. Otherwise its meaning is unknown.
  34. * Likely useful to alternatively send temperature or other info. */
  35. int temp = raw[5] & 0x80 ? 0 : raw[5]-56;
  36. int flags = raw[5] & 0x7f;
  37. int car_moving = (raw[6] & 0x44) == 0x44;
  38. #if 0
  39. snprintf(info->name,sizeof(info->name),"%s","Ford TPMS");
  40. snprintf(info->raw,sizeof(info->raw),"%02X%02X%02X%02X%02X%02X%02X%02X",
  41. raw[0],raw[1],raw[2],raw[3],raw[4],raw[5],
  42. raw[6],raw[7]);
  43. snprintf(info->info1,sizeof(info->info1),"Tire ID %02X%02X%02X%02X",
  44. raw[0],raw[1],raw[2],raw[3]);
  45. snprintf(info->info2,sizeof(info->info2),"Pressure %.2f psi", (double)psi);
  46. if (temp)
  47. snprintf(info->info3,sizeof(info->info3),"Temperature %d C", temp);
  48. else
  49. snprintf(info->info3,sizeof(info->info3),"Flags %d", flags);
  50. snprintf(info->info4,sizeof(info->info4),"Moving %s", car_moving ? "yes" : "no");
  51. #endif
  52. return true;
  53. }
  54. ProtoViewDecoder FordTPMSDecoder = {
  55. .name = "Ford TPMS",
  56. .decode = decode,
  57. .get_fields = NULL,
  58. .build_message = NULL
  59. };