ibutton_worker_modes.c 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. #include <furi.h>
  2. #include <furi_hal.h>
  3. #include "ibutton_worker_i.h"
  4. #include "ibutton_key_command.h"
  5. void ibutton_worker_mode_idle_start(iButtonWorker* worker);
  6. void ibutton_worker_mode_idle_tick(iButtonWorker* worker);
  7. void ibutton_worker_mode_idle_stop(iButtonWorker* worker);
  8. void ibutton_worker_mode_emulate_start(iButtonWorker* worker);
  9. void ibutton_worker_mode_emulate_tick(iButtonWorker* worker);
  10. void ibutton_worker_mode_emulate_stop(iButtonWorker* worker);
  11. void ibutton_worker_mode_read_start(iButtonWorker* worker);
  12. void ibutton_worker_mode_read_tick(iButtonWorker* worker);
  13. void ibutton_worker_mode_read_stop(iButtonWorker* worker);
  14. void ibutton_worker_mode_write_start(iButtonWorker* worker);
  15. void ibutton_worker_mode_write_tick(iButtonWorker* worker);
  16. void ibutton_worker_mode_write_stop(iButtonWorker* worker);
  17. const iButtonWorkerModeType ibutton_worker_modes[] = {
  18. {
  19. .quant = osWaitForever,
  20. .start = ibutton_worker_mode_idle_start,
  21. .tick = ibutton_worker_mode_idle_tick,
  22. .stop = ibutton_worker_mode_idle_stop,
  23. },
  24. {
  25. .quant = 100,
  26. .start = ibutton_worker_mode_read_start,
  27. .tick = ibutton_worker_mode_read_tick,
  28. .stop = ibutton_worker_mode_read_stop,
  29. },
  30. {
  31. .quant = 1000,
  32. .start = ibutton_worker_mode_write_start,
  33. .tick = ibutton_worker_mode_write_tick,
  34. .stop = ibutton_worker_mode_write_stop,
  35. },
  36. {
  37. .quant = 1000,
  38. .start = ibutton_worker_mode_emulate_start,
  39. .tick = ibutton_worker_mode_emulate_tick,
  40. .stop = ibutton_worker_mode_emulate_stop,
  41. },
  42. };
  43. /*********************** IDLE ***********************/
  44. void ibutton_worker_mode_idle_start(iButtonWorker* worker) {
  45. }
  46. void ibutton_worker_mode_idle_tick(iButtonWorker* worker) {
  47. }
  48. void ibutton_worker_mode_idle_stop(iButtonWorker* worker) {
  49. }
  50. /*********************** READ ***********************/
  51. void ibutton_worker_comparator_callback(bool level, void* context) {
  52. iButtonWorker* worker = context;
  53. uint32_t current_dwt_value = DWT->CYCCNT;
  54. pulse_decoder_process_pulse(
  55. worker->pulse_decoder, level, current_dwt_value - worker->last_dwt_value);
  56. worker->last_dwt_value = current_dwt_value;
  57. }
  58. bool ibutton_worker_read_comparator(iButtonWorker* worker) {
  59. bool result = false;
  60. pulse_decoder_reset(worker->pulse_decoder);
  61. furi_hal_rfid_pins_reset();
  62. // pulldown pull pin, we sense the signal through the analog part of the RFID schematic
  63. furi_hal_rfid_pin_pull_pulldown();
  64. furi_hal_rfid_comp_set_callback(ibutton_worker_comparator_callback, worker);
  65. worker->last_dwt_value = DWT->CYCCNT;
  66. furi_hal_rfid_comp_start();
  67. // TODO: rework with thread events, "pulse_decoder_get_decoded_index_with_timeout"
  68. furi_hal_delay_ms(100);
  69. int32_t decoded_index = pulse_decoder_get_decoded_index(worker->pulse_decoder);
  70. if(decoded_index >= 0) {
  71. pulse_decoder_get_data(
  72. worker->pulse_decoder, decoded_index, worker->key_data, ibutton_key_get_max_size());
  73. }
  74. switch(decoded_index) {
  75. case PulseProtocolCyfral:
  76. furi_check(worker->key_p != NULL);
  77. ibutton_key_set_type(worker->key_p, iButtonKeyCyfral);
  78. ibutton_key_set_data(worker->key_p, worker->key_data, ibutton_key_get_max_size());
  79. result = true;
  80. break;
  81. case PulseProtocolMetakom:
  82. furi_check(worker->key_p != NULL);
  83. ibutton_key_set_type(worker->key_p, iButtonKeyMetakom);
  84. ibutton_key_set_data(worker->key_p, worker->key_data, ibutton_key_get_max_size());
  85. result = true;
  86. break;
  87. break;
  88. default:
  89. break;
  90. }
  91. furi_hal_rfid_comp_stop();
  92. furi_hal_rfid_comp_set_callback(NULL, NULL);
  93. furi_hal_rfid_pins_reset();
  94. return result;
  95. }
  96. bool ibutton_worker_read_dallas(iButtonWorker* worker) {
  97. bool result = false;
  98. onewire_host_start(worker->host);
  99. furi_hal_delay_ms(100);
  100. FURI_CRITICAL_ENTER();
  101. if(onewire_host_search(worker->host, worker->key_data, NORMAL_SEARCH)) {
  102. onewire_host_reset_search(worker->host);
  103. // key found, verify
  104. if(onewire_host_reset(worker->host)) {
  105. onewire_host_write(worker->host, DS1990_CMD_READ_ROM);
  106. bool key_valid = true;
  107. for(uint8_t i = 0; i < ibutton_key_get_max_size(); i++) {
  108. if(onewire_host_read(worker->host) != worker->key_data[i]) {
  109. key_valid = false;
  110. break;
  111. }
  112. }
  113. if(key_valid) {
  114. result = true;
  115. furi_check(worker->key_p != NULL);
  116. ibutton_key_set_type(worker->key_p, iButtonKeyDS1990);
  117. ibutton_key_set_data(worker->key_p, worker->key_data, ibutton_key_get_max_size());
  118. }
  119. }
  120. } else {
  121. onewire_host_reset_search(worker->host);
  122. }
  123. onewire_host_stop(worker->host);
  124. FURI_CRITICAL_EXIT();
  125. return result;
  126. }
  127. void ibutton_worker_mode_read_start(iButtonWorker* worker) {
  128. furi_hal_power_enable_otg();
  129. }
  130. void ibutton_worker_mode_read_tick(iButtonWorker* worker) {
  131. bool valid = false;
  132. if(ibutton_worker_read_dallas(worker)) {
  133. valid = true;
  134. } else if(ibutton_worker_read_comparator(worker)) {
  135. valid = true;
  136. }
  137. if(valid) {
  138. if(worker->read_cb != NULL) {
  139. worker->read_cb(worker->cb_ctx);
  140. }
  141. ibutton_worker_switch_mode(worker, iButtonWorkerIdle);
  142. }
  143. }
  144. void ibutton_worker_mode_read_stop(iButtonWorker* worker) {
  145. furi_hal_power_disable_otg();
  146. }
  147. /*********************** EMULATE ***********************/
  148. static void onewire_slave_callback(void* context) {
  149. furi_assert(context);
  150. iButtonWorker* worker = context;
  151. if(worker->emulate_cb != NULL) {
  152. worker->emulate_cb(worker->cb_ctx, true);
  153. }
  154. }
  155. void ibutton_worker_emulate_dallas_start(iButtonWorker* worker) {
  156. uint8_t* device_id = onewire_device_get_id_p(worker->device);
  157. const uint8_t* key_id = ibutton_key_get_data_p(worker->key_p);
  158. const uint8_t key_size = ibutton_key_get_max_size();
  159. memcpy(device_id, key_id, key_size);
  160. onewire_slave_attach(worker->slave, worker->device);
  161. onewire_slave_start(worker->slave);
  162. onewire_slave_set_result_callback(worker->slave, onewire_slave_callback, worker);
  163. }
  164. void ibutton_worker_emulate_dallas_stop(iButtonWorker* worker) {
  165. onewire_slave_stop(worker->slave);
  166. onewire_slave_detach(worker->slave);
  167. }
  168. void ibutton_worker_emulate_timer_cb(void* context) {
  169. furi_assert(context);
  170. iButtonWorker* worker = context;
  171. bool polarity;
  172. uint32_t length;
  173. switch(worker->emulate_mode) {
  174. case iButtonEmulateModeCyfral:
  175. encoder_cyfral_get_pulse(worker->encoder_cyfral, &polarity, &length);
  176. break;
  177. case iButtonEmulateModeMetakom:
  178. encoder_metakom_get_pulse(worker->encoder_metakom, &polarity, &length);
  179. break;
  180. }
  181. furi_hal_ibutton_emulate_set_next(length);
  182. if(polarity) {
  183. furi_hal_ibutton_pin_high();
  184. } else {
  185. furi_hal_ibutton_pin_low();
  186. }
  187. }
  188. void ibutton_worker_emulate_timer_start(iButtonWorker* worker) {
  189. furi_assert(worker->key_p);
  190. const uint8_t* key_id = ibutton_key_get_data_p(worker->key_p);
  191. const uint8_t key_size = ibutton_key_get_max_size();
  192. switch(ibutton_key_get_type(worker->key_p)) {
  193. case iButtonKeyDS1990:
  194. return;
  195. break;
  196. case iButtonKeyCyfral:
  197. worker->emulate_mode = iButtonEmulateModeCyfral;
  198. encoder_cyfral_reset(worker->encoder_cyfral);
  199. encoder_cyfral_set_data(worker->encoder_cyfral, key_id, key_size);
  200. break;
  201. case iButtonKeyMetakom:
  202. worker->emulate_mode = iButtonEmulateModeMetakom;
  203. encoder_metakom_reset(worker->encoder_metakom);
  204. encoder_metakom_set_data(worker->encoder_metakom, key_id, key_size);
  205. break;
  206. }
  207. furi_hal_ibutton_start_drive();
  208. furi_hal_ibutton_emulate_start(0, ibutton_worker_emulate_timer_cb, worker);
  209. }
  210. void ibutton_worker_emulate_timer_stop(iButtonWorker* worker) {
  211. furi_hal_ibutton_emulate_stop();
  212. }
  213. void ibutton_worker_mode_emulate_start(iButtonWorker* worker) {
  214. furi_assert(worker->key_p);
  215. furi_hal_rfid_pins_reset();
  216. furi_hal_rfid_pin_pull_pulldown();
  217. switch(ibutton_key_get_type(worker->key_p)) {
  218. case iButtonKeyDS1990:
  219. ibutton_worker_emulate_dallas_start(worker);
  220. break;
  221. case iButtonKeyCyfral:
  222. case iButtonKeyMetakom:
  223. ibutton_worker_emulate_timer_start(worker);
  224. break;
  225. }
  226. }
  227. void ibutton_worker_mode_emulate_tick(iButtonWorker* worker) {
  228. }
  229. void ibutton_worker_mode_emulate_stop(iButtonWorker* worker) {
  230. furi_assert(worker->key_p);
  231. furi_hal_rfid_pins_reset();
  232. switch(ibutton_key_get_type(worker->key_p)) {
  233. case iButtonKeyDS1990:
  234. ibutton_worker_emulate_dallas_stop(worker);
  235. break;
  236. case iButtonKeyCyfral:
  237. case iButtonKeyMetakom:
  238. ibutton_worker_emulate_timer_stop(worker);
  239. break;
  240. }
  241. }
  242. /*********************** WRITE ***********************/
  243. void ibutton_worker_mode_write_start(iButtonWorker* worker) {
  244. furi_hal_power_enable_otg();
  245. onewire_host_start(worker->host);
  246. }
  247. void ibutton_worker_mode_write_tick(iButtonWorker* worker) {
  248. furi_check(worker->key_p != NULL);
  249. iButtonWriterResult writer_result = ibutton_writer_write(worker->writer, worker->key_p);
  250. iButtonWorkerWriteResult result;
  251. switch(writer_result) {
  252. case iButtonWriterOK:
  253. result = iButtonWorkerWriteOK;
  254. break;
  255. case iButtonWriterSameKey:
  256. result = iButtonWorkerWriteSameKey;
  257. break;
  258. case iButtonWriterNoDetect:
  259. result = iButtonWorkerWriteNoDetect;
  260. break;
  261. case iButtonWriterCannotWrite:
  262. result = iButtonWorkerWriteCannotWrite;
  263. break;
  264. default:
  265. result = iButtonWorkerWriteNoDetect;
  266. break;
  267. }
  268. if(worker->write_cb != NULL) {
  269. worker->write_cb(worker->cb_ctx, result);
  270. }
  271. }
  272. void ibutton_worker_mode_write_stop(iButtonWorker* worker) {
  273. furi_hal_power_disable_otg();
  274. onewire_host_stop(worker->host);
  275. }