blob: 2f06c35ae42777a66a051f9bf7b90bd51867fded [file] [log] [blame]
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001/*
2 * WPA Supplicant - command line interface for wpa_supplicant daemon
Dmitry Shmidt807291d2015-01-27 13:40:23 -08003 * Copyright (c) 2004-2015, Jouni Malinen <j@w1.fi>
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004 *
Dmitry Shmidtc5ec7f52012-03-06 16:33:24 -08005 * This software may be distributed under the terms of the BSD license.
6 * See README for more details.
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07007 */
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"
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -080021#include "utils/list.h"
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070022#include "common/version.h"
Dmitry Shmidt04949592012-07-19 12:16:46 -070023#include "common/ieee802_11_defs.h"
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070024#ifdef ANDROID
25#include <cutils/properties.h>
26#endif /* ANDROID */
27
28
29static const char *wpa_cli_version =
30"wpa_cli v" VERSION_STR "\n"
Dmitry Shmidt807291d2015-01-27 13:40:23 -080031"Copyright (c) 2004-2015, Jouni Malinen <j@w1.fi> and contributors";
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070032
33
34static const char *wpa_cli_license =
Dmitry Shmidtc5ec7f52012-03-06 16:33:24 -080035"This software may be distributed under the terms of the BSD license.\n"
36"See README for more details.\n";
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070037
38static const char *wpa_cli_full_license =
Dmitry Shmidtc5ec7f52012-03-06 16:33:24 -080039"This software may be distributed under the terms of the BSD license.\n"
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070040"\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
69static struct wpa_ctrl *ctrl_conn;
70static struct wpa_ctrl *mon_conn;
71static int wpa_cli_quit = 0;
72static int wpa_cli_attached = 0;
Dmitry Shmidt391c59f2013-09-03 12:16:28 -070073static int wpa_cli_connected = -1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070074static 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 */
78static const char *ctrl_iface_dir = CONFIG_CTRL_IFACE_DIR;
79static char *ctrl_ifname = NULL;
80static const char *pid_file = NULL;
81static const char *action_file = NULL;
82static int ping_interval = 5;
83static int interactive = 0;
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -070084static char *ifname_prefix = NULL;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070085
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -080086struct cli_txt_entry {
87 struct dl_list list;
88 char *txt;
89};
90
91static DEFINE_DL_LIST(bsses); /* struct cli_txt_entry */
92static DEFINE_DL_LIST(p2p_peers); /* struct cli_txt_entry */
93static DEFINE_DL_LIST(p2p_groups); /* struct cli_txt_entry */
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -070094static DEFINE_DL_LIST(ifnames); /* struct cli_txt_entry */
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -080095
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070096
Dmitry Shmidt61d9df32012-08-29 16:22:06 -070097static void print_help(const char *cmd);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070098static void wpa_cli_mon_receive(int sock, void *eloop_ctx, void *sock_ctx);
Dmitry Shmidt61d9df32012-08-29 16:22:06 -070099static void wpa_cli_close_connection(void);
100static char * wpa_cli_get_default_ifname(void);
101static char ** wpa_list_cmd_list(void);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700102
103
104static 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");
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700118 print_help(NULL);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700119}
120
121
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -0800122static 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
130static 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
138static 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
150static 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
159static 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
171static 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);
Dmitry Shmidt4b060592013-04-29 16:42:49 -0700178 buf = dup_binstr(txt, end - txt);
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -0800179 if (buf == NULL)
180 return;
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -0800181 cli_txt_list_del(txt_list, buf);
182 os_free(buf);
183}
184#endif /* CONFIG_P2P */
185
186
187static 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
207static 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
218static 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);
Dmitry Shmidt4b060592013-04-29 16:42:49 -0700226 buf = dup_binstr(txt, end - txt);
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -0800227 if (buf == NULL)
228 return -1;
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -0800229 ret = cli_txt_list_add(txt_list, buf);
230 os_free(buf);
231 return ret;
232}
233#endif /* CONFIG_P2P */
234
235
236static 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
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700242 res = os_calloc(count + 1, sizeof(char *));
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -0800243 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
258static 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
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700276static int str_starts(const char *src, const char *match)
277{
278 return os_strncmp(src, match, os_strlen(match)) == 0;
279}
280
281
282static 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
303static 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);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800336 if (os_snprintf_error(flen, res)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700337 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");
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700365 wpa_cli_close_connection();
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700366 return -1;
367 }
368 }
369
370 return 0;
371}
372
373
374static 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
393static void wpa_cli_msg_cb(char *msg, size_t len)
394{
395 printf("%s\n", msg);
396}
397
398
399static int _wpa_ctrl_command(struct wpa_ctrl *ctrl, char *cmd, int print)
400{
Dmitry Shmidtc97d8bf2011-08-30 11:10:13 -0700401 char buf[4096];
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700402 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 }
Dmitry Shmidtd3e385e2013-06-05 11:06:13 -0700409 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 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700415 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
435static int wpa_ctrl_command(struct wpa_ctrl *ctrl, char *cmd)
436{
437 return _wpa_ctrl_command(ctrl, cmd, 1);
438}
439
440
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700441static 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);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800451 if (os_snprintf_error(end - pos, res))
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700452 goto fail;
453 pos += res;
454
455 for (i = 0; i < argc; i++) {
456 res = os_snprintf(pos, end - pos, " %s", argv[i]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800457 if (os_snprintf_error(end - pos, res))
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700458 goto fail;
459 pos += res;
460 }
461
462 buf[buflen - 1] = '\0';
463 return 0;
464
465fail:
466 printf("Too long command\n");
467 return -1;
468}
469
470
471static int wpa_cli_cmd(struct wpa_ctrl *ctrl, const char *cmd, int min_args,
472 int argc, char *argv[])
473{
Dmitry Shmidt4b060592013-04-29 16:42:49 -0700474 char buf[4096];
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700475 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
487static int wpa_cli_cmd_ifname(struct wpa_ctrl *ctrl, int argc, char *argv[])
488{
489 return wpa_ctrl_command(ctrl, "IFNAME");
490}
491
492
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700493static int wpa_cli_cmd_status(struct wpa_ctrl *ctrl, int argc, char *argv[])
494{
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -0800495 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");
Dmitry Shmidt56052862013-10-04 10:23:25 -0700499 if (argc > 0 && os_strcmp(argv[0], "driver") == 0)
500 return wpa_ctrl_command(ctrl, "STATUS-DRIVER");
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -0800501 return wpa_ctrl_command(ctrl, "STATUS");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700502}
503
504
505static int wpa_cli_cmd_ping(struct wpa_ctrl *ctrl, int argc, char *argv[])
506{
507 return wpa_ctrl_command(ctrl, "PING");
508}
509
510
511static int wpa_cli_cmd_relog(struct wpa_ctrl *ctrl, int argc, char *argv[])
512{
513 return wpa_ctrl_command(ctrl, "RELOG");
514}
515
516
517static int wpa_cli_cmd_note(struct wpa_ctrl *ctrl, int argc, char *argv[])
518{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700519 return wpa_cli_cmd(ctrl, "NOTE", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700520}
521
522
523static int wpa_cli_cmd_mib(struct wpa_ctrl *ctrl, int argc, char *argv[])
524{
525 return wpa_ctrl_command(ctrl, "MIB");
526}
527
528
529static int wpa_cli_cmd_pmksa(struct wpa_ctrl *ctrl, int argc, char *argv[])
530{
531 return wpa_ctrl_command(ctrl, "PMKSA");
532}
533
534
Dmitry Shmidt661b4f72014-09-29 14:58:27 -0700535static 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
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700542static int wpa_cli_cmd_help(struct wpa_ctrl *ctrl, int argc, char *argv[])
543{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700544 print_help(argc > 0 ? argv[0] : NULL);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700545 return 0;
546}
547
548
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700549static 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
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700564static 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
571static 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
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700580static int wpa_cli_cmd_set(struct wpa_ctrl *ctrl, int argc, char *argv[])
581{
582 char cmd[256];
583 int res;
584
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -0700585 if (argc == 1) {
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -0800586 res = os_snprintf(cmd, sizeof(cmd), "SET %s ", argv[0]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800587 if (os_snprintf_error(sizeof(cmd), res)) {
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -0700588 printf("Too long SET command.\n");
589 return -1;
590 }
591 return wpa_ctrl_command(ctrl, cmd);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700592 }
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -0700593
594 return wpa_cli_cmd(ctrl, "SET", 2, argc, argv);
595}
596
597
598static 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",
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800613 "pkcs11_module_path", "openssl_ciphers",
614 "pcsc_reader", "pcsc_pin",
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -0700615 "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",
Dmitry Shmidt68d0e3e2013-10-28 17:59:21 -0700626 "p2p_no_go_freq",
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -0700627 "p2p_go_ht40", "p2p_disabled", "p2p_no_group_iface",
Dmitry Shmidt68d0e3e2013-10-28 17:59:21 -0700628 "p2p_go_vht",
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -0700629 "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",
Dmitry Shmidt97672262014-02-03 13:02:54 -0800638 "ignore_old_scan_res", "freq_list", "external_sim",
Dmitry Shmidt09f57ba2014-06-10 16:07:13 -0700639 "tdls_external_control", "p2p_search_delay"
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -0700640 };
Dmitry Shmidt68d0e3e2013-10-28 17:59:21 -0700641 int i, num_fields = ARRAY_SIZE(fields);
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -0700642
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;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700659}
660
661
662static int wpa_cli_cmd_get(struct wpa_ctrl *ctrl, int argc, char *argv[])
663{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700664 return wpa_cli_cmd(ctrl, "GET", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700665}
666
667
668static int wpa_cli_cmd_logoff(struct wpa_ctrl *ctrl, int argc, char *argv[])
669{
670 return wpa_ctrl_command(ctrl, "LOGOFF");
671}
672
673
674static int wpa_cli_cmd_logon(struct wpa_ctrl *ctrl, int argc, char *argv[])
675{
676 return wpa_ctrl_command(ctrl, "LOGON");
677}
678
679
680static 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
Dmitry Shmidt98660862014-03-11 17:26:21 -0700687static int wpa_cli_cmd_reattach(struct wpa_ctrl *ctrl, int argc, char *argv[])
688{
689 return wpa_ctrl_command(ctrl, "REATTACH");
690}
691
692
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700693static int wpa_cli_cmd_preauthenticate(struct wpa_ctrl *ctrl, int argc,
694 char *argv[])
695{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700696 return wpa_cli_cmd(ctrl, "PREAUTH", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700697}
698
699
700static int wpa_cli_cmd_ap_scan(struct wpa_ctrl *ctrl, int argc, char *argv[])
701{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700702 return wpa_cli_cmd(ctrl, "AP_SCAN", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700703}
704
705
706static int wpa_cli_cmd_scan_interval(struct wpa_ctrl *ctrl, int argc,
707 char *argv[])
708{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700709 return wpa_cli_cmd(ctrl, "SCAN_INTERVAL", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700710}
711
712
713static int wpa_cli_cmd_bss_expire_age(struct wpa_ctrl *ctrl, int argc,
714 char *argv[])
715{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700716 return wpa_cli_cmd(ctrl, "BSS_EXPIRE_AGE", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700717}
718
719
720static int wpa_cli_cmd_bss_expire_count(struct wpa_ctrl *ctrl, int argc,
721 char *argv[])
722{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700723 return wpa_cli_cmd(ctrl, "BSS_EXPIRE_COUNT", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700724}
725
726
Dmitry Shmidtf48e4f92012-08-24 11:14:44 -0700727static 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800736 if (os_snprintf_error(sizeof(cmd), res)) {
Dmitry Shmidtf48e4f92012-08-24 11:14:44 -0700737 printf("Too long BSS_FLUSH command.\n");
738 return -1;
739 }
740 return wpa_ctrl_command(ctrl, cmd);
741}
742
743
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700744static int wpa_cli_cmd_stkstart(struct wpa_ctrl *ctrl, int argc,
745 char *argv[])
746{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700747 return wpa_cli_cmd(ctrl, "STKSTART", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700748}
749
750
751static int wpa_cli_cmd_ft_ds(struct wpa_ctrl *ctrl, int argc, char *argv[])
752{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700753 return wpa_cli_cmd(ctrl, "FT_DS", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700754}
755
756
757static int wpa_cli_cmd_wps_pbc(struct wpa_ctrl *ctrl, int argc, char *argv[])
758{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700759 return wpa_cli_cmd(ctrl, "WPS_PBC", 0, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700760}
761
762
763static int wpa_cli_cmd_wps_pin(struct wpa_ctrl *ctrl, int argc, char *argv[])
764{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700765 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
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700773 return wpa_cli_cmd(ctrl, "WPS_PIN", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700774}
775
776
777static int wpa_cli_cmd_wps_check_pin(struct wpa_ctrl *ctrl, int argc,
778 char *argv[])
779{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700780 return wpa_cli_cmd(ctrl, "WPS_CHECK_PIN", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700781}
782
783
784static 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
Dmitry Shmidt04949592012-07-19 12:16:46 -0700791#ifdef CONFIG_WPS_NFC
792
793static int wpa_cli_cmd_wps_nfc(struct wpa_ctrl *ctrl, int argc, char *argv[])
794{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700795 return wpa_cli_cmd(ctrl, "WPS_NFC", 0, argc, argv);
Dmitry Shmidt04949592012-07-19 12:16:46 -0700796}
797
798
Dmitry Shmidtf8623282013-02-20 14:34:59 -0800799static 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
Dmitry Shmidt04949592012-07-19 12:16:46 -0700806static int wpa_cli_cmd_wps_nfc_token(struct wpa_ctrl *ctrl, int argc,
807 char *argv[])
808{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700809 return wpa_cli_cmd(ctrl, "WPS_NFC_TOKEN", 1, argc, argv);
Dmitry Shmidt04949592012-07-19 12:16:46 -0700810}
811
812
813static 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
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800838
839static 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
846static 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
Dmitry Shmidtf8623282013-02-20 14:34:59 -0800853static 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
Dmitry Shmidt04949592012-07-19 12:16:46 -0700859#endif /* CONFIG_WPS_NFC */
860
861
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700862static 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
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800911 if (os_snprintf_error(sizeof(cmd), res)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700912 printf("Too long WPS_REG command.\n");
913 return -1;
914 }
915 return wpa_ctrl_command(ctrl, cmd);
916}
917
918
919static int wpa_cli_cmd_wps_ap_pin(struct wpa_ctrl *ctrl, int argc,
920 char *argv[])
921{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700922 return wpa_cli_cmd(ctrl, "WPS_AP_PIN", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700923}
924
925
926static int wpa_cli_cmd_wps_er_start(struct wpa_ctrl *ctrl, int argc,
927 char *argv[])
928{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700929 return wpa_cli_cmd(ctrl, "WPS_ER_START", 0, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700930}
931
932
933static 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
941static int wpa_cli_cmd_wps_er_pin(struct wpa_ctrl *ctrl, int argc,
942 char *argv[])
943{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700944 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
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700953 return wpa_cli_cmd(ctrl, "WPS_ER_PIN", 2, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700954}
955
956
957static int wpa_cli_cmd_wps_er_pbc(struct wpa_ctrl *ctrl, int argc,
958 char *argv[])
959{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700960 return wpa_cli_cmd(ctrl, "WPS_ER_PBC", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700961}
962
963
964static int wpa_cli_cmd_wps_er_learn(struct wpa_ctrl *ctrl, int argc,
965 char *argv[])
966{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700967 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
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700974 return wpa_cli_cmd(ctrl, "WPS_ER_LEARN", 2, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700975}
976
977
978static int wpa_cli_cmd_wps_er_set_config(struct wpa_ctrl *ctrl, int argc,
979 char *argv[])
980{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700981 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
Dmitry Shmidt61d9df32012-08-29 16:22:06 -0700989 return wpa_cli_cmd(ctrl, "WPS_ER_SET_CONFIG", 2, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700990}
991
992
993static 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
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001036 if (os_snprintf_error(sizeof(cmd), res)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001037 printf("Too long WPS_ER_CONFIG command.\n");
1038 return -1;
1039 }
1040 return wpa_ctrl_command(ctrl, cmd);
1041}
1042
1043
Dmitry Shmidt04949592012-07-19 12:16:46 -07001044#ifdef CONFIG_WPS_NFC
1045static int wpa_cli_cmd_wps_er_nfc_config_token(struct wpa_ctrl *ctrl, int argc,
1046 char *argv[])
1047{
Dmitry Shmidt04949592012-07-19 12:16:46 -07001048 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
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001056 return wpa_cli_cmd(ctrl, "WPS_ER_NFC_CONFIG_TOKEN", 2, argc, argv);
Dmitry Shmidt04949592012-07-19 12:16:46 -07001057}
1058#endif /* CONFIG_WPS_NFC */
1059
1060
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001061static int wpa_cli_cmd_ibss_rsn(struct wpa_ctrl *ctrl, int argc, char *argv[])
1062{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001063 return wpa_cli_cmd(ctrl, "IBSS_RSN", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001064}
1065
1066
1067static int wpa_cli_cmd_level(struct wpa_ctrl *ctrl, int argc, char *argv[])
1068{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001069 return wpa_cli_cmd(ctrl, "LEVEL", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001070}
1071
1072
1073static 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001088 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001089 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001095 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001096 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
1106static 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001121 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001122 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001128 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001129 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
1139static 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001155 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001156 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001162 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001163 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
1173static 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001188 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001189 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001195 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001196 printf("Too long PIN command.\n");
1197 return -1;
1198 }
1199 pos += ret;
1200 }
1201 return wpa_ctrl_command(ctrl, cmd);
1202}
1203
1204
1205static 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001220 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001221 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001227 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001228 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
Dmitry Shmidt051af732013-10-22 13:52:46 -07001238static 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001253 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt051af732013-10-22 13:52:46 -07001254 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001260 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt051af732013-10-22 13:52:46 -07001261 printf("Too long SIM command.\n");
1262 return -1;
1263 }
1264 pos += ret;
1265 }
1266 return wpa_ctrl_command(ctrl, cmd);
1267}
1268
1269
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001270static 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001286 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001287 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]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001293 if (os_snprintf_error(end - pos, ret)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001294 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
1304static int wpa_cli_cmd_bssid(struct wpa_ctrl *ctrl, int argc, char *argv[])
1305{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001306 if (argc < 2) {
1307 printf("Invalid BSSID command: needs two arguments (network "
1308 "id and BSSID)\n");
1309 return -1;
1310 }
1311
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001312 return wpa_cli_cmd(ctrl, "BSSID", 2, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001313}
1314
1315
Dmitry Shmidt696359e2011-03-16 15:04:31 -07001316static int wpa_cli_cmd_blacklist(struct wpa_ctrl *ctrl, int argc, char *argv[])
1317{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001318 return wpa_cli_cmd(ctrl, "BLACKLIST", 0, argc, argv);
Dmitry Shmidt696359e2011-03-16 15:04:31 -07001319}
1320
1321
1322static int wpa_cli_cmd_log_level(struct wpa_ctrl *ctrl, int argc, char *argv[])
1323{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001324 return wpa_cli_cmd(ctrl, "LOG_LEVEL", 0, argc, argv);
Dmitry Shmidt696359e2011-03-16 15:04:31 -07001325}
1326
1327
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001328static 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
1335static int wpa_cli_cmd_select_network(struct wpa_ctrl *ctrl, int argc,
1336 char *argv[])
1337{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001338 return wpa_cli_cmd(ctrl, "SELECT_NETWORK", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001339}
1340
1341
1342static int wpa_cli_cmd_enable_network(struct wpa_ctrl *ctrl, int argc,
1343 char *argv[])
1344{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001345 return wpa_cli_cmd(ctrl, "ENABLE_NETWORK", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001346}
1347
1348
1349static int wpa_cli_cmd_disable_network(struct wpa_ctrl *ctrl, int argc,
1350 char *argv[])
1351{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001352 return wpa_cli_cmd(ctrl, "DISABLE_NETWORK", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001353}
1354
1355
1356static 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
1363static int wpa_cli_cmd_remove_network(struct wpa_ctrl *ctrl, int argc,
1364 char *argv[])
1365{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001366 return wpa_cli_cmd(ctrl, "REMOVE_NETWORK", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001367}
1368
1369
1370static 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
1391static int wpa_cli_cmd_set_network(struct wpa_ctrl *ctrl, int argc,
1392 char *argv[])
1393{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001394 if (argc == 0) {
1395 wpa_cli_show_network_variables();
1396 return 0;
1397 }
1398
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001399 if (argc < 3) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001400 printf("Invalid SET_NETWORK command: needs three arguments\n"
1401 "(network id, variable name, and value)\n");
1402 return -1;
1403 }
1404
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001405 return wpa_cli_cmd(ctrl, "SET_NETWORK", 3, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001406}
1407
1408
1409static int wpa_cli_cmd_get_network(struct wpa_ctrl *ctrl, int argc,
1410 char *argv[])
1411{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001412 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
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001423 return wpa_cli_cmd(ctrl, "GET_NETWORK", 2, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001424}
1425
1426
Dmitry Shmidt684785c2014-05-12 13:34:29 -07001427static 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
Dmitry Shmidt04949592012-07-19 12:16:46 -07001445static 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
1452static 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
1458static int wpa_cli_cmd_remove_cred(struct wpa_ctrl *ctrl, int argc,
1459 char *argv[])
1460{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001461 return wpa_cli_cmd(ctrl, "REMOVE_CRED", 1, argc, argv);
Dmitry Shmidt04949592012-07-19 12:16:46 -07001462}
1463
1464
1465static int wpa_cli_cmd_set_cred(struct wpa_ctrl *ctrl, int argc, char *argv[])
1466{
Dmitry Shmidt04949592012-07-19 12:16:46 -07001467 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
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001473 return wpa_cli_cmd(ctrl, "SET_CRED", 3, argc, argv);
Dmitry Shmidt04949592012-07-19 12:16:46 -07001474}
1475
1476
Dmitry Shmidt0cfd5f72014-04-04 14:48:05 -07001477static 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
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001489static 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
1496static 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
1503static 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
1510static int wpa_cli_cmd_scan(struct wpa_ctrl *ctrl, int argc, char *argv[])
1511{
Dmitry Shmidt4b9d52f2013-02-05 17:44:43 -08001512 return wpa_cli_cmd(ctrl, "SCAN", 0, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001513}
1514
1515
1516static 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
1523static int wpa_cli_cmd_bss(struct wpa_ctrl *ctrl, int argc, char *argv[])
1524{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001525 return wpa_cli_cmd(ctrl, "BSS", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001526}
1527
1528
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08001529static 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
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001544static int wpa_cli_cmd_get_capability(struct wpa_ctrl *ctrl, int argc,
1545 char *argv[])
1546{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001547 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
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001559 return wpa_cli_cmd(ctrl, "GET_CAPABILITY", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001560}
1561
1562
1563static 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
1570static 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]);
Dmitry Shmidt2f74e362015-01-21 13:19:05 -08001580 if (!ctrl_ifname) {
1581 printf("Failed to allocate memory\n");
1582 return 0;
1583 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001584
Dmitry Shmidt413dde72014-04-11 10:23:22 -07001585 if (wpa_cli_open_connection(ctrl_ifname, 1) == 0) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001586 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
1595static 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
1602static 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
1609static 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] : "");
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001633 if (os_snprintf_error(sizeof(cmd), res))
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001634 return -1;
1635 cmd[sizeof(cmd) - 1] = '\0';
1636 return wpa_ctrl_command(ctrl, cmd);
1637}
1638
1639
1640static int wpa_cli_cmd_interface_remove(struct wpa_ctrl *ctrl, int argc,
1641 char *argv[])
1642{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001643 return wpa_cli_cmd(ctrl, "INTERFACE_REMOVE", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001644}
1645
1646
1647static 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
1655static int wpa_cli_cmd_sta(struct wpa_ctrl *ctrl, int argc, char *argv[])
1656{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001657 return wpa_cli_cmd(ctrl, "STA", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001658}
1659
1660
1661static 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;
Dmitry Shmidt04949592012-07-19 12:16:46 -07001673 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001674 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';
Dmitry Shmidt04949592012-07-19 12:16:46 -07001684 if (os_memcmp(buf, "FAIL", 4) == 0)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001685 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
1697static 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}
Dmitry Shmidt04949592012-07-19 12:16:46 -07001709
1710
1711static int wpa_cli_cmd_deauthenticate(struct wpa_ctrl *ctrl, int argc,
1712 char *argv[])
1713{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001714 return wpa_cli_cmd(ctrl, "DEAUTHENTICATE", 1, argc, argv);
Dmitry Shmidt04949592012-07-19 12:16:46 -07001715}
1716
1717
1718static int wpa_cli_cmd_disassociate(struct wpa_ctrl *ctrl, int argc,
1719 char *argv[])
1720{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001721 return wpa_cli_cmd(ctrl, "DISASSOCIATE", 1, argc, argv);
Dmitry Shmidt04949592012-07-19 12:16:46 -07001722}
Dmitry Shmidte0e48dc2013-11-18 12:00:06 -08001723
1724static 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
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001730#endif /* CONFIG_AP */
1731
1732
1733static int wpa_cli_cmd_suspend(struct wpa_ctrl *ctrl, int argc, char *argv[])
1734{
1735 return wpa_ctrl_command(ctrl, "SUSPEND");
1736}
1737
1738
1739static int wpa_cli_cmd_resume(struct wpa_ctrl *ctrl, int argc, char *argv[])
1740{
1741 return wpa_ctrl_command(ctrl, "RESUME");
1742}
1743
1744
Dmitry Shmidt21de2142014-04-08 10:50:52 -07001745#ifdef CONFIG_TESTING_OPTIONS
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001746static int wpa_cli_cmd_drop_sa(struct wpa_ctrl *ctrl, int argc, char *argv[])
1747{
1748 return wpa_ctrl_command(ctrl, "DROP_SA");
1749}
Dmitry Shmidt21de2142014-04-08 10:50:52 -07001750#endif /* CONFIG_TESTING_OPTIONS */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001751
1752
1753static int wpa_cli_cmd_roam(struct wpa_ctrl *ctrl, int argc, char *argv[])
1754{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001755 return wpa_cli_cmd(ctrl, "ROAM", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001756}
1757
1758
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001759#ifdef CONFIG_MESH
1760
1761static 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
1768static 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
1775static 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
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001784#ifdef CONFIG_P2P
1785
1786static int wpa_cli_cmd_p2p_find(struct wpa_ctrl *ctrl, int argc, char *argv[])
1787{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001788 return wpa_cli_cmd(ctrl, "P2P_FIND", 0, argc, argv);
1789}
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001790
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001791
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001792static 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;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001818}
1819
1820
1821static 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
Dmitry Shmidt216983b2015-02-06 10:50:36 -08001828static int wpa_cli_cmd_p2p_asp_provision(struct wpa_ctrl *ctrl, int argc,
1829 char *argv[])
1830{
1831 return wpa_cli_cmd(ctrl, "P2P_ASP_PROVISION", 3, argc, argv);
1832}
1833
1834
1835static int wpa_cli_cmd_p2p_asp_provision_resp(struct wpa_ctrl *ctrl, int argc,
1836 char *argv[])
1837{
1838 return wpa_cli_cmd(ctrl, "P2P_ASP_PROVISION_RESP", 2, argc, argv);
1839}
1840
1841
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001842static int wpa_cli_cmd_p2p_connect(struct wpa_ctrl *ctrl, int argc,
1843 char *argv[])
1844{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001845 return wpa_cli_cmd(ctrl, "P2P_CONNECT", 2, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001846}
1847
1848
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08001849static char ** wpa_cli_complete_p2p_connect(const char *str, int pos)
1850{
1851 int arg = get_cmd_arg_num(str, pos);
1852 char **res = NULL;
1853
1854 switch (arg) {
1855 case 1:
1856 res = cli_txt_list_array(&p2p_peers);
1857 break;
1858 }
1859
1860 return res;
1861}
1862
1863
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001864static int wpa_cli_cmd_p2p_listen(struct wpa_ctrl *ctrl, int argc,
1865 char *argv[])
1866{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001867 return wpa_cli_cmd(ctrl, "P2P_LISTEN", 0, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001868}
1869
1870
1871static int wpa_cli_cmd_p2p_group_remove(struct wpa_ctrl *ctrl, int argc,
1872 char *argv[])
1873{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001874 return wpa_cli_cmd(ctrl, "P2P_GROUP_REMOVE", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001875}
1876
1877
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08001878static char ** wpa_cli_complete_p2p_group_remove(const char *str, int pos)
1879{
1880 int arg = get_cmd_arg_num(str, pos);
1881 char **res = NULL;
1882
1883 switch (arg) {
1884 case 1:
1885 res = cli_txt_list_array(&p2p_groups);
1886 break;
1887 }
1888
1889 return res;
1890}
1891
1892
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001893static int wpa_cli_cmd_p2p_group_add(struct wpa_ctrl *ctrl, int argc,
1894 char *argv[])
1895{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001896 return wpa_cli_cmd(ctrl, "P2P_GROUP_ADD", 0, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001897}
1898
1899
1900static int wpa_cli_cmd_p2p_prov_disc(struct wpa_ctrl *ctrl, int argc,
1901 char *argv[])
1902{
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08001903 if (argc != 2 && argc != 3) {
1904 printf("Invalid P2P_PROV_DISC command: needs at least "
1905 "two arguments, address and config method\n"
1906 "(display, keypad, or pbc) and an optional join\n");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001907 return -1;
1908 }
1909
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001910 return wpa_cli_cmd(ctrl, "P2P_PROV_DISC", 2, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001911}
1912
1913
1914static int wpa_cli_cmd_p2p_get_passphrase(struct wpa_ctrl *ctrl, int argc,
1915 char *argv[])
1916{
1917 return wpa_ctrl_command(ctrl, "P2P_GET_PASSPHRASE");
1918}
1919
1920
1921static int wpa_cli_cmd_p2p_serv_disc_req(struct wpa_ctrl *ctrl, int argc,
1922 char *argv[])
1923{
1924 char cmd[4096];
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001925
Dmitry Shmidt216983b2015-02-06 10:50:36 -08001926 if (argc < 2) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001927 printf("Invalid P2P_SERV_DISC_REQ command: needs two "
Dmitry Shmidt216983b2015-02-06 10:50:36 -08001928 "or more arguments (address and TLVs)\n");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001929 return -1;
1930 }
1931
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001932 if (write_cmd(cmd, sizeof(cmd), "P2P_SERV_DISC_REQ", argc, argv) < 0)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001933 return -1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001934 return wpa_ctrl_command(ctrl, cmd);
1935}
1936
1937
1938static int wpa_cli_cmd_p2p_serv_disc_cancel_req(struct wpa_ctrl *ctrl,
1939 int argc, char *argv[])
1940{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001941 return wpa_cli_cmd(ctrl, "P2P_SERV_DISC_CANCEL_REQ", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001942}
1943
1944
1945static int wpa_cli_cmd_p2p_serv_disc_resp(struct wpa_ctrl *ctrl, int argc,
1946 char *argv[])
1947{
1948 char cmd[4096];
1949 int res;
1950
1951 if (argc != 4) {
1952 printf("Invalid P2P_SERV_DISC_RESP command: needs four "
1953 "arguments (freq, address, dialog token, and TLVs)\n");
1954 return -1;
1955 }
1956
1957 res = os_snprintf(cmd, sizeof(cmd), "P2P_SERV_DISC_RESP %s %s %s %s",
1958 argv[0], argv[1], argv[2], argv[3]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001959 if (os_snprintf_error(sizeof(cmd), res))
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001960 return -1;
1961 cmd[sizeof(cmd) - 1] = '\0';
1962 return wpa_ctrl_command(ctrl, cmd);
1963}
1964
1965
1966static int wpa_cli_cmd_p2p_service_update(struct wpa_ctrl *ctrl, int argc,
1967 char *argv[])
1968{
1969 return wpa_ctrl_command(ctrl, "P2P_SERVICE_UPDATE");
1970}
1971
1972
1973static int wpa_cli_cmd_p2p_serv_disc_external(struct wpa_ctrl *ctrl,
1974 int argc, char *argv[])
1975{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07001976 return wpa_cli_cmd(ctrl, "P2P_SERV_DISC_EXTERNAL", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001977}
1978
1979
1980static int wpa_cli_cmd_p2p_service_flush(struct wpa_ctrl *ctrl, int argc,
1981 char *argv[])
1982{
1983 return wpa_ctrl_command(ctrl, "P2P_SERVICE_FLUSH");
1984}
1985
1986
1987static int wpa_cli_cmd_p2p_service_add(struct wpa_ctrl *ctrl, int argc,
1988 char *argv[])
1989{
Dmitry Shmidt216983b2015-02-06 10:50:36 -08001990 if (argc < 3) {
1991 printf("Invalid P2P_SERVICE_ADD command: needs 3-6 arguments\n");
1992 return -1;
1993 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001994
Dmitry Shmidt216983b2015-02-06 10:50:36 -08001995 return wpa_cli_cmd(ctrl, "P2P_SERVICE_ADD", 3, argc, argv);
1996}
1997
1998
1999static int wpa_cli_cmd_p2p_service_rep(struct wpa_ctrl *ctrl, int argc,
2000 char *argv[])
2001{
2002 if (argc < 5 || argc > 6) {
2003 printf("Invalid P2P_SERVICE_REP command: needs 5-6 "
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002004 "arguments\n");
2005 return -1;
2006 }
2007
Dmitry Shmidt216983b2015-02-06 10:50:36 -08002008 return wpa_cli_cmd(ctrl, "P2P_SERVICE_REP", 5, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002009}
2010
2011
2012static int wpa_cli_cmd_p2p_service_del(struct wpa_ctrl *ctrl, int argc,
2013 char *argv[])
2014{
2015 char cmd[4096];
2016 int res;
2017
2018 if (argc != 2 && argc != 3) {
2019 printf("Invalid P2P_SERVICE_DEL command: needs two or three "
2020 "arguments\n");
2021 return -1;
2022 }
2023
2024 if (argc == 3)
2025 res = os_snprintf(cmd, sizeof(cmd),
2026 "P2P_SERVICE_DEL %s %s %s",
2027 argv[0], argv[1], argv[2]);
2028 else
2029 res = os_snprintf(cmd, sizeof(cmd),
2030 "P2P_SERVICE_DEL %s %s",
2031 argv[0], argv[1]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002032 if (os_snprintf_error(sizeof(cmd), res))
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002033 return -1;
2034 cmd[sizeof(cmd) - 1] = '\0';
2035 return wpa_ctrl_command(ctrl, cmd);
2036}
2037
2038
2039static int wpa_cli_cmd_p2p_reject(struct wpa_ctrl *ctrl,
2040 int argc, char *argv[])
2041{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002042 return wpa_cli_cmd(ctrl, "P2P_REJECT", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002043}
2044
2045
2046static int wpa_cli_cmd_p2p_invite(struct wpa_ctrl *ctrl,
2047 int argc, char *argv[])
2048{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002049 return wpa_cli_cmd(ctrl, "P2P_INVITE", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002050}
2051
2052
2053static int wpa_cli_cmd_p2p_peer(struct wpa_ctrl *ctrl, int argc, char *argv[])
2054{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002055 return wpa_cli_cmd(ctrl, "P2P_PEER", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002056}
2057
2058
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002059static char ** wpa_cli_complete_p2p_peer(const char *str, int pos)
2060{
2061 int arg = get_cmd_arg_num(str, pos);
2062 char **res = NULL;
2063
2064 switch (arg) {
2065 case 1:
2066 res = cli_txt_list_array(&p2p_peers);
2067 break;
2068 }
2069
2070 return res;
2071}
2072
2073
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002074static int wpa_ctrl_command_p2p_peer(struct wpa_ctrl *ctrl, char *cmd,
2075 char *addr, size_t addr_len,
2076 int discovered)
2077{
2078 char buf[4096], *pos;
2079 size_t len;
2080 int ret;
2081
2082 if (ctrl_conn == NULL)
2083 return -1;
2084 len = sizeof(buf) - 1;
Dmitry Shmidt04949592012-07-19 12:16:46 -07002085 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002086 wpa_cli_msg_cb);
2087 if (ret == -2) {
2088 printf("'%s' command timed out.\n", cmd);
2089 return -2;
2090 } else if (ret < 0) {
2091 printf("'%s' command failed.\n", cmd);
2092 return -1;
2093 }
2094
2095 buf[len] = '\0';
Dmitry Shmidt04949592012-07-19 12:16:46 -07002096 if (os_memcmp(buf, "FAIL", 4) == 0)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002097 return -1;
2098
2099 pos = buf;
2100 while (*pos != '\0' && *pos != '\n')
2101 pos++;
2102 *pos++ = '\0';
2103 os_strlcpy(addr, buf, addr_len);
2104 if (!discovered || os_strstr(pos, "[PROBE_REQ_ONLY]") == NULL)
2105 printf("%s\n", addr);
2106 return 0;
2107}
2108
2109
2110static int wpa_cli_cmd_p2p_peers(struct wpa_ctrl *ctrl, int argc, char *argv[])
2111{
2112 char addr[32], cmd[64];
2113 int discovered;
2114
2115 discovered = argc > 0 && os_strcmp(argv[0], "discovered") == 0;
2116
2117 if (wpa_ctrl_command_p2p_peer(ctrl, "P2P_PEER FIRST",
2118 addr, sizeof(addr), discovered))
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002119 return -1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002120 do {
2121 os_snprintf(cmd, sizeof(cmd), "P2P_PEER NEXT-%s", addr);
2122 } while (wpa_ctrl_command_p2p_peer(ctrl, cmd, addr, sizeof(addr),
2123 discovered) == 0);
2124
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002125 return 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002126}
2127
2128
2129static int wpa_cli_cmd_p2p_set(struct wpa_ctrl *ctrl, int argc, char *argv[])
2130{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002131 return wpa_cli_cmd(ctrl, "P2P_SET", 2, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002132}
2133
2134
Dmitry Shmidt391c59f2013-09-03 12:16:28 -07002135static char ** wpa_cli_complete_p2p_set(const char *str, int pos)
2136{
2137 int arg = get_cmd_arg_num(str, pos);
2138 const char *fields[] = {
2139 "discoverability",
2140 "managed",
2141 "listen_channel",
2142 "ssid_postfix",
2143 "noa",
2144 "ps",
2145 "oppps",
2146 "ctwindow",
2147 "disabled",
2148 "conc_pref",
2149 "force_long_sd",
2150 "peer_filter",
2151 "cross_connect",
2152 "go_apsd",
2153 "client_apsd",
2154 "disallow_freq",
2155 "disc_int",
2156 "per_sta_psk",
2157 };
Dmitry Shmidt68d0e3e2013-10-28 17:59:21 -07002158 int i, num_fields = ARRAY_SIZE(fields);
Dmitry Shmidt391c59f2013-09-03 12:16:28 -07002159
2160 if (arg == 1) {
2161 char **res = os_calloc(num_fields + 1, sizeof(char *));
2162 if (res == NULL)
2163 return NULL;
2164 for (i = 0; i < num_fields; i++) {
2165 res[i] = os_strdup(fields[i]);
2166 if (res[i] == NULL)
2167 return res;
2168 }
2169 return res;
2170 }
2171
2172 if (arg == 2 && os_strncasecmp(str, "p2p_set peer_filter ", 20) == 0)
2173 return cli_txt_list_array(&p2p_peers);
2174
2175 return NULL;
2176}
2177
2178
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002179static int wpa_cli_cmd_p2p_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
2180{
2181 return wpa_ctrl_command(ctrl, "P2P_FLUSH");
2182}
2183
2184
2185static int wpa_cli_cmd_p2p_cancel(struct wpa_ctrl *ctrl, int argc,
2186 char *argv[])
2187{
2188 return wpa_ctrl_command(ctrl, "P2P_CANCEL");
2189}
2190
2191
2192static int wpa_cli_cmd_p2p_unauthorize(struct wpa_ctrl *ctrl, int argc,
2193 char *argv[])
2194{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002195 return wpa_cli_cmd(ctrl, "P2P_UNAUTHORIZE", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002196}
2197
2198
2199static int wpa_cli_cmd_p2p_presence_req(struct wpa_ctrl *ctrl, int argc,
2200 char *argv[])
2201{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002202 if (argc != 0 && argc != 2 && argc != 4) {
2203 printf("Invalid P2P_PRESENCE_REQ command: needs two arguments "
2204 "(preferred duration, interval; in microsecods).\n"
2205 "Optional second pair can be used to provide "
2206 "acceptable values.\n");
2207 return -1;
2208 }
2209
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002210 return wpa_cli_cmd(ctrl, "P2P_PRESENCE_REQ", 0, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002211}
2212
2213
2214static int wpa_cli_cmd_p2p_ext_listen(struct wpa_ctrl *ctrl, int argc,
2215 char *argv[])
2216{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002217 if (argc != 0 && argc != 2) {
2218 printf("Invalid P2P_EXT_LISTEN command: needs two arguments "
2219 "(availability period, availability interval; in "
2220 "millisecods).\n"
2221 "Extended Listen Timing can be cancelled with this "
2222 "command when used without parameters.\n");
2223 return -1;
2224 }
2225
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002226 return wpa_cli_cmd(ctrl, "P2P_EXT_LISTEN", 0, argc, argv);
2227}
2228
Dmitry Shmidt391c59f2013-09-03 12:16:28 -07002229
2230static int wpa_cli_cmd_p2p_remove_client(struct wpa_ctrl *ctrl, int argc,
2231 char *argv[])
2232{
2233 return wpa_cli_cmd(ctrl, "P2P_REMOVE_CLIENT", 1, argc, argv);
2234}
2235
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002236#endif /* CONFIG_P2P */
2237
2238#ifdef CONFIG_WIFI_DISPLAY
2239
2240static int wpa_cli_cmd_wfd_subelem_set(struct wpa_ctrl *ctrl, int argc,
2241 char *argv[])
2242{
2243 char cmd[100];
2244 int res;
2245
2246 if (argc != 1 && argc != 2) {
2247 printf("Invalid WFD_SUBELEM_SET command: needs one or two "
2248 "arguments (subelem, hexdump)\n");
2249 return -1;
2250 }
2251
2252 res = os_snprintf(cmd, sizeof(cmd), "WFD_SUBELEM_SET %s %s",
2253 argv[0], argc > 1 ? argv[1] : "");
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002254 if (os_snprintf_error(sizeof(cmd), res))
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002255 return -1;
2256 cmd[sizeof(cmd) - 1] = '\0';
2257 return wpa_ctrl_command(ctrl, cmd);
2258}
2259
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002260
2261static int wpa_cli_cmd_wfd_subelem_get(struct wpa_ctrl *ctrl, int argc,
2262 char *argv[])
2263{
2264 char cmd[100];
2265 int res;
2266
2267 if (argc != 1) {
2268 printf("Invalid WFD_SUBELEM_GET command: needs one "
2269 "argument (subelem)\n");
2270 return -1;
2271 }
2272
2273 res = os_snprintf(cmd, sizeof(cmd), "WFD_SUBELEM_GET %s",
2274 argv[0]);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002275 if (os_snprintf_error(sizeof(cmd), res))
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002276 return -1;
2277 cmd[sizeof(cmd) - 1] = '\0';
2278 return wpa_ctrl_command(ctrl, cmd);
2279}
2280#endif /* CONFIG_WIFI_DISPLAY */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002281
2282
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002283#ifdef CONFIG_INTERWORKING
2284static int wpa_cli_cmd_fetch_anqp(struct wpa_ctrl *ctrl, int argc,
2285 char *argv[])
2286{
2287 return wpa_ctrl_command(ctrl, "FETCH_ANQP");
2288}
2289
2290
2291static int wpa_cli_cmd_stop_fetch_anqp(struct wpa_ctrl *ctrl, int argc,
2292 char *argv[])
2293{
2294 return wpa_ctrl_command(ctrl, "STOP_FETCH_ANQP");
2295}
2296
2297
2298static int wpa_cli_cmd_interworking_select(struct wpa_ctrl *ctrl, int argc,
2299 char *argv[])
2300{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002301 return wpa_cli_cmd(ctrl, "INTERWORKING_SELECT", 0, argc, argv);
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002302}
2303
2304
2305static int wpa_cli_cmd_interworking_connect(struct wpa_ctrl *ctrl, int argc,
2306 char *argv[])
2307{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002308 return wpa_cli_cmd(ctrl, "INTERWORKING_CONNECT", 1, argc, argv);
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002309}
2310
2311
2312static int wpa_cli_cmd_anqp_get(struct wpa_ctrl *ctrl, int argc, char *argv[])
2313{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002314 return wpa_cli_cmd(ctrl, "ANQP_GET", 2, argc, argv);
2315}
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002316
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002317
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002318static int wpa_cli_cmd_gas_request(struct wpa_ctrl *ctrl, int argc,
2319 char *argv[])
2320{
2321 return wpa_cli_cmd(ctrl, "GAS_REQUEST", 2, argc, argv);
2322}
2323
2324
2325static int wpa_cli_cmd_gas_response_get(struct wpa_ctrl *ctrl, int argc,
2326 char *argv[])
2327{
2328 return wpa_cli_cmd(ctrl, "GAS_RESPONSE_GET", 2, argc, argv);
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002329}
2330#endif /* CONFIG_INTERWORKING */
2331
2332
Dmitry Shmidt04949592012-07-19 12:16:46 -07002333#ifdef CONFIG_HS20
2334
2335static int wpa_cli_cmd_hs20_anqp_get(struct wpa_ctrl *ctrl, int argc,
2336 char *argv[])
2337{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002338 return wpa_cli_cmd(ctrl, "HS20_ANQP_GET", 2, argc, argv);
Dmitry Shmidt04949592012-07-19 12:16:46 -07002339}
2340
2341
2342static int wpa_cli_cmd_get_nai_home_realm_list(struct wpa_ctrl *ctrl, int argc,
2343 char *argv[])
2344{
2345 char cmd[512];
Dmitry Shmidt04949592012-07-19 12:16:46 -07002346
2347 if (argc == 0) {
2348 printf("Command needs one or two arguments (dst mac addr and "
2349 "optional home realm)\n");
2350 return -1;
2351 }
2352
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002353 if (write_cmd(cmd, sizeof(cmd), "HS20_GET_NAI_HOME_REALM_LIST",
2354 argc, argv) < 0)
Dmitry Shmidt04949592012-07-19 12:16:46 -07002355 return -1;
Dmitry Shmidt04949592012-07-19 12:16:46 -07002356
2357 return wpa_ctrl_command(ctrl, cmd);
2358}
2359
Dmitry Shmidtf21452a2014-02-26 10:55:25 -08002360
2361static int wpa_cli_cmd_hs20_icon_request(struct wpa_ctrl *ctrl, int argc,
2362 char *argv[])
2363{
2364 char cmd[512];
2365
2366 if (argc < 2) {
2367 printf("Command needs two arguments (dst mac addr and "
2368 "icon name)\n");
2369 return -1;
2370 }
2371
2372 if (write_cmd(cmd, sizeof(cmd), "HS20_ICON_REQUEST", argc, argv) < 0)
2373 return -1;
2374
2375 return wpa_ctrl_command(ctrl, cmd);
2376}
2377
2378
2379static int wpa_cli_cmd_fetch_osu(struct wpa_ctrl *ctrl, int argc, char *argv[])
2380{
2381 return wpa_ctrl_command(ctrl, "FETCH_OSU");
2382}
2383
2384
2385static int wpa_cli_cmd_cancel_fetch_osu(struct wpa_ctrl *ctrl, int argc,
2386 char *argv[])
2387{
2388 return wpa_ctrl_command(ctrl, "CANCEL_FETCH_OSU");
2389}
2390
Dmitry Shmidt04949592012-07-19 12:16:46 -07002391#endif /* CONFIG_HS20 */
2392
2393
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002394static int wpa_cli_cmd_sta_autoconnect(struct wpa_ctrl *ctrl, int argc,
2395 char *argv[])
2396{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002397 return wpa_cli_cmd(ctrl, "STA_AUTOCONNECT", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002398}
2399
2400
2401static int wpa_cli_cmd_tdls_discover(struct wpa_ctrl *ctrl, int argc,
2402 char *argv[])
2403{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002404 return wpa_cli_cmd(ctrl, "TDLS_DISCOVER", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002405}
2406
2407
2408static int wpa_cli_cmd_tdls_setup(struct wpa_ctrl *ctrl, int argc,
2409 char *argv[])
2410{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002411 return wpa_cli_cmd(ctrl, "TDLS_SETUP", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002412}
2413
2414
2415static int wpa_cli_cmd_tdls_teardown(struct wpa_ctrl *ctrl, int argc,
2416 char *argv[])
2417{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002418 return wpa_cli_cmd(ctrl, "TDLS_TEARDOWN", 1, argc, argv);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002419}
2420
2421
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002422static int wpa_cli_cmd_wmm_ac_addts(struct wpa_ctrl *ctrl, int argc,
2423 char *argv[])
2424{
2425 return wpa_cli_cmd(ctrl, "WMM_AC_ADDTS", 3, argc, argv);
2426}
2427
2428
2429static int wpa_cli_cmd_wmm_ac_delts(struct wpa_ctrl *ctrl, int argc,
2430 char *argv[])
2431{
2432 return wpa_cli_cmd(ctrl, "WMM_AC_DELTS", 1, argc, argv);
2433}
2434
2435
2436static int wpa_cli_cmd_wmm_ac_status(struct wpa_ctrl *ctrl, int argc,
2437 char *argv[])
2438{
2439 return wpa_ctrl_command(ctrl, "WMM_AC_STATUS");
2440}
2441
2442
2443static int wpa_cli_cmd_tdls_chan_switch(struct wpa_ctrl *ctrl, int argc,
2444 char *argv[])
2445{
2446 return wpa_cli_cmd(ctrl, "TDLS_CHAN_SWITCH", 2, argc, argv);
2447}
2448
2449
2450static int wpa_cli_cmd_tdls_cancel_chan_switch(struct wpa_ctrl *ctrl, int argc,
2451 char *argv[])
2452{
2453 return wpa_cli_cmd(ctrl, "TDLS_CANCEL_CHAN_SWITCH", 1, argc, argv);
2454}
2455
2456
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002457static int wpa_cli_cmd_signal_poll(struct wpa_ctrl *ctrl, int argc,
2458 char *argv[])
2459{
2460 return wpa_ctrl_command(ctrl, "SIGNAL_POLL");
2461}
2462
2463
Yuhao Zhengfcd6f212012-07-27 10:37:52 -07002464static int wpa_cli_cmd_pktcnt_poll(struct wpa_ctrl *ctrl, int argc,
2465 char *argv[])
2466{
2467 return wpa_ctrl_command(ctrl, "PKTCNT_POLL");
2468}
2469
2470
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002471static int wpa_cli_cmd_reauthenticate(struct wpa_ctrl *ctrl, int argc,
2472 char *argv[])
2473{
2474 return wpa_ctrl_command(ctrl, "REAUTHENTICATE");
2475}
2476
Dmitry Shmidt04949592012-07-19 12:16:46 -07002477
2478#ifdef CONFIG_AUTOSCAN
2479
2480static int wpa_cli_cmd_autoscan(struct wpa_ctrl *ctrl, int argc, char *argv[])
2481{
Dmitry Shmidt04949592012-07-19 12:16:46 -07002482 if (argc == 0)
2483 return wpa_ctrl_command(ctrl, "AUTOSCAN ");
2484
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002485 return wpa_cli_cmd(ctrl, "AUTOSCAN", 0, argc, argv);
Dmitry Shmidt04949592012-07-19 12:16:46 -07002486}
2487
2488#endif /* CONFIG_AUTOSCAN */
2489
2490
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08002491#ifdef CONFIG_WNM
2492
2493static int wpa_cli_cmd_wnm_sleep(struct wpa_ctrl *ctrl, int argc, char *argv[])
2494{
2495 return wpa_cli_cmd(ctrl, "WNM_SLEEP", 0, argc, argv);
2496}
2497
Dmitry Shmidt44c95782013-05-17 09:51:35 -07002498
2499static int wpa_cli_cmd_wnm_bss_query(struct wpa_ctrl *ctrl, int argc, char *argv[])
2500{
2501 return wpa_cli_cmd(ctrl, "WNM_BSS_QUERY", 1, argc, argv);
2502}
2503
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08002504#endif /* CONFIG_WNM */
2505
2506
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002507static int wpa_cli_cmd_raw(struct wpa_ctrl *ctrl, int argc, char *argv[])
2508{
2509 if (argc == 0)
2510 return -1;
2511 return wpa_cli_cmd(ctrl, argv[0], 0, argc - 1, &argv[1]);
2512}
2513
2514
Dmitry Shmidtc5ec7f52012-03-06 16:33:24 -08002515#ifdef ANDROID
Dmitry Shmidt0716c122011-04-08 15:03:17 -07002516static int wpa_cli_cmd_driver(struct wpa_ctrl *ctrl, int argc, char *argv[])
2517{
Dmitry Shmidt292b0c32013-11-22 12:54:42 -08002518 return wpa_cli_cmd(ctrl, "DRIVER", 1, argc, argv);
Dmitry Shmidt0716c122011-04-08 15:03:17 -07002519}
Dmitry Shmidt292b0c32013-11-22 12:54:42 -08002520#endif /* ANDROID */
Dmitry Shmidt0716c122011-04-08 15:03:17 -07002521
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002522
Dmitry Shmidta38abf92014-03-06 13:38:44 -08002523static int wpa_cli_cmd_vendor(struct wpa_ctrl *ctrl, int argc, char *argv[])
2524{
2525 return wpa_cli_cmd(ctrl, "VENDOR", 1, argc, argv);
2526}
2527
2528
Dmitry Shmidt444d5672013-04-01 13:08:44 -07002529static int wpa_cli_cmd_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
2530{
2531 return wpa_ctrl_command(ctrl, "FLUSH");
2532}
2533
2534
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08002535static int wpa_cli_cmd_radio_work(struct wpa_ctrl *ctrl, int argc, char *argv[])
2536{
2537 return wpa_cli_cmd(ctrl, "RADIO_WORK", 1, argc, argv);
2538}
2539
2540
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002541static int wpa_cli_cmd_neighbor_rep_request(struct wpa_ctrl *ctrl, int argc,
2542 char *argv[])
2543{
2544 return wpa_cli_cmd(ctrl, "NEIGHBOR_REP_REQUEST", 0, argc, argv);
2545}
2546
2547
2548static int wpa_cli_cmd_erp_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
2549{
2550 return wpa_ctrl_command(ctrl, "ERP_FLUSH");
2551}
2552
2553
2554static int wpa_cli_cmd_mac_rand_scan(struct wpa_ctrl *ctrl, int argc,
2555 char *argv[])
2556{
2557 return wpa_cli_cmd(ctrl, "MAC_RAND_SCAN", 1, argc, argv);
2558}
2559
2560
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002561enum wpa_cli_cmd_flags {
2562 cli_cmd_flag_none = 0x00,
2563 cli_cmd_flag_sensitive = 0x01
2564};
2565
2566struct wpa_cli_cmd {
2567 const char *cmd;
2568 int (*handler)(struct wpa_ctrl *ctrl, int argc, char *argv[]);
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002569 char ** (*completion)(const char *str, int pos);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002570 enum wpa_cli_cmd_flags flags;
2571 const char *usage;
2572};
2573
2574static struct wpa_cli_cmd wpa_cli_commands[] = {
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002575 { "status", wpa_cli_cmd_status, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002576 cli_cmd_flag_none,
2577 "[verbose] = get current WPA/EAPOL/EAP status" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002578 { "ifname", wpa_cli_cmd_ifname, NULL,
2579 cli_cmd_flag_none,
2580 "= get current interface name" },
2581 { "ping", wpa_cli_cmd_ping, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002582 cli_cmd_flag_none,
2583 "= pings wpa_supplicant" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002584 { "relog", wpa_cli_cmd_relog, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002585 cli_cmd_flag_none,
2586 "= re-open log-file (allow rolling logs)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002587 { "note", wpa_cli_cmd_note, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002588 cli_cmd_flag_none,
2589 "<text> = add a note to wpa_supplicant debug log" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002590 { "mib", wpa_cli_cmd_mib, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002591 cli_cmd_flag_none,
2592 "= get MIB variables (dot1x, dot11)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002593 { "help", wpa_cli_cmd_help, wpa_cli_complete_help,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002594 cli_cmd_flag_none,
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002595 "[command] = show usage help" },
2596 { "interface", wpa_cli_cmd_interface, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002597 cli_cmd_flag_none,
2598 "[ifname] = show interfaces/select interface" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002599 { "level", wpa_cli_cmd_level, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002600 cli_cmd_flag_none,
2601 "<debug level> = change debug level" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002602 { "license", wpa_cli_cmd_license, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002603 cli_cmd_flag_none,
2604 "= show full wpa_cli license" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002605 { "quit", wpa_cli_cmd_quit, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002606 cli_cmd_flag_none,
2607 "= exit wpa_cli" },
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -07002608 { "set", wpa_cli_cmd_set, wpa_cli_complete_set,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002609 cli_cmd_flag_none,
2610 "= set variables (shows list of variables when run without "
2611 "arguments)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002612 { "get", wpa_cli_cmd_get, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002613 cli_cmd_flag_none,
2614 "<name> = get information" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002615 { "logon", wpa_cli_cmd_logon, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002616 cli_cmd_flag_none,
2617 "= IEEE 802.1X EAPOL state machine logon" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002618 { "logoff", wpa_cli_cmd_logoff, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002619 cli_cmd_flag_none,
2620 "= IEEE 802.1X EAPOL state machine logoff" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002621 { "pmksa", wpa_cli_cmd_pmksa, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002622 cli_cmd_flag_none,
2623 "= show PMKSA cache" },
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07002624 { "pmksa_flush", wpa_cli_cmd_pmksa_flush, NULL,
2625 cli_cmd_flag_none,
2626 "= flush PMKSA cache entries" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002627 { "reassociate", wpa_cli_cmd_reassociate, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002628 cli_cmd_flag_none,
2629 "= force reassociation" },
Dmitry Shmidt98660862014-03-11 17:26:21 -07002630 { "reattach", wpa_cli_cmd_reattach, NULL,
2631 cli_cmd_flag_none,
2632 "= force reassociation back to the same BSS" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002633 { "preauthenticate", wpa_cli_cmd_preauthenticate, wpa_cli_complete_bss,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002634 cli_cmd_flag_none,
2635 "<BSSID> = force preauthentication" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002636 { "identity", wpa_cli_cmd_identity, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002637 cli_cmd_flag_none,
2638 "<network id> <identity> = configure identity for an SSID" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002639 { "password", wpa_cli_cmd_password, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002640 cli_cmd_flag_sensitive,
2641 "<network id> <password> = configure password for an SSID" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002642 { "new_password", wpa_cli_cmd_new_password, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002643 cli_cmd_flag_sensitive,
2644 "<network id> <password> = change password for an SSID" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002645 { "pin", wpa_cli_cmd_pin, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002646 cli_cmd_flag_sensitive,
2647 "<network id> <pin> = configure pin for an SSID" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002648 { "otp", wpa_cli_cmd_otp, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002649 cli_cmd_flag_sensitive,
2650 "<network id> <password> = configure one-time-password for an SSID"
2651 },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002652 { "passphrase", wpa_cli_cmd_passphrase, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002653 cli_cmd_flag_sensitive,
2654 "<network id> <passphrase> = configure private key passphrase\n"
2655 " for an SSID" },
Dmitry Shmidt051af732013-10-22 13:52:46 -07002656 { "sim", wpa_cli_cmd_sim, NULL,
2657 cli_cmd_flag_sensitive,
2658 "<network id> <pin> = report SIM operation result" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002659 { "bssid", wpa_cli_cmd_bssid, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002660 cli_cmd_flag_none,
2661 "<network id> <BSSID> = set preferred BSSID for an SSID" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002662 { "blacklist", wpa_cli_cmd_blacklist, wpa_cli_complete_bss,
Dmitry Shmidt696359e2011-03-16 15:04:31 -07002663 cli_cmd_flag_none,
2664 "<BSSID> = add a BSSID to the blacklist\n"
2665 "blacklist clear = clear the blacklist\n"
2666 "blacklist = display the blacklist" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002667 { "log_level", wpa_cli_cmd_log_level, NULL,
Dmitry Shmidt696359e2011-03-16 15:04:31 -07002668 cli_cmd_flag_none,
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002669 "<level> [<timestamp>] = update the log level/timestamp\n"
Dmitry Shmidt696359e2011-03-16 15:04:31 -07002670 "log_level = display the current log level and log options" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002671 { "list_networks", wpa_cli_cmd_list_networks, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002672 cli_cmd_flag_none,
2673 "= list configured networks" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002674 { "select_network", wpa_cli_cmd_select_network, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002675 cli_cmd_flag_none,
2676 "<network id> = select a network (disable others)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002677 { "enable_network", wpa_cli_cmd_enable_network, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002678 cli_cmd_flag_none,
2679 "<network id> = enable a network" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002680 { "disable_network", wpa_cli_cmd_disable_network, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002681 cli_cmd_flag_none,
2682 "<network id> = disable a network" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002683 { "add_network", wpa_cli_cmd_add_network, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002684 cli_cmd_flag_none,
2685 "= add a network" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002686 { "remove_network", wpa_cli_cmd_remove_network, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002687 cli_cmd_flag_none,
2688 "<network id> = remove a network" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002689 { "set_network", wpa_cli_cmd_set_network, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002690 cli_cmd_flag_sensitive,
2691 "<network id> <variable> <value> = set network variables (shows\n"
2692 " list of variables when run without arguments)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002693 { "get_network", wpa_cli_cmd_get_network, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002694 cli_cmd_flag_none,
2695 "<network id> <variable> = get network variables" },
Dmitry Shmidt684785c2014-05-12 13:34:29 -07002696 { "dup_network", wpa_cli_cmd_dup_network, NULL,
2697 cli_cmd_flag_none,
2698 "<src network id> <dst network id> <variable> = duplicate network variables"
2699 },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002700 { "list_creds", wpa_cli_cmd_list_creds, NULL,
Dmitry Shmidt04949592012-07-19 12:16:46 -07002701 cli_cmd_flag_none,
2702 "= list configured credentials" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002703 { "add_cred", wpa_cli_cmd_add_cred, NULL,
Dmitry Shmidt04949592012-07-19 12:16:46 -07002704 cli_cmd_flag_none,
2705 "= add a credential" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002706 { "remove_cred", wpa_cli_cmd_remove_cred, NULL,
Dmitry Shmidt04949592012-07-19 12:16:46 -07002707 cli_cmd_flag_none,
2708 "<cred id> = remove a credential" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002709 { "set_cred", wpa_cli_cmd_set_cred, NULL,
Dmitry Shmidt04949592012-07-19 12:16:46 -07002710 cli_cmd_flag_sensitive,
2711 "<cred id> <variable> <value> = set credential variables" },
Dmitry Shmidt0cfd5f72014-04-04 14:48:05 -07002712 { "get_cred", wpa_cli_cmd_get_cred, NULL,
2713 cli_cmd_flag_none,
2714 "<cred id> <variable> = get credential variables" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002715 { "save_config", wpa_cli_cmd_save_config, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002716 cli_cmd_flag_none,
2717 "= save the current configuration" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002718 { "disconnect", wpa_cli_cmd_disconnect, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002719 cli_cmd_flag_none,
2720 "= disconnect and wait for reassociate/reconnect command before\n"
2721 " connecting" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002722 { "reconnect", wpa_cli_cmd_reconnect, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002723 cli_cmd_flag_none,
2724 "= like reassociate, but only takes effect if already disconnected"
2725 },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002726 { "scan", wpa_cli_cmd_scan, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002727 cli_cmd_flag_none,
2728 "= request new BSS scan" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002729 { "scan_results", wpa_cli_cmd_scan_results, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002730 cli_cmd_flag_none,
2731 "= get latest scan results" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002732 { "bss", wpa_cli_cmd_bss, wpa_cli_complete_bss,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002733 cli_cmd_flag_none,
2734 "<<idx> | <bssid>> = get detailed scan result info" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002735 { "get_capability", wpa_cli_cmd_get_capability, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002736 cli_cmd_flag_none,
Dmitry Shmidt8da800a2013-04-24 12:57:01 -07002737 "<eap/pairwise/group/key_mgmt/proto/auth_alg/channels/freq/modes> "
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002738 "= get capabilies" },
2739 { "reconfigure", wpa_cli_cmd_reconfigure, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002740 cli_cmd_flag_none,
2741 "= force wpa_supplicant to re-read its configuration file" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002742 { "terminate", wpa_cli_cmd_terminate, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002743 cli_cmd_flag_none,
2744 "= terminate wpa_supplicant" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002745 { "interface_add", wpa_cli_cmd_interface_add, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002746 cli_cmd_flag_none,
2747 "<ifname> <confname> <driver> <ctrl_interface> <driver_param>\n"
2748 " <bridge_name> = adds new interface, all parameters but <ifname>\n"
2749 " are optional" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002750 { "interface_remove", wpa_cli_cmd_interface_remove, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002751 cli_cmd_flag_none,
2752 "<ifname> = removes the interface" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002753 { "interface_list", wpa_cli_cmd_interface_list, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002754 cli_cmd_flag_none,
2755 "= list available interfaces" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002756 { "ap_scan", wpa_cli_cmd_ap_scan, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002757 cli_cmd_flag_none,
2758 "<value> = set ap_scan parameter" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002759 { "scan_interval", wpa_cli_cmd_scan_interval, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002760 cli_cmd_flag_none,
2761 "<value> = set scan_interval parameter (in seconds)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002762 { "bss_expire_age", wpa_cli_cmd_bss_expire_age, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002763 cli_cmd_flag_none,
2764 "<value> = set BSS expiration age parameter" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002765 { "bss_expire_count", wpa_cli_cmd_bss_expire_count, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002766 cli_cmd_flag_none,
2767 "<value> = set BSS expiration scan count parameter" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002768 { "bss_flush", wpa_cli_cmd_bss_flush, NULL,
Dmitry Shmidtf48e4f92012-08-24 11:14:44 -07002769 cli_cmd_flag_none,
2770 "<value> = set BSS flush age (0 by default)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002771 { "stkstart", wpa_cli_cmd_stkstart, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002772 cli_cmd_flag_none,
2773 "<addr> = request STK negotiation with <addr>" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002774 { "ft_ds", wpa_cli_cmd_ft_ds, wpa_cli_complete_bss,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002775 cli_cmd_flag_none,
2776 "<addr> = request over-the-DS FT with <addr>" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002777 { "wps_pbc", wpa_cli_cmd_wps_pbc, wpa_cli_complete_bss,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002778 cli_cmd_flag_none,
2779 "[BSSID] = start Wi-Fi Protected Setup: Push Button Configuration" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002780 { "wps_pin", wpa_cli_cmd_wps_pin, wpa_cli_complete_bss,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002781 cli_cmd_flag_sensitive,
2782 "<BSSID> [PIN] = start WPS PIN method (returns PIN, if not "
2783 "hardcoded)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002784 { "wps_check_pin", wpa_cli_cmd_wps_check_pin, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002785 cli_cmd_flag_sensitive,
2786 "<PIN> = verify PIN checksum" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002787 { "wps_cancel", wpa_cli_cmd_wps_cancel, NULL, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002788 "Cancels the pending WPS operation" },
Dmitry Shmidt04949592012-07-19 12:16:46 -07002789#ifdef CONFIG_WPS_NFC
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002790 { "wps_nfc", wpa_cli_cmd_wps_nfc, wpa_cli_complete_bss,
Dmitry Shmidt04949592012-07-19 12:16:46 -07002791 cli_cmd_flag_none,
2792 "[BSSID] = start Wi-Fi Protected Setup: NFC" },
Dmitry Shmidtf8623282013-02-20 14:34:59 -08002793 { "wps_nfc_config_token", wpa_cli_cmd_wps_nfc_config_token, NULL,
2794 cli_cmd_flag_none,
2795 "<WPS|NDEF> = build configuration token" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002796 { "wps_nfc_token", wpa_cli_cmd_wps_nfc_token, NULL,
Dmitry Shmidt04949592012-07-19 12:16:46 -07002797 cli_cmd_flag_none,
2798 "<WPS|NDEF> = create password token" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002799 { "wps_nfc_tag_read", wpa_cli_cmd_wps_nfc_tag_read, NULL,
Dmitry Shmidt04949592012-07-19 12:16:46 -07002800 cli_cmd_flag_sensitive,
2801 "<hexdump of payload> = report read NFC tag with WPS data" },
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08002802 { "nfc_get_handover_req", wpa_cli_cmd_nfc_get_handover_req, NULL,
2803 cli_cmd_flag_none,
2804 "<NDEF> <WPS> = create NFC handover request" },
2805 { "nfc_get_handover_sel", wpa_cli_cmd_nfc_get_handover_sel, NULL,
2806 cli_cmd_flag_none,
2807 "<NDEF> <WPS> = create NFC handover select" },
Dmitry Shmidtf8623282013-02-20 14:34:59 -08002808 { "nfc_report_handover", wpa_cli_cmd_nfc_report_handover, NULL,
2809 cli_cmd_flag_none,
2810 "<role> <type> <hexdump of req> <hexdump of sel> = report completed "
2811 "NFC handover" },
Dmitry Shmidt04949592012-07-19 12:16:46 -07002812#endif /* CONFIG_WPS_NFC */
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002813 { "wps_reg", wpa_cli_cmd_wps_reg, wpa_cli_complete_bss,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002814 cli_cmd_flag_sensitive,
2815 "<BSSID> <AP PIN> = start WPS Registrar to configure an AP" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002816 { "wps_ap_pin", wpa_cli_cmd_wps_ap_pin, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002817 cli_cmd_flag_sensitive,
2818 "[params..] = enable/disable AP PIN" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002819 { "wps_er_start", wpa_cli_cmd_wps_er_start, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002820 cli_cmd_flag_none,
2821 "[IP address] = start Wi-Fi Protected Setup External Registrar" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002822 { "wps_er_stop", wpa_cli_cmd_wps_er_stop, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002823 cli_cmd_flag_none,
2824 "= stop Wi-Fi Protected Setup External Registrar" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002825 { "wps_er_pin", wpa_cli_cmd_wps_er_pin, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002826 cli_cmd_flag_sensitive,
2827 "<UUID> <PIN> = add an Enrollee PIN to External Registrar" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002828 { "wps_er_pbc", wpa_cli_cmd_wps_er_pbc, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002829 cli_cmd_flag_none,
2830 "<UUID> = accept an Enrollee PBC using External Registrar" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002831 { "wps_er_learn", wpa_cli_cmd_wps_er_learn, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002832 cli_cmd_flag_sensitive,
2833 "<UUID> <PIN> = learn AP configuration" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002834 { "wps_er_set_config", wpa_cli_cmd_wps_er_set_config, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002835 cli_cmd_flag_none,
2836 "<UUID> <network id> = set AP configuration for enrolling" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002837 { "wps_er_config", wpa_cli_cmd_wps_er_config, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002838 cli_cmd_flag_sensitive,
2839 "<UUID> <PIN> <SSID> <auth> <encr> <key> = configure AP" },
Dmitry Shmidt04949592012-07-19 12:16:46 -07002840#ifdef CONFIG_WPS_NFC
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002841 { "wps_er_nfc_config_token", wpa_cli_cmd_wps_er_nfc_config_token, NULL,
Dmitry Shmidt04949592012-07-19 12:16:46 -07002842 cli_cmd_flag_none,
2843 "<WPS/NDEF> <UUID> = build NFC configuration token" },
2844#endif /* CONFIG_WPS_NFC */
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002845 { "ibss_rsn", wpa_cli_cmd_ibss_rsn, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002846 cli_cmd_flag_none,
2847 "<addr> = request RSN authentication with <addr> in IBSS" },
2848#ifdef CONFIG_AP
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002849 { "sta", wpa_cli_cmd_sta, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002850 cli_cmd_flag_none,
2851 "<addr> = get information about an associated station (AP)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002852 { "all_sta", wpa_cli_cmd_all_sta, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002853 cli_cmd_flag_none,
2854 "= get information about all associated stations (AP)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002855 { "deauthenticate", wpa_cli_cmd_deauthenticate, NULL,
Dmitry Shmidt04949592012-07-19 12:16:46 -07002856 cli_cmd_flag_none,
2857 "<addr> = deauthenticate a station" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002858 { "disassociate", wpa_cli_cmd_disassociate, NULL,
Dmitry Shmidt04949592012-07-19 12:16:46 -07002859 cli_cmd_flag_none,
2860 "<addr> = disassociate a station" },
Dmitry Shmidte0e48dc2013-11-18 12:00:06 -08002861 { "chan_switch", wpa_cli_cmd_chanswitch, NULL,
2862 cli_cmd_flag_none,
2863 "<cs_count> <freq> [sec_channel_offset=] [center_freq1=]"
2864 " [center_freq2=] [bandwidth=] [blocktx] [ht|vht]"
2865 " = CSA parameters" },
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002866#endif /* CONFIG_AP */
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002867 { "suspend", wpa_cli_cmd_suspend, NULL, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002868 "= notification of suspend/hibernate" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002869 { "resume", wpa_cli_cmd_resume, NULL, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002870 "= notification of resume/thaw" },
Dmitry Shmidt21de2142014-04-08 10:50:52 -07002871#ifdef CONFIG_TESTING_OPTIONS
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002872 { "drop_sa", wpa_cli_cmd_drop_sa, NULL, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002873 "= drop SA without deauth/disassoc (test command)" },
Dmitry Shmidt21de2142014-04-08 10:50:52 -07002874#endif /* CONFIG_TESTING_OPTIONS */
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002875 { "roam", wpa_cli_cmd_roam, wpa_cli_complete_bss,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002876 cli_cmd_flag_none,
2877 "<addr> = roam to the specified BSS" },
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002878#ifdef CONFIG_MESH
2879 { "mesh_interface_add", wpa_cli_cmd_mesh_interface_add, NULL,
2880 cli_cmd_flag_none,
2881 "[ifname] = Create a new mesh interface" },
2882 { "mesh_group_add", wpa_cli_cmd_mesh_group_add, NULL,
2883 cli_cmd_flag_none,
2884 "<network id> = join a mesh network (disable others)" },
2885 { "mesh_group_remove", wpa_cli_cmd_mesh_group_remove, NULL,
2886 cli_cmd_flag_none,
2887 "<ifname> = Remove mesh group interface" },
2888#endif /* CONFIG_MESH */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002889#ifdef CONFIG_P2P
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002890 { "p2p_find", wpa_cli_cmd_p2p_find, wpa_cli_complete_p2p_find,
2891 cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002892 "[timeout] [type=*] = find P2P Devices for up-to timeout seconds" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002893 { "p2p_stop_find", wpa_cli_cmd_p2p_stop_find, NULL, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002894 "= stop P2P Devices search" },
Dmitry Shmidt216983b2015-02-06 10:50:36 -08002895 { "p2p_asp_provision", wpa_cli_cmd_p2p_asp_provision, NULL,
2896 cli_cmd_flag_none,
2897 "<addr> adv_id=<adv_id> conncap=<conncap> [info=<infodata>] = provision with a P2P ASP Device" },
2898 { "p2p_asp_provision_resp", wpa_cli_cmd_p2p_asp_provision_resp, NULL,
2899 cli_cmd_flag_none,
2900 "<addr> adv_id=<adv_id> [role<conncap>] [info=<infodata>] = provision with a P2P ASP Device" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002901 { "p2p_connect", wpa_cli_cmd_p2p_connect, wpa_cli_complete_p2p_connect,
2902 cli_cmd_flag_none,
2903 "<addr> <\"pbc\"|PIN> [ht40] = connect to a P2P Device" },
2904 { "p2p_listen", wpa_cli_cmd_p2p_listen, NULL, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002905 "[timeout] = listen for P2P Devices for up-to timeout seconds" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002906 { "p2p_group_remove", wpa_cli_cmd_p2p_group_remove,
2907 wpa_cli_complete_p2p_group_remove, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002908 "<ifname> = remove P2P group interface (terminate group if GO)" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002909 { "p2p_group_add", wpa_cli_cmd_p2p_group_add, NULL, cli_cmd_flag_none,
2910 "[ht40] = add a new P2P group (local end as GO)" },
2911 { "p2p_prov_disc", wpa_cli_cmd_p2p_prov_disc,
2912 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002913 "<addr> <method> = request provisioning discovery" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002914 { "p2p_get_passphrase", wpa_cli_cmd_p2p_get_passphrase, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002915 cli_cmd_flag_none,
2916 "= get the passphrase for a group (GO only)" },
2917 { "p2p_serv_disc_req", wpa_cli_cmd_p2p_serv_disc_req,
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002918 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002919 "<addr> <TLVs> = schedule service discovery request" },
2920 { "p2p_serv_disc_cancel_req", wpa_cli_cmd_p2p_serv_disc_cancel_req,
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002921 NULL, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002922 "<id> = cancel pending service discovery request" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002923 { "p2p_serv_disc_resp", wpa_cli_cmd_p2p_serv_disc_resp, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002924 cli_cmd_flag_none,
2925 "<freq> <addr> <dialog token> <TLVs> = service discovery response" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002926 { "p2p_service_update", wpa_cli_cmd_p2p_service_update, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002927 cli_cmd_flag_none,
2928 "= indicate change in local services" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002929 { "p2p_serv_disc_external", wpa_cli_cmd_p2p_serv_disc_external, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002930 cli_cmd_flag_none,
2931 "<external> = set external processing of service discovery" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002932 { "p2p_service_flush", wpa_cli_cmd_p2p_service_flush, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002933 cli_cmd_flag_none,
2934 "= remove all stored service entries" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002935 { "p2p_service_add", wpa_cli_cmd_p2p_service_add, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002936 cli_cmd_flag_none,
Dmitry Shmidt216983b2015-02-06 10:50:36 -08002937 "<bonjour|upnp|asp> <query|version> <response|service> = add a local "
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002938 "service" },
Dmitry Shmidt216983b2015-02-06 10:50:36 -08002939 { "p2p_service_rep", wpa_cli_cmd_p2p_service_rep, NULL,
2940 cli_cmd_flag_none,
2941 "asp <auto> <adv_id> <svc_state> <svc_string> [<svc_info>] = replace "
2942 "local ASP service" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002943 { "p2p_service_del", wpa_cli_cmd_p2p_service_del, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002944 cli_cmd_flag_none,
2945 "<bonjour|upnp> <query|version> [|service] = remove a local "
2946 "service" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002947 { "p2p_reject", wpa_cli_cmd_p2p_reject, wpa_cli_complete_p2p_peer,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002948 cli_cmd_flag_none,
2949 "<addr> = reject connection attempts from a specific peer" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002950 { "p2p_invite", wpa_cli_cmd_p2p_invite, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002951 cli_cmd_flag_none,
2952 "<cmd> [peer=addr] = invite peer" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002953 { "p2p_peers", wpa_cli_cmd_p2p_peers, NULL, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002954 "[discovered] = list known (optionally, only fully discovered) P2P "
2955 "peers" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002956 { "p2p_peer", wpa_cli_cmd_p2p_peer, wpa_cli_complete_p2p_peer,
2957 cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002958 "<address> = show information about known P2P peer" },
Dmitry Shmidt391c59f2013-09-03 12:16:28 -07002959 { "p2p_set", wpa_cli_cmd_p2p_set, wpa_cli_complete_p2p_set,
2960 cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002961 "<field> <value> = set a P2P parameter" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002962 { "p2p_flush", wpa_cli_cmd_p2p_flush, NULL, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002963 "= flush P2P state" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002964 { "p2p_cancel", wpa_cli_cmd_p2p_cancel, NULL, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002965 "= cancel P2P group formation" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002966 { "p2p_unauthorize", wpa_cli_cmd_p2p_unauthorize,
2967 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002968 "<address> = unauthorize a peer" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002969 { "p2p_presence_req", wpa_cli_cmd_p2p_presence_req, NULL,
2970 cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002971 "[<duration> <interval>] [<duration> <interval>] = request GO "
2972 "presence" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002973 { "p2p_ext_listen", wpa_cli_cmd_p2p_ext_listen, NULL,
2974 cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002975 "[<period> <interval>] = set extended listen timing" },
Dmitry Shmidt391c59f2013-09-03 12:16:28 -07002976 { "p2p_remove_client", wpa_cli_cmd_p2p_remove_client,
2977 wpa_cli_complete_p2p_peer, cli_cmd_flag_none,
2978 "<address|iface=address> = remove a peer from all groups" },
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002979#endif /* CONFIG_P2P */
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002980#ifdef CONFIG_WIFI_DISPLAY
2981 { "wfd_subelem_set", wpa_cli_cmd_wfd_subelem_set, NULL,
2982 cli_cmd_flag_none,
2983 "<subelem> [contents] = set Wi-Fi Display subelement" },
2984 { "wfd_subelem_get", wpa_cli_cmd_wfd_subelem_get, NULL,
2985 cli_cmd_flag_none,
2986 "<subelem> = get Wi-Fi Display subelement" },
2987#endif /* CONFIG_WIFI_DISPLAY */
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002988#ifdef CONFIG_INTERWORKING
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002989 { "fetch_anqp", wpa_cli_cmd_fetch_anqp, NULL, cli_cmd_flag_none,
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002990 "= fetch ANQP information for all APs" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002991 { "stop_fetch_anqp", wpa_cli_cmd_stop_fetch_anqp, NULL,
2992 cli_cmd_flag_none,
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002993 "= stop fetch_anqp operation" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002994 { "interworking_select", wpa_cli_cmd_interworking_select, NULL,
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002995 cli_cmd_flag_none,
2996 "[auto] = perform Interworking network selection" },
2997 { "interworking_connect", wpa_cli_cmd_interworking_connect,
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002998 wpa_cli_complete_bss, cli_cmd_flag_none,
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002999 "<BSSID> = connect using Interworking credentials" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003000 { "anqp_get", wpa_cli_cmd_anqp_get, wpa_cli_complete_bss,
3001 cli_cmd_flag_none,
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003002 "<addr> <info id>[,<info id>]... = request ANQP information" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003003 { "gas_request", wpa_cli_cmd_gas_request, wpa_cli_complete_bss,
3004 cli_cmd_flag_none,
3005 "<addr> <AdvProtoID> [QueryReq] = GAS request" },
3006 { "gas_response_get", wpa_cli_cmd_gas_response_get,
3007 wpa_cli_complete_bss, cli_cmd_flag_none,
3008 "<addr> <dialog token> [start,len] = Fetch last GAS response" },
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003009#endif /* CONFIG_INTERWORKING */
Dmitry Shmidt04949592012-07-19 12:16:46 -07003010#ifdef CONFIG_HS20
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003011 { "hs20_anqp_get", wpa_cli_cmd_hs20_anqp_get, wpa_cli_complete_bss,
3012 cli_cmd_flag_none,
Dmitry Shmidt04949592012-07-19 12:16:46 -07003013 "<addr> <subtype>[,<subtype>]... = request HS 2.0 ANQP information"
3014 },
3015 { "nai_home_realm_list", wpa_cli_cmd_get_nai_home_realm_list,
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003016 wpa_cli_complete_bss, cli_cmd_flag_none,
Dmitry Shmidt04949592012-07-19 12:16:46 -07003017 "<addr> <home realm> = get HS20 nai home realm list" },
Dmitry Shmidtf21452a2014-02-26 10:55:25 -08003018 { "hs20_icon_request", wpa_cli_cmd_hs20_icon_request,
3019 wpa_cli_complete_bss, cli_cmd_flag_none,
3020 "<addr> <icon name> = get Hotspot 2.0 OSU icon" },
3021 { "fetch_osu", wpa_cli_cmd_fetch_osu, NULL, cli_cmd_flag_none,
3022 "= fetch OSU provider information from all APs" },
3023 { "cancel_fetch_osu", wpa_cli_cmd_cancel_fetch_osu, NULL,
3024 cli_cmd_flag_none,
3025 "= cancel fetch_osu command" },
Dmitry Shmidt04949592012-07-19 12:16:46 -07003026#endif /* CONFIG_HS20 */
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003027 { "sta_autoconnect", wpa_cli_cmd_sta_autoconnect, NULL,
3028 cli_cmd_flag_none,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003029 "<0/1> = disable/enable automatic reconnection" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003030 { "tdls_discover", wpa_cli_cmd_tdls_discover, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003031 cli_cmd_flag_none,
3032 "<addr> = request TDLS discovery with <addr>" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003033 { "tdls_setup", wpa_cli_cmd_tdls_setup, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003034 cli_cmd_flag_none,
3035 "<addr> = request TDLS setup with <addr>" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003036 { "tdls_teardown", wpa_cli_cmd_tdls_teardown, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003037 cli_cmd_flag_none,
3038 "<addr> = tear down TDLS with <addr>" },
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08003039 { "wmm_ac_addts", wpa_cli_cmd_wmm_ac_addts, NULL,
3040 cli_cmd_flag_none,
3041 "<uplink/downlink/bidi> <tsid=0..7> <up=0..7> [nominal_msdu_size=#] "
3042 "[mean_data_rate=#] [min_phy_rate=#] [sba=#] [fixed_nominal_msdu] "
3043 "= add WMM-AC traffic stream" },
3044 { "wmm_ac_delts", wpa_cli_cmd_wmm_ac_delts, NULL,
3045 cli_cmd_flag_none,
3046 "<tsid> = delete WMM-AC traffic stream" },
3047 { "wmm_ac_status", wpa_cli_cmd_wmm_ac_status, NULL,
3048 cli_cmd_flag_none,
3049 "= show status for Wireless Multi-Media Admission-Control" },
3050 { "tdls_chan_switch", wpa_cli_cmd_tdls_chan_switch, NULL,
3051 cli_cmd_flag_none,
3052 "<addr> <oper class> <freq> [sec_channel_offset=] [center_freq1=] "
3053 "[center_freq2=] [bandwidth=] [ht|vht] = enable channel switching "
3054 "with TDLS peer" },
3055 { "tdls_cancel_chan_switch", wpa_cli_cmd_tdls_cancel_chan_switch, NULL,
3056 cli_cmd_flag_none,
3057 "<addr> = disable channel switching with TDLS peer <addr>" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003058 { "signal_poll", wpa_cli_cmd_signal_poll, NULL,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003059 cli_cmd_flag_none,
3060 "= get signal parameters" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003061 { "pktcnt_poll", wpa_cli_cmd_pktcnt_poll, NULL,
Yuhao Zhengfcd6f212012-07-27 10:37:52 -07003062 cli_cmd_flag_none,
3063 "= get TX/RX packet counters" },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003064 { "reauthenticate", wpa_cli_cmd_reauthenticate, NULL,
3065 cli_cmd_flag_none,
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003066 "= trigger IEEE 802.1X/EAPOL reauthentication" },
Dmitry Shmidt04949592012-07-19 12:16:46 -07003067#ifdef CONFIG_AUTOSCAN
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003068 { "autoscan", wpa_cli_cmd_autoscan, NULL, cli_cmd_flag_none,
Dmitry Shmidt04949592012-07-19 12:16:46 -07003069 "[params] = Set or unset (if none) autoscan parameters" },
3070#endif /* CONFIG_AUTOSCAN */
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08003071#ifdef CONFIG_WNM
3072 { "wnm_sleep", wpa_cli_cmd_wnm_sleep, NULL, cli_cmd_flag_none,
3073 "<enter/exit> [interval=#] = enter/exit WNM-Sleep mode" },
Dmitry Shmidt44c95782013-05-17 09:51:35 -07003074 { "wnm_bss_query", wpa_cli_cmd_wnm_bss_query, NULL, cli_cmd_flag_none,
3075 "<query reason> = Send BSS Transition Management Query" },
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08003076#endif /* CONFIG_WNM */
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003077 { "raw", wpa_cli_cmd_raw, NULL, cli_cmd_flag_sensitive,
3078 "<params..> = Sent unprocessed command" },
Dmitry Shmidt444d5672013-04-01 13:08:44 -07003079 { "flush", wpa_cli_cmd_flush, NULL, cli_cmd_flag_none,
3080 "= flush wpa_supplicant state" },
Dmitry Shmidtc5ec7f52012-03-06 16:33:24 -08003081#ifdef ANDROID
Dmitry Shmidt292b0c32013-11-22 12:54:42 -08003082 { "driver", wpa_cli_cmd_driver, NULL, cli_cmd_flag_none,
Dmitry Shmidt0716c122011-04-08 15:03:17 -07003083 "<command> = driver private commands" },
Dmitry Shmidt292b0c32013-11-22 12:54:42 -08003084#endif /* ANDROID */
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08003085 { "radio_work", wpa_cli_cmd_radio_work, NULL, cli_cmd_flag_none,
3086 "= radio_work <show/add/done>" },
Dmitry Shmidta38abf92014-03-06 13:38:44 -08003087 { "vendor", wpa_cli_cmd_vendor, NULL, cli_cmd_flag_none,
3088 "<vendor id> <command id> [<hex formatted command argument>] = Send vendor command"
3089 },
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08003090 { "neighbor_rep_request",
3091 wpa_cli_cmd_neighbor_rep_request, NULL, cli_cmd_flag_none,
3092 "[ssid=<SSID>] = Trigger request to AP for neighboring AP report "
3093 "(with optional given SSID, default: current SSID)"
3094 },
3095 { "erp_flush", wpa_cli_cmd_erp_flush, NULL, cli_cmd_flag_none,
3096 "= flush ERP keys" },
3097 { "mac_rand_scan",
3098 wpa_cli_cmd_mac_rand_scan, NULL, cli_cmd_flag_none,
3099 "<scan|sched|pno|all> enable=<0/1> [addr=mac-address "
3100 "mask=mac-address-mask] = scan MAC randomization"
3101 },
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003102 { NULL, NULL, NULL, cli_cmd_flag_none, NULL }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003103};
3104
3105
3106/*
3107 * Prints command usage, lines are padded with the specified string.
3108 */
3109static void print_cmd_help(struct wpa_cli_cmd *cmd, const char *pad)
3110{
3111 char c;
3112 size_t n;
3113
3114 printf("%s%s ", pad, cmd->cmd);
3115 for (n = 0; (c = cmd->usage[n]); n++) {
3116 printf("%c", c);
3117 if (c == '\n')
3118 printf("%s", pad);
3119 }
3120 printf("\n");
3121}
3122
3123
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003124static void print_help(const char *cmd)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003125{
3126 int n;
3127 printf("commands:\n");
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003128 for (n = 0; wpa_cli_commands[n].cmd; n++) {
3129 if (cmd == NULL || str_starts(wpa_cli_commands[n].cmd, cmd))
3130 print_cmd_help(&wpa_cli_commands[n], " ");
3131 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003132}
3133
3134
3135static int wpa_cli_edit_filter_history_cb(void *ctx, const char *cmd)
3136{
3137 const char *c, *delim;
3138 int n;
3139 size_t len;
3140
3141 delim = os_strchr(cmd, ' ');
3142 if (delim)
3143 len = delim - cmd;
3144 else
3145 len = os_strlen(cmd);
3146
3147 for (n = 0; (c = wpa_cli_commands[n].cmd); n++) {
3148 if (os_strncasecmp(cmd, c, len) == 0 && len == os_strlen(c))
3149 return (wpa_cli_commands[n].flags &
3150 cli_cmd_flag_sensitive);
3151 }
3152 return 0;
3153}
3154
3155
3156static char ** wpa_list_cmd_list(void)
3157{
3158 char **res;
3159 int i, count;
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -07003160 struct cli_txt_entry *e;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003161
Dmitry Shmidt68d0e3e2013-10-28 17:59:21 -07003162 count = ARRAY_SIZE(wpa_cli_commands);
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -07003163 count += dl_list_len(&p2p_groups);
3164 count += dl_list_len(&ifnames);
3165 res = os_calloc(count + 1, sizeof(char *));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003166 if (res == NULL)
3167 return NULL;
3168
3169 for (i = 0; wpa_cli_commands[i].cmd; i++) {
3170 res[i] = os_strdup(wpa_cli_commands[i].cmd);
3171 if (res[i] == NULL)
3172 break;
3173 }
3174
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -07003175 dl_list_for_each(e, &p2p_groups, struct cli_txt_entry, list) {
3176 size_t len = 8 + os_strlen(e->txt);
3177 res[i] = os_malloc(len);
3178 if (res[i] == NULL)
3179 break;
3180 os_snprintf(res[i], len, "ifname=%s", e->txt);
3181 i++;
3182 }
3183
3184 dl_list_for_each(e, &ifnames, struct cli_txt_entry, list) {
3185 res[i] = os_strdup(e->txt);
3186 if (res[i] == NULL)
3187 break;
3188 i++;
3189 }
3190
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003191 return res;
3192}
3193
3194
3195static char ** wpa_cli_cmd_completion(const char *cmd, const char *str,
3196 int pos)
3197{
3198 int i;
3199
3200 for (i = 0; wpa_cli_commands[i].cmd; i++) {
3201 if (os_strcasecmp(wpa_cli_commands[i].cmd, cmd) == 0) {
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003202 if (wpa_cli_commands[i].completion)
3203 return wpa_cli_commands[i].completion(str,
3204 pos);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003205 edit_clear_line();
3206 printf("\r%s\n", wpa_cli_commands[i].usage);
3207 edit_redraw();
3208 break;
3209 }
3210 }
3211
3212 return NULL;
3213}
3214
3215
3216static char ** wpa_cli_edit_completion_cb(void *ctx, const char *str, int pos)
3217{
3218 char **res;
3219 const char *end;
3220 char *cmd;
3221
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -07003222 if (pos > 7 && os_strncasecmp(str, "IFNAME=", 7) == 0) {
3223 end = os_strchr(str, ' ');
3224 if (end && pos > end - str) {
3225 pos -= end - str + 1;
3226 str = end + 1;
3227 }
3228 }
3229
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003230 end = os_strchr(str, ' ');
3231 if (end == NULL || str + pos < end)
3232 return wpa_list_cmd_list();
3233
3234 cmd = os_malloc(pos + 1);
3235 if (cmd == NULL)
3236 return NULL;
3237 os_memcpy(cmd, str, pos);
3238 cmd[end - str] = '\0';
3239 res = wpa_cli_cmd_completion(cmd, str, pos);
3240 os_free(cmd);
3241 return res;
3242}
3243
3244
3245static int wpa_request(struct wpa_ctrl *ctrl, int argc, char *argv[])
3246{
3247 struct wpa_cli_cmd *cmd, *match = NULL;
3248 int count;
3249 int ret = 0;
3250
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -07003251 if (argc > 1 && os_strncasecmp(argv[0], "IFNAME=", 7) == 0) {
3252 ifname_prefix = argv[0] + 7;
3253 argv = &argv[1];
3254 argc--;
3255 } else
3256 ifname_prefix = NULL;
3257
3258 if (argc == 0)
3259 return -1;
3260
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003261 count = 0;
3262 cmd = wpa_cli_commands;
3263 while (cmd->cmd) {
3264 if (os_strncasecmp(cmd->cmd, argv[0], os_strlen(argv[0])) == 0)
3265 {
3266 match = cmd;
3267 if (os_strcasecmp(cmd->cmd, argv[0]) == 0) {
3268 /* we have an exact match */
3269 count = 1;
3270 break;
3271 }
3272 count++;
3273 }
3274 cmd++;
3275 }
3276
3277 if (count > 1) {
3278 printf("Ambiguous command '%s'; possible commands:", argv[0]);
3279 cmd = wpa_cli_commands;
3280 while (cmd->cmd) {
3281 if (os_strncasecmp(cmd->cmd, argv[0],
3282 os_strlen(argv[0])) == 0) {
3283 printf(" %s", cmd->cmd);
3284 }
3285 cmd++;
3286 }
3287 printf("\n");
3288 ret = 1;
3289 } else if (count == 0) {
3290 printf("Unknown command '%s'\n", argv[0]);
3291 ret = 1;
3292 } else {
3293 ret = match->handler(ctrl, argc - 1, &argv[1]);
3294 }
3295
3296 return ret;
3297}
3298
3299
3300static int str_match(const char *a, const char *b)
3301{
3302 return os_strncmp(a, b, os_strlen(b)) == 0;
3303}
3304
3305
3306static int wpa_cli_exec(const char *program, const char *arg1,
3307 const char *arg2)
3308{
Jouni Malinen772e12c2014-10-07 10:29:35 -07003309 char *arg;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003310 size_t len;
3311 int res;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003312
Jouni Malinen772e12c2014-10-07 10:29:35 -07003313 len = os_strlen(arg1) + os_strlen(arg2) + 2;
3314 arg = os_malloc(len);
3315 if (arg == NULL)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003316 return -1;
Jouni Malinen772e12c2014-10-07 10:29:35 -07003317 os_snprintf(arg, len, "%s %s", arg1, arg2);
3318 res = os_exec(program, arg, 1);
3319 os_free(arg);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003320
Jouni Malinen772e12c2014-10-07 10:29:35 -07003321 return res;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003322}
3323
3324
3325static void wpa_cli_action_process(const char *msg)
3326{
3327 const char *pos;
3328 char *copy = NULL, *id, *pos2;
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003329 const char *ifname = ctrl_ifname;
3330 char ifname_buf[100];
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003331
3332 pos = msg;
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003333 if (os_strncmp(pos, "IFNAME=", 7) == 0) {
3334 const char *end;
3335 end = os_strchr(pos + 7, ' ');
3336 if (end && (unsigned int) (end - pos) < sizeof(ifname_buf)) {
3337 pos += 7;
3338 os_memcpy(ifname_buf, pos, end - pos);
3339 ifname_buf[end - pos] = '\0';
3340 ifname = ifname_buf;
3341 pos = end + 1;
3342 }
3343 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003344 if (*pos == '<') {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003345 const char *prev = pos;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003346 /* skip priority */
3347 pos = os_strchr(pos, '>');
3348 if (pos)
3349 pos++;
3350 else
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003351 pos = prev;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003352 }
3353
3354 if (str_match(pos, WPA_EVENT_CONNECTED)) {
3355 int new_id = -1;
3356 os_unsetenv("WPA_ID");
3357 os_unsetenv("WPA_ID_STR");
3358 os_unsetenv("WPA_CTRL_DIR");
3359
3360 pos = os_strstr(pos, "[id=");
3361 if (pos)
3362 copy = os_strdup(pos + 4);
3363
3364 if (copy) {
3365 pos2 = id = copy;
3366 while (*pos2 && *pos2 != ' ')
3367 pos2++;
3368 *pos2++ = '\0';
3369 new_id = atoi(id);
3370 os_setenv("WPA_ID", id, 1);
3371 while (*pos2 && *pos2 != '=')
3372 pos2++;
3373 if (*pos2 == '=')
3374 pos2++;
3375 id = pos2;
3376 while (*pos2 && *pos2 != ']')
3377 pos2++;
3378 *pos2 = '\0';
3379 os_setenv("WPA_ID_STR", id, 1);
3380 os_free(copy);
3381 }
3382
3383 os_setenv("WPA_CTRL_DIR", ctrl_iface_dir, 1);
3384
Dmitry Shmidt391c59f2013-09-03 12:16:28 -07003385 if (wpa_cli_connected <= 0 || new_id != wpa_cli_last_id) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003386 wpa_cli_connected = 1;
3387 wpa_cli_last_id = new_id;
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003388 wpa_cli_exec(action_file, ifname, "CONNECTED");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003389 }
3390 } else if (str_match(pos, WPA_EVENT_DISCONNECTED)) {
3391 if (wpa_cli_connected) {
3392 wpa_cli_connected = 0;
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003393 wpa_cli_exec(action_file, ifname, "DISCONNECTED");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003394 }
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08003395 } else if (str_match(pos, MESH_GROUP_STARTED)) {
3396 wpa_cli_exec(action_file, ctrl_ifname, pos);
3397 } else if (str_match(pos, MESH_GROUP_REMOVED)) {
3398 wpa_cli_exec(action_file, ctrl_ifname, pos);
3399 } else if (str_match(pos, MESH_PEER_CONNECTED)) {
3400 wpa_cli_exec(action_file, ctrl_ifname, pos);
3401 } else if (str_match(pos, MESH_PEER_DISCONNECTED)) {
3402 wpa_cli_exec(action_file, ctrl_ifname, pos);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003403 } else if (str_match(pos, P2P_EVENT_GROUP_STARTED)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003404 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003405 } else if (str_match(pos, P2P_EVENT_GROUP_REMOVED)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003406 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003407 } else if (str_match(pos, P2P_EVENT_CROSS_CONNECT_ENABLE)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003408 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003409 } else if (str_match(pos, P2P_EVENT_CROSS_CONNECT_DISABLE)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003410 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003411 } else if (str_match(pos, P2P_EVENT_GO_NEG_FAILURE)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003412 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003413 } else if (str_match(pos, WPS_EVENT_SUCCESS)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003414 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003415 } else if (str_match(pos, WPS_EVENT_FAIL)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003416 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003417 } else if (str_match(pos, AP_STA_CONNECTED)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003418 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003419 } else if (str_match(pos, AP_STA_DISCONNECTED)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003420 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -07003421 } else if (str_match(pos, ESS_DISASSOC_IMMINENT)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003422 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidtf21452a2014-02-26 10:55:25 -08003423 } else if (str_match(pos, HS20_SUBSCRIPTION_REMEDIATION)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003424 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidtf21452a2014-02-26 10:55:25 -08003425 } else if (str_match(pos, HS20_DEAUTH_IMMINENT_NOTICE)) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003426 wpa_cli_exec(action_file, ifname, pos);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003427 } else if (str_match(pos, WPA_EVENT_TERMINATING)) {
3428 printf("wpa_supplicant is terminating - stop monitoring\n");
3429 wpa_cli_quit = 1;
3430 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003431}
3432
3433
3434#ifndef CONFIG_ANSI_C_EXTRA
3435static void wpa_cli_action_cb(char *msg, size_t len)
3436{
3437 wpa_cli_action_process(msg);
3438}
3439#endif /* CONFIG_ANSI_C_EXTRA */
3440
3441
3442static void wpa_cli_reconnect(void)
3443{
3444 wpa_cli_close_connection();
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003445 if (wpa_cli_open_connection(ctrl_ifname, 1) < 0)
3446 return;
3447
3448 if (interactive) {
3449 edit_clear_line();
3450 printf("\rConnection to wpa_supplicant re-established\n");
3451 edit_redraw();
3452 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003453}
3454
3455
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003456static void cli_event(const char *str)
3457{
3458 const char *start, *s;
3459
3460 start = os_strchr(str, '>');
3461 if (start == NULL)
3462 return;
3463
3464 start++;
3465
3466 if (str_starts(start, WPA_EVENT_BSS_ADDED)) {
3467 s = os_strchr(start, ' ');
3468 if (s == NULL)
3469 return;
3470 s = os_strchr(s + 1, ' ');
3471 if (s == NULL)
3472 return;
3473 cli_txt_list_add(&bsses, s + 1);
3474 return;
3475 }
3476
3477 if (str_starts(start, WPA_EVENT_BSS_REMOVED)) {
3478 s = os_strchr(start, ' ');
3479 if (s == NULL)
3480 return;
3481 s = os_strchr(s + 1, ' ');
3482 if (s == NULL)
3483 return;
3484 cli_txt_list_del_addr(&bsses, s + 1);
3485 return;
3486 }
3487
3488#ifdef CONFIG_P2P
3489 if (str_starts(start, P2P_EVENT_DEVICE_FOUND)) {
3490 s = os_strstr(start, " p2p_dev_addr=");
3491 if (s == NULL)
3492 return;
3493 cli_txt_list_add_addr(&p2p_peers, s + 14);
3494 return;
3495 }
3496
3497 if (str_starts(start, P2P_EVENT_DEVICE_LOST)) {
3498 s = os_strstr(start, " p2p_dev_addr=");
3499 if (s == NULL)
3500 return;
3501 cli_txt_list_del_addr(&p2p_peers, s + 14);
3502 return;
3503 }
3504
3505 if (str_starts(start, P2P_EVENT_GROUP_STARTED)) {
3506 s = os_strchr(start, ' ');
3507 if (s == NULL)
3508 return;
3509 cli_txt_list_add_word(&p2p_groups, s + 1);
3510 return;
3511 }
3512
3513 if (str_starts(start, P2P_EVENT_GROUP_REMOVED)) {
3514 s = os_strchr(start, ' ');
3515 if (s == NULL)
3516 return;
3517 cli_txt_list_del_word(&p2p_groups, s + 1);
3518 return;
3519 }
3520#endif /* CONFIG_P2P */
3521}
3522
3523
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003524static int check_terminating(const char *msg)
3525{
3526 const char *pos = msg;
3527
3528 if (*pos == '<') {
3529 /* skip priority */
3530 pos = os_strchr(pos, '>');
3531 if (pos)
3532 pos++;
3533 else
3534 pos = msg;
3535 }
3536
3537 if (str_match(pos, WPA_EVENT_TERMINATING) && ctrl_conn) {
3538 edit_clear_line();
3539 printf("\rConnection to wpa_supplicant lost - trying to "
3540 "reconnect\n");
3541 edit_redraw();
3542 wpa_cli_attached = 0;
3543 wpa_cli_close_connection();
3544 return 1;
3545 }
3546
3547 return 0;
3548}
3549
3550
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003551static void wpa_cli_recv_pending(struct wpa_ctrl *ctrl, int action_monitor)
3552{
3553 if (ctrl_conn == NULL) {
3554 wpa_cli_reconnect();
3555 return;
3556 }
3557 while (wpa_ctrl_pending(ctrl) > 0) {
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003558 char buf[4096];
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003559 size_t len = sizeof(buf) - 1;
3560 if (wpa_ctrl_recv(ctrl, buf, &len) == 0) {
3561 buf[len] = '\0';
3562 if (action_monitor)
3563 wpa_cli_action_process(buf);
3564 else {
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003565 cli_event(buf);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003566 if (wpa_cli_show_event(buf)) {
3567 edit_clear_line();
3568 printf("\r%s\n", buf);
3569 edit_redraw();
3570 }
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003571
3572 if (interactive && check_terminating(buf) > 0)
3573 return;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003574 }
3575 } else {
3576 printf("Could not read pending message.\n");
3577 break;
3578 }
3579 }
3580
3581 if (wpa_ctrl_pending(ctrl) < 0) {
3582 printf("Connection to wpa_supplicant lost - trying to "
3583 "reconnect\n");
3584 wpa_cli_reconnect();
3585 }
3586}
3587
3588#define max_args 10
3589
3590static int tokenize_cmd(char *cmd, char *argv[])
3591{
3592 char *pos;
3593 int argc = 0;
3594
3595 pos = cmd;
3596 for (;;) {
3597 while (*pos == ' ')
3598 pos++;
3599 if (*pos == '\0')
3600 break;
3601 argv[argc] = pos;
3602 argc++;
3603 if (argc == max_args)
3604 break;
3605 if (*pos == '"') {
3606 char *pos2 = os_strrchr(pos, '"');
3607 if (pos2)
3608 pos = pos2 + 1;
3609 }
3610 while (*pos != '\0' && *pos != ' ')
3611 pos++;
3612 if (*pos == ' ')
3613 *pos++ = '\0';
3614 }
3615
3616 return argc;
3617}
3618
3619
3620static void wpa_cli_ping(void *eloop_ctx, void *timeout_ctx)
3621{
Dmitry Shmidt661b4f72014-09-29 14:58:27 -07003622 if (ctrl_conn) {
3623 int res;
3624 char *prefix = ifname_prefix;
3625
3626 ifname_prefix = NULL;
3627 res = _wpa_ctrl_command(ctrl_conn, "PING", 0);
3628 ifname_prefix = prefix;
3629 if (res) {
3630 printf("Connection to wpa_supplicant lost - trying to "
3631 "reconnect\n");
3632 wpa_cli_close_connection();
3633 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003634 }
3635 if (!ctrl_conn)
3636 wpa_cli_reconnect();
3637 eloop_register_timeout(ping_interval, 0, wpa_cli_ping, NULL, NULL);
3638}
3639
3640
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003641static void wpa_cli_mon_receive(int sock, void *eloop_ctx, void *sock_ctx)
3642{
3643 wpa_cli_recv_pending(mon_conn, 0);
3644}
3645
3646
3647static void wpa_cli_edit_cmd_cb(void *ctx, char *cmd)
3648{
3649 char *argv[max_args];
3650 int argc;
3651 argc = tokenize_cmd(cmd, argv);
3652 if (argc)
3653 wpa_request(ctrl_conn, argc, argv);
3654}
3655
3656
3657static void wpa_cli_edit_eof_cb(void *ctx)
3658{
3659 eloop_terminate();
3660}
3661
3662
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003663static int warning_displayed = 0;
3664static char *hfile = NULL;
3665static int edit_started = 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003666
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003667static void start_edit(void)
3668{
3669 char *home;
3670 char *ps = NULL;
3671
3672#ifdef CONFIG_CTRL_IFACE_UDP_REMOTE
3673 ps = wpa_ctrl_get_remote_ifname(ctrl_conn);
3674#endif /* CONFIG_CTRL_IFACE_UDP_REMOTE */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003675
3676 home = getenv("HOME");
3677 if (home) {
3678 const char *fname = ".wpa_cli_history";
3679 int hfile_len = os_strlen(home) + 1 + os_strlen(fname) + 1;
3680 hfile = os_malloc(hfile_len);
3681 if (hfile)
3682 os_snprintf(hfile, hfile_len, "%s/%s", home, fname);
3683 }
3684
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003685 if (edit_init(wpa_cli_edit_cmd_cb, wpa_cli_edit_eof_cb,
3686 wpa_cli_edit_completion_cb, NULL, hfile, ps) < 0) {
3687 eloop_terminate();
3688 return;
3689 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003690
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003691 edit_started = 1;
3692 eloop_register_timeout(ping_interval, 0, wpa_cli_ping, NULL, NULL);
3693}
3694
3695
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -07003696static void update_bssid_list(struct wpa_ctrl *ctrl)
3697{
3698 char buf[4096];
3699 size_t len = sizeof(buf);
3700 int ret;
3701 char *cmd = "BSS RANGE=ALL MASK=0x2";
3702 char *pos, *end;
3703
3704 if (ctrl == NULL)
3705 return;
3706 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len, NULL);
3707 if (ret < 0)
3708 return;
3709 buf[len] = '\0';
3710
3711 pos = buf;
3712 while (pos) {
3713 pos = os_strstr(pos, "bssid=");
3714 if (pos == NULL)
3715 break;
3716 pos += 6;
3717 end = os_strchr(pos, '\n');
3718 if (end == NULL)
3719 break;
3720 *end = '\0';
3721 cli_txt_list_add(&bsses, pos);
3722 pos = end + 1;
3723 }
3724}
3725
3726
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -07003727static void update_ifnames(struct wpa_ctrl *ctrl)
3728{
3729 char buf[4096];
3730 size_t len = sizeof(buf);
3731 int ret;
3732 char *cmd = "INTERFACES";
3733 char *pos, *end;
3734 char txt[200];
3735
3736 cli_txt_list_flush(&ifnames);
3737
3738 if (ctrl == NULL)
3739 return;
3740 ret = wpa_ctrl_request(ctrl, cmd, os_strlen(cmd), buf, &len, NULL);
3741 if (ret < 0)
3742 return;
3743 buf[len] = '\0';
3744
3745 pos = buf;
3746 while (pos) {
3747 end = os_strchr(pos, '\n');
3748 if (end == NULL)
3749 break;
3750 *end = '\0';
3751 ret = os_snprintf(txt, sizeof(txt), "ifname=%s", pos);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08003752 if (!os_snprintf_error(sizeof(txt), ret))
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -07003753 cli_txt_list_add(&ifnames, txt);
3754 pos = end + 1;
3755 }
3756}
3757
3758
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003759static void try_connection(void *eloop_ctx, void *timeout_ctx)
3760{
Dmitry Shmidtb6e9aaf2013-05-20 14:49:44 -07003761 if (ctrl_conn)
3762 goto done;
3763
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003764 if (ctrl_ifname == NULL)
3765 ctrl_ifname = wpa_cli_get_default_ifname();
3766
3767 if (!wpa_cli_open_connection(ctrl_ifname, 1) == 0) {
3768 if (!warning_displayed) {
3769 printf("Could not connect to wpa_supplicant: "
Dmitry Shmidt2f74e362015-01-21 13:19:05 -08003770 "%s - re-trying\n",
3771 ctrl_ifname ? ctrl_ifname : "(nil)");
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003772 warning_displayed = 1;
3773 }
3774 eloop_register_timeout(1, 0, try_connection, NULL, NULL);
3775 return;
3776 }
3777
Dmitry Shmidt51b6ea82013-05-08 10:42:09 -07003778 update_bssid_list(ctrl_conn);
3779
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003780 if (warning_displayed)
3781 printf("Connection established.\n");
3782
Dmitry Shmidtb6e9aaf2013-05-20 14:49:44 -07003783done:
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003784 start_edit();
3785}
3786
3787
3788static void wpa_cli_interactive(void)
3789{
3790 printf("\nInteractive mode\n\n");
3791
3792 eloop_register_timeout(0, 0, try_connection, NULL, NULL);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003793 eloop_run();
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003794 eloop_cancel_timeout(try_connection, NULL, NULL);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003795
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003796 cli_txt_list_flush(&p2p_peers);
3797 cli_txt_list_flush(&p2p_groups);
3798 cli_txt_list_flush(&bsses);
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -07003799 cli_txt_list_flush(&ifnames);
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003800 if (edit_started)
3801 edit_deinit(hfile, wpa_cli_edit_filter_history_cb);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003802 os_free(hfile);
3803 eloop_cancel_timeout(wpa_cli_ping, NULL, NULL);
3804 wpa_cli_close_connection();
3805}
3806
3807
3808static void wpa_cli_action(struct wpa_ctrl *ctrl)
3809{
3810#ifdef CONFIG_ANSI_C_EXTRA
3811 /* TODO: ANSI C version(?) */
3812 printf("Action processing not supported in ANSI C build.\n");
3813#else /* CONFIG_ANSI_C_EXTRA */
3814 fd_set rfds;
3815 int fd, res;
3816 struct timeval tv;
3817 char buf[256]; /* note: large enough to fit in unsolicited messages */
3818 size_t len;
3819
3820 fd = wpa_ctrl_get_fd(ctrl);
3821
3822 while (!wpa_cli_quit) {
3823 FD_ZERO(&rfds);
3824 FD_SET(fd, &rfds);
3825 tv.tv_sec = ping_interval;
3826 tv.tv_usec = 0;
3827 res = select(fd + 1, &rfds, NULL, NULL, &tv);
3828 if (res < 0 && errno != EINTR) {
3829 perror("select");
3830 break;
3831 }
3832
3833 if (FD_ISSET(fd, &rfds))
3834 wpa_cli_recv_pending(ctrl, 1);
3835 else {
3836 /* verify that connection is still working */
3837 len = sizeof(buf) - 1;
3838 if (wpa_ctrl_request(ctrl, "PING", 4, buf, &len,
3839 wpa_cli_action_cb) < 0 ||
3840 len < 4 || os_memcmp(buf, "PONG", 4) != 0) {
3841 printf("wpa_supplicant did not reply to PING "
3842 "command - exiting\n");
3843 break;
3844 }
3845 }
3846 }
3847#endif /* CONFIG_ANSI_C_EXTRA */
3848}
3849
3850
3851static void wpa_cli_cleanup(void)
3852{
3853 wpa_cli_close_connection();
3854 if (pid_file)
3855 os_daemonize_terminate(pid_file);
3856
3857 os_program_deinit();
3858}
3859
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003860
3861static void wpa_cli_terminate(int sig, void *ctx)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003862{
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003863 eloop_terminate();
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003864}
3865
3866
3867static char * wpa_cli_get_default_ifname(void)
3868{
3869 char *ifname = NULL;
3870
3871#ifdef CONFIG_CTRL_IFACE_UNIX
3872 struct dirent *dent;
3873 DIR *dir = opendir(ctrl_iface_dir);
3874 if (!dir) {
3875#ifdef ANDROID
3876 char ifprop[PROPERTY_VALUE_MAX];
3877 if (property_get("wifi.interface", ifprop, NULL) != 0) {
3878 ifname = os_strdup(ifprop);
3879 printf("Using interface '%s'\n", ifname);
3880 return ifname;
3881 }
3882#endif /* ANDROID */
3883 return NULL;
3884 }
3885 while ((dent = readdir(dir))) {
3886#ifdef _DIRENT_HAVE_D_TYPE
3887 /*
3888 * Skip the file if it is not a socket. Also accept
3889 * DT_UNKNOWN (0) in case the C library or underlying
3890 * file system does not support d_type.
3891 */
3892 if (dent->d_type != DT_SOCK && dent->d_type != DT_UNKNOWN)
3893 continue;
3894#endif /* _DIRENT_HAVE_D_TYPE */
3895 if (os_strcmp(dent->d_name, ".") == 0 ||
3896 os_strcmp(dent->d_name, "..") == 0)
3897 continue;
3898 printf("Selected interface '%s'\n", dent->d_name);
3899 ifname = os_strdup(dent->d_name);
3900 break;
3901 }
3902 closedir(dir);
3903#endif /* CONFIG_CTRL_IFACE_UNIX */
3904
3905#ifdef CONFIG_CTRL_IFACE_NAMED_PIPE
Dmitry Shmidtb6e9aaf2013-05-20 14:49:44 -07003906 char buf[4096], *pos;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003907 size_t len;
3908 struct wpa_ctrl *ctrl;
3909 int ret;
3910
3911 ctrl = wpa_ctrl_open(NULL);
3912 if (ctrl == NULL)
3913 return NULL;
3914
3915 len = sizeof(buf) - 1;
3916 ret = wpa_ctrl_request(ctrl, "INTERFACES", 10, buf, &len, NULL);
3917 if (ret >= 0) {
3918 buf[len] = '\0';
3919 pos = os_strchr(buf, '\n');
3920 if (pos)
3921 *pos = '\0';
3922 ifname = os_strdup(buf);
3923 }
3924 wpa_ctrl_close(ctrl);
3925#endif /* CONFIG_CTRL_IFACE_NAMED_PIPE */
3926
3927 return ifname;
3928}
3929
3930
3931int main(int argc, char *argv[])
3932{
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003933 int c;
3934 int daemonize = 0;
3935 int ret = 0;
3936 const char *global = NULL;
3937
3938 if (os_program_init())
3939 return -1;
3940
3941 for (;;) {
3942 c = getopt(argc, argv, "a:Bg:G:hi:p:P:v");
3943 if (c < 0)
3944 break;
3945 switch (c) {
3946 case 'a':
3947 action_file = optarg;
3948 break;
3949 case 'B':
3950 daemonize = 1;
3951 break;
3952 case 'g':
3953 global = optarg;
3954 break;
3955 case 'G':
3956 ping_interval = atoi(optarg);
3957 break;
3958 case 'h':
3959 usage();
3960 return 0;
3961 case 'v':
3962 printf("%s\n", wpa_cli_version);
3963 return 0;
3964 case 'i':
3965 os_free(ctrl_ifname);
3966 ctrl_ifname = os_strdup(optarg);
3967 break;
3968 case 'p':
3969 ctrl_iface_dir = optarg;
3970 break;
3971 case 'P':
3972 pid_file = optarg;
3973 break;
3974 default:
3975 usage();
3976 return -1;
3977 }
3978 }
3979
3980 interactive = (argc == optind) && (action_file == NULL);
3981
3982 if (interactive)
3983 printf("%s\n\n%s\n\n", wpa_cli_version, wpa_cli_license);
3984
3985 if (eloop_init())
3986 return -1;
3987
3988 if (global) {
3989#ifdef CONFIG_CTRL_IFACE_NAMED_PIPE
3990 ctrl_conn = wpa_ctrl_open(NULL);
3991#else /* CONFIG_CTRL_IFACE_NAMED_PIPE */
3992 ctrl_conn = wpa_ctrl_open(global);
3993#endif /* CONFIG_CTRL_IFACE_NAMED_PIPE */
3994 if (ctrl_conn == NULL) {
Dmitry Shmidt04949592012-07-19 12:16:46 -07003995 fprintf(stderr, "Failed to connect to wpa_supplicant "
3996 "global interface: %s error: %s\n",
3997 global, strerror(errno));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003998 return -1;
3999 }
Dmitry Shmidtb6e9aaf2013-05-20 14:49:44 -07004000
4001 if (interactive) {
Dmitry Shmidtf7e0a992013-05-23 11:03:10 -07004002 update_ifnames(ctrl_conn);
Dmitry Shmidtb6e9aaf2013-05-20 14:49:44 -07004003 mon_conn = wpa_ctrl_open(global);
4004 if (mon_conn) {
4005 if (wpa_ctrl_attach(mon_conn) == 0) {
4006 wpa_cli_attached = 1;
4007 eloop_register_read_sock(
4008 wpa_ctrl_get_fd(mon_conn),
4009 wpa_cli_mon_receive,
4010 NULL, NULL);
4011 } else {
4012 printf("Failed to open monitor "
4013 "connection through global "
4014 "control interface\n");
4015 }
4016 }
4017 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004018 }
4019
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07004020 eloop_register_signal_terminate(wpa_cli_terminate, NULL);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004021
4022 if (ctrl_ifname == NULL)
4023 ctrl_ifname = wpa_cli_get_default_ifname();
4024
4025 if (interactive) {
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07004026 wpa_cli_interactive();
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004027 } else {
4028 if (!global &&
4029 wpa_cli_open_connection(ctrl_ifname, 0) < 0) {
Dmitry Shmidt04949592012-07-19 12:16:46 -07004030 fprintf(stderr, "Failed to connect to non-global "
4031 "ctrl_ifname: %s error: %s\n",
Dmitry Shmidt2f74e362015-01-21 13:19:05 -08004032 ctrl_ifname ? ctrl_ifname : "(nil)",
4033 strerror(errno));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004034 return -1;
4035 }
4036
4037 if (action_file) {
4038 if (wpa_ctrl_attach(ctrl_conn) == 0) {
4039 wpa_cli_attached = 1;
4040 } else {
4041 printf("Warning: Failed to attach to "
4042 "wpa_supplicant.\n");
4043 return -1;
4044 }
4045 }
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07004046
4047 if (daemonize && os_daemonize(pid_file))
4048 return -1;
4049
4050 if (action_file)
4051 wpa_cli_action(ctrl_conn);
4052 else
4053 ret = wpa_request(ctrl_conn, argc - optind,
4054 &argv[optind]);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004055 }
4056
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004057 os_free(ctrl_ifname);
4058 eloop_destroy();
4059 wpa_cli_cleanup();
4060
4061 return ret;
4062}
4063
4064#else /* CONFIG_CTRL_IFACE */
4065int main(int argc, char *argv[])
4066{
4067 printf("CONFIG_CTRL_IFACE not defined - wpa_cli disabled\n");
4068 return -1;
4069}
4070#endif /* CONFIG_CTRL_IFACE */