[iwmgmt] Add user-level 802.11 management commands and common error tables
authorJoshua Oreman <oremanj@xenon.get-linux.org>
Sun, 21 Jun 2009 01:08:14 +0000 (18:08 -0700)
committerJoshua Oreman <oremanj@xenon.get-linux.org>
Sun, 21 Jun 2009 01:13:23 +0000 (18:13 -0700)
Since these commands should be compiled in by default when support
for a wireless card is included, never compiled in when 802.11 support
is not, and able to be disabled even when the base wireless support
is included, they are conditionally linked in not by config.o (as is
usual) but by the main 802.11 stack file net80211.o.

src/config/general.h
src/core/config.c
src/hci/commands/iwmgmt_cmd.c [new file with mode: 0644]
src/hci/wireless_errors.c [new file with mode: 0644]
src/include/gpxe/errfile.h
src/include/usr/iwmgmt.h [new file with mode: 0644]
src/net/80211/net80211.c
src/usr/iwmgmt.c [new file with mode: 0644]

index dfc8700..6daefc9 100644 (file)
@@ -98,6 +98,7 @@ FILE_LICENCE ( GPL2_OR_LATER );
 #define        NVO_CMD                 /* Non-volatile option storage commands */
 #define        CONFIG_CMD              /* Option configuration console */
 #define        IFMGMT_CMD              /* Interface management commands */
+#define        IWMGMT_CMD              /* Wireless interface management commands */
 #define        ROUTE_CMD               /* Routing table management commands */
 #define IMAGE_CMD              /* Image management commands */
 #define DHCP_CMD               /* DHCP management commands */
@@ -106,6 +107,12 @@ FILE_LICENCE ( GPL2_OR_LATER );
 #undef TIME_CMD                /* Time commands */
 #undef DIGEST_CMD              /* Image crypto digest commands */
 
+/*
+ * Error message tables to include
+ *
+ */
+#undef ERRMSG_80211            /* All 802.11 error descriptions (~3.3kb) */
+
 /*
  * Obscure configuration options
  *
index ecaf781..082d3f3 100644 (file)
@@ -183,6 +183,7 @@ REQUIRE_OBJECT ( config_cmd );
 #ifdef IFMGMT_CMD
 REQUIRE_OBJECT ( ifmgmt_cmd );
 #endif
+/* IWMGMT_CMD is brought in by net80211.c if requested */
 #ifdef ROUTE_CMD
 REQUIRE_OBJECT ( route_cmd );
 #endif
@@ -205,6 +206,14 @@ REQUIRE_OBJECT ( time_cmd );
 REQUIRE_OBJECT ( digest_cmd );
 #endif
 
+/*
+ * Drag in error message tables
+ *
+ */
+#ifdef ERRMSG_80211
+REQUIRE_OBJECT ( wireless_errors );
+#endif
+
 /*
  * Drag in miscellaneous objects
  *
diff --git a/src/hci/commands/iwmgmt_cmd.c b/src/hci/commands/iwmgmt_cmd.c
new file mode 100644 (file)
index 0000000..4cfe292
--- /dev/null
@@ -0,0 +1,91 @@
+/*
+ * Copyright (C) 2009 Joshua Oreman <oremanj@rwcr.net>.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either version 2 of the
+ * License, or any later version.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ */
+
+FILE_LICENCE ( GPL2_OR_LATER );
+
+#include <gpxe/netdevice.h>
+#include <gpxe/net80211.h>
+#include <gpxe/command.h>
+#include <usr/iwmgmt.h>
+#include "ifmgmt_cmd.h"
+
+/* "iwstat" command */
+
+static int iwstat_payload ( struct net_device *netdev ) {
+       struct net80211_device *dev = net80211_get ( netdev );
+
+       if ( dev )
+               iwstat ( dev );
+
+       return 0;
+}
+
+static int iwstat_exec ( int argc, char **argv ) {
+       return ifcommon_exec ( iwstat_payload, "Display wireless status of",
+                              argc, argv );
+}
+
+/* "iwlist" command */
+
+static int iwlist_payload ( struct net_device *netdev ) {
+       struct net80211_device *dev = net80211_get ( netdev );
+
+       if ( dev )
+               return iwlist ( dev );
+
+       return 0;
+}
+
+static int iwlist_exec ( int argc, char **argv ) {
+       return ifcommon_exec ( iwlist_payload,
+                              "List wireless networks available via",
+                              argc, argv );
+}
+
+/* "iwassoc" command */
+
+static int iwassoc_payload ( struct net_device *netdev ) {
+       struct net80211_device *dev = net80211_get ( netdev );
+
+       if ( dev )
+               iwassoc ( dev );
+
+       return 0;
+}
+
+static int iwassoc_exec ( int argc, char **argv ) {
+       return ifcommon_exec ( iwassoc_payload,
+                              "Retry 802.11 association for",
+                              argc, argv );
+}
+
+/** Wireless interface management commands */
+struct command iwmgmt_commands[] __command = {
+       {
+               .name = "iwstat",
+               .exec = iwstat_exec,
+       },
+       {
+               .name = "iwlist",
+               .exec = iwlist_exec,
+       },
+       {
+               .name = "iwassoc",
+               .exec = iwassoc_exec,
+       },
+};
diff --git a/src/hci/wireless_errors.c b/src/hci/wireless_errors.c
new file mode 100644 (file)
index 0000000..834fe9f
--- /dev/null
@@ -0,0 +1,118 @@
+/*
+ * Copyright (C) 2009 Joshua Oreman <oremanj@rwcr.net>.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either version 2 of the
+ * License, or any later version.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ */
+
+FILE_LICENCE ( GPL2_OR_LATER );
+
+#include <errno.h>
+#include <gpxe/errortab.h>
+
+/* Record errors as though they come from the 802.11 stack */
+#undef ERRFILE
+#define ERRFILE ERRFILE_net80211
+
+/** All 802.11 errors
+ *
+ * These follow the 802.11 standard as much as is feasible, but most
+ * have been abbreviated to fit the 50-character limit imposed by
+ * strerror.
+ */
+struct errortab wireless_errors[] __errortab = {
+       /* gPXE 802.11 stack errors */
+       { EINVAL | EUNIQ_01, "Packet too short" },
+       { EINVAL | EUNIQ_02, "Packet 802.11 version not supported" },
+       { EINVAL | EUNIQ_03, "Packet not a data packet" },
+       { EINVAL | EUNIQ_04, "Packet not from an Access Point" },
+       { EINVAL | EUNIQ_05, "Packet has invalid LLC header" },
+       { EINVAL | EUNIQ_06, "Cannot encrypt frame without a cryptosystem", },
+       { EINVAL | EUNIQ_07, "Invalid active scan requested" },
+
+       /* 802.11 status codes (IEEE Std 802.11-2007, Table 7-23) */
+       /* Maximum error length: 50 chars                                            | */
+       { ECONNREFUSED | EUNIQ_01, "Unspecified failure" },
+       { ECONNREFUSED | EUNIQ_0A, "Cannot support all requested capabilities" },
+       { ECONNREFUSED | EUNIQ_0B, "Reassociation denied due to lack of association" },
+       { ECONNREFUSED | EUNIQ_0C, "Association denied for another reason" },
+       { ECONNREFUSED | EUNIQ_0D, "Authentication algorithm unsupported" },
+       { ECONNREFUSED | EUNIQ_0E, "Authentication sequence number unexpected" },
+       { ECONNREFUSED | EUNIQ_0F, "Authentication rejected due to challenge failure" },
+       { ECONNREFUSED | EUNIQ_10, "Authentication rejected due to timeout" },
+       { ECONNREFUSED | EUNIQ_11, "Association denied because AP is out of resources" },
+       { ECONNREFUSED | EUNIQ_12, "Association denied; basic rate support required" },
+       { ECONNREFUSED | EUNIQ_13, "Association denied; short preamble support req'd" },
+       { ECONNREFUSED | EUNIQ_14, "Association denied; PBCC modulation support req'd" },
+       { ECONNREFUSED | EUNIQ_15, "Association denied; Channel Agility support req'd" },
+       { ECONNREFUSED | EUNIQ_16, "Association denied; Spectrum Management required" },
+       { ECONNREFUSED | EUNIQ_17, "Association denied; Power Capability unacceptable" },
+       { ECONNREFUSED | EUNIQ_18, "Association denied; Supported Channels unacceptable" },
+       { ECONNREFUSED | EUNIQ_19, "Association denied; Short Slot Tume support req'd" },
+       { ECONNREFUSED | EUNIQ_1A, "Association denied; DSSS-OFDM support required" },
+       { EHOSTUNREACH,            "Unspecified, QoS-related failure" },
+       { EHOSTUNREACH | EUNIQ_01, "Association denied; QoS AP out of QoS resources" },
+       { EHOSTUNREACH | EUNIQ_02, "Association denied due to excessively poor link" },
+       { EHOSTUNREACH | EUNIQ_03, "Association denied; QoS support required" },
+       { EHOSTUNREACH | EUNIQ_05, "The request has been declined" },
+       { EHOSTUNREACH | EUNIQ_06, "Request unsuccessful due to invalid parameters" },
+       { EHOSTUNREACH | EUNIQ_07, "TS not created due to bad specification" },
+       { EHOSTUNREACH | EUNIQ_08, "Invalid information element" },
+       { EHOSTUNREACH | EUNIQ_09, "Invalid group cipher" },
+       { EHOSTUNREACH | EUNIQ_0A, "Invalid pairwise cipher" },
+       { EHOSTUNREACH | EUNIQ_0B, "Invalid AKMP" },
+       { EHOSTUNREACH | EUNIQ_0C, "Unsupported RSN information element version" },
+       { EHOSTUNREACH | EUNIQ_0D, "Invalid RSN information element capabilities" },
+       { EHOSTUNREACH | EUNIQ_0E, "Cipher suite rejected because of security policy" },
+       { EHOSTUNREACH | EUNIQ_0F, "TS not created due to insufficient delay" },
+       { EHOSTUNREACH | EUNIQ_10, "Direct link is not allowed in the BSS by policy" },
+       { EHOSTUNREACH | EUNIQ_11, "The Destination STA is not present within the BSS" },
+       { EHOSTUNREACH | EUNIQ_12, "The Destination STA is not a QoS STA" },
+       { EHOSTUNREACH | EUNIQ_13, "Association denied; Listen Interval is too large" },
+
+       /* 802.11 reason codes (IEEE Std 802.11-2007, Table 7-22) */
+       /* Maximum error length: 50 chars                                          | */
+       { ECONNRESET | EUNIQ_01, "Unspecified reason" },
+       { ECONNRESET | EUNIQ_02, "Previous authentication no longer valid" },
+       { ECONNRESET | EUNIQ_03, "Deauthenticated due to leaving network" },
+       { ECONNRESET | EUNIQ_04, "Disassociated due to inactivity" },
+       { ECONNRESET | EUNIQ_05, "Disassociated because AP is out of resources" },
+       { ECONNRESET | EUNIQ_06, "Class 2 frame received from nonauthenticated STA" },
+       { ECONNRESET | EUNIQ_07, "Class 3 frame received from nonassociated STA" },
+       { ECONNRESET | EUNIQ_08, "Disassociated due to roaming" },
+       { ECONNRESET | EUNIQ_09, "STA requesting (re)association not authenticated" },
+       { ECONNRESET | EUNIQ_0A, "Disassociated; Power Capability unacceptable" },
+       { ECONNRESET | EUNIQ_0B, "Disassociated; Supported Channels unacceptable" },
+       { ECONNRESET | EUNIQ_0D, "Invalid information element" },
+       { ECONNRESET | EUNIQ_0E, "Message integrity code (MIC) failure" },
+       { ECONNRESET | EUNIQ_0F, "4-Way Handshake timeout" },
+       { ECONNRESET | EUNIQ_10, "Group Key Handshake timeout" },
+       { ECONNRESET | EUNIQ_11, "4-Way Handshake information element changed unduly" },
+       { ECONNRESET | EUNIQ_12, "Invalid group cipher" },
+       { ECONNRESET | EUNIQ_13, "Invalid pairwise cipher" },
+       { ECONNRESET | EUNIQ_14, "Invalid AKMP" },
+       { ECONNRESET | EUNIQ_15, "Unsupported RSN information element version" },
+       { ECONNRESET | EUNIQ_16, "Invalid RSN information element capabilities" },
+       { ECONNRESET | EUNIQ_17, "IEEE 802.1X authentication failed" },
+       { ECONNRESET | EUNIQ_18, "Cipher suite rejected because of security policy" },
+       { ENETRESET,            "Disassociated for unspecified, QoS-related reason" },
+       { ENETRESET | EUNIQ_01, "Disassociated; QoS AP is out of QoS resources" },
+       { ENETRESET | EUNIQ_02, "Disassociated due to excessively poor link" },
+       { ENETRESET | EUNIQ_03, "Disassociated due to TXOP limit violation" },
+       { ENETRESET | EUNIQ_04, "Requested; STA is leaving the BSS (or resetting)" },
+       { ENETRESET | EUNIQ_05, "Requested; does not want to use the mechanism" },
+       { ENETRESET | EUNIQ_06, "Requested; setup is required" },
+       { ENETRESET | EUNIQ_07, "Requested from peer STA due to timeout" },
+       { ENETRESET | EUNIQ_0D, "Peer STA does not support requested cipher suite" },
+};
index 0f29a67..7a74f50 100644 (file)
@@ -176,6 +176,7 @@ FILE_LICENCE ( GPL2_OR_LATER );
 #define ERRFILE_pxemenu                      ( ERRFILE_OTHER | 0x00150000 )
 #define ERRFILE_x509                 ( ERRFILE_OTHER | 0x00160000 )
 #define ERRFILE_login_ui             ( ERRFILE_OTHER | 0x00170000 )
+#define ERRFILE_iwmgmt               ( ERRFILE_OTHER | 0x00180000 )
 
 /** @} */
 
diff --git a/src/include/usr/iwmgmt.h b/src/include/usr/iwmgmt.h
new file mode 100644 (file)
index 0000000..8a52f03
--- /dev/null
@@ -0,0 +1,18 @@
+#ifndef _USR_IWMGMT_H
+#define _USR_IWMGMT_H
+
+/** @file
+ *
+ * Wireless network interface management
+ *
+ */
+
+FILE_LICENCE ( GPL2_OR_LATER );
+
+struct net80211_device;
+
+extern void iwstat ( struct net80211_device *dev );
+extern int iwlist ( struct net80211_device *dev );
+extern void iwassoc ( struct net80211_device *dev );
+
+#endif /* _USR_IWMGMT_H */
index 1dad8f9..2473372 100644 (file)
@@ -39,6 +39,13 @@ FILE_LICENCE ( GPL2_OR_LATER );
  * 802.11 device management
  */
 
+/* Bring in IWMGMT_CMD here if enabled; the commands are useless if
+   nothing else requires wireless. */
+#include <config/general.h>
+#ifdef IWMGMT_CMD
+REQUIRE_OBJECT ( iwmgmt_cmd );
+#endif
+
 /* Disambiguate the EINVAL's a bit */
 #define EINVAL_PKT_TOO_SHORT   ( EINVAL | EUNIQ_01 )
 #define EINVAL_PKT_VERSION     ( EINVAL | EUNIQ_02 )
diff --git a/src/usr/iwmgmt.c b/src/usr/iwmgmt.c
new file mode 100644 (file)
index 0000000..17aca60
--- /dev/null
@@ -0,0 +1,249 @@
+/*
+ * Copyright (C) 2009 Joshua Oreman <oremanj@rwcr.net>.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either version 2 of the
+ * License, or any later version.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ */
+
+FILE_LICENCE ( GPL2_OR_LATER );
+
+#include <stdio.h>
+#include <console.h>
+#include <string.h>
+#include <errno.h>
+#include <gpxe/net80211.h>
+#include <gpxe/ethernet.h>
+#include <usr/ifmgmt.h>
+#include <usr/iwmgmt.h>
+#include <gpxe/errortab.h>
+
+/** @file
+ *
+ * Wireless network interface management
+ *
+ */
+
+/**
+ * Print status of 802.11 device
+ *
+ * @v dev      802.11 device
+ */
+void iwstat ( struct net80211_device *dev ) {
+
+       ifstat ( dev->netdev );
+
+       printf ( "  [802.11 ");
+       if ( dev->state & NET80211_ASSOCIATED ) {
+               printf ( "SSID '%s', ", dev->essid );
+       } else {
+               printf ( "not associated, " );
+       }
+       if ( dev->channel < dev->nr_channels && dev->rate < dev->nr_rates ) {
+               printf ( "Ch:%d Sig:%d", dev->channels[dev->channel].channel_nr,
+                        dev->last_signal );
+               switch ( dev->hw->signal_type ) {
+               case NET80211_SIGNAL_NONE:
+                       printf ( "?" );
+                       break;
+               case NET80211_SIGNAL_ARBITRARY:
+                       printf ( "/%d", dev->hw->signal_max );
+                       break;
+               case NET80211_SIGNAL_DB:
+                       printf ( "/%d dB", dev->hw->signal_max );
+                       break;
+               case NET80211_SIGNAL_DBM:
+                       printf ( " dBm" );
+                       break;
+               }
+               printf ( ", Rate:%d Mbps]\n", dev->rates[dev->rate] / 10 );
+       } else {
+               printf ( "antenna off]\n" );
+       }
+
+       if ( dev->state & NET80211_WORKING ) {
+               printf ( "  [associating" );
+               if ( dev->associating )
+                       printf ( " to '%s'", dev->associating->essid );
+               printf ( "...]\n" );
+       } else if ( dev->assoc_rc ) {
+               printf ( "  [association error: %s]\n",
+                        strerror ( dev->assoc_rc ) );
+       }
+}
+
+/** Identifiers for 802.11 cryptography types, indexed by type number */
+static const char *crypto_types[] = {
+       [NET80211_CRYPT_NONE] = "Open",
+       [NET80211_CRYPT_WEP] = "WEP",
+       [NET80211_CRYPT_TKIP] = "WPA",
+       [NET80211_CRYPT_CCMP] = "WPA2",
+};
+
+/** Number of 802.11 cryptography types defined */
+#define NR_CRYPTO_TYPES ( sizeof ( crypto_types ) / sizeof ( crypto_types[0] ) )
+
+/** Identifiers for 802.11 authentication types, indexed by type number */
+static const char *auth_types[] = {
+       [NET80211_SECPROT_NONE] = "",
+       [NET80211_SECPROT_PSK] = "PSK",
+       [NET80211_SECPROT_EAP] = "802.1X",
+};
+
+/** Number of 802.11 authentication types defined */
+#define NR_AUTH_TYPES ( sizeof ( auth_types ) / sizeof ( auth_types[0] ) )
+
+/**
+ * Scan for wireless networks using 802.11 device
+ *
+ * @v dev      802.11 device
+ * @v active   Whether to use active scanning
+ *
+ * The list of networks found will be printed in tabular format.
+ *
+ * This function is safe to call at all times, whether the 802.11
+ * device is open or not, but if called while the auto-association
+ * task is running it will return an error indication.
+ */
+int iwlist ( struct net80211_device *dev ) {
+       struct net80211_probe_ctx *ctx;
+       struct list_head *networks;
+       struct net80211_wlan *wlan;
+       char ssid_buf[22];
+       int rc;
+       unsigned i;
+       int was_opened = dev->netdev->state & NETDEV_OPEN;
+       int was_channel = dev->channels[dev->channel].channel_nr;
+
+       if ( ! was_opened ) {
+               dev->state |= NET80211_NO_ASSOC;
+               rc = netdev_open ( dev->netdev );
+               if ( rc < 0 )
+                       goto err;
+       }
+
+       if ( dev->state & NET80211_WORKING ) {
+               rc = -EINVAL;
+               goto err_close_netdev;
+       }
+
+       if ( ! was_opened ) {
+               rc = net80211_prepare_probe ( dev, dev->hw->bands, 0 );
+               if ( rc < 0 )
+                       goto err_close_netdev;
+       }
+
+       ctx = net80211_probe_start ( dev, "", 0 );
+       if ( ! ctx ) {
+               rc = -ENOMEM;
+               goto err_close_netdev;
+       }
+
+       while ( ! ( rc = net80211_probe_step ( ctx ) ) ) {
+               step();
+       }
+
+       networks = net80211_probe_finish_all ( ctx );
+
+       if ( list_empty ( networks ) ) {
+               goto err_free_networks;
+       }
+
+       rc = 0;
+
+       printf ( "Networks on %s:\n\n", dev->netdev->name );
+
+       /* Output format:
+        * 0         1         2         3         4         5         6
+        * 0123456789012345678901234567890123456789012345678901234567890
+        * [Sig] SSID                  BSSID              Ch  Crypt/Auth
+        * -------------------------------------------------------------
+        * [ 15] abcdefghijklmnopqrst> 00:00:00:00:00:00  11  Open
+        *                                             ... or WPA   PSK etc.
+        */
+
+       /* Quoting the dashes and spaces verbatim uses less code space
+          than generating them programmatically. */
+       printf ( "[Sig] SSID                  BSSID              Ch  Crypt/Auth\n"
+                "-------------------------------------------------------------\n" );
+
+       list_for_each_entry ( wlan, networks, list ) {
+
+               /* Format SSID into 22-character string, space-padded,
+                  with '>' indicating truncation */
+
+               snprintf ( ssid_buf, sizeof ( ssid_buf ), "%s", wlan->essid );
+               for ( i = strlen ( ssid_buf ); i < sizeof ( ssid_buf ) - 1;
+                     i++ )
+                       ssid_buf[i] = ' ';
+               if ( ssid_buf[sizeof ( ssid_buf ) - 2] != ' ' )
+                       ssid_buf[sizeof ( ssid_buf ) - 2] = '>';
+               ssid_buf[sizeof ( ssid_buf ) - 1] = 0;
+
+               /* Sanity check */
+               if ( wlan->crypto >= NR_CRYPTO_TYPES ||
+                    wlan->handshaking >= NR_AUTH_TYPES )
+                       continue;
+
+               printf ( "[%3d] %s %s  %2d  %s%s  %s\n",
+                        wlan->signal < 0 ? 100 + wlan->signal : wlan->signal,
+                        ssid_buf, eth_ntoa ( wlan->bssid ), wlan->channel,
+                        crypto_types[wlan->crypto],
+                        wlan->crypto == NET80211_CRYPT_TKIP? " " : "",
+                        auth_types[wlan->handshaking] );
+       }
+       printf ( "\n" );
+
+ err_free_networks:
+       net80211_free_wlanlist ( networks );
+
+ err_close_netdev:
+       if ( ! was_opened ) {
+               dev->state &= ~NET80211_NO_ASSOC;
+               netdev_close ( dev->netdev );
+       } else {
+               net80211_change_channel ( dev, was_channel );
+       }
+
+       if ( ! rc )
+               return 0;
+
+ err:
+       printf ( "Scanning for networks on %s: %s\n",
+                dev->netdev->name, strerror ( rc ) );
+       return rc;
+}
+
+/**
+ * Cause reassociation of 802.11 device
+ *
+ * @v dev      802.11 device
+ */
+void iwassoc ( struct net80211_device *dev ) {
+       net80211_autoassociate ( dev );
+}
+
+/* Record error codes as though they come from the 802.11 stack */
+#undef ERRFILE
+#define ERRFILE ERRFILE_net80211
+
+/** Common 802.11 errors */
+struct errortab common_wireless_errors[] __errortab = {
+       { ECONNRESET | EUNIQ_01, "Unspecified reason" },
+       { ECONNRESET | EUNIQ_04, "Disassociated due to inactivity" },
+       { ECONNRESET | EUNIQ_0F, "4-Way Handshake timeout" },
+       { ECONNRESET | EUNIQ_17, "IEEE 802.1X authentication failed" },
+       { ECONNREFUSED | EUNIQ_01, "Unspecified failure" },
+       { ECONNREFUSED | EUNIQ_0C, "Association denied" },
+       { ECONNREFUSED | EUNIQ_0D, "Authentication method not supported" },
+};