infrared_brute_force.c 4.6 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. InfraredBruteForceRecordDict_t records;
  22. };
  23. InfraredBruteForce* infrared_brute_force_alloc() {
  24. InfraredBruteForce* brute_force = malloc(sizeof(InfraredBruteForce));
  25. brute_force->ff = NULL;
  26. brute_force->db_filename = NULL;
  27. string_init(brute_force->current_record_name);
  28. InfraredBruteForceRecordDict_init(brute_force->records);
  29. return brute_force;
  30. }
  31. void infrared_brute_force_free(InfraredBruteForce* brute_force) {
  32. furi_assert(!brute_force->ff);
  33. InfraredBruteForceRecordDict_clear(brute_force->records);
  34. string_clear(brute_force->current_record_name);
  35. free(brute_force);
  36. }
  37. void infrared_brute_force_set_db_filename(InfraredBruteForce* brute_force, const char* db_filename) {
  38. brute_force->db_filename = db_filename;
  39. }
  40. bool infrared_brute_force_calculate_messages(InfraredBruteForce* brute_force) {
  41. furi_assert(brute_force->db_filename);
  42. bool success = false;
  43. Storage* storage = furi_record_open("storage");
  44. FlipperFormat* ff = flipper_format_file_alloc(storage);
  45. success = flipper_format_file_open_existing(ff, brute_force->db_filename);
  46. if(success) {
  47. string_t signal_name;
  48. string_init(signal_name);
  49. while(flipper_format_read_string(ff, "name", signal_name)) {
  50. InfraredBruteForceRecord* record =
  51. InfraredBruteForceRecordDict_get(brute_force->records, signal_name);
  52. if(record) {
  53. ++(record->count);
  54. }
  55. }
  56. string_clear(signal_name);
  57. }
  58. flipper_format_free(ff);
  59. furi_record_close("storage");
  60. return success;
  61. }
  62. bool infrared_brute_force_start(
  63. InfraredBruteForce* brute_force,
  64. uint32_t index,
  65. uint32_t* record_count) {
  66. bool success = false;
  67. *record_count = 0;
  68. InfraredBruteForceRecordDict_it_t it;
  69. for(InfraredBruteForceRecordDict_it(it, brute_force->records);
  70. !InfraredBruteForceRecordDict_end_p(it);
  71. InfraredBruteForceRecordDict_next(it)) {
  72. const InfraredBruteForceRecordDict_itref_t* record = InfraredBruteForceRecordDict_cref(it);
  73. if(record->value.index == index) {
  74. *record_count = record->value.count;
  75. if(*record_count) {
  76. string_set(brute_force->current_record_name, record->key);
  77. }
  78. break;
  79. }
  80. }
  81. if(*record_count) {
  82. Storage* storage = furi_record_open("storage");
  83. brute_force->ff = flipper_format_file_alloc(storage);
  84. success = flipper_format_file_open_existing(brute_force->ff, brute_force->db_filename);
  85. if(!success) {
  86. flipper_format_free(brute_force->ff);
  87. brute_force->ff = NULL;
  88. furi_record_close("storage");
  89. }
  90. }
  91. return success;
  92. }
  93. bool infrared_brute_force_is_started(InfraredBruteForce* brute_force) {
  94. return brute_force->ff;
  95. }
  96. void infrared_brute_force_stop(InfraredBruteForce* brute_force) {
  97. furi_assert(string_size(brute_force->current_record_name));
  98. furi_assert(brute_force->ff);
  99. string_reset(brute_force->current_record_name);
  100. flipper_format_free(brute_force->ff);
  101. furi_record_close("storage");
  102. brute_force->ff = NULL;
  103. }
  104. bool infrared_brute_force_send_next(InfraredBruteForce* brute_force) {
  105. furi_assert(string_size(brute_force->current_record_name));
  106. furi_assert(brute_force->ff);
  107. bool success = false;
  108. string_t signal_name;
  109. string_init(signal_name);
  110. InfraredSignal* signal = infrared_signal_alloc();
  111. do {
  112. success = infrared_signal_read(signal, brute_force->ff, signal_name);
  113. } while(success && !string_equal_p(brute_force->current_record_name, signal_name));
  114. if(success) {
  115. infrared_signal_transmit(signal);
  116. }
  117. infrared_signal_free(signal);
  118. string_clear(signal_name);
  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. }