wardriver.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360
  1. #include "wardriver.h"
  2. #include "wardriver_uart.h"
  3. #include <expansion/expansion.h>
  4. void save_file(Context* ctx) {
  5. Storage* storage = furi_record_open(RECORD_STORAGE);
  6. DateTime datetime;
  7. furi_hal_rtc_get_datetime(&datetime);
  8. FuriString* filename = furi_string_alloc();
  9. furi_string_printf(
  10. filename,
  11. "%s/%s_%d_%d_%d_%d_%d_%d.txt",
  12. FILE_PATH,
  13. "wigle",
  14. datetime.year,
  15. datetime.month,
  16. datetime.day,
  17. datetime.hour,
  18. datetime.minute,
  19. datetime.second);
  20. File* file = storage_file_alloc(storage);
  21. if(!storage_common_exists(storage, FILE_PATH)) {
  22. storage_common_mkdir(storage, FILE_PATH);
  23. }
  24. if(!storage_file_open(file, furi_string_get_cstr(filename), FSAM_WRITE, FSOM_OPEN_ALWAYS)) {
  25. FURI_LOG_I(appname, "Failed to open file");
  26. storage_file_close(file);
  27. storage_file_free(file);
  28. furi_string_free(filename);
  29. furi_record_close(RECORD_STORAGE);
  30. furi_hal_light_blink_start(LightRed, 100, 100, 5000);
  31. return;
  32. }
  33. furi_string_reset(ctx->buffer);
  34. // WIGLE HEADERS DONT CHANGE THIS ITS IMPORTANT!
  35. furi_string_printf(
  36. ctx->buffer,
  37. "%s,%s,%s,%s,%s,%s,%s,%s\r\n",
  38. "WigleWifi-1.4",
  39. "appRelease=v2.0",
  40. "model=S33",
  41. "release=XtremeFW",
  42. "Flipper Zero",
  43. "",
  44. "Wardriver",
  45. "S33");
  46. furi_string_cat_printf(
  47. ctx->buffer,
  48. "%s,%s,%s,%s,%s,%s,%s\r\n",
  49. "MAC",
  50. "SSID",
  51. "FirstSeen",
  52. "Channel",
  53. "RSSI",
  54. "CurrentLatitude",
  55. "CurrentLongitude");
  56. if(!storage_file_write(
  57. file, furi_string_get_cstr(ctx->buffer), strlen(furi_string_get_cstr(ctx->buffer)))) {
  58. FURI_LOG_I(appname, "Failed to write header to file");
  59. }
  60. for(int i = 0; i < ctx->access_points_count; i++) {
  61. AccessPoint ap = ctx->access_points[i];
  62. furi_string_printf(
  63. ctx->buffer,
  64. "%s,%s,%04d-%02d-%02d %02d:%02d:%02d,%d,%d,%f,%f\r\n",
  65. ap.bssid,
  66. ap.ssid,
  67. ap.datetime.year,
  68. ap.datetime.month,
  69. ap.datetime.day,
  70. ap.datetime.hour,
  71. ap.datetime.minute,
  72. ap.datetime.second,
  73. ap.rssi,
  74. ap.channel,
  75. (double)ap.latitude,
  76. (double)ap.longitude);
  77. if(!storage_file_write(
  78. file,
  79. furi_string_get_cstr(ctx->buffer),
  80. strlen(furi_string_get_cstr(ctx->buffer)))) {
  81. FURI_LOG_I(appname, "Failed to write AP to file");
  82. }
  83. free(ap.ssid);
  84. free(ap.bssid);
  85. }
  86. storage_file_close(file);
  87. storage_file_free(file);
  88. furi_string_free(filename);
  89. furi_record_close(RECORD_STORAGE);
  90. return;
  91. }
  92. static void tick_callback(void* ctx_q) {
  93. furi_assert(ctx_q);
  94. FuriMessageQueue* queue = ctx_q;
  95. Event event = {.type = EventTypeTick};
  96. furi_message_queue_put(queue, &event, 0);
  97. }
  98. static void input_callback(InputEvent* input_event, FuriMessageQueue* queue) {
  99. furi_assert(queue);
  100. Event event = {.type = EventTypeKey, .input = *input_event};
  101. furi_message_queue_put(queue, &event, FuriWaitForever);
  102. }
  103. static void draw_access_point(Canvas* canvas, Context* context) {
  104. Context* ctx = context;
  105. AccessPoint ap = ctx->active_access_point;
  106. canvas_draw_str_aligned(canvas, 62, 25, AlignCenter, AlignBottom, ap.ssid);
  107. canvas_set_font(canvas, FontSecondary);
  108. canvas_draw_str_aligned(canvas, 38, 12, AlignLeft, AlignBottom, ap.bssid);
  109. furi_string_printf(ctx->buffer, "Signal strength: %ddBm", ap.rssi);
  110. canvas_draw_str_aligned(
  111. canvas, 3, 35, AlignLeft, AlignBottom, furi_string_get_cstr(ctx->buffer));
  112. furi_string_printf(ctx->buffer, "CH: %d", ap.channel);
  113. canvas_draw_str_aligned(
  114. canvas, 3, 47, AlignLeft, AlignBottom, furi_string_get_cstr(ctx->buffer));
  115. furi_string_printf(ctx->buffer, "%d", ap.packetRxCount);
  116. canvas_draw_icon(canvas, 35, 39, &I_down);
  117. canvas_draw_str_aligned(
  118. canvas, 45, 47, AlignLeft, AlignBottom, furi_string_get_cstr(ctx->buffer));
  119. furi_string_printf(ctx->buffer, "%d", ap.packetTxCount);
  120. canvas_draw_icon(canvas, 85, 38, &I_up);
  121. canvas_draw_str_aligned(
  122. canvas, 95, 47, AlignLeft, AlignBottom, furi_string_get_cstr(ctx->buffer));
  123. furi_string_printf(
  124. ctx->buffer,
  125. "Seen: %02d:%02d:%02d (%lds ago)",
  126. ap.datetime.hour,
  127. ap.datetime.minute,
  128. ap.datetime.second,
  129. furi_hal_rtc_get_timestamp() - datetime_datetime_to_timestamp(&ap.datetime));
  130. canvas_draw_str_aligned(
  131. canvas, 3, 59, AlignLeft, AlignBottom, furi_string_get_cstr(ctx->buffer));
  132. }
  133. static void render_callback(Canvas* canvas, void* context) {
  134. Context* ctx = context;
  135. canvas_set_font(canvas, FontPrimary);
  136. if(ctx->access_points_count >= MAX_ACCESS_POINTS) {
  137. canvas_draw_str(canvas, 118, 10, "!");
  138. }
  139. switch(ctx->view_state) {
  140. case SHOW_NMEA:
  141. if(UART_CH_ESP == UART_CH_GPS) {
  142. canvas_draw_str(canvas, 0, 10, "GPS channel invalid!");
  143. canvas_draw_str(canvas, 0, 20, "Change UART");
  144. canvas_draw_str(canvas, 0, 30, "channel");
  145. canvas_draw_str(canvas, 0, 40, "in the Xtreme");
  146. canvas_draw_str(canvas, 0, 50, "app");
  147. } else {
  148. furi_string_printf(
  149. ctx->buffer,
  150. "%f",
  151. isnan(ctx->gps_data.latitude) ? 0 : (double)ctx->gps_data.latitude);
  152. canvas_draw_str(canvas, 0, 10, furi_string_get_cstr(ctx->buffer));
  153. furi_string_printf(
  154. ctx->buffer,
  155. "%f",
  156. isnan(ctx->gps_data.longitude) ? 0 : (double)ctx->gps_data.longitude);
  157. canvas_draw_str(canvas, 0, 20, furi_string_get_cstr(ctx->buffer));
  158. furi_string_printf(ctx->buffer, "%d sats", ctx->gps_data.satelites);
  159. canvas_draw_str(canvas, 0, 30, furi_string_get_cstr(ctx->buffer));
  160. furi_string_printf(
  161. ctx->buffer,
  162. "%02d:%02d:%02dZ",
  163. ctx->gps_data.hour,
  164. ctx->gps_data.minute,
  165. ctx->gps_data.second);
  166. canvas_draw_str(canvas, 0, 40, furi_string_get_cstr(ctx->buffer));
  167. canvas_draw_str(canvas, 70, 10, "GPS DATA");
  168. }
  169. elements_button_left(canvas, "Back");
  170. canvas_draw_icon(canvas, 82, 20, &I_WarningDolphinFlip_45x42);
  171. break;
  172. case NO_APS:
  173. canvas_draw_str(canvas, 80, 30, "No AP's");
  174. canvas_draw_str(canvas, 80, 40, "Found!");
  175. canvas_draw_icon(canvas, 1, 4, &I_DolphinWait_61x59);
  176. break;
  177. case NORMAL:
  178. default:
  179. canvas_draw_frame(canvas, 0, 0, 128, 64);
  180. furi_string_printf(
  181. ctx->buffer, "%d/%d", ctx->access_points_index + 1, ctx->access_points_count);
  182. canvas_draw_str(canvas, 3, 12, furi_string_get_cstr(ctx->buffer));
  183. draw_access_point(canvas, ctx);
  184. break;
  185. }
  186. furi_mutex_release(ctx->mutex);
  187. }
  188. int32_t wardriver_app() {
  189. // Disable expansion protocol to avoid interference with UART Handle
  190. Expansion* expansion = furi_record_open(RECORD_EXPANSION);
  191. expansion_disable(expansion);
  192. // turn off 5v, so it gets reset on startup
  193. if(furi_hal_power_is_otg_enabled()) {
  194. furi_hal_power_disable_otg();
  195. }
  196. // Enable 5v on startup
  197. uint8_t attempts = 0;
  198. while(!furi_hal_power_is_otg_enabled() && attempts++ < 5) {
  199. furi_hal_power_enable_otg();
  200. furi_delay_ms(10);
  201. }
  202. furi_delay_ms(200);
  203. Context* ctx = malloc(sizeof(Context));
  204. ctx->queue = furi_message_queue_alloc(8, sizeof(Event));
  205. ctx->mutex = furi_mutex_alloc(FuriMutexTypeNormal);
  206. ctx->buffer = furi_string_alloc();
  207. ctx->access_points_count = 0;
  208. ctx->access_points_index = 0;
  209. ctx->pressedButton = false;
  210. ctx->view_state = NO_APS;
  211. wardriver_uart_init(ctx);
  212. ViewPort* view_port = view_port_alloc();
  213. view_port_draw_callback_set(view_port, render_callback, ctx);
  214. view_port_input_callback_set(view_port, input_callback, ctx->queue);
  215. Gui* gui = furi_record_open(RECORD_GUI);
  216. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  217. FuriTimer* timer = furi_timer_alloc(tick_callback, FuriTimerTypePeriodic, ctx->queue);
  218. furi_timer_start(timer, 100);
  219. // application loop
  220. Event event;
  221. bool processing = true;
  222. do {
  223. if(furi_message_queue_get(ctx->queue, &event, FuriWaitForever) == FuriStatusOk) {
  224. furi_mutex_acquire(ctx->mutex, FuriWaitForever);
  225. switch(event.type) {
  226. case EventTypeKey:
  227. if(event.input.type == InputTypeShort && event.input.key == InputKeyBack) {
  228. if(ctx->view_state == SHOW_NMEA) {
  229. ctx->view_state = NORMAL;
  230. } else {
  231. processing = false;
  232. }
  233. } else if(event.input.type == InputTypeLong && event.input.key == InputKeyBack) {
  234. if(ctx->view_state == SHOW_NMEA) {
  235. ctx->view_state = NORMAL;
  236. } else {
  237. processing = false;
  238. }
  239. } else if(event.input.type == InputTypeLong && event.input.key == InputKeyOk) {
  240. } else if(event.input.type == InputTypePress && event.input.key == InputKeyDown) {
  241. ctx->access_points_index--;
  242. if(ctx->access_points_index < 0) {
  243. ctx->access_points_index = ctx->access_points_count - 1;
  244. }
  245. ctx->active_access_point = ctx->access_points[ctx->access_points_index];
  246. ctx->pressedButton = true;
  247. } else if(event.input.type == InputTypePress && event.input.key == InputKeyUp) {
  248. ctx->access_points_index++;
  249. if(ctx->access_points_index >= ctx->access_points_count) {
  250. ctx->access_points_index = 0;
  251. }
  252. ctx->active_access_point = ctx->access_points[ctx->access_points_index];
  253. ctx->pressedButton = true;
  254. } else if(event.input.type == InputTypePress && event.input.key == InputKeyLeft) {
  255. if(ctx->view_state == NORMAL) {
  256. ctx->view_state = SHOW_NMEA;
  257. } else if(ctx->view_state == SHOW_NMEA) {
  258. ctx->view_state = NORMAL;
  259. }
  260. } else if(event.input.type == InputTypePress && event.input.key == InputKeyRight) {
  261. }
  262. break;
  263. case EventTypeTick:
  264. // fix for the empty active access point when there was no interaction
  265. if(!ctx->pressedButton) {
  266. ctx->access_points_index = 0;
  267. ctx->active_access_point = ctx->access_points[ctx->access_points_index];
  268. }
  269. break;
  270. default:
  271. break;
  272. }
  273. view_port_update(view_port);
  274. } else {
  275. processing = false;
  276. }
  277. } while(processing);
  278. furi_timer_free(timer);
  279. view_port_enabled_set(view_port, false);
  280. gui_remove_view_port(gui, view_port);
  281. view_port_free(view_port);
  282. furi_record_close(RECORD_GUI);
  283. furi_message_queue_free(ctx->queue);
  284. furi_mutex_free(ctx->mutex);
  285. furi_string_free(ctx->buffer);
  286. wardriver_uart_deinit(ctx);
  287. save_file(ctx);
  288. free(ctx);
  289. furi_hal_light_set(LightBlue, 0);
  290. furi_hal_light_set(LightGreen, 0);
  291. if(furi_hal_power_is_otg_enabled()) {
  292. furi_hal_power_disable_otg();
  293. }
  294. // Return previous state of expansion
  295. expansion_enable(expansion);
  296. furi_record_close(RECORD_EXPANSION);
  297. return 0;
  298. }