2 * WPA Supplicant - command line interface for wpa_supplicant daemon
3 * Copyright (c) 2004-2015, Jouni Malinen <j@w1.fi>
5 * This software may be distributed under the terms of the BSD license.
6 * See README for more details.
11 #ifdef CONFIG_CTRL_IFACE
13 #ifdef CONFIG_CTRL_IFACE_UNIX
15 #endif /* CONFIG_CTRL_IFACE_UNIX */
17 #include "common/wpa_ctrl.h"
18 #include "utils/common.h"
19 #include "utils/eloop.h"
20 #include "utils/edit.h"
21 #include "utils/list.h"
22 #include "common/version.h"
23 #include "common/ieee802_11_defs.h"
25 #include <cutils/properties.h>
29 static const char *const wpa_cli_version =
30 "wpa_cli v" VERSION_STR "\n"
31 "Copyright (c) 2004-2015, Jouni Malinen <j@w1.fi> and contributors";
34 static const char *const wpa_cli_license =
35 "This software may be distributed under the terms of the BSD license.\n"
36 "See README for more details.\n";
38 static const char *const wpa_cli_full_license =
39 "This software may be distributed under the terms of the BSD license.\n"
41 "Redistribution and use in source and binary forms, with or without\n"
42 "modification, are permitted provided that the following conditions are\n"
45 "1. Redistributions of source code must retain the above copyright\n"
46 " notice, this list of conditions and the following disclaimer.\n"
48 "2. Redistributions in binary form must reproduce the above copyright\n"
49 " notice, this list of conditions and the following disclaimer in the\n"
50 " documentation and/or other materials provided with the distribution.\n"
52 "3. Neither the name(s) of the above-listed copyright holder(s) nor the\n"
53 " names of its contributors may be used to endorse or promote products\n"
54 " derived from this software without specific prior written permission.\n"
56 "THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS\n"
57 "\"AS IS\" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT\n"
58 "LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR\n"
59 "A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT\n"
60 "OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,\n"
61 "SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT\n"
62 "LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,\n"
63 "DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY\n"
64 "THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT\n"
65 "(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE\n"
66 "OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\n"
69 static struct wpa_ctrl *ctrl_conn;
70 static struct wpa_ctrl *mon_conn;
71 static int wpa_cli_quit = 0;
72 static int wpa_cli_attached = 0;
73 static int wpa_cli_connected = -1;
74 static int wpa_cli_last_id = 0;
75 #ifndef CONFIG_CTRL_IFACE_DIR
76 #define CONFIG_CTRL_IFACE_DIR "/var/run/wpa_supplicant"
77 #endif /* CONFIG_CTRL_IFACE_DIR */
78 static const char *ctrl_iface_dir = CONFIG_CTRL_IFACE_DIR;
79 static char *ctrl_ifname = NULL;
80 static const char *pid_file = NULL;
81 static const char *action_file = NULL;
82 static int ping_interval = 5;
83 static int interactive = 0;
84 static char *ifname_prefix = NULL;
86 struct cli_txt_entry {
91 static DEFINE_DL_LIST(bsses); /* struct cli_txt_entry */
92 static DEFINE_DL_LIST(p2p_peers); /* struct cli_txt_entry */
93 static DEFINE_DL_LIST(p2p_groups); /* struct cli_txt_entry */
94 static DEFINE_DL_LIST(ifnames); /* struct cli_txt_entry */
95 static DEFINE_DL_LIST(networks); /* struct cli_txt_entry */
98 static void print_help(const char *cmd);
99 static void wpa_cli_mon_receive(int sock, void *eloop_ctx, void *sock_ctx);
100 static void wpa_cli_close_connection(void);
101 static char * wpa_cli_get_default_ifname(void);
102 static char ** wpa_list_cmd_list(void);
103 static void update_networks(struct wpa_ctrl *ctrl);
106 static void usage(void)
108 printf("wpa_cli [-p<path to ctrl sockets>] [-i<ifname>] [-hvB] "
109 "[-a<action file>] \\\n"
110 " [-P<pid file>] [-g<global ctrl>] [-G<ping interval>] "
112 " -h = help (show this usage text)\n"
113 " -v = shown version information\n"
114 " -a = run in daemon mode executing the action file based on "
117 " -B = run a daemon in the background\n"
118 " default path: " CONFIG_CTRL_IFACE_DIR "\n"
119 " default interface: first interface found in socket path\n");
124 static void cli_txt_list_free(struct cli_txt_entry *e)
126 dl_list_del(&e->list);
132 static void cli_txt_list_flush(struct dl_list *list)
134 struct cli_txt_entry *e;
135 while ((e = dl_list_first(list, struct cli_txt_entry, list)))
136 cli_txt_list_free(e);
140 static struct cli_txt_entry * cli_txt_list_get(struct dl_list *txt_list,
143 struct cli_txt_entry *e;
144 dl_list_for_each(e, txt_list, struct cli_txt_entry, list) {
145 if (os_strcmp(e->txt, txt) == 0)
152 static void cli_txt_list_del(struct dl_list *txt_list, const char *txt)
154 struct cli_txt_entry *e;
155 e = cli_txt_list_get(txt_list, txt);
157 cli_txt_list_free(e);
161 static void cli_txt_list_del_addr(struct dl_list *txt_list, const char *txt)
165 if (hwaddr_aton(txt, addr) < 0)
167 os_snprintf(buf, sizeof(buf), MACSTR, MAC2STR(addr));
168 cli_txt_list_del(txt_list, buf);
173 static void cli_txt_list_del_word(struct dl_list *txt_list, const char *txt,
178 end = os_strchr(txt, separator);
180 end = txt + os_strlen(txt);
181 buf = dup_binstr(txt, end - txt);
184 cli_txt_list_del(txt_list, buf);
187 #endif /* CONFIG_P2P */
190 static int cli_txt_list_add(struct dl_list *txt_list, const char *txt)
192 struct cli_txt_entry *e;
193 e = cli_txt_list_get(txt_list, txt);
196 e = os_zalloc(sizeof(*e));
199 e->txt = os_strdup(txt);
200 if (e->txt == NULL) {
204 dl_list_add(txt_list, &e->list);
210 static int cli_txt_list_add_addr(struct dl_list *txt_list, const char *txt)
214 if (hwaddr_aton(txt, addr) < 0)
216 os_snprintf(buf, sizeof(buf), MACSTR, MAC2STR(addr));
217 return cli_txt_list_add(txt_list, buf);
219 #endif /* CONFIG_P2P */
222 static int cli_txt_list_add_word(struct dl_list *txt_list, const char *txt,
228 end = os_strchr(txt, separator);
230 end = txt + os_strlen(txt);
231 buf = dup_binstr(txt, end - txt);
234 ret = cli_txt_list_add(txt_list, buf);
240 static char ** cli_txt_list_array(struct dl_list *txt_list)
242 unsigned int i, count = dl_list_len(txt_list);
244 struct cli_txt_entry *e;
246 res = os_calloc(count + 1, sizeof(char *));
251 dl_list_for_each(e, txt_list, struct cli_txt_entry, list) {
252 res[i] = os_strdup(e->txt);
262 static int get_cmd_arg_num(const char *str, int pos)
266 for (i = 0; i <= pos; i++) {
269 while (i <= pos && str[i] != ' ')
280 static int str_starts(const char *src, const char *match)
282 return os_strncmp(src, match, os_strlen(match)) == 0;
286 static int wpa_cli_show_event(const char *event)
290 start = os_strchr(event, '>');
296 * Skip BSS added/removed events since they can be relatively frequent
297 * and are likely of not much use for an interactive user.
299 if (str_starts(start, WPA_EVENT_BSS_ADDED) ||
300 str_starts(start, WPA_EVENT_BSS_REMOVED))
307 static int wpa_cli_open_connection(const char *ifname, int attach)
309 #if defined(CONFIG_CTRL_IFACE_UDP) || defined(CONFIG_CTRL_IFACE_NAMED_PIPE)
310 ctrl_conn = wpa_ctrl_open(ifname);
311 if (ctrl_conn == NULL)
314 if (attach && interactive)
315 mon_conn = wpa_ctrl_open(ifname);
318 #else /* CONFIG_CTRL_IFACE_UDP || CONFIG_CTRL_IFACE_NAMED_PIPE */
326 if (access(ctrl_iface_dir, F_OK) < 0) {
327 cfile = os_strdup(ifname);
334 flen = os_strlen(ctrl_iface_dir) + os_strlen(ifname) + 2;
335 cfile = os_malloc(flen);
338 res = os_snprintf(cfile, flen, "%s/%s", ctrl_iface_dir,
340 if (os_snprintf_error(flen, res)) {
346 ctrl_conn = wpa_ctrl_open(cfile);
347 if (ctrl_conn == NULL) {
352 if (attach && interactive)
353 mon_conn = wpa_ctrl_open(cfile);
357 #endif /* CONFIG_CTRL_IFACE_UDP || CONFIG_CTRL_IFACE_NAMED_PIPE */
360 if (wpa_ctrl_attach(mon_conn) == 0) {
361 wpa_cli_attached = 1;
363 eloop_register_read_sock(
364 wpa_ctrl_get_fd(mon_conn),
365 wpa_cli_mon_receive, NULL, NULL);
367 printf("Warning: Failed to attach to "
368 "wpa_supplicant.\n");
369 wpa_cli_close_connection();
378 static void wpa_cli_close_connection(void)
380 if (ctrl_conn == NULL)
383 if (wpa_cli_attached) {
384 wpa_ctrl_detach(interactive ? mon_conn : ctrl_conn);
385 wpa_cli_attached = 0;
387 wpa_ctrl_close(ctrl_conn);
390 eloop_unregister_read_sock(wpa_ctrl_get_fd(mon_conn));
391 wpa_ctrl_close(mon_conn);
397 static void wpa_cli_msg_cb(char *msg, size_t len)
403 static int _wpa_ctrl_command(struct wpa_ctrl *ctrl, char *cmd, int print)
409 if (ctrl_conn == NULL) {
410 printf("Not connected to wpa_supplicant - command dropped.\n");
414 os_snprintf(buf, sizeof(buf), "IFNAME=%s %s",
416 buf[sizeof(buf) - 1] = '\0';
419 len = sizeof(buf) - 1;
420 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len,
423 printf("'%s' command timed out.\n", cmd);
425 } else if (ret < 0) {
426 printf("'%s' command failed.\n", cmd);
432 if (interactive && len > 0 && buf[len - 1] != '\n')
439 static int wpa_ctrl_command(struct wpa_ctrl *ctrl, char *cmd)
441 return _wpa_ctrl_command(ctrl, cmd, 1);
445 static int write_cmd(char *buf, size_t buflen, const char *cmd, int argc,
454 res = os_snprintf(pos, end - pos, "%s", cmd);
455 if (os_snprintf_error(end - pos, res))
459 for (i = 0; i < argc; i++) {
460 res = os_snprintf(pos, end - pos, " %s", argv[i]);
461 if (os_snprintf_error(end - pos, res))
466 buf[buflen - 1] = '\0';
470 printf("Too long command\n");
475 static int wpa_cli_cmd(struct wpa_ctrl *ctrl, const char *cmd, int min_args,
476 int argc, char *argv[])
479 if (argc < min_args) {
480 printf("Invalid %s command - at least %d argument%s "
481 "required.\n", cmd, min_args,
482 min_args > 1 ? "s are" : " is");
485 if (write_cmd(buf, sizeof(buf), cmd, argc, argv) < 0)
487 return wpa_ctrl_command(ctrl, buf);
491 static int wpa_cli_cmd_ifname(struct wpa_ctrl *ctrl, int argc, char *argv[])
493 return wpa_ctrl_command(ctrl, "IFNAME");
497 static int wpa_cli_cmd_status(struct wpa_ctrl *ctrl, int argc, char *argv[])
499 if (argc > 0 && os_strcmp(argv[0], "verbose") == 0)
500 return wpa_ctrl_command(ctrl, "STATUS-VERBOSE");
501 if (argc > 0 && os_strcmp(argv[0], "wps") == 0)
502 return wpa_ctrl_command(ctrl, "STATUS-WPS");
503 if (argc > 0 && os_strcmp(argv[0], "driver") == 0)
504 return wpa_ctrl_command(ctrl, "STATUS-DRIVER");
505 return wpa_ctrl_command(ctrl, "STATUS");
509 static int wpa_cli_cmd_ping(struct wpa_ctrl *ctrl, int argc, char *argv[])
511 return wpa_ctrl_command(ctrl, "PING");
515 static int wpa_cli_cmd_relog(struct wpa_ctrl *ctrl, int argc, char *argv[])
517 return wpa_ctrl_command(ctrl, "RELOG");
521 static int wpa_cli_cmd_note(struct wpa_ctrl *ctrl, int argc, char *argv[])
523 return wpa_cli_cmd(ctrl, "NOTE", 1, argc, argv);
527 static int wpa_cli_cmd_mib(struct wpa_ctrl *ctrl, int argc, char *argv[])
529 return wpa_ctrl_command(ctrl, "MIB");
533 static int wpa_cli_cmd_pmksa(struct wpa_ctrl *ctrl, int argc, char *argv[])
535 return wpa_ctrl_command(ctrl, "PMKSA");
539 static int wpa_cli_cmd_pmksa_flush(struct wpa_ctrl *ctrl, int argc,
542 return wpa_ctrl_command(ctrl, "PMKSA_FLUSH");
546 static int wpa_cli_cmd_help(struct wpa_ctrl *ctrl, int argc, char *argv[])
548 print_help(argc > 0 ? argv[0] : NULL);
553 static char ** wpa_cli_complete_help(const char *str, int pos)
555 int arg = get_cmd_arg_num(str, pos);
560 res = wpa_list_cmd_list();
568 static int wpa_cli_cmd_license(struct wpa_ctrl *ctrl, int argc, char *argv[])
570 printf("%s\n\n%s\n", wpa_cli_version, wpa_cli_full_license);
575 static int wpa_cli_cmd_quit(struct wpa_ctrl *ctrl, int argc, char *argv[])
584 static int wpa_cli_cmd_set(struct wpa_ctrl *ctrl, int argc, char *argv[])
590 res = os_snprintf(cmd, sizeof(cmd), "SET %s ", argv[0]);
591 if (os_snprintf_error(sizeof(cmd), res)) {
592 printf("Too long SET command.\n");
595 return wpa_ctrl_command(ctrl, cmd);
598 return wpa_cli_cmd(ctrl, "SET", 2, argc, argv);
602 static char ** wpa_cli_complete_set(const char *str, int pos)
604 int arg = get_cmd_arg_num(str, pos);
605 const char *fields[] = {
607 "EAPOL::heldPeriod", "EAPOL::authPeriod", "EAPOL::startPeriod",
608 "EAPOL::maxStart", "dot11RSNAConfigPMKLifetime",
609 "dot11RSNAConfigPMKReauthThreshold", "dot11RSNAConfigSATimeout",
610 "wps_fragment_size", "wps_version_number", "ampdu",
611 "tdls_testing", "tdls_disabled", "pno", "radio_disabled",
612 "uapsd", "ps", "wifi_display", "bssid_filter", "disallow_aps",
614 /* global configuration parameters */
615 #ifdef CONFIG_CTRL_IFACE
616 "ctrl_interface", "no_ctrl_interface", "ctrl_interface_group",
617 #endif /* CONFIG_CTRL_IFACE */
618 "eapol_version", "ap_scan", "bgscan",
620 "user_mpm", "max_peer_links", "mesh_max_inactivity",
621 #endif /* CONFIG_MESH */
622 "disable_scan_offload", "fast_reauth", "opensc_engine_path",
623 "pkcs11_engine_path", "pkcs11_module_path", "openssl_ciphers",
624 "pcsc_reader", "pcsc_pin", "external_sim", "driver_param",
625 "dot11RSNAConfigPMKLifetime",
626 "dot11RSNAConfigPMKReauthThreshold",
627 "dot11RSNAConfigSATimeout",
628 #ifndef CONFIG_NO_CONFIG_WRITE
630 #endif /* CONFIG_NO_CONFIG_WRITE */
633 "uuid", "device_name", "manufacturer", "model_name",
634 "model_number", "serial_number", "device_type", "os_version",
635 "config_methods", "wps_cred_processing", "wps_vendor_ext_m1",
636 #endif /* CONFIG_WPS */
639 "p2p_listen_reg_class", "p2p_listen_channel",
640 "p2p_oper_reg_class", "p2p_oper_channel", "p2p_go_intent",
641 "p2p_ssid_postfix", "persistent_reconnect", "p2p_intra_bss",
642 "p2p_group_idle", "p2p_passphrase_len", "p2p_pref_chan",
643 "p2p_no_go_freq", "p2p_add_cli_chan",
644 "p2p_optimize_listen_chan", "p2p_go_ht40", "p2p_go_vht",
645 "p2p_disabled", "p2p_go_ctwindow", "p2p_no_group_iface",
646 "p2p_ignore_shared_freq", "ip_addr_go", "ip_addr_mask",
647 "ip_addr_start", "ip_addr_end",
648 #endif /* CONFIG_P2P */
649 "country", "bss_max_count", "bss_expiration_age",
650 "bss_expiration_scan_count", "filter_ssids", "filter_rssi",
651 "max_num_sta", "disassoc_low_ack",
654 #endif /* CONFIG_HS20 */
655 "interworking", "hessid", "access_network_type", "pbc_in_m1",
656 "autoscan", "wps_nfc_dev_pw_id", "wps_nfc_dh_pubkey",
657 "wps_nfc_dh_privkey", "wps_nfc_dev_pw", "ext_password_backend",
658 "p2p_go_max_inactivity", "auto_interworking", "okc", "pmf",
659 "sae_groups", "dtim_period", "beacon_int",
660 "ap_vendor_elements", "ignore_old_scan_res", "freq_list",
661 "scan_cur_freq", "sched_scan_interval",
662 "tdls_external_control", "osu_dir", "wowlan_triggers",
663 "p2p_search_delay", "mac_addr", "rand_addr_lifetime",
664 "preassoc_mac_addr", "key_mgmt_offload", "passive_scan",
665 "reassoc_same_bss_optim", "wps_priority"
667 int i, num_fields = ARRAY_SIZE(fields);
670 char **res = os_calloc(num_fields + 1, sizeof(char *));
673 for (i = 0; i < num_fields; i++) {
674 res[i] = os_strdup(fields[i]);
681 if (arg > 1 && os_strncasecmp(str, "set bssid_filter ", 17) == 0)
682 return cli_txt_list_array(&bsses);
687 static int wpa_cli_cmd_dump(struct wpa_ctrl *ctrl, int argc, char *argv[])
689 return wpa_ctrl_command(ctrl, "DUMP");
693 static int wpa_cli_cmd_get(struct wpa_ctrl *ctrl, int argc, char *argv[])
695 return wpa_cli_cmd(ctrl, "GET", 1, argc, argv);
699 static char ** wpa_cli_complete_get(const char *str, int pos)
701 int arg = get_cmd_arg_num(str, pos);
702 const char *fields[] = {
703 #ifdef CONFIG_CTRL_IFACE
704 "ctrl_interface", "ctrl_interface_group",
705 #endif /* CONFIG_CTRL_IFACE */
706 "eapol_version", "ap_scan",
708 "user_mpm", "max_peer_links", "mesh_max_inactivity",
709 #endif /* CONFIG_MESH */
710 "disable_scan_offload", "fast_reauth", "opensc_engine_path",
711 "pkcs11_engine_path", "pkcs11_module_path", "openssl_ciphers",
712 "pcsc_reader", "pcsc_pin", "external_sim", "driver_param",
713 "dot11RSNAConfigPMKLifetime",
714 "dot11RSNAConfigPMKReauthThreshold",
715 "dot11RSNAConfigSATimeout",
716 #ifndef CONFIG_NO_CONFIG_WRITE
718 #endif /* CONFIG_NO_CONFIG_WRITE */
720 "device_name", "manufacturer", "model_name", "model_number",
721 "serial_number", "config_methods", "wps_cred_processing",
722 #endif /* CONFIG_WPS */
724 "p2p_listen_reg_class", "p2p_listen_channel",
725 "p2p_oper_reg_class", "p2p_oper_channel", "p2p_go_intent",
726 "p2p_ssid_postfix", "persistent_reconnect", "p2p_intra_bss",
727 "p2p_group_idle", "p2p_passphrase_len", "p2p_add_cli_chan",
728 "p2p_optimize_listen_chan", "p2p_go_ht40", "p2p_go_vht",
729 "p2p_disabled", "p2p_go_ctwindow", "p2p_no_group_iface",
730 "p2p_ignore_shared_freq", "ip_addr_go", "ip_addr_mask",
731 "ip_addr_start", "ip_addr_end",
732 #endif /* CONFIG_P2P */
733 "bss_max_count", "bss_expiration_age",
734 "bss_expiration_scan_count", "filter_ssids", "filter_rssi",
735 "max_num_sta", "disassoc_low_ack",
738 #endif /* CONFIG_HS20 */
739 "interworking", "access_network_type", "pbc_in_m1", "autoscan",
740 "wps_nfc_dev_pw_id", "ext_password_backend",
741 "p2p_go_max_inactivity", "auto_interworking", "okc", "pmf",
742 "dtim_period", "beacon_int", "ignore_old_scan_res",
743 "scan_cur_freq", "sched_scan_interval",
744 "tdls_external_control", "osu_dir", "wowlan_triggers",
745 "p2p_search_delay", "mac_addr", "rand_addr_lifetime",
746 "preassoc_mac_addr", "key_mgmt_offload", "passive_scan",
747 "reassoc_same_bss_optim"
749 int i, num_fields = ARRAY_SIZE(fields);
752 char **res = os_calloc(num_fields + 1, sizeof(char *));
755 for (i = 0; i < num_fields; i++) {
756 res[i] = os_strdup(fields[i]);
767 static int wpa_cli_cmd_logoff(struct wpa_ctrl *ctrl, int argc, char *argv[])
769 return wpa_ctrl_command(ctrl, "LOGOFF");
773 static int wpa_cli_cmd_logon(struct wpa_ctrl *ctrl, int argc, char *argv[])
775 return wpa_ctrl_command(ctrl, "LOGON");
779 static int wpa_cli_cmd_reassociate(struct wpa_ctrl *ctrl, int argc,
782 return wpa_ctrl_command(ctrl, "REASSOCIATE");
786 static int wpa_cli_cmd_reattach(struct wpa_ctrl *ctrl, int argc, char *argv[])
788 return wpa_ctrl_command(ctrl, "REATTACH");
792 static int wpa_cli_cmd_preauthenticate(struct wpa_ctrl *ctrl, int argc,
795 return wpa_cli_cmd(ctrl, "PREAUTH", 1, argc, argv);
799 static int wpa_cli_cmd_ap_scan(struct wpa_ctrl *ctrl, int argc, char *argv[])
801 return wpa_cli_cmd(ctrl, "AP_SCAN", 1, argc, argv);
805 static int wpa_cli_cmd_scan_interval(struct wpa_ctrl *ctrl, int argc,
808 return wpa_cli_cmd(ctrl, "SCAN_INTERVAL", 1, argc, argv);
812 static int wpa_cli_cmd_bss_expire_age(struct wpa_ctrl *ctrl, int argc,
815 return wpa_cli_cmd(ctrl, "BSS_EXPIRE_AGE", 1, argc, argv);
819 static int wpa_cli_cmd_bss_expire_count(struct wpa_ctrl *ctrl, int argc,
822 return wpa_cli_cmd(ctrl, "BSS_EXPIRE_COUNT", 1, argc, argv);
826 static int wpa_cli_cmd_bss_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
832 res = os_snprintf(cmd, sizeof(cmd), "BSS_FLUSH 0");
834 res = os_snprintf(cmd, sizeof(cmd), "BSS_FLUSH %s", argv[0]);
835 if (os_snprintf_error(sizeof(cmd), res)) {
836 printf("Too long BSS_FLUSH command.\n");
839 return wpa_ctrl_command(ctrl, cmd);
843 static int wpa_cli_cmd_stkstart(struct wpa_ctrl *ctrl, int argc,
846 return wpa_cli_cmd(ctrl, "STKSTART", 1, argc, argv);
850 static int wpa_cli_cmd_ft_ds(struct wpa_ctrl *ctrl, int argc, char *argv[])
852 return wpa_cli_cmd(ctrl, "FT_DS", 1, argc, argv);
856 static int wpa_cli_cmd_wps_pbc(struct wpa_ctrl *ctrl, int argc, char *argv[])
858 return wpa_cli_cmd(ctrl, "WPS_PBC", 0, argc, argv);
862 static int wpa_cli_cmd_wps_pin(struct wpa_ctrl *ctrl, int argc, char *argv[])
865 printf("Invalid WPS_PIN command: need one or two arguments:\n"
866 "- BSSID: use 'any' to select any\n"
867 "- PIN: optional, used only with devices that have no "
872 return wpa_cli_cmd(ctrl, "WPS_PIN", 1, argc, argv);
876 static int wpa_cli_cmd_wps_check_pin(struct wpa_ctrl *ctrl, int argc,
879 return wpa_cli_cmd(ctrl, "WPS_CHECK_PIN", 1, argc, argv);
883 static int wpa_cli_cmd_wps_cancel(struct wpa_ctrl *ctrl, int argc,
886 return wpa_ctrl_command(ctrl, "WPS_CANCEL");
890 #ifdef CONFIG_WPS_NFC
892 static int wpa_cli_cmd_wps_nfc(struct wpa_ctrl *ctrl, int argc, char *argv[])
894 return wpa_cli_cmd(ctrl, "WPS_NFC", 0, argc, argv);
898 static int wpa_cli_cmd_wps_nfc_config_token(struct wpa_ctrl *ctrl, int argc,
901 return wpa_cli_cmd(ctrl, "WPS_NFC_CONFIG_TOKEN", 1, argc, argv);
905 static int wpa_cli_cmd_wps_nfc_token(struct wpa_ctrl *ctrl, int argc,
908 return wpa_cli_cmd(ctrl, "WPS_NFC_TOKEN", 1, argc, argv);
912 static int wpa_cli_cmd_wps_nfc_tag_read(struct wpa_ctrl *ctrl, int argc,
920 printf("Invalid 'wps_nfc_tag_read' command - one argument "
925 buflen = 18 + os_strlen(argv[0]);
926 buf = os_malloc(buflen);
929 os_snprintf(buf, buflen, "WPS_NFC_TAG_READ %s", argv[0]);
931 ret = wpa_ctrl_command(ctrl, buf);
938 static int wpa_cli_cmd_nfc_get_handover_req(struct wpa_ctrl *ctrl, int argc,
941 return wpa_cli_cmd(ctrl, "NFC_GET_HANDOVER_REQ", 2, argc, argv);
945 static int wpa_cli_cmd_nfc_get_handover_sel(struct wpa_ctrl *ctrl, int argc,
948 return wpa_cli_cmd(ctrl, "NFC_GET_HANDOVER_SEL", 2, argc, argv);
952 static int wpa_cli_cmd_nfc_report_handover(struct wpa_ctrl *ctrl, int argc,
955 return wpa_cli_cmd(ctrl, "NFC_REPORT_HANDOVER", 4, argc, argv);
958 #endif /* CONFIG_WPS_NFC */
961 static int wpa_cli_cmd_wps_reg(struct wpa_ctrl *ctrl, int argc, char *argv[])
967 res = os_snprintf(cmd, sizeof(cmd), "WPS_REG %s %s",
969 else if (argc == 5 || argc == 6) {
970 char ssid_hex[2 * SSID_MAX_LEN + 1];
971 char key_hex[2 * 64 + 1];
975 for (i = 0; i < SSID_MAX_LEN; i++) {
976 if (argv[2][i] == '\0')
978 os_snprintf(&ssid_hex[i * 2], 3, "%02x", argv[2][i]);
983 for (i = 0; i < 64; i++) {
984 if (argv[5][i] == '\0')
986 os_snprintf(&key_hex[i * 2], 3, "%02x",
991 res = os_snprintf(cmd, sizeof(cmd),
992 "WPS_REG %s %s %s %s %s %s",
993 argv[0], argv[1], ssid_hex, argv[3], argv[4],
996 printf("Invalid WPS_REG command: need two arguments:\n"
997 "- BSSID of the target AP\n"
999 printf("Alternatively, six arguments can be used to "
1000 "reconfigure the AP:\n"
1001 "- BSSID of the target AP\n"
1004 "- new auth (OPEN, WPAPSK, WPA2PSK)\n"
1005 "- new encr (NONE, WEP, TKIP, CCMP)\n"
1010 if (os_snprintf_error(sizeof(cmd), res)) {
1011 printf("Too long WPS_REG command.\n");
1014 return wpa_ctrl_command(ctrl, cmd);
1018 static int wpa_cli_cmd_wps_ap_pin(struct wpa_ctrl *ctrl, int argc,
1021 return wpa_cli_cmd(ctrl, "WPS_AP_PIN", 1, argc, argv);
1025 static int wpa_cli_cmd_wps_er_start(struct wpa_ctrl *ctrl, int argc,
1028 return wpa_cli_cmd(ctrl, "WPS_ER_START", 0, argc, argv);
1032 static int wpa_cli_cmd_wps_er_stop(struct wpa_ctrl *ctrl, int argc,
1035 return wpa_ctrl_command(ctrl, "WPS_ER_STOP");
1040 static int wpa_cli_cmd_wps_er_pin(struct wpa_ctrl *ctrl, int argc,
1044 printf("Invalid WPS_ER_PIN command: need at least two "
1046 "- UUID: use 'any' to select any\n"
1047 "- PIN: Enrollee PIN\n"
1048 "optional: - Enrollee MAC address\n");
1052 return wpa_cli_cmd(ctrl, "WPS_ER_PIN", 2, argc, argv);
1056 static int wpa_cli_cmd_wps_er_pbc(struct wpa_ctrl *ctrl, int argc,
1059 return wpa_cli_cmd(ctrl, "WPS_ER_PBC", 1, argc, argv);
1063 static int wpa_cli_cmd_wps_er_learn(struct wpa_ctrl *ctrl, int argc,
1067 printf("Invalid WPS_ER_LEARN command: need two arguments:\n"
1068 "- UUID: specify which AP to use\n"
1073 return wpa_cli_cmd(ctrl, "WPS_ER_LEARN", 2, argc, argv);
1077 static int wpa_cli_cmd_wps_er_set_config(struct wpa_ctrl *ctrl, int argc,
1081 printf("Invalid WPS_ER_SET_CONFIG command: need two "
1083 "- UUID: specify which AP to use\n"
1084 "- Network configuration id\n");
1088 return wpa_cli_cmd(ctrl, "WPS_ER_SET_CONFIG", 2, argc, argv);
1092 static int wpa_cli_cmd_wps_er_config(struct wpa_ctrl *ctrl, int argc,
1098 if (argc == 5 || argc == 6) {
1099 char ssid_hex[2 * SSID_MAX_LEN + 1];
1100 char key_hex[2 * 64 + 1];
1104 for (i = 0; i < SSID_MAX_LEN; i++) {
1105 if (argv[2][i] == '\0')
1107 os_snprintf(&ssid_hex[i * 2], 3, "%02x", argv[2][i]);
1112 for (i = 0; i < 64; i++) {
1113 if (argv[5][i] == '\0')
1115 os_snprintf(&key_hex[i * 2], 3, "%02x",
1120 res = os_snprintf(cmd, sizeof(cmd),
1121 "WPS_ER_CONFIG %s %s %s %s %s %s",
1122 argv[0], argv[1], ssid_hex, argv[3], argv[4],
1125 printf("Invalid WPS_ER_CONFIG command: need six arguments:\n"
1129 "- new auth (OPEN, WPAPSK, WPA2PSK)\n"
1130 "- new encr (NONE, WEP, TKIP, CCMP)\n"
1135 if (os_snprintf_error(sizeof(cmd), res)) {
1136 printf("Too long WPS_ER_CONFIG command.\n");
1139 return wpa_ctrl_command(ctrl, cmd);
1143 #ifdef CONFIG_WPS_NFC
1144 static int wpa_cli_cmd_wps_er_nfc_config_token(struct wpa_ctrl *ctrl, int argc,
1148 printf("Invalid WPS_ER_NFC_CONFIG_TOKEN command: need two "
1150 "- WPS/NDEF: token format\n"
1151 "- UUID: specify which AP to use\n");
1155 return wpa_cli_cmd(ctrl, "WPS_ER_NFC_CONFIG_TOKEN", 2, argc, argv);
1157 #endif /* CONFIG_WPS_NFC */
1160 static int wpa_cli_cmd_ibss_rsn(struct wpa_ctrl *ctrl, int argc, char *argv[])
1162 return wpa_cli_cmd(ctrl, "IBSS_RSN", 1, argc, argv);
1166 static int wpa_cli_cmd_level(struct wpa_ctrl *ctrl, int argc, char *argv[])
1168 return wpa_cli_cmd(ctrl, "LEVEL", 1, argc, argv);
1172 static int wpa_cli_cmd_identity(struct wpa_ctrl *ctrl, int argc, char *argv[])
1174 char cmd[256], *pos, *end;
1178 printf("Invalid IDENTITY command: needs two arguments "
1179 "(network id and identity)\n");
1183 end = cmd + sizeof(cmd);
1185 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "IDENTITY-%s:%s",
1187 if (os_snprintf_error(end - pos, ret)) {
1188 printf("Too long IDENTITY command.\n");
1192 for (i = 2; i < argc; i++) {
1193 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1194 if (os_snprintf_error(end - pos, ret)) {
1195 printf("Too long IDENTITY command.\n");
1201 return wpa_ctrl_command(ctrl, cmd);
1205 static int wpa_cli_cmd_password(struct wpa_ctrl *ctrl, int argc, char *argv[])
1207 char cmd[256], *pos, *end;
1211 printf("Invalid PASSWORD command: needs two arguments "
1212 "(network id and password)\n");
1216 end = cmd + sizeof(cmd);
1218 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "PASSWORD-%s:%s",
1220 if (os_snprintf_error(end - pos, ret)) {
1221 printf("Too long PASSWORD command.\n");
1225 for (i = 2; i < argc; i++) {
1226 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1227 if (os_snprintf_error(end - pos, ret)) {
1228 printf("Too long PASSWORD command.\n");
1234 return wpa_ctrl_command(ctrl, cmd);
1238 static int wpa_cli_cmd_new_password(struct wpa_ctrl *ctrl, int argc,
1241 char cmd[256], *pos, *end;
1245 printf("Invalid NEW_PASSWORD command: needs two arguments "
1246 "(network id and password)\n");
1250 end = cmd + sizeof(cmd);
1252 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "NEW_PASSWORD-%s:%s",
1254 if (os_snprintf_error(end - pos, ret)) {
1255 printf("Too long NEW_PASSWORD command.\n");
1259 for (i = 2; i < argc; i++) {
1260 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1261 if (os_snprintf_error(end - pos, ret)) {
1262 printf("Too long NEW_PASSWORD command.\n");
1268 return wpa_ctrl_command(ctrl, cmd);
1272 static int wpa_cli_cmd_pin(struct wpa_ctrl *ctrl, int argc, char *argv[])
1274 char cmd[256], *pos, *end;
1278 printf("Invalid PIN command: needs two arguments "
1279 "(network id and pin)\n");
1283 end = cmd + sizeof(cmd);
1285 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "PIN-%s:%s",
1287 if (os_snprintf_error(end - pos, ret)) {
1288 printf("Too long PIN command.\n");
1292 for (i = 2; i < argc; i++) {
1293 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1294 if (os_snprintf_error(end - pos, ret)) {
1295 printf("Too long PIN command.\n");
1300 return wpa_ctrl_command(ctrl, cmd);
1304 static int wpa_cli_cmd_otp(struct wpa_ctrl *ctrl, int argc, char *argv[])
1306 char cmd[256], *pos, *end;
1310 printf("Invalid OTP command: needs two arguments (network "
1311 "id and password)\n");
1315 end = cmd + sizeof(cmd);
1317 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "OTP-%s:%s",
1319 if (os_snprintf_error(end - pos, ret)) {
1320 printf("Too long OTP command.\n");
1324 for (i = 2; i < argc; i++) {
1325 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1326 if (os_snprintf_error(end - pos, ret)) {
1327 printf("Too long OTP command.\n");
1333 return wpa_ctrl_command(ctrl, cmd);
1337 static int wpa_cli_cmd_sim(struct wpa_ctrl *ctrl, int argc, char *argv[])
1339 char cmd[256], *pos, *end;
1343 printf("Invalid SIM command: needs two arguments "
1344 "(network id and SIM operation response)\n");
1348 end = cmd + sizeof(cmd);
1350 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "SIM-%s:%s",
1352 if (os_snprintf_error(end - pos, ret)) {
1353 printf("Too long SIM command.\n");
1357 for (i = 2; i < argc; i++) {
1358 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1359 if (os_snprintf_error(end - pos, ret)) {
1360 printf("Too long SIM command.\n");
1365 return wpa_ctrl_command(ctrl, cmd);
1369 static int wpa_cli_cmd_passphrase(struct wpa_ctrl *ctrl, int argc,
1372 char cmd[256], *pos, *end;
1376 printf("Invalid PASSPHRASE command: needs two arguments "
1377 "(network id and passphrase)\n");
1381 end = cmd + sizeof(cmd);
1383 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "PASSPHRASE-%s:%s",
1385 if (os_snprintf_error(end - pos, ret)) {
1386 printf("Too long PASSPHRASE command.\n");
1390 for (i = 2; i < argc; i++) {
1391 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1392 if (os_snprintf_error(end - pos, ret)) {
1393 printf("Too long PASSPHRASE command.\n");
1399 return wpa_ctrl_command(ctrl, cmd);
1403 static int wpa_cli_cmd_bssid(struct wpa_ctrl *ctrl, int argc, char *argv[])
1406 printf("Invalid BSSID command: needs two arguments (network "
1411 return wpa_cli_cmd(ctrl, "BSSID", 2, argc, argv);
1415 static int wpa_cli_cmd_blacklist(struct wpa_ctrl *ctrl, int argc, char *argv[])
1417 return wpa_cli_cmd(ctrl, "BLACKLIST", 0, argc, argv);
1421 static int wpa_cli_cmd_log_level(struct wpa_ctrl *ctrl, int argc, char *argv[])
1423 return wpa_cli_cmd(ctrl, "LOG_LEVEL", 0, argc, argv);
1427 static int wpa_cli_cmd_list_networks(struct wpa_ctrl *ctrl, int argc,
1430 return wpa_ctrl_command(ctrl, "LIST_NETWORKS");
1434 static int wpa_cli_cmd_select_network(struct wpa_ctrl *ctrl, int argc,
1437 return wpa_cli_cmd(ctrl, "SELECT_NETWORK", 1, argc, argv);
1441 static int wpa_cli_cmd_enable_network(struct wpa_ctrl *ctrl, int argc,
1444 return wpa_cli_cmd(ctrl, "ENABLE_NETWORK", 1, argc, argv);
1448 static int wpa_cli_cmd_disable_network(struct wpa_ctrl *ctrl, int argc,
1451 return wpa_cli_cmd(ctrl, "DISABLE_NETWORK", 1, argc, argv);
1455 static int wpa_cli_cmd_add_network(struct wpa_ctrl *ctrl, int argc,
1458 int res = wpa_ctrl_command(ctrl, "ADD_NETWORK");
1460 update_networks(ctrl);
1465 static int wpa_cli_cmd_remove_network(struct wpa_ctrl *ctrl, int argc,
1468 int res = wpa_cli_cmd(ctrl, "REMOVE_NETWORK", 1, argc, argv);
1470 update_networks(ctrl);
1475 static void wpa_cli_show_network_variables(void)
1477 printf("set_network variables:\n"
1478 " ssid (network name, SSID)\n"
1479 " psk (WPA passphrase or pre-shared key)\n"
1480 " key_mgmt (key management protocol)\n"
1481 " identity (EAP identity)\n"
1482 " password (EAP password)\n"
1485 "Note: Values are entered in the same format as the "
1486 "configuration file is using,\n"
1487 "i.e., strings values need to be inside double quotation "
1489 "For example: set_network 1 ssid \"network name\"\n"
1491 "Please see wpa_supplicant.conf documentation for full list "
1492 "of\navailable variables.\n");
1496 static int wpa_cli_cmd_set_network(struct wpa_ctrl *ctrl, int argc,
1500 wpa_cli_show_network_variables();
1505 printf("Invalid SET_NETWORK command: needs three arguments\n"
1506 "(network id, variable name, and value)\n");
1510 return wpa_cli_cmd(ctrl, "SET_NETWORK", 3, argc, argv);
1514 static int wpa_cli_cmd_get_network(struct wpa_ctrl *ctrl, int argc,
1518 wpa_cli_show_network_variables();
1523 printf("Invalid GET_NETWORK command: needs two arguments\n"
1524 "(network id and variable name)\n");
1528 return wpa_cli_cmd(ctrl, "GET_NETWORK", 2, argc, argv);
1532 static const char *network_fields[] = {
1533 "ssid", "scan_ssid", "bssid", "bssid_blacklist",
1534 "bssid_whitelist", "psk", "proto", "key_mgmt",
1535 "bg_scan_period", "pairwise", "group", "auth_alg", "scan_freq",
1537 #ifdef IEEE8021X_EAPOL
1538 "eap", "identity", "anonymous_identity", "password", "ca_cert",
1539 "ca_path", "client_cert", "private_key", "private_key_passwd",
1540 "dh_file", "subject_match", "altsubject_match",
1541 "domain_suffix_match", "domain_match", "ca_cert2", "ca_path2",
1542 "client_cert2", "private_key2", "private_key2_passwd",
1543 "dh_file2", "subject_match2", "altsubject_match2",
1544 "domain_suffix_match2", "domain_match2", "phase1", "phase2",
1545 "pcsc", "pin", "engine_id", "key_id", "cert_id", "ca_cert_id",
1546 "pin2", "engine2_id", "key2_id", "cert2_id", "ca_cert2_id",
1547 "engine", "engine2", "eapol_flags", "sim_num",
1548 "openssl_ciphers", "erp",
1549 #endif /* IEEE8021X_EAPOL */
1550 "wep_key0", "wep_key1", "wep_key2", "wep_key3",
1551 "wep_tx_keyidx", "priority",
1552 #ifdef IEEE8021X_EAPOL
1553 "eap_workaround", "pac_file", "fragment_size", "ocsp",
1554 #endif /* IEEE8021X_EAPOL */
1556 "mode", "no_auto_peer",
1557 #else /* CONFIG_MESH */
1559 #endif /* CONFIG_MESH */
1560 "proactive_key_caching", "disabled", "id_str",
1561 #ifdef CONFIG_IEEE80211W
1563 #endif /* CONFIG_IEEE80211W */
1564 "peerkey", "mixed_cell", "frequency", "fixed_freq",
1566 "mesh_basic_rates", "dot11MeshMaxRetries",
1567 "dot11MeshRetryTimeout", "dot11MeshConfirmTimeout",
1568 "dot11MeshHoldingTimeout",
1569 #endif /* CONFIG_MESH */
1570 "wpa_ptk_rekey", "bgscan", "ignore_broadcast_ssid",
1572 "go_p2p_dev_addr", "p2p_client_list", "psk_list",
1573 #endif /* CONFIG_P2P */
1574 #ifdef CONFIG_HT_OVERRIDES
1575 "disable_ht", "disable_ht40", "disable_sgi", "disable_ldpc",
1576 "ht40_intolerant", "disable_max_amsdu", "ampdu_factor",
1577 "ampdu_density", "ht_mcs",
1578 #endif /* CONFIG_HT_OVERRIDES */
1579 #ifdef CONFIG_VHT_OVERRIDES
1580 "disable_vht", "vht_capa", "vht_capa_mask", "vht_rx_mcs_nss_1",
1581 "vht_rx_mcs_nss_2", "vht_rx_mcs_nss_3", "vht_rx_mcs_nss_4",
1582 "vht_rx_mcs_nss_5", "vht_rx_mcs_nss_6", "vht_rx_mcs_nss_7",
1583 "vht_rx_mcs_nss_8", "vht_tx_mcs_nss_1", "vht_tx_mcs_nss_2",
1584 "vht_tx_mcs_nss_3", "vht_tx_mcs_nss_4", "vht_tx_mcs_nss_5",
1585 "vht_tx_mcs_nss_6", "vht_tx_mcs_nss_7", "vht_tx_mcs_nss_8",
1586 #endif /* CONFIG_VHT_OVERRIDES */
1587 "ap_max_inactivity", "dtim_period", "beacon_int",
1588 #ifdef CONFIG_MACSEC
1590 #endif /* CONFIG_MACSEC */
1592 "update_identifier",
1593 #endif /* CONFIG_HS20 */
1598 static char ** wpa_cli_complete_network(const char *str, int pos)
1600 int arg = get_cmd_arg_num(str, pos);
1601 int i, num_fields = ARRAY_SIZE(network_fields);
1606 res = cli_txt_list_array(&networks);
1609 res = os_calloc(num_fields + 1, sizeof(char *));
1612 for (i = 0; i < num_fields; i++) {
1613 res[i] = os_strdup(network_fields[i]);
1622 static char ** wpa_cli_complete_network_id(const char *str, int pos)
1624 int arg = get_cmd_arg_num(str, pos);
1626 return cli_txt_list_array(&networks);
1631 static int wpa_cli_cmd_dup_network(struct wpa_ctrl *ctrl, int argc,
1635 wpa_cli_show_network_variables();
1640 printf("Invalid DUP_NETWORK command: needs three arguments\n"
1641 "(src netid, dest netid, and variable name)\n");
1645 return wpa_cli_cmd(ctrl, "DUP_NETWORK", 3, argc, argv);
1649 static char ** wpa_cli_complete_dup_network(const char *str, int pos)
1651 int arg = get_cmd_arg_num(str, pos);
1652 int i, num_fields = ARRAY_SIZE(network_fields);
1658 res = cli_txt_list_array(&networks);
1661 res = os_calloc(num_fields + 1, sizeof(char *));
1664 for (i = 0; i < num_fields; i++) {
1665 res[i] = os_strdup(network_fields[i]);
1674 static int wpa_cli_cmd_list_creds(struct wpa_ctrl *ctrl, int argc,
1677 return wpa_ctrl_command(ctrl, "LIST_CREDS");
1681 static int wpa_cli_cmd_add_cred(struct wpa_ctrl *ctrl, int argc, char *argv[])
1683 return wpa_ctrl_command(ctrl, "ADD_CRED");
1687 static int wpa_cli_cmd_remove_cred(struct wpa_ctrl *ctrl, int argc,
1690 return wpa_cli_cmd(ctrl, "REMOVE_CRED", 1, argc, argv);
1694 static int wpa_cli_cmd_set_cred(struct wpa_ctrl *ctrl, int argc, char *argv[])
1697 printf("Invalid SET_CRED command: needs three arguments\n"
1698 "(cred id, variable name, and value)\n");
1702 return wpa_cli_cmd(ctrl, "SET_CRED", 3, argc, argv);
1706 static int wpa_cli_cmd_get_cred(struct wpa_ctrl *ctrl, int argc, char *argv[])
1709 printf("Invalid GET_CRED command: needs two arguments\n"
1710 "(cred id, variable name)\n");
1714 return wpa_cli_cmd(ctrl, "GET_CRED", 2, argc, argv);
1718 static int wpa_cli_cmd_disconnect(struct wpa_ctrl *ctrl, int argc,
1721 return wpa_ctrl_command(ctrl, "DISCONNECT");
1725 static int wpa_cli_cmd_reconnect(struct wpa_ctrl *ctrl, int argc,
1728 return wpa_ctrl_command(ctrl, "RECONNECT");
1732 static int wpa_cli_cmd_save_config(struct wpa_ctrl *ctrl, int argc,
1735 return wpa_ctrl_command(ctrl, "SAVE_CONFIG");
1739 static int wpa_cli_cmd_scan(struct wpa_ctrl *ctrl, int argc, char *argv[])
1741 return wpa_cli_cmd(ctrl, "SCAN", 0, argc, argv);
1745 static int wpa_cli_cmd_scan_results(struct wpa_ctrl *ctrl, int argc,
1748 return wpa_ctrl_command(ctrl, "SCAN_RESULTS");
1752 static int wpa_cli_cmd_bss(struct wpa_ctrl *ctrl, int argc, char *argv[])
1754 return wpa_cli_cmd(ctrl, "BSS", 1, argc, argv);
1758 static char ** wpa_cli_complete_bss(const char *str, int pos)
1760 int arg = get_cmd_arg_num(str, pos);
1765 res = cli_txt_list_array(&bsses);
1773 static int wpa_cli_cmd_get_capability(struct wpa_ctrl *ctrl, int argc,
1776 if (argc < 1 || argc > 2) {
1777 printf("Invalid GET_CAPABILITY command: need either one or "
1782 if ((argc == 2) && os_strcmp(argv[1], "strict") != 0) {
1783 printf("Invalid GET_CAPABILITY command: second argument, "
1784 "if any, must be 'strict'\n");
1788 return wpa_cli_cmd(ctrl, "GET_CAPABILITY", 1, argc, argv);
1792 static int wpa_cli_list_interfaces(struct wpa_ctrl *ctrl)
1794 printf("Available interfaces:\n");
1795 return wpa_ctrl_command(ctrl, "INTERFACES");
1799 static int wpa_cli_cmd_interface(struct wpa_ctrl *ctrl, int argc, char *argv[])
1802 wpa_cli_list_interfaces(ctrl);
1806 wpa_cli_close_connection();
1807 os_free(ctrl_ifname);
1808 ctrl_ifname = os_strdup(argv[0]);
1810 printf("Failed to allocate memory\n");
1814 if (wpa_cli_open_connection(ctrl_ifname, 1) == 0) {
1815 printf("Connected to interface '%s.\n", ctrl_ifname);
1817 printf("Could not connect to interface '%s' - re-trying\n",
1824 static int wpa_cli_cmd_reconfigure(struct wpa_ctrl *ctrl, int argc,
1827 return wpa_ctrl_command(ctrl, "RECONFIGURE");
1831 static int wpa_cli_cmd_terminate(struct wpa_ctrl *ctrl, int argc,
1834 return wpa_ctrl_command(ctrl, "TERMINATE");
1838 static int wpa_cli_cmd_interface_add(struct wpa_ctrl *ctrl, int argc,
1845 printf("Invalid INTERFACE_ADD command: needs at least one "
1846 "argument (interface name)\n"
1847 "All arguments: ifname confname driver ctrl_interface "
1848 "driver_param bridge_name [create]\n");
1853 * INTERFACE_ADD <ifname>TAB<confname>TAB<driver>TAB<ctrl_interface>TAB
1854 * <driver_param>TAB<bridge_name>[TAB<create>]
1856 res = os_snprintf(cmd, sizeof(cmd),
1857 "INTERFACE_ADD %s\t%s\t%s\t%s\t%s\t%s\t%s",
1859 argc > 1 ? argv[1] : "", argc > 2 ? argv[2] : "",
1860 argc > 3 ? argv[3] : "", argc > 4 ? argv[4] : "",
1861 argc > 5 ? argv[5] : "", argc > 6 ? argv[6] : "");
1862 if (os_snprintf_error(sizeof(cmd), res))
1864 cmd[sizeof(cmd) - 1] = '\0';
1865 return wpa_ctrl_command(ctrl, cmd);
1869 static int wpa_cli_cmd_interface_remove(struct wpa_ctrl *ctrl, int argc,
1872 return wpa_cli_cmd(ctrl, "INTERFACE_REMOVE", 1, argc, argv);
1876 static int wpa_cli_cmd_interface_list(struct wpa_ctrl *ctrl, int argc,
1879 return wpa_ctrl_command(ctrl, "INTERFACE_LIST");
1884 static int wpa_cli_cmd_sta(struct wpa_ctrl *ctrl, int argc, char *argv[])
1886 return wpa_cli_cmd(ctrl, "STA", 1, argc, argv);
1890 static int wpa_ctrl_command_sta(struct wpa_ctrl *ctrl, char *cmd,
1891 char *addr, size_t addr_len)
1893 char buf[4096], *pos;
1897 if (ctrl_conn == NULL) {
1898 printf("Not connected to hostapd - command dropped.\n");
1901 len = sizeof(buf) - 1;
1902 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len,
1905 printf("'%s' command timed out.\n", cmd);
1907 } else if (ret < 0) {
1908 printf("'%s' command failed.\n", cmd);
1913 if (os_memcmp(buf, "FAIL", 4) == 0)
1918 while (*pos != '\0' && *pos != '\n')
1921 os_strlcpy(addr, buf, addr_len);
1926 static int wpa_cli_cmd_all_sta(struct wpa_ctrl *ctrl, int argc, char *argv[])
1928 char addr[32], cmd[64];
1930 if (wpa_ctrl_command_sta(ctrl, "STA-FIRST", addr, sizeof(addr)))
1933 os_snprintf(cmd, sizeof(cmd), "STA-NEXT %s", addr);
1934 } while (wpa_ctrl_command_sta(ctrl, cmd, addr, sizeof(addr)) == 0);
1940 static int wpa_cli_cmd_deauthenticate(struct wpa_ctrl *ctrl, int argc,
1943 return wpa_cli_cmd(ctrl, "DEAUTHENTICATE", 1, argc, argv);
1947 static int wpa_cli_cmd_disassociate(struct wpa_ctrl *ctrl, int argc,
1950 return wpa_cli_cmd(ctrl, "DISASSOCIATE", 1, argc, argv);
1953 static int wpa_cli_cmd_chanswitch(struct wpa_ctrl *ctrl, int argc,
1956 return wpa_cli_cmd(ctrl, "CHAN_SWITCH", 2, argc, argv);
1959 #endif /* CONFIG_AP */
1962 static int wpa_cli_cmd_suspend(struct wpa_ctrl *ctrl, int argc, char *argv[])
1964 return wpa_ctrl_command(ctrl, "SUSPEND");
1968 static int wpa_cli_cmd_resume(struct wpa_ctrl *ctrl, int argc, char *argv[])
1970 return wpa_ctrl_command(ctrl, "RESUME");
1974 #ifdef CONFIG_TESTING_OPTIONS
1975 static int wpa_cli_cmd_drop_sa(struct wpa_ctrl *ctrl, int argc, char *argv[])
1977 return wpa_ctrl_command(ctrl, "DROP_SA");
1979 #endif /* CONFIG_TESTING_OPTIONS */
1982 static int wpa_cli_cmd_roam(struct wpa_ctrl *ctrl, int argc, char *argv[])
1984 return wpa_cli_cmd(ctrl, "ROAM", 1, argc, argv);
1990 static int wpa_cli_cmd_mesh_interface_add(struct wpa_ctrl *ctrl, int argc,
1993 return wpa_cli_cmd(ctrl, "MESH_INTERFACE_ADD", 0, argc, argv);
1997 static int wpa_cli_cmd_mesh_group_add(struct wpa_ctrl *ctrl, int argc,
2000 return wpa_cli_cmd(ctrl, "MESH_GROUP_ADD", 1, argc, argv);
2004 static int wpa_cli_cmd_mesh_group_remove(struct wpa_ctrl *ctrl, int argc,
2007 return wpa_cli_cmd(ctrl, "MESH_GROUP_REMOVE", 1, argc, argv);
2010 #endif /* CONFIG_MESH */
2015 static int wpa_cli_cmd_p2p_find(struct wpa_ctrl *ctrl, int argc, char *argv[])
2017 return wpa_cli_cmd(ctrl, "P2P_FIND", 0, argc, argv);
2021 static char ** wpa_cli_complete_p2p_find(const char *str, int pos)
2024 int arg = get_cmd_arg_num(str, pos);
2026 res = os_calloc(6, sizeof(char *));
2029 res[0] = os_strdup("type=social");
2030 if (res[0] == NULL) {
2034 res[1] = os_strdup("type=progressive");
2037 res[2] = os_strdup("delay=");
2040 res[3] = os_strdup("dev_id=");
2044 res[4] = os_strdup("[timeout]");
2050 static int wpa_cli_cmd_p2p_stop_find(struct wpa_ctrl *ctrl, int argc,
2053 return wpa_ctrl_command(ctrl, "P2P_STOP_FIND");
2057 static int wpa_cli_cmd_p2p_asp_provision(struct wpa_ctrl *ctrl, int argc,
2060 return wpa_cli_cmd(ctrl, "P2P_ASP_PROVISION", 3, argc, argv);
2064 static int wpa_cli_cmd_p2p_asp_provision_resp(struct wpa_ctrl *ctrl, int argc,
2067 return wpa_cli_cmd(ctrl, "P2P_ASP_PROVISION_RESP", 2, argc, argv);
2071 static int wpa_cli_cmd_p2p_connect(struct wpa_ctrl *ctrl, int argc,
2074 return wpa_cli_cmd(ctrl, "P2P_CONNECT", 2, argc, argv);
2078 static char ** wpa_cli_complete_p2p_connect(const char *str, int pos)
2080 int arg = get_cmd_arg_num(str, pos);
2085 res = cli_txt_list_array(&p2p_peers);
2093 static int wpa_cli_cmd_p2p_listen(struct wpa_ctrl *ctrl, int argc,
2096 return wpa_cli_cmd(ctrl, "P2P_LISTEN", 0, argc, argv);
2100 static int wpa_cli_cmd_p2p_group_remove(struct wpa_ctrl *ctrl, int argc,
2103 return wpa_cli_cmd(ctrl, "P2P_GROUP_REMOVE", 1, argc, argv);
2107 static char ** wpa_cli_complete_p2p_group_remove(const char *str, int pos)
2109 int arg = get_cmd_arg_num(str, pos);
2114 res = cli_txt_list_array(&p2p_groups);
2122 static int wpa_cli_cmd_p2p_group_add(struct wpa_ctrl *ctrl, int argc,
2125 return wpa_cli_cmd(ctrl, "P2P_GROUP_ADD", 0, argc, argv);
2129 static int wpa_cli_cmd_p2p_prov_disc(struct wpa_ctrl *ctrl, int argc,
2132 if (argc != 2 && argc != 3) {
2133 printf("Invalid P2P_PROV_DISC command: needs at least "
2134 "two arguments, address and config method\n"
2135 "(display, keypad, or pbc) and an optional join\n");
2139 return wpa_cli_cmd(ctrl, "P2P_PROV_DISC", 2, argc, argv);
2143 static int wpa_cli_cmd_p2p_get_passphrase(struct wpa_ctrl *ctrl, int argc,
2146 return wpa_ctrl_command(ctrl, "P2P_GET_PASSPHRASE");
2150 static int wpa_cli_cmd_p2p_serv_disc_req(struct wpa_ctrl *ctrl, int argc,
2156 printf("Invalid P2P_SERV_DISC_REQ command: needs two "
2157 "or more arguments (address and TLVs)\n");
2161 if (write_cmd(cmd, sizeof(cmd), "P2P_SERV_DISC_REQ", argc, argv) < 0)
2163 return wpa_ctrl_command(ctrl, cmd);
2167 static int wpa_cli_cmd_p2p_serv_disc_cancel_req(struct wpa_ctrl *ctrl,
2168 int argc, char *argv[])
2170 return wpa_cli_cmd(ctrl, "P2P_SERV_DISC_CANCEL_REQ", 1, argc, argv);
2174 static int wpa_cli_cmd_p2p_serv_disc_resp(struct wpa_ctrl *ctrl, int argc,
2181 printf("Invalid P2P_SERV_DISC_RESP command: needs four "
2182 "arguments (freq, address, dialog token, and TLVs)\n");
2186 res = os_snprintf(cmd, sizeof(cmd), "P2P_SERV_DISC_RESP %s %s %s %s",
2187 argv[0], argv[1], argv[2], argv[3]);
2188 if (os_snprintf_error(sizeof(cmd), res))
2190 cmd[sizeof(cmd) - 1] = '\0';
2191 return wpa_ctrl_command(ctrl, cmd);
2195 static int wpa_cli_cmd_p2p_service_update(struct wpa_ctrl *ctrl, int argc,
2198 return wpa_ctrl_command(ctrl, "P2P_SERVICE_UPDATE");
2202 static int wpa_cli_cmd_p2p_serv_disc_external(struct wpa_ctrl *ctrl,
2203 int argc, char *argv[])
2205 return wpa_cli_cmd(ctrl, "P2P_SERV_DISC_EXTERNAL", 1, argc, argv);
2209 static int wpa_cli_cmd_p2p_service_flush(struct wpa_ctrl *ctrl, int argc,
2212 return wpa_ctrl_command(ctrl, "P2P_SERVICE_FLUSH");
2216 static int wpa_cli_cmd_p2p_service_add(struct wpa_ctrl *ctrl, int argc,
2220 printf("Invalid P2P_SERVICE_ADD command: needs 3-6 arguments\n");
2224 return wpa_cli_cmd(ctrl, "P2P_SERVICE_ADD", 3, argc, argv);
2228 static int wpa_cli_cmd_p2p_service_rep(struct wpa_ctrl *ctrl, int argc,
2231 if (argc < 5 || argc > 6) {
2232 printf("Invalid P2P_SERVICE_REP command: needs 5-6 "
2237 return wpa_cli_cmd(ctrl, "P2P_SERVICE_REP", 5, argc, argv);
2241 static int wpa_cli_cmd_p2p_service_del(struct wpa_ctrl *ctrl, int argc,
2247 if (argc != 2 && argc != 3) {
2248 printf("Invalid P2P_SERVICE_DEL command: needs two or three "
2254 res = os_snprintf(cmd, sizeof(cmd),
2255 "P2P_SERVICE_DEL %s %s %s",
2256 argv[0], argv[1], argv[2]);
2258 res = os_snprintf(cmd, sizeof(cmd),
2259 "P2P_SERVICE_DEL %s %s",
2261 if (os_snprintf_error(sizeof(cmd), res))
2263 cmd[sizeof(cmd) - 1] = '\0';
2264 return wpa_ctrl_command(ctrl, cmd);
2268 static int wpa_cli_cmd_p2p_reject(struct wpa_ctrl *ctrl,
2269 int argc, char *argv[])
2271 return wpa_cli_cmd(ctrl, "P2P_REJECT", 1, argc, argv);
2275 static int wpa_cli_cmd_p2p_invite(struct wpa_ctrl *ctrl,
2276 int argc, char *argv[])
2278 return wpa_cli_cmd(ctrl, "P2P_INVITE", 1, argc, argv);
2282 static int wpa_cli_cmd_p2p_peer(struct wpa_ctrl *ctrl, int argc, char *argv[])
2284 return wpa_cli_cmd(ctrl, "P2P_PEER", 1, argc, argv);
2288 static char ** wpa_cli_complete_p2p_peer(const char *str, int pos)
2290 int arg = get_cmd_arg_num(str, pos);
2295 res = cli_txt_list_array(&p2p_peers);
2303 static int wpa_ctrl_command_p2p_peer(struct wpa_ctrl *ctrl, char *cmd,
2304 char *addr, size_t addr_len,
2307 char buf[4096], *pos;
2311 if (ctrl_conn == NULL)
2313 len = sizeof(buf) - 1;
2314 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len,
2317 printf("'%s' command timed out.\n", cmd);
2319 } else if (ret < 0) {
2320 printf("'%s' command failed.\n", cmd);
2325 if (os_memcmp(buf, "FAIL", 4) == 0)
2329 while (*pos != '\0' && *pos != '\n')
2332 os_strlcpy(addr, buf, addr_len);
2333 if (!discovered || os_strstr(pos, "[PROBE_REQ_ONLY]") == NULL)
2334 printf("%s\n", addr);
2339 static int wpa_cli_cmd_p2p_peers(struct wpa_ctrl *ctrl, int argc, char *argv[])
2341 char addr[32], cmd[64];
2344 discovered = argc > 0 && os_strcmp(argv[0], "discovered") == 0;
2346 if (wpa_ctrl_command_p2p_peer(ctrl, "P2P_PEER FIRST",
2347 addr, sizeof(addr), discovered))
2350 os_snprintf(cmd, sizeof(cmd), "P2P_PEER NEXT-%s", addr);
2351 } while (wpa_ctrl_command_p2p_peer(ctrl, cmd, addr, sizeof(addr),
2358 static int wpa_cli_cmd_p2p_set(struct wpa_ctrl *ctrl, int argc, char *argv[])
2360 return wpa_cli_cmd(ctrl, "P2P_SET", 2, argc, argv);
2364 static char ** wpa_cli_complete_p2p_set(const char *str, int pos)
2366 int arg = get_cmd_arg_num(str, pos);
2367 const char *fields[] = {
2387 int i, num_fields = ARRAY_SIZE(fields);
2390 char **res = os_calloc(num_fields + 1, sizeof(char *));
2393 for (i = 0; i < num_fields; i++) {
2394 res[i] = os_strdup(fields[i]);
2401 if (arg == 2 && os_strncasecmp(str, "p2p_set peer_filter ", 20) == 0)
2402 return cli_txt_list_array(&p2p_peers);
2408 static int wpa_cli_cmd_p2p_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
2410 return wpa_ctrl_command(ctrl, "P2P_FLUSH");
2414 static int wpa_cli_cmd_p2p_cancel(struct wpa_ctrl *ctrl, int argc,
2417 return wpa_ctrl_command(ctrl, "P2P_CANCEL");
2421 static int wpa_cli_cmd_p2p_unauthorize(struct wpa_ctrl *ctrl, int argc,
2424 return wpa_cli_cmd(ctrl, "P2P_UNAUTHORIZE", 1, argc, argv);
2428 static int wpa_cli_cmd_p2p_presence_req(struct wpa_ctrl *ctrl, int argc,
2431 if (argc != 0 && argc != 2 && argc != 4) {
2432 printf("Invalid P2P_PRESENCE_REQ command: needs two arguments "
2433 "(preferred duration, interval; in microsecods).\n"
2434 "Optional second pair can be used to provide "
2435 "acceptable values.\n");
2439 return wpa_cli_cmd(ctrl, "P2P_PRESENCE_REQ", 0, argc, argv);
2443 static int wpa_cli_cmd_p2p_ext_listen(struct wpa_ctrl *ctrl, int argc,
2446 if (argc != 0 && argc != 2) {
2447 printf("Invalid P2P_EXT_LISTEN command: needs two arguments "
2448 "(availability period, availability interval; in "
2450 "Extended Listen Timing can be cancelled with this "
2451 "command when used without parameters.\n");
2455 return wpa_cli_cmd(ctrl, "P2P_EXT_LISTEN", 0, argc, argv);
2459 static int wpa_cli_cmd_p2p_remove_client(struct wpa_ctrl *ctrl, int argc,
2462 return wpa_cli_cmd(ctrl, "P2P_REMOVE_CLIENT", 1, argc, argv);
2465 #endif /* CONFIG_P2P */
2467 #ifdef CONFIG_WIFI_DISPLAY
2469 static int wpa_cli_cmd_wfd_subelem_set(struct wpa_ctrl *ctrl, int argc,
2475 if (argc != 1 && argc != 2) {
2476 printf("Invalid WFD_SUBELEM_SET command: needs one or two "
2477 "arguments (subelem, hexdump)\n");
2481 res = os_snprintf(cmd, sizeof(cmd), "WFD_SUBELEM_SET %s %s",
2482 argv[0], argc > 1 ? argv[1] : "");
2483 if (os_snprintf_error(sizeof(cmd), res))
2485 cmd[sizeof(cmd) - 1] = '\0';
2486 return wpa_ctrl_command(ctrl, cmd);
2490 static int wpa_cli_cmd_wfd_subelem_get(struct wpa_ctrl *ctrl, int argc,
2497 printf("Invalid WFD_SUBELEM_GET command: needs one "
2498 "argument (subelem)\n");
2502 res = os_snprintf(cmd, sizeof(cmd), "WFD_SUBELEM_GET %s",
2504 if (os_snprintf_error(sizeof(cmd), res))
2506 cmd[sizeof(cmd) - 1] = '\0';
2507 return wpa_ctrl_command(ctrl, cmd);
2509 #endif /* CONFIG_WIFI_DISPLAY */
2512 #ifdef CONFIG_INTERWORKING
2513 static int wpa_cli_cmd_fetch_anqp(struct wpa_ctrl *ctrl, int argc,
2516 return wpa_ctrl_command(ctrl, "FETCH_ANQP");
2520 static int wpa_cli_cmd_stop_fetch_anqp(struct wpa_ctrl *ctrl, int argc,
2523 return wpa_ctrl_command(ctrl, "STOP_FETCH_ANQP");
2527 static int wpa_cli_cmd_interworking_select(struct wpa_ctrl *ctrl, int argc,
2530 return wpa_cli_cmd(ctrl, "INTERWORKING_SELECT", 0, argc, argv);
2534 static int wpa_cli_cmd_interworking_connect(struct wpa_ctrl *ctrl, int argc,
2537 return wpa_cli_cmd(ctrl, "INTERWORKING_CONNECT", 1, argc, argv);
2541 static int wpa_cli_cmd_interworking_add_network(struct wpa_ctrl *ctrl, int argc,
2544 return wpa_cli_cmd(ctrl, "INTERWORKING_ADD_NETWORK", 1, argc, argv);
2548 static int wpa_cli_cmd_anqp_get(struct wpa_ctrl *ctrl, int argc, char *argv[])
2550 return wpa_cli_cmd(ctrl, "ANQP_GET", 2, argc, argv);
2554 static int wpa_cli_cmd_gas_request(struct wpa_ctrl *ctrl, int argc,
2557 return wpa_cli_cmd(ctrl, "GAS_REQUEST", 2, argc, argv);
2561 static int wpa_cli_cmd_gas_response_get(struct wpa_ctrl *ctrl, int argc,
2564 return wpa_cli_cmd(ctrl, "GAS_RESPONSE_GET", 2, argc, argv);
2566 #endif /* CONFIG_INTERWORKING */
2571 static int wpa_cli_cmd_hs20_anqp_get(struct wpa_ctrl *ctrl, int argc,
2574 return wpa_cli_cmd(ctrl, "HS20_ANQP_GET", 2, argc, argv);
2578 static int wpa_cli_cmd_get_nai_home_realm_list(struct wpa_ctrl *ctrl, int argc,
2584 printf("Command needs one or two arguments (dst mac addr and "
2585 "optional home realm)\n");
2589 if (write_cmd(cmd, sizeof(cmd), "HS20_GET_NAI_HOME_REALM_LIST",
2593 return wpa_ctrl_command(ctrl, cmd);
2597 static int wpa_cli_cmd_hs20_icon_request(struct wpa_ctrl *ctrl, int argc,
2603 printf("Command needs two arguments (dst mac addr and "
2608 if (write_cmd(cmd, sizeof(cmd), "HS20_ICON_REQUEST", argc, argv) < 0)
2611 return wpa_ctrl_command(ctrl, cmd);
2615 static int wpa_cli_cmd_fetch_osu(struct wpa_ctrl *ctrl, int argc, char *argv[])
2617 return wpa_ctrl_command(ctrl, "FETCH_OSU");
2621 static int wpa_cli_cmd_cancel_fetch_osu(struct wpa_ctrl *ctrl, int argc,
2624 return wpa_ctrl_command(ctrl, "CANCEL_FETCH_OSU");
2627 #endif /* CONFIG_HS20 */
2630 static int wpa_cli_cmd_sta_autoconnect(struct wpa_ctrl *ctrl, int argc,
2633 return wpa_cli_cmd(ctrl, "STA_AUTOCONNECT", 1, argc, argv);
2637 static int wpa_cli_cmd_tdls_discover(struct wpa_ctrl *ctrl, int argc,
2640 return wpa_cli_cmd(ctrl, "TDLS_DISCOVER", 1, argc, argv);
2644 static int wpa_cli_cmd_tdls_setup(struct wpa_ctrl *ctrl, int argc,
2647 return wpa_cli_cmd(ctrl, "TDLS_SETUP", 1, argc, argv);
2651 static int wpa_cli_cmd_tdls_teardown(struct wpa_ctrl *ctrl, int argc,
2654 return wpa_cli_cmd(ctrl, "TDLS_TEARDOWN", 1, argc, argv);
2658 static int wpa_cli_cmd_tdls_link_status(struct wpa_ctrl *ctrl, int argc,
2661 return wpa_cli_cmd(ctrl, "TDLS_LINK_STATUS", 1, argc, argv);
2665 static int wpa_cli_cmd_wmm_ac_addts(struct wpa_ctrl *ctrl, int argc,
2668 return wpa_cli_cmd(ctrl, "WMM_AC_ADDTS", 3, argc, argv);
2672 static int wpa_cli_cmd_wmm_ac_delts(struct wpa_ctrl *ctrl, int argc,
2675 return wpa_cli_cmd(ctrl, "WMM_AC_DELTS", 1, argc, argv);
2679 static int wpa_cli_cmd_wmm_ac_status(struct wpa_ctrl *ctrl, int argc,
2682 return wpa_ctrl_command(ctrl, "WMM_AC_STATUS");
2686 static int wpa_cli_cmd_tdls_chan_switch(struct wpa_ctrl *ctrl, int argc,
2689 return wpa_cli_cmd(ctrl, "TDLS_CHAN_SWITCH", 2, argc, argv);
2693 static int wpa_cli_cmd_tdls_cancel_chan_switch(struct wpa_ctrl *ctrl, int argc,
2696 return wpa_cli_cmd(ctrl, "TDLS_CANCEL_CHAN_SWITCH", 1, argc, argv);
2700 static int wpa_cli_cmd_signal_poll(struct wpa_ctrl *ctrl, int argc,
2703 return wpa_ctrl_command(ctrl, "SIGNAL_POLL");
2707 static int wpa_cli_cmd_pktcnt_poll(struct wpa_ctrl *ctrl, int argc,
2710 return wpa_ctrl_command(ctrl, "PKTCNT_POLL");
2714 static int wpa_cli_cmd_reauthenticate(struct wpa_ctrl *ctrl, int argc,
2717 return wpa_ctrl_command(ctrl, "REAUTHENTICATE");
2721 #ifdef CONFIG_AUTOSCAN
2723 static int wpa_cli_cmd_autoscan(struct wpa_ctrl *ctrl, int argc, char *argv[])
2726 return wpa_ctrl_command(ctrl, "AUTOSCAN ");
2728 return wpa_cli_cmd(ctrl, "AUTOSCAN", 0, argc, argv);
2731 #endif /* CONFIG_AUTOSCAN */
2736 static int wpa_cli_cmd_wnm_sleep(struct wpa_ctrl *ctrl, int argc, char *argv[])
2738 return wpa_cli_cmd(ctrl, "WNM_SLEEP", 0, argc, argv);
2742 static int wpa_cli_cmd_wnm_bss_query(struct wpa_ctrl *ctrl, int argc, char *argv[])
2744 return wpa_cli_cmd(ctrl, "WNM_BSS_QUERY", 1, argc, argv);
2747 #endif /* CONFIG_WNM */
2750 static int wpa_cli_cmd_raw(struct wpa_ctrl *ctrl, int argc, char *argv[])
2754 return wpa_cli_cmd(ctrl, argv[0], 0, argc - 1, &argv[1]);
2759 static int wpa_cli_cmd_driver(struct wpa_ctrl *ctrl, int argc, char *argv[])
2761 return wpa_cli_cmd(ctrl, "DRIVER", 1, argc, argv);
2763 #endif /* ANDROID */
2766 static int wpa_cli_cmd_vendor(struct wpa_ctrl *ctrl, int argc, char *argv[])
2768 return wpa_cli_cmd(ctrl, "VENDOR", 1, argc, argv);
2772 static int wpa_cli_cmd_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
2774 return wpa_ctrl_command(ctrl, "FLUSH");
2778 static int wpa_cli_cmd_radio_work(struct wpa_ctrl *ctrl, int argc, char *argv[])
2780 return wpa_cli_cmd(ctrl, "RADIO_WORK", 1, argc, argv);
2784 static int wpa_cli_cmd_neighbor_rep_request(struct wpa_ctrl *ctrl, int argc,
2787 return wpa_cli_cmd(ctrl, "NEIGHBOR_REP_REQUEST", 0, argc, argv);
2791 static int wpa_cli_cmd_erp_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
2793 return wpa_ctrl_command(ctrl, "ERP_FLUSH");
2797 static int wpa_cli_cmd_mac_rand_scan(struct wpa_ctrl *ctrl, int argc,
2800 return wpa_cli_cmd(ctrl, "MAC_RAND_SCAN", 1, argc, argv);
2804 enum wpa_cli_cmd_flags {
2805 cli_cmd_flag_none = 0x00,
2806 cli_cmd_flag_sensitive = 0x01
2809 struct wpa_cli_cmd {
2811 int (*handler)(struct wpa_ctrl *ctrl, int argc, char *argv[]);
2812 char ** (*completion)(const char *str, int pos);
2813 enum wpa_cli_cmd_flags flags;
2817 static const struct wpa_cli_cmd wpa_cli_commands[] = {
2818 { "status", wpa_cli_cmd_status, NULL,
2820 "[verbose] = get current WPA/EAPOL/EAP status" },
2821 { "ifname", wpa_cli_cmd_ifname, NULL,
2823 "= get current interface name" },
2824 { "ping", wpa_cli_cmd_ping, NULL,
2826 "= pings wpa_supplicant" },
2827 { "relog", wpa_cli_cmd_relog, NULL,
2829 "= re-open log-file (allow rolling logs)" },
2830 { "note", wpa_cli_cmd_note, NULL,
2832 "<text> = add a note to wpa_supplicant debug log" },
2833 { "mib", wpa_cli_cmd_mib, NULL,
2835 "= get MIB variables (dot1x, dot11)" },
2836 { "help", wpa_cli_cmd_help, wpa_cli_complete_help,
2838 "[command] = show usage help" },
2839 { "interface", wpa_cli_cmd_interface, NULL,
2841 "[ifname] = show interfaces/select interface" },
2842 { "level", wpa_cli_cmd_level, NULL,
2844 "<debug level> = change debug level" },
2845 { "license", wpa_cli_cmd_license, NULL,
2847 "= show full wpa_cli license" },
2848 { "quit", wpa_cli_cmd_quit, NULL,
2851 { "set", wpa_cli_cmd_set, wpa_cli_complete_set,
2853 "= set variables (shows list of variables when run without "
2855 { "dump", wpa_cli_cmd_dump, NULL,
2857 "= dump config variables" },
2858 { "get", wpa_cli_cmd_get, wpa_cli_complete_get,
2860 "<name> = get information" },
2861 { "logon", wpa_cli_cmd_logon, NULL,
2863 "= IEEE 802.1X EAPOL state machine logon" },
2864 { "logoff", wpa_cli_cmd_logoff, NULL,
2866 "= IEEE 802.1X EAPOL state machine logoff" },
2867 { "pmksa", wpa_cli_cmd_pmksa, NULL,
2869 "= show PMKSA cache" },
2870 { "pmksa_flush", wpa_cli_cmd_pmksa_flush, NULL,
2872 "= flush PMKSA cache entries" },
2873 { "reassociate", wpa_cli_cmd_reassociate, NULL,
2875 "= force reassociation" },
2876 { "reattach", wpa_cli_cmd_reattach, NULL,
2878 "= force reassociation back to the same BSS" },
2879 { "preauthenticate", wpa_cli_cmd_preauthenticate, wpa_cli_complete_bss,
2881 "<BSSID> = force preauthentication" },
2882 { "identity", wpa_cli_cmd_identity, NULL,
2884 "<network id> <identity> = configure identity for an SSID" },
2885 { "password", wpa_cli_cmd_password, NULL,
2886 cli_cmd_flag_sensitive,
2887 "<network id> <password> = configure password for an SSID" },
2888 { "new_password", wpa_cli_cmd_new_password, NULL,
2889 cli_cmd_flag_sensitive,
2890 "<network id> <password> = change password for an SSID" },
2891 { "pin", wpa_cli_cmd_pin, NULL,
2892 cli_cmd_flag_sensitive,
2893 "<network id> <pin> = configure pin for an SSID" },
2894 { "otp", wpa_cli_cmd_otp, NULL,
2895 cli_cmd_flag_sensitive,
2896 "<network id> <password> = configure one-time-password for an SSID"
2898 { "passphrase", wpa_cli_cmd_passphrase, NULL,
2899 cli_cmd_flag_sensitive,
2900 "<network id> <passphrase> = configure private key passphrase\n"
2902 { "sim", wpa_cli_cmd_sim, NULL,
2903 cli_cmd_flag_sensitive,
2904 "<network id> <pin> = report SIM operation result" },
2905 { "bssid", wpa_cli_cmd_bssid, NULL,
2907 "<network id> <BSSID> = set preferred BSSID for an SSID" },
2908 { "blacklist", wpa_cli_cmd_blacklist, wpa_cli_complete_bss,
2910 "<BSSID> = add a BSSID to the blacklist\n"
2911 "blacklist clear = clear the blacklist\n"
2912 "blacklist = display the blacklist" },
2913 { "log_level", wpa_cli_cmd_log_level, NULL,
2915 "<level> [<timestamp>] = update the log level/timestamp\n"
2916 "log_level = display the current log level and log options" },
2917 { "list_networks", wpa_cli_cmd_list_networks, NULL,
2919 "= list configured networks" },
2920 { "select_network", wpa_cli_cmd_select_network,
2921 wpa_cli_complete_network_id,
2923 "<network id> = select a network (disable others)" },
2924 { "enable_network", wpa_cli_cmd_enable_network,
2925 wpa_cli_complete_network_id,
2927 "<network id> = enable a network" },
2928 { "disable_network", wpa_cli_cmd_disable_network,
2929 wpa_cli_complete_network_id,
2931 "<network id> = disable a network" },
2932 { "add_network", wpa_cli_cmd_add_network, NULL,
2934 "= add a network" },
2935 { "remove_network", wpa_cli_cmd_remove_network,
2936 wpa_cli_complete_network_id,
2938 "<network id> = remove a network" },
2939 { "set_network", wpa_cli_cmd_set_network, wpa_cli_complete_network,
2940 cli_cmd_flag_sensitive,
2941 "<network id> <variable> <value> = set network variables (shows\n"
2942 " list of variables when run without arguments)" },
2943 { "get_network", wpa_cli_cmd_get_network, wpa_cli_complete_network,
2945 "<network id> <variable> = get network variables" },
2946 { "dup_network", wpa_cli_cmd_dup_network, wpa_cli_complete_dup_network,
2948 "<src network id> <dst network id> <variable> = duplicate network variables"
2950 { "list_creds", wpa_cli_cmd_list_creds, NULL,
2952 "= list configured credentials" },
2953 { "add_cred", wpa_cli_cmd_add_cred, NULL,
2955 "= add a credential" },
2956 { "remove_cred", wpa_cli_cmd_remove_cred, NULL,
2958 "<cred id> = remove a credential" },
2959 { "set_cred", wpa_cli_cmd_set_cred, NULL,
2960 cli_cmd_flag_sensitive,
2961 "<cred id> <variable> <value> = set credential variables" },
2962 { "get_cred", wpa_cli_cmd_get_cred, NULL,
2964 "<cred id> <variable> = get credential variables" },
2965 { "save_config", wpa_cli_cmd_save_config, NULL,
2967 "= save the current configuration" },
2968 { "disconnect", wpa_cli_cmd_disconnect, NULL,
2970 "= disconnect and wait for reassociate/reconnect command before\n"
2972 { "reconnect", wpa_cli_cmd_reconnect, NULL,
2974 "= like reassociate, but only takes effect if already disconnected"
2976 { "scan", wpa_cli_cmd_scan, NULL,
2978 "= request new BSS scan" },
2979 { "scan_results", wpa_cli_cmd_scan_results, NULL,
2981 "= get latest scan results" },
2982 { "bss", wpa_cli_cmd_bss, wpa_cli_complete_bss,
2984 "<<idx> | <bssid>> = get detailed scan result info" },
2985 { "get_capability", wpa_cli_cmd_get_capability, NULL,
2987 "<eap/pairwise/group/key_mgmt/proto/auth_alg/channels/freq/modes> "
2988 "= get capabilities" },
2989 { "reconfigure", wpa_cli_cmd_reconfigure, NULL,
2991 "= force wpa_supplicant to re-read its configuration file" },
2992 { "terminate", wpa_cli_cmd_terminate, NULL,
2994 "= terminate wpa_supplicant" },
2995 { "interface_add", wpa_cli_cmd_interface_add, NULL,
2997 "<ifname> <confname> <driver> <ctrl_interface> <driver_param>\n"
2998 " <bridge_name> = adds new interface, all parameters but <ifname>\n"
3000 { "interface_remove", wpa_cli_cmd_interface_remove, NULL,
3002 "<ifname> = removes the interface" },
3003 { "interface_list", wpa_cli_cmd_interface_list, NULL,
3005 "= list available interfaces" },
3006 { "ap_scan", wpa_cli_cmd_ap_scan, NULL,
3008 "<value> = set ap_scan parameter" },
3009 { "scan_interval", wpa_cli_cmd_scan_interval, NULL,
3011 "<value> = set scan_interval parameter (in seconds)" },
3012 { "bss_expire_age", wpa_cli_cmd_bss_expire_age, NULL,
3014 "<value> = set BSS expiration age parameter" },
3015 { "bss_expire_count", wpa_cli_cmd_bss_expire_count, NULL,
3017 "<value> = set BSS expiration scan count parameter" },
3018 { "bss_flush", wpa_cli_cmd_bss_flush, NULL,
3020 "<value> = set BSS flush age (0 by default)" },
3021 { "stkstart", wpa_cli_cmd_stkstart, NULL,
3023 "<addr> = request STK negotiation with <addr>" },
3024 { "ft_ds", wpa_cli_cmd_ft_ds, wpa_cli_complete_bss,
3026 "<addr> = request over-the-DS FT with <addr>" },
3027 { "wps_pbc", wpa_cli_cmd_wps_pbc, wpa_cli_complete_bss,
3029 "[BSSID] = start Wi-Fi Protected Setup: Push Button Configuration" },
3030 { "wps_pin", wpa_cli_cmd_wps_pin, wpa_cli_complete_bss,
3031 cli_cmd_flag_sensitive,
3032 "<BSSID> [PIN] = start WPS PIN method (returns PIN, if not "
3034 { "wps_check_pin", wpa_cli_cmd_wps_check_pin, NULL,
3035 cli_cmd_flag_sensitive,
3036 "<PIN> = verify PIN checksum" },
3037 { "wps_cancel", wpa_cli_cmd_wps_cancel, NULL, cli_cmd_flag_none,
3038 "Cancels the pending WPS operation" },
3039 #ifdef CONFIG_WPS_NFC
3040 { "wps_nfc", wpa_cli_cmd_wps_nfc, wpa_cli_complete_bss,
3042 "[BSSID] = start Wi-Fi Protected Setup: NFC" },
3043 { "wps_nfc_config_token", wpa_cli_cmd_wps_nfc_config_token, NULL,
3045 "<WPS|NDEF> = build configuration token" },
3046 { "wps_nfc_token", wpa_cli_cmd_wps_nfc_token, NULL,
3048 "<WPS|NDEF> = create password token" },
3049 { "wps_nfc_tag_read", wpa_cli_cmd_wps_nfc_tag_read, NULL,
3050 cli_cmd_flag_sensitive,
3051 "<hexdump of payload> = report read NFC tag with WPS data" },
3052 { "nfc_get_handover_req", wpa_cli_cmd_nfc_get_handover_req, NULL,
3054 "<NDEF> <WPS> = create NFC handover request" },
3055 { "nfc_get_handover_sel", wpa_cli_cmd_nfc_get_handover_sel, NULL,
3057 "<NDEF> <WPS> = create NFC handover select" },
3058 { "nfc_report_handover", wpa_cli_cmd_nfc_report_handover, NULL,
3060 "<role> <type> <hexdump of req> <hexdump of sel> = report completed "
3062 #endif /* CONFIG_WPS_NFC */
3063 { "wps_reg", wpa_cli_cmd_wps_reg, wpa_cli_complete_bss,
3064 cli_cmd_flag_sensitive,
3065 "<BSSID> <AP PIN> = start WPS Registrar to configure an AP" },
3066 { "wps_ap_pin", wpa_cli_cmd_wps_ap_pin, NULL,
3067 cli_cmd_flag_sensitive,
3068 "[params..] = enable/disable AP PIN" },
3069 { "wps_er_start", wpa_cli_cmd_wps_er_start, NULL,
3071 "[IP address] = start Wi-Fi Protected Setup External Registrar" },
3072 { "wps_er_stop", wpa_cli_cmd_wps_er_stop, NULL,
3074 "= stop Wi-Fi Protected Setup External Registrar" },
3075 { "wps_er_pin", wpa_cli_cmd_wps_er_pin, NULL,
3076 cli_cmd_flag_sensitive,
3077 "<UUID> <PIN> = add an Enrollee PIN to External Registrar" },
3078 { "wps_er_pbc", wpa_cli_cmd_wps_er_pbc, NULL,
3080 "<UUID> = accept an Enrollee PBC using External Registrar" },
3081 { "wps_er_learn", wpa_cli_cmd_wps_er_learn, NULL,
3082 cli_cmd_flag_sensitive,
3083 "<UUID> <PIN> = learn AP configuration" },
3084 { "wps_er_set_config", wpa_cli_cmd_wps_er_set_config, NULL,
3086 "<UUID> <network id> = set AP configuration for enrolling" },
3087 { "wps_er_config", wpa_cli_cmd_wps_er_config, NULL,
3088 cli_cmd_flag_sensitive,
3089 "<UUID> <PIN> <SSID> <auth> <encr> <key> = configure AP" },
3090 #ifdef CONFIG_WPS_NFC
3091 { "wps_er_nfc_config_token", wpa_cli_cmd_wps_er_nfc_config_token, NULL,
3093 "<WPS/NDEF> <UUID> = build NFC configuration token" },
3094 #endif /* CONFIG_WPS_NFC */
3095 { "ibss_rsn", wpa_cli_cmd_ibss_rsn, NULL,
3097 "<addr> = request RSN authentication with <addr> in IBSS" },
3099 { "sta", wpa_cli_cmd_sta, NULL,
3101 "<addr> = get information about an associated station (AP)" },
3102 { "all_sta", wpa_cli_cmd_all_sta, NULL,
3104 "= get information about all associated stations (AP)" },
3105 { "deauthenticate", wpa_cli_cmd_deauthenticate, NULL,
3107 "<addr> = deauthenticate a station" },
3108 { "disassociate", wpa_cli_cmd_disassociate, NULL,
3110 "<addr> = disassociate a station" },
3111 { "chan_switch", wpa_cli_cmd_chanswitch, NULL,
3113 "<cs_count> <freq> [sec_channel_offset=] [center_freq1=]"
3114 " [center_freq2=] [bandwidth=] [blocktx] [ht|vht]"
3115 " = CSA parameters" },
3116 #endif /* CONFIG_AP */
3117 { "suspend", wpa_cli_cmd_suspend, NULL, cli_cmd_flag_none,
3118 "= notification of suspend/hibernate" },
3119 { "resume", wpa_cli_cmd_resume, NULL, cli_cmd_flag_none,
3120 "= notification of resume/thaw" },
3121 #ifdef CONFIG_TESTING_OPTIONS
3122 { "drop_sa", wpa_cli_cmd_drop_sa, NULL, cli_cmd_flag_none,
3123 "= drop SA without deauth/disassoc (test command)" },
3124 #endif /* CONFIG_TESTING_OPTIONS */
3125 { "roam", wpa_cli_cmd_roam, wpa_cli_complete_bss,
3127 "<addr> = roam to the specified BSS" },
3129 { "mesh_interface_add", wpa_cli_cmd_mesh_interface_add, NULL,
3131 "[ifname] = Create a new mesh interface" },
3132 { "mesh_group_add", wpa_cli_cmd_mesh_group_add, NULL,
3134 "<network id> = join a mesh network (disable others)" },
3135 { "mesh_group_remove", wpa_cli_cmd_mesh_group_remove, NULL,
3137 "<ifname> = Remove mesh group interface" },
3138 #endif /* CONFIG_MESH */
3140 { "p2p_find", wpa_cli_cmd_p2p_find, wpa_cli_complete_p2p_find,
3142 "[timeout] [type=*] = find P2P Devices for up-to timeout seconds" },
3143 { "p2p_stop_find", wpa_cli_cmd_p2p_stop_find, NULL, cli_cmd_flag_none,
3144 "= stop P2P Devices search" },
3145 { "p2p_asp_provision", wpa_cli_cmd_p2p_asp_provision, NULL,
3147 "<addr> adv_id=<adv_id> conncap=<conncap> [info=<infodata>] = provision with a P2P ASP Device" },
3148 { "p2p_asp_provision_resp", wpa_cli_cmd_p2p_asp_provision_resp, NULL,
3150 "<addr> adv_id=<adv_id> [role<conncap>] [info=<infodata>] = provision with a P2P ASP Device" },
3151 { "p2p_connect", wpa_cli_cmd_p2p_connect, wpa_cli_complete_p2p_connect,
3153 "<addr> <\"pbc\"|PIN> [ht40] = connect to a P2P Device" },
3154 { "p2p_listen", wpa_cli_cmd_p2p_listen, NULL, cli_cmd_flag_none,
3155 "[timeout] = listen for P2P Devices for up-to timeout seconds" },
3156 { "p2p_group_remove", wpa_cli_cmd_p2p_group_remove,
3157 wpa_cli_complete_p2p_group_remove, cli_cmd_flag_none,
3158 "<ifname> = remove P2P group interface (terminate group if GO)" },
3159 { "p2p_group_add", wpa_cli_cmd_p2p_group_add, NULL, cli_cmd_flag_none,
3160 "[ht40] = add a new P2P group (local end as GO)" },
3161 { "p2p_prov_disc", wpa_cli_cmd_p2p_prov_disc,
3162 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
3163 "<addr> <method> = request provisioning discovery" },
3164 { "p2p_get_passphrase", wpa_cli_cmd_p2p_get_passphrase, NULL,
3166 "= get the passphrase for a group (GO only)" },
3167 { "p2p_serv_disc_req", wpa_cli_cmd_p2p_serv_disc_req,
3168 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
3169 "<addr> <TLVs> = schedule service discovery request" },
3170 { "p2p_serv_disc_cancel_req", wpa_cli_cmd_p2p_serv_disc_cancel_req,
3171 NULL, cli_cmd_flag_none,
3172 "<id> = cancel pending service discovery request" },
3173 { "p2p_serv_disc_resp", wpa_cli_cmd_p2p_serv_disc_resp, NULL,
3175 "<freq> <addr> <dialog token> <TLVs> = service discovery response" },
3176 { "p2p_service_update", wpa_cli_cmd_p2p_service_update, NULL,
3178 "= indicate change in local services" },
3179 { "p2p_serv_disc_external", wpa_cli_cmd_p2p_serv_disc_external, NULL,
3181 "<external> = set external processing of service discovery" },
3182 { "p2p_service_flush", wpa_cli_cmd_p2p_service_flush, NULL,
3184 "= remove all stored service entries" },
3185 { "p2p_service_add", wpa_cli_cmd_p2p_service_add, NULL,
3187 "<bonjour|upnp|asp> <query|version> <response|service> = add a local "
3189 { "p2p_service_rep", wpa_cli_cmd_p2p_service_rep, NULL,
3191 "asp <auto> <adv_id> <svc_state> <svc_string> [<svc_info>] = replace "
3192 "local ASP service" },
3193 { "p2p_service_del", wpa_cli_cmd_p2p_service_del, NULL,
3195 "<bonjour|upnp> <query|version> [|service] = remove a local "
3197 { "p2p_reject", wpa_cli_cmd_p2p_reject, wpa_cli_complete_p2p_peer,
3199 "<addr> = reject connection attempts from a specific peer" },
3200 { "p2p_invite", wpa_cli_cmd_p2p_invite, NULL,
3202 "<cmd> [peer=addr] = invite peer" },
3203 { "p2p_peers", wpa_cli_cmd_p2p_peers, NULL, cli_cmd_flag_none,
3204 "[discovered] = list known (optionally, only fully discovered) P2P "
3206 { "p2p_peer", wpa_cli_cmd_p2p_peer, wpa_cli_complete_p2p_peer,
3208 "<address> = show information about known P2P peer" },
3209 { "p2p_set", wpa_cli_cmd_p2p_set, wpa_cli_complete_p2p_set,
3211 "<field> <value> = set a P2P parameter" },
3212 { "p2p_flush", wpa_cli_cmd_p2p_flush, NULL, cli_cmd_flag_none,
3213 "= flush P2P state" },
3214 { "p2p_cancel", wpa_cli_cmd_p2p_cancel, NULL, cli_cmd_flag_none,
3215 "= cancel P2P group formation" },
3216 { "p2p_unauthorize", wpa_cli_cmd_p2p_unauthorize,
3217 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
3218 "<address> = unauthorize a peer" },
3219 { "p2p_presence_req", wpa_cli_cmd_p2p_presence_req, NULL,
3221 "[<duration> <interval>] [<duration> <interval>] = request GO "
3223 { "p2p_ext_listen", wpa_cli_cmd_p2p_ext_listen, NULL,
3225 "[<period> <interval>] = set extended listen timing" },
3226 { "p2p_remove_client", wpa_cli_cmd_p2p_remove_client,
3227 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
3228 "<address|iface=address> = remove a peer from all groups" },
3229 #endif /* CONFIG_P2P */
3230 #ifdef CONFIG_WIFI_DISPLAY
3231 { "wfd_subelem_set", wpa_cli_cmd_wfd_subelem_set, NULL,
3233 "<subelem> [contents] = set Wi-Fi Display subelement" },
3234 { "wfd_subelem_get", wpa_cli_cmd_wfd_subelem_get, NULL,
3236 "<subelem> = get Wi-Fi Display subelement" },
3237 #endif /* CONFIG_WIFI_DISPLAY */
3238 #ifdef CONFIG_INTERWORKING
3239 { "fetch_anqp", wpa_cli_cmd_fetch_anqp, NULL, cli_cmd_flag_none,
3240 "= fetch ANQP information for all APs" },
3241 { "stop_fetch_anqp", wpa_cli_cmd_stop_fetch_anqp, NULL,
3243 "= stop fetch_anqp operation" },
3244 { "interworking_select", wpa_cli_cmd_interworking_select, NULL,
3246 "[auto] = perform Interworking network selection" },
3247 { "interworking_connect", wpa_cli_cmd_interworking_connect,
3248 wpa_cli_complete_bss, cli_cmd_flag_none,
3249 "<BSSID> = connect using Interworking credentials" },
3250 { "interworking_add_network", wpa_cli_cmd_interworking_add_network,
3251 wpa_cli_complete_bss, cli_cmd_flag_none,
3252 "<BSSID> = connect using Interworking credentials" },
3253 { "anqp_get", wpa_cli_cmd_anqp_get, wpa_cli_complete_bss,
3255 "<addr> <info id>[,<info id>]... = request ANQP information" },
3256 { "gas_request", wpa_cli_cmd_gas_request, wpa_cli_complete_bss,
3258 "<addr> <AdvProtoID> [QueryReq] = GAS request" },
3259 { "gas_response_get", wpa_cli_cmd_gas_response_get,
3260 wpa_cli_complete_bss, cli_cmd_flag_none,
3261 "<addr> <dialog token> [start,len] = Fetch last GAS response" },
3262 #endif /* CONFIG_INTERWORKING */
3264 { "hs20_anqp_get", wpa_cli_cmd_hs20_anqp_get, wpa_cli_complete_bss,
3266 "<addr> <subtype>[,<subtype>]... = request HS 2.0 ANQP information"
3268 { "nai_home_realm_list", wpa_cli_cmd_get_nai_home_realm_list,
3269 wpa_cli_complete_bss, cli_cmd_flag_none,
3270 "<addr> <home realm> = get HS20 nai home realm list" },
3271 { "hs20_icon_request", wpa_cli_cmd_hs20_icon_request,
3272 wpa_cli_complete_bss, cli_cmd_flag_none,
3273 "<addr> <icon name> = get Hotspot 2.0 OSU icon" },
3274 { "fetch_osu", wpa_cli_cmd_fetch_osu, NULL, cli_cmd_flag_none,
3275 "= fetch OSU provider information from all APs" },
3276 { "cancel_fetch_osu", wpa_cli_cmd_cancel_fetch_osu, NULL,
3278 "= cancel fetch_osu command" },
3279 #endif /* CONFIG_HS20 */
3280 { "sta_autoconnect", wpa_cli_cmd_sta_autoconnect, NULL,
3282 "<0/1> = disable/enable automatic reconnection" },
3283 { "tdls_discover", wpa_cli_cmd_tdls_discover, NULL,
3285 "<addr> = request TDLS discovery with <addr>" },
3286 { "tdls_setup", wpa_cli_cmd_tdls_setup, NULL,
3288 "<addr> = request TDLS setup with <addr>" },
3289 { "tdls_teardown", wpa_cli_cmd_tdls_teardown, NULL,
3291 "<addr> = tear down TDLS with <addr>" },
3292 { "tdls_link_status", wpa_cli_cmd_tdls_link_status, NULL,
3294 "<addr> = TDLS link status with <addr>" },
3295 { "wmm_ac_addts", wpa_cli_cmd_wmm_ac_addts, NULL,
3297 "<uplink/downlink/bidi> <tsid=0..7> <up=0..7> [nominal_msdu_size=#] "
3298 "[mean_data_rate=#] [min_phy_rate=#] [sba=#] [fixed_nominal_msdu] "
3299 "= add WMM-AC traffic stream" },
3300 { "wmm_ac_delts", wpa_cli_cmd_wmm_ac_delts, NULL,
3302 "<tsid> = delete WMM-AC traffic stream" },
3303 { "wmm_ac_status", wpa_cli_cmd_wmm_ac_status, NULL,
3305 "= show status for Wireless Multi-Media Admission-Control" },
3306 { "tdls_chan_switch", wpa_cli_cmd_tdls_chan_switch, NULL,
3308 "<addr> <oper class> <freq> [sec_channel_offset=] [center_freq1=] "
3309 "[center_freq2=] [bandwidth=] [ht|vht] = enable channel switching "
3311 { "tdls_cancel_chan_switch", wpa_cli_cmd_tdls_cancel_chan_switch, NULL,
3313 "<addr> = disable channel switching with TDLS peer <addr>" },
3314 { "signal_poll", wpa_cli_cmd_signal_poll, NULL,
3316 "= get signal parameters" },
3317 { "pktcnt_poll", wpa_cli_cmd_pktcnt_poll, NULL,
3319 "= get TX/RX packet counters" },
3320 { "reauthenticate", wpa_cli_cmd_reauthenticate, NULL,
3322 "= trigger IEEE 802.1X/EAPOL reauthentication" },
3323 #ifdef CONFIG_AUTOSCAN
3324 { "autoscan", wpa_cli_cmd_autoscan, NULL, cli_cmd_flag_none,
3325 "[params] = Set or unset (if none) autoscan parameters" },
3326 #endif /* CONFIG_AUTOSCAN */
3328 { "wnm_sleep", wpa_cli_cmd_wnm_sleep, NULL, cli_cmd_flag_none,
3329 "<enter/exit> [interval=#] = enter/exit WNM-Sleep mode" },
3330 { "wnm_bss_query", wpa_cli_cmd_wnm_bss_query, NULL, cli_cmd_flag_none,
3331 "<query reason> = Send BSS Transition Management Query" },
3332 #endif /* CONFIG_WNM */
3333 { "raw", wpa_cli_cmd_raw, NULL, cli_cmd_flag_sensitive,
3334 "<params..> = Sent unprocessed command" },
3335 { "flush", wpa_cli_cmd_flush, NULL, cli_cmd_flag_none,
3336 "= flush wpa_supplicant state" },
3338 { "driver", wpa_cli_cmd_driver, NULL, cli_cmd_flag_none,
3339 "<command> = driver private commands" },
3340 #endif /* ANDROID */
3341 { "radio_work", wpa_cli_cmd_radio_work, NULL, cli_cmd_flag_none,
3342 "= radio_work <show/add/done>" },
3343 { "vendor", wpa_cli_cmd_vendor, NULL, cli_cmd_flag_none,
3344 "<vendor id> <command id> [<hex formatted command argument>] = Send vendor command"
3346 { "neighbor_rep_request",
3347 wpa_cli_cmd_neighbor_rep_request, NULL, cli_cmd_flag_none,
3348 "[ssid=<SSID>] = Trigger request to AP for neighboring AP report "
3349 "(with optional given SSID, default: current SSID)"
3351 { "erp_flush", wpa_cli_cmd_erp_flush, NULL, cli_cmd_flag_none,
3352 "= flush ERP keys" },
3354 wpa_cli_cmd_mac_rand_scan, NULL, cli_cmd_flag_none,
3355 "<scan|sched|pno|all> enable=<0/1> [addr=mac-address "
3356 "mask=mac-address-mask] = scan MAC randomization"
3358 { NULL, NULL, NULL, cli_cmd_flag_none, NULL }
3363 * Prints command usage, lines are padded with the specified string.
3365 static void print_cmd_help(const struct wpa_cli_cmd *cmd, const char *pad)
3370 printf("%s%s ", pad, cmd->cmd);
3371 for (n = 0; (c = cmd->usage[n]); n++) {
3380 static void print_help(const char *cmd)
3383 printf("commands:\n");
3384 for (n = 0; wpa_cli_commands[n].cmd; n++) {
3385 if (cmd == NULL || str_starts(wpa_cli_commands[n].cmd, cmd))
3386 print_cmd_help(&wpa_cli_commands[n], " ");
3391 static int wpa_cli_edit_filter_history_cb(void *ctx, const char *cmd)
3393 const char *c, *delim;
3397 delim = os_strchr(cmd, ' ');
3401 len = os_strlen(cmd);
3403 for (n = 0; (c = wpa_cli_commands[n].cmd); n++) {
3404 if (os_strncasecmp(cmd, c, len) == 0 && len == os_strlen(c))
3405 return (wpa_cli_commands[n].flags &
3406 cli_cmd_flag_sensitive);
3412 static char ** wpa_list_cmd_list(void)
3416 struct cli_txt_entry *e;
3418 count = ARRAY_SIZE(wpa_cli_commands);
3419 count += dl_list_len(&p2p_groups);
3420 count += dl_list_len(&ifnames);
3421 res = os_calloc(count + 1, sizeof(char *));
3425 for (i = 0; wpa_cli_commands[i].cmd; i++) {
3426 res[i] = os_strdup(wpa_cli_commands[i].cmd);
3431 dl_list_for_each(e, &p2p_groups, struct cli_txt_entry, list) {
3432 size_t len = 8 + os_strlen(e->txt);
3433 res[i] = os_malloc(len);
3436 os_snprintf(res[i], len, "ifname=%s", e->txt);
3440 dl_list_for_each(e, &ifnames, struct cli_txt_entry, list) {
3441 res[i] = os_strdup(e->txt);
3451 static char ** wpa_cli_cmd_completion(const char *cmd, const char *str,
3456 for (i = 0; wpa_cli_commands[i].cmd; i++) {
3457 if (os_strcasecmp(wpa_cli_commands[i].cmd, cmd) == 0) {
3458 if (wpa_cli_commands[i].completion)
3459 return wpa_cli_commands[i].completion(str,
3462 printf("\r%s\n", wpa_cli_commands[i].usage);
3472 static char ** wpa_cli_edit_completion_cb(void *ctx, const char *str, int pos)
3478 if (pos > 7 && os_strncasecmp(str, "IFNAME=", 7) == 0) {
3479 end = os_strchr(str, ' ');
3480 if (end && pos > end - str) {
3481 pos -= end - str + 1;
3486 end = os_strchr(str, ' ');
3487 if (end == NULL || str + pos < end)
3488 return wpa_list_cmd_list();
3490 cmd = os_malloc(pos + 1);
3493 os_memcpy(cmd, str, pos);
3494 cmd[end - str] = '\0';
3495 res = wpa_cli_cmd_completion(cmd, str, pos);
3501 static int wpa_request(struct wpa_ctrl *ctrl, int argc, char *argv[])
3503 const struct wpa_cli_cmd *cmd, *match = NULL;
3507 if (argc > 1 && os_strncasecmp(argv[0], "IFNAME=", 7) == 0) {
3508 ifname_prefix = argv[0] + 7;
3512 ifname_prefix = NULL;
3518 cmd = wpa_cli_commands;
3520 if (os_strncasecmp(cmd->cmd, argv[0], os_strlen(argv[0])) == 0)
3523 if (os_strcasecmp(cmd->cmd, argv[0]) == 0) {
3524 /* we have an exact match */
3534 printf("Ambiguous command '%s'; possible commands:", argv[0]);
3535 cmd = wpa_cli_commands;
3537 if (os_strncasecmp(cmd->cmd, argv[0],
3538 os_strlen(argv[0])) == 0) {
3539 printf(" %s", cmd->cmd);
3545 } else if (count == 0) {
3546 printf("Unknown command '%s'\n", argv[0]);
3549 ret = match->handler(ctrl, argc - 1, &argv[1]);
3556 static int str_match(const char *a, const char *b)
3558 return os_strncmp(a, b, os_strlen(b)) == 0;
3562 static int wpa_cli_exec(const char *program, const char *arg1,
3569 len = os_strlen(arg1) + os_strlen(arg2) + 2;
3570 arg = os_malloc(len);
3573 os_snprintf(arg, len, "%s %s", arg1, arg2);
3574 res = os_exec(program, arg, 1);
3581 static void wpa_cli_action_process(const char *msg)
3584 char *copy = NULL, *id, *pos2;
3585 const char *ifname = ctrl_ifname;
3586 char ifname_buf[100];
3588 if (eloop_terminated())
3592 if (os_strncmp(pos, "IFNAME=", 7) == 0) {
3594 end = os_strchr(pos + 7, ' ');
3595 if (end && (unsigned int) (end - pos) < sizeof(ifname_buf)) {
3597 os_memcpy(ifname_buf, pos, end - pos);
3598 ifname_buf[end - pos] = '\0';
3599 ifname = ifname_buf;
3604 const char *prev = pos;
3606 pos = os_strchr(pos, '>');
3613 if (str_match(pos, WPA_EVENT_CONNECTED)) {
3615 os_unsetenv("WPA_ID");
3616 os_unsetenv("WPA_ID_STR");
3617 os_unsetenv("WPA_CTRL_DIR");
3619 pos = os_strstr(pos, "[id=");
3621 copy = os_strdup(pos + 4);
3625 while (*pos2 && *pos2 != ' ')
3629 os_setenv("WPA_ID", id, 1);
3630 while (*pos2 && *pos2 != '=')
3635 while (*pos2 && *pos2 != ']')
3638 os_setenv("WPA_ID_STR", id, 1);
3642 os_setenv("WPA_CTRL_DIR", ctrl_iface_dir, 1);
3644 if (wpa_cli_connected <= 0 || new_id != wpa_cli_last_id) {
3645 wpa_cli_connected = 1;
3646 wpa_cli_last_id = new_id;
3647 wpa_cli_exec(action_file, ifname, "CONNECTED");
3649 } else if (str_match(pos, WPA_EVENT_DISCONNECTED)) {
3650 if (wpa_cli_connected) {
3651 wpa_cli_connected = 0;
3652 wpa_cli_exec(action_file, ifname, "DISCONNECTED");
3654 } else if (str_match(pos, MESH_GROUP_STARTED)) {
3655 wpa_cli_exec(action_file, ctrl_ifname, pos);
3656 } else if (str_match(pos, MESH_GROUP_REMOVED)) {
3657 wpa_cli_exec(action_file, ctrl_ifname, pos);
3658 } else if (str_match(pos, MESH_PEER_CONNECTED)) {
3659 wpa_cli_exec(action_file, ctrl_ifname, pos);
3660 } else if (str_match(pos, MESH_PEER_DISCONNECTED)) {
3661 wpa_cli_exec(action_file, ctrl_ifname, pos);
3662 } else if (str_match(pos, P2P_EVENT_GROUP_STARTED)) {
3663 wpa_cli_exec(action_file, ifname, pos);
3664 } else if (str_match(pos, P2P_EVENT_GROUP_REMOVED)) {
3665 wpa_cli_exec(action_file, ifname, pos);
3666 } else if (str_match(pos, P2P_EVENT_CROSS_CONNECT_ENABLE)) {
3667 wpa_cli_exec(action_file, ifname, pos);
3668 } else if (str_match(pos, P2P_EVENT_CROSS_CONNECT_DISABLE)) {
3669 wpa_cli_exec(action_file, ifname, pos);
3670 } else if (str_match(pos, P2P_EVENT_GO_NEG_FAILURE)) {
3671 wpa_cli_exec(action_file, ifname, pos);
3672 } else if (str_match(pos, WPS_EVENT_SUCCESS)) {
3673 wpa_cli_exec(action_file, ifname, pos);
3674 } else if (str_match(pos, WPS_EVENT_FAIL)) {
3675 wpa_cli_exec(action_file, ifname, pos);
3676 } else if (str_match(pos, AP_STA_CONNECTED)) {
3677 wpa_cli_exec(action_file, ifname, pos);
3678 } else if (str_match(pos, AP_STA_DISCONNECTED)) {
3679 wpa_cli_exec(action_file, ifname, pos);
3680 } else if (str_match(pos, ESS_DISASSOC_IMMINENT)) {
3681 wpa_cli_exec(action_file, ifname, pos);
3682 } else if (str_match(pos, HS20_SUBSCRIPTION_REMEDIATION)) {
3683 wpa_cli_exec(action_file, ifname, pos);
3684 } else if (str_match(pos, HS20_DEAUTH_IMMINENT_NOTICE)) {
3685 wpa_cli_exec(action_file, ifname, pos);
3686 } else if (str_match(pos, WPA_EVENT_TERMINATING)) {
3687 printf("wpa_supplicant is terminating - stop monitoring\n");
3693 #ifndef CONFIG_ANSI_C_EXTRA
3694 static void wpa_cli_action_cb(char *msg, size_t len)
3696 wpa_cli_action_process(msg);
3698 #endif /* CONFIG_ANSI_C_EXTRA */
3701 static void wpa_cli_reconnect(void)
3703 wpa_cli_close_connection();
3704 if (wpa_cli_open_connection(ctrl_ifname, 1) < 0)
3709 printf("\rConnection to wpa_supplicant re-established\n");
3715 static void cli_event(const char *str)
3717 const char *start, *s;
3719 start = os_strchr(str, '>');
3725 if (str_starts(start, WPA_EVENT_BSS_ADDED)) {
3726 s = os_strchr(start, ' ');
3729 s = os_strchr(s + 1, ' ');
3732 cli_txt_list_add(&bsses, s + 1);
3736 if (str_starts(start, WPA_EVENT_BSS_REMOVED)) {
3737 s = os_strchr(start, ' ');
3740 s = os_strchr(s + 1, ' ');
3743 cli_txt_list_del_addr(&bsses, s + 1);
3748 if (str_starts(start, P2P_EVENT_DEVICE_FOUND)) {
3749 s = os_strstr(start, " p2p_dev_addr=");
3752 cli_txt_list_add_addr(&p2p_peers, s + 14);
3756 if (str_starts(start, P2P_EVENT_DEVICE_LOST)) {
3757 s = os_strstr(start, " p2p_dev_addr=");
3760 cli_txt_list_del_addr(&p2p_peers, s + 14);
3764 if (str_starts(start, P2P_EVENT_GROUP_STARTED)) {
3765 s = os_strchr(start, ' ');
3768 cli_txt_list_add_word(&p2p_groups, s + 1, ' ');
3772 if (str_starts(start, P2P_EVENT_GROUP_REMOVED)) {
3773 s = os_strchr(start, ' ');
3776 cli_txt_list_del_word(&p2p_groups, s + 1, ' ');
3779 #endif /* CONFIG_P2P */
3783 static int check_terminating(const char *msg)
3785 const char *pos = msg;
3789 pos = os_strchr(pos, '>');
3796 if (str_match(pos, WPA_EVENT_TERMINATING) && ctrl_conn) {
3798 printf("\rConnection to wpa_supplicant lost - trying to "
3801 wpa_cli_attached = 0;
3802 wpa_cli_close_connection();
3810 static void wpa_cli_recv_pending(struct wpa_ctrl *ctrl, int action_monitor)
3812 if (ctrl_conn == NULL) {
3813 wpa_cli_reconnect();
3816 while (wpa_ctrl_pending(ctrl) > 0) {
3818 size_t len = sizeof(buf) - 1;
3819 if (wpa_ctrl_recv(ctrl, buf, &len) == 0) {
3822 wpa_cli_action_process(buf);
3825 if (wpa_cli_show_event(buf)) {
3827 printf("\r%s\n", buf);
3831 if (interactive && check_terminating(buf) > 0)
3835 printf("Could not read pending message.\n");
3840 if (wpa_ctrl_pending(ctrl) < 0) {
3841 printf("Connection to wpa_supplicant lost - trying to "
3843 wpa_cli_reconnect();
3849 static int tokenize_cmd(char *cmd, char *argv[])
3862 if (argc == max_args)
3865 char *pos2 = os_strrchr(pos, '"');
3869 while (*pos != '\0' && *pos != ' ')
3879 static void wpa_cli_ping(void *eloop_ctx, void *timeout_ctx)
3883 char *prefix = ifname_prefix;
3885 ifname_prefix = NULL;
3886 res = _wpa_ctrl_command(ctrl_conn, "PING", 0);
3887 ifname_prefix = prefix;
3889 printf("Connection to wpa_supplicant lost - trying to "
3891 wpa_cli_close_connection();
3895 wpa_cli_reconnect();
3896 eloop_register_timeout(ping_interval, 0, wpa_cli_ping, NULL, NULL);
3900 static void wpa_cli_mon_receive(int sock, void *eloop_ctx, void *sock_ctx)
3902 wpa_cli_recv_pending(mon_conn, 0);
3906 static void wpa_cli_edit_cmd_cb(void *ctx, char *cmd)
3908 char *argv[max_args];
3910 argc = tokenize_cmd(cmd, argv);
3912 wpa_request(ctrl_conn, argc, argv);
3916 static void wpa_cli_edit_eof_cb(void *ctx)
3922 static int warning_displayed = 0;
3923 static char *hfile = NULL;
3924 static int edit_started = 0;
3926 static void start_edit(void)
3931 #ifdef CONFIG_CTRL_IFACE_UDP_REMOTE
3932 ps = wpa_ctrl_get_remote_ifname(ctrl_conn);
3933 #endif /* CONFIG_CTRL_IFACE_UDP_REMOTE */
3935 #ifdef CONFIG_WPA_CLI_HISTORY_DIR
3936 home = CONFIG_WPA_CLI_HISTORY_DIR;
3937 #else /* CONFIG_WPA_CLI_HISTORY_DIR */
3938 home = getenv("HOME");
3939 #endif /* CONFIG_WPA_CLI_HISTORY_DIR */
3941 const char *fname = ".wpa_cli_history";
3942 int hfile_len = os_strlen(home) + 1 + os_strlen(fname) + 1;
3943 hfile = os_malloc(hfile_len);
3945 os_snprintf(hfile, hfile_len, "%s/%s", home, fname);
3948 if (edit_init(wpa_cli_edit_cmd_cb, wpa_cli_edit_eof_cb,
3949 wpa_cli_edit_completion_cb, NULL, hfile, ps) < 0) {
3955 eloop_register_timeout(ping_interval, 0, wpa_cli_ping, NULL, NULL);
3959 static void update_bssid_list(struct wpa_ctrl *ctrl)
3962 size_t len = sizeof(buf);
3964 char *cmd = "BSS RANGE=ALL MASK=0x2";
3969 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len, NULL);
3976 pos = os_strstr(pos, "bssid=");
3980 end = os_strchr(pos, '\n');
3984 cli_txt_list_add(&bsses, pos);
3990 static void update_ifnames(struct wpa_ctrl *ctrl)
3993 size_t len = sizeof(buf);
3995 char *cmd = "INTERFACES";
3999 cli_txt_list_flush(&ifnames);
4003 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len, NULL);
4010 end = os_strchr(pos, '\n');
4014 ret = os_snprintf(txt, sizeof(txt), "ifname=%s", pos);
4015 if (!os_snprintf_error(sizeof(txt), ret))
4016 cli_txt_list_add(&ifnames, txt);
4022 static void update_networks(struct wpa_ctrl *ctrl)
4025 size_t len = sizeof(buf);
4027 char *cmd = "LIST_NETWORKS";
4031 cli_txt_list_flush(&networks);
4035 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len, NULL);
4042 end = os_strchr(pos, '\n');
4047 cli_txt_list_add_word(&networks, pos, '\t');
4054 static void try_connection(void *eloop_ctx, void *timeout_ctx)
4059 if (ctrl_ifname == NULL)
4060 ctrl_ifname = wpa_cli_get_default_ifname();
4062 if (!wpa_cli_open_connection(ctrl_ifname, 1) == 0) {
4063 if (!warning_displayed) {
4064 printf("Could not connect to wpa_supplicant: "
4066 ctrl_ifname ? ctrl_ifname : "(nil)");
4067 warning_displayed = 1;
4069 eloop_register_timeout(1, 0, try_connection, NULL, NULL);
4073 update_bssid_list(ctrl_conn);
4074 update_networks(ctrl_conn);
4076 if (warning_displayed)
4077 printf("Connection established.\n");
4084 static void wpa_cli_interactive(void)
4086 printf("\nInteractive mode\n\n");
4088 eloop_register_timeout(0, 0, try_connection, NULL, NULL);
4090 eloop_cancel_timeout(try_connection, NULL, NULL);
4092 cli_txt_list_flush(&p2p_peers);
4093 cli_txt_list_flush(&p2p_groups);
4094 cli_txt_list_flush(&bsses);
4095 cli_txt_list_flush(&ifnames);
4096 cli_txt_list_flush(&networks);
4098 edit_deinit(hfile, wpa_cli_edit_filter_history_cb);
4100 eloop_cancel_timeout(wpa_cli_ping, NULL, NULL);
4101 wpa_cli_close_connection();
4105 static void wpa_cli_action_ping(void *eloop_ctx, void *timeout_ctx)
4107 struct wpa_ctrl *ctrl = eloop_ctx;
4111 /* verify that connection is still working */
4112 len = sizeof(buf) - 1;
4113 if (wpa_ctrl_request(ctrl, "PING", 4, buf, &len,
4114 wpa_cli_action_cb) < 0 ||
4115 len < 4 || os_memcmp(buf, "PONG", 4) != 0) {
4116 printf("wpa_supplicant did not reply to PING command - exiting\n");
4120 eloop_register_timeout(ping_interval, 0, wpa_cli_action_ping,
4125 static void wpa_cli_action_receive(int sock, void *eloop_ctx, void *sock_ctx)
4127 struct wpa_ctrl *ctrl = eloop_ctx;
4129 wpa_cli_recv_pending(ctrl, 1);
4133 static void wpa_cli_action(struct wpa_ctrl *ctrl)
4135 #ifdef CONFIG_ANSI_C_EXTRA
4136 /* TODO: ANSI C version(?) */
4137 printf("Action processing not supported in ANSI C build.\n");
4138 #else /* CONFIG_ANSI_C_EXTRA */
4141 fd = wpa_ctrl_get_fd(ctrl);
4142 eloop_register_timeout(ping_interval, 0, wpa_cli_action_ping,
4144 eloop_register_read_sock(fd, wpa_cli_action_receive, ctrl, NULL);
4146 eloop_cancel_timeout(wpa_cli_action_ping, ctrl, NULL);
4147 eloop_unregister_read_sock(fd);
4148 #endif /* CONFIG_ANSI_C_EXTRA */
4152 static void wpa_cli_cleanup(void)
4154 wpa_cli_close_connection();
4156 os_daemonize_terminate(pid_file);
4158 os_program_deinit();
4162 static void wpa_cli_terminate(int sig, void *ctx)
4168 static char * wpa_cli_get_default_ifname(void)
4170 char *ifname = NULL;
4172 #ifdef CONFIG_CTRL_IFACE_UNIX
4173 struct dirent *dent;
4174 DIR *dir = opendir(ctrl_iface_dir);
4177 char ifprop[PROPERTY_VALUE_MAX];
4178 if (property_get("wifi.interface", ifprop, NULL) != 0) {
4179 ifname = os_strdup(ifprop);
4180 printf("Using interface '%s'\n", ifname);
4183 #endif /* ANDROID */
4186 while ((dent = readdir(dir))) {
4187 #ifdef _DIRENT_HAVE_D_TYPE
4189 * Skip the file if it is not a socket. Also accept
4190 * DT_UNKNOWN (0) in case the C library or underlying
4191 * file system does not support d_type.
4193 if (dent->d_type != DT_SOCK && dent->d_type != DT_UNKNOWN)
4195 #endif /* _DIRENT_HAVE_D_TYPE */
4196 if (os_strcmp(dent->d_name, ".") == 0 ||
4197 os_strcmp(dent->d_name, "..") == 0)
4199 printf("Selected interface '%s'\n", dent->d_name);
4200 ifname = os_strdup(dent->d_name);
4204 #endif /* CONFIG_CTRL_IFACE_UNIX */
4206 #ifdef CONFIG_CTRL_IFACE_NAMED_PIPE
4207 char buf[4096], *pos;
4209 struct wpa_ctrl *ctrl;
4212 ctrl = wpa_ctrl_open(NULL);
4216 len = sizeof(buf) - 1;
4217 ret = wpa_ctrl_request(ctrl, "INTERFACES", 10, buf, &len, NULL);
4220 pos = os_strchr(buf, '\n');
4223 ifname = os_strdup(buf);
4225 wpa_ctrl_close(ctrl);
4226 #endif /* CONFIG_CTRL_IFACE_NAMED_PIPE */
4232 int main(int argc, char *argv[])
4237 const char *global = NULL;
4239 if (os_program_init())
4243 c = getopt(argc, argv, "a:Bg:G:hi:p:P:v");
4248 action_file = optarg;
4257 ping_interval = atoi(optarg);
4263 printf("%s\n", wpa_cli_version);
4266 os_free(ctrl_ifname);
4267 ctrl_ifname = os_strdup(optarg);
4270 ctrl_iface_dir = optarg;
4281 interactive = (argc == optind) && (action_file == NULL);
4284 printf("%s\n\n%s\n\n", wpa_cli_version, wpa_cli_license);
4290 #ifdef CONFIG_CTRL_IFACE_NAMED_PIPE
4291 ctrl_conn = wpa_ctrl_open(NULL);
4292 #else /* CONFIG_CTRL_IFACE_NAMED_PIPE */
4293 ctrl_conn = wpa_ctrl_open(global);
4294 #endif /* CONFIG_CTRL_IFACE_NAMED_PIPE */
4295 if (ctrl_conn == NULL) {
4296 fprintf(stderr, "Failed to connect to wpa_supplicant "
4297 "global interface: %s error: %s\n",
4298 global, strerror(errno));
4303 update_ifnames(ctrl_conn);
4304 mon_conn = wpa_ctrl_open(global);
4306 if (wpa_ctrl_attach(mon_conn) == 0) {
4307 wpa_cli_attached = 1;
4308 eloop_register_read_sock(
4309 wpa_ctrl_get_fd(mon_conn),
4310 wpa_cli_mon_receive,
4313 printf("Failed to open monitor "
4314 "connection through global "
4315 "control interface\n");
4321 eloop_register_signal_terminate(wpa_cli_terminate, NULL);
4323 if (ctrl_ifname == NULL)
4324 ctrl_ifname = wpa_cli_get_default_ifname();
4327 wpa_cli_interactive();
4330 wpa_cli_open_connection(ctrl_ifname, 0) < 0) {
4331 fprintf(stderr, "Failed to connect to non-global "
4332 "ctrl_ifname: %s error: %s\n",
4333 ctrl_ifname ? ctrl_ifname : "(nil)",
4339 if (wpa_ctrl_attach(ctrl_conn) == 0) {
4340 wpa_cli_attached = 1;
4342 printf("Warning: Failed to attach to "
4343 "wpa_supplicant.\n");
4348 if (daemonize && os_daemonize(pid_file))
4352 wpa_cli_action(ctrl_conn);
4354 ret = wpa_request(ctrl_conn, argc - optind,
4358 os_free(ctrl_ifname);
4365 #else /* CONFIG_CTRL_IFACE */
4366 int main(int argc, char *argv[])
4368 printf("CONFIG_CTRL_IFACE not defined - wpa_cli disabled\n");
4371 #endif /* CONFIG_CTRL_IFACE */