infrared_brute_force.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. #include "infrared_brute_force.h"
  2. #include <stdlib.h>
  3. #include <m-dict.h>
  4. #include <flipper_format/flipper_format.h>
  5. #include "infrared_signal.h"
  6. typedef struct {
  7. uint32_t index;
  8. uint32_t count;
  9. } InfraredBruteForceRecord;
  10. DICT_DEF2(
  11. InfraredBruteForceRecordDict,
  12. FuriString*,
  13. FURI_STRING_OPLIST,
  14. InfraredBruteForceRecord,
  15. M_POD_OPLIST);
  16. struct InfraredBruteForce {
  17. FlipperFormat* ff;
  18. const char* db_filename;
  19. FuriString* current_record_name;
  20. InfraredSignal* current_signal;
  21. InfraredBruteForceRecordDict_t records;
  22. bool is_started;
  23. };
  24. InfraredBruteForce* infrared_brute_force_alloc() {
  25. InfraredBruteForce* brute_force = malloc(sizeof(InfraredBruteForce));
  26. brute_force->ff = NULL;
  27. brute_force->db_filename = NULL;
  28. brute_force->current_signal = NULL;
  29. brute_force->is_started = false;
  30. brute_force->current_record_name = furi_string_alloc();
  31. InfraredBruteForceRecordDict_init(brute_force->records);
  32. return brute_force;
  33. }
  34. void infrared_brute_force_free(InfraredBruteForce* brute_force) {
  35. furi_assert(!brute_force->is_started);
  36. InfraredBruteForceRecordDict_clear(brute_force->records);
  37. furi_string_free(brute_force->current_record_name);
  38. free(brute_force);
  39. }
  40. void infrared_brute_force_set_db_filename(InfraredBruteForce* brute_force, const char* db_filename) {
  41. furi_assert(!brute_force->is_started);
  42. brute_force->db_filename = db_filename;
  43. }
  44. bool infrared_brute_force_calculate_messages(InfraredBruteForce* brute_force) {
  45. furi_assert(!brute_force->is_started);
  46. furi_assert(brute_force->db_filename);
  47. bool success = false;
  48. Storage* storage = furi_record_open(RECORD_STORAGE);
  49. FlipperFormat* ff = flipper_format_buffered_file_alloc(storage);
  50. success = flipper_format_buffered_file_open_existing(ff, brute_force->db_filename);
  51. if(success) {
  52. FuriString* signal_name;
  53. signal_name = furi_string_alloc();
  54. while(flipper_format_read_string(ff, "name", signal_name)) {
  55. InfraredBruteForceRecord* record =
  56. InfraredBruteForceRecordDict_get(brute_force->records, signal_name);
  57. if(record) { //-V547
  58. ++(record->count);
  59. }
  60. }
  61. furi_string_free(signal_name);
  62. }
  63. flipper_format_free(ff);
  64. furi_record_close(RECORD_STORAGE);
  65. return success;
  66. }
  67. bool infrared_brute_force_start(
  68. InfraredBruteForce* brute_force,
  69. uint32_t index,
  70. uint32_t* record_count) {
  71. furi_assert(!brute_force->is_started);
  72. bool success = false;
  73. *record_count = 0;
  74. InfraredBruteForceRecordDict_it_t it;
  75. for(InfraredBruteForceRecordDict_it(it, brute_force->records);
  76. !InfraredBruteForceRecordDict_end_p(it);
  77. InfraredBruteForceRecordDict_next(it)) {
  78. const InfraredBruteForceRecordDict_itref_t* record = InfraredBruteForceRecordDict_cref(it);
  79. if(record->value.index == index) {
  80. *record_count = record->value.count;
  81. if(*record_count) {
  82. furi_string_set(brute_force->current_record_name, record->key);
  83. }
  84. break;
  85. }
  86. }
  87. if(*record_count) {
  88. Storage* storage = furi_record_open(RECORD_STORAGE);
  89. brute_force->ff = flipper_format_buffered_file_alloc(storage);
  90. brute_force->current_signal = infrared_signal_alloc();
  91. brute_force->is_started = true;
  92. success =
  93. flipper_format_buffered_file_open_existing(brute_force->ff, brute_force->db_filename);
  94. if(!success) infrared_brute_force_stop(brute_force);
  95. }
  96. return success;
  97. }
  98. bool infrared_brute_force_is_started(InfraredBruteForce* brute_force) {
  99. return brute_force->is_started;
  100. }
  101. void infrared_brute_force_stop(InfraredBruteForce* brute_force) {
  102. furi_assert(brute_force->is_started);
  103. furi_string_reset(brute_force->current_record_name);
  104. infrared_signal_free(brute_force->current_signal);
  105. flipper_format_free(brute_force->ff);
  106. brute_force->current_signal = NULL;
  107. brute_force->ff = NULL;
  108. brute_force->is_started = false;
  109. furi_record_close(RECORD_STORAGE);
  110. }
  111. bool infrared_brute_force_send_next(InfraredBruteForce* brute_force) {
  112. furi_assert(brute_force->is_started);
  113. const bool success = infrared_signal_search_and_read(
  114. brute_force->current_signal, brute_force->ff, brute_force->current_record_name);
  115. if(success) {
  116. infrared_signal_transmit(brute_force->current_signal);
  117. }
  118. return success;
  119. }
  120. void infrared_brute_force_add_record(
  121. InfraredBruteForce* brute_force,
  122. uint32_t index,
  123. const char* name) {
  124. InfraredBruteForceRecord value = {.index = index, .count = 0};
  125. FuriString* key;
  126. key = furi_string_alloc_set(name);
  127. InfraredBruteForceRecordDict_set_at(brute_force->records, key, value);
  128. furi_string_free(key);
  129. }
  130. void infrared_brute_force_reset(InfraredBruteForce* brute_force) {
  131. furi_assert(!brute_force->is_started);
  132. InfraredBruteForceRecordDict_reset(brute_force->records);
  133. }