#include "app_console.h" #include "esp_console.h" #include "esp_log.h" #include "argtable3/argtable3.h" #include "wifi_cfg.h" #include "iperf.h" #include "gps_sync.h" #include "wifi_controller.h" #include "esp_wifi.h" #include "nvs.h" #include "nvs_flash.h" #include #include #include #include // --- Helper: Prompt Update --- // Updates the "esp32>" vs "esp32*>" prompt based on dirty state static void end_cmd(void) { app_console_update_prompt(); } // ============================================================================ // COMMAND: nvs (Storage Management) // ============================================================================ static struct { struct arg_lit *dump; struct arg_lit *clear_all; struct arg_lit *help; struct arg_end *end; } nvs_args; static void print_nvs_key_str(nvs_handle_t h, const char *key, const char *label) { char buf[64] = {0}; size_t len = sizeof(buf); if (nvs_get_str(h, key, buf, &len) == ESP_OK) { printf(" %-12s : %s\n", label, buf); } else { printf(" %-12s : \n", label); } } static void print_nvs_key_u32(nvs_handle_t h, const char *key, const char *label) { uint32_t val = 0; if (nvs_get_u32(h, key, &val) == ESP_OK) { printf(" %-12s : %" PRIu32 "\n", label, val); } else { printf(" %-12s : \n", label); } } static void print_nvs_key_u8(nvs_handle_t h, const char *key, const char *label) { uint8_t val = 0; if (nvs_get_u8(h, key, &val) == ESP_OK) { printf(" %-12s : %u\n", label, val); } else { printf(" %-12s : \n", label); } } static int cmd_nvs(int argc, char **argv) { int nerrors = arg_parse(argc, argv, (void **)&nvs_args); if (nerrors > 0) { arg_print_errors(stderr, nvs_args.end, argv[0]); return 1; } if (nvs_args.help->count > 0) { printf("Usage: nvs [--dump] [--clear-all]\n"); return 0; } // --- CLEAR ALL --- if (nvs_args.clear_all->count > 0) { printf("Erasing ALL settings from NVS...\n"); wifi_cfg_clear_credentials(); wifi_cfg_clear_monitor_channel(); iperf_param_clear(); printf("Done. Please reboot.\n"); end_cmd(); return 0; } // --- DUMP --- printf("\n--- [WiFi Config (netcfg)] ---\n"); nvs_handle_t h; if (nvs_open("netcfg", NVS_READONLY, &h) == ESP_OK) { print_nvs_key_str(h, "ssid", "SSID"); print_nvs_key_str(h, "pass", "Password"); print_nvs_key_str(h, "ip", "Static IP"); print_nvs_key_str(h, "mask", "Netmask"); print_nvs_key_str(h, "gw", "Gateway"); print_nvs_key_u8 (h, "dhcp", "DHCP"); print_nvs_key_u8 (h, "mon_ch", "Monitor Ch"); nvs_close(h); } else { printf("Failed to open 'netcfg' namespace.\n"); } printf("\n--- [iPerf Config (storage)] ---\n"); if (nvs_open("storage", NVS_READONLY, &h) == ESP_OK) { print_nvs_key_str(h, "iperf_dst_ip", "Dest IP"); print_nvs_key_u32(h, "iperf_port", "Port"); print_nvs_key_u32(h, "iperf_pps", "Target PPS"); print_nvs_key_u32(h, "iperf_len", "Packet Len"); print_nvs_key_u32(h, "iperf_burst", "Burst"); nvs_close(h); } else { printf("Failed to open 'storage' namespace.\n"); } printf("\n"); end_cmd(); return 0; } static void register_nvs_cmd(void) { nvs_args.dump = arg_lit0(NULL, "dump", "Show all"); nvs_args.clear_all = arg_lit0(NULL, "clear-all", "Factory Reset"); nvs_args.help = arg_lit0("h", "help", "Help"); nvs_args.end = arg_end(1); const esp_console_cmd_t cmd = { .command = "nvs", .help = "Storage Management", .func = &cmd_nvs, .argtable = &nvs_args }; ESP_ERROR_CHECK(esp_console_cmd_register(&cmd)); } // ============================================================================ // COMMAND: gps (Configure & Status) // ============================================================================ static struct { struct arg_lit *enable; struct arg_lit *disable; struct arg_lit *status; struct arg_lit *help; struct arg_end *end; } gps_args; static int cmd_gps(int argc, char **argv) { int nerrors = arg_parse(argc, argv, (void **)&gps_args); if (nerrors > 0) { arg_print_errors(stderr, gps_args.end, argv[0]); return 1; } if (gps_args.help->count > 0) { printf("Usage: gps [--enable|--disable|--status]\n"); return 0; } nvs_handle_t h; esp_err_t err = nvs_open("storage", NVS_READWRITE, &h); if (err != ESP_OK) { printf("Error opening NVS: %s\n", esp_err_to_name(err)); return 1; } // --- HANDLE SETTERS --- bool changed = false; if (gps_args.enable->count > 0) { nvs_set_u8(h, "gps_enabled", 1); printf("GPS set to ENABLED. (Reboot required)\n"); changed = true; } else if (gps_args.disable->count > 0) { nvs_set_u8(h, "gps_enabled", 0); printf("GPS set to DISABLED. (Reboot required)\n"); changed = true; } if (changed) nvs_commit(h); // --- DISPLAY STATUS --- // 1. NVS State uint8_t val = 1; if (nvs_get_u8(h, "gps_enabled", &val) != ESP_OK) val = 1; printf("GPS NVS State: %s\n", val ? "ENABLED" : "DISABLED"); nvs_close(h); // 2. Runtime Status if (val) { gps_timestamp_t ts = gps_get_timestamp(); int64_t pps_age = gps_get_pps_age_ms(); printf("Status: %s\n", ts.synced ? "SYNCED" : "SEARCHING"); // Print Raw NMEA --- char nmea_buf[128]; gps_get_last_nmea(nmea_buf, sizeof(nmea_buf)); // Remove trailing \r\n for cleaner printing size_t len = strlen(nmea_buf); if (len > 0 && (nmea_buf[len-1] == '\r' || nmea_buf[len-1] == '\n')) nmea_buf[len-1] = 0; if (len > 1 && (nmea_buf[len-2] == '\r' || nmea_buf[len-2] == '\n')) nmea_buf[len-2] = 0; printf("Last NMEA: [%s]\n", nmea_buf); // --------------------------- if (pps_age < 0) { printf("PPS Signal: NEVER DETECTED\n"); } else if (pps_age > 1100) { printf("PPS Signal: LOST (Last seen %" PRId64 " ms ago)\n", pps_age); } else { printf("PPS Signal: ACTIVE (Age: %" PRId64 " ms)\n", pps_age); } if (ts.gps_us > 0) { time_t now_sec = ts.gps_us / 1000000; struct tm tm_info; gmtime_r(&now_sec, &tm_info); char time_buf[64]; strftime(time_buf, sizeof(time_buf), "%Y-%m-%d %H:%M:%S UTC", &tm_info); printf("Current Time: %s\n", time_buf); } else { printf("Current Time: (System Epoch)\n"); } } end_cmd(); return 0; } static void register_gps_cmd(void) { gps_args.enable = arg_lit0(NULL, "enable", "Enable GPS"); gps_args.disable = arg_lit0(NULL, "disable", "Disable GPS"); gps_args.status = arg_lit0(NULL, "status", "Show Status"); gps_args.help = arg_lit0("h", "help", "Help"); gps_args.end = arg_end(2); const esp_console_cmd_t cmd = { .command = "gps", .help = "Configure GPS", .func = &cmd_gps, .argtable = &gps_args }; ESP_ERROR_CHECK(esp_console_cmd_register(&cmd)); } // ============================================================================ // COMMAND: iperf // ============================================================================ static struct { struct arg_lit *start, *stop, *status, *save, *reload; struct arg_lit *clear_nvs; struct arg_str *ip; struct arg_int *port, *pps, *len, *burst; struct arg_lit *help; struct arg_end *end; } iperf_args; static int cmd_iperf(int argc, char **argv) { int nerrors = arg_parse(argc, argv, (void **)&iperf_args); if (nerrors > 0) { arg_print_errors(stderr, iperf_args.end, argv[0]); return 1; } if (iperf_args.help->count > 0) { printf("Usage: iperf [options]\n"); printf(" --start Start traffic\n"); printf(" --stop Stop traffic\n"); printf(" --save Save config to NVS\n"); printf(" --clear-nvs Reset to defaults\n"); printf(" -c Set Dest IP\n"); printf(" --pps Set Packets Per Sec\n"); return 0; } if (iperf_args.clear_nvs->count > 0) { iperf_param_clear(); printf("iPerf Configuration cleared (Reset to defaults).\n"); end_cmd(); return 0; } if (iperf_args.reload->count > 0) { iperf_param_init(); printf("Configuration reloaded from NVS.\n"); } bool config_changed = false; iperf_cfg_t cfg; iperf_param_get(&cfg); if (iperf_args.ip->count > 0) { cfg.dip = inet_addr(iperf_args.ip->sval[0]); config_changed = true; } if (iperf_args.port->count > 0) { cfg.dport = (uint16_t)iperf_args.port->ival[0]; config_changed = true; } if (iperf_args.len->count > 0) { cfg.send_len = (uint32_t)iperf_args.len->ival[0]; config_changed = true; } if (iperf_args.burst->count > 0) { cfg.burst_count = (uint32_t)iperf_args.burst->ival[0]; config_changed = true; } if (iperf_args.pps->count > 0) { if (iperf_args.pps->ival[0] > 0) { cfg.target_pps = (uint32_t)iperf_args.pps->ival[0]; config_changed = true; } } if (config_changed) { iperf_param_set(&cfg); printf("RAM configuration updated.\n"); } if (iperf_args.save->count > 0) { bool changed = false; if (iperf_param_save(&changed) == ESP_OK) { printf(changed ? "Configuration saved to NVS.\n" : "No changes to save (NVS matches RAM).\n"); } else { printf("Error saving to NVS.\n"); } } if (iperf_args.stop->count > 0) iperf_stop(); if (iperf_args.start->count > 0) iperf_start(); if (iperf_args.status->count > 0) iperf_print_status(); end_cmd(); return 0; } static void register_iperf_cmd(void) { iperf_args.start = arg_lit0(NULL, "start", "Start"); iperf_args.stop = arg_lit0(NULL, "stop", "Stop"); iperf_args.status = arg_lit0(NULL, "status", "Status"); iperf_args.save = arg_lit0(NULL, "save", "Save"); iperf_args.reload = arg_lit0(NULL, "reload", "Reload"); iperf_args.clear_nvs = arg_lit0(NULL, "clear-nvs", "Clear NVS"); iperf_args.ip = arg_str0("c", "client", "", "IP"); iperf_args.port = arg_int0("p", "port", "", "Port"); iperf_args.pps = arg_int0(NULL, "pps", "", "PPS"); iperf_args.len = arg_int0(NULL, "len", "", "Len"); iperf_args.burst = arg_int0(NULL, "burst", "", "Burst"); iperf_args.help = arg_lit0("h", "help", "Help"); iperf_args.end = arg_end(20); const esp_console_cmd_t cmd = { .command = "iperf", .help = "Traffic Gen", .func = &cmd_iperf, .argtable = &iperf_args }; ESP_ERROR_CHECK(esp_console_cmd_register(&cmd)); } // ============================================================================ // COMMAND: monitor // ============================================================================ static struct { struct arg_lit *start, *stop, *status, *save, *reload; struct arg_lit *clear_nvs; struct arg_int *channel; struct arg_lit *help; struct arg_end *end; } mon_args; static int cmd_monitor(int argc, char **argv) { int nerrors = arg_parse(argc, argv, (void **)&mon_args); if (nerrors > 0) { arg_print_errors(stderr, mon_args.end, argv[0]); return 1; } if (mon_args.help->count > 0) { printf("Usage: monitor [--start|--stop] [-c ] [--save|--reload]\n"); return 0; } if (mon_args.clear_nvs->count > 0) { wifi_ctl_param_clear(); printf("Monitor config cleared (Defaulting to Ch 6).\n"); end_cmd(); return 0; } if (mon_args.reload->count > 0) { wifi_ctl_param_reload(); printf("Config reloaded from NVS.\n"); } if (mon_args.channel->count > 0) { wifi_ctl_param_set_monitor_channel((uint8_t)mon_args.channel->ival[0]); printf("Channel set to %d (RAM).\n", mon_args.channel->ival[0]); } if (mon_args.save->count > 0) { if (wifi_ctl_param_save()) printf("Configuration saved to NVS.\n"); else printf("No changes to save (NVS matches RAM).\n"); } if (mon_args.stop->count > 0) { wifi_ctl_switch_to_sta(WIFI_BW_HT20); printf("Switched to Station Mode.\n"); } if (mon_args.start->count > 0) { if (wifi_ctl_switch_to_monitor(0, WIFI_BW_HT20) == ESP_OK) printf("Monitor Mode Started.\n"); else printf("Failed to start Monitor Mode.\n"); } if (mon_args.status->count > 0) { wifi_ctl_mode_t mode = wifi_ctl_get_mode(); printf("MONITOR STATUS:\n"); printf(" Mode: %s\n", (mode == WIFI_CTL_MODE_MONITOR) ? "MONITOR" : "STATION"); printf(" Active: Ch %d\n", (mode == WIFI_CTL_MODE_MONITOR) ? wifi_ctl_get_monitor_channel() : 0); printf(" Staged: Ch %d\n", wifi_ctl_param_get_monitor_channel()); printf(" Frames: %" PRIu32 "\n", wifi_ctl_get_monitor_frame_count()); } end_cmd(); return 0; } static void register_monitor_cmd(void) { mon_args.start = arg_lit0(NULL, "start", "Start"); mon_args.stop = arg_lit0(NULL, "stop", "Stop"); mon_args.status = arg_lit0(NULL, "status", "Status"); mon_args.save = arg_lit0(NULL, "save", "Save"); mon_args.reload = arg_lit0(NULL, "reload", "Reload"); mon_args.clear_nvs = arg_lit0(NULL, "clear-nvs", "Clear NVS"); mon_args.channel = arg_int0("c", "channel", "", "Chan"); mon_args.help = arg_lit0("h", "help", "Help"); mon_args.end = arg_end(20); const esp_console_cmd_t cmd = { .command = "monitor", .help = "Monitor Mode", .func = &cmd_monitor, .argtable = &mon_args }; ESP_ERROR_CHECK(esp_console_cmd_register(&cmd)); } // ============================================================================ // COMMAND: scan // ============================================================================ static struct { struct arg_lit *help; struct arg_end *end; } scan_args; static int cmd_scan(int argc, char **argv) { int nerrors = arg_parse(argc, argv, (void **)&scan_args); if (nerrors > 0) { arg_print_errors(stderr, scan_args.end, argv[0]); return 1; } if (scan_args.help->count > 0) { printf("Usage: scan\n"); return 0; } printf("Starting WiFi Scan...\n"); // Force STA mode to allow scanning wifi_ctl_switch_to_sta(WIFI_BW_HT20); wifi_scan_config_t scan_config = { .show_hidden = true }; esp_err_t err = esp_wifi_scan_start(&scan_config, true); if (err != ESP_OK) { printf("Scan failed: %s\n", esp_err_to_name(err)); return 1; } uint16_t ap_count = 0; esp_wifi_scan_get_ap_num(&ap_count); printf("Found %d APs:\n", ap_count); if (ap_count > 0) { wifi_ap_record_t *ap_list = (wifi_ap_record_t *)malloc(sizeof(wifi_ap_record_t) * ap_count); if (ap_list) { esp_wifi_scan_get_ap_records(&ap_count, ap_list); printf("%-32s | %-4s | %-4s | %-3s\n", "SSID", "RSSI", "CH", "Auth"); printf("----------------------------------------------------------\n"); for (int i = 0; i < ap_count; i++) { printf("%-32s | %-4d | %-4d | %d\n", (char *)ap_list[i].ssid, ap_list[i].rssi, ap_list[i].primary, ap_list[i].authmode); } free(ap_list); } } return 0; } static void register_scan_cmd(void) { scan_args.help = arg_lit0("h", "help", "Help"); scan_args.end = arg_end(1); const esp_console_cmd_t cmd = { .command = "scan", .help = "WiFi Scan", .func = &cmd_scan, .argtable = &scan_args }; ESP_ERROR_CHECK(esp_console_cmd_register(&cmd)); } // ============================================================================ // COMMAND: wifi_config // ============================================================================ static struct { struct arg_str *ssid; struct arg_str *pass; struct arg_str *ip; struct arg_lit *dhcp; struct arg_lit *clear_nvs; struct arg_lit *help; struct arg_end *end; } wifi_args; static int cmd_wifi_config(int argc, char **argv) { int nerrors = arg_parse(argc, argv, (void **)&wifi_args); if (nerrors > 0) { arg_print_errors(stderr, wifi_args.end, argv[0]); return 1; } if (wifi_args.help->count > 0) { printf("Usage: wifi_config -s [-p ] [--clear-nvs]\n"); return 0; } if (wifi_args.clear_nvs->count > 0) { wifi_cfg_clear_credentials(); printf("WiFi Credentials CLEARED from NVS.\n"); return 0; } if (wifi_args.ssid->count == 0) { printf("Error: SSID is required (-s)\n"); return 1; } const char* ssid = wifi_args.ssid->sval[0]; const char* pass = (wifi_args.pass->count > 0) ? wifi_args.pass->sval[0] : ""; const char* ip = (wifi_args.ip->count > 0) ? wifi_args.ip->sval[0] : NULL; bool dhcp = (wifi_args.dhcp->count > 0); printf("Saving WiFi Config: SSID='%s' DHCP=%d\n", ssid, dhcp); wifi_cfg_set_credentials(ssid, pass); if (ip) { char mask[] = "255.255.255.0"; char gw[32]; strlcpy(gw, ip, sizeof(gw)); char *last_dot = strrchr(gw, '.'); if (last_dot) strcpy(last_dot, ".1"); wifi_cfg_set_static_ip(ip, mask, gw); wifi_cfg_set_dhcp(false); } else { wifi_cfg_set_dhcp(dhcp); } printf("Config saved. Rebooting to apply...\n"); esp_restart(); return 0; } static void register_wifi_cmd(void) { wifi_args.ssid = arg_str0("s", "ssid", "", "SSID"); wifi_args.pass = arg_str0("p", "password", "", "Pass"); wifi_args.ip = arg_str0("i", "ip", "", "Static IP"); wifi_args.dhcp = arg_lit0("d", "dhcp", "Enable DHCP"); wifi_args.clear_nvs = arg_lit0(NULL, "clear-nvs", "Clear NVS"); wifi_args.help = arg_lit0("h", "help", "Help"); wifi_args.end = arg_end(20); const esp_console_cmd_t cmd = { .command = "wifi_config", .help = "Configure WiFi", .func = &cmd_wifi_config, .argtable = &wifi_args }; ESP_ERROR_CHECK(esp_console_cmd_register(&cmd)); } void app_console_register_commands(void) { register_iperf_cmd(); register_monitor_cmd(); register_scan_cmd(); register_wifi_cmd(); register_nvs_cmd(); register_gps_cmd(); }