Merge branch 'moonshot' of ssh://moonshot.suchdamage.org:822/srv/git/libeap into...
[libeap.git] / wpa_supplicant / wpa_cli.c
index bae0d6f..00fb0dc 100644 (file)
@@ -1,6 +1,6 @@
 /*
  * WPA Supplicant - command line interface for wpa_supplicant daemon
- * Copyright (c) 2004-2009, Jouni Malinen <j@w1.fi>
+ * Copyright (c) 2004-2010, Jouni Malinen <j@w1.fi>
  *
  * This program is free software; you can redistribute it and/or modify
  * it under the terms of the GNU General Public License version 2 as
 #include <readline/readline.h>
 #include <readline/history.h>
 #endif /* CONFIG_READLINE */
+#ifdef CONFIG_WPA_CLI_FORK
+#include <sys/wait.h>
+#endif /* CONFIG_WPA_CLI_FORK */
 
-#include "wpa_ctrl.h"
+#include "common/wpa_ctrl.h"
 #include "common.h"
-#include "version.h"
+#include "common/version.h"
 
 
 static const char *wpa_cli_version =
 "wpa_cli v" VERSION_STR "\n"
-"Copyright (c) 2004-2009, Jouni Malinen <j@w1.fi> and contributors";
+"Copyright (c) 2004-2010, Jouni Malinen <j@w1.fi> and contributors";
 
 
 static const char *wpa_cli_license =
@@ -87,6 +90,10 @@ static const char *wpa_cli_full_license =
 "\n";
 
 static struct wpa_ctrl *ctrl_conn;
+static struct wpa_ctrl *mon_conn;
+#ifdef CONFIG_WPA_CLI_FORK
+static pid_t mon_pid = 0;
+#endif /* CONFIG_WPA_CLI_FORK */
 static int wpa_cli_quit = 0;
 static int wpa_cli_attached = 0;
 static int wpa_cli_connected = 0;
@@ -96,6 +103,7 @@ static char *ctrl_ifname = NULL;
 static const char *pid_file = NULL;
 static const char *action_file = NULL;
 static int ping_interval = 5;
+static int interactive = 0;
 
 
 static void print_help();
@@ -119,32 +127,167 @@ static void usage(void)
 }
 
 
-static struct wpa_ctrl * wpa_cli_open_connection(const char *ifname)
+static void readline_redraw()
+{
+#ifdef CONFIG_READLINE
+       rl_on_new_line();
+       rl_redisplay();
+#endif /* CONFIG_READLINE */
+}
+
+
+static int str_starts(const char *src, const char *match)
+{
+       return os_strncmp(src, match, os_strlen(match)) == 0;
+}
+
+
+static int wpa_cli_show_event(const char *event)
+{
+       const char *start;
+
+       start = os_strchr(event, '>');
+       if (start == NULL)
+               return 1;
+
+       start++;
+       /*
+        * Skip BSS added/removed events since they can be relatively frequent
+        * and are likely of not much use for an interactive user.
+        */
+       if (str_starts(start, WPA_EVENT_BSS_ADDED) ||
+           str_starts(start, WPA_EVENT_BSS_REMOVED))
+               return 0;
+
+       return 1;
+}
+
+
+#ifdef CONFIG_WPA_CLI_FORK
+static int in_query = 0;
+
+static void wpa_cli_monitor_sig(int sig)
+{
+       if (sig == SIGUSR1)
+               in_query = 1;
+       else if (sig == SIGUSR2)
+               in_query = 0;
+}
+
+
+static void wpa_cli_monitor(void)
+{
+       char buf[256];
+       size_t len = sizeof(buf) - 1;
+       struct timeval tv;
+       fd_set rfds;
+       int ppid;
+
+       signal(SIGUSR1, wpa_cli_monitor_sig);
+       signal(SIGUSR2, wpa_cli_monitor_sig);
+
+       ppid = getppid();
+       while (mon_conn) {
+               int s = wpa_ctrl_get_fd(mon_conn);
+               tv.tv_sec = 5;
+               tv.tv_usec = 0;
+               FD_ZERO(&rfds);
+               FD_SET(s, &rfds);
+               if (select(s + 1, &rfds, NULL, NULL, &tv) < 0) {
+                       if (errno == EINTR)
+                               continue;
+                       perror("select");
+                       break;
+               }
+               if (mon_conn == NULL)
+                       break;
+               if (FD_ISSET(s, &rfds)) {
+                       len = sizeof(buf) - 1;
+                       int res = wpa_ctrl_recv(mon_conn, buf, &len);
+                       if (res < 0) {
+                               perror("wpa_ctrl_recv");
+                               break;
+                       }
+                       buf[len] = '\0';
+                       if (wpa_cli_show_event(buf)) {
+                               if (in_query)
+                                       printf("\r");
+                               printf("%s\n", buf);
+                               kill(ppid, SIGUSR1);
+                       }
+               }
+       }
+}
+#endif /* CONFIG_WPA_CLI_FORK */
+
+
+static int wpa_cli_open_connection(const char *ifname, int attach)
 {
 #if defined(CONFIG_CTRL_IFACE_UDP) || defined(CONFIG_CTRL_IFACE_NAMED_PIPE)
        ctrl_conn = wpa_ctrl_open(ifname);
-       return ctrl_conn;
+       if (ctrl_conn == NULL)
+               return -1;
+
+       if (attach && interactive)
+               mon_conn = wpa_ctrl_open(ifname);
+       else
+               mon_conn = NULL;
 #else /* CONFIG_CTRL_IFACE_UDP || CONFIG_CTRL_IFACE_NAMED_PIPE */
        char *cfile;
        int flen, res;
 
        if (ifname == NULL)
-               return NULL;
+               return -1;
 
        flen = os_strlen(ctrl_iface_dir) + os_strlen(ifname) + 2;
        cfile = os_malloc(flen);
        if (cfile == NULL)
-               return NULL;
+               return -1L;
        res = os_snprintf(cfile, flen, "%s/%s", ctrl_iface_dir, ifname);
        if (res < 0 || res >= flen) {
                os_free(cfile);
-               return NULL;
+               return -1;
        }
 
        ctrl_conn = wpa_ctrl_open(cfile);
+       if (ctrl_conn == NULL) {
+               os_free(cfile);
+               return -1;
+       }
+
+       if (attach && interactive)
+               mon_conn = wpa_ctrl_open(cfile);
+       else
+               mon_conn = NULL;
        os_free(cfile);
-       return ctrl_conn;
 #endif /* CONFIG_CTRL_IFACE_UDP || CONFIG_CTRL_IFACE_NAMED_PIPE */
+
+       if (mon_conn) {
+               if (wpa_ctrl_attach(mon_conn) == 0) {
+                       wpa_cli_attached = 1;
+               } else {
+                       printf("Warning: Failed to attach to "
+                              "wpa_supplicant.\n");
+                       return -1;
+               }
+
+#ifdef CONFIG_WPA_CLI_FORK
+               {
+                       pid_t p = fork();
+                       if (p < 0) {
+                               perror("fork");
+                               return -1;
+                       }
+                       if (p == 0) {
+                               wpa_cli_monitor();
+                               exit(0);
+                       } else
+                               mon_pid = p;
+               }
+#endif /* CONFIG_WPA_CLI_FORK */
+       }
+
+       return 0;
 }
 
 
@@ -153,12 +296,25 @@ static void wpa_cli_close_connection(void)
        if (ctrl_conn == NULL)
                return;
 
+#ifdef CONFIG_WPA_CLI_FORK
+       if (mon_pid) {
+               int status;
+               kill(mon_pid, SIGPIPE);
+               wait(&status);
+               mon_pid = 0;
+       }
+#endif /* CONFIG_WPA_CLI_FORK */
+
        if (wpa_cli_attached) {
-               wpa_ctrl_detach(ctrl_conn);
+               wpa_ctrl_detach(interactive ? mon_conn : ctrl_conn);
                wpa_cli_attached = 0;
        }
        wpa_ctrl_close(ctrl_conn);
        ctrl_conn = NULL;
+       if (mon_conn) {
+               wpa_ctrl_close(mon_conn);
+               mon_conn = NULL;
+       }
 }
 
 
@@ -215,6 +371,19 @@ static int wpa_cli_cmd_ping(struct wpa_ctrl *ctrl, int argc, char *argv[])
 }
 
 
+static int wpa_cli_cmd_note(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       char cmd[256];
+       int ret;
+       if (argc == 0)
+               return -1;
+       ret = os_snprintf(cmd, sizeof(cmd), "NOTE %s", argv[0]);
+       if (ret < 0 || (size_t) ret >= sizeof(cmd))
+               return -1;
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
 static int wpa_cli_cmd_mib(struct wpa_ctrl *ctrl, int argc, char *argv[])
 {
        return wpa_ctrl_command(ctrl, "MIB");
@@ -446,28 +615,70 @@ static int wpa_cli_cmd_wps_pin(struct wpa_ctrl *ctrl, int argc, char *argv[])
 }
 
 
+static int wpa_cli_cmd_wps_check_pin(struct wpa_ctrl *ctrl, int argc,
+                                    char *argv[])
+{
+       char cmd[256];
+       int res;
+
+       if (argc != 1 && argc != 2) {
+               printf("Invalid WPS_CHECK_PIN command: needs one argument:\n"
+                      "- PIN to be verified\n");
+               return -1;
+       }
+
+       if (argc == 2)
+               res = os_snprintf(cmd, sizeof(cmd), "WPS_CHECK_PIN %s %s",
+                                 argv[0], argv[1]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd), "WPS_CHECK_PIN %s",
+                                 argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd) - 1) {
+               printf("Too long WPS_CHECK_PIN command.\n");
+               return -1;
+       }
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_wps_cancel(struct wpa_ctrl *ctrl, int argc,
+                                 char *argv[])
+{
+       return wpa_ctrl_command(ctrl, "WPS_CANCEL");
+}
+
+
+#ifdef CONFIG_WPS_OOB
 static int wpa_cli_cmd_wps_oob(struct wpa_ctrl *ctrl, int argc, char *argv[])
 {
        char cmd[256];
        int res;
 
-       if (argc != 3) {
-               printf("Invalid WPS_OOB command: need three arguments:\n"
-                      "- OOB_DEV_TYPE: use 'ufd'\n"
-                      "- OOB_PATH: path of OOB device like '/mnt'\n"
-                      "- OOB_METHOD: OOB method 'pin-e' or 'pin-r', "
-                      "'cred'\n");
+       if (argc != 3 && argc != 4) {
+               printf("Invalid WPS_OOB command: need three or four "
+                      "arguments:\n"
+                      "- DEV_TYPE: use 'ufd' or 'nfc'\n"
+                      "- PATH: path of OOB device like '/mnt'\n"
+                      "- METHOD: OOB method 'pin-e' or 'pin-r', "
+                      "'cred'\n"
+                      "- DEV_NAME: (only for NFC) device name like "
+                      "'pn531'\n");
                return -1;
        }
 
-       res = os_snprintf(cmd, sizeof(cmd), "WPS_OOB %s %s %s",
-                         argv[0], argv[1], argv[2]);
+       if (argc == 3)
+               res = os_snprintf(cmd, sizeof(cmd), "WPS_OOB %s %s %s",
+                                 argv[0], argv[1], argv[2]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd), "WPS_OOB %s %s %s %s",
+                                 argv[0], argv[1], argv[2], argv[3]);
        if (res < 0 || (size_t) res >= sizeof(cmd) - 1) {
                printf("Too long WPS_OOB command.\n");
                return -1;
        }
        return wpa_ctrl_command(ctrl, cmd);
 }
+#endif /* CONFIG_WPS_OOB */
 
 
 static int wpa_cli_cmd_wps_reg(struct wpa_ctrl *ctrl, int argc, char *argv[])
@@ -475,14 +686,50 @@ static int wpa_cli_cmd_wps_reg(struct wpa_ctrl *ctrl, int argc, char *argv[])
        char cmd[256];
        int res;
 
-       if (argc != 2) {
+       if (argc == 2)
+               res = os_snprintf(cmd, sizeof(cmd), "WPS_REG %s %s",
+                                 argv[0], argv[1]);
+       else if (argc == 5 || argc == 6) {
+               char ssid_hex[2 * 32 + 1];
+               char key_hex[2 * 64 + 1];
+               int i;
+
+               ssid_hex[0] = '\0';
+               for (i = 0; i < 32; i++) {
+                       if (argv[2][i] == '\0')
+                               break;
+                       os_snprintf(&ssid_hex[i * 2], 3, "%02x", argv[2][i]);
+               }
+
+               key_hex[0] = '\0';
+               if (argc == 6) {
+                       for (i = 0; i < 64; i++) {
+                               if (argv[5][i] == '\0')
+                                       break;
+                               os_snprintf(&key_hex[i * 2], 3, "%02x",
+                                           argv[5][i]);
+                       }
+               }
+
+               res = os_snprintf(cmd, sizeof(cmd),
+                                 "WPS_REG %s %s %s %s %s %s",
+                                 argv[0], argv[1], ssid_hex, argv[3], argv[4],
+                                 key_hex);
+       } else {
                printf("Invalid WPS_REG command: need two arguments:\n"
                       "- BSSID: use 'any' to select any\n"
                       "- AP PIN\n");
+               printf("Alternatively, six arguments can be used to "
+                      "reconfigure the AP:\n"
+                      "- BSSID: use 'any' to select any\n"
+                      "- AP PIN\n"
+                      "- new SSID\n"
+                      "- new auth (OPEN, WPAPSK, WPA2PSK)\n"
+                      "- new encr (NONE, WEP, TKIP, CCMP)\n"
+                      "- new key\n");
                return -1;
        }
 
-       res = os_snprintf(cmd, sizeof(cmd), "WPS_REG %s %s", argv[0], argv[1]);
        if (res < 0 || (size_t) res >= sizeof(cmd) - 1) {
                printf("Too long WPS_REG command.\n");
                return -1;
@@ -491,6 +738,151 @@ static int wpa_cli_cmd_wps_reg(struct wpa_ctrl *ctrl, int argc, char *argv[])
 }
 
 
+static int wpa_cli_cmd_wps_er_start(struct wpa_ctrl *ctrl, int argc,
+                                   char *argv[])
+{
+       char cmd[100];
+       if (argc > 0) {
+               os_snprintf(cmd, sizeof(cmd), "WPS_ER_START %s", argv[0]);
+               return wpa_ctrl_command(ctrl, cmd);
+       }
+       return wpa_ctrl_command(ctrl, "WPS_ER_START");
+}
+
+
+static int wpa_cli_cmd_wps_er_stop(struct wpa_ctrl *ctrl, int argc,
+                                  char *argv[])
+{
+       return wpa_ctrl_command(ctrl, "WPS_ER_STOP");
+
+}
+
+
+static int wpa_cli_cmd_wps_er_pin(struct wpa_ctrl *ctrl, int argc,
+                                 char *argv[])
+{
+       char cmd[256];
+       int res;
+
+       if (argc < 2) {
+               printf("Invalid WPS_ER_PIN command: need at least two "
+                      "arguments:\n"
+                      "- UUID: use 'any' to select any\n"
+                      "- PIN: Enrollee PIN\n"
+                      "optional: - Enrollee MAC address\n");
+               return -1;
+       }
+
+       if (argc > 2)
+               res = os_snprintf(cmd, sizeof(cmd), "WPS_ER_PIN %s %s %s",
+                                 argv[0], argv[1], argv[2]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd), "WPS_ER_PIN %s %s",
+                                 argv[0], argv[1]);
+       if (res < 0 || (size_t) res >= sizeof(cmd) - 1) {
+               printf("Too long WPS_ER_PIN command.\n");
+               return -1;
+       }
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_wps_er_pbc(struct wpa_ctrl *ctrl, int argc,
+                                 char *argv[])
+{
+       char cmd[256];
+       int res;
+
+       if (argc != 1) {
+               printf("Invalid WPS_ER_PBC command: need one argument:\n"
+                      "- UUID: Specify the Enrollee\n");
+               return -1;
+       }
+
+       res = os_snprintf(cmd, sizeof(cmd), "WPS_ER_PBC %s",
+                         argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd) - 1) {
+               printf("Too long WPS_ER_PBC command.\n");
+               return -1;
+       }
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_wps_er_learn(struct wpa_ctrl *ctrl, int argc,
+                                   char *argv[])
+{
+       char cmd[256];
+       int res;
+
+       if (argc != 2) {
+               printf("Invalid WPS_ER_LEARN command: need two arguments:\n"
+                      "- UUID: specify which AP to use\n"
+                      "- PIN: AP PIN\n");
+               return -1;
+       }
+
+       res = os_snprintf(cmd, sizeof(cmd), "WPS_ER_LEARN %s %s",
+                         argv[0], argv[1]);
+       if (res < 0 || (size_t) res >= sizeof(cmd) - 1) {
+               printf("Too long WPS_ER_LEARN command.\n");
+               return -1;
+       }
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_wps_er_config(struct wpa_ctrl *ctrl, int argc,
+                                    char *argv[])
+{
+       char cmd[256];
+       int res;
+
+       if (argc == 5 || argc == 6) {
+               char ssid_hex[2 * 32 + 1];
+               char key_hex[2 * 64 + 1];
+               int i;
+
+               ssid_hex[0] = '\0';
+               for (i = 0; i < 32; i++) {
+                       if (argv[2][i] == '\0')
+                               break;
+                       os_snprintf(&ssid_hex[i * 2], 3, "%02x", argv[2][i]);
+               }
+
+               key_hex[0] = '\0';
+               if (argc == 6) {
+                       for (i = 0; i < 64; i++) {
+                               if (argv[5][i] == '\0')
+                                       break;
+                               os_snprintf(&key_hex[i * 2], 3, "%02x",
+                                           argv[5][i]);
+                       }
+               }
+
+               res = os_snprintf(cmd, sizeof(cmd),
+                                 "WPS_ER_CONFIG %s %s %s %s %s %s",
+                                 argv[0], argv[1], ssid_hex, argv[3], argv[4],
+                                 key_hex);
+       } else {
+               printf("Invalid WPS_ER_CONFIG command: need six arguments:\n"
+                      "- AP UUID\n"
+                      "- AP PIN\n"
+                      "- new SSID\n"
+                      "- new auth (OPEN, WPAPSK, WPA2PSK)\n"
+                      "- new encr (NONE, WEP, TKIP, CCMP)\n"
+                      "- new key\n");
+               return -1;
+       }
+
+       if (res < 0 || (size_t) res >= sizeof(cmd) - 1) {
+               printf("Too long WPS_ER_CONFIG command.\n");
+               return -1;
+       }
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
 static int wpa_cli_cmd_ibss_rsn(struct wpa_ctrl *ctrl, int argc, char *argv[])
 {
        char cmd[256];
@@ -1034,14 +1426,8 @@ static int wpa_cli_cmd_interface(struct wpa_ctrl *ctrl, int argc, char *argv[])
        os_free(ctrl_ifname);
        ctrl_ifname = os_strdup(argv[0]);
 
-       if (wpa_cli_open_connection(ctrl_ifname)) {
+       if (wpa_cli_open_connection(ctrl_ifname, 1)) {
                printf("Connected to interface '%s.\n", ctrl_ifname);
-               if (wpa_ctrl_attach(ctrl_conn) == 0) {
-                       wpa_cli_attached = 1;
-               } else {
-                       printf("Warning: Failed to attach to "
-                              "wpa_supplicant.\n");
-               }
        } else {
                printf("Could not connect to interface '%s' - re-trying\n",
                       ctrl_ifname);
@@ -1122,45 +1508,674 @@ static int wpa_cli_cmd_interface_list(struct wpa_ctrl *ctrl, int argc,
 }
 
 
-enum wpa_cli_cmd_flags {
-       cli_cmd_flag_none               = 0x00,
-       cli_cmd_flag_sensitive          = 0x01
-};
+#ifdef CONFIG_AP
+static int wpa_cli_cmd_sta(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       char buf[64];
+       if (argc != 1) {
+               printf("Invalid 'sta' command - exactly one argument, STA "
+                      "address, is required.\n");
+               return -1;
+       }
+       os_snprintf(buf, sizeof(buf), "STA %s", argv[0]);
+       return wpa_ctrl_command(ctrl, buf);
+}
 
-struct wpa_cli_cmd {
-       const char *cmd;
-       int (*handler)(struct wpa_ctrl *ctrl, int argc, char *argv[]);
-       enum wpa_cli_cmd_flags flags;
-       const char *usage;
-};
 
-static struct wpa_cli_cmd wpa_cli_commands[] = {
-       { "status", wpa_cli_cmd_status,
-         cli_cmd_flag_none,
-         "[verbose] = get current WPA/EAPOL/EAP status" },
-       { "ping", wpa_cli_cmd_ping,
-         cli_cmd_flag_none,
-         "= pings wpa_supplicant" },
-       { "mib", wpa_cli_cmd_mib,
-         cli_cmd_flag_none,
-         "= get MIB variables (dot1x, dot11)" },
-       { "help", wpa_cli_cmd_help,
-         cli_cmd_flag_none,
-         "= show this usage help" },
-       { "interface", wpa_cli_cmd_interface,
-         cli_cmd_flag_none,
-         "[ifname] = show interfaces/select interface" },
-       { "level", wpa_cli_cmd_level,
-         cli_cmd_flag_none,
-         "<debug level> = change debug level" },
-       { "license", wpa_cli_cmd_license,
-         cli_cmd_flag_none,
-         "= show full wpa_cli license" },
-       { "quit", wpa_cli_cmd_quit,
-         cli_cmd_flag_none,
-         "= exit wpa_cli" },
-       { "set", wpa_cli_cmd_set,
-         cli_cmd_flag_none,
+static int wpa_ctrl_command_sta(struct wpa_ctrl *ctrl, char *cmd,
+                               char *addr, size_t addr_len)
+{
+       char buf[4096], *pos;
+       size_t len;
+       int ret;
+
+       if (ctrl_conn == NULL) {
+               printf("Not connected to hostapd - command dropped.\n");
+               return -1;
+       }
+       len = sizeof(buf) - 1;
+       ret = wpa_ctrl_request(ctrl, cmd, strlen(cmd), buf, &len,
+                              wpa_cli_msg_cb);
+       if (ret == -2) {
+               printf("'%s' command timed out.\n", cmd);
+               return -2;
+       } else if (ret < 0) {
+               printf("'%s' command failed.\n", cmd);
+               return -1;
+       }
+
+       buf[len] = '\0';
+       if (memcmp(buf, "FAIL", 4) == 0)
+               return -1;
+       printf("%s", buf);
+
+       pos = buf;
+       while (*pos != '\0' && *pos != '\n')
+               pos++;
+       *pos = '\0';
+       os_strlcpy(addr, buf, addr_len);
+       return 0;
+}
+
+
+static int wpa_cli_cmd_all_sta(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       char addr[32], cmd[64];
+
+       if (wpa_ctrl_command_sta(ctrl, "STA-FIRST", addr, sizeof(addr)))
+               return 0;
+       do {
+               os_snprintf(cmd, sizeof(cmd), "STA-NEXT %s", addr);
+       } while (wpa_ctrl_command_sta(ctrl, cmd, addr, sizeof(addr)) == 0);
+
+       return -1;
+}
+#endif /* CONFIG_AP */
+
+
+static int wpa_cli_cmd_suspend(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       return wpa_ctrl_command(ctrl, "SUSPEND");
+}
+
+
+static int wpa_cli_cmd_resume(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       return wpa_ctrl_command(ctrl, "RESUME");
+}
+
+
+static int wpa_cli_cmd_drop_sa(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       return wpa_ctrl_command(ctrl, "DROP_SA");
+}
+
+
+static int wpa_cli_cmd_roam(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc != 1) {
+               printf("Invalid ROAM command: needs one argument "
+                      "(target AP's BSSID)\n");
+               return -1;
+       }
+
+       res = os_snprintf(cmd, sizeof(cmd), "ROAM %s", argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd) - 1) {
+               printf("Too long ROAM command.\n");
+               return -1;
+       }
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+#ifdef CONFIG_P2P
+
+static int wpa_cli_cmd_p2p_find(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc == 0)
+               return wpa_ctrl_command(ctrl, "P2P_FIND");
+
+       if (argc > 1)
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_FIND %s %s",
+                                 argv[0], argv[1]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_FIND %s", argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_stop_find(struct wpa_ctrl *ctrl, int argc,
+                                    char *argv[])
+{
+       return wpa_ctrl_command(ctrl, "P2P_STOP_FIND");
+}
+
+
+static int wpa_cli_cmd_p2p_connect(struct wpa_ctrl *ctrl, int argc,
+                                  char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc < 2) {
+               printf("Invalid P2P_CONNECT command: needs at least two "
+                      "arguments (address and pbc/PIN)\n");
+               return -1;
+       }
+
+       if (argc > 4)
+               res = os_snprintf(cmd, sizeof(cmd),
+                                 "P2P_CONNECT %s %s %s %s %s",
+                                 argv[0], argv[1], argv[2], argv[3],
+                                 argv[4]);
+       else if (argc > 3)
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_CONNECT %s %s %s %s",
+                                 argv[0], argv[1], argv[2], argv[3]);
+       else if (argc > 2)
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_CONNECT %s %s %s",
+                                 argv[0], argv[1], argv[2]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_CONNECT %s %s",
+                                 argv[0], argv[1]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_listen(struct wpa_ctrl *ctrl, int argc,
+                                 char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc == 0)
+               return wpa_ctrl_command(ctrl, "P2P_LISTEN");
+
+       res = os_snprintf(cmd, sizeof(cmd), "P2P_LISTEN %s", argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_group_remove(struct wpa_ctrl *ctrl, int argc,
+                                       char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc != 1) {
+               printf("Invalid P2P_GROUP_REMOVE command: needs one argument "
+                      "(interface name)\n");
+               return -1;
+       }
+
+       res = os_snprintf(cmd, sizeof(cmd), "P2P_GROUP_REMOVE %s", argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_group_add(struct wpa_ctrl *ctrl, int argc,
+                                       char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc == 0)
+               return wpa_ctrl_command(ctrl, "P2P_GROUP_ADD");
+
+       res = os_snprintf(cmd, sizeof(cmd), "P2P_GROUP_ADD %s", argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_prov_disc(struct wpa_ctrl *ctrl, int argc,
+                                    char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc != 2) {
+               printf("Invalid P2P_PROV_DISC command: needs two arguments "
+                      "(address and config method\n"
+                      "(display, keypad, or pbc)\n");
+               return -1;
+       }
+
+       res = os_snprintf(cmd, sizeof(cmd), "P2P_PROV_DISC %s %s",
+                         argv[0], argv[1]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_get_passphrase(struct wpa_ctrl *ctrl, int argc,
+                                         char *argv[])
+{
+       return wpa_ctrl_command(ctrl, "P2P_GET_PASSPHRASE");
+}
+
+
+static int wpa_cli_cmd_p2p_serv_disc_req(struct wpa_ctrl *ctrl, int argc,
+                                        char *argv[])
+{
+       char cmd[4096];
+       int res;
+
+       if (argc != 2 && argc != 4) {
+               printf("Invalid P2P_SERV_DISC_REQ command: needs two "
+                      "arguments (address and TLVs) or four arguments "
+                      "(address, \"upnp\", version, search target "
+                      "(SSDP ST:)\n");
+               return -1;
+       }
+
+       if (argc == 4)
+               res = os_snprintf(cmd, sizeof(cmd),
+                                 "P2P_SERV_DISC_REQ %s %s %s %s",
+                                 argv[0], argv[1], argv[2], argv[3]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_SERV_DISC_REQ %s %s",
+                                 argv[0], argv[1]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_serv_disc_cancel_req(struct wpa_ctrl *ctrl,
+                                               int argc, char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc != 1) {
+               printf("Invalid P2P_SERV_DISC_CANCEL_REQ command: needs one "
+                      "argument (pending request identifier)\n");
+               return -1;
+       }
+
+       res = os_snprintf(cmd, sizeof(cmd), "P2P_SERV_DISC_CANCEL_REQ %s",
+                         argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_serv_disc_resp(struct wpa_ctrl *ctrl, int argc,
+                                         char *argv[])
+{
+       char cmd[4096];
+       int res;
+
+       if (argc != 4) {
+               printf("Invalid P2P_SERV_DISC_RESP command: needs four "
+                      "arguments (freq, address, dialog token, and TLVs)\n");
+               return -1;
+       }
+
+       res = os_snprintf(cmd, sizeof(cmd), "P2P_SERV_DISC_RESP %s %s %s %s",
+                         argv[0], argv[1], argv[2], argv[3]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_service_update(struct wpa_ctrl *ctrl, int argc,
+                                         char *argv[])
+{
+       return wpa_ctrl_command(ctrl, "P2P_SERVICE_UPDATE");
+}
+
+
+static int wpa_cli_cmd_p2p_serv_disc_external(struct wpa_ctrl *ctrl,
+                                             int argc, char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc != 1) {
+               printf("Invalid P2P_SERV_DISC_EXTERNAL command: needs one "
+                      "argument (external processing: 0/1)\n");
+               return -1;
+       }
+
+       res = os_snprintf(cmd, sizeof(cmd), "P2P_SERV_DISC_EXTERNAL %s",
+                         argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_service_flush(struct wpa_ctrl *ctrl, int argc,
+                                        char *argv[])
+{
+       return wpa_ctrl_command(ctrl, "P2P_SERVICE_FLUSH");
+}
+
+
+static int wpa_cli_cmd_p2p_service_add(struct wpa_ctrl *ctrl, int argc,
+                                      char *argv[])
+{
+       char cmd[4096];
+       int res;
+
+       if (argc != 3 && argc != 4) {
+               printf("Invalid P2P_SERVICE_ADD command: needs three or four "
+                      "arguments\n");
+               return -1;
+       }
+
+       if (argc == 4)
+               res = os_snprintf(cmd, sizeof(cmd),
+                                 "P2P_SERVICE_ADD %s %s %s %s",
+                                 argv[0], argv[1], argv[2], argv[3]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd),
+                                 "P2P_SERVICE_ADD %s %s %s",
+                                 argv[0], argv[1], argv[2]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_service_del(struct wpa_ctrl *ctrl, int argc,
+                                      char *argv[])
+{
+       char cmd[4096];
+       int res;
+
+       if (argc != 2 && argc != 3) {
+               printf("Invalid P2P_SERVICE_DEL command: needs two or three "
+                      "arguments\n");
+               return -1;
+       }
+
+       if (argc == 3)
+               res = os_snprintf(cmd, sizeof(cmd),
+                                 "P2P_SERVICE_DEL %s %s %s",
+                                 argv[0], argv[1], argv[2]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd),
+                                 "P2P_SERVICE_DEL %s %s",
+                                 argv[0], argv[1]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_reject(struct wpa_ctrl *ctrl,
+                                 int argc, char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc != 1) {
+               printf("Invalid P2P_REJECT command: needs one argument "
+                      "(peer address)\n");
+               return -1;
+       }
+
+       res = os_snprintf(cmd, sizeof(cmd), "P2P_REJECT %s", argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_invite(struct wpa_ctrl *ctrl,
+                                 int argc, char *argv[])
+{
+       char cmd[128];
+       int res;
+
+       if (argc < 1) {
+               printf("Invalid P2P_INVITE command: needs at least one "
+                      "argument\n");
+               return -1;
+       }
+
+       if (argc > 2)
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_INVITE %s %s %s",
+                                 argv[0], argv[1], argv[2]);
+       else if (argc > 1)
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_INVITE %s %s",
+                                 argv[0], argv[1]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_INVITE %s", argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_peer(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       char buf[64];
+       if (argc != 1) {
+               printf("Invalid 'p2p_peer' command - exactly one argument, "
+                      "P2P peer device address, is required.\n");
+               return -1;
+       }
+       os_snprintf(buf, sizeof(buf), "P2P_PEER %s", argv[0]);
+       return wpa_ctrl_command(ctrl, buf);
+}
+
+
+static int wpa_ctrl_command_p2p_peer(struct wpa_ctrl *ctrl, char *cmd,
+                                    char *addr, size_t addr_len,
+                                    int discovered)
+{
+       char buf[4096], *pos;
+       size_t len;
+       int ret;
+
+       if (ctrl_conn == NULL)
+               return -1;
+       len = sizeof(buf) - 1;
+       ret = wpa_ctrl_request(ctrl, cmd, strlen(cmd), buf, &len,
+                              wpa_cli_msg_cb);
+       if (ret == -2) {
+               printf("'%s' command timed out.\n", cmd);
+               return -2;
+       } else if (ret < 0) {
+               printf("'%s' command failed.\n", cmd);
+               return -1;
+       }
+
+       buf[len] = '\0';
+       if (memcmp(buf, "FAIL", 4) == 0)
+               return -1;
+
+       pos = buf;
+       while (*pos != '\0' && *pos != '\n')
+               pos++;
+       *pos++ = '\0';
+       os_strlcpy(addr, buf, addr_len);
+       if (!discovered || os_strstr(pos, "[PROBE_REQ_ONLY]") == NULL)
+               printf("%s\n", addr);
+       return 0;
+}
+
+
+static int wpa_cli_cmd_p2p_peers(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       char addr[32], cmd[64];
+       int discovered;
+
+       discovered = argc > 0 && os_strcmp(argv[0], "discovered") == 0;
+
+       if (wpa_ctrl_command_p2p_peer(ctrl, "P2P_PEER FIRST",
+                                     addr, sizeof(addr), discovered))
+               return 0;
+       do {
+               os_snprintf(cmd, sizeof(cmd), "P2P_PEER NEXT-%s", addr);
+       } while (wpa_ctrl_command_p2p_peer(ctrl, cmd, addr, sizeof(addr),
+                        discovered) == 0);
+
+       return -1;
+}
+
+
+static int wpa_cli_cmd_p2p_set(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       char cmd[100];
+       int res;
+
+       if (argc != 2) {
+               printf("Invalid P2P_SET command: needs two arguments (field, "
+                      "value)\n");
+               return -1;
+       }
+
+       res = os_snprintf(cmd, sizeof(cmd), "P2P_SET %s %s", argv[0], argv[1]);
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_flush(struct wpa_ctrl *ctrl, int argc, char *argv[])
+{
+       return wpa_ctrl_command(ctrl, "P2P_FLUSH");
+}
+
+
+static int wpa_cli_cmd_p2p_presence_req(struct wpa_ctrl *ctrl, int argc,
+                                       char *argv[])
+{
+       char cmd[100];
+       int res;
+
+       if (argc != 0 && argc != 2 && argc != 4) {
+               printf("Invalid P2P_PRESENCE_REQ command: needs two arguments "
+                      "(preferred duration, interval; in microsecods).\n"
+                      "Optional second pair can be used to provide "
+                      "acceptable values.\n");
+               return -1;
+       }
+
+       if (argc == 4)
+               res = os_snprintf(cmd, sizeof(cmd),
+                                 "P2P_PRESENCE_REQ %s %s %s %s",
+                                 argv[0], argv[1], argv[2], argv[3]);
+       else if (argc == 2)
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_PRESENCE_REQ %s %s",
+                                 argv[0], argv[1]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_PRESENCE_REQ");
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+static int wpa_cli_cmd_p2p_ext_listen(struct wpa_ctrl *ctrl, int argc,
+                                     char *argv[])
+{
+       char cmd[100];
+       int res;
+
+       if (argc != 0 && argc != 2) {
+               printf("Invalid P2P_EXT_LISTEN command: needs two arguments "
+                      "(availability period, availability interval; in "
+                      "millisecods).\n"
+                      "Extended Listen Timing can be cancelled with this "
+                      "command when used without parameters.\n");
+               return -1;
+       }
+
+       if (argc == 2)
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_EXT_LISTEN %s %s",
+                                 argv[0], argv[1]);
+       else
+               res = os_snprintf(cmd, sizeof(cmd), "P2P_EXT_LISTEN");
+       if (res < 0 || (size_t) res >= sizeof(cmd))
+               return -1;
+       cmd[sizeof(cmd) - 1] = '\0';
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+#endif /* CONFIG_P2P */
+
+
+static int wpa_cli_cmd_sta_autoconnect(struct wpa_ctrl *ctrl, int argc,
+                                      char *argv[])
+{
+       char cmd[256];
+       int res;
+
+       if (argc != 1) {
+               printf("Invalid STA_AUTOCONNECT command: needs one argument "
+                      "(0/1 = disable/enable automatic reconnection)\n");
+               return -1;
+       }
+       res = os_snprintf(cmd, sizeof(cmd), "STA_AUTOCONNECT %s", argv[0]);
+       if (res < 0 || (size_t) res >= sizeof(cmd) - 1) {
+               printf("Too long STA_AUTOCONNECT command.\n");
+               return -1;
+       }
+       return wpa_ctrl_command(ctrl, cmd);
+}
+
+
+enum wpa_cli_cmd_flags {
+       cli_cmd_flag_none               = 0x00,
+       cli_cmd_flag_sensitive          = 0x01
+};
+
+struct wpa_cli_cmd {
+       const char *cmd;
+       int (*handler)(struct wpa_ctrl *ctrl, int argc, char *argv[]);
+       enum wpa_cli_cmd_flags flags;
+       const char *usage;
+};
+
+static struct wpa_cli_cmd wpa_cli_commands[] = {
+       { "status", wpa_cli_cmd_status,
+         cli_cmd_flag_none,
+         "[verbose] = get current WPA/EAPOL/EAP status" },
+       { "ping", wpa_cli_cmd_ping,
+         cli_cmd_flag_none,
+         "= pings wpa_supplicant" },
+       { "note", wpa_cli_cmd_note,
+         cli_cmd_flag_none,
+         "<text> = add a note to wpa_supplicant debug log" },
+       { "mib", wpa_cli_cmd_mib,
+         cli_cmd_flag_none,
+         "= get MIB variables (dot1x, dot11)" },
+       { "help", wpa_cli_cmd_help,
+         cli_cmd_flag_none,
+         "= show this usage help" },
+       { "interface", wpa_cli_cmd_interface,
+         cli_cmd_flag_none,
+         "[ifname] = show interfaces/select interface" },
+       { "level", wpa_cli_cmd_level,
+         cli_cmd_flag_none,
+         "<debug level> = change debug level" },
+       { "license", wpa_cli_cmd_license,
+         cli_cmd_flag_none,
+         "= show full wpa_cli license" },
+       { "quit", wpa_cli_cmd_quit,
+         cli_cmd_flag_none,
+         "= exit wpa_cli" },
+       { "set", wpa_cli_cmd_set,
+         cli_cmd_flag_none,
          "= set variables (shows list of variables when run without "
          "arguments)" },
        { "logon", wpa_cli_cmd_logon,
@@ -1282,15 +2297,124 @@ static struct wpa_cli_cmd wpa_cli_commands[] = {
          cli_cmd_flag_sensitive,
          "<BSSID> [PIN] = start WPS PIN method (returns PIN, if not "
          "hardcoded)" },
+       { "wps_check_pin", wpa_cli_cmd_wps_check_pin,
+         cli_cmd_flag_sensitive,
+         "<PIN> = verify PIN checksum" },
+       { "wps_cancel", wpa_cli_cmd_wps_cancel, cli_cmd_flag_none,
+         "Cancels the pending WPS operation" },
+#ifdef CONFIG_WPS_OOB
        { "wps_oob", wpa_cli_cmd_wps_oob,
          cli_cmd_flag_sensitive,
-         "<OOB_DEV_TYPE> <OOB_PATH> <OOB_METHOD> = start WPS OOB" },
+         "<DEV_TYPE> <PATH> <METHOD> [DEV_NAME] = start WPS OOB" },
+#endif /* CONFIG_WPS_OOB */
        { "wps_reg", wpa_cli_cmd_wps_reg,
          cli_cmd_flag_sensitive,
          "<BSSID> <AP PIN> = start WPS Registrar to configure an AP" },
+       { "wps_er_start", wpa_cli_cmd_wps_er_start,
+         cli_cmd_flag_none,
+         "[IP address] = start Wi-Fi Protected Setup External Registrar" },
+       { "wps_er_stop", wpa_cli_cmd_wps_er_stop,
+         cli_cmd_flag_none,
+         "= stop Wi-Fi Protected Setup External Registrar" },
+       { "wps_er_pin", wpa_cli_cmd_wps_er_pin,
+         cli_cmd_flag_sensitive,
+         "<UUID> <PIN> = add an Enrollee PIN to External Registrar" },
+       { "wps_er_pbc", wpa_cli_cmd_wps_er_pbc,
+         cli_cmd_flag_none,
+         "<UUID> = accept an Enrollee PBC using External Registrar" },
+       { "wps_er_learn", wpa_cli_cmd_wps_er_learn,
+         cli_cmd_flag_sensitive,
+         "<UUID> <PIN> = learn AP configuration" },
+       { "wps_er_config", wpa_cli_cmd_wps_er_config,
+         cli_cmd_flag_sensitive,
+         "<UUID> <PIN> <SSID> <auth> <encr> <key> = configure AP" },
        { "ibss_rsn", wpa_cli_cmd_ibss_rsn,
          cli_cmd_flag_none,
          "<addr> = request RSN authentication with <addr> in IBSS" },
+#ifdef CONFIG_AP
+       { "sta", wpa_cli_cmd_sta,
+         cli_cmd_flag_none,
+         "<addr> = get information about an associated station (AP)" },
+       { "all_sta", wpa_cli_cmd_all_sta,
+         cli_cmd_flag_none,
+         "= get information about all associated stations (AP)" },
+#endif /* CONFIG_AP */
+       { "suspend", wpa_cli_cmd_suspend, cli_cmd_flag_none,
+         "= notification of suspend/hibernate" },
+       { "resume", wpa_cli_cmd_resume, cli_cmd_flag_none,
+         "= notification of resume/thaw" },
+       { "drop_sa", wpa_cli_cmd_drop_sa, cli_cmd_flag_none,
+         "= drop SA without deauth/disassoc (test command)" },
+       { "roam", wpa_cli_cmd_roam,
+         cli_cmd_flag_none,
+         "<addr> = roam to the specified BSS" },
+#ifdef CONFIG_P2P
+       { "p2p_find", wpa_cli_cmd_p2p_find, cli_cmd_flag_none,
+         "[timeout] [type=*] = find P2P Devices for up-to timeout seconds" },
+       { "p2p_stop_find", wpa_cli_cmd_p2p_stop_find, cli_cmd_flag_none,
+         "= stop P2P Devices search" },
+       { "p2p_connect", wpa_cli_cmd_p2p_connect, cli_cmd_flag_none,
+         "<addr> <\"pbc\"|PIN> = connect to a P2P Devices" },
+       { "p2p_listen", wpa_cli_cmd_p2p_listen, cli_cmd_flag_none,
+         "[timeout] = listen for P2P Devices for up-to timeout seconds" },
+       { "p2p_group_remove", wpa_cli_cmd_p2p_group_remove, cli_cmd_flag_none,
+         "<ifname> = remote P2P group interface (terminate group if GO)" },
+       { "p2p_group_add", wpa_cli_cmd_p2p_group_add, cli_cmd_flag_none,
+         "= add a new P2P group (local end as GO)" },
+       { "p2p_prov_disc", wpa_cli_cmd_p2p_prov_disc, cli_cmd_flag_none,
+         "<addr> <method> = request provisioning discovery" },
+       { "p2p_get_passphrase", wpa_cli_cmd_p2p_get_passphrase,
+         cli_cmd_flag_none,
+         "= get the passphrase for a group (GO only)" },
+       { "p2p_serv_disc_req", wpa_cli_cmd_p2p_serv_disc_req,
+         cli_cmd_flag_none,
+         "<addr> <TLVs> = schedule service discovery request" },
+       { "p2p_serv_disc_cancel_req", wpa_cli_cmd_p2p_serv_disc_cancel_req,
+         cli_cmd_flag_none,
+         "<id> = cancel pending service discovery request" },
+       { "p2p_serv_disc_resp", wpa_cli_cmd_p2p_serv_disc_resp,
+         cli_cmd_flag_none,
+         "<freq> <addr> <dialog token> <TLVs> = service discovery response" },
+       { "p2p_service_update", wpa_cli_cmd_p2p_service_update,
+         cli_cmd_flag_none,
+         "= indicate change in local services" },
+       { "p2p_serv_disc_external", wpa_cli_cmd_p2p_serv_disc_external,
+         cli_cmd_flag_none,
+         "<external> = set external processing of service discovery" },
+       { "p2p_service_flush", wpa_cli_cmd_p2p_service_flush,
+         cli_cmd_flag_none,
+         "= remove all stored service entries" },
+       { "p2p_service_add", wpa_cli_cmd_p2p_service_add,
+         cli_cmd_flag_none,
+         "<bonjour|upnp> <query|version> <response|service> = add a local "
+         "service" },
+       { "p2p_service_del", wpa_cli_cmd_p2p_service_del,
+         cli_cmd_flag_none,
+         "<bonjour|upnp> <query|version> [|service] = remove a local "
+         "service" },
+       { "p2p_reject", wpa_cli_cmd_p2p_reject,
+         cli_cmd_flag_none,
+         "<addr> = reject connection attempts from a specific peer" },
+       { "p2p_invite", wpa_cli_cmd_p2p_invite,
+         cli_cmd_flag_none,
+         "<cmd> [peer=addr] = invite peer" },
+       { "p2p_peers", wpa_cli_cmd_p2p_peers, cli_cmd_flag_none,
+         "[discovered] = list known (optionally, only fully discovered) P2P "
+         "peers" },
+       { "p2p_peer", wpa_cli_cmd_p2p_peer, cli_cmd_flag_none,
+         "<address> = show information about known P2P peer" },
+       { "p2p_set", wpa_cli_cmd_p2p_set, cli_cmd_flag_none,
+         "<field> <value> = set a P2P parameter" },
+       { "p2p_flush", wpa_cli_cmd_p2p_flush, cli_cmd_flag_none,
+         "= flush P2P state" },
+       { "p2p_presence_req", wpa_cli_cmd_p2p_presence_req, cli_cmd_flag_none,
+         "[<duration> <interval>] [<duration> <interval>] = request GO "
+         "presence" },
+       { "p2p_ext_listen", wpa_cli_cmd_p2p_ext_listen, cli_cmd_flag_none,
+         "[<period> <interval>] = set extended listen timing" },
+#endif /* CONFIG_P2P */
+       { "sta_autoconnect", wpa_cli_cmd_sta_autoconnect, cli_cmd_flag_none,
+         "<0/1> = disable/enable automatic reconnection" },
        { NULL, NULL, cli_cmd_flag_none, NULL }
 };
 
@@ -1480,6 +2604,18 @@ static void wpa_cli_action_process(const char *msg)
                        wpa_cli_connected = 0;
                        wpa_cli_exec(action_file, ctrl_ifname, "DISCONNECTED");
                }
+       } else if (str_match(pos, P2P_EVENT_GROUP_STARTED)) {
+               wpa_cli_exec(action_file, ctrl_ifname, pos);
+       } else if (str_match(pos, P2P_EVENT_GROUP_REMOVED)) {
+               wpa_cli_exec(action_file, ctrl_ifname, pos);
+       } else if (str_match(pos, P2P_EVENT_CROSS_CONNECT_ENABLE)) {
+               wpa_cli_exec(action_file, ctrl_ifname, pos);
+       } else if (str_match(pos, P2P_EVENT_CROSS_CONNECT_DISABLE)) {
+               wpa_cli_exec(action_file, ctrl_ifname, pos);
+       } else if (str_match(pos, WPS_EVENT_SUCCESS)) {
+               wpa_cli_exec(action_file, ctrl_ifname, pos);
+       } else if (str_match(pos, WPS_EVENT_FAIL)) {
+               wpa_cli_exec(action_file, ctrl_ifname, pos);
        } else if (str_match(pos, WPA_EVENT_TERMINATING)) {
                printf("wpa_supplicant is terminating - stop monitoring\n");
                wpa_cli_quit = 1;
@@ -1498,16 +2634,7 @@ static void wpa_cli_action_cb(char *msg, size_t len)
 static void wpa_cli_reconnect(void)
 {
        wpa_cli_close_connection();
-       ctrl_conn = wpa_cli_open_connection(ctrl_ifname);
-       if (ctrl_conn) {
-               printf("Connection to wpa_supplicant re-established\n");
-               if (wpa_ctrl_attach(ctrl_conn) == 0) {
-                       wpa_cli_attached = 1;
-               } else {
-                       printf("Warning: Failed to attach to "
-                              "wpa_supplicant.\n");
-               }
-       }
+       wpa_cli_open_connection(ctrl_ifname, 1);
 }
 
 
@@ -1527,10 +2654,13 @@ static void wpa_cli_recv_pending(struct wpa_ctrl *ctrl, int in_read,
                        if (action_monitor)
                                wpa_cli_action_process(buf);
                        else {
-                               if (in_read && first)
-                                       printf("\n");
-                               first = 0;
-                               printf("%s\n", buf);
+                               if (wpa_cli_show_event(buf)) {
+                                       if (in_read && first)
+                                               printf("\r");
+                                       first = 0;
+                                       printf("%s\n", buf);
+                                       readline_redraw();
+                               }
                        }
                } else {
                        printf("Could not read pending message.\n");
@@ -1560,7 +2690,7 @@ static char * wpa_cli_cmd_gen(const char *text, int state)
        while ((cmd = wpa_cli_commands[i].cmd)) {
                i++;
                if (os_strncasecmp(cmd, text, len) == 0)
-                       return os_strdup(cmd);
+                       return strdup(cmd);
        }
 
        return NULL;
@@ -1569,14 +2699,59 @@ static char * wpa_cli_cmd_gen(const char *text, int state)
 
 static char * wpa_cli_dummy_gen(const char *text, int state)
 {
+       int i;
+
+       for (i = 0; wpa_cli_commands[i].cmd; i++) {
+               const char *cmd = wpa_cli_commands[i].cmd;
+               size_t len = os_strlen(cmd);
+               if (os_strncasecmp(rl_line_buffer, cmd, len) == 0 &&
+                   rl_line_buffer[len] == ' ') {
+                       printf("\n%s\n", wpa_cli_commands[i].usage);
+                       readline_redraw();
+                       break;
+               }
+       }
+
+       rl_attempted_completion_over = 1;
+       return NULL;
+}
+
+
+static char * wpa_cli_status_gen(const char *text, int state)
+{
+       static int i, len;
+       char *options[] = {
+               "verbose", NULL
+       };
+       char *t;
+
+       if (state == 0) {
+               i = 0;
+               len = os_strlen(text);
+       }
+
+       while ((t = options[i])) {
+               i++;
+               if (os_strncasecmp(t, text, len) == 0)
+                       return strdup(t);
+       }
+
+       rl_attempted_completion_over = 1;
        return NULL;
 }
 
 
 static char ** wpa_cli_completion(const char *text, int start, int end)
 {
-       return rl_completion_matches(text, start == 0 ?
-                                    wpa_cli_cmd_gen : wpa_cli_dummy_gen);
+       char * (*func)(const char *text, int state);
+
+       if (start == 0)
+               func = wpa_cli_cmd_gen;
+       else if (os_strncasecmp(rl_line_buffer, "status ", 7) == 0)
+               func = wpa_cli_status_gen;
+       else
+               func = wpa_cli_dummy_gen;
+       return rl_completion_matches(text, func);
 }
 #endif /* CONFIG_READLINE */
 
@@ -1613,10 +2788,14 @@ static void wpa_cli_interactive(void)
 #endif /* CONFIG_READLINE */
 
        do {
-               wpa_cli_recv_pending(ctrl_conn, 0, 0);
+               wpa_cli_recv_pending(mon_conn, 0, 0);
 #ifndef CONFIG_NATIVE_WINDOWS
                alarm(ping_interval);
 #endif /* CONFIG_NATIVE_WINDOWS */
+#ifdef CONFIG_WPA_CLI_FORK
+               if (mon_pid)
+                       kill(mon_pid, SIGUSR1);
+#endif /* CONFIG_WPA_CLI_FORK */
 #ifdef CONFIG_READLINE
                cmd = readline("> ");
                if (cmd && *cmd) {
@@ -1637,7 +2816,7 @@ static void wpa_cli_interactive(void)
 #endif /* CONFIG_NATIVE_WINDOWS */
                if (cmd == NULL)
                        break;
-               wpa_cli_recv_pending(ctrl_conn, 0, 0);
+               wpa_cli_recv_pending(mon_conn, 0, 0);
                pos = cmd;
                while (*pos != '\0') {
                        if (*pos == '\n') {
@@ -1671,7 +2850,11 @@ static void wpa_cli_interactive(void)
                        wpa_request(ctrl_conn, argc, argv);
 
                if (cmd != cmdbuf)
-                       os_free(cmd);
+                       free(cmd);
+#ifdef CONFIG_WPA_CLI_FORK
+               if (mon_pid)
+                       kill(mon_pid, SIGUSR2);
+#endif /* CONFIG_WPA_CLI_FORK */
        } while (!wpa_cli_quit);
 
 #ifdef CONFIG_READLINE
@@ -1761,6 +2944,14 @@ static void wpa_cli_terminate(int sig)
 }
 
 
+#ifdef CONFIG_WPA_CLI_FORK
+static void wpa_cli_usr1(int sig)
+{
+       readline_redraw();
+}
+#endif /* CONFIG_WPA_CLI_FORK */
+
+
 #ifndef CONFIG_NATIVE_WINDOWS
 static void wpa_cli_alarm(int sig)
 {
@@ -1771,8 +2962,8 @@ static void wpa_cli_alarm(int sig)
        }
        if (!ctrl_conn)
                wpa_cli_reconnect();
-       if (ctrl_conn)
-               wpa_cli_recv_pending(ctrl_conn, 1, 0);
+       if (mon_conn)
+               wpa_cli_recv_pending(mon_conn, 1, 0);
        alarm(ping_interval);
 }
 #endif /* CONFIG_NATIVE_WINDOWS */
@@ -1835,7 +3026,6 @@ static char * wpa_cli_get_default_ifname(void)
 
 int main(int argc, char *argv[])
 {
-       int interactive;
        int warning_displayed = 0;
        int c;
        int daemonize = 0;
@@ -1902,31 +3092,6 @@ int main(int argc, char *argv[])
                }
        }
 
-       for (; !global;) {
-               if (ctrl_ifname == NULL)
-                       ctrl_ifname = wpa_cli_get_default_ifname();
-               ctrl_conn = wpa_cli_open_connection(ctrl_ifname);
-               if (ctrl_conn) {
-                       if (warning_displayed)
-                               printf("Connection established.\n");
-                       break;
-               }
-
-               if (!interactive) {
-                       perror("Failed to connect to wpa_supplicant - "
-                              "wpa_ctrl_open");
-                       return -1;
-               }
-
-               if (!warning_displayed) {
-                       printf("Could not connect to wpa_supplicant - "
-                              "re-trying\n");
-                       warning_displayed = 1;
-               }
-               os_sleep(1, 0);
-               continue;
-       }
-
 #ifndef _WIN32_WCE
        signal(SIGINT, wpa_cli_terminate);
        signal(SIGTERM, wpa_cli_terminate);
@@ -1934,15 +3099,45 @@ int main(int argc, char *argv[])
 #ifndef CONFIG_NATIVE_WINDOWS
        signal(SIGALRM, wpa_cli_alarm);
 #endif /* CONFIG_NATIVE_WINDOWS */
+#ifdef CONFIG_WPA_CLI_FORK
+       signal(SIGUSR1, wpa_cli_usr1);
+#endif /* CONFIG_WPA_CLI_FORK */
+
+       if (ctrl_ifname == NULL)
+               ctrl_ifname = wpa_cli_get_default_ifname();
+
+       if (interactive) {
+               for (; !global;) {
+                       if (wpa_cli_open_connection(ctrl_ifname, 1) == 0) {
+                               if (warning_displayed)
+                                       printf("Connection established.\n");
+                               break;
+                       }
 
-       if (interactive || action_file) {
-               if (wpa_ctrl_attach(ctrl_conn) == 0) {
-                       wpa_cli_attached = 1;
-               } else {
-                       printf("Warning: Failed to attach to "
-                              "wpa_supplicant.\n");
-                       if (!interactive)
+                       if (!warning_displayed) {
+                               printf("Could not connect to wpa_supplicant - "
+                                      "re-trying\n");
+                               warning_displayed = 1;
+                       }
+                       os_sleep(1, 0);
+                       continue;
+               }
+       } else {
+               if (!global &&
+                   wpa_cli_open_connection(ctrl_ifname, 0) < 0) {
+                       perror("Failed to connect to wpa_supplicant - "
+                              "wpa_ctrl_open");
+                       return -1;
+               }
+
+               if (action_file) {
+                       if (wpa_ctrl_attach(ctrl_conn) == 0) {
+                               wpa_cli_attached = 1;
+                       } else {
+                               printf("Warning: Failed to attach to "
+                                      "wpa_supplicant.\n");
                                return -1;
+                       }
                }
        }