lfrfid_worker.c 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. #include <furi.h>
  2. #include <furi_hal.h>
  3. #include <atomic.h>
  4. #include "lfrfid_worker_i.h"
  5. typedef enum {
  6. LFRFIDEventStopThread = (1 << 0),
  7. LFRFIDEventStopMode = (1 << 1),
  8. LFRFIDEventRead = (1 << 2),
  9. LFRFIDEventWrite = (1 << 3),
  10. LFRFIDEventEmulate = (1 << 4),
  11. LFRFIDEventReadRaw = (1 << 5),
  12. LFRFIDEventEmulateRaw = (1 << 6),
  13. LFRFIDEventAll =
  14. (LFRFIDEventStopThread | LFRFIDEventStopMode | LFRFIDEventRead | LFRFIDEventWrite |
  15. LFRFIDEventEmulate | LFRFIDEventReadRaw | LFRFIDEventEmulateRaw),
  16. } LFRFIDEventType;
  17. static int32_t lfrfid_worker_thread(void* thread_context);
  18. LFRFIDWorker* lfrfid_worker_alloc(ProtocolDict* dict) {
  19. furi_assert(dict);
  20. LFRFIDWorker* worker = malloc(sizeof(LFRFIDWorker));
  21. worker->mode_index = LFRFIDWorkerIdle;
  22. worker->read_cb = NULL;
  23. worker->write_cb = NULL;
  24. worker->cb_ctx = NULL;
  25. worker->raw_filename = NULL;
  26. worker->mode_storage = NULL;
  27. worker->thread = furi_thread_alloc_ex("LfrfidWorker", 2048, lfrfid_worker_thread, worker);
  28. worker->protocols = dict;
  29. return worker;
  30. }
  31. void lfrfid_worker_free(LFRFIDWorker* worker) {
  32. if(worker->raw_filename) {
  33. free(worker->raw_filename);
  34. }
  35. furi_thread_free(worker->thread);
  36. free(worker);
  37. }
  38. void lfrfid_worker_read_start(
  39. LFRFIDWorker* worker,
  40. LFRFIDWorkerReadType type,
  41. LFRFIDWorkerReadCallback callback,
  42. void* context) {
  43. furi_assert(worker->mode_index == LFRFIDWorkerIdle);
  44. worker->read_type = type;
  45. worker->read_cb = callback;
  46. worker->cb_ctx = context;
  47. furi_thread_flags_set(furi_thread_get_id(worker->thread), LFRFIDEventRead);
  48. }
  49. void lfrfid_worker_write_start(
  50. LFRFIDWorker* worker,
  51. LFRFIDProtocol protocol,
  52. LFRFIDWorkerWriteCallback callback,
  53. void* context) {
  54. furi_assert(worker->mode_index == LFRFIDWorkerIdle);
  55. worker->protocol = protocol;
  56. worker->write_cb = callback;
  57. worker->cb_ctx = context;
  58. furi_thread_flags_set(furi_thread_get_id(worker->thread), LFRFIDEventWrite);
  59. }
  60. void lfrfid_worker_emulate_start(LFRFIDWorker* worker, LFRFIDProtocol protocol) {
  61. furi_assert(worker->mode_index == LFRFIDWorkerIdle);
  62. worker->protocol = protocol;
  63. furi_thread_flags_set(furi_thread_get_id(worker->thread), LFRFIDEventEmulate);
  64. }
  65. void lfrfid_worker_set_filename(LFRFIDWorker* worker, const char* filename) {
  66. if(worker->raw_filename) {
  67. free(worker->raw_filename);
  68. }
  69. worker->raw_filename = strdup(filename);
  70. }
  71. void lfrfid_worker_read_raw_start(
  72. LFRFIDWorker* worker,
  73. const char* filename,
  74. LFRFIDWorkerReadType type,
  75. LFRFIDWorkerReadRawCallback callback,
  76. void* context) {
  77. furi_assert(worker->mode_index == LFRFIDWorkerIdle);
  78. worker->read_type = type;
  79. worker->read_raw_cb = callback;
  80. worker->cb_ctx = context;
  81. lfrfid_worker_set_filename(worker, filename);
  82. furi_thread_flags_set(furi_thread_get_id(worker->thread), LFRFIDEventReadRaw);
  83. }
  84. void lfrfid_worker_emulate_raw_start(
  85. LFRFIDWorker* worker,
  86. const char* filename,
  87. LFRFIDWorkerEmulateRawCallback callback,
  88. void* context) {
  89. furi_assert(worker->mode_index == LFRFIDWorkerIdle);
  90. lfrfid_worker_set_filename(worker, filename);
  91. worker->emulate_raw_cb = callback;
  92. worker->cb_ctx = context;
  93. furi_thread_flags_set(furi_thread_get_id(worker->thread), LFRFIDEventEmulateRaw);
  94. }
  95. void lfrfid_worker_stop(LFRFIDWorker* worker) {
  96. furi_thread_flags_set(furi_thread_get_id(worker->thread), LFRFIDEventStopMode);
  97. }
  98. void lfrfid_worker_start_thread(LFRFIDWorker* worker) {
  99. furi_thread_start(worker->thread);
  100. }
  101. void lfrfid_worker_stop_thread(LFRFIDWorker* worker) {
  102. furi_assert(worker->mode_index == LFRFIDWorkerIdle);
  103. furi_thread_flags_set(furi_thread_get_id(worker->thread), LFRFIDEventStopThread);
  104. furi_thread_join(worker->thread);
  105. }
  106. bool lfrfid_worker_check_for_stop(LFRFIDWorker* worker) {
  107. UNUSED(worker);
  108. uint32_t flags = furi_thread_flags_get();
  109. return (flags & LFRFIDEventStopMode);
  110. }
  111. size_t lfrfid_worker_dict_get_data_size(LFRFIDWorker* worker, LFRFIDProtocol protocol) {
  112. furi_assert(worker->mode_index == LFRFIDWorkerIdle);
  113. return protocol_dict_get_data_size(worker->protocols, protocol);
  114. }
  115. static int32_t lfrfid_worker_thread(void* thread_context) {
  116. LFRFIDWorker* worker = thread_context;
  117. while(true) {
  118. uint32_t flags = furi_thread_flags_wait(LFRFIDEventAll, FuriFlagWaitAny, FuriWaitForever);
  119. if(flags != (unsigned)FuriFlagErrorTimeout) {
  120. // stop thread
  121. if(flags & LFRFIDEventStopThread) break;
  122. // switch mode
  123. if(flags & LFRFIDEventRead) worker->mode_index = LFRFIDWorkerRead;
  124. if(flags & LFRFIDEventWrite) worker->mode_index = LFRFIDWorkerWrite;
  125. if(flags & LFRFIDEventEmulate) worker->mode_index = LFRFIDWorkerEmulate;
  126. if(flags & LFRFIDEventReadRaw) worker->mode_index = LFRFIDWorkerReadRaw;
  127. if(flags & LFRFIDEventEmulateRaw) worker->mode_index = LFRFIDWorkerEmulateRaw;
  128. // do mode, if it exists
  129. if(lfrfid_worker_modes[worker->mode_index].process) {
  130. lfrfid_worker_modes[worker->mode_index].process(worker);
  131. }
  132. // reset mode
  133. worker->mode_index = LFRFIDWorkerIdle;
  134. }
  135. }
  136. return 0;
  137. }