subghz_protocol_raw.c 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. #include "subghz_protocol_raw.h"
  2. #include "../subghz_file_encoder_worker.h"
  3. #define SUBGHZ_DOWNLOAD_MAX_SIZE 512
  4. struct SubGhzProtocolRAW {
  5. SubGhzProtocolCommon common;
  6. int32_t* upload_raw;
  7. uint16_t ind_write;
  8. Storage* storage;
  9. FlipperFile* flipper_file;
  10. SubGhzFileEncoderWorker* file_worker_encoder;
  11. uint32_t file_is_open;
  12. string_t file_name;
  13. size_t sample_write;
  14. bool last_level;
  15. };
  16. typedef enum {
  17. RAWFileIsOpenClose = 0,
  18. RAWFileIsOpenWrite,
  19. RAWFileIsOpenRead,
  20. } RAWFilIsOpen;
  21. SubGhzProtocolRAW* subghz_protocol_raw_alloc(void) {
  22. SubGhzProtocolRAW* instance = furi_alloc(sizeof(SubGhzProtocolRAW));
  23. instance->upload_raw = NULL;
  24. instance->ind_write = 0;
  25. instance->last_level = false;
  26. instance->storage = furi_record_open("storage");
  27. instance->flipper_file = flipper_file_alloc(instance->storage);
  28. instance->file_is_open = RAWFileIsOpenClose;
  29. string_init(instance->file_name);
  30. instance->common.name = "RAW";
  31. instance->common.code_min_count_bit_for_found = 0;
  32. instance->common.te_short = 80;
  33. instance->common.te_long = 32700;
  34. instance->common.te_delta = 0;
  35. instance->common.type_protocol = SubGhzProtocolCommonTypeRAW;
  36. instance->common.to_load_protocol_from_file =
  37. (SubGhzProtocolCommonLoadFromFile)subghz_protocol_raw_to_load_protocol_from_file;
  38. instance->common.to_string = (SubGhzProtocolCommonToStr)subghz_protocol_raw_to_str;
  39. //instance->common.to_load_protocol =
  40. // (SubGhzProtocolCommonLoadFromRAW)subghz_decoder_raw_to_load_protocol;
  41. instance->common.get_upload_protocol =
  42. (SubGhzProtocolCommonEncoderGetUpLoad)subghz_protocol_raw_send_key;
  43. return instance;
  44. }
  45. void subghz_protocol_raw_free(SubGhzProtocolRAW* instance) {
  46. furi_assert(instance);
  47. string_clear(instance->file_name);
  48. flipper_file_free(instance->flipper_file);
  49. furi_record_close("storage");
  50. free(instance);
  51. }
  52. void subghz_protocol_raw_file_encoder_worker_stop(void* context) {
  53. furi_assert(context);
  54. SubGhzProtocolRAW* instance = context;
  55. if(subghz_file_encoder_worker_is_running(instance->file_worker_encoder)) {
  56. subghz_file_encoder_worker_stop(instance->file_worker_encoder);
  57. subghz_file_encoder_worker_free(instance->file_worker_encoder);
  58. instance->file_is_open = RAWFileIsOpenClose;
  59. }
  60. }
  61. bool subghz_protocol_raw_send_key(
  62. SubGhzProtocolRAW* instance,
  63. SubGhzProtocolCommonEncoder* encoder) {
  64. furi_assert(instance);
  65. furi_assert(encoder);
  66. bool loaded = false;
  67. instance->file_worker_encoder = subghz_file_encoder_worker_alloc();
  68. if(subghz_file_encoder_worker_start(
  69. instance->file_worker_encoder, string_get_cstr(instance->file_name))) {
  70. //the worker needs a file in order to open and read part of the file
  71. osDelay(100);
  72. instance->file_is_open = RAWFileIsOpenRead;
  73. subghz_protocol_encoder_common_set_callback(
  74. encoder, subghz_file_encoder_worker_get_level_duration, instance->file_worker_encoder);
  75. subghz_protocol_encoder_common_set_callback_end(
  76. encoder, subghz_protocol_raw_file_encoder_worker_stop, instance);
  77. loaded = true;
  78. } else {
  79. subghz_protocol_raw_file_encoder_worker_stop(instance);
  80. }
  81. return loaded;
  82. }
  83. void subghz_protocol_raw_reset(SubGhzProtocolRAW* instance) {
  84. instance->ind_write = 0;
  85. }
  86. void subghz_protocol_raw_parse(SubGhzProtocolRAW* instance, bool level, uint32_t duration) {
  87. if(instance->upload_raw != NULL) {
  88. if(duration > instance->common.te_short) {
  89. if(duration > instance->common.te_long) duration = instance->common.te_long;
  90. if(instance->last_level != level) {
  91. instance->last_level = (level ? true : false);
  92. instance->upload_raw[instance->ind_write++] = (level ? duration : -duration);
  93. }
  94. }
  95. if(instance->ind_write == SUBGHZ_DOWNLOAD_MAX_SIZE) {
  96. subghz_protocol_raw_save_to_file_write(instance);
  97. }
  98. }
  99. }
  100. void subghz_protocol_raw_to_str(SubGhzProtocolRAW* instance, string_t output) {
  101. string_cat_printf(output, "RAW Date");
  102. }
  103. const char* subghz_protocol_raw_get_last_file_name(SubGhzProtocolRAW* instance) {
  104. return string_get_cstr(instance->file_name);
  105. }
  106. void subghz_protocol_raw_set_last_file_name(SubGhzProtocolRAW* instance, const char* name) {
  107. string_printf(instance->file_name, "%s", name);
  108. }
  109. bool subghz_protocol_raw_save_to_file_init(
  110. SubGhzProtocolRAW* instance,
  111. const char* dev_name,
  112. uint32_t frequency,
  113. const char* preset) {
  114. furi_assert(instance);
  115. //instance->flipper_file = flipper_file_alloc(instance->storage);
  116. string_t dev_file_name;
  117. string_init(dev_file_name);
  118. bool init = false;
  119. do {
  120. // Create subghz folder directory if necessary
  121. if(!storage_simply_mkdir(instance->storage, SUBGHZ_RAW_FOLDER)) {
  122. break;
  123. }
  124. // Create saved directory if necessary
  125. if(!storage_simply_mkdir(instance->storage, SUBGHZ_RAW_PATH_FOLDER)) {
  126. break;
  127. }
  128. string_set(instance->file_name, dev_name);
  129. // First remove subghz device file if it was saved
  130. string_printf(
  131. dev_file_name, "%s/%s%s", SUBGHZ_APP_PATH_FOLDER, dev_name, SUBGHZ_APP_EXTENSION);
  132. if(!storage_simply_remove(instance->storage, string_get_cstr(dev_file_name))) {
  133. break;
  134. }
  135. // Open file
  136. if(!flipper_file_open_always(instance->flipper_file, string_get_cstr(dev_file_name))) {
  137. FURI_LOG_E(SUBGHZ_RAW_TAG, "Unable to open file for write: %s", dev_file_name);
  138. break;
  139. }
  140. if(!flipper_file_write_header_cstr(
  141. instance->flipper_file, SUBGHZ_RAW_FILE_TYPE, SUBGHZ_RAW_FILE_VERSION)) {
  142. FURI_LOG_E(SUBGHZ_RAW_TAG, "Unable to add header");
  143. break;
  144. }
  145. if(!flipper_file_write_uint32(instance->flipper_file, "Frequency", (uint32_t*)&frequency, 1)) {
  146. FURI_LOG_E(SUBGHZ_RAW_TAG, "Unable to add Frequency");
  147. break;
  148. }
  149. if(!flipper_file_write_string_cstr(instance->flipper_file, "Preset", preset)) {
  150. FURI_LOG_E(SUBGHZ_RAW_TAG, "Unable to add Preset");
  151. break;
  152. }
  153. if(!flipper_file_write_string_cstr(instance->flipper_file, "Protocol", instance->common.name)) {
  154. FURI_LOG_E(SUBGHZ_RAW_TAG, "Unable to add Protocol");
  155. break;
  156. }
  157. instance->upload_raw = furi_alloc(SUBGHZ_DOWNLOAD_MAX_SIZE * sizeof(int32_t));
  158. instance->file_is_open = RAWFileIsOpenWrite;
  159. instance->sample_write = 0;
  160. init = true;
  161. } while(0);
  162. string_clear(dev_file_name);
  163. return init;
  164. }
  165. void subghz_protocol_raw_save_to_file_stop(SubGhzProtocolRAW* instance) {
  166. furi_assert(instance);
  167. if(instance->file_is_open == RAWFileIsOpenWrite && instance->ind_write)
  168. subghz_protocol_raw_save_to_file_write(instance);
  169. if(instance->file_is_open != RAWFileIsOpenClose) {
  170. free(instance->upload_raw);
  171. instance->upload_raw = NULL;
  172. }
  173. flipper_file_close(instance->flipper_file);
  174. instance->file_is_open = RAWFileIsOpenClose;
  175. }
  176. bool subghz_protocol_raw_save_to_file_write(SubGhzProtocolRAW* instance) {
  177. furi_assert(instance);
  178. bool is_write = false;
  179. if(instance->file_is_open == RAWFileIsOpenWrite) {
  180. if(!flipper_file_write_int32(
  181. instance->flipper_file, "RAW_Data", instance->upload_raw, instance->ind_write)) {
  182. FURI_LOG_E(SUBGHZ_RAW_TAG, "Unable to add RAW_Data");
  183. } else {
  184. instance->sample_write += instance->ind_write;
  185. instance->ind_write = 0;
  186. is_write = true;
  187. }
  188. }
  189. return is_write;
  190. }
  191. size_t subghz_protocol_raw_get_sample_write(SubGhzProtocolRAW* instance) {
  192. return instance->sample_write + instance->ind_write;
  193. }
  194. bool subghz_protocol_raw_to_load_protocol_from_file(
  195. FlipperFile* flipper_file,
  196. SubGhzProtocolRAW* instance,
  197. const char* file_path) {
  198. furi_assert(file_path);
  199. subghz_protocol_raw_set_last_file_name(instance, file_path);
  200. return true;
  201. }