Dmitry Shmidt | 1f69aa5 | 2012-01-24 16:10:04 -0800 | [diff] [blame] | 1 | /* |
| 2 | * Common driver-related functions |
| 3 | * Copyright (c) 2003-2011, Jouni Malinen <j@w1.fi> |
| 4 | * |
Dmitry Shmidt | c5ec7f5 | 2012-03-06 16:33:24 -0800 | [diff] [blame] | 5 | * This software may be distributed under the terms of the BSD license. |
| 6 | * See README for more details. |
Dmitry Shmidt | 1f69aa5 | 2012-01-24 16:10:04 -0800 | [diff] [blame] | 7 | */ |
| 8 | |
| 9 | #include "includes.h" |
| 10 | #include "utils/common.h" |
| 11 | #include "driver.h" |
| 12 | |
| 13 | void wpa_scan_results_free(struct wpa_scan_results *res) |
| 14 | { |
| 15 | size_t i; |
| 16 | |
| 17 | if (res == NULL) |
| 18 | return; |
| 19 | |
| 20 | for (i = 0; i < res->num; i++) |
| 21 | os_free(res->res[i]); |
| 22 | os_free(res->res); |
| 23 | os_free(res); |
| 24 | } |
| 25 | |
| 26 | |
| 27 | const char * event_to_string(enum wpa_event_type event) |
| 28 | { |
| 29 | #define E2S(n) case EVENT_ ## n: return #n |
| 30 | switch (event) { |
| 31 | E2S(ASSOC); |
| 32 | E2S(DISASSOC); |
| 33 | E2S(MICHAEL_MIC_FAILURE); |
| 34 | E2S(SCAN_RESULTS); |
| 35 | E2S(ASSOCINFO); |
| 36 | E2S(INTERFACE_STATUS); |
| 37 | E2S(PMKID_CANDIDATE); |
| 38 | E2S(STKSTART); |
| 39 | E2S(TDLS); |
| 40 | E2S(FT_RESPONSE); |
| 41 | E2S(IBSS_RSN_START); |
| 42 | E2S(AUTH); |
| 43 | E2S(DEAUTH); |
| 44 | E2S(ASSOC_REJECT); |
| 45 | E2S(AUTH_TIMED_OUT); |
| 46 | E2S(ASSOC_TIMED_OUT); |
Dmitry Shmidt | 1f69aa5 | 2012-01-24 16:10:04 -0800 | [diff] [blame] | 47 | E2S(WPS_BUTTON_PUSHED); |
| 48 | E2S(TX_STATUS); |
| 49 | E2S(RX_FROM_UNKNOWN); |
| 50 | E2S(RX_MGMT); |
Dmitry Shmidt | 1f69aa5 | 2012-01-24 16:10:04 -0800 | [diff] [blame] | 51 | E2S(REMAIN_ON_CHANNEL); |
| 52 | E2S(CANCEL_REMAIN_ON_CHANNEL); |
Dmitry Shmidt | 1f69aa5 | 2012-01-24 16:10:04 -0800 | [diff] [blame] | 53 | E2S(RX_PROBE_REQ); |
| 54 | E2S(NEW_STA); |
| 55 | E2S(EAPOL_RX); |
| 56 | E2S(SIGNAL_CHANGE); |
| 57 | E2S(INTERFACE_ENABLED); |
| 58 | E2S(INTERFACE_DISABLED); |
| 59 | E2S(CHANNEL_LIST_CHANGED); |
| 60 | E2S(INTERFACE_UNAVAILABLE); |
| 61 | E2S(BEST_CHANNEL); |
| 62 | E2S(UNPROT_DEAUTH); |
| 63 | E2S(UNPROT_DISASSOC); |
| 64 | E2S(STATION_LOW_ACK); |
Dmitry Shmidt | 1f69aa5 | 2012-01-24 16:10:04 -0800 | [diff] [blame] | 65 | E2S(IBSS_PEER_LOST); |
| 66 | E2S(DRIVER_GTK_REKEY); |
| 67 | E2S(SCHED_SCAN_STOPPED); |
| 68 | E2S(DRIVER_CLIENT_POLL_OK); |
| 69 | E2S(EAPOL_TX_STATUS); |
Dmitry Shmidt | 0494959 | 2012-07-19 12:16:46 -0700 | [diff] [blame] | 70 | E2S(CH_SWITCH); |
Dmitry Shmidt | 61d9df3 | 2012-08-29 16:22:06 -0700 | [diff] [blame] | 71 | E2S(WNM); |
Dmitry Shmidt | f862328 | 2013-02-20 14:34:59 -0800 | [diff] [blame] | 72 | E2S(CONNECT_FAILED_REASON); |
Dmitry Shmidt | ea69e84 | 2013-05-13 14:52:28 -0700 | [diff] [blame] | 73 | E2S(DFS_RADAR_DETECTED); |
| 74 | E2S(DFS_CAC_FINISHED); |
| 75 | E2S(DFS_CAC_ABORTED); |
| 76 | E2S(DFS_NOP_FINISHED); |
Dmitry Shmidt | b7b4d0e | 2013-08-26 12:09:05 -0700 | [diff] [blame] | 77 | E2S(SURVEY); |
Dmitry Shmidt | fb79edc | 2014-01-10 10:45:54 -0800 | [diff] [blame] | 78 | E2S(SCAN_STARTED); |
Dmitry Shmidt | cf32e60 | 2014-01-28 10:57:39 -0800 | [diff] [blame] | 79 | E2S(AVOID_FREQUENCIES); |
Dmitry Shmidt | 6c0da2b | 2015-01-05 13:08:17 -0800 | [diff] [blame] | 80 | E2S(NEW_PEER_CANDIDATE); |
| 81 | E2S(ACS_CHANNEL_SELECTED); |
Dmitry Shmidt | 7f65602 | 2015-02-25 14:36:37 -0800 | [diff] [blame] | 82 | E2S(DFS_CAC_STARTED); |
Dmitry Shmidt | 1f69aa5 | 2012-01-24 16:10:04 -0800 | [diff] [blame] | 83 | } |
| 84 | |
| 85 | return "UNKNOWN"; |
| 86 | #undef E2S |
| 87 | } |
Dmitry Shmidt | 661b4f7 | 2014-09-29 14:58:27 -0700 | [diff] [blame] | 88 | |
| 89 | |
| 90 | const char * channel_width_to_string(enum chan_width width) |
| 91 | { |
| 92 | switch (width) { |
| 93 | case CHAN_WIDTH_20_NOHT: |
| 94 | return "20 MHz (no HT)"; |
| 95 | case CHAN_WIDTH_20: |
| 96 | return "20 MHz"; |
| 97 | case CHAN_WIDTH_40: |
| 98 | return "40 MHz"; |
| 99 | case CHAN_WIDTH_80: |
| 100 | return "80 MHz"; |
| 101 | case CHAN_WIDTH_80P80: |
| 102 | return "80+80 MHz"; |
| 103 | case CHAN_WIDTH_160: |
| 104 | return "160 MHz"; |
| 105 | default: |
| 106 | return "unknown"; |
| 107 | } |
| 108 | } |
Dmitry Shmidt | 6c0da2b | 2015-01-05 13:08:17 -0800 | [diff] [blame] | 109 | |
| 110 | |
| 111 | int ht_supported(const struct hostapd_hw_modes *mode) |
| 112 | { |
| 113 | if (!(mode->flags & HOSTAPD_MODE_FLAG_HT_INFO_KNOWN)) { |
| 114 | /* |
| 115 | * The driver did not indicate whether it supports HT. Assume |
| 116 | * it does to avoid connection issues. |
| 117 | */ |
| 118 | return 1; |
| 119 | } |
| 120 | |
| 121 | /* |
| 122 | * IEEE Std 802.11n-2009 20.1.1: |
| 123 | * An HT non-AP STA shall support all EQM rates for one spatial stream. |
| 124 | */ |
| 125 | return mode->mcs_set[0] == 0xff; |
| 126 | } |
| 127 | |
| 128 | |
| 129 | int vht_supported(const struct hostapd_hw_modes *mode) |
| 130 | { |
| 131 | if (!(mode->flags & HOSTAPD_MODE_FLAG_VHT_INFO_KNOWN)) { |
| 132 | /* |
| 133 | * The driver did not indicate whether it supports VHT. Assume |
| 134 | * it does to avoid connection issues. |
| 135 | */ |
| 136 | return 1; |
| 137 | } |
| 138 | |
| 139 | /* |
| 140 | * A VHT non-AP STA shall support MCS 0-7 for one spatial stream. |
| 141 | * TODO: Verify if this complies with the standard |
| 142 | */ |
| 143 | return (mode->vht_mcs_set[0] & 0x3) != 3; |
| 144 | } |
| 145 | |
| 146 | |
| 147 | static int wpa_check_wowlan_trigger(const char *start, const char *trigger, |
| 148 | int capa_trigger, u8 *param_trigger) |
| 149 | { |
| 150 | if (os_strcmp(start, trigger) != 0) |
| 151 | return 0; |
| 152 | if (!capa_trigger) |
| 153 | return 0; |
| 154 | |
| 155 | *param_trigger = 1; |
| 156 | return 1; |
| 157 | } |
| 158 | |
| 159 | |
| 160 | struct wowlan_triggers * |
| 161 | wpa_get_wowlan_triggers(const char *wowlan_triggers, |
| 162 | const struct wpa_driver_capa *capa) |
| 163 | { |
| 164 | struct wowlan_triggers *triggers; |
| 165 | char *start, *end, *buf; |
| 166 | int last; |
| 167 | |
| 168 | if (!wowlan_triggers) |
| 169 | return NULL; |
| 170 | |
| 171 | buf = os_strdup(wowlan_triggers); |
| 172 | if (buf == NULL) |
| 173 | return NULL; |
| 174 | |
| 175 | triggers = os_zalloc(sizeof(*triggers)); |
| 176 | if (triggers == NULL) |
| 177 | goto out; |
| 178 | |
| 179 | #define CHECK_TRIGGER(trigger) \ |
| 180 | wpa_check_wowlan_trigger(start, #trigger, \ |
| 181 | capa->wowlan_triggers.trigger, \ |
| 182 | &triggers->trigger) |
| 183 | |
| 184 | start = buf; |
| 185 | while (*start != '\0') { |
Dmitry Shmidt | 57c2d39 | 2016-02-23 13:40:19 -0800 | [diff] [blame^] | 186 | while (isblank((unsigned char) *start)) |
Dmitry Shmidt | 6c0da2b | 2015-01-05 13:08:17 -0800 | [diff] [blame] | 187 | start++; |
| 188 | if (*start == '\0') |
| 189 | break; |
| 190 | end = start; |
Dmitry Shmidt | 57c2d39 | 2016-02-23 13:40:19 -0800 | [diff] [blame^] | 191 | while (!isblank((unsigned char) *end) && *end != '\0') |
Dmitry Shmidt | 6c0da2b | 2015-01-05 13:08:17 -0800 | [diff] [blame] | 192 | end++; |
| 193 | last = *end == '\0'; |
| 194 | *end = '\0'; |
| 195 | |
| 196 | if (!CHECK_TRIGGER(any) && |
| 197 | !CHECK_TRIGGER(disconnect) && |
| 198 | !CHECK_TRIGGER(magic_pkt) && |
| 199 | !CHECK_TRIGGER(gtk_rekey_failure) && |
| 200 | !CHECK_TRIGGER(eap_identity_req) && |
| 201 | !CHECK_TRIGGER(four_way_handshake) && |
| 202 | !CHECK_TRIGGER(rfkill_release)) { |
| 203 | wpa_printf(MSG_DEBUG, |
| 204 | "Unknown/unsupported wowlan trigger '%s'", |
| 205 | start); |
| 206 | os_free(triggers); |
| 207 | triggers = NULL; |
| 208 | goto out; |
| 209 | } |
| 210 | |
| 211 | if (last) |
| 212 | break; |
| 213 | start = end + 1; |
| 214 | } |
| 215 | #undef CHECK_TRIGGER |
| 216 | |
| 217 | out: |
| 218 | os_free(buf); |
| 219 | return triggers; |
| 220 | } |