diff options
author | Jouni Malinen <jouni.malinen@atheros.com> | 2008-11-25 11:56:28 +0200 |
---|---|---|
committer | Jouni Malinen <j@w1.fi> | 2008-11-25 11:56:28 +0200 |
commit | df73d284fbe60721ddedf565f53dae543348e306 (patch) | |
tree | 3424b3f7601868cc168a163eaa9515fc57609943 /hostapd/beacon.c | |
parent | bf01d8bc2bd202c8a2b57a951fc7a2bc39c32400 (diff) | |
download | external_wpa_supplicant_8_ti-df73d284fbe60721ddedf565f53dae543348e306.zip external_wpa_supplicant_8_ti-df73d284fbe60721ddedf565f53dae543348e306.tar.gz external_wpa_supplicant_8_ti-df73d284fbe60721ddedf565f53dae543348e306.tar.bz2 |
Added support for generating Country IE based on nl80211 regulatory info
Diffstat (limited to 'hostapd/beacon.c')
-rw-r--r-- | hostapd/beacon.c | 64 |
1 files changed, 61 insertions, 3 deletions
diff --git a/hostapd/beacon.c b/hostapd/beacon.c index a464db7..35c7028 100644 --- a/hostapd/beacon.c +++ b/hostapd/beacon.c @@ -96,12 +96,36 @@ static u8 * hostapd_eid_erp_info(struct hostapd_data *hapd, u8 *eid) } +static u8 * hostapd_eid_country_add(u8 *pos, u8 *end, int chan_spacing, + struct hostapd_channel_data *start, + struct hostapd_channel_data *prev) +{ + if (end - pos < 3) + return pos; + + /* first channel number */ + *pos++ = start->chan; + /* number of channels */ + *pos++ = (prev->chan - start->chan) / chan_spacing + 1; + /* maximum transmit power level */ + *pos++ = start->max_tx_power; + + return pos; +} + + static u8 * hostapd_eid_country(struct hostapd_data *hapd, u8 *eid, int max_len) { u8 *pos = eid; - - if (!hapd->iconf->ieee80211d || max_len < 6) + u8 *end = eid + max_len; + int i; + struct hostapd_hw_modes *mode; + struct hostapd_channel_data *start, *prev; + int chan_spacing = 1; + + if (!hapd->iconf->ieee80211d || max_len < 6 || + hapd->iface->current_mode == NULL) return eid; *pos++ = WLAN_EID_COUNTRY; @@ -109,8 +133,42 @@ static u8 * hostapd_eid_country(struct hostapd_data *hapd, u8 *eid, os_memcpy(pos, hapd->iconf->country, 3); /* e.g., 'US ' */ pos += 3; - if ((pos - eid) & 1) + mode = hapd->iface->current_mode; + if (mode->mode == HOSTAPD_MODE_IEEE80211A) + chan_spacing = 4; + + start = prev = NULL; + for (i = 0; i < mode->num_channels; i++) { + struct hostapd_channel_data *chan = &mode->channels[i]; + if (chan->flag & HOSTAPD_CHAN_DISABLED) + continue; + if (start && prev && + prev->chan + chan_spacing == chan->chan && + start->max_tx_power == chan->max_tx_power) { + prev = chan; + continue; /* can use same entry */ + } + + if (start) { + pos = hostapd_eid_country_add(pos, end, chan_spacing, + start, prev); + start = NULL; + } + + /* Start new group */ + start = prev = chan; + } + + if (start) { + pos = hostapd_eid_country_add(pos, end, chan_spacing, + start, prev); + } + + if ((pos - eid) & 1) { + if (end - pos < 1) + return eid; *pos++ = 0; /* pad for 16-bit alignment */ + } eid[1] = (pos - eid) - 2; |