]> git.ipfire.org Git - thirdparty/hostap.git/blob - wpa_supplicant/wpa_cli.c
wpa_cli: Fix NULL dereference on printf string argument
[thirdparty/hostap.git] / wpa_supplicant / wpa_cli.c
1 /*
2 * WPA Supplicant - command line interface for wpa_supplicant daemon
3 * Copyright (c) 2004-2013, Jouni Malinen <j@w1.fi>
4 *
5 * This software may be distributed under the terms of the BSD license.
6 * See README for more details.
7 */
8
9 #include "includes.h"
10
11 #ifdef CONFIG_CTRL_IFACE
12
13 #ifdef CONFIG_CTRL_IFACE_UNIX
14 #include <dirent.h>
15 #endif /* CONFIG_CTRL_IFACE_UNIX */
16
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"
24 #ifdef ANDROID
25 #include <cutils/properties.h>
26 #endif /* ANDROID */
27
28
29 static const char *wpa_cli_version =
30 "wpa_cli v" VERSION_STR "\n"
31 "Copyright (c) 2004-2014, Jouni Malinen <j@w1.fi> and contributors";
32
33
34 static const char *wpa_cli_license =
35 "This software may be distributed under the terms of the BSD license.\n"
36 "See README for more details.\n";
37
38 static const char *wpa_cli_full_license =
39 "This software may be distributed under the terms of the BSD license.\n"
40 "\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"
43 "met:\n"
44 "\n"
45 "1. Redistributions of source code must retain the above copyright\n"
46 " notice, this list of conditions and the following disclaimer.\n"
47 "\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"
51 "\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"
55 "\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"
67 "\n";
68
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;
85
86 struct cli_txt_entry {
87 struct dl_list list;
88 char *txt;
89 };
90
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
96
97 static void print_help(const char *cmd);
98 static void wpa_cli_mon_receive(int sock, void *eloop_ctx, void *sock_ctx);
99 static void wpa_cli_close_connection(void);
100 static char * wpa_cli_get_default_ifname(void);
101 static char ** wpa_list_cmd_list(void);
102
103
104 static void usage(void)
105 {
106 printf("wpa_cli [-p<path to ctrl sockets>] [-i<ifname>] [-hvB] "
107 "[-a<action file>] \\\n"
108 " [-P<pid file>] [-g<global ctrl>] [-G<ping interval>] "
109 "[command..]\n"
110 " -h = help (show this usage text)\n"
111 " -v = shown version information\n"
112 " -a = run in daemon mode executing the action file based on "
113 "events from\n"
114 " wpa_supplicant\n"
115 " -B = run a daemon in the background\n"
116 " default path: " CONFIG_CTRL_IFACE_DIR "\n"
117 " default interface: first interface found in socket path\n");
118 print_help(NULL);
119 }
120
121
122 static void cli_txt_list_free(struct cli_txt_entry *e)
123 {
124 dl_list_del(&e->list);
125 os_free(e->txt);
126 os_free(e);
127 }
128
129
130 static void cli_txt_list_flush(struct dl_list *list)
131 {
132 struct cli_txt_entry *e;
133 while ((e = dl_list_first(list, struct cli_txt_entry, list)))
134 cli_txt_list_free(e);
135 }
136
137
138 static struct cli_txt_entry * cli_txt_list_get(struct dl_list *txt_list,
139 const char *txt)
140 {
141 struct cli_txt_entry *e;
142 dl_list_for_each(e, txt_list, struct cli_txt_entry, list) {
143 if (os_strcmp(e->txt, txt) == 0)
144 return e;
145 }
146 return NULL;
147 }
148
149
150 static void cli_txt_list_del(struct dl_list *txt_list, const char *txt)
151 {
152 struct cli_txt_entry *e;
153 e = cli_txt_list_get(txt_list, txt);
154 if (e)
155 cli_txt_list_free(e);
156 }
157
158
159 static void cli_txt_list_del_addr(struct dl_list *txt_list, const char *txt)
160 {
161 u8 addr[ETH_ALEN];
162 char buf[18];
163 if (hwaddr_aton(txt, addr) < 0)
164 return;
165 os_snprintf(buf, sizeof(buf), MACSTR, MAC2STR(addr));
166 cli_txt_list_del(txt_list, buf);
167 }
168
169
170 #ifdef CONFIG_P2P
171 static void cli_txt_list_del_word(struct dl_list *txt_list, const char *txt)
172 {
173 const char *end;
174 char *buf;
175 end = os_strchr(txt, ' ');
176 if (end == NULL)
177 end = txt + os_strlen(txt);
178 buf = dup_binstr(txt, end - txt);
179 if (buf == NULL)
180 return;
181 cli_txt_list_del(txt_list, buf);
182 os_free(buf);
183 }
184 #endif /* CONFIG_P2P */
185
186
187 static int cli_txt_list_add(struct dl_list *txt_list, const char *txt)
188 {
189 struct cli_txt_entry *e;
190 e = cli_txt_list_get(txt_list, txt);
191 if (e)
192 return 0;
193 e = os_zalloc(sizeof(*e));
194 if (e == NULL)
195 return -1;
196 e->txt = os_strdup(txt);
197 if (e->txt == NULL) {
198 os_free(e);
199 return -1;
200 }
201 dl_list_add(txt_list, &e->list);
202 return 0;
203 }
204
205
206 #ifdef CONFIG_P2P
207 static int cli_txt_list_add_addr(struct dl_list *txt_list, const char *txt)
208 {
209 u8 addr[ETH_ALEN];
210 char buf[18];
211 if (hwaddr_aton(txt, addr) < 0)
212 return -1;
213 os_snprintf(buf, sizeof(buf), MACSTR, MAC2STR(addr));
214 return cli_txt_list_add(txt_list, buf);
215 }
216
217
218 static int cli_txt_list_add_word(struct dl_list *txt_list, const char *txt)
219 {
220 const char *end;
221 char *buf;
222 int ret;
223 end = os_strchr(txt, ' ');
224 if (end == NULL)
225 end = txt + os_strlen(txt);
226 buf = dup_binstr(txt, end - txt);
227 if (buf == NULL)
228 return -1;
229 ret = cli_txt_list_add(txt_list, buf);
230 os_free(buf);
231 return ret;
232 }
233 #endif /* CONFIG_P2P */
234
235
236 static char ** cli_txt_list_array(struct dl_list *txt_list)
237 {
238 unsigned int i, count = dl_list_len(txt_list);
239 char **res;
240 struct cli_txt_entry *e;
241
242 res = os_calloc(count + 1, sizeof(char *));
243 if (res == NULL)
244 return NULL;
245
246 i = 0;
247 dl_list_for_each(e, txt_list, struct cli_txt_entry, list) {
248 res[i] = os_strdup(e->txt);
249 if (res[i] == NULL)
250 break;
251 i++;
252 }
253
254 return res;
255 }
256
257
258 static int get_cmd_arg_num(const char *str, int pos)
259 {
260 int arg = 0, i;
261
262 for (i = 0; i <= pos; i++) {
263 if (str[i] != ' ') {
264 arg++;
265 while (i <= pos && str[i] != ' ')
266 i++;
267 }
268 }
269
270 if (arg > 0)
271 arg--;
272 return arg;
273 }
274
275
276 static int str_starts(const char *src, const char *match)
277 {
278 return os_strncmp(src, match, os_strlen(match)) == 0;
279 }
280
281
282 static int wpa_cli_show_event(const char *event)
283 {
284 const char *start;
285
286 start = os_strchr(event, '>');
287 if (start == NULL)
288 return 1;
289
290 start++;
291 /*
292 * Skip BSS added/removed events since they can be relatively frequent
293 * and are likely of not much use for an interactive user.
294 */
295 if (str_starts(start, WPA_EVENT_BSS_ADDED) ||
296 str_starts(start, WPA_EVENT_BSS_REMOVED))
297 return 0;
298
299 return 1;
300 }
301
302
303 static int wpa_cli_open_connection(const char *ifname, int attach)
304 {
305 #if defined(CONFIG_CTRL_IFACE_UDP) || defined(CONFIG_CTRL_IFACE_NAMED_PIPE)
306 ctrl_conn = wpa_ctrl_open(ifname);
307 if (ctrl_conn == NULL)
308 return -1;
309
310 if (attach && interactive)
311 mon_conn = wpa_ctrl_open(ifname);
312 else
313 mon_conn = NULL;
314 #else /* CONFIG_CTRL_IFACE_UDP || CONFIG_CTRL_IFACE_NAMED_PIPE */
315 char *cfile = NULL;
316 int flen, res;
317
318 if (ifname == NULL)
319 return -1;
320
321 #ifdef ANDROID
322 if (access(ctrl_iface_dir, F_OK) < 0) {
323 cfile = os_strdup(ifname);
324 if (cfile == NULL)
325 return -1;
326 }
327 #endif /* ANDROID */
328
329 if (cfile == NULL) {
330 flen = os_strlen(ctrl_iface_dir) + os_strlen(ifname) + 2;
331 cfile = os_malloc(flen);
332 if (cfile == NULL)
333 return -1;
334 res = os_snprintf(cfile, flen, "%s/%s", ctrl_iface_dir,
335 ifname);
336 if (os_snprintf_error(flen, res)) {
337 os_free(cfile);
338 return -1;
339 }
340 }
341
342 ctrl_conn = wpa_ctrl_open(cfile);
343 if (ctrl_conn == NULL) {
344 os_free(cfile);
345 return -1;
346 }
347
348 if (attach && interactive)
349 mon_conn = wpa_ctrl_open(cfile);
350 else
351 mon_conn = NULL;
352 os_free(cfile);
353 #endif /* CONFIG_CTRL_IFACE_UDP || CONFIG_CTRL_IFACE_NAMED_PIPE */
354
355 if (mon_conn) {
356 if (wpa_ctrl_attach(mon_conn) == 0) {
357 wpa_cli_attached = 1;
358 if (interactive)
359 eloop_register_read_sock(
360 wpa_ctrl_get_fd(mon_conn),
361 wpa_cli_mon_receive, NULL, NULL);
362 } else {
363 printf("Warning: Failed to attach to "
364 "wpa_supplicant.\n");
365 wpa_cli_close_connection();
366 return -1;
367 }
368 }
369
370 return 0;
371 }
372
373
374 static void wpa_cli_close_connection(void)
375 {
376 if (ctrl_conn == NULL)
377 return;
378
379 if (wpa_cli_attached) {
380 wpa_ctrl_detach(interactive ? mon_conn : ctrl_conn);
381 wpa_cli_attached = 0;
382 }
383 wpa_ctrl_close(ctrl_conn);
384 ctrl_conn = NULL;
385 if (mon_conn) {
386 eloop_unregister_read_sock(wpa_ctrl_get_fd(mon_conn));
387 wpa_ctrl_close(mon_conn);
388 mon_conn = NULL;
389 }
390 }
391
392
393 static void wpa_cli_msg_cb(char *msg, size_t len)
394 {
395 printf("%s\n", msg);
396 }
397
398
399 static int _wpa_ctrl_command(struct wpa_ctrl *ctrl, char *cmd, int print)
400 {
401 char buf[4096];
402 size_t len;
403 int ret;
404
405 if (ctrl_conn == NULL) {
406 printf("Not connected to wpa_supplicant - command dropped.\n");
407 return -1;
408 }
409 if (ifname_prefix) {
410 os_snprintf(buf, sizeof(buf), "IFNAME=%s %s",
411 ifname_prefix, cmd);
412 buf[sizeof(buf) - 1] = '\0';
413 cmd = buf;
414 }
415 len = sizeof(buf) - 1;
416 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len,
417 wpa_cli_msg_cb);
418 if (ret == -2) {
419 printf("'%s' command timed out.\n", cmd);
420 return -2;
421 } else if (ret < 0) {
422 printf("'%s' command failed.\n", cmd);
423 return -1;
424 }
425 if (print) {
426 buf[len] = '\0';
427 printf("%s", buf);
428 if (interactive && len > 0 && buf[len - 1] != '\n')
429 printf("\n");
430 }
431 return 0;
432 }
433
434
435 static int wpa_ctrl_command(struct wpa_ctrl *ctrl, char *cmd)
436 {
437 return _wpa_ctrl_command(ctrl, cmd, 1);
438 }
439
440
441 static int write_cmd(char *buf, size_t buflen, const char *cmd, int argc,
442 char *argv[])
443 {
444 int i, res;
445 char *pos, *end;
446
447 pos = buf;
448 end = buf + buflen;
449
450 res = os_snprintf(pos, end - pos, "%s", cmd);
451 if (os_snprintf_error(end - pos, res))
452 goto fail;
453 pos += res;
454
455 for (i = 0; i < argc; i++) {
456 res = os_snprintf(pos, end - pos, " %s", argv[i]);
457 if (os_snprintf_error(end - pos, res))
458 goto fail;
459 pos += res;
460 }
461
462 buf[buflen - 1] = '\0';
463 return 0;
464
465 fail:
466 printf("Too long command\n");
467 return -1;
468 }
469
470
471 static int wpa_cli_cmd(struct wpa_ctrl *ctrl, const char *cmd, int min_args,
472 int argc, char *argv[])
473 {
474 char buf[4096];
475 if (argc < min_args) {
476 printf("Invalid %s command - at least %d argument%s "
477 "required.\n", cmd, min_args,
478 min_args > 1 ? "s are" : " is");
479 return -1;
480 }
481 if (write_cmd(buf, sizeof(buf), cmd, argc, argv) < 0)
482 return -1;
483 return wpa_ctrl_command(ctrl, buf);
484 }
485
486
487 static int wpa_cli_cmd_ifname(struct wpa_ctrl *ctrl, int argc, char *argv[])
488 {
489 return wpa_ctrl_command(ctrl, "IFNAME");
490 }
491
492
493 static int wpa_cli_cmd_status(struct wpa_ctrl *ctrl, int argc, char *argv[])
494 {
495 if (argc > 0 && os_strcmp(argv[0], "verbose") == 0)
496 return wpa_ctrl_command(ctrl, "STATUS-VERBOSE");
497 if (argc > 0 && os_strcmp(argv[0], "wps") == 0)
498 return wpa_ctrl_command(ctrl, "STATUS-WPS");
499 if (argc > 0 && os_strcmp(argv[0], "driver") == 0)
500 return wpa_ctrl_command(ctrl, "STATUS-DRIVER");
501 return wpa_ctrl_command(ctrl, "STATUS");
502 }
503
504
505 static int wpa_cli_cmd_ping(struct wpa_ctrl *ctrl, int argc, char *argv[])
506 {
507 return wpa_ctrl_command(ctrl, "PING");
508 }
509
510
511 static int wpa_cli_cmd_relog(struct wpa_ctrl *ctrl, int argc, char *argv[])
512 {
513 return wpa_ctrl_command(ctrl, "RELOG");
514 }
515
516
517 static int wpa_cli_cmd_note(struct wpa_ctrl *ctrl, int argc, char *argv[])
518 {
519 return wpa_cli_cmd(ctrl, "NOTE", 1, argc, argv);
520 }
521
522
523 static int wpa_cli_cmd_mib(struct wpa_ctrl *ctrl, int argc, char *argv[])
524 {
525 return wpa_ctrl_command(ctrl, "MIB");
526 }
527
528
529 static int wpa_cli_cmd_pmksa(struct wpa_ctrl *ctrl, int argc, char *argv[])
530 {
531 return wpa_ctrl_command(ctrl, "PMKSA");
532 }
533
534
535 static int wpa_cli_cmd_pmksa_flush(struct wpa_ctrl *ctrl, int argc,
536 char *argv[])
537 {
538 return wpa_ctrl_command(ctrl, "PMKSA_FLUSH");
539 }
540
541
542 static int wpa_cli_cmd_help(struct wpa_ctrl *ctrl, int argc, char *argv[])
543 {
544 print_help(argc > 0 ? argv[0] : NULL);
545 return 0;
546 }
547
548
549 static char ** wpa_cli_complete_help(const char *str, int pos)
550 {
551 int arg = get_cmd_arg_num(str, pos);
552 char **res = NULL;
553
554 switch (arg) {
555 case 1:
556 res = wpa_list_cmd_list();
557 break;
558 }
559
560 return res;
561 }
562
563
564 static int wpa_cli_cmd_license(struct wpa_ctrl *ctrl, int argc, char *argv[])
565 {
566 printf("%s\n\n%s\n", wpa_cli_version, wpa_cli_full_license);
567 return 0;
568 }
569
570
571 static int wpa_cli_cmd_quit(struct wpa_ctrl *ctrl, int argc, char *argv[])
572 {
573 wpa_cli_quit = 1;
574 if (interactive)
575 eloop_terminate();
576 return 0;
577 }
578
579
580 static int wpa_cli_cmd_set(struct wpa_ctrl *ctrl, int argc, char *argv[])
581 {
582 char cmd[256];
583 int res;
584
585 if (argc == 1) {
586 res = os_snprintf(cmd, sizeof(cmd), "SET %s ", argv[0]);
587 if (os_snprintf_error(sizeof(cmd), res)) {
588 printf("Too long SET command.\n");
589 return -1;
590 }
591 return wpa_ctrl_command(ctrl, cmd);
592 }
593
594 return wpa_cli_cmd(ctrl, "SET", 2, argc, argv);
595 }
596
597
598 static char ** wpa_cli_complete_set(const char *str, int pos)
599 {
600 int arg = get_cmd_arg_num(str, pos);
601 const char *fields[] = {
602 /* runtime values */
603 "EAPOL::heldPeriod", "EAPOL::authPeriod", "EAPOL::startPeriod",
604 "EAPOL::maxStart", "dot11RSNAConfigPMKLifetime",
605 "dot11RSNAConfigPMKReauthThreshold", "dot11RSNAConfigSATimeout",
606 "wps_fragment_size", "wps_version_number", "ampdu",
607 "tdls_testing", "tdls_disabled", "pno", "radio_disabled",
608 "uapsd", "ps", "wifi_display", "bssid_filter", "disallow_aps",
609 "no_keep_alive",
610 /* global configuration parameters */
611 "eapol_version", "ap_scan", "disable_scan_offload",
612 "fast_reauth", "opensc_engine_path", "pkcs11_engine_path",
613 "pkcs11_module_path", "openssl_ciphers",
614 "pcsc_reader", "pcsc_pin",
615 "driver_param", "dot11RSNAConfigPMKLifetime",
616 "dot11RSNAConfigPMKReauthThreshold",
617 "dot11RSNAConfigSATimeout",
618 "update_config", "load_dynamic_eap", "uuid", "device_name",
619 "manufacturer", "model_name", "model_number", "serial_number",
620 "device_type", "os_version", "config_methods",
621 "wps_cred_processing", "wps_vendor_ext_m1", "sec_device_type",
622 "p2p_listen_reg_class", "p2p_listen_channel",
623 "p2p_oper_reg_class", "p2p_oper_channel",
624 "p2p_go_intent", "p2p_ssid_postfix", "persistent_reconnect",
625 "p2p_intra_bss", "p2p_group_idle", "p2p_pref_chan",
626 "p2p_no_go_freq",
627 "p2p_go_ht40", "p2p_disabled", "p2p_no_group_iface",
628 "p2p_go_vht",
629 "p2p_ignore_shared_freq", "country", "bss_max_count",
630 "bss_expiration_age", "bss_expiration_scan_count",
631 "filter_ssids", "filter_rssi", "max_num_sta",
632 "disassoc_low_ack", "hs20", "interworking", "hessid",
633 "access_network_type", "pbc_in_m1", "autoscan",
634 "wps_nfc_dev_pw_id", "wps_nfc_dh_pubkey", "wps_nfc_dh_privkey",
635 "wps_nfc_dev_pw", "ext_password_backend",
636 "p2p_go_max_inactivity", "auto_interworking", "okc", "pmf",
637 "sae_groups", "dtim_period", "beacon_int", "ap_vendor_elements",
638 "ignore_old_scan_res", "freq_list", "external_sim",
639 "tdls_external_control", "p2p_search_delay"
640 };
641 int i, num_fields = ARRAY_SIZE(fields);
642
643 if (arg == 1) {
644 char **res = os_calloc(num_fields + 1, sizeof(char *));
645 if (res == NULL)
646 return NULL;
647 for (i = 0; i < num_fields; i++) {
648 res[i] = os_strdup(fields[i]);
649 if (res[i] == NULL)
650 return res;
651 }
652 return res;
653 }
654
655 if (arg > 1 && os_strncasecmp(str, "set bssid_filter ", 17) == 0)
656 return cli_txt_list_array(&bsses);
657
658 return NULL;
659 }
660
661
662 static int wpa_cli_cmd_get(struct wpa_ctrl *ctrl, int argc, char *argv[])
663 {
664 return wpa_cli_cmd(ctrl, "GET", 1, argc, argv);
665 }
666
667
668 static int wpa_cli_cmd_logoff(struct wpa_ctrl *ctrl, int argc, char *argv[])
669 {
670 return wpa_ctrl_command(ctrl, "LOGOFF");
671 }
672
673
674 static int wpa_cli_cmd_logon(struct wpa_ctrl *ctrl, int argc, char *argv[])
675 {
676 return wpa_ctrl_command(ctrl, "LOGON");
677 }
678
679
680 static int wpa_cli_cmd_reassociate(struct wpa_ctrl *ctrl, int argc,
681 char *argv[])
682 {
683 return wpa_ctrl_command(ctrl, "REASSOCIATE");
684 }
685
686
687 static int wpa_cli_cmd_reattach(struct wpa_ctrl *ctrl, int argc, char *argv[])
688 {
689 return wpa_ctrl_command(ctrl, "REATTACH");
690 }
691
692
693 static int wpa_cli_cmd_preauthenticate(struct wpa_ctrl *ctrl, int argc,
694 char *argv[])
695 {
696 return wpa_cli_cmd(ctrl, "PREAUTH", 1, argc, argv);
697 }
698
699
700 static int wpa_cli_cmd_ap_scan(struct wpa_ctrl *ctrl, int argc, char *argv[])
701 {
702 return wpa_cli_cmd(ctrl, "AP_SCAN", 1, argc, argv);
703 }
704
705
706 static int wpa_cli_cmd_scan_interval(struct wpa_ctrl *ctrl, int argc,
707 char *argv[])
708 {
709 return wpa_cli_cmd(ctrl, "SCAN_INTERVAL", 1, argc, argv);
710 }
711
712
713 static int wpa_cli_cmd_bss_expire_age(struct wpa_ctrl *ctrl, int argc,
714 char *argv[])
715 {
716 return wpa_cli_cmd(ctrl, "BSS_EXPIRE_AGE", 1, argc, argv);
717 }
718
719
720 static int wpa_cli_cmd_bss_expire_count(struct wpa_ctrl *ctrl, int argc,
721 char *argv[])
722 {
723 return wpa_cli_cmd(ctrl, "BSS_EXPIRE_COUNT", 1, argc, argv);
724 }
725
726
727 static int wpa_cli_cmd_bss_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
728 {
729 char cmd[256];
730 int res;
731
732 if (argc < 1)
733 res = os_snprintf(cmd, sizeof(cmd), "BSS_FLUSH 0");
734 else
735 res = os_snprintf(cmd, sizeof(cmd), "BSS_FLUSH %s", argv[0]);
736 if (os_snprintf_error(sizeof(cmd), res)) {
737 printf("Too long BSS_FLUSH command.\n");
738 return -1;
739 }
740 return wpa_ctrl_command(ctrl, cmd);
741 }
742
743
744 static int wpa_cli_cmd_stkstart(struct wpa_ctrl *ctrl, int argc,
745 char *argv[])
746 {
747 return wpa_cli_cmd(ctrl, "STKSTART", 1, argc, argv);
748 }
749
750
751 static int wpa_cli_cmd_ft_ds(struct wpa_ctrl *ctrl, int argc, char *argv[])
752 {
753 return wpa_cli_cmd(ctrl, "FT_DS", 1, argc, argv);
754 }
755
756
757 static int wpa_cli_cmd_wps_pbc(struct wpa_ctrl *ctrl, int argc, char *argv[])
758 {
759 return wpa_cli_cmd(ctrl, "WPS_PBC", 0, argc, argv);
760 }
761
762
763 static int wpa_cli_cmd_wps_pin(struct wpa_ctrl *ctrl, int argc, char *argv[])
764 {
765 if (argc == 0) {
766 printf("Invalid WPS_PIN command: need one or two arguments:\n"
767 "- BSSID: use 'any' to select any\n"
768 "- PIN: optional, used only with devices that have no "
769 "display\n");
770 return -1;
771 }
772
773 return wpa_cli_cmd(ctrl, "WPS_PIN", 1, argc, argv);
774 }
775
776
777 static int wpa_cli_cmd_wps_check_pin(struct wpa_ctrl *ctrl, int argc,
778 char *argv[])
779 {
780 return wpa_cli_cmd(ctrl, "WPS_CHECK_PIN", 1, argc, argv);
781 }
782
783
784 static int wpa_cli_cmd_wps_cancel(struct wpa_ctrl *ctrl, int argc,
785 char *argv[])
786 {
787 return wpa_ctrl_command(ctrl, "WPS_CANCEL");
788 }
789
790
791 #ifdef CONFIG_WPS_NFC
792
793 static int wpa_cli_cmd_wps_nfc(struct wpa_ctrl *ctrl, int argc, char *argv[])
794 {
795 return wpa_cli_cmd(ctrl, "WPS_NFC", 0, argc, argv);
796 }
797
798
799 static int wpa_cli_cmd_wps_nfc_config_token(struct wpa_ctrl *ctrl, int argc,
800 char *argv[])
801 {
802 return wpa_cli_cmd(ctrl, "WPS_NFC_CONFIG_TOKEN", 1, argc, argv);
803 }
804
805
806 static int wpa_cli_cmd_wps_nfc_token(struct wpa_ctrl *ctrl, int argc,
807 char *argv[])
808 {
809 return wpa_cli_cmd(ctrl, "WPS_NFC_TOKEN", 1, argc, argv);
810 }
811
812
813 static int wpa_cli_cmd_wps_nfc_tag_read(struct wpa_ctrl *ctrl, int argc,
814 char *argv[])
815 {
816 int ret;
817 char *buf;
818 size_t buflen;
819
820 if (argc != 1) {
821 printf("Invalid 'wps_nfc_tag_read' command - one argument "
822 "is required.\n");
823 return -1;
824 }
825
826 buflen = 18 + os_strlen(argv[0]);
827 buf = os_malloc(buflen);
828 if (buf == NULL)
829 return -1;
830 os_snprintf(buf, buflen, "WPS_NFC_TAG_READ %s", argv[0]);
831
832 ret = wpa_ctrl_command(ctrl, buf);
833 os_free(buf);
834
835 return ret;
836 }
837
838
839 static int wpa_cli_cmd_nfc_get_handover_req(struct wpa_ctrl *ctrl, int argc,
840 char *argv[])
841 {
842 return wpa_cli_cmd(ctrl, "NFC_GET_HANDOVER_REQ", 2, argc, argv);
843 }
844
845
846 static int wpa_cli_cmd_nfc_get_handover_sel(struct wpa_ctrl *ctrl, int argc,
847 char *argv[])
848 {
849 return wpa_cli_cmd(ctrl, "NFC_GET_HANDOVER_SEL", 2, argc, argv);
850 }
851
852
853 static int wpa_cli_cmd_nfc_report_handover(struct wpa_ctrl *ctrl, int argc,
854 char *argv[])
855 {
856 return wpa_cli_cmd(ctrl, "NFC_REPORT_HANDOVER", 4, argc, argv);
857 }
858
859 #endif /* CONFIG_WPS_NFC */
860
861
862 static int wpa_cli_cmd_wps_reg(struct wpa_ctrl *ctrl, int argc, char *argv[])
863 {
864 char cmd[256];
865 int res;
866
867 if (argc == 2)
868 res = os_snprintf(cmd, sizeof(cmd), "WPS_REG %s %s",
869 argv[0], argv[1]);
870 else if (argc == 5 || argc == 6) {
871 char ssid_hex[2 * 32 + 1];
872 char key_hex[2 * 64 + 1];
873 int i;
874
875 ssid_hex[0] = '\0';
876 for (i = 0; i < 32; i++) {
877 if (argv[2][i] == '\0')
878 break;
879 os_snprintf(&ssid_hex[i * 2], 3, "%02x", argv[2][i]);
880 }
881
882 key_hex[0] = '\0';
883 if (argc == 6) {
884 for (i = 0; i < 64; i++) {
885 if (argv[5][i] == '\0')
886 break;
887 os_snprintf(&key_hex[i * 2], 3, "%02x",
888 argv[5][i]);
889 }
890 }
891
892 res = os_snprintf(cmd, sizeof(cmd),
893 "WPS_REG %s %s %s %s %s %s",
894 argv[0], argv[1], ssid_hex, argv[3], argv[4],
895 key_hex);
896 } else {
897 printf("Invalid WPS_REG command: need two arguments:\n"
898 "- BSSID of the target AP\n"
899 "- AP PIN\n");
900 printf("Alternatively, six arguments can be used to "
901 "reconfigure the AP:\n"
902 "- BSSID of the target AP\n"
903 "- AP PIN\n"
904 "- new SSID\n"
905 "- new auth (OPEN, WPAPSK, WPA2PSK)\n"
906 "- new encr (NONE, WEP, TKIP, CCMP)\n"
907 "- new key\n");
908 return -1;
909 }
910
911 if (os_snprintf_error(sizeof(cmd), res)) {
912 printf("Too long WPS_REG command.\n");
913 return -1;
914 }
915 return wpa_ctrl_command(ctrl, cmd);
916 }
917
918
919 static int wpa_cli_cmd_wps_ap_pin(struct wpa_ctrl *ctrl, int argc,
920 char *argv[])
921 {
922 return wpa_cli_cmd(ctrl, "WPS_AP_PIN", 1, argc, argv);
923 }
924
925
926 static int wpa_cli_cmd_wps_er_start(struct wpa_ctrl *ctrl, int argc,
927 char *argv[])
928 {
929 return wpa_cli_cmd(ctrl, "WPS_ER_START", 0, argc, argv);
930 }
931
932
933 static int wpa_cli_cmd_wps_er_stop(struct wpa_ctrl *ctrl, int argc,
934 char *argv[])
935 {
936 return wpa_ctrl_command(ctrl, "WPS_ER_STOP");
937
938 }
939
940
941 static int wpa_cli_cmd_wps_er_pin(struct wpa_ctrl *ctrl, int argc,
942 char *argv[])
943 {
944 if (argc < 2) {
945 printf("Invalid WPS_ER_PIN command: need at least two "
946 "arguments:\n"
947 "- UUID: use 'any' to select any\n"
948 "- PIN: Enrollee PIN\n"
949 "optional: - Enrollee MAC address\n");
950 return -1;
951 }
952
953 return wpa_cli_cmd(ctrl, "WPS_ER_PIN", 2, argc, argv);
954 }
955
956
957 static int wpa_cli_cmd_wps_er_pbc(struct wpa_ctrl *ctrl, int argc,
958 char *argv[])
959 {
960 return wpa_cli_cmd(ctrl, "WPS_ER_PBC", 1, argc, argv);
961 }
962
963
964 static int wpa_cli_cmd_wps_er_learn(struct wpa_ctrl *ctrl, int argc,
965 char *argv[])
966 {
967 if (argc != 2) {
968 printf("Invalid WPS_ER_LEARN command: need two arguments:\n"
969 "- UUID: specify which AP to use\n"
970 "- PIN: AP PIN\n");
971 return -1;
972 }
973
974 return wpa_cli_cmd(ctrl, "WPS_ER_LEARN", 2, argc, argv);
975 }
976
977
978 static int wpa_cli_cmd_wps_er_set_config(struct wpa_ctrl *ctrl, int argc,
979 char *argv[])
980 {
981 if (argc != 2) {
982 printf("Invalid WPS_ER_SET_CONFIG command: need two "
983 "arguments:\n"
984 "- UUID: specify which AP to use\n"
985 "- Network configuration id\n");
986 return -1;
987 }
988
989 return wpa_cli_cmd(ctrl, "WPS_ER_SET_CONFIG", 2, argc, argv);
990 }
991
992
993 static int wpa_cli_cmd_wps_er_config(struct wpa_ctrl *ctrl, int argc,
994 char *argv[])
995 {
996 char cmd[256];
997 int res;
998
999 if (argc == 5 || argc == 6) {
1000 char ssid_hex[2 * 32 + 1];
1001 char key_hex[2 * 64 + 1];
1002 int i;
1003
1004 ssid_hex[0] = '\0';
1005 for (i = 0; i < 32; i++) {
1006 if (argv[2][i] == '\0')
1007 break;
1008 os_snprintf(&ssid_hex[i * 2], 3, "%02x", argv[2][i]);
1009 }
1010
1011 key_hex[0] = '\0';
1012 if (argc == 6) {
1013 for (i = 0; i < 64; i++) {
1014 if (argv[5][i] == '\0')
1015 break;
1016 os_snprintf(&key_hex[i * 2], 3, "%02x",
1017 argv[5][i]);
1018 }
1019 }
1020
1021 res = os_snprintf(cmd, sizeof(cmd),
1022 "WPS_ER_CONFIG %s %s %s %s %s %s",
1023 argv[0], argv[1], ssid_hex, argv[3], argv[4],
1024 key_hex);
1025 } else {
1026 printf("Invalid WPS_ER_CONFIG command: need six arguments:\n"
1027 "- AP UUID\n"
1028 "- AP PIN\n"
1029 "- new SSID\n"
1030 "- new auth (OPEN, WPAPSK, WPA2PSK)\n"
1031 "- new encr (NONE, WEP, TKIP, CCMP)\n"
1032 "- new key\n");
1033 return -1;
1034 }
1035
1036 if (os_snprintf_error(sizeof(cmd), res)) {
1037 printf("Too long WPS_ER_CONFIG command.\n");
1038 return -1;
1039 }
1040 return wpa_ctrl_command(ctrl, cmd);
1041 }
1042
1043
1044 #ifdef CONFIG_WPS_NFC
1045 static int wpa_cli_cmd_wps_er_nfc_config_token(struct wpa_ctrl *ctrl, int argc,
1046 char *argv[])
1047 {
1048 if (argc != 2) {
1049 printf("Invalid WPS_ER_NFC_CONFIG_TOKEN command: need two "
1050 "arguments:\n"
1051 "- WPS/NDEF: token format\n"
1052 "- UUID: specify which AP to use\n");
1053 return -1;
1054 }
1055
1056 return wpa_cli_cmd(ctrl, "WPS_ER_NFC_CONFIG_TOKEN", 2, argc, argv);
1057 }
1058 #endif /* CONFIG_WPS_NFC */
1059
1060
1061 static int wpa_cli_cmd_ibss_rsn(struct wpa_ctrl *ctrl, int argc, char *argv[])
1062 {
1063 return wpa_cli_cmd(ctrl, "IBSS_RSN", 1, argc, argv);
1064 }
1065
1066
1067 static int wpa_cli_cmd_level(struct wpa_ctrl *ctrl, int argc, char *argv[])
1068 {
1069 return wpa_cli_cmd(ctrl, "LEVEL", 1, argc, argv);
1070 }
1071
1072
1073 static int wpa_cli_cmd_identity(struct wpa_ctrl *ctrl, int argc, char *argv[])
1074 {
1075 char cmd[256], *pos, *end;
1076 int i, ret;
1077
1078 if (argc < 2) {
1079 printf("Invalid IDENTITY command: needs two arguments "
1080 "(network id and identity)\n");
1081 return -1;
1082 }
1083
1084 end = cmd + sizeof(cmd);
1085 pos = cmd;
1086 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "IDENTITY-%s:%s",
1087 argv[0], argv[1]);
1088 if (os_snprintf_error(end - pos, ret)) {
1089 printf("Too long IDENTITY command.\n");
1090 return -1;
1091 }
1092 pos += ret;
1093 for (i = 2; i < argc; i++) {
1094 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1095 if (os_snprintf_error(end - pos, ret)) {
1096 printf("Too long IDENTITY command.\n");
1097 return -1;
1098 }
1099 pos += ret;
1100 }
1101
1102 return wpa_ctrl_command(ctrl, cmd);
1103 }
1104
1105
1106 static int wpa_cli_cmd_password(struct wpa_ctrl *ctrl, int argc, char *argv[])
1107 {
1108 char cmd[256], *pos, *end;
1109 int i, ret;
1110
1111 if (argc < 2) {
1112 printf("Invalid PASSWORD command: needs two arguments "
1113 "(network id and password)\n");
1114 return -1;
1115 }
1116
1117 end = cmd + sizeof(cmd);
1118 pos = cmd;
1119 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "PASSWORD-%s:%s",
1120 argv[0], argv[1]);
1121 if (os_snprintf_error(end - pos, ret)) {
1122 printf("Too long PASSWORD command.\n");
1123 return -1;
1124 }
1125 pos += ret;
1126 for (i = 2; i < argc; i++) {
1127 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1128 if (os_snprintf_error(end - pos, ret)) {
1129 printf("Too long PASSWORD command.\n");
1130 return -1;
1131 }
1132 pos += ret;
1133 }
1134
1135 return wpa_ctrl_command(ctrl, cmd);
1136 }
1137
1138
1139 static int wpa_cli_cmd_new_password(struct wpa_ctrl *ctrl, int argc,
1140 char *argv[])
1141 {
1142 char cmd[256], *pos, *end;
1143 int i, ret;
1144
1145 if (argc < 2) {
1146 printf("Invalid NEW_PASSWORD command: needs two arguments "
1147 "(network id and password)\n");
1148 return -1;
1149 }
1150
1151 end = cmd + sizeof(cmd);
1152 pos = cmd;
1153 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "NEW_PASSWORD-%s:%s",
1154 argv[0], argv[1]);
1155 if (os_snprintf_error(end - pos, ret)) {
1156 printf("Too long NEW_PASSWORD command.\n");
1157 return -1;
1158 }
1159 pos += ret;
1160 for (i = 2; i < argc; i++) {
1161 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1162 if (os_snprintf_error(end - pos, ret)) {
1163 printf("Too long NEW_PASSWORD command.\n");
1164 return -1;
1165 }
1166 pos += ret;
1167 }
1168
1169 return wpa_ctrl_command(ctrl, cmd);
1170 }
1171
1172
1173 static int wpa_cli_cmd_pin(struct wpa_ctrl *ctrl, int argc, char *argv[])
1174 {
1175 char cmd[256], *pos, *end;
1176 int i, ret;
1177
1178 if (argc < 2) {
1179 printf("Invalid PIN command: needs two arguments "
1180 "(network id and pin)\n");
1181 return -1;
1182 }
1183
1184 end = cmd + sizeof(cmd);
1185 pos = cmd;
1186 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "PIN-%s:%s",
1187 argv[0], argv[1]);
1188 if (os_snprintf_error(end - pos, ret)) {
1189 printf("Too long PIN command.\n");
1190 return -1;
1191 }
1192 pos += ret;
1193 for (i = 2; i < argc; i++) {
1194 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1195 if (os_snprintf_error(end - pos, ret)) {
1196 printf("Too long PIN command.\n");
1197 return -1;
1198 }
1199 pos += ret;
1200 }
1201 return wpa_ctrl_command(ctrl, cmd);
1202 }
1203
1204
1205 static int wpa_cli_cmd_otp(struct wpa_ctrl *ctrl, int argc, char *argv[])
1206 {
1207 char cmd[256], *pos, *end;
1208 int i, ret;
1209
1210 if (argc < 2) {
1211 printf("Invalid OTP command: needs two arguments (network "
1212 "id and password)\n");
1213 return -1;
1214 }
1215
1216 end = cmd + sizeof(cmd);
1217 pos = cmd;
1218 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "OTP-%s:%s",
1219 argv[0], argv[1]);
1220 if (os_snprintf_error(end - pos, ret)) {
1221 printf("Too long OTP command.\n");
1222 return -1;
1223 }
1224 pos += ret;
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 OTP command.\n");
1229 return -1;
1230 }
1231 pos += ret;
1232 }
1233
1234 return wpa_ctrl_command(ctrl, cmd);
1235 }
1236
1237
1238 static int wpa_cli_cmd_sim(struct wpa_ctrl *ctrl, int argc, char *argv[])
1239 {
1240 char cmd[256], *pos, *end;
1241 int i, ret;
1242
1243 if (argc < 2) {
1244 printf("Invalid SIM command: needs two arguments "
1245 "(network id and SIM operation response)\n");
1246 return -1;
1247 }
1248
1249 end = cmd + sizeof(cmd);
1250 pos = cmd;
1251 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "SIM-%s:%s",
1252 argv[0], argv[1]);
1253 if (os_snprintf_error(end - pos, ret)) {
1254 printf("Too long SIM command.\n");
1255 return -1;
1256 }
1257 pos += ret;
1258 for (i = 2; i < argc; i++) {
1259 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1260 if (os_snprintf_error(end - pos, ret)) {
1261 printf("Too long SIM command.\n");
1262 return -1;
1263 }
1264 pos += ret;
1265 }
1266 return wpa_ctrl_command(ctrl, cmd);
1267 }
1268
1269
1270 static int wpa_cli_cmd_passphrase(struct wpa_ctrl *ctrl, int argc,
1271 char *argv[])
1272 {
1273 char cmd[256], *pos, *end;
1274 int i, ret;
1275
1276 if (argc < 2) {
1277 printf("Invalid PASSPHRASE command: needs two arguments "
1278 "(network id and passphrase)\n");
1279 return -1;
1280 }
1281
1282 end = cmd + sizeof(cmd);
1283 pos = cmd;
1284 ret = os_snprintf(pos, end - pos, WPA_CTRL_RSP "PASSPHRASE-%s:%s",
1285 argv[0], argv[1]);
1286 if (os_snprintf_error(end - pos, ret)) {
1287 printf("Too long PASSPHRASE command.\n");
1288 return -1;
1289 }
1290 pos += ret;
1291 for (i = 2; i < argc; i++) {
1292 ret = os_snprintf(pos, end - pos, " %s", argv[i]);
1293 if (os_snprintf_error(end - pos, ret)) {
1294 printf("Too long PASSPHRASE command.\n");
1295 return -1;
1296 }
1297 pos += ret;
1298 }
1299
1300 return wpa_ctrl_command(ctrl, cmd);
1301 }
1302
1303
1304 static int wpa_cli_cmd_bssid(struct wpa_ctrl *ctrl, int argc, char *argv[])
1305 {
1306 if (argc < 2) {
1307 printf("Invalid BSSID command: needs two arguments (network "
1308 "id and BSSID)\n");
1309 return -1;
1310 }
1311
1312 return wpa_cli_cmd(ctrl, "BSSID", 2, argc, argv);
1313 }
1314
1315
1316 static int wpa_cli_cmd_blacklist(struct wpa_ctrl *ctrl, int argc, char *argv[])
1317 {
1318 return wpa_cli_cmd(ctrl, "BLACKLIST", 0, argc, argv);
1319 }
1320
1321
1322 static int wpa_cli_cmd_log_level(struct wpa_ctrl *ctrl, int argc, char *argv[])
1323 {
1324 return wpa_cli_cmd(ctrl, "LOG_LEVEL", 0, argc, argv);
1325 }
1326
1327
1328 static int wpa_cli_cmd_list_networks(struct wpa_ctrl *ctrl, int argc,
1329 char *argv[])
1330 {
1331 return wpa_ctrl_command(ctrl, "LIST_NETWORKS");
1332 }
1333
1334
1335 static int wpa_cli_cmd_select_network(struct wpa_ctrl *ctrl, int argc,
1336 char *argv[])
1337 {
1338 return wpa_cli_cmd(ctrl, "SELECT_NETWORK", 1, argc, argv);
1339 }
1340
1341
1342 static int wpa_cli_cmd_enable_network(struct wpa_ctrl *ctrl, int argc,
1343 char *argv[])
1344 {
1345 return wpa_cli_cmd(ctrl, "ENABLE_NETWORK", 1, argc, argv);
1346 }
1347
1348
1349 static int wpa_cli_cmd_disable_network(struct wpa_ctrl *ctrl, int argc,
1350 char *argv[])
1351 {
1352 return wpa_cli_cmd(ctrl, "DISABLE_NETWORK", 1, argc, argv);
1353 }
1354
1355
1356 static int wpa_cli_cmd_add_network(struct wpa_ctrl *ctrl, int argc,
1357 char *argv[])
1358 {
1359 return wpa_ctrl_command(ctrl, "ADD_NETWORK");
1360 }
1361
1362
1363 static int wpa_cli_cmd_remove_network(struct wpa_ctrl *ctrl, int argc,
1364 char *argv[])
1365 {
1366 return wpa_cli_cmd(ctrl, "REMOVE_NETWORK", 1, argc, argv);
1367 }
1368
1369
1370 static void wpa_cli_show_network_variables(void)
1371 {
1372 printf("set_network variables:\n"
1373 " ssid (network name, SSID)\n"
1374 " psk (WPA passphrase or pre-shared key)\n"
1375 " key_mgmt (key management protocol)\n"
1376 " identity (EAP identity)\n"
1377 " password (EAP password)\n"
1378 " ...\n"
1379 "\n"
1380 "Note: Values are entered in the same format as the "
1381 "configuration file is using,\n"
1382 "i.e., strings values need to be inside double quotation "
1383 "marks.\n"
1384 "For example: set_network 1 ssid \"network name\"\n"
1385 "\n"
1386 "Please see wpa_supplicant.conf documentation for full list "
1387 "of\navailable variables.\n");
1388 }
1389
1390
1391 static int wpa_cli_cmd_set_network(struct wpa_ctrl *ctrl, int argc,
1392 char *argv[])
1393 {
1394 if (argc == 0) {
1395 wpa_cli_show_network_variables();
1396 return 0;
1397 }
1398
1399 if (argc < 3) {
1400 printf("Invalid SET_NETWORK command: needs three arguments\n"
1401 "(network id, variable name, and value)\n");
1402 return -1;
1403 }
1404
1405 return wpa_cli_cmd(ctrl, "SET_NETWORK", 3, argc, argv);
1406 }
1407
1408
1409 static int wpa_cli_cmd_get_network(struct wpa_ctrl *ctrl, int argc,
1410 char *argv[])
1411 {
1412 if (argc == 0) {
1413 wpa_cli_show_network_variables();
1414 return 0;
1415 }
1416
1417 if (argc != 2) {
1418 printf("Invalid GET_NETWORK command: needs two arguments\n"
1419 "(network id and variable name)\n");
1420 return -1;
1421 }
1422
1423 return wpa_cli_cmd(ctrl, "GET_NETWORK", 2, argc, argv);
1424 }
1425
1426
1427 static int wpa_cli_cmd_dup_network(struct wpa_ctrl *ctrl, int argc,
1428 char *argv[])
1429 {
1430 if (argc == 0) {
1431 wpa_cli_show_network_variables();
1432 return 0;
1433 }
1434
1435 if (argc < 3) {
1436 printf("Invalid DUP_NETWORK command: needs three arguments\n"
1437 "(src netid, dest netid, and variable name)\n");
1438 return -1;
1439 }
1440
1441 return wpa_cli_cmd(ctrl, "DUP_NETWORK", 3, argc, argv);
1442 }
1443
1444
1445 static int wpa_cli_cmd_list_creds(struct wpa_ctrl *ctrl, int argc,
1446 char *argv[])
1447 {
1448 return wpa_ctrl_command(ctrl, "LIST_CREDS");
1449 }
1450
1451
1452 static int wpa_cli_cmd_add_cred(struct wpa_ctrl *ctrl, int argc, char *argv[])
1453 {
1454 return wpa_ctrl_command(ctrl, "ADD_CRED");
1455 }
1456
1457
1458 static int wpa_cli_cmd_remove_cred(struct wpa_ctrl *ctrl, int argc,
1459 char *argv[])
1460 {
1461 return wpa_cli_cmd(ctrl, "REMOVE_CRED", 1, argc, argv);
1462 }
1463
1464
1465 static int wpa_cli_cmd_set_cred(struct wpa_ctrl *ctrl, int argc, char *argv[])
1466 {
1467 if (argc != 3) {
1468 printf("Invalid SET_CRED command: needs three arguments\n"
1469 "(cred id, variable name, and value)\n");
1470 return -1;
1471 }
1472
1473 return wpa_cli_cmd(ctrl, "SET_CRED", 3, argc, argv);
1474 }
1475
1476
1477 static int wpa_cli_cmd_get_cred(struct wpa_ctrl *ctrl, int argc, char *argv[])
1478 {
1479 if (argc != 2) {
1480 printf("Invalid GET_CRED command: needs two arguments\n"
1481 "(cred id, variable name)\n");
1482 return -1;
1483 }
1484
1485 return wpa_cli_cmd(ctrl, "GET_CRED", 2, argc, argv);
1486 }
1487
1488
1489 static int wpa_cli_cmd_disconnect(struct wpa_ctrl *ctrl, int argc,
1490 char *argv[])
1491 {
1492 return wpa_ctrl_command(ctrl, "DISCONNECT");
1493 }
1494
1495
1496 static int wpa_cli_cmd_reconnect(struct wpa_ctrl *ctrl, int argc,
1497 char *argv[])
1498 {
1499 return wpa_ctrl_command(ctrl, "RECONNECT");
1500 }
1501
1502
1503 static int wpa_cli_cmd_save_config(struct wpa_ctrl *ctrl, int argc,
1504 char *argv[])
1505 {
1506 return wpa_ctrl_command(ctrl, "SAVE_CONFIG");
1507 }
1508
1509
1510 static int wpa_cli_cmd_scan(struct wpa_ctrl *ctrl, int argc, char *argv[])
1511 {
1512 return wpa_cli_cmd(ctrl, "SCAN", 0, argc, argv);
1513 }
1514
1515
1516 static int wpa_cli_cmd_scan_results(struct wpa_ctrl *ctrl, int argc,
1517 char *argv[])
1518 {
1519 return wpa_ctrl_command(ctrl, "SCAN_RESULTS");
1520 }
1521
1522
1523 static int wpa_cli_cmd_bss(struct wpa_ctrl *ctrl, int argc, char *argv[])
1524 {
1525 return wpa_cli_cmd(ctrl, "BSS", 1, argc, argv);
1526 }
1527
1528
1529 static char ** wpa_cli_complete_bss(const char *str, int pos)
1530 {
1531 int arg = get_cmd_arg_num(str, pos);
1532 char **res = NULL;
1533
1534 switch (arg) {
1535 case 1:
1536 res = cli_txt_list_array(&bsses);
1537 break;
1538 }
1539
1540 return res;
1541 }
1542
1543
1544 static int wpa_cli_cmd_get_capability(struct wpa_ctrl *ctrl, int argc,
1545 char *argv[])
1546 {
1547 if (argc < 1 || argc > 2) {
1548 printf("Invalid GET_CAPABILITY command: need either one or "
1549 "two arguments\n");
1550 return -1;
1551 }
1552
1553 if ((argc == 2) && os_strcmp(argv[1], "strict") != 0) {
1554 printf("Invalid GET_CAPABILITY command: second argument, "
1555 "if any, must be 'strict'\n");
1556 return -1;
1557 }
1558
1559 return wpa_cli_cmd(ctrl, "GET_CAPABILITY", 1, argc, argv);
1560 }
1561
1562
1563 static int wpa_cli_list_interfaces(struct wpa_ctrl *ctrl)
1564 {
1565 printf("Available interfaces:\n");
1566 return wpa_ctrl_command(ctrl, "INTERFACES");
1567 }
1568
1569
1570 static int wpa_cli_cmd_interface(struct wpa_ctrl *ctrl, int argc, char *argv[])
1571 {
1572 if (argc < 1) {
1573 wpa_cli_list_interfaces(ctrl);
1574 return 0;
1575 }
1576
1577 wpa_cli_close_connection();
1578 os_free(ctrl_ifname);
1579 ctrl_ifname = os_strdup(argv[0]);
1580 if (!ctrl_ifname) {
1581 printf("Failed to allocate memory\n");
1582 return 0;
1583 }
1584
1585 if (wpa_cli_open_connection(ctrl_ifname, 1) == 0) {
1586 printf("Connected to interface '%s.\n", ctrl_ifname);
1587 } else {
1588 printf("Could not connect to interface '%s' - re-trying\n",
1589 ctrl_ifname);
1590 }
1591 return 0;
1592 }
1593
1594
1595 static int wpa_cli_cmd_reconfigure(struct wpa_ctrl *ctrl, int argc,
1596 char *argv[])
1597 {
1598 return wpa_ctrl_command(ctrl, "RECONFIGURE");
1599 }
1600
1601
1602 static int wpa_cli_cmd_terminate(struct wpa_ctrl *ctrl, int argc,
1603 char *argv[])
1604 {
1605 return wpa_ctrl_command(ctrl, "TERMINATE");
1606 }
1607
1608
1609 static int wpa_cli_cmd_interface_add(struct wpa_ctrl *ctrl, int argc,
1610 char *argv[])
1611 {
1612 char cmd[256];
1613 int res;
1614
1615 if (argc < 1) {
1616 printf("Invalid INTERFACE_ADD command: needs at least one "
1617 "argument (interface name)\n"
1618 "All arguments: ifname confname driver ctrl_interface "
1619 "driver_param bridge_name\n");
1620 return -1;
1621 }
1622
1623 /*
1624 * INTERFACE_ADD <ifname>TAB<confname>TAB<driver>TAB<ctrl_interface>TAB
1625 * <driver_param>TAB<bridge_name>
1626 */
1627 res = os_snprintf(cmd, sizeof(cmd),
1628 "INTERFACE_ADD %s\t%s\t%s\t%s\t%s\t%s",
1629 argv[0],
1630 argc > 1 ? argv[1] : "", argc > 2 ? argv[2] : "",
1631 argc > 3 ? argv[3] : "", argc > 4 ? argv[4] : "",
1632 argc > 5 ? argv[5] : "");
1633 if (os_snprintf_error(sizeof(cmd), res))
1634 return -1;
1635 cmd[sizeof(cmd) - 1] = '\0';
1636 return wpa_ctrl_command(ctrl, cmd);
1637 }
1638
1639
1640 static int wpa_cli_cmd_interface_remove(struct wpa_ctrl *ctrl, int argc,
1641 char *argv[])
1642 {
1643 return wpa_cli_cmd(ctrl, "INTERFACE_REMOVE", 1, argc, argv);
1644 }
1645
1646
1647 static int wpa_cli_cmd_interface_list(struct wpa_ctrl *ctrl, int argc,
1648 char *argv[])
1649 {
1650 return wpa_ctrl_command(ctrl, "INTERFACE_LIST");
1651 }
1652
1653
1654 #ifdef CONFIG_AP
1655 static int wpa_cli_cmd_sta(struct wpa_ctrl *ctrl, int argc, char *argv[])
1656 {
1657 return wpa_cli_cmd(ctrl, "STA", 1, argc, argv);
1658 }
1659
1660
1661 static int wpa_ctrl_command_sta(struct wpa_ctrl *ctrl, char *cmd,
1662 char *addr, size_t addr_len)
1663 {
1664 char buf[4096], *pos;
1665 size_t len;
1666 int ret;
1667
1668 if (ctrl_conn == NULL) {
1669 printf("Not connected to hostapd - command dropped.\n");
1670 return -1;
1671 }
1672 len = sizeof(buf) - 1;
1673 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len,
1674 wpa_cli_msg_cb);
1675 if (ret == -2) {
1676 printf("'%s' command timed out.\n", cmd);
1677 return -2;
1678 } else if (ret < 0) {
1679 printf("'%s' command failed.\n", cmd);
1680 return -1;
1681 }
1682
1683 buf[len] = '\0';
1684 if (os_memcmp(buf, "FAIL", 4) == 0)
1685 return -1;
1686 printf("%s", buf);
1687
1688 pos = buf;
1689 while (*pos != '\0' && *pos != '\n')
1690 pos++;
1691 *pos = '\0';
1692 os_strlcpy(addr, buf, addr_len);
1693 return 0;
1694 }
1695
1696
1697 static int wpa_cli_cmd_all_sta(struct wpa_ctrl *ctrl, int argc, char *argv[])
1698 {
1699 char addr[32], cmd[64];
1700
1701 if (wpa_ctrl_command_sta(ctrl, "STA-FIRST", addr, sizeof(addr)))
1702 return 0;
1703 do {
1704 os_snprintf(cmd, sizeof(cmd), "STA-NEXT %s", addr);
1705 } while (wpa_ctrl_command_sta(ctrl, cmd, addr, sizeof(addr)) == 0);
1706
1707 return -1;
1708 }
1709
1710
1711 static int wpa_cli_cmd_deauthenticate(struct wpa_ctrl *ctrl, int argc,
1712 char *argv[])
1713 {
1714 return wpa_cli_cmd(ctrl, "DEAUTHENTICATE", 1, argc, argv);
1715 }
1716
1717
1718 static int wpa_cli_cmd_disassociate(struct wpa_ctrl *ctrl, int argc,
1719 char *argv[])
1720 {
1721 return wpa_cli_cmd(ctrl, "DISASSOCIATE", 1, argc, argv);
1722 }
1723
1724 static int wpa_cli_cmd_chanswitch(struct wpa_ctrl *ctrl, int argc,
1725 char *argv[])
1726 {
1727 return wpa_cli_cmd(ctrl, "CHAN_SWITCH", 2, argc, argv);
1728 }
1729
1730 #endif /* CONFIG_AP */
1731
1732
1733 static int wpa_cli_cmd_suspend(struct wpa_ctrl *ctrl, int argc, char *argv[])
1734 {
1735 return wpa_ctrl_command(ctrl, "SUSPEND");
1736 }
1737
1738
1739 static int wpa_cli_cmd_resume(struct wpa_ctrl *ctrl, int argc, char *argv[])
1740 {
1741 return wpa_ctrl_command(ctrl, "RESUME");
1742 }
1743
1744
1745 #ifdef CONFIG_TESTING_OPTIONS
1746 static int wpa_cli_cmd_drop_sa(struct wpa_ctrl *ctrl, int argc, char *argv[])
1747 {
1748 return wpa_ctrl_command(ctrl, "DROP_SA");
1749 }
1750 #endif /* CONFIG_TESTING_OPTIONS */
1751
1752
1753 static int wpa_cli_cmd_roam(struct wpa_ctrl *ctrl, int argc, char *argv[])
1754 {
1755 return wpa_cli_cmd(ctrl, "ROAM", 1, argc, argv);
1756 }
1757
1758
1759 #ifdef CONFIG_MESH
1760
1761 static int wpa_cli_cmd_mesh_interface_add(struct wpa_ctrl *ctrl, int argc,
1762 char *argv[])
1763 {
1764 return wpa_cli_cmd(ctrl, "MESH_INTERFACE_ADD", 0, argc, argv);
1765 }
1766
1767
1768 static int wpa_cli_cmd_mesh_group_add(struct wpa_ctrl *ctrl, int argc,
1769 char *argv[])
1770 {
1771 return wpa_cli_cmd(ctrl, "MESH_GROUP_ADD", 1, argc, argv);
1772 }
1773
1774
1775 static int wpa_cli_cmd_mesh_group_remove(struct wpa_ctrl *ctrl, int argc,
1776 char *argv[])
1777 {
1778 return wpa_cli_cmd(ctrl, "MESH_GROUP_REMOVE", 1, argc, argv);
1779 }
1780
1781 #endif /* CONFIG_MESH */
1782
1783
1784 #ifdef CONFIG_P2P
1785
1786 static int wpa_cli_cmd_p2p_find(struct wpa_ctrl *ctrl, int argc, char *argv[])
1787 {
1788 return wpa_cli_cmd(ctrl, "P2P_FIND", 0, argc, argv);
1789 }
1790
1791
1792 static char ** wpa_cli_complete_p2p_find(const char *str, int pos)
1793 {
1794 char **res = NULL;
1795 int arg = get_cmd_arg_num(str, pos);
1796
1797 res = os_calloc(6, sizeof(char *));
1798 if (res == NULL)
1799 return NULL;
1800 res[0] = os_strdup("type=social");
1801 if (res[0] == NULL) {
1802 os_free(res);
1803 return NULL;
1804 }
1805 res[1] = os_strdup("type=progressive");
1806 if (res[1] == NULL)
1807 return res;
1808 res[2] = os_strdup("delay=");
1809 if (res[2] == NULL)
1810 return res;
1811 res[3] = os_strdup("dev_id=");
1812 if (res[3] == NULL)
1813 return res;
1814 if (arg == 1)
1815 res[4] = os_strdup("[timeout]");
1816
1817 return res;
1818 }
1819
1820
1821 static int wpa_cli_cmd_p2p_stop_find(struct wpa_ctrl *ctrl, int argc,
1822 char *argv[])
1823 {
1824 return wpa_ctrl_command(ctrl, "P2P_STOP_FIND");
1825 }
1826
1827
1828 static int wpa_cli_cmd_p2p_connect(struct wpa_ctrl *ctrl, int argc,
1829 char *argv[])
1830 {
1831 return wpa_cli_cmd(ctrl, "P2P_CONNECT", 2, argc, argv);
1832 }
1833
1834
1835 static char ** wpa_cli_complete_p2p_connect(const char *str, int pos)
1836 {
1837 int arg = get_cmd_arg_num(str, pos);
1838 char **res = NULL;
1839
1840 switch (arg) {
1841 case 1:
1842 res = cli_txt_list_array(&p2p_peers);
1843 break;
1844 }
1845
1846 return res;
1847 }
1848
1849
1850 static int wpa_cli_cmd_p2p_listen(struct wpa_ctrl *ctrl, int argc,
1851 char *argv[])
1852 {
1853 return wpa_cli_cmd(ctrl, "P2P_LISTEN", 0, argc, argv);
1854 }
1855
1856
1857 static int wpa_cli_cmd_p2p_group_remove(struct wpa_ctrl *ctrl, int argc,
1858 char *argv[])
1859 {
1860 return wpa_cli_cmd(ctrl, "P2P_GROUP_REMOVE", 1, argc, argv);
1861 }
1862
1863
1864 static char ** wpa_cli_complete_p2p_group_remove(const char *str, int pos)
1865 {
1866 int arg = get_cmd_arg_num(str, pos);
1867 char **res = NULL;
1868
1869 switch (arg) {
1870 case 1:
1871 res = cli_txt_list_array(&p2p_groups);
1872 break;
1873 }
1874
1875 return res;
1876 }
1877
1878
1879 static int wpa_cli_cmd_p2p_group_add(struct wpa_ctrl *ctrl, int argc,
1880 char *argv[])
1881 {
1882 return wpa_cli_cmd(ctrl, "P2P_GROUP_ADD", 0, argc, argv);
1883 }
1884
1885
1886 static int wpa_cli_cmd_p2p_prov_disc(struct wpa_ctrl *ctrl, int argc,
1887 char *argv[])
1888 {
1889 if (argc != 2 && argc != 3) {
1890 printf("Invalid P2P_PROV_DISC command: needs at least "
1891 "two arguments, address and config method\n"
1892 "(display, keypad, or pbc) and an optional join\n");
1893 return -1;
1894 }
1895
1896 return wpa_cli_cmd(ctrl, "P2P_PROV_DISC", 2, argc, argv);
1897 }
1898
1899
1900 static int wpa_cli_cmd_p2p_get_passphrase(struct wpa_ctrl *ctrl, int argc,
1901 char *argv[])
1902 {
1903 return wpa_ctrl_command(ctrl, "P2P_GET_PASSPHRASE");
1904 }
1905
1906
1907 static int wpa_cli_cmd_p2p_serv_disc_req(struct wpa_ctrl *ctrl, int argc,
1908 char *argv[])
1909 {
1910 char cmd[4096];
1911
1912 if (argc != 2 && argc != 4) {
1913 printf("Invalid P2P_SERV_DISC_REQ command: needs two "
1914 "arguments (address and TLVs) or four arguments "
1915 "(address, \"upnp\", version, search target "
1916 "(SSDP ST:)\n");
1917 return -1;
1918 }
1919
1920 if (write_cmd(cmd, sizeof(cmd), "P2P_SERV_DISC_REQ", argc, argv) < 0)
1921 return -1;
1922 return wpa_ctrl_command(ctrl, cmd);
1923 }
1924
1925
1926 static int wpa_cli_cmd_p2p_serv_disc_cancel_req(struct wpa_ctrl *ctrl,
1927 int argc, char *argv[])
1928 {
1929 return wpa_cli_cmd(ctrl, "P2P_SERV_DISC_CANCEL_REQ", 1, argc, argv);
1930 }
1931
1932
1933 static int wpa_cli_cmd_p2p_serv_disc_resp(struct wpa_ctrl *ctrl, int argc,
1934 char *argv[])
1935 {
1936 char cmd[4096];
1937 int res;
1938
1939 if (argc != 4) {
1940 printf("Invalid P2P_SERV_DISC_RESP command: needs four "
1941 "arguments (freq, address, dialog token, and TLVs)\n");
1942 return -1;
1943 }
1944
1945 res = os_snprintf(cmd, sizeof(cmd), "P2P_SERV_DISC_RESP %s %s %s %s",
1946 argv[0], argv[1], argv[2], argv[3]);
1947 if (os_snprintf_error(sizeof(cmd), res))
1948 return -1;
1949 cmd[sizeof(cmd) - 1] = '\0';
1950 return wpa_ctrl_command(ctrl, cmd);
1951 }
1952
1953
1954 static int wpa_cli_cmd_p2p_service_update(struct wpa_ctrl *ctrl, int argc,
1955 char *argv[])
1956 {
1957 return wpa_ctrl_command(ctrl, "P2P_SERVICE_UPDATE");
1958 }
1959
1960
1961 static int wpa_cli_cmd_p2p_serv_disc_external(struct wpa_ctrl *ctrl,
1962 int argc, char *argv[])
1963 {
1964 return wpa_cli_cmd(ctrl, "P2P_SERV_DISC_EXTERNAL", 1, argc, argv);
1965 }
1966
1967
1968 static int wpa_cli_cmd_p2p_service_flush(struct wpa_ctrl *ctrl, int argc,
1969 char *argv[])
1970 {
1971 return wpa_ctrl_command(ctrl, "P2P_SERVICE_FLUSH");
1972 }
1973
1974
1975 static int wpa_cli_cmd_p2p_service_add(struct wpa_ctrl *ctrl, int argc,
1976 char *argv[])
1977 {
1978 char cmd[4096];
1979 int res;
1980
1981 if (argc != 3 && argc != 4) {
1982 printf("Invalid P2P_SERVICE_ADD command: needs three or four "
1983 "arguments\n");
1984 return -1;
1985 }
1986
1987 if (argc == 4)
1988 res = os_snprintf(cmd, sizeof(cmd),
1989 "P2P_SERVICE_ADD %s %s %s %s",
1990 argv[0], argv[1], argv[2], argv[3]);
1991 else
1992 res = os_snprintf(cmd, sizeof(cmd),
1993 "P2P_SERVICE_ADD %s %s %s",
1994 argv[0], argv[1], argv[2]);
1995 if (os_snprintf_error(sizeof(cmd), res))
1996 return -1;
1997 cmd[sizeof(cmd) - 1] = '\0';
1998 return wpa_ctrl_command(ctrl, cmd);
1999 }
2000
2001
2002 static int wpa_cli_cmd_p2p_service_del(struct wpa_ctrl *ctrl, int argc,
2003 char *argv[])
2004 {
2005 char cmd[4096];
2006 int res;
2007
2008 if (argc != 2 && argc != 3) {
2009 printf("Invalid P2P_SERVICE_DEL command: needs two or three "
2010 "arguments\n");
2011 return -1;
2012 }
2013
2014 if (argc == 3)
2015 res = os_snprintf(cmd, sizeof(cmd),
2016 "P2P_SERVICE_DEL %s %s %s",
2017 argv[0], argv[1], argv[2]);
2018 else
2019 res = os_snprintf(cmd, sizeof(cmd),
2020 "P2P_SERVICE_DEL %s %s",
2021 argv[0], argv[1]);
2022 if (os_snprintf_error(sizeof(cmd), res))
2023 return -1;
2024 cmd[sizeof(cmd) - 1] = '\0';
2025 return wpa_ctrl_command(ctrl, cmd);
2026 }
2027
2028
2029 static int wpa_cli_cmd_p2p_reject(struct wpa_ctrl *ctrl,
2030 int argc, char *argv[])
2031 {
2032 return wpa_cli_cmd(ctrl, "P2P_REJECT", 1, argc, argv);
2033 }
2034
2035
2036 static int wpa_cli_cmd_p2p_invite(struct wpa_ctrl *ctrl,
2037 int argc, char *argv[])
2038 {
2039 return wpa_cli_cmd(ctrl, "P2P_INVITE", 1, argc, argv);
2040 }
2041
2042
2043 static int wpa_cli_cmd_p2p_peer(struct wpa_ctrl *ctrl, int argc, char *argv[])
2044 {
2045 return wpa_cli_cmd(ctrl, "P2P_PEER", 1, argc, argv);
2046 }
2047
2048
2049 static char ** wpa_cli_complete_p2p_peer(const char *str, int pos)
2050 {
2051 int arg = get_cmd_arg_num(str, pos);
2052 char **res = NULL;
2053
2054 switch (arg) {
2055 case 1:
2056 res = cli_txt_list_array(&p2p_peers);
2057 break;
2058 }
2059
2060 return res;
2061 }
2062
2063
2064 static int wpa_ctrl_command_p2p_peer(struct wpa_ctrl *ctrl, char *cmd,
2065 char *addr, size_t addr_len,
2066 int discovered)
2067 {
2068 char buf[4096], *pos;
2069 size_t len;
2070 int ret;
2071
2072 if (ctrl_conn == NULL)
2073 return -1;
2074 len = sizeof(buf) - 1;
2075 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len,
2076 wpa_cli_msg_cb);
2077 if (ret == -2) {
2078 printf("'%s' command timed out.\n", cmd);
2079 return -2;
2080 } else if (ret < 0) {
2081 printf("'%s' command failed.\n", cmd);
2082 return -1;
2083 }
2084
2085 buf[len] = '\0';
2086 if (os_memcmp(buf, "FAIL", 4) == 0)
2087 return -1;
2088
2089 pos = buf;
2090 while (*pos != '\0' && *pos != '\n')
2091 pos++;
2092 *pos++ = '\0';
2093 os_strlcpy(addr, buf, addr_len);
2094 if (!discovered || os_strstr(pos, "[PROBE_REQ_ONLY]") == NULL)
2095 printf("%s\n", addr);
2096 return 0;
2097 }
2098
2099
2100 static int wpa_cli_cmd_p2p_peers(struct wpa_ctrl *ctrl, int argc, char *argv[])
2101 {
2102 char addr[32], cmd[64];
2103 int discovered;
2104
2105 discovered = argc > 0 && os_strcmp(argv[0], "discovered") == 0;
2106
2107 if (wpa_ctrl_command_p2p_peer(ctrl, "P2P_PEER FIRST",
2108 addr, sizeof(addr), discovered))
2109 return -1;
2110 do {
2111 os_snprintf(cmd, sizeof(cmd), "P2P_PEER NEXT-%s", addr);
2112 } while (wpa_ctrl_command_p2p_peer(ctrl, cmd, addr, sizeof(addr),
2113 discovered) == 0);
2114
2115 return 0;
2116 }
2117
2118
2119 static int wpa_cli_cmd_p2p_set(struct wpa_ctrl *ctrl, int argc, char *argv[])
2120 {
2121 return wpa_cli_cmd(ctrl, "P2P_SET", 2, argc, argv);
2122 }
2123
2124
2125 static char ** wpa_cli_complete_p2p_set(const char *str, int pos)
2126 {
2127 int arg = get_cmd_arg_num(str, pos);
2128 const char *fields[] = {
2129 "discoverability",
2130 "managed",
2131 "listen_channel",
2132 "ssid_postfix",
2133 "noa",
2134 "ps",
2135 "oppps",
2136 "ctwindow",
2137 "disabled",
2138 "conc_pref",
2139 "force_long_sd",
2140 "peer_filter",
2141 "cross_connect",
2142 "go_apsd",
2143 "client_apsd",
2144 "disallow_freq",
2145 "disc_int",
2146 "per_sta_psk",
2147 };
2148 int i, num_fields = ARRAY_SIZE(fields);
2149
2150 if (arg == 1) {
2151 char **res = os_calloc(num_fields + 1, sizeof(char *));
2152 if (res == NULL)
2153 return NULL;
2154 for (i = 0; i < num_fields; i++) {
2155 res[i] = os_strdup(fields[i]);
2156 if (res[i] == NULL)
2157 return res;
2158 }
2159 return res;
2160 }
2161
2162 if (arg == 2 && os_strncasecmp(str, "p2p_set peer_filter ", 20) == 0)
2163 return cli_txt_list_array(&p2p_peers);
2164
2165 return NULL;
2166 }
2167
2168
2169 static int wpa_cli_cmd_p2p_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
2170 {
2171 return wpa_ctrl_command(ctrl, "P2P_FLUSH");
2172 }
2173
2174
2175 static int wpa_cli_cmd_p2p_cancel(struct wpa_ctrl *ctrl, int argc,
2176 char *argv[])
2177 {
2178 return wpa_ctrl_command(ctrl, "P2P_CANCEL");
2179 }
2180
2181
2182 static int wpa_cli_cmd_p2p_unauthorize(struct wpa_ctrl *ctrl, int argc,
2183 char *argv[])
2184 {
2185 return wpa_cli_cmd(ctrl, "P2P_UNAUTHORIZE", 1, argc, argv);
2186 }
2187
2188
2189 static int wpa_cli_cmd_p2p_presence_req(struct wpa_ctrl *ctrl, int argc,
2190 char *argv[])
2191 {
2192 if (argc != 0 && argc != 2 && argc != 4) {
2193 printf("Invalid P2P_PRESENCE_REQ command: needs two arguments "
2194 "(preferred duration, interval; in microsecods).\n"
2195 "Optional second pair can be used to provide "
2196 "acceptable values.\n");
2197 return -1;
2198 }
2199
2200 return wpa_cli_cmd(ctrl, "P2P_PRESENCE_REQ", 0, argc, argv);
2201 }
2202
2203
2204 static int wpa_cli_cmd_p2p_ext_listen(struct wpa_ctrl *ctrl, int argc,
2205 char *argv[])
2206 {
2207 if (argc != 0 && argc != 2) {
2208 printf("Invalid P2P_EXT_LISTEN command: needs two arguments "
2209 "(availability period, availability interval; in "
2210 "millisecods).\n"
2211 "Extended Listen Timing can be cancelled with this "
2212 "command when used without parameters.\n");
2213 return -1;
2214 }
2215
2216 return wpa_cli_cmd(ctrl, "P2P_EXT_LISTEN", 0, argc, argv);
2217 }
2218
2219
2220 static int wpa_cli_cmd_p2p_remove_client(struct wpa_ctrl *ctrl, int argc,
2221 char *argv[])
2222 {
2223 return wpa_cli_cmd(ctrl, "P2P_REMOVE_CLIENT", 1, argc, argv);
2224 }
2225
2226 #endif /* CONFIG_P2P */
2227
2228 #ifdef CONFIG_WIFI_DISPLAY
2229
2230 static int wpa_cli_cmd_wfd_subelem_set(struct wpa_ctrl *ctrl, int argc,
2231 char *argv[])
2232 {
2233 char cmd[100];
2234 int res;
2235
2236 if (argc != 1 && argc != 2) {
2237 printf("Invalid WFD_SUBELEM_SET command: needs one or two "
2238 "arguments (subelem, hexdump)\n");
2239 return -1;
2240 }
2241
2242 res = os_snprintf(cmd, sizeof(cmd), "WFD_SUBELEM_SET %s %s",
2243 argv[0], argc > 1 ? argv[1] : "");
2244 if (os_snprintf_error(sizeof(cmd), res))
2245 return -1;
2246 cmd[sizeof(cmd) - 1] = '\0';
2247 return wpa_ctrl_command(ctrl, cmd);
2248 }
2249
2250
2251 static int wpa_cli_cmd_wfd_subelem_get(struct wpa_ctrl *ctrl, int argc,
2252 char *argv[])
2253 {
2254 char cmd[100];
2255 int res;
2256
2257 if (argc != 1) {
2258 printf("Invalid WFD_SUBELEM_GET command: needs one "
2259 "argument (subelem)\n");
2260 return -1;
2261 }
2262
2263 res = os_snprintf(cmd, sizeof(cmd), "WFD_SUBELEM_GET %s",
2264 argv[0]);
2265 if (os_snprintf_error(sizeof(cmd), res))
2266 return -1;
2267 cmd[sizeof(cmd) - 1] = '\0';
2268 return wpa_ctrl_command(ctrl, cmd);
2269 }
2270 #endif /* CONFIG_WIFI_DISPLAY */
2271
2272
2273 #ifdef CONFIG_INTERWORKING
2274 static int wpa_cli_cmd_fetch_anqp(struct wpa_ctrl *ctrl, int argc,
2275 char *argv[])
2276 {
2277 return wpa_ctrl_command(ctrl, "FETCH_ANQP");
2278 }
2279
2280
2281 static int wpa_cli_cmd_stop_fetch_anqp(struct wpa_ctrl *ctrl, int argc,
2282 char *argv[])
2283 {
2284 return wpa_ctrl_command(ctrl, "STOP_FETCH_ANQP");
2285 }
2286
2287
2288 static int wpa_cli_cmd_interworking_select(struct wpa_ctrl *ctrl, int argc,
2289 char *argv[])
2290 {
2291 return wpa_cli_cmd(ctrl, "INTERWORKING_SELECT", 0, argc, argv);
2292 }
2293
2294
2295 static int wpa_cli_cmd_interworking_connect(struct wpa_ctrl *ctrl, int argc,
2296 char *argv[])
2297 {
2298 return wpa_cli_cmd(ctrl, "INTERWORKING_CONNECT", 1, argc, argv);
2299 }
2300
2301
2302 static int wpa_cli_cmd_anqp_get(struct wpa_ctrl *ctrl, int argc, char *argv[])
2303 {
2304 return wpa_cli_cmd(ctrl, "ANQP_GET", 2, argc, argv);
2305 }
2306
2307
2308 static int wpa_cli_cmd_gas_request(struct wpa_ctrl *ctrl, int argc,
2309 char *argv[])
2310 {
2311 return wpa_cli_cmd(ctrl, "GAS_REQUEST", 2, argc, argv);
2312 }
2313
2314
2315 static int wpa_cli_cmd_gas_response_get(struct wpa_ctrl *ctrl, int argc,
2316 char *argv[])
2317 {
2318 return wpa_cli_cmd(ctrl, "GAS_RESPONSE_GET", 2, argc, argv);
2319 }
2320 #endif /* CONFIG_INTERWORKING */
2321
2322
2323 #ifdef CONFIG_HS20
2324
2325 static int wpa_cli_cmd_hs20_anqp_get(struct wpa_ctrl *ctrl, int argc,
2326 char *argv[])
2327 {
2328 return wpa_cli_cmd(ctrl, "HS20_ANQP_GET", 2, argc, argv);
2329 }
2330
2331
2332 static int wpa_cli_cmd_get_nai_home_realm_list(struct wpa_ctrl *ctrl, int argc,
2333 char *argv[])
2334 {
2335 char cmd[512];
2336
2337 if (argc == 0) {
2338 printf("Command needs one or two arguments (dst mac addr and "
2339 "optional home realm)\n");
2340 return -1;
2341 }
2342
2343 if (write_cmd(cmd, sizeof(cmd), "HS20_GET_NAI_HOME_REALM_LIST",
2344 argc, argv) < 0)
2345 return -1;
2346
2347 return wpa_ctrl_command(ctrl, cmd);
2348 }
2349
2350
2351 static int wpa_cli_cmd_hs20_icon_request(struct wpa_ctrl *ctrl, int argc,
2352 char *argv[])
2353 {
2354 char cmd[512];
2355
2356 if (argc < 2) {
2357 printf("Command needs two arguments (dst mac addr and "
2358 "icon name)\n");
2359 return -1;
2360 }
2361
2362 if (write_cmd(cmd, sizeof(cmd), "HS20_ICON_REQUEST", argc, argv) < 0)
2363 return -1;
2364
2365 return wpa_ctrl_command(ctrl, cmd);
2366 }
2367
2368
2369 static int wpa_cli_cmd_fetch_osu(struct wpa_ctrl *ctrl, int argc, char *argv[])
2370 {
2371 return wpa_ctrl_command(ctrl, "FETCH_OSU");
2372 }
2373
2374
2375 static int wpa_cli_cmd_cancel_fetch_osu(struct wpa_ctrl *ctrl, int argc,
2376 char *argv[])
2377 {
2378 return wpa_ctrl_command(ctrl, "CANCEL_FETCH_OSU");
2379 }
2380
2381 #endif /* CONFIG_HS20 */
2382
2383
2384 static int wpa_cli_cmd_sta_autoconnect(struct wpa_ctrl *ctrl, int argc,
2385 char *argv[])
2386 {
2387 return wpa_cli_cmd(ctrl, "STA_AUTOCONNECT", 1, argc, argv);
2388 }
2389
2390
2391 static int wpa_cli_cmd_tdls_discover(struct wpa_ctrl *ctrl, int argc,
2392 char *argv[])
2393 {
2394 return wpa_cli_cmd(ctrl, "TDLS_DISCOVER", 1, argc, argv);
2395 }
2396
2397
2398 static int wpa_cli_cmd_tdls_setup(struct wpa_ctrl *ctrl, int argc,
2399 char *argv[])
2400 {
2401 return wpa_cli_cmd(ctrl, "TDLS_SETUP", 1, argc, argv);
2402 }
2403
2404
2405 static int wpa_cli_cmd_tdls_teardown(struct wpa_ctrl *ctrl, int argc,
2406 char *argv[])
2407 {
2408 return wpa_cli_cmd(ctrl, "TDLS_TEARDOWN", 1, argc, argv);
2409 }
2410
2411
2412 static int wpa_cli_cmd_wmm_ac_addts(struct wpa_ctrl *ctrl, int argc,
2413 char *argv[])
2414 {
2415 return wpa_cli_cmd(ctrl, "WMM_AC_ADDTS", 3, argc, argv);
2416 }
2417
2418
2419 static int wpa_cli_cmd_wmm_ac_delts(struct wpa_ctrl *ctrl, int argc,
2420 char *argv[])
2421 {
2422 return wpa_cli_cmd(ctrl, "WMM_AC_DELTS", 1, argc, argv);
2423 }
2424
2425
2426 static int wpa_cli_cmd_wmm_ac_status(struct wpa_ctrl *ctrl, int argc,
2427 char *argv[])
2428 {
2429 return wpa_ctrl_command(ctrl, "WMM_AC_STATUS");
2430 }
2431
2432
2433 static int wpa_cli_cmd_tdls_chan_switch(struct wpa_ctrl *ctrl, int argc,
2434 char *argv[])
2435 {
2436 return wpa_cli_cmd(ctrl, "TDLS_CHAN_SWITCH", 2, argc, argv);
2437 }
2438
2439
2440 static int wpa_cli_cmd_tdls_cancel_chan_switch(struct wpa_ctrl *ctrl, int argc,
2441 char *argv[])
2442 {
2443 return wpa_cli_cmd(ctrl, "TDLS_CANCEL_CHAN_SWITCH", 1, argc, argv);
2444 }
2445
2446
2447 static int wpa_cli_cmd_signal_poll(struct wpa_ctrl *ctrl, int argc,
2448 char *argv[])
2449 {
2450 return wpa_ctrl_command(ctrl, "SIGNAL_POLL");
2451 }
2452
2453
2454 static int wpa_cli_cmd_pktcnt_poll(struct wpa_ctrl *ctrl, int argc,
2455 char *argv[])
2456 {
2457 return wpa_ctrl_command(ctrl, "PKTCNT_POLL");
2458 }
2459
2460
2461 static int wpa_cli_cmd_reauthenticate(struct wpa_ctrl *ctrl, int argc,
2462 char *argv[])
2463 {
2464 return wpa_ctrl_command(ctrl, "REAUTHENTICATE");
2465 }
2466
2467
2468 #ifdef CONFIG_AUTOSCAN
2469
2470 static int wpa_cli_cmd_autoscan(struct wpa_ctrl *ctrl, int argc, char *argv[])
2471 {
2472 if (argc == 0)
2473 return wpa_ctrl_command(ctrl, "AUTOSCAN ");
2474
2475 return wpa_cli_cmd(ctrl, "AUTOSCAN", 0, argc, argv);
2476 }
2477
2478 #endif /* CONFIG_AUTOSCAN */
2479
2480
2481 #ifdef CONFIG_WNM
2482
2483 static int wpa_cli_cmd_wnm_sleep(struct wpa_ctrl *ctrl, int argc, char *argv[])
2484 {
2485 return wpa_cli_cmd(ctrl, "WNM_SLEEP", 0, argc, argv);
2486 }
2487
2488
2489 static int wpa_cli_cmd_wnm_bss_query(struct wpa_ctrl *ctrl, int argc, char *argv[])
2490 {
2491 return wpa_cli_cmd(ctrl, "WNM_BSS_QUERY", 1, argc, argv);
2492 }
2493
2494 #endif /* CONFIG_WNM */
2495
2496
2497 static int wpa_cli_cmd_raw(struct wpa_ctrl *ctrl, int argc, char *argv[])
2498 {
2499 if (argc == 0)
2500 return -1;
2501 return wpa_cli_cmd(ctrl, argv[0], 0, argc - 1, &argv[1]);
2502 }
2503
2504
2505 #ifdef ANDROID
2506 static int wpa_cli_cmd_driver(struct wpa_ctrl *ctrl, int argc, char *argv[])
2507 {
2508 return wpa_cli_cmd(ctrl, "DRIVER", 1, argc, argv);
2509 }
2510 #endif /* ANDROID */
2511
2512
2513 static int wpa_cli_cmd_vendor(struct wpa_ctrl *ctrl, int argc, char *argv[])
2514 {
2515 return wpa_cli_cmd(ctrl, "VENDOR", 1, argc, argv);
2516 }
2517
2518
2519 static int wpa_cli_cmd_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
2520 {
2521 return wpa_ctrl_command(ctrl, "FLUSH");
2522 }
2523
2524
2525 static int wpa_cli_cmd_radio_work(struct wpa_ctrl *ctrl, int argc, char *argv[])
2526 {
2527 return wpa_cli_cmd(ctrl, "RADIO_WORK", 1, argc, argv);
2528 }
2529
2530
2531 static int wpa_cli_cmd_neighbor_rep_request(struct wpa_ctrl *ctrl, int argc,
2532 char *argv[])
2533 {
2534 return wpa_cli_cmd(ctrl, "NEIGHBOR_REP_REQUEST", 0, argc, argv);
2535 }
2536
2537
2538 static int wpa_cli_cmd_erp_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
2539 {
2540 return wpa_ctrl_command(ctrl, "ERP_FLUSH");
2541 }
2542
2543
2544 static int wpa_cli_cmd_mac_rand_scan(struct wpa_ctrl *ctrl, int argc,
2545 char *argv[])
2546 {
2547 return wpa_cli_cmd(ctrl, "MAC_RAND_SCAN", 1, argc, argv);
2548 }
2549
2550
2551 enum wpa_cli_cmd_flags {
2552 cli_cmd_flag_none = 0x00,
2553 cli_cmd_flag_sensitive = 0x01
2554 };
2555
2556 struct wpa_cli_cmd {
2557 const char *cmd;
2558 int (*handler)(struct wpa_ctrl *ctrl, int argc, char *argv[]);
2559 char ** (*completion)(const char *str, int pos);
2560 enum wpa_cli_cmd_flags flags;
2561 const char *usage;
2562 };
2563
2564 static struct wpa_cli_cmd wpa_cli_commands[] = {
2565 { "status", wpa_cli_cmd_status, NULL,
2566 cli_cmd_flag_none,
2567 "[verbose] = get current WPA/EAPOL/EAP status" },
2568 { "ifname", wpa_cli_cmd_ifname, NULL,
2569 cli_cmd_flag_none,
2570 "= get current interface name" },
2571 { "ping", wpa_cli_cmd_ping, NULL,
2572 cli_cmd_flag_none,
2573 "= pings wpa_supplicant" },
2574 { "relog", wpa_cli_cmd_relog, NULL,
2575 cli_cmd_flag_none,
2576 "= re-open log-file (allow rolling logs)" },
2577 { "note", wpa_cli_cmd_note, NULL,
2578 cli_cmd_flag_none,
2579 "<text> = add a note to wpa_supplicant debug log" },
2580 { "mib", wpa_cli_cmd_mib, NULL,
2581 cli_cmd_flag_none,
2582 "= get MIB variables (dot1x, dot11)" },
2583 { "help", wpa_cli_cmd_help, wpa_cli_complete_help,
2584 cli_cmd_flag_none,
2585 "[command] = show usage help" },
2586 { "interface", wpa_cli_cmd_interface, NULL,
2587 cli_cmd_flag_none,
2588 "[ifname] = show interfaces/select interface" },
2589 { "level", wpa_cli_cmd_level, NULL,
2590 cli_cmd_flag_none,
2591 "<debug level> = change debug level" },
2592 { "license", wpa_cli_cmd_license, NULL,
2593 cli_cmd_flag_none,
2594 "= show full wpa_cli license" },
2595 { "quit", wpa_cli_cmd_quit, NULL,
2596 cli_cmd_flag_none,
2597 "= exit wpa_cli" },
2598 { "set", wpa_cli_cmd_set, wpa_cli_complete_set,
2599 cli_cmd_flag_none,
2600 "= set variables (shows list of variables when run without "
2601 "arguments)" },
2602 { "get", wpa_cli_cmd_get, NULL,
2603 cli_cmd_flag_none,
2604 "<name> = get information" },
2605 { "logon", wpa_cli_cmd_logon, NULL,
2606 cli_cmd_flag_none,
2607 "= IEEE 802.1X EAPOL state machine logon" },
2608 { "logoff", wpa_cli_cmd_logoff, NULL,
2609 cli_cmd_flag_none,
2610 "= IEEE 802.1X EAPOL state machine logoff" },
2611 { "pmksa", wpa_cli_cmd_pmksa, NULL,
2612 cli_cmd_flag_none,
2613 "= show PMKSA cache" },
2614 { "pmksa_flush", wpa_cli_cmd_pmksa_flush, NULL,
2615 cli_cmd_flag_none,
2616 "= flush PMKSA cache entries" },
2617 { "reassociate", wpa_cli_cmd_reassociate, NULL,
2618 cli_cmd_flag_none,
2619 "= force reassociation" },
2620 { "reattach", wpa_cli_cmd_reattach, NULL,
2621 cli_cmd_flag_none,
2622 "= force reassociation back to the same BSS" },
2623 { "preauthenticate", wpa_cli_cmd_preauthenticate, wpa_cli_complete_bss,
2624 cli_cmd_flag_none,
2625 "<BSSID> = force preauthentication" },
2626 { "identity", wpa_cli_cmd_identity, NULL,
2627 cli_cmd_flag_none,
2628 "<network id> <identity> = configure identity for an SSID" },
2629 { "password", wpa_cli_cmd_password, NULL,
2630 cli_cmd_flag_sensitive,
2631 "<network id> <password> = configure password for an SSID" },
2632 { "new_password", wpa_cli_cmd_new_password, NULL,
2633 cli_cmd_flag_sensitive,
2634 "<network id> <password> = change password for an SSID" },
2635 { "pin", wpa_cli_cmd_pin, NULL,
2636 cli_cmd_flag_sensitive,
2637 "<network id> <pin> = configure pin for an SSID" },
2638 { "otp", wpa_cli_cmd_otp, NULL,
2639 cli_cmd_flag_sensitive,
2640 "<network id> <password> = configure one-time-password for an SSID"
2641 },
2642 { "passphrase", wpa_cli_cmd_passphrase, NULL,
2643 cli_cmd_flag_sensitive,
2644 "<network id> <passphrase> = configure private key passphrase\n"
2645 " for an SSID" },
2646 { "sim", wpa_cli_cmd_sim, NULL,
2647 cli_cmd_flag_sensitive,
2648 "<network id> <pin> = report SIM operation result" },
2649 { "bssid", wpa_cli_cmd_bssid, NULL,
2650 cli_cmd_flag_none,
2651 "<network id> <BSSID> = set preferred BSSID for an SSID" },
2652 { "blacklist", wpa_cli_cmd_blacklist, wpa_cli_complete_bss,
2653 cli_cmd_flag_none,
2654 "<BSSID> = add a BSSID to the blacklist\n"
2655 "blacklist clear = clear the blacklist\n"
2656 "blacklist = display the blacklist" },
2657 { "log_level", wpa_cli_cmd_log_level, NULL,
2658 cli_cmd_flag_none,
2659 "<level> [<timestamp>] = update the log level/timestamp\n"
2660 "log_level = display the current log level and log options" },
2661 { "list_networks", wpa_cli_cmd_list_networks, NULL,
2662 cli_cmd_flag_none,
2663 "= list configured networks" },
2664 { "select_network", wpa_cli_cmd_select_network, NULL,
2665 cli_cmd_flag_none,
2666 "<network id> = select a network (disable others)" },
2667 { "enable_network", wpa_cli_cmd_enable_network, NULL,
2668 cli_cmd_flag_none,
2669 "<network id> = enable a network" },
2670 { "disable_network", wpa_cli_cmd_disable_network, NULL,
2671 cli_cmd_flag_none,
2672 "<network id> = disable a network" },
2673 { "add_network", wpa_cli_cmd_add_network, NULL,
2674 cli_cmd_flag_none,
2675 "= add a network" },
2676 { "remove_network", wpa_cli_cmd_remove_network, NULL,
2677 cli_cmd_flag_none,
2678 "<network id> = remove a network" },
2679 { "set_network", wpa_cli_cmd_set_network, NULL,
2680 cli_cmd_flag_sensitive,
2681 "<network id> <variable> <value> = set network variables (shows\n"
2682 " list of variables when run without arguments)" },
2683 { "get_network", wpa_cli_cmd_get_network, NULL,
2684 cli_cmd_flag_none,
2685 "<network id> <variable> = get network variables" },
2686 { "dup_network", wpa_cli_cmd_dup_network, NULL,
2687 cli_cmd_flag_none,
2688 "<src network id> <dst network id> <variable> = duplicate network variables"
2689 },
2690 { "list_creds", wpa_cli_cmd_list_creds, NULL,
2691 cli_cmd_flag_none,
2692 "= list configured credentials" },
2693 { "add_cred", wpa_cli_cmd_add_cred, NULL,
2694 cli_cmd_flag_none,
2695 "= add a credential" },
2696 { "remove_cred", wpa_cli_cmd_remove_cred, NULL,
2697 cli_cmd_flag_none,
2698 "<cred id> = remove a credential" },
2699 { "set_cred", wpa_cli_cmd_set_cred, NULL,
2700 cli_cmd_flag_sensitive,
2701 "<cred id> <variable> <value> = set credential variables" },
2702 { "get_cred", wpa_cli_cmd_get_cred, NULL,
2703 cli_cmd_flag_none,
2704 "<cred id> <variable> = get credential variables" },
2705 { "save_config", wpa_cli_cmd_save_config, NULL,
2706 cli_cmd_flag_none,
2707 "= save the current configuration" },
2708 { "disconnect", wpa_cli_cmd_disconnect, NULL,
2709 cli_cmd_flag_none,
2710 "= disconnect and wait for reassociate/reconnect command before\n"
2711 " connecting" },
2712 { "reconnect", wpa_cli_cmd_reconnect, NULL,
2713 cli_cmd_flag_none,
2714 "= like reassociate, but only takes effect if already disconnected"
2715 },
2716 { "scan", wpa_cli_cmd_scan, NULL,
2717 cli_cmd_flag_none,
2718 "= request new BSS scan" },
2719 { "scan_results", wpa_cli_cmd_scan_results, NULL,
2720 cli_cmd_flag_none,
2721 "= get latest scan results" },
2722 { "bss", wpa_cli_cmd_bss, wpa_cli_complete_bss,
2723 cli_cmd_flag_none,
2724 "<<idx> | <bssid>> = get detailed scan result info" },
2725 { "get_capability", wpa_cli_cmd_get_capability, NULL,
2726 cli_cmd_flag_none,
2727 "<eap/pairwise/group/key_mgmt/proto/auth_alg/channels/freq/modes> "
2728 "= get capabilies" },
2729 { "reconfigure", wpa_cli_cmd_reconfigure, NULL,
2730 cli_cmd_flag_none,
2731 "= force wpa_supplicant to re-read its configuration file" },
2732 { "terminate", wpa_cli_cmd_terminate, NULL,
2733 cli_cmd_flag_none,
2734 "= terminate wpa_supplicant" },
2735 { "interface_add", wpa_cli_cmd_interface_add, NULL,
2736 cli_cmd_flag_none,
2737 "<ifname> <confname> <driver> <ctrl_interface> <driver_param>\n"
2738 " <bridge_name> = adds new interface, all parameters but <ifname>\n"
2739 " are optional" },
2740 { "interface_remove", wpa_cli_cmd_interface_remove, NULL,
2741 cli_cmd_flag_none,
2742 "<ifname> = removes the interface" },
2743 { "interface_list", wpa_cli_cmd_interface_list, NULL,
2744 cli_cmd_flag_none,
2745 "= list available interfaces" },
2746 { "ap_scan", wpa_cli_cmd_ap_scan, NULL,
2747 cli_cmd_flag_none,
2748 "<value> = set ap_scan parameter" },
2749 { "scan_interval", wpa_cli_cmd_scan_interval, NULL,
2750 cli_cmd_flag_none,
2751 "<value> = set scan_interval parameter (in seconds)" },
2752 { "bss_expire_age", wpa_cli_cmd_bss_expire_age, NULL,
2753 cli_cmd_flag_none,
2754 "<value> = set BSS expiration age parameter" },
2755 { "bss_expire_count", wpa_cli_cmd_bss_expire_count, NULL,
2756 cli_cmd_flag_none,
2757 "<value> = set BSS expiration scan count parameter" },
2758 { "bss_flush", wpa_cli_cmd_bss_flush, NULL,
2759 cli_cmd_flag_none,
2760 "<value> = set BSS flush age (0 by default)" },
2761 { "stkstart", wpa_cli_cmd_stkstart, NULL,
2762 cli_cmd_flag_none,
2763 "<addr> = request STK negotiation with <addr>" },
2764 { "ft_ds", wpa_cli_cmd_ft_ds, wpa_cli_complete_bss,
2765 cli_cmd_flag_none,
2766 "<addr> = request over-the-DS FT with <addr>" },
2767 { "wps_pbc", wpa_cli_cmd_wps_pbc, wpa_cli_complete_bss,
2768 cli_cmd_flag_none,
2769 "[BSSID] = start Wi-Fi Protected Setup: Push Button Configuration" },
2770 { "wps_pin", wpa_cli_cmd_wps_pin, wpa_cli_complete_bss,
2771 cli_cmd_flag_sensitive,
2772 "<BSSID> [PIN] = start WPS PIN method (returns PIN, if not "
2773 "hardcoded)" },
2774 { "wps_check_pin", wpa_cli_cmd_wps_check_pin, NULL,
2775 cli_cmd_flag_sensitive,
2776 "<PIN> = verify PIN checksum" },
2777 { "wps_cancel", wpa_cli_cmd_wps_cancel, NULL, cli_cmd_flag_none,
2778 "Cancels the pending WPS operation" },
2779 #ifdef CONFIG_WPS_NFC
2780 { "wps_nfc", wpa_cli_cmd_wps_nfc, wpa_cli_complete_bss,
2781 cli_cmd_flag_none,
2782 "[BSSID] = start Wi-Fi Protected Setup: NFC" },
2783 { "wps_nfc_config_token", wpa_cli_cmd_wps_nfc_config_token, NULL,
2784 cli_cmd_flag_none,
2785 "<WPS|NDEF> = build configuration token" },
2786 { "wps_nfc_token", wpa_cli_cmd_wps_nfc_token, NULL,
2787 cli_cmd_flag_none,
2788 "<WPS|NDEF> = create password token" },
2789 { "wps_nfc_tag_read", wpa_cli_cmd_wps_nfc_tag_read, NULL,
2790 cli_cmd_flag_sensitive,
2791 "<hexdump of payload> = report read NFC tag with WPS data" },
2792 { "nfc_get_handover_req", wpa_cli_cmd_nfc_get_handover_req, NULL,
2793 cli_cmd_flag_none,
2794 "<NDEF> <WPS> = create NFC handover request" },
2795 { "nfc_get_handover_sel", wpa_cli_cmd_nfc_get_handover_sel, NULL,
2796 cli_cmd_flag_none,
2797 "<NDEF> <WPS> = create NFC handover select" },
2798 { "nfc_report_handover", wpa_cli_cmd_nfc_report_handover, NULL,
2799 cli_cmd_flag_none,
2800 "<role> <type> <hexdump of req> <hexdump of sel> = report completed "
2801 "NFC handover" },
2802 #endif /* CONFIG_WPS_NFC */
2803 { "wps_reg", wpa_cli_cmd_wps_reg, wpa_cli_complete_bss,
2804 cli_cmd_flag_sensitive,
2805 "<BSSID> <AP PIN> = start WPS Registrar to configure an AP" },
2806 { "wps_ap_pin", wpa_cli_cmd_wps_ap_pin, NULL,
2807 cli_cmd_flag_sensitive,
2808 "[params..] = enable/disable AP PIN" },
2809 { "wps_er_start", wpa_cli_cmd_wps_er_start, NULL,
2810 cli_cmd_flag_none,
2811 "[IP address] = start Wi-Fi Protected Setup External Registrar" },
2812 { "wps_er_stop", wpa_cli_cmd_wps_er_stop, NULL,
2813 cli_cmd_flag_none,
2814 "= stop Wi-Fi Protected Setup External Registrar" },
2815 { "wps_er_pin", wpa_cli_cmd_wps_er_pin, NULL,
2816 cli_cmd_flag_sensitive,
2817 "<UUID> <PIN> = add an Enrollee PIN to External Registrar" },
2818 { "wps_er_pbc", wpa_cli_cmd_wps_er_pbc, NULL,
2819 cli_cmd_flag_none,
2820 "<UUID> = accept an Enrollee PBC using External Registrar" },
2821 { "wps_er_learn", wpa_cli_cmd_wps_er_learn, NULL,
2822 cli_cmd_flag_sensitive,
2823 "<UUID> <PIN> = learn AP configuration" },
2824 { "wps_er_set_config", wpa_cli_cmd_wps_er_set_config, NULL,
2825 cli_cmd_flag_none,
2826 "<UUID> <network id> = set AP configuration for enrolling" },
2827 { "wps_er_config", wpa_cli_cmd_wps_er_config, NULL,
2828 cli_cmd_flag_sensitive,
2829 "<UUID> <PIN> <SSID> <auth> <encr> <key> = configure AP" },
2830 #ifdef CONFIG_WPS_NFC
2831 { "wps_er_nfc_config_token", wpa_cli_cmd_wps_er_nfc_config_token, NULL,
2832 cli_cmd_flag_none,
2833 "<WPS/NDEF> <UUID> = build NFC configuration token" },
2834 #endif /* CONFIG_WPS_NFC */
2835 { "ibss_rsn", wpa_cli_cmd_ibss_rsn, NULL,
2836 cli_cmd_flag_none,
2837 "<addr> = request RSN authentication with <addr> in IBSS" },
2838 #ifdef CONFIG_AP
2839 { "sta", wpa_cli_cmd_sta, NULL,
2840 cli_cmd_flag_none,
2841 "<addr> = get information about an associated station (AP)" },
2842 { "all_sta", wpa_cli_cmd_all_sta, NULL,
2843 cli_cmd_flag_none,
2844 "= get information about all associated stations (AP)" },
2845 { "deauthenticate", wpa_cli_cmd_deauthenticate, NULL,
2846 cli_cmd_flag_none,
2847 "<addr> = deauthenticate a station" },
2848 { "disassociate", wpa_cli_cmd_disassociate, NULL,
2849 cli_cmd_flag_none,
2850 "<addr> = disassociate a station" },
2851 { "chan_switch", wpa_cli_cmd_chanswitch, NULL,
2852 cli_cmd_flag_none,
2853 "<cs_count> <freq> [sec_channel_offset=] [center_freq1=]"
2854 " [center_freq2=] [bandwidth=] [blocktx] [ht|vht]"
2855 " = CSA parameters" },
2856 #endif /* CONFIG_AP */
2857 { "suspend", wpa_cli_cmd_suspend, NULL, cli_cmd_flag_none,
2858 "= notification of suspend/hibernate" },
2859 { "resume", wpa_cli_cmd_resume, NULL, cli_cmd_flag_none,
2860 "= notification of resume/thaw" },
2861 #ifdef CONFIG_TESTING_OPTIONS
2862 { "drop_sa", wpa_cli_cmd_drop_sa, NULL, cli_cmd_flag_none,
2863 "= drop SA without deauth/disassoc (test command)" },
2864 #endif /* CONFIG_TESTING_OPTIONS */
2865 { "roam", wpa_cli_cmd_roam, wpa_cli_complete_bss,
2866 cli_cmd_flag_none,
2867 "<addr> = roam to the specified BSS" },
2868 #ifdef CONFIG_MESH
2869 { "mesh_interface_add", wpa_cli_cmd_mesh_interface_add, NULL,
2870 cli_cmd_flag_none,
2871 "[ifname] = Create a new mesh interface" },
2872 { "mesh_group_add", wpa_cli_cmd_mesh_group_add, NULL,
2873 cli_cmd_flag_none,
2874 "<network id> = join a mesh network (disable others)" },
2875 { "mesh_group_remove", wpa_cli_cmd_mesh_group_remove, NULL,
2876 cli_cmd_flag_none,
2877 "<ifname> = Remove mesh group interface" },
2878 #endif /* CONFIG_MESH */
2879 #ifdef CONFIG_P2P
2880 { "p2p_find", wpa_cli_cmd_p2p_find, wpa_cli_complete_p2p_find,
2881 cli_cmd_flag_none,
2882 "[timeout] [type=*] = find P2P Devices for up-to timeout seconds" },
2883 { "p2p_stop_find", wpa_cli_cmd_p2p_stop_find, NULL, cli_cmd_flag_none,
2884 "= stop P2P Devices search" },
2885 { "p2p_connect", wpa_cli_cmd_p2p_connect, wpa_cli_complete_p2p_connect,
2886 cli_cmd_flag_none,
2887 "<addr> <\"pbc\"|PIN> [ht40] = connect to a P2P Device" },
2888 { "p2p_listen", wpa_cli_cmd_p2p_listen, NULL, cli_cmd_flag_none,
2889 "[timeout] = listen for P2P Devices for up-to timeout seconds" },
2890 { "p2p_group_remove", wpa_cli_cmd_p2p_group_remove,
2891 wpa_cli_complete_p2p_group_remove, cli_cmd_flag_none,
2892 "<ifname> = remove P2P group interface (terminate group if GO)" },
2893 { "p2p_group_add", wpa_cli_cmd_p2p_group_add, NULL, cli_cmd_flag_none,
2894 "[ht40] = add a new P2P group (local end as GO)" },
2895 { "p2p_prov_disc", wpa_cli_cmd_p2p_prov_disc,
2896 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
2897 "<addr> <method> = request provisioning discovery" },
2898 { "p2p_get_passphrase", wpa_cli_cmd_p2p_get_passphrase, NULL,
2899 cli_cmd_flag_none,
2900 "= get the passphrase for a group (GO only)" },
2901 { "p2p_serv_disc_req", wpa_cli_cmd_p2p_serv_disc_req,
2902 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
2903 "<addr> <TLVs> = schedule service discovery request" },
2904 { "p2p_serv_disc_cancel_req", wpa_cli_cmd_p2p_serv_disc_cancel_req,
2905 NULL, cli_cmd_flag_none,
2906 "<id> = cancel pending service discovery request" },
2907 { "p2p_serv_disc_resp", wpa_cli_cmd_p2p_serv_disc_resp, NULL,
2908 cli_cmd_flag_none,
2909 "<freq> <addr> <dialog token> <TLVs> = service discovery response" },
2910 { "p2p_service_update", wpa_cli_cmd_p2p_service_update, NULL,
2911 cli_cmd_flag_none,
2912 "= indicate change in local services" },
2913 { "p2p_serv_disc_external", wpa_cli_cmd_p2p_serv_disc_external, NULL,
2914 cli_cmd_flag_none,
2915 "<external> = set external processing of service discovery" },
2916 { "p2p_service_flush", wpa_cli_cmd_p2p_service_flush, NULL,
2917 cli_cmd_flag_none,
2918 "= remove all stored service entries" },
2919 { "p2p_service_add", wpa_cli_cmd_p2p_service_add, NULL,
2920 cli_cmd_flag_none,
2921 "<bonjour|upnp> <query|version> <response|service> = add a local "
2922 "service" },
2923 { "p2p_service_del", wpa_cli_cmd_p2p_service_del, NULL,
2924 cli_cmd_flag_none,
2925 "<bonjour|upnp> <query|version> [|service] = remove a local "
2926 "service" },
2927 { "p2p_reject", wpa_cli_cmd_p2p_reject, wpa_cli_complete_p2p_peer,
2928 cli_cmd_flag_none,
2929 "<addr> = reject connection attempts from a specific peer" },
2930 { "p2p_invite", wpa_cli_cmd_p2p_invite, NULL,
2931 cli_cmd_flag_none,
2932 "<cmd> [peer=addr] = invite peer" },
2933 { "p2p_peers", wpa_cli_cmd_p2p_peers, NULL, cli_cmd_flag_none,
2934 "[discovered] = list known (optionally, only fully discovered) P2P "
2935 "peers" },
2936 { "p2p_peer", wpa_cli_cmd_p2p_peer, wpa_cli_complete_p2p_peer,
2937 cli_cmd_flag_none,
2938 "<address> = show information about known P2P peer" },
2939 { "p2p_set", wpa_cli_cmd_p2p_set, wpa_cli_complete_p2p_set,
2940 cli_cmd_flag_none,
2941 "<field> <value> = set a P2P parameter" },
2942 { "p2p_flush", wpa_cli_cmd_p2p_flush, NULL, cli_cmd_flag_none,
2943 "= flush P2P state" },
2944 { "p2p_cancel", wpa_cli_cmd_p2p_cancel, NULL, cli_cmd_flag_none,
2945 "= cancel P2P group formation" },
2946 { "p2p_unauthorize", wpa_cli_cmd_p2p_unauthorize,
2947 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
2948 "<address> = unauthorize a peer" },
2949 { "p2p_presence_req", wpa_cli_cmd_p2p_presence_req, NULL,
2950 cli_cmd_flag_none,
2951 "[<duration> <interval>] [<duration> <interval>] = request GO "
2952 "presence" },
2953 { "p2p_ext_listen", wpa_cli_cmd_p2p_ext_listen, NULL,
2954 cli_cmd_flag_none,
2955 "[<period> <interval>] = set extended listen timing" },
2956 { "p2p_remove_client", wpa_cli_cmd_p2p_remove_client,
2957 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
2958 "<address|iface=address> = remove a peer from all groups" },
2959 #endif /* CONFIG_P2P */
2960 #ifdef CONFIG_WIFI_DISPLAY
2961 { "wfd_subelem_set", wpa_cli_cmd_wfd_subelem_set, NULL,
2962 cli_cmd_flag_none,
2963 "<subelem> [contents] = set Wi-Fi Display subelement" },
2964 { "wfd_subelem_get", wpa_cli_cmd_wfd_subelem_get, NULL,
2965 cli_cmd_flag_none,
2966 "<subelem> = get Wi-Fi Display subelement" },
2967 #endif /* CONFIG_WIFI_DISPLAY */
2968 #ifdef CONFIG_INTERWORKING
2969 { "fetch_anqp", wpa_cli_cmd_fetch_anqp, NULL, cli_cmd_flag_none,
2970 "= fetch ANQP information for all APs" },
2971 { "stop_fetch_anqp", wpa_cli_cmd_stop_fetch_anqp, NULL,
2972 cli_cmd_flag_none,
2973 "= stop fetch_anqp operation" },
2974 { "interworking_select", wpa_cli_cmd_interworking_select, NULL,
2975 cli_cmd_flag_none,
2976 "[auto] = perform Interworking network selection" },
2977 { "interworking_connect", wpa_cli_cmd_interworking_connect,
2978 wpa_cli_complete_bss, cli_cmd_flag_none,
2979 "<BSSID> = connect using Interworking credentials" },
2980 { "anqp_get", wpa_cli_cmd_anqp_get, wpa_cli_complete_bss,
2981 cli_cmd_flag_none,
2982 "<addr> <info id>[,<info id>]... = request ANQP information" },
2983 { "gas_request", wpa_cli_cmd_gas_request, wpa_cli_complete_bss,
2984 cli_cmd_flag_none,
2985 "<addr> <AdvProtoID> [QueryReq] = GAS request" },
2986 { "gas_response_get", wpa_cli_cmd_gas_response_get,
2987 wpa_cli_complete_bss, cli_cmd_flag_none,
2988 "<addr> <dialog token> [start,len] = Fetch last GAS response" },
2989 #endif /* CONFIG_INTERWORKING */
2990 #ifdef CONFIG_HS20
2991 { "hs20_anqp_get", wpa_cli_cmd_hs20_anqp_get, wpa_cli_complete_bss,
2992 cli_cmd_flag_none,
2993 "<addr> <subtype>[,<subtype>]... = request HS 2.0 ANQP information"
2994 },
2995 { "nai_home_realm_list", wpa_cli_cmd_get_nai_home_realm_list,
2996 wpa_cli_complete_bss, cli_cmd_flag_none,
2997 "<addr> <home realm> = get HS20 nai home realm list" },
2998 { "hs20_icon_request", wpa_cli_cmd_hs20_icon_request,
2999 wpa_cli_complete_bss, cli_cmd_flag_none,
3000 "<addr> <icon name> = get Hotspot 2.0 OSU icon" },
3001 { "fetch_osu", wpa_cli_cmd_fetch_osu, NULL, cli_cmd_flag_none,
3002 "= fetch OSU provider information from all APs" },
3003 { "cancel_fetch_osu", wpa_cli_cmd_cancel_fetch_osu, NULL,
3004 cli_cmd_flag_none,
3005 "= cancel fetch_osu command" },
3006 #endif /* CONFIG_HS20 */
3007 { "sta_autoconnect", wpa_cli_cmd_sta_autoconnect, NULL,
3008 cli_cmd_flag_none,
3009 "<0/1> = disable/enable automatic reconnection" },
3010 { "tdls_discover", wpa_cli_cmd_tdls_discover, NULL,
3011 cli_cmd_flag_none,
3012 "<addr> = request TDLS discovery with <addr>" },
3013 { "tdls_setup", wpa_cli_cmd_tdls_setup, NULL,
3014 cli_cmd_flag_none,
3015 "<addr> = request TDLS setup with <addr>" },
3016 { "tdls_teardown", wpa_cli_cmd_tdls_teardown, NULL,
3017 cli_cmd_flag_none,
3018 "<addr> = tear down TDLS with <addr>" },
3019 { "wmm_ac_addts", wpa_cli_cmd_wmm_ac_addts, NULL,
3020 cli_cmd_flag_none,
3021 "<uplink/downlink/bidi> <tsid=0..7> <up=0..7> [nominal_msdu_size=#] "
3022 "[mean_data_rate=#] [min_phy_rate=#] [sba=#] [fixed_nominal_msdu] "
3023 "= add WMM-AC traffic stream" },
3024 { "wmm_ac_delts", wpa_cli_cmd_wmm_ac_delts, NULL,
3025 cli_cmd_flag_none,
3026 "<tsid> = delete WMM-AC traffic stream" },
3027 { "wmm_ac_status", wpa_cli_cmd_wmm_ac_status, NULL,
3028 cli_cmd_flag_none,
3029 "= show status for Wireless Multi-Media Admission-Control" },
3030 { "tdls_chan_switch", wpa_cli_cmd_tdls_chan_switch, NULL,
3031 cli_cmd_flag_none,
3032 "<addr> <oper class> <freq> [sec_channel_offset=] [center_freq1=] "
3033 "[center_freq2=] [bandwidth=] [ht|vht] = enable channel switching "
3034 "with TDLS peer" },
3035 { "tdls_cancel_chan_switch", wpa_cli_cmd_tdls_cancel_chan_switch, NULL,
3036 cli_cmd_flag_none,
3037 "<addr> = disable channel switching with TDLS peer <addr>" },
3038 { "signal_poll", wpa_cli_cmd_signal_poll, NULL,
3039 cli_cmd_flag_none,
3040 "= get signal parameters" },
3041 { "pktcnt_poll", wpa_cli_cmd_pktcnt_poll, NULL,
3042 cli_cmd_flag_none,
3043 "= get TX/RX packet counters" },
3044 { "reauthenticate", wpa_cli_cmd_reauthenticate, NULL,
3045 cli_cmd_flag_none,
3046 "= trigger IEEE 802.1X/EAPOL reauthentication" },
3047 #ifdef CONFIG_AUTOSCAN
3048 { "autoscan", wpa_cli_cmd_autoscan, NULL, cli_cmd_flag_none,
3049 "[params] = Set or unset (if none) autoscan parameters" },
3050 #endif /* CONFIG_AUTOSCAN */
3051 #ifdef CONFIG_WNM
3052 { "wnm_sleep", wpa_cli_cmd_wnm_sleep, NULL, cli_cmd_flag_none,
3053 "<enter/exit> [interval=#] = enter/exit WNM-Sleep mode" },
3054 { "wnm_bss_query", wpa_cli_cmd_wnm_bss_query, NULL, cli_cmd_flag_none,
3055 "<query reason> = Send BSS Transition Management Query" },
3056 #endif /* CONFIG_WNM */
3057 { "raw", wpa_cli_cmd_raw, NULL, cli_cmd_flag_sensitive,
3058 "<params..> = Sent unprocessed command" },
3059 { "flush", wpa_cli_cmd_flush, NULL, cli_cmd_flag_none,
3060 "= flush wpa_supplicant state" },
3061 #ifdef ANDROID
3062 { "driver", wpa_cli_cmd_driver, NULL, cli_cmd_flag_none,
3063 "<command> = driver private commands" },
3064 #endif /* ANDROID */
3065 { "radio_work", wpa_cli_cmd_radio_work, NULL, cli_cmd_flag_none,
3066 "= radio_work <show/add/done>" },
3067 { "vendor", wpa_cli_cmd_vendor, NULL, cli_cmd_flag_none,
3068 "<vendor id> <command id> [<hex formatted command argument>] = Send vendor command"
3069 },
3070 { "neighbor_rep_request",
3071 wpa_cli_cmd_neighbor_rep_request, NULL, cli_cmd_flag_none,
3072 "[ssid=<SSID>] = Trigger request to AP for neighboring AP report "
3073 "(with optional given SSID, default: current SSID)"
3074 },
3075 { "erp_flush", wpa_cli_cmd_erp_flush, NULL, cli_cmd_flag_none,
3076 "= flush ERP keys" },
3077 { "mac_rand_scan",
3078 wpa_cli_cmd_mac_rand_scan, NULL, cli_cmd_flag_none,
3079 "<scan|sched|pno|all> enable=<0/1> [addr=mac-address "
3080 "mask=mac-address-mask] = scan MAC randomization"
3081 },
3082 { NULL, NULL, NULL, cli_cmd_flag_none, NULL }
3083 };
3084
3085
3086 /*
3087 * Prints command usage, lines are padded with the specified string.
3088 */
3089 static void print_cmd_help(struct wpa_cli_cmd *cmd, const char *pad)
3090 {
3091 char c;
3092 size_t n;
3093
3094 printf("%s%s ", pad, cmd->cmd);
3095 for (n = 0; (c = cmd->usage[n]); n++) {
3096 printf("%c", c);
3097 if (c == '\n')
3098 printf("%s", pad);
3099 }
3100 printf("\n");
3101 }
3102
3103
3104 static void print_help(const char *cmd)
3105 {
3106 int n;
3107 printf("commands:\n");
3108 for (n = 0; wpa_cli_commands[n].cmd; n++) {
3109 if (cmd == NULL || str_starts(wpa_cli_commands[n].cmd, cmd))
3110 print_cmd_help(&wpa_cli_commands[n], " ");
3111 }
3112 }
3113
3114
3115 static int wpa_cli_edit_filter_history_cb(void *ctx, const char *cmd)
3116 {
3117 const char *c, *delim;
3118 int n;
3119 size_t len;
3120
3121 delim = os_strchr(cmd, ' ');
3122 if (delim)
3123 len = delim - cmd;
3124 else
3125 len = os_strlen(cmd);
3126
3127 for (n = 0; (c = wpa_cli_commands[n].cmd); n++) {
3128 if (os_strncasecmp(cmd, c, len) == 0 && len == os_strlen(c))
3129 return (wpa_cli_commands[n].flags &
3130 cli_cmd_flag_sensitive);
3131 }
3132 return 0;
3133 }
3134
3135
3136 static char ** wpa_list_cmd_list(void)
3137 {
3138 char **res;
3139 int i, count;
3140 struct cli_txt_entry *e;
3141
3142 count = ARRAY_SIZE(wpa_cli_commands);
3143 count += dl_list_len(&p2p_groups);
3144 count += dl_list_len(&ifnames);
3145 res = os_calloc(count + 1, sizeof(char *));
3146 if (res == NULL)
3147 return NULL;
3148
3149 for (i = 0; wpa_cli_commands[i].cmd; i++) {
3150 res[i] = os_strdup(wpa_cli_commands[i].cmd);
3151 if (res[i] == NULL)
3152 break;
3153 }
3154
3155 dl_list_for_each(e, &p2p_groups, struct cli_txt_entry, list) {
3156 size_t len = 8 + os_strlen(e->txt);
3157 res[i] = os_malloc(len);
3158 if (res[i] == NULL)
3159 break;
3160 os_snprintf(res[i], len, "ifname=%s", e->txt);
3161 i++;
3162 }
3163
3164 dl_list_for_each(e, &ifnames, struct cli_txt_entry, list) {
3165 res[i] = os_strdup(e->txt);
3166 if (res[i] == NULL)
3167 break;
3168 i++;
3169 }
3170
3171 return res;
3172 }
3173
3174
3175 static char ** wpa_cli_cmd_completion(const char *cmd, const char *str,
3176 int pos)
3177 {
3178 int i;
3179
3180 for (i = 0; wpa_cli_commands[i].cmd; i++) {
3181 if (os_strcasecmp(wpa_cli_commands[i].cmd, cmd) == 0) {
3182 if (wpa_cli_commands[i].completion)
3183 return wpa_cli_commands[i].completion(str,
3184 pos);
3185 edit_clear_line();
3186 printf("\r%s\n", wpa_cli_commands[i].usage);
3187 edit_redraw();
3188 break;
3189 }
3190 }
3191
3192 return NULL;
3193 }
3194
3195
3196 static char ** wpa_cli_edit_completion_cb(void *ctx, const char *str, int pos)
3197 {
3198 char **res;
3199 const char *end;
3200 char *cmd;
3201
3202 if (pos > 7 && os_strncasecmp(str, "IFNAME=", 7) == 0) {
3203 end = os_strchr(str, ' ');
3204 if (end && pos > end - str) {
3205 pos -= end - str + 1;
3206 str = end + 1;
3207 }
3208 }
3209
3210 end = os_strchr(str, ' ');
3211 if (end == NULL || str + pos < end)
3212 return wpa_list_cmd_list();
3213
3214 cmd = os_malloc(pos + 1);
3215 if (cmd == NULL)
3216 return NULL;
3217 os_memcpy(cmd, str, pos);
3218 cmd[end - str] = '\0';
3219 res = wpa_cli_cmd_completion(cmd, str, pos);
3220 os_free(cmd);
3221 return res;
3222 }
3223
3224
3225 static int wpa_request(struct wpa_ctrl *ctrl, int argc, char *argv[])
3226 {
3227 struct wpa_cli_cmd *cmd, *match = NULL;
3228 int count;
3229 int ret = 0;
3230
3231 if (argc > 1 && os_strncasecmp(argv[0], "IFNAME=", 7) == 0) {
3232 ifname_prefix = argv[0] + 7;
3233 argv = &argv[1];
3234 argc--;
3235 } else
3236 ifname_prefix = NULL;
3237
3238 if (argc == 0)
3239 return -1;
3240
3241 count = 0;
3242 cmd = wpa_cli_commands;
3243 while (cmd->cmd) {
3244 if (os_strncasecmp(cmd->cmd, argv[0], os_strlen(argv[0])) == 0)
3245 {
3246 match = cmd;
3247 if (os_strcasecmp(cmd->cmd, argv[0]) == 0) {
3248 /* we have an exact match */
3249 count = 1;
3250 break;
3251 }
3252 count++;
3253 }
3254 cmd++;
3255 }
3256
3257 if (count > 1) {
3258 printf("Ambiguous command '%s'; possible commands:", argv[0]);
3259 cmd = wpa_cli_commands;
3260 while (cmd->cmd) {
3261 if (os_strncasecmp(cmd->cmd, argv[0],
3262 os_strlen(argv[0])) == 0) {
3263 printf(" %s", cmd->cmd);
3264 }
3265 cmd++;
3266 }
3267 printf("\n");
3268 ret = 1;
3269 } else if (count == 0) {
3270 printf("Unknown command '%s'\n", argv[0]);
3271 ret = 1;
3272 } else {
3273 ret = match->handler(ctrl, argc - 1, &argv[1]);
3274 }
3275
3276 return ret;
3277 }
3278
3279
3280 static int str_match(const char *a, const char *b)
3281 {
3282 return os_strncmp(a, b, os_strlen(b)) == 0;
3283 }
3284
3285
3286 static int wpa_cli_exec(const char *program, const char *arg1,
3287 const char *arg2)
3288 {
3289 char *arg;
3290 size_t len;
3291 int res;
3292
3293 len = os_strlen(arg1) + os_strlen(arg2) + 2;
3294 arg = os_malloc(len);
3295 if (arg == NULL)
3296 return -1;
3297 os_snprintf(arg, len, "%s %s", arg1, arg2);
3298 res = os_exec(program, arg, 1);
3299 os_free(arg);
3300
3301 return res;
3302 }
3303
3304
3305 static void wpa_cli_action_process(const char *msg)
3306 {
3307 const char *pos;
3308 char *copy = NULL, *id, *pos2;
3309 const char *ifname = ctrl_ifname;
3310 char ifname_buf[100];
3311
3312 pos = msg;
3313 if (os_strncmp(pos, "IFNAME=", 7) == 0) {
3314 const char *end;
3315 end = os_strchr(pos + 7, ' ');
3316 if (end && (unsigned int) (end - pos) < sizeof(ifname_buf)) {
3317 pos += 7;
3318 os_memcpy(ifname_buf, pos, end - pos);
3319 ifname_buf[end - pos] = '\0';
3320 ifname = ifname_buf;
3321 pos = end + 1;
3322 }
3323 }
3324 if (*pos == '<') {
3325 const char *prev = pos;
3326 /* skip priority */
3327 pos = os_strchr(pos, '>');
3328 if (pos)
3329 pos++;
3330 else
3331 pos = prev;
3332 }
3333
3334 if (str_match(pos, WPA_EVENT_CONNECTED)) {
3335 int new_id = -1;
3336 os_unsetenv("WPA_ID");
3337 os_unsetenv("WPA_ID_STR");
3338 os_unsetenv("WPA_CTRL_DIR");
3339
3340 pos = os_strstr(pos, "[id=");
3341 if (pos)
3342 copy = os_strdup(pos + 4);
3343
3344 if (copy) {
3345 pos2 = id = copy;
3346 while (*pos2 && *pos2 != ' ')
3347 pos2++;
3348 *pos2++ = '\0';
3349 new_id = atoi(id);
3350 os_setenv("WPA_ID", id, 1);
3351 while (*pos2 && *pos2 != '=')
3352 pos2++;
3353 if (*pos2 == '=')
3354 pos2++;
3355 id = pos2;
3356 while (*pos2 && *pos2 != ']')
3357 pos2++;
3358 *pos2 = '\0';
3359 os_setenv("WPA_ID_STR", id, 1);
3360 os_free(copy);
3361 }
3362
3363 os_setenv("WPA_CTRL_DIR", ctrl_iface_dir, 1);
3364
3365 if (wpa_cli_connected <= 0 || new_id != wpa_cli_last_id) {
3366 wpa_cli_connected = 1;
3367 wpa_cli_last_id = new_id;
3368 wpa_cli_exec(action_file, ifname, "CONNECTED");
3369 }
3370 } else if (str_match(pos, WPA_EVENT_DISCONNECTED)) {
3371 if (wpa_cli_connected) {
3372 wpa_cli_connected = 0;
3373 wpa_cli_exec(action_file, ifname, "DISCONNECTED");
3374 }
3375 } else if (str_match(pos, MESH_GROUP_STARTED)) {
3376 wpa_cli_exec(action_file, ctrl_ifname, pos);
3377 } else if (str_match(pos, MESH_GROUP_REMOVED)) {
3378 wpa_cli_exec(action_file, ctrl_ifname, pos);
3379 } else if (str_match(pos, MESH_PEER_CONNECTED)) {
3380 wpa_cli_exec(action_file, ctrl_ifname, pos);
3381 } else if (str_match(pos, MESH_PEER_DISCONNECTED)) {
3382 wpa_cli_exec(action_file, ctrl_ifname, pos);
3383 } else if (str_match(pos, P2P_EVENT_GROUP_STARTED)) {
3384 wpa_cli_exec(action_file, ifname, pos);
3385 } else if (str_match(pos, P2P_EVENT_GROUP_REMOVED)) {
3386 wpa_cli_exec(action_file, ifname, pos);
3387 } else if (str_match(pos, P2P_EVENT_CROSS_CONNECT_ENABLE)) {
3388 wpa_cli_exec(action_file, ifname, pos);
3389 } else if (str_match(pos, P2P_EVENT_CROSS_CONNECT_DISABLE)) {
3390 wpa_cli_exec(action_file, ifname, pos);
3391 } else if (str_match(pos, P2P_EVENT_GO_NEG_FAILURE)) {
3392 wpa_cli_exec(action_file, ifname, pos);
3393 } else if (str_match(pos, WPS_EVENT_SUCCESS)) {
3394 wpa_cli_exec(action_file, ifname, pos);
3395 } else if (str_match(pos, WPS_EVENT_FAIL)) {
3396 wpa_cli_exec(action_file, ifname, pos);
3397 } else if (str_match(pos, AP_STA_CONNECTED)) {
3398 wpa_cli_exec(action_file, ifname, pos);
3399 } else if (str_match(pos, AP_STA_DISCONNECTED)) {
3400 wpa_cli_exec(action_file, ifname, pos);
3401 } else if (str_match(pos, ESS_DISASSOC_IMMINENT)) {
3402 wpa_cli_exec(action_file, ifname, pos);
3403 } else if (str_match(pos, HS20_SUBSCRIPTION_REMEDIATION)) {
3404 wpa_cli_exec(action_file, ifname, pos);
3405 } else if (str_match(pos, HS20_DEAUTH_IMMINENT_NOTICE)) {
3406 wpa_cli_exec(action_file, ifname, pos);
3407 } else if (str_match(pos, WPA_EVENT_TERMINATING)) {
3408 printf("wpa_supplicant is terminating - stop monitoring\n");
3409 wpa_cli_quit = 1;
3410 }
3411 }
3412
3413
3414 #ifndef CONFIG_ANSI_C_EXTRA
3415 static void wpa_cli_action_cb(char *msg, size_t len)
3416 {
3417 wpa_cli_action_process(msg);
3418 }
3419 #endif /* CONFIG_ANSI_C_EXTRA */
3420
3421
3422 static void wpa_cli_reconnect(void)
3423 {
3424 wpa_cli_close_connection();
3425 if (wpa_cli_open_connection(ctrl_ifname, 1) < 0)
3426 return;
3427
3428 if (interactive) {
3429 edit_clear_line();
3430 printf("\rConnection to wpa_supplicant re-established\n");
3431 edit_redraw();
3432 }
3433 }
3434
3435
3436 static void cli_event(const char *str)
3437 {
3438 const char *start, *s;
3439
3440 start = os_strchr(str, '>');
3441 if (start == NULL)
3442 return;
3443
3444 start++;
3445
3446 if (str_starts(start, WPA_EVENT_BSS_ADDED)) {
3447 s = os_strchr(start, ' ');
3448 if (s == NULL)
3449 return;
3450 s = os_strchr(s + 1, ' ');
3451 if (s == NULL)
3452 return;
3453 cli_txt_list_add(&bsses, s + 1);
3454 return;
3455 }
3456
3457 if (str_starts(start, WPA_EVENT_BSS_REMOVED)) {
3458 s = os_strchr(start, ' ');
3459 if (s == NULL)
3460 return;
3461 s = os_strchr(s + 1, ' ');
3462 if (s == NULL)
3463 return;
3464 cli_txt_list_del_addr(&bsses, s + 1);
3465 return;
3466 }
3467
3468 #ifdef CONFIG_P2P
3469 if (str_starts(start, P2P_EVENT_DEVICE_FOUND)) {
3470 s = os_strstr(start, " p2p_dev_addr=");
3471 if (s == NULL)
3472 return;
3473 cli_txt_list_add_addr(&p2p_peers, s + 14);
3474 return;
3475 }
3476
3477 if (str_starts(start, P2P_EVENT_DEVICE_LOST)) {
3478 s = os_strstr(start, " p2p_dev_addr=");
3479 if (s == NULL)
3480 return;
3481 cli_txt_list_del_addr(&p2p_peers, s + 14);
3482 return;
3483 }
3484
3485 if (str_starts(start, P2P_EVENT_GROUP_STARTED)) {
3486 s = os_strchr(start, ' ');
3487 if (s == NULL)
3488 return;
3489 cli_txt_list_add_word(&p2p_groups, s + 1);
3490 return;
3491 }
3492
3493 if (str_starts(start, P2P_EVENT_GROUP_REMOVED)) {
3494 s = os_strchr(start, ' ');
3495 if (s == NULL)
3496 return;
3497 cli_txt_list_del_word(&p2p_groups, s + 1);
3498 return;
3499 }
3500 #endif /* CONFIG_P2P */
3501 }
3502
3503
3504 static int check_terminating(const char *msg)
3505 {
3506 const char *pos = msg;
3507
3508 if (*pos == '<') {
3509 /* skip priority */
3510 pos = os_strchr(pos, '>');
3511 if (pos)
3512 pos++;
3513 else
3514 pos = msg;
3515 }
3516
3517 if (str_match(pos, WPA_EVENT_TERMINATING) && ctrl_conn) {
3518 edit_clear_line();
3519 printf("\rConnection to wpa_supplicant lost - trying to "
3520 "reconnect\n");
3521 edit_redraw();
3522 wpa_cli_attached = 0;
3523 wpa_cli_close_connection();
3524 return 1;
3525 }
3526
3527 return 0;
3528 }
3529
3530
3531 static void wpa_cli_recv_pending(struct wpa_ctrl *ctrl, int action_monitor)
3532 {
3533 if (ctrl_conn == NULL) {
3534 wpa_cli_reconnect();
3535 return;
3536 }
3537 while (wpa_ctrl_pending(ctrl) > 0) {
3538 char buf[4096];
3539 size_t len = sizeof(buf) - 1;
3540 if (wpa_ctrl_recv(ctrl, buf, &len) == 0) {
3541 buf[len] = '\0';
3542 if (action_monitor)
3543 wpa_cli_action_process(buf);
3544 else {
3545 cli_event(buf);
3546 if (wpa_cli_show_event(buf)) {
3547 edit_clear_line();
3548 printf("\r%s\n", buf);
3549 edit_redraw();
3550 }
3551
3552 if (interactive && check_terminating(buf) > 0)
3553 return;
3554 }
3555 } else {
3556 printf("Could not read pending message.\n");
3557 break;
3558 }
3559 }
3560
3561 if (wpa_ctrl_pending(ctrl) < 0) {
3562 printf("Connection to wpa_supplicant lost - trying to "
3563 "reconnect\n");
3564 wpa_cli_reconnect();
3565 }
3566 }
3567
3568 #define max_args 10
3569
3570 static int tokenize_cmd(char *cmd, char *argv[])
3571 {
3572 char *pos;
3573 int argc = 0;
3574
3575 pos = cmd;
3576 for (;;) {
3577 while (*pos == ' ')
3578 pos++;
3579 if (*pos == '\0')
3580 break;
3581 argv[argc] = pos;
3582 argc++;
3583 if (argc == max_args)
3584 break;
3585 if (*pos == '"') {
3586 char *pos2 = os_strrchr(pos, '"');
3587 if (pos2)
3588 pos = pos2 + 1;
3589 }
3590 while (*pos != '\0' && *pos != ' ')
3591 pos++;
3592 if (*pos == ' ')
3593 *pos++ = '\0';
3594 }
3595
3596 return argc;
3597 }
3598
3599
3600 static void wpa_cli_ping(void *eloop_ctx, void *timeout_ctx)
3601 {
3602 if (ctrl_conn) {
3603 int res;
3604 char *prefix = ifname_prefix;
3605
3606 ifname_prefix = NULL;
3607 res = _wpa_ctrl_command(ctrl_conn, "PING", 0);
3608 ifname_prefix = prefix;
3609 if (res) {
3610 printf("Connection to wpa_supplicant lost - trying to "
3611 "reconnect\n");
3612 wpa_cli_close_connection();
3613 }
3614 }
3615 if (!ctrl_conn)
3616 wpa_cli_reconnect();
3617 eloop_register_timeout(ping_interval, 0, wpa_cli_ping, NULL, NULL);
3618 }
3619
3620
3621 static void wpa_cli_mon_receive(int sock, void *eloop_ctx, void *sock_ctx)
3622 {
3623 wpa_cli_recv_pending(mon_conn, 0);
3624 }
3625
3626
3627 static void wpa_cli_edit_cmd_cb(void *ctx, char *cmd)
3628 {
3629 char *argv[max_args];
3630 int argc;
3631 argc = tokenize_cmd(cmd, argv);
3632 if (argc)
3633 wpa_request(ctrl_conn, argc, argv);
3634 }
3635
3636
3637 static void wpa_cli_edit_eof_cb(void *ctx)
3638 {
3639 eloop_terminate();
3640 }
3641
3642
3643 static int warning_displayed = 0;
3644 static char *hfile = NULL;
3645 static int edit_started = 0;
3646
3647 static void start_edit(void)
3648 {
3649 char *home;
3650 char *ps = NULL;
3651
3652 #ifdef CONFIG_CTRL_IFACE_UDP_REMOTE
3653 ps = wpa_ctrl_get_remote_ifname(ctrl_conn);
3654 #endif /* CONFIG_CTRL_IFACE_UDP_REMOTE */
3655
3656 home = getenv("HOME");
3657 if (home) {
3658 const char *fname = ".wpa_cli_history";
3659 int hfile_len = os_strlen(home) + 1 + os_strlen(fname) + 1;
3660 hfile = os_malloc(hfile_len);
3661 if (hfile)
3662 os_snprintf(hfile, hfile_len, "%s/%s", home, fname);
3663 }
3664
3665 if (edit_init(wpa_cli_edit_cmd_cb, wpa_cli_edit_eof_cb,
3666 wpa_cli_edit_completion_cb, NULL, hfile, ps) < 0) {
3667 eloop_terminate();
3668 return;
3669 }
3670
3671 edit_started = 1;
3672 eloop_register_timeout(ping_interval, 0, wpa_cli_ping, NULL, NULL);
3673 }
3674
3675
3676 static void update_bssid_list(struct wpa_ctrl *ctrl)
3677 {
3678 char buf[4096];
3679 size_t len = sizeof(buf);
3680 int ret;
3681 char *cmd = "BSS RANGE=ALL MASK=0x2";
3682 char *pos, *end;
3683
3684 if (ctrl == NULL)
3685 return;
3686 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len, NULL);
3687 if (ret < 0)
3688 return;
3689 buf[len] = '\0';
3690
3691 pos = buf;
3692 while (pos) {
3693 pos = os_strstr(pos, "bssid=");
3694 if (pos == NULL)
3695 break;
3696 pos += 6;
3697 end = os_strchr(pos, '\n');
3698 if (end == NULL)
3699 break;
3700 *end = '\0';
3701 cli_txt_list_add(&bsses, pos);
3702 pos = end + 1;
3703 }
3704 }
3705
3706
3707 static void update_ifnames(struct wpa_ctrl *ctrl)
3708 {
3709 char buf[4096];
3710 size_t len = sizeof(buf);
3711 int ret;
3712 char *cmd = "INTERFACES";
3713 char *pos, *end;
3714 char txt[200];
3715
3716 cli_txt_list_flush(&ifnames);
3717
3718 if (ctrl == NULL)
3719 return;
3720 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len, NULL);
3721 if (ret < 0)
3722 return;
3723 buf[len] = '\0';
3724
3725 pos = buf;
3726 while (pos) {
3727 end = os_strchr(pos, '\n');
3728 if (end == NULL)
3729 break;
3730 *end = '\0';
3731 ret = os_snprintf(txt, sizeof(txt), "ifname=%s", pos);
3732 if (!os_snprintf_error(sizeof(txt), ret))
3733 cli_txt_list_add(&ifnames, txt);
3734 pos = end + 1;
3735 }
3736 }
3737
3738
3739 static void try_connection(void *eloop_ctx, void *timeout_ctx)
3740 {
3741 if (ctrl_conn)
3742 goto done;
3743
3744 if (ctrl_ifname == NULL)
3745 ctrl_ifname = wpa_cli_get_default_ifname();
3746
3747 if (!wpa_cli_open_connection(ctrl_ifname, 1) == 0) {
3748 if (!warning_displayed) {
3749 printf("Could not connect to wpa_supplicant: "
3750 "%s - re-trying\n",
3751 ctrl_ifname ? ctrl_ifname : "(nil)");
3752 warning_displayed = 1;
3753 }
3754 eloop_register_timeout(1, 0, try_connection, NULL, NULL);
3755 return;
3756 }
3757
3758 update_bssid_list(ctrl_conn);
3759
3760 if (warning_displayed)
3761 printf("Connection established.\n");
3762
3763 done:
3764 start_edit();
3765 }
3766
3767
3768 static void wpa_cli_interactive(void)
3769 {
3770 printf("\nInteractive mode\n\n");
3771
3772 eloop_register_timeout(0, 0, try_connection, NULL, NULL);
3773 eloop_run();
3774 eloop_cancel_timeout(try_connection, NULL, NULL);
3775
3776 cli_txt_list_flush(&p2p_peers);
3777 cli_txt_list_flush(&p2p_groups);
3778 cli_txt_list_flush(&bsses);
3779 cli_txt_list_flush(&ifnames);
3780 if (edit_started)
3781 edit_deinit(hfile, wpa_cli_edit_filter_history_cb);
3782 os_free(hfile);
3783 eloop_cancel_timeout(wpa_cli_ping, NULL, NULL);
3784 wpa_cli_close_connection();
3785 }
3786
3787
3788 static void wpa_cli_action(struct wpa_ctrl *ctrl)
3789 {
3790 #ifdef CONFIG_ANSI_C_EXTRA
3791 /* TODO: ANSI C version(?) */
3792 printf("Action processing not supported in ANSI C build.\n");
3793 #else /* CONFIG_ANSI_C_EXTRA */
3794 fd_set rfds;
3795 int fd, res;
3796 struct timeval tv;
3797 char buf[256]; /* note: large enough to fit in unsolicited messages */
3798 size_t len;
3799
3800 fd = wpa_ctrl_get_fd(ctrl);
3801
3802 while (!wpa_cli_quit) {
3803 FD_ZERO(&rfds);
3804 FD_SET(fd, &rfds);
3805 tv.tv_sec = ping_interval;
3806 tv.tv_usec = 0;
3807 res = select(fd + 1, &rfds, NULL, NULL, &tv);
3808 if (res < 0 && errno != EINTR) {
3809 perror("select");
3810 break;
3811 }
3812
3813 if (FD_ISSET(fd, &rfds))
3814 wpa_cli_recv_pending(ctrl, 1);
3815 else {
3816 /* verify that connection is still working */
3817 len = sizeof(buf) - 1;
3818 if (wpa_ctrl_request(ctrl, "PING", 4, buf, &len,
3819 wpa_cli_action_cb) < 0 ||
3820 len < 4 || os_memcmp(buf, "PONG", 4) != 0) {
3821 printf("wpa_supplicant did not reply to PING "
3822 "command - exiting\n");
3823 break;
3824 }
3825 }
3826 }
3827 #endif /* CONFIG_ANSI_C_EXTRA */
3828 }
3829
3830
3831 static void wpa_cli_cleanup(void)
3832 {
3833 wpa_cli_close_connection();
3834 if (pid_file)
3835 os_daemonize_terminate(pid_file);
3836
3837 os_program_deinit();
3838 }
3839
3840
3841 static void wpa_cli_terminate(int sig, void *ctx)
3842 {
3843 eloop_terminate();
3844 }
3845
3846
3847 static char * wpa_cli_get_default_ifname(void)
3848 {
3849 char *ifname = NULL;
3850
3851 #ifdef CONFIG_CTRL_IFACE_UNIX
3852 struct dirent *dent;
3853 DIR *dir = opendir(ctrl_iface_dir);
3854 if (!dir) {
3855 #ifdef ANDROID
3856 char ifprop[PROPERTY_VALUE_MAX];
3857 if (property_get("wifi.interface", ifprop, NULL) != 0) {
3858 ifname = os_strdup(ifprop);
3859 printf("Using interface '%s'\n", ifname);
3860 return ifname;
3861 }
3862 #endif /* ANDROID */
3863 return NULL;
3864 }
3865 while ((dent = readdir(dir))) {
3866 #ifdef _DIRENT_HAVE_D_TYPE
3867 /*
3868 * Skip the file if it is not a socket. Also accept
3869 * DT_UNKNOWN (0) in case the C library or underlying
3870 * file system does not support d_type.
3871 */
3872 if (dent->d_type != DT_SOCK && dent->d_type != DT_UNKNOWN)
3873 continue;
3874 #endif /* _DIRENT_HAVE_D_TYPE */
3875 if (os_strcmp(dent->d_name, ".") == 0 ||
3876 os_strcmp(dent->d_name, "..") == 0)
3877 continue;
3878 printf("Selected interface '%s'\n", dent->d_name);
3879 ifname = os_strdup(dent->d_name);
3880 break;
3881 }
3882 closedir(dir);
3883 #endif /* CONFIG_CTRL_IFACE_UNIX */
3884
3885 #ifdef CONFIG_CTRL_IFACE_NAMED_PIPE
3886 char buf[4096], *pos;
3887 size_t len;
3888 struct wpa_ctrl *ctrl;
3889 int ret;
3890
3891 ctrl = wpa_ctrl_open(NULL);
3892 if (ctrl == NULL)
3893 return NULL;
3894
3895 len = sizeof(buf) - 1;
3896 ret = wpa_ctrl_request(ctrl, "INTERFACES", 10, buf, &len, NULL);
3897 if (ret >= 0) {
3898 buf[len] = '\0';
3899 pos = os_strchr(buf, '\n');
3900 if (pos)
3901 *pos = '\0';
3902 ifname = os_strdup(buf);
3903 }
3904 wpa_ctrl_close(ctrl);
3905 #endif /* CONFIG_CTRL_IFACE_NAMED_PIPE */
3906
3907 return ifname;
3908 }
3909
3910
3911 int main(int argc, char *argv[])
3912 {
3913 int c;
3914 int daemonize = 0;
3915 int ret = 0;
3916 const char *global = NULL;
3917
3918 if (os_program_init())
3919 return -1;
3920
3921 for (;;) {
3922 c = getopt(argc, argv, "a:Bg:G:hi:p:P:v");
3923 if (c < 0)
3924 break;
3925 switch (c) {
3926 case 'a':
3927 action_file = optarg;
3928 break;
3929 case 'B':
3930 daemonize = 1;
3931 break;
3932 case 'g':
3933 global = optarg;
3934 break;
3935 case 'G':
3936 ping_interval = atoi(optarg);
3937 break;
3938 case 'h':
3939 usage();
3940 return 0;
3941 case 'v':
3942 printf("%s\n", wpa_cli_version);
3943 return 0;
3944 case 'i':
3945 os_free(ctrl_ifname);
3946 ctrl_ifname = os_strdup(optarg);
3947 break;
3948 case 'p':
3949 ctrl_iface_dir = optarg;
3950 break;
3951 case 'P':
3952 pid_file = optarg;
3953 break;
3954 default:
3955 usage();
3956 return -1;
3957 }
3958 }
3959
3960 interactive = (argc == optind) && (action_file == NULL);
3961
3962 if (interactive)
3963 printf("%s\n\n%s\n\n", wpa_cli_version, wpa_cli_license);
3964
3965 if (eloop_init())
3966 return -1;
3967
3968 if (global) {
3969 #ifdef CONFIG_CTRL_IFACE_NAMED_PIPE
3970 ctrl_conn = wpa_ctrl_open(NULL);
3971 #else /* CONFIG_CTRL_IFACE_NAMED_PIPE */
3972 ctrl_conn = wpa_ctrl_open(global);
3973 #endif /* CONFIG_CTRL_IFACE_NAMED_PIPE */
3974 if (ctrl_conn == NULL) {
3975 fprintf(stderr, "Failed to connect to wpa_supplicant "
3976 "global interface: %s error: %s\n",
3977 global, strerror(errno));
3978 return -1;
3979 }
3980
3981 if (interactive) {
3982 update_ifnames(ctrl_conn);
3983 mon_conn = wpa_ctrl_open(global);
3984 if (mon_conn) {
3985 if (wpa_ctrl_attach(mon_conn) == 0) {
3986 wpa_cli_attached = 1;
3987 eloop_register_read_sock(
3988 wpa_ctrl_get_fd(mon_conn),
3989 wpa_cli_mon_receive,
3990 NULL, NULL);
3991 } else {
3992 printf("Failed to open monitor "
3993 "connection through global "
3994 "control interface\n");
3995 }
3996 }
3997 }
3998 }
3999
4000 eloop_register_signal_terminate(wpa_cli_terminate, NULL);
4001
4002 if (ctrl_ifname == NULL)
4003 ctrl_ifname = wpa_cli_get_default_ifname();
4004
4005 if (interactive) {
4006 wpa_cli_interactive();
4007 } else {
4008 if (!global &&
4009 wpa_cli_open_connection(ctrl_ifname, 0) < 0) {
4010 fprintf(stderr, "Failed to connect to non-global "
4011 "ctrl_ifname: %s error: %s\n",
4012 ctrl_ifname ? ctrl_ifname : "(nil)",
4013 strerror(errno));
4014 return -1;
4015 }
4016
4017 if (action_file) {
4018 if (wpa_ctrl_attach(ctrl_conn) == 0) {
4019 wpa_cli_attached = 1;
4020 } else {
4021 printf("Warning: Failed to attach to "
4022 "wpa_supplicant.\n");
4023 return -1;
4024 }
4025 }
4026
4027 if (daemonize && os_daemonize(pid_file))
4028 return -1;
4029
4030 if (action_file)
4031 wpa_cli_action(ctrl_conn);
4032 else
4033 ret = wpa_request(ctrl_conn, argc - optind,
4034 &argv[optind]);
4035 }
4036
4037 os_free(ctrl_ifname);
4038 eloop_destroy();
4039 wpa_cli_cleanup();
4040
4041 return ret;
4042 }
4043
4044 #else /* CONFIG_CTRL_IFACE */
4045 int main(int argc, char *argv[])
4046 {
4047 printf("CONFIG_CTRL_IFACE not defined - wpa_cli disabled\n");
4048 return -1;
4049 }
4050 #endif /* CONFIG_CTRL_IFACE */