infrared_brute_force.c 4.9 KB

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