app.h 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. /* Copyright (C) 2022-2023 Salvatore Sanfilippo -- All Rights Reserved
  2. * See the LICENSE file for information about the license. */
  3. #pragma once
  4. #include <furi.h>
  5. #include <furi_hal.h>
  6. #include <input/input.h>
  7. #include <gui/gui.h>
  8. #include <stdlib.h>
  9. #include <gui/gui.h>
  10. #include <gui/view_dispatcher.h>
  11. #include <gui/scene_manager.h>
  12. #include <gui/modules/submenu.h>
  13. #include <gui/modules/variable_item_list.h>
  14. #include <gui/modules/widget.h>
  15. #include <gui/modules/text_input.h>
  16. #include <notification/notification_messages.h>
  17. #include <lib/subghz/subghz_setting.h>
  18. #include <lib/subghz/subghz_worker.h>
  19. #include <lib/subghz/receiver.h>
  20. #include <lib/subghz/transmitter.h>
  21. #include <lib/subghz/registry.h>
  22. #include "app_buffer.h"
  23. #define TAG "ProtoView"
  24. #define PROTOVIEW_RAW_VIEW_DEFAULT_SCALE 100 // 100us is 1 pixel by default
  25. #define BITMAP_SEEK_NOT_FOUND UINT32_MAX // Returned by function as sentinel
  26. #define PROTOVIEW_VIEW_PRIVDATA_LEN 64 // View specific private data len
  27. #define DEBUG_MSG 1
  28. /* Forward declarations. */
  29. typedef struct ProtoViewApp ProtoViewApp;
  30. typedef struct ProtoViewMsgInfo ProtoViewMsgInfo;
  31. typedef struct ProtoViewFieldSet ProtoViewFieldSet;
  32. typedef struct ProtoViewDecoder ProtoViewDecoder;
  33. /* ============================== enumerations ============================== */
  34. /* Subghz system state */
  35. typedef enum {
  36. TxRxStateIDLE,
  37. TxRxStateRx,
  38. TxRxStateTx,
  39. TxRxStateSleep,
  40. } TxRxState;
  41. /* Currently active view. */
  42. typedef enum {
  43. ViewRawPulses,
  44. ViewInfo,
  45. ViewFrequencySettings,
  46. ViewModulationSettings,
  47. ViewDirectSampling,
  48. ViewLast, /* Just a sentinel to wrap around. */
  49. /* The following are special views that are not iterated, but
  50. * have meaning for the API. */
  51. ViewGoNext,
  52. ViewGoPrev,
  53. } ProtoViewCurrentView;
  54. /* ================================== RX/TX ================================= */
  55. typedef struct {
  56. const char *name; // Name to show to the user.
  57. const char *id; // Identifier in the Flipper API/file.
  58. FuriHalSubGhzPreset preset; // The preset ID.
  59. uint8_t *custom; // If not null, a set of registers for
  60. // the CC1101, specifying a custom preset.
  61. } ProtoViewModulation;
  62. extern ProtoViewModulation ProtoViewModulations[]; /* In app_subghz.c */
  63. /* This is the context of our subghz worker and associated thread.
  64. * It receives data and we get our protocol "feed" callback called
  65. * with the level (1 or 0) and duration. */
  66. struct ProtoViewTxRx {
  67. bool freq_mod_changed; /* The user changed frequency and/or modulation
  68. from the interface. There is to restart the
  69. radio with the right parameters. */
  70. SubGhzWorker* worker; /* Our background worker. */
  71. SubGhzEnvironment* environment;
  72. SubGhzReceiver* receiver;
  73. TxRxState txrx_state; /* Receiving, idle or sleeping? */
  74. /* Timer sampling mode state. */
  75. bool debug_timer_sampling; /* Read data from GDO0 in a busy loop. Only
  76. for testing. */
  77. uint32_t last_g0_change_time; /* Last high->low (or reverse) switch. */
  78. bool last_g0_value; /* Current value (high or low): we are
  79. checking the duration in the timer
  80. handler. */
  81. };
  82. typedef struct ProtoViewTxRx ProtoViewTxRx;
  83. /* ============================== Main app state ============================ */
  84. #define ALERT_MAX_LEN 32
  85. struct ProtoViewApp {
  86. /* GUI */
  87. Gui *gui;
  88. NotificationApp *notification;
  89. ViewPort *view_port; /* We just use a raw viewport and we render
  90. everything into the low level canvas. */
  91. ProtoViewCurrentView current_view; /* Active left-right view ID. */
  92. int current_subview[ViewLast]; /* Active up-down subview ID. */
  93. FuriMessageQueue *event_queue; /* Keypress events go here. */
  94. /* Input text state. */
  95. ViewDispatcher *view_dispatcher; /* Used only when we want to show
  96. the text_input view for a moment.
  97. Otherwise it is set to null. */
  98. TextInput *text_input;
  99. bool show_text_input;
  100. char *text_input_buffer;
  101. uint32_t text_input_buffer_len;
  102. void (*text_input_done_callback)(void*);
  103. /* Alert state. */
  104. uint32_t alert_dismiss_time; /* Millisecond when the alert will be
  105. no longer shown. Or zero if the alert
  106. is currently not set at all. */
  107. char alert_text[ALERT_MAX_LEN]; /* Alert content. */
  108. /* Radio related. */
  109. ProtoViewTxRx *txrx; /* Radio state. */
  110. SubGhzSetting *setting; /* A list of valid frequencies. */
  111. /* Generic app state. */
  112. int running; /* Once false exists the app. */
  113. uint32_t signal_bestlen; /* Longest coherent signal observed so far. */
  114. uint32_t signal_last_scan_idx; /* Index of the buffer last time we
  115. performed the scan. */
  116. bool signal_decoded; /* Was the current signal decoded? */
  117. ProtoViewMsgInfo *msg_info; /* Decoded message info if not NULL. */
  118. bool direct_sampling_enabled; /* This special view needs an explicit
  119. acknowledge to work. */
  120. void *view_privdata; /* This is a piece of memory of total size
  121. PROTOVIEW_VIEW_PRIVDATA_LEN that it is
  122. initialized to zero when we switch to
  123. a a new view. While the view we are using
  124. is the same, it can be used by the view to
  125. store any kind of info inside, just casting
  126. the pointer to a few specific-data structure. */
  127. /* Raw view apps state. */
  128. uint32_t us_scale; /* microseconds per pixel. */
  129. uint32_t signal_offset; /* Long press left/right panning in raw view. */
  130. /* Configuration view app state. */
  131. uint32_t frequency; /* Current frequency. */
  132. uint8_t modulation; /* Current modulation ID, array index in the
  133. ProtoViewModulations table. */
  134. };
  135. /* =========================== Protocols decoders =========================== */
  136. /* This stucture is filled by the decoder for specific protocols with the
  137. * informations about the message. ProtoView will display such information
  138. * in the message info view. */
  139. #define PROTOVIEW_MSG_STR_LEN 32
  140. typedef struct ProtoViewMsgInfo {
  141. ProtoViewDecoder *decoder; /* The decoder that decoded the message. */
  142. ProtoViewFieldSet *fieldset; /* Decoded fields. */
  143. /* Low level information of the detected signal: the following are filled
  144. * by the protocol decoding function: */
  145. uint32_t start_off; /* Pulses start offset in the bitmap. */
  146. uint32_t pulses_count; /* Number of pulses of the full message. */
  147. /* The following are passed already filled to the decoder. */
  148. uint32_t short_pulse_dur; /* Microseconds duration of the short pulse. */
  149. /* The following are filled by ProtoView core after the decoder returned
  150. * success. */
  151. uint8_t *bits; /* Bitmap with the signal. */
  152. uint32_t bits_bytes; /* Number of full bytes in the bitmap, that
  153. is 'pulses_count/8' rounded to the next
  154. integer. */
  155. } ProtoViewMsgInfo;
  156. /* This structures describe a set of protocol fields. It is used by decoders
  157. * supporting message building to receive and return information about the
  158. * protocol. */
  159. typedef enum {
  160. FieldTypeStr,
  161. FieldTypeSignedInt,
  162. FieldTypeUnsignedInt,
  163. FieldTypeBinary,
  164. FieldTypeHex,
  165. FieldTypeBytes,
  166. FieldTypeFloat,
  167. } ProtoViewFieldType;
  168. typedef struct {
  169. ProtoViewFieldType type;
  170. uint32_t len; // Depends on type:
  171. // Bits for integers (signed,unsigned,binary,hex).
  172. // Number of characters for strings.
  173. // Number of nibbles for bytes (1 for each 4 bits).
  174. // Number of digits after dot for floats.
  175. char *name; // Field name.
  176. union {
  177. char *str; // String type.
  178. int64_t value; // Signed integer type.
  179. uint64_t uvalue; // Unsigned integer type.
  180. uint8_t *bytes; // Raw bytes type.
  181. float fvalue; // Float type.
  182. };
  183. } ProtoViewField;
  184. typedef struct ProtoViewFieldSet {
  185. ProtoViewField **fields;
  186. uint32_t numfields;
  187. } ProtoViewFieldSet;
  188. typedef struct ProtoViewDecoder {
  189. const char *name; /* Protocol name. */
  190. /* The decode function takes a buffer that is actually a bitmap, with
  191. * high and low levels represented as 0 and 1. The number of high/low
  192. * pulses represented by the bitmap is passed as the 'numbits' argument,
  193. * while 'numbytes' represents the total size of the bitmap pointed by
  194. * 'bits'. So 'numbytes' is mainly useful to pass as argument to other
  195. * functions that perform bit extraction with bound checking, such as
  196. * bitmap_get() and so forth. */
  197. bool (*decode)(uint8_t *bits, uint32_t numbytes, uint32_t numbits, ProtoViewMsgInfo *info);
  198. void (*get_fields)(ProtoViewMsgInfo *info, ProtoViewFieldSet *fields);
  199. void (*build_message)(RawSamplesBuffer *samples, ProtoViewFieldSet *fields);
  200. } ProtoViewDecoder;
  201. extern RawSamplesBuffer *RawSamples, *DetectedSamples;
  202. /* app_radio.c */
  203. void radio_begin(ProtoViewApp* app);
  204. uint32_t radio_rx(ProtoViewApp* app);
  205. void radio_idle(ProtoViewApp* app);
  206. void radio_rx_end(ProtoViewApp* app);
  207. void radio_sleep(ProtoViewApp* app);
  208. void raw_sampling_worker_start(ProtoViewApp *app);
  209. void raw_sampling_worker_stop(ProtoViewApp *app);
  210. void radio_tx_signal(ProtoViewApp *app, FuriHalSubGhzAsyncTxCallback data_feeder, void *ctx);
  211. /* signal.c */
  212. uint32_t duration_delta(uint32_t a, uint32_t b);
  213. void reset_current_signal(ProtoViewApp *app);
  214. void scan_for_signal(ProtoViewApp *app);
  215. bool bitmap_get(uint8_t *b, uint32_t blen, uint32_t bitpos);
  216. void bitmap_set(uint8_t *b, uint32_t blen, uint32_t bitpos, bool val);
  217. void bitmap_copy(uint8_t *d, uint32_t dlen, uint32_t doff, uint8_t *s, uint32_t slen, uint32_t soff, uint32_t count);
  218. void bitmap_set_pattern(uint8_t *b, uint32_t blen, uint32_t off, const char *pat);
  219. void bitmap_reverse_bytes(uint8_t *p, uint32_t len);
  220. bool bitmap_match_bits(uint8_t *b, uint32_t blen, uint32_t bitpos, const char *bits);
  221. uint32_t bitmap_seek_bits(uint8_t *b, uint32_t blen, uint32_t startpos, uint32_t maxbits, const char *bits);
  222. uint32_t convert_from_line_code(uint8_t *buf, uint64_t buflen, uint8_t *bits, uint32_t len, uint32_t offset, const char *zero_pattern, const char *one_pattern);
  223. uint32_t convert_from_diff_manchester(uint8_t *buf, uint64_t buflen, uint8_t *bits, uint32_t len, uint32_t off, bool previous);
  224. void init_msg_info(ProtoViewMsgInfo *i, ProtoViewApp *app);
  225. void free_msg_info(ProtoViewMsgInfo *i);
  226. /* signal_file.c */
  227. bool save_signal(ProtoViewApp *app, const char *filename);
  228. /* view_*.c */
  229. void render_view_raw_pulses(Canvas *const canvas, ProtoViewApp *app);
  230. void process_input_raw_pulses(ProtoViewApp *app, InputEvent input);
  231. void render_view_settings(Canvas *const canvas, ProtoViewApp *app);
  232. void process_input_settings(ProtoViewApp *app, InputEvent input);
  233. void render_view_info(Canvas *const canvas, ProtoViewApp *app);
  234. void process_input_info(ProtoViewApp *app, InputEvent input);
  235. void render_view_direct_sampling(Canvas *const canvas, ProtoViewApp *app);
  236. void process_input_direct_sampling(ProtoViewApp *app, InputEvent input);
  237. void view_enter_direct_sampling(ProtoViewApp *app);
  238. void view_exit_direct_sampling(ProtoViewApp *app);
  239. void view_exit_settings(ProtoViewApp *app);
  240. void adjust_raw_view_scale(ProtoViewApp *app, uint32_t short_pulse_dur);
  241. /* ui.c */
  242. int ui_get_current_subview(ProtoViewApp *app);
  243. void ui_show_available_subviews(Canvas *canvas, ProtoViewApp *app, int last_subview);
  244. bool ui_process_subview_updown(ProtoViewApp *app, InputEvent input, int last_subview);
  245. void ui_show_keyboard(ProtoViewApp *app, char *buffer, uint32_t buflen,
  246. void (*done_callback)(void*));
  247. void ui_dismiss_keyboard(ProtoViewApp *app);
  248. void ui_show_alert(ProtoViewApp *app, const char *text, uint32_t ttl);
  249. void ui_dismiss_alert(ProtoViewApp *app);
  250. void ui_draw_alert_if_needed(Canvas *canvas, ProtoViewApp *app);
  251. void canvas_draw_str_with_border(Canvas* canvas, uint8_t x, uint8_t y, const char* str, Color text_color, Color border_color);
  252. /* fields.c */
  253. void fieldset_free(ProtoViewFieldSet *fs);
  254. ProtoViewFieldSet *fieldset_new(void);
  255. void fieldset_add_int(ProtoViewFieldSet *fs, const char *name, int64_t val, uint8_t bits);
  256. void fieldset_add_uint(ProtoViewFieldSet *fs, const char *name, uint64_t uval, uint8_t bits);
  257. void fieldset_add_hex(ProtoViewFieldSet *fs, const char *name, uint64_t uval, uint8_t bits);
  258. void fieldset_add_bin(ProtoViewFieldSet *fs, const char *name, uint64_t uval, uint8_t bits);
  259. void fieldset_add_str(ProtoViewFieldSet *fs, const char *name, const char *s);
  260. void fieldset_add_bytes(ProtoViewFieldSet *fs, const char *name, const uint8_t *bytes, uint32_t count);
  261. void fieldset_add_float(ProtoViewFieldSet *fs, const char *name, float val, uint32_t digits_after_dot);
  262. /* crc.c */
  263. uint8_t crc8(const uint8_t *data, size_t len, uint8_t init, uint8_t poly);