keeloq.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. /* Microchip HCS200/HCS300/HSC301 KeeLoq, rolling code remotes.
  2. *
  3. * Usually 443.92 Mhz OOK, ~200us or ~400us pulse len, depending
  4. * on the configuration.
  5. *
  6. * Preamble: 12 pairs of alternating pulse/gap.
  7. * Sync: long gap of around 10 times the duration of the short-pulse.
  8. * Data: pulse width encoded data. Each bit takes three cycles:
  9. *
  10. * 0 = 110
  11. * 1 = 100
  12. *
  13. * There are a total of 66 bits transmitted.
  14. * 0..31: 32 bits of encrypted rolling code.
  15. * 32..59: Remote ID, 28 bits
  16. * 60..63: Buttons pressed
  17. * 64..64: Low battery if set
  18. * 65..65: Always set to 1
  19. *
  20. * Bits in bytes are inverted: least significant bit is first.
  21. * For some reason there is no checksum whatsoever, so we only decode
  22. * if we find everything well formed.
  23. */
  24. #include "../app.h"
  25. static bool decode(uint8_t *bits, uint32_t numbytes, uint32_t numbits, ProtoViewMsgInfo *info) {
  26. /* In the sync pattern, we require the 12 high/low pulses and at least
  27. * half the gap we expect (5 pulses times, one is the final zero in the
  28. * 24 symbols high/low sequence, then other 4). */
  29. const char *sync_pattern = "101010101010101010101010" "0000";
  30. uint8_t sync_len = 24+4;
  31. if (numbits-sync_len+sync_len < 3*66) return false;
  32. uint32_t off = bitmap_seek_bits(bits,numbytes,0,numbits,sync_pattern);
  33. if (off == BITMAP_SEEK_NOT_FOUND) return false;
  34. info->start_off = off;
  35. off += sync_len; // Seek start of message.
  36. /* Now there is half the gap left, but we allow from 3 to 7, instead of 5
  37. * symbols of gap, to avoid missing the signal for a matter of wrong
  38. * timing. */
  39. uint8_t gap_len = 0;
  40. while(gap_len <= 7 && bitmap_get(bits,numbytes,off+gap_len) == 0)
  41. gap_len++;
  42. if (gap_len < 3 || gap_len > 7) return false;
  43. off += gap_len;
  44. FURI_LOG_E(TAG, "Keeloq preamble+sync found");
  45. uint8_t raw[9] = {0};
  46. uint32_t decoded =
  47. convert_from_line_code(raw,sizeof(raw),bits,numbytes,off,
  48. "110","100"); /* Pulse width modulation. */
  49. FURI_LOG_E(TAG, "Keeloq decoded bits: %lu", decoded);
  50. if (decoded < 66) return false; /* Require the full 66 bits. */
  51. info->pulses_count = (off+66*3) - info->start_off;
  52. bitmap_reverse_bytes_bits(raw,sizeof(raw)); /* Keeloq is LSB first. */
  53. int buttons = raw[7]>>4;
  54. int lowbat = (raw[8]&0x1) == 0; // Actual bit meaning: good battery level
  55. int alwaysone = (raw[8]&0x2) != 0;
  56. fieldset_add_bytes(info->fieldset,"encr",raw,4*2);
  57. raw[7] = raw[7]<<4; // Make ID bits contiguous
  58. fieldset_add_bytes(info->fieldset,"id",raw+4,7); // 28 bits, 7 nibbles
  59. fieldset_add_bin(info->fieldset,"s2 s1 s0 s3",buttons,4);
  60. fieldset_add_bin(info->fieldset,"low battery",lowbat,0);
  61. fieldset_add_bin(info->fieldset,"always one",alwaysone,1);
  62. return true;
  63. }
  64. static void get_fields(ProtoViewFieldSet *fieldset) {
  65. uint8_t remote_id[4] = {0xab, 0xcd, 0xef, 0xa0};
  66. uint8_t encr[4] = {0xab, 0xab, 0xab, 0xab};
  67. fieldset_add_bytes(fieldset,"encr",encr,8);
  68. fieldset_add_bytes(fieldset,"id",remote_id,7);
  69. fieldset_add_bin(fieldset,"s[2,1,0,3]",2,4);
  70. fieldset_add_bin(fieldset,"low battery",0,1);
  71. fieldset_add_bin(fieldset,"always one",1,1);
  72. }
  73. static void build_message(RawSamplesBuffer *samples, ProtoViewFieldSet *fieldset)
  74. {
  75. uint32_t te = 380; // Short pulse duration in microseconds.
  76. // Sync: 12 pairs of pulse/gap + 9 times gap
  77. for (int j = 0; j < 12; j++) {
  78. raw_samples_add(samples,true,te);
  79. raw_samples_add(samples,false,te);
  80. }
  81. raw_samples_add(samples,false,te*9);
  82. // Data, 66 bits.
  83. uint8_t data[9] = {0};
  84. memcpy(data,fieldset->fields[0]->bytes,4); // Encrypted part.
  85. memcpy(data+4,fieldset->fields[1]->bytes,4); // ID.
  86. data[7] = data[7]>>4 | fieldset->fields[2]->uvalue << 4; // s[2,1,0,3]
  87. int low_battery = fieldset->fields[3] != 0;
  88. int always_one = fieldset->fields[4] != 0;
  89. low_battery = !low_battery; // Bit real meaning is good battery level.
  90. data[8] |= low_battery;
  91. data[8] |= (always_one << 1);
  92. bitmap_reverse_bytes_bits(data,sizeof(data)); /* Keeloq is LSB first. */
  93. for (int j = 0; j < 66; j++) {
  94. if (bitmap_get(data,9,j)) {
  95. raw_samples_add(samples,true,te);
  96. raw_samples_add(samples,false,te*2);
  97. } else {
  98. raw_samples_add(samples,true,te*2);
  99. raw_samples_add(samples,false,te);
  100. }
  101. }
  102. }
  103. ProtoViewDecoder KeeloqDecoder = {
  104. .name = "Keeloq",
  105. .decode = decode,
  106. .get_fields = get_fields,
  107. .build_message = build_message
  108. };