subghz_protocol_keeloq.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. #include "subghz_protocol_keeloq.h"
  2. #include "subghz_protocol_keeloq_common.h"
  3. #include "../subghz_keystore.h"
  4. #include <furi.h>
  5. #include <m-string.h>
  6. struct SubGhzProtocolKeeloq {
  7. SubGhzProtocolCommon common;
  8. SubGhzKeystore* keystore;
  9. const char* manufacture_name;
  10. };
  11. SubGhzProtocolKeeloq* subghz_protocol_keeloq_alloc(SubGhzKeystore* keystore) {
  12. SubGhzProtocolKeeloq* instance = furi_alloc(sizeof(SubGhzProtocolKeeloq));
  13. instance->keystore = keystore;
  14. instance->common.name = "KeeLoq";
  15. instance->common.code_min_count_bit_for_found = 64;
  16. instance->common.te_short = 400;
  17. instance->common.te_long = 800;
  18. instance->common.te_delta = 140;
  19. instance->common.type_protocol = TYPE_PROTOCOL_DYNAMIC;
  20. instance->common.to_string = (SubGhzProtocolCommonToStr)subghz_protocol_keeloq_to_str;
  21. instance->common.to_save_string =
  22. (SubGhzProtocolCommonGetStrSave)subghz_protocol_keeloq_to_save_str;
  23. instance->common.to_load_protocol_from_file =
  24. (SubGhzProtocolCommonLoadFromFile)subghz_protocol_keeloq_to_load_protocol_from_file;
  25. instance->common.to_load_protocol =
  26. (SubGhzProtocolCommonLoadFromRAW)subghz_decoder_keeloq_to_load_protocol;
  27. instance->common.get_upload_protocol =
  28. (SubGhzProtocolCommonEncoderGetUpLoad)subghz_protocol_keeloq_send_key;
  29. return instance;
  30. }
  31. void subghz_protocol_keeloq_free(SubGhzProtocolKeeloq* instance) {
  32. furi_assert(instance);
  33. free(instance);
  34. }
  35. /** Checking the accepted code against the database manafacture key
  36. *
  37. * @param instance SubGhzProtocolKeeloq instance
  38. * @param fix fix part of the parcel
  39. * @param hop hop encrypted part of the parcel
  40. * @return true on successful search
  41. */
  42. uint8_t subghz_protocol_keeloq_check_remote_controller_selector(
  43. SubGhzProtocolKeeloq* instance,
  44. uint32_t fix,
  45. uint32_t hop) {
  46. uint16_t end_serial = (uint16_t)(fix & 0x3FF);
  47. uint8_t btn = (uint8_t)(fix >> 28);
  48. uint32_t decrypt = 0;
  49. uint64_t man_normal_learning;
  50. for
  51. M_EACH(manufacture_code, *subghz_keystore_get_data(instance->keystore), SubGhzKeyArray_t) {
  52. switch(manufacture_code->type) {
  53. case KEELOQ_LEARNING_SIMPLE:
  54. //Simple Learning
  55. decrypt = subghz_protocol_keeloq_common_decrypt(hop, manufacture_code->key);
  56. if((decrypt >> 28 == btn) &&
  57. (((((uint16_t)(decrypt >> 16)) & 0x3FF) == end_serial) ||
  58. ((((uint16_t)(decrypt >> 16)) & 0x3FF) == 0))) {
  59. instance->manufacture_name = string_get_cstr(manufacture_code->name);
  60. instance->common.cnt = decrypt & 0x0000FFFF;
  61. return 1;
  62. }
  63. break;
  64. case KEELOQ_LEARNING_NORMAL:
  65. // Normal_Learning
  66. // https://phreakerclub.com/forum/showpost.php?p=43557&postcount=37
  67. man_normal_learning =
  68. subghz_protocol_keeloq_common_normal_learning(fix, manufacture_code->key);
  69. decrypt = subghz_protocol_keeloq_common_decrypt(hop, man_normal_learning);
  70. if((decrypt >> 28 == btn) &&
  71. (((((uint16_t)(decrypt >> 16)) & 0x3FF) == end_serial) ||
  72. ((((uint16_t)(decrypt >> 16)) & 0x3FF) == 0))) {
  73. instance->manufacture_name = string_get_cstr(manufacture_code->name);
  74. instance->common.cnt = decrypt & 0x0000FFFF;
  75. return 1;
  76. }
  77. break;
  78. case KEELOQ_LEARNING_UNKNOWN:
  79. // Simple Learning
  80. decrypt = subghz_protocol_keeloq_common_decrypt(hop, manufacture_code->key);
  81. if((decrypt >> 28 == btn) &&
  82. (((((uint16_t)(decrypt >> 16)) & 0x3FF) == end_serial) ||
  83. ((((uint16_t)(decrypt >> 16)) & 0x3FF) == 0))) {
  84. instance->manufacture_name = string_get_cstr(manufacture_code->name);
  85. instance->common.cnt = decrypt & 0x0000FFFF;
  86. return 1;
  87. }
  88. // Check for mirrored man
  89. uint64_t man_rev = 0;
  90. uint64_t man_rev_byte = 0;
  91. for(uint8_t i = 0; i < 64; i += 8) {
  92. man_rev_byte = (uint8_t)(manufacture_code->key >> i);
  93. man_rev = man_rev | man_rev_byte << (56 - i);
  94. }
  95. decrypt = subghz_protocol_keeloq_common_decrypt(hop, man_rev);
  96. if((decrypt >> 28 == btn) &&
  97. (((((uint16_t)(decrypt >> 16)) & 0x3FF) == end_serial) ||
  98. ((((uint16_t)(decrypt >> 16)) & 0x3FF) == 0))) {
  99. instance->manufacture_name = string_get_cstr(manufacture_code->name);
  100. instance->common.cnt = decrypt & 0x0000FFFF;
  101. return 1;
  102. }
  103. //###########################
  104. // Normal_Learning
  105. // https://phreakerclub.com/forum/showpost.php?p=43557&postcount=37
  106. man_normal_learning =
  107. subghz_protocol_keeloq_common_normal_learning(fix, manufacture_code->key);
  108. decrypt = subghz_protocol_keeloq_common_decrypt(hop, man_normal_learning);
  109. if((decrypt >> 28 == btn) &&
  110. (((((uint16_t)(decrypt >> 16)) & 0x3FF) == end_serial) ||
  111. ((((uint16_t)(decrypt >> 16)) & 0x3FF) == 0))) {
  112. instance->manufacture_name = string_get_cstr(manufacture_code->name);
  113. instance->common.cnt = decrypt & 0x0000FFFF;
  114. return 1;
  115. }
  116. // Check for mirrored man
  117. man_rev = 0;
  118. man_rev_byte = 0;
  119. for(uint8_t i = 0; i < 64; i += 8) {
  120. man_rev_byte = (uint8_t)(manufacture_code->key >> i);
  121. man_rev = man_rev | man_rev_byte << (56 - i);
  122. }
  123. man_normal_learning = subghz_protocol_keeloq_common_normal_learning(fix, man_rev);
  124. decrypt = subghz_protocol_keeloq_common_decrypt(hop, man_normal_learning);
  125. if((decrypt >> 28 == btn) &&
  126. (((((uint16_t)(decrypt >> 16)) & 0x3FF) == end_serial) ||
  127. ((((uint16_t)(decrypt >> 16)) & 0x3FF) == 0))) {
  128. instance->manufacture_name = string_get_cstr(manufacture_code->name);
  129. instance->common.cnt = decrypt & 0x0000FFFF;
  130. return 1;
  131. }
  132. break;
  133. }
  134. }
  135. instance->manufacture_name = "Unknown";
  136. instance->common.cnt = 0;
  137. return 0;
  138. }
  139. /** Analysis of received data
  140. *
  141. * @param instance SubGhzProtocolKeeloq instance
  142. */
  143. void subghz_protocol_keeloq_check_remote_controller(SubGhzProtocolKeeloq* instance) {
  144. uint64_t key = subghz_protocol_common_reverse_key(
  145. instance->common.code_last_found, instance->common.code_last_count_bit);
  146. uint32_t key_fix = key >> 32;
  147. uint32_t key_hop = key & 0x00000000ffffffff;
  148. // Check key AN-Motors
  149. if((key_hop >> 24) == ((key_hop >> 16) & 0x00ff) &&
  150. (key_fix >> 28) == ((key_hop >> 12) & 0x0f) && (key_hop & 0xFFF) == 0x404) {
  151. instance->manufacture_name = "AN-Motors";
  152. instance->common.cnt = key_hop >> 16;
  153. } else if((key_hop & 0xFFF) == (0x000) && (key_fix >> 28) == ((key_hop >> 12) & 0x0f)) {
  154. instance->manufacture_name = "HCS101";
  155. instance->common.cnt = key_hop >> 16;
  156. } else {
  157. subghz_protocol_keeloq_check_remote_controller_selector(instance, key_fix, key_hop);
  158. }
  159. instance->common.serial = key_fix & 0x0FFFFFFF;
  160. instance->common.btn = key_fix >> 28;
  161. }
  162. const char* subghz_protocol_keeloq_get_manufacture_name(void* context) {
  163. SubGhzProtocolKeeloq* instance = context;
  164. subghz_protocol_keeloq_check_remote_controller(instance);
  165. return instance->manufacture_name;
  166. }
  167. void subghz_protocol_keeloq_set_manufacture_name(void* context, const char* manufacture_name) {
  168. SubGhzProtocolKeeloq* instance = context;
  169. instance->manufacture_name = manufacture_name;
  170. }
  171. uint64_t subghz_protocol_keeloq_gen_key(void* context) {
  172. SubGhzProtocolKeeloq* instance = context;
  173. uint32_t fix = instance->common.btn << 28 | instance->common.serial;
  174. uint32_t decrypt = instance->common.btn << 28 | (instance->common.serial & 0x3FF) << 16 |
  175. instance->common.cnt;
  176. uint32_t hop = 0;
  177. uint64_t man_normal_learning = 0;
  178. int res = 0;
  179. for
  180. M_EACH(manufacture_code, *subghz_keystore_get_data(instance->keystore), SubGhzKeyArray_t) {
  181. res = strcmp(string_get_cstr(manufacture_code->name), instance->manufacture_name);
  182. if(res == 0) {
  183. switch(manufacture_code->type) {
  184. case KEELOQ_LEARNING_SIMPLE:
  185. //Simple Learning
  186. hop = subghz_protocol_keeloq_common_encrypt(decrypt, manufacture_code->key);
  187. break;
  188. case KEELOQ_LEARNING_NORMAL:
  189. //Simple Learning
  190. man_normal_learning =
  191. subghz_protocol_keeloq_common_normal_learning(fix, manufacture_code->key);
  192. hop = subghz_protocol_keeloq_common_encrypt(decrypt, man_normal_learning);
  193. break;
  194. case KEELOQ_LEARNING_UNKNOWN:
  195. hop = 0; //todo
  196. break;
  197. }
  198. break;
  199. }
  200. }
  201. uint64_t yek = (uint64_t)fix << 32 | hop;
  202. return subghz_protocol_common_reverse_key(yek, instance->common.code_last_count_bit);
  203. }
  204. bool subghz_protocol_keeloq_send_key(
  205. SubGhzProtocolKeeloq* instance,
  206. SubGhzProtocolCommonEncoder* encoder) {
  207. furi_assert(instance);
  208. furi_assert(encoder);
  209. //gen new key
  210. instance->common.cnt++;
  211. instance->common.code_last_found = subghz_protocol_keeloq_gen_key(instance);
  212. if(instance->common.callback)
  213. instance->common.callback((SubGhzProtocolCommon*)instance, instance->common.context);
  214. size_t index = 0;
  215. encoder->size_upload = 11 * 2 + 2 + (instance->common.code_last_count_bit * 2) + 4;
  216. if(encoder->size_upload > SUBGHZ_ENCODER_UPLOAD_MAX_SIZE) return false;
  217. //Send header
  218. for(uint8_t i = 11; i > 0; i--) {
  219. encoder->upload[index++] = level_duration_make(true, (uint32_t)instance->common.te_short);
  220. encoder->upload[index++] = level_duration_make(false, (uint32_t)instance->common.te_short);
  221. }
  222. encoder->upload[index++] = level_duration_make(true, (uint32_t)instance->common.te_short);
  223. encoder->upload[index++] =
  224. level_duration_make(false, (uint32_t)instance->common.te_short * 10);
  225. //Send key data
  226. for(uint8_t i = instance->common.code_last_count_bit; i > 0; i--) {
  227. if(bit_read(instance->common.code_last_found, i - 1)) {
  228. //send bit 1
  229. encoder->upload[index++] =
  230. level_duration_make(true, (uint32_t)instance->common.te_short);
  231. encoder->upload[index++] =
  232. level_duration_make(false, (uint32_t)instance->common.te_long);
  233. } else {
  234. //send bit 0
  235. encoder->upload[index++] =
  236. level_duration_make(true, (uint32_t)instance->common.te_long);
  237. encoder->upload[index++] =
  238. level_duration_make(false, (uint32_t)instance->common.te_short);
  239. }
  240. }
  241. // +send 2 status bit
  242. encoder->upload[index++] = level_duration_make(true, (uint32_t)instance->common.te_short);
  243. encoder->upload[index++] = level_duration_make(false, (uint32_t)instance->common.te_long);
  244. //encoder->upload[index++] = level_duration_make(true, (uint32_t)instance->common.te_long);
  245. //encoder->upload[index++] = level_duration_make(false, (uint32_t)instance->common.te_short);
  246. // send end
  247. encoder->upload[index++] = level_duration_make(true, (uint32_t)instance->common.te_short);
  248. encoder->upload[index++] =
  249. level_duration_make(false, (uint32_t)instance->common.te_short * 40);
  250. return true;
  251. }
  252. void subghz_protocol_keeloq_reset(SubGhzProtocolKeeloq* instance) {
  253. instance->common.parser_step = 0;
  254. }
  255. void subghz_protocol_keeloq_parse(SubGhzProtocolKeeloq* instance, bool level, uint32_t duration) {
  256. switch(instance->common.parser_step) {
  257. case 0:
  258. if((level) &&
  259. DURATION_DIFF(duration, instance->common.te_short) < instance->common.te_delta) {
  260. instance->common.parser_step = 1;
  261. instance->common.header_count++;
  262. } else {
  263. instance->common.parser_step = 0;
  264. }
  265. break;
  266. case 1:
  267. if((!level) &&
  268. (DURATION_DIFF(duration, instance->common.te_short) < instance->common.te_delta)) {
  269. instance->common.parser_step = 0;
  270. break;
  271. }
  272. if((instance->common.header_count > 2) &&
  273. (DURATION_DIFF(duration, instance->common.te_short * 10) <
  274. instance->common.te_delta * 10)) {
  275. // Found header
  276. instance->common.parser_step = 2;
  277. instance->common.code_found = 0;
  278. instance->common.code_count_bit = 0;
  279. } else {
  280. instance->common.parser_step = 0;
  281. instance->common.header_count = 0;
  282. }
  283. break;
  284. case 2:
  285. if(level) {
  286. instance->common.te_last = duration;
  287. instance->common.parser_step = 3;
  288. }
  289. break;
  290. case 3:
  291. if(!level) {
  292. if(duration >= (instance->common.te_short * 2 + instance->common.te_delta)) {
  293. // Found end TX
  294. instance->common.parser_step = 0;
  295. if(instance->common.code_count_bit >=
  296. instance->common.code_min_count_bit_for_found) {
  297. if(instance->common.code_last_found != instance->common.code_found) {
  298. instance->common.code_last_found = instance->common.code_found;
  299. instance->common.code_last_count_bit = instance->common.code_count_bit;
  300. if(instance->common.callback)
  301. instance->common.callback(
  302. (SubGhzProtocolCommon*)instance, instance->common.context);
  303. }
  304. instance->common.code_found = 0;
  305. instance->common.code_count_bit = 0;
  306. instance->common.header_count = 0;
  307. }
  308. break;
  309. } else if(
  310. (DURATION_DIFF(instance->common.te_last, instance->common.te_short) <
  311. instance->common.te_delta) &&
  312. (DURATION_DIFF(duration, instance->common.te_long) < instance->common.te_delta)) {
  313. if(instance->common.code_count_bit <
  314. instance->common.code_min_count_bit_for_found) {
  315. subghz_protocol_common_add_bit(&instance->common, 1);
  316. }
  317. instance->common.parser_step = 2;
  318. } else if(
  319. (DURATION_DIFF(instance->common.te_last, instance->common.te_long) <
  320. instance->common.te_delta) &&
  321. (DURATION_DIFF(duration, instance->common.te_short) < instance->common.te_delta)) {
  322. if(instance->common.code_count_bit <
  323. instance->common.code_min_count_bit_for_found) {
  324. subghz_protocol_common_add_bit(&instance->common, 0);
  325. }
  326. instance->common.parser_step = 2;
  327. } else {
  328. instance->common.parser_step = 0;
  329. instance->common.header_count = 0;
  330. }
  331. } else {
  332. instance->common.parser_step = 0;
  333. instance->common.header_count = 0;
  334. }
  335. break;
  336. }
  337. }
  338. void subghz_protocol_keeloq_to_str(SubGhzProtocolKeeloq* instance, string_t output) {
  339. subghz_protocol_keeloq_check_remote_controller(instance);
  340. uint32_t code_found_hi = instance->common.code_last_found >> 32;
  341. uint32_t code_found_lo = instance->common.code_last_found & 0x00000000ffffffff;
  342. uint64_t code_found_reverse = subghz_protocol_common_reverse_key(
  343. instance->common.code_last_found, instance->common.code_last_count_bit);
  344. uint32_t code_found_reverse_hi = code_found_reverse >> 32;
  345. uint32_t code_found_reverse_lo = code_found_reverse & 0x00000000ffffffff;
  346. string_cat_printf(
  347. output,
  348. "%s %dbit\r\n"
  349. "Key:0x%lX%lX\r\n"
  350. "Fix:0x%08lX Cnt:%04X\r\n"
  351. "Hop:0x%08lX Btn:%02lX\r\n"
  352. "MF:%s\r\n"
  353. "Sn:0x%07lX \r\n",
  354. instance->common.name,
  355. instance->common.code_last_count_bit,
  356. code_found_hi,
  357. code_found_lo,
  358. code_found_reverse_hi,
  359. instance->common.cnt,
  360. code_found_reverse_lo,
  361. instance->common.btn,
  362. instance->manufacture_name,
  363. instance->common.serial
  364. );
  365. }
  366. void subghz_protocol_keeloq_to_save_str(SubGhzProtocolKeeloq* instance, string_t output) {
  367. string_printf(
  368. output,
  369. "Protocol: %s\n"
  370. "Bit: %d\n"
  371. "Key: %08lX%08lX\n",
  372. instance->common.name,
  373. instance->common.code_last_count_bit,
  374. (uint32_t)(instance->common.code_last_found >> 32),
  375. (uint32_t)(instance->common.code_last_found & 0xFFFFFFFF));
  376. }
  377. bool subghz_protocol_keeloq_to_load_protocol_from_file(
  378. FileWorker* file_worker,
  379. SubGhzProtocolKeeloq* instance) {
  380. bool loaded = false;
  381. string_t temp_str;
  382. string_init(temp_str);
  383. int res = 0;
  384. int data = 0;
  385. do {
  386. // Read and parse bit data from 2nd line
  387. if(!file_worker_read_until(file_worker, temp_str, '\n')) {
  388. break;
  389. }
  390. res = sscanf(string_get_cstr(temp_str), "Bit: %d\n", &data);
  391. if(res != 1) {
  392. break;
  393. }
  394. instance->common.code_last_count_bit = (uint8_t)data;
  395. // Read and parse key data from 3nd line
  396. if(!file_worker_read_until(file_worker, temp_str, '\n')) {
  397. break;
  398. }
  399. // strlen("Key: ") = 5
  400. string_right(temp_str, 5);
  401. uint8_t buf_key[8] = {0};
  402. if(!subghz_protocol_common_read_hex(temp_str, buf_key, 8)) {
  403. break;
  404. }
  405. for(uint8_t i = 0; i < 8; i++) {
  406. instance->common.code_last_found = instance->common.code_last_found << 8 | buf_key[i];
  407. }
  408. loaded = true;
  409. } while(0);
  410. string_clear(temp_str);
  411. return loaded;
  412. }
  413. void subghz_decoder_keeloq_to_load_protocol(
  414. SubGhzProtocolKeeloq* instance,
  415. void* context) {
  416. furi_assert(context);
  417. furi_assert(instance);
  418. SubGhzProtocolCommonLoad* data = context;
  419. instance->common.code_last_found = data->code_found;
  420. instance->common.code_last_count_bit = data->code_count_bit;
  421. subghz_protocol_keeloq_check_remote_controller(instance);
  422. }