mirror of
https://git.FreeBSD.org/src.git
synced 2025-01-16 15:11:52 +00:00
split Atheros SuperG support out into it's own file that's included only
with a new IEEE80211_SUPPORT_SUPERG option
This commit is contained in:
parent
d0f083bca0
commit
616190d079
Notes:
svn2git
2020-12-20 02:59:44 +00:00
svn path=/head/; revision=190391
@ -2215,6 +2215,7 @@ net80211/ieee80211_rssadapt.c optional wlan wlan_rssadapt
|
||||
net80211/ieee80211_scan.c optional wlan
|
||||
net80211/ieee80211_scan_sta.c optional wlan
|
||||
net80211/ieee80211_sta.c optional wlan
|
||||
net80211/ieee80211_superg.c optional wlan ieee80211_support_superg
|
||||
net80211/ieee80211_tdma.c optional wlan ieee80211_support_tdma
|
||||
net80211/ieee80211_wds.c optional wlan
|
||||
net80211/ieee80211_xauth.c optional wlan wlan_xauth
|
||||
|
@ -789,6 +789,7 @@ INTR_FILTER
|
||||
IEEE80211_DEBUG opt_wlan.h
|
||||
IEEE80211_DEBUG_REFCNT opt_wlan.h
|
||||
IEEE80211_AMPDU_AGE opt_wlan.h
|
||||
IEEE80211_SUPPORT_SUPERG opt_wlan.h
|
||||
IEEE80211_SUPPORT_TDMA opt_wlan.h
|
||||
|
||||
# 802.11 TDMA support
|
||||
|
@ -46,6 +46,9 @@ __FBSDID("$FreeBSD$");
|
||||
|
||||
#include <net80211/ieee80211_var.h>
|
||||
#include <net80211/ieee80211_regdomain.h>
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
#include <net80211/ieee80211_superg.h>
|
||||
#endif
|
||||
|
||||
#include <net/bpf.h>
|
||||
|
||||
@ -264,6 +267,9 @@ ieee80211_ifattach(struct ieee80211com *ic)
|
||||
ieee80211_node_attach(ic);
|
||||
ieee80211_power_attach(ic);
|
||||
ieee80211_proto_attach(ic);
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
ieee80211_superg_attach(ic);
|
||||
#endif
|
||||
ieee80211_ht_attach(ic);
|
||||
ieee80211_scan_attach(ic);
|
||||
ieee80211_regdomain_attach(ic);
|
||||
@ -306,6 +312,9 @@ ieee80211_ifdetach(struct ieee80211com *ic)
|
||||
ieee80211_sysctl_detach(ic);
|
||||
ieee80211_regdomain_detach(ic);
|
||||
ieee80211_scan_detach(ic);
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
ieee80211_superg_detach(ic);
|
||||
#endif
|
||||
ieee80211_ht_detach(ic);
|
||||
/* NB: must be called before ieee80211_node_detach */
|
||||
ieee80211_proto_detach(ic);
|
||||
@ -415,10 +424,6 @@ ieee80211_vap_setup(struct ieee80211com *ic, struct ieee80211vap *vap,
|
||||
vap->iv_flags |= IEEE80211_F_WME;
|
||||
if (vap->iv_caps & IEEE80211_C_BURST)
|
||||
vap->iv_flags |= IEEE80211_F_BURST;
|
||||
if (vap->iv_caps & IEEE80211_C_FF)
|
||||
vap->iv_flags |= IEEE80211_F_FF;
|
||||
if (vap->iv_caps & IEEE80211_C_TURBOP)
|
||||
vap->iv_flags |= IEEE80211_F_TURBOP;
|
||||
/* NB: bg scanning only makes sense for station mode right now */
|
||||
if (vap->iv_opmode == IEEE80211_M_STA &&
|
||||
(vap->iv_caps & IEEE80211_C_BGSCAN))
|
||||
@ -445,6 +450,9 @@ ieee80211_vap_setup(struct ieee80211com *ic, struct ieee80211vap *vap,
|
||||
ieee80211_node_vattach(vap);
|
||||
ieee80211_power_vattach(vap);
|
||||
ieee80211_proto_vattach(vap);
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
ieee80211_superg_vattach(vap);
|
||||
#endif
|
||||
ieee80211_ht_vattach(vap);
|
||||
ieee80211_scan_vattach(vap);
|
||||
ieee80211_regdomain_vattach(vap);
|
||||
@ -497,7 +505,9 @@ ieee80211_vap_attach(struct ieee80211vap *vap,
|
||||
IEEE80211_LOCK(ic);
|
||||
TAILQ_INSERT_TAIL(&ic->ic_vaps, vap, iv_next);
|
||||
ieee80211_syncflag_locked(ic, IEEE80211_F_WME);
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
ieee80211_syncflag_locked(ic, IEEE80211_F_TURBOP);
|
||||
#endif
|
||||
ieee80211_syncflag_locked(ic, IEEE80211_F_PCF);
|
||||
ieee80211_syncflag_locked(ic, IEEE80211_F_BURST);
|
||||
ieee80211_syncflag_ext_locked(ic, IEEE80211_FEXT_HT);
|
||||
@ -545,7 +555,9 @@ ieee80211_vap_detach(struct ieee80211vap *vap)
|
||||
|
||||
TAILQ_REMOVE(&ic->ic_vaps, vap, iv_next);
|
||||
ieee80211_syncflag_locked(ic, IEEE80211_F_WME);
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
ieee80211_syncflag_locked(ic, IEEE80211_F_TURBOP);
|
||||
#endif
|
||||
ieee80211_syncflag_locked(ic, IEEE80211_F_PCF);
|
||||
ieee80211_syncflag_locked(ic, IEEE80211_F_BURST);
|
||||
ieee80211_syncflag_ext_locked(ic, IEEE80211_FEXT_HT);
|
||||
@ -562,6 +574,9 @@ ieee80211_vap_detach(struct ieee80211vap *vap)
|
||||
|
||||
ieee80211_regdomain_vdetach(vap);
|
||||
ieee80211_scan_vdetach(vap);
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
ieee80211_superg_vdetach(vap);
|
||||
#endif
|
||||
ieee80211_ht_vdetach(vap);
|
||||
/* NB: must be before ieee80211_node_vdetach */
|
||||
ieee80211_proto_vdetach(vap);
|
||||
|
@ -57,6 +57,9 @@ __FBSDID("$FreeBSD$");
|
||||
#include <net80211/ieee80211_var.h>
|
||||
#include <net80211/ieee80211_adhoc.h>
|
||||
#include <net80211/ieee80211_input.h>
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
#include <net80211/ieee80211_superg.h>
|
||||
#endif
|
||||
#ifdef IEEE80211_SUPPORT_TDMA
|
||||
#include <net80211/ieee80211_tdma.h>
|
||||
#endif
|
||||
@ -590,32 +593,13 @@ adhoc_input(struct ieee80211_node *ni, struct mbuf *m,
|
||||
m = ieee80211_decap_amsdu(ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
} else if (IEEE80211_ATH_CAP(vap, ni, IEEE80211_NODE_FF) &&
|
||||
#define FF_LLC_SIZE (sizeof(struct ether_header) + sizeof(struct llc))
|
||||
m->m_pkthdr.len >= 3*FF_LLC_SIZE) {
|
||||
struct llc *llc;
|
||||
|
||||
/*
|
||||
* Check for fast-frame tunnel encapsulation.
|
||||
*/
|
||||
if (m->m_len < FF_LLC_SIZE &&
|
||||
(m = m_pullup(m, FF_LLC_SIZE)) == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame",
|
||||
"%s", "m_pullup(llc) failed");
|
||||
vap->iv_stats.is_rx_tooshort++;
|
||||
} else {
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
m = ieee80211_decap_fastframe(vap, ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
}
|
||||
llc = (struct llc *)(mtod(m, uint8_t *) +
|
||||
sizeof(struct ether_header));
|
||||
if (llc->llc_snap.ether_type == htons(ATH_FF_ETH_TYPE)) {
|
||||
m_adj(m, FF_LLC_SIZE);
|
||||
m = ieee80211_decap_fastframe(ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
#undef FF_LLC_SIZE
|
||||
if (dir == IEEE80211_FC1_DIR_DSTODS && ni->ni_wdsvap != NULL)
|
||||
ieee80211_deliver_data(ni->ni_wdsvap, ni, m);
|
||||
else
|
||||
|
@ -57,6 +57,9 @@ __FBSDID("$FreeBSD$");
|
||||
#include <net80211/ieee80211_var.h>
|
||||
#include <net80211/ieee80211_hostap.h>
|
||||
#include <net80211/ieee80211_input.h>
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
#include <net80211/ieee80211_superg.h>
|
||||
#endif
|
||||
#include <net80211/ieee80211_wds.h>
|
||||
|
||||
#define IEEE80211_RATE2MBS(r) (((r) & IEEE80211_RATE_VAL) / 2)
|
||||
@ -752,32 +755,13 @@ hostap_input(struct ieee80211_node *ni, struct mbuf *m,
|
||||
m = ieee80211_decap_amsdu(ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
} else if (IEEE80211_ATH_CAP(vap, ni, IEEE80211_NODE_FF) &&
|
||||
#define FF_LLC_SIZE (sizeof(struct ether_header) + sizeof(struct llc))
|
||||
m->m_pkthdr.len >= 3*FF_LLC_SIZE) {
|
||||
struct llc *llc;
|
||||
|
||||
/*
|
||||
* Check for fast-frame tunnel encapsulation.
|
||||
*/
|
||||
if (m->m_len < FF_LLC_SIZE &&
|
||||
(m = m_pullup(m, FF_LLC_SIZE)) == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame",
|
||||
"%s", "m_pullup(llc) failed");
|
||||
vap->iv_stats.is_rx_tooshort++;
|
||||
} else {
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
m = ieee80211_decap_fastframe(vap, ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
}
|
||||
llc = (struct llc *)(mtod(m, uint8_t *) +
|
||||
sizeof(struct ether_header));
|
||||
if (llc->llc_snap.ether_type == htons(ATH_FF_ETH_TYPE)) {
|
||||
m_adj(m, FF_LLC_SIZE);
|
||||
m = ieee80211_decap_fastframe(ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
#undef FF_LLC_SIZE
|
||||
if (dir == IEEE80211_FC1_DIR_DSTODS && ni->ni_wdsvap != NULL)
|
||||
ieee80211_deliver_data(ni->ni_wdsvap, ni, m);
|
||||
else
|
||||
@ -1967,8 +1951,10 @@ hostap_recv_mgmt(struct ieee80211_node *ni, struct mbuf *m0,
|
||||
wpa = frm;
|
||||
else if (iswmeinfo(frm))
|
||||
wme = frm;
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
else if (isatherosoui(frm))
|
||||
ath = frm;
|
||||
#endif
|
||||
else if (vap->iv_flags_ext & IEEE80211_FEXT_HTCOMPAT) {
|
||||
if (ishtcapoui(frm) && htcap == NULL)
|
||||
htcap = frm;
|
||||
@ -2106,6 +2092,7 @@ hostap_recv_mgmt(struct ieee80211_node *ni, struct mbuf *m0,
|
||||
ni->ni_flags |= IEEE80211_NODE_QOS;
|
||||
} else
|
||||
ni->ni_flags &= ~IEEE80211_NODE_QOS;
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
if (ath != NULL) {
|
||||
setie(ath_ie, ath - sfrm);
|
||||
/*
|
||||
@ -2113,6 +2100,7 @@ hostap_recv_mgmt(struct ieee80211_node *ni, struct mbuf *m0,
|
||||
*/
|
||||
ieee80211_parse_ath(ni, ni->ni_ies.ath_ie);
|
||||
} else
|
||||
#endif
|
||||
ni->ni_ath_flags = 0;
|
||||
#undef setie
|
||||
} else {
|
||||
|
@ -357,73 +357,6 @@ ieee80211_decap1(struct mbuf *m, int *framelen)
|
||||
#undef FF_LLC_SIZE
|
||||
}
|
||||
|
||||
/*
|
||||
* Decap the encapsulated frame pair and dispatch the first
|
||||
* for delivery. The second frame is returned for delivery
|
||||
* via the normal path.
|
||||
*/
|
||||
struct mbuf *
|
||||
ieee80211_decap_fastframe(struct ieee80211_node *ni, struct mbuf *m)
|
||||
{
|
||||
#define MS(x,f) (((x) & f) >> f##_S)
|
||||
struct ieee80211vap *vap = ni->ni_vap;
|
||||
uint32_t ath;
|
||||
struct mbuf *n;
|
||||
int framelen;
|
||||
|
||||
m_copydata(m, 0, sizeof(uint32_t), (caddr_t) &ath);
|
||||
if (MS(ath, ATH_FF_PROTO) != ATH_FF_PROTO_L2TUNNEL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame",
|
||||
"unsupport tunnel protocol, header 0x%x", ath);
|
||||
vap->iv_stats.is_ff_badhdr++;
|
||||
m_freem(m);
|
||||
return NULL;
|
||||
}
|
||||
/* NB: skip header and alignment padding */
|
||||
m_adj(m, roundup(sizeof(uint32_t) - 2, 4) + 2);
|
||||
|
||||
vap->iv_stats.is_ff_decap++;
|
||||
|
||||
/*
|
||||
* Decap the first frame, bust it apart from the
|
||||
* second and deliver; then decap the second frame
|
||||
* and return it to the caller for normal delivery.
|
||||
*/
|
||||
m = ieee80211_decap1(m, &framelen);
|
||||
if (m == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame", "%s", "first decap failed");
|
||||
vap->iv_stats.is_ff_tooshort++;
|
||||
return NULL;
|
||||
}
|
||||
n = m_split(m, framelen, M_NOWAIT);
|
||||
if (n == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame",
|
||||
"%s", "unable to split encapsulated frames");
|
||||
vap->iv_stats.is_ff_split++;
|
||||
m_freem(m); /* NB: must reclaim */
|
||||
return NULL;
|
||||
}
|
||||
/* XXX not right for WDS */
|
||||
vap->iv_deliver_data(vap, ni, m); /* 1st of pair */
|
||||
|
||||
/*
|
||||
* Decap second frame.
|
||||
*/
|
||||
m_adj(n, roundup2(framelen, 4) - framelen); /* padding */
|
||||
n = ieee80211_decap1(n, &framelen);
|
||||
if (n == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame", "%s", "second decap failed");
|
||||
vap->iv_stats.is_ff_tooshort++;
|
||||
}
|
||||
/* XXX verify framelen against mbuf contents */
|
||||
return n; /* 2nd delivered by caller */
|
||||
#undef MS
|
||||
}
|
||||
|
||||
/*
|
||||
* Install received rate set information in the node's state block.
|
||||
*/
|
||||
@ -510,16 +443,6 @@ ieee80211_alloc_challenge(struct ieee80211_node *ni)
|
||||
return (ni->ni_challenge != NULL);
|
||||
}
|
||||
|
||||
void
|
||||
ieee80211_parse_ath(struct ieee80211_node *ni, uint8_t *ie)
|
||||
{
|
||||
const struct ieee80211_ath_ie *ath =
|
||||
(const struct ieee80211_ath_ie *) ie;
|
||||
|
||||
ni->ni_ath_flags = ath->ath_capability;
|
||||
ni->ni_ath_defkeyix = LE_READ_2(&ath->ath_defkeyix);
|
||||
}
|
||||
|
||||
/*
|
||||
* Parse a Beacon or ProbeResponse frame and return the
|
||||
* useful information in an ieee80211_scanparams structure.
|
||||
@ -633,8 +556,10 @@ ieee80211_parse_beacon(struct ieee80211_node *ni, struct mbuf *m,
|
||||
scan->wpa = frm;
|
||||
else if (iswmeparam(frm) || iswmeinfo(frm))
|
||||
scan->wme = frm;
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
else if (isatherosoui(frm))
|
||||
scan->ath = frm;
|
||||
#endif
|
||||
#ifdef IEEE80211_SUPPORT_TDMA
|
||||
else if (istdmaoui(frm))
|
||||
scan->tdma = frm;
|
||||
|
@ -148,14 +148,11 @@ struct mbuf *ieee80211_defrag(struct ieee80211_node *,
|
||||
struct mbuf *, int);
|
||||
struct mbuf *ieee80211_decap(struct ieee80211vap *, struct mbuf *, int);
|
||||
struct mbuf *ieee80211_decap1(struct mbuf *, int *);
|
||||
struct mbuf *ieee80211_decap_fastframe(struct ieee80211_node *,
|
||||
struct mbuf *);
|
||||
int ieee80211_setup_rates(struct ieee80211_node *ni,
|
||||
const uint8_t *rates, const uint8_t *xrates, int flags);
|
||||
void ieee80211_send_error(struct ieee80211_node *,
|
||||
const uint8_t mac[IEEE80211_ADDR_LEN], int subtype, int arg);
|
||||
int ieee80211_alloc_challenge(struct ieee80211_node *);
|
||||
void ieee80211_parse_ath(struct ieee80211_node *, uint8_t *);
|
||||
int ieee80211_parse_beacon(struct ieee80211_node *, struct mbuf *,
|
||||
struct ieee80211_scanparams *);
|
||||
int ieee80211_parse_action(struct ieee80211_node *, struct mbuf *);
|
||||
|
@ -584,21 +584,6 @@ ieee80211_ioctl_getmaccmd(struct ieee80211vap *vap, struct ieee80211req *ireq)
|
||||
return (acl == NULL ? EINVAL : acl->iac_getioctl(vap, ireq));
|
||||
}
|
||||
|
||||
/*
|
||||
* Return the current ``state'' of an Atheros capbility.
|
||||
* If associated in station mode report the negotiated
|
||||
* setting. Otherwise report the current setting.
|
||||
*/
|
||||
static int
|
||||
getathcap(struct ieee80211vap *vap, int cap)
|
||||
{
|
||||
if (vap->iv_opmode == IEEE80211_M_STA &&
|
||||
vap->iv_state == IEEE80211_S_RUN)
|
||||
return IEEE80211_ATH_CAP(vap, vap->iv_bss, cap) != 0;
|
||||
else
|
||||
return (vap->iv_flags & cap) != 0;
|
||||
}
|
||||
|
||||
static __noinline int
|
||||
ieee80211_ioctl_getcurchan(struct ieee80211vap *vap, struct ieee80211req *ireq)
|
||||
{
|
||||
@ -974,12 +959,6 @@ ieee80211_ioctl_get80211(struct ieee80211vap *vap, u_long cmd,
|
||||
case IEEE80211_IOC_PUREG:
|
||||
ireq->i_val = (vap->iv_flags & IEEE80211_F_PUREG) != 0;
|
||||
break;
|
||||
case IEEE80211_IOC_FF:
|
||||
ireq->i_val = getathcap(vap, IEEE80211_F_FF);
|
||||
break;
|
||||
case IEEE80211_IOC_TURBOP:
|
||||
ireq->i_val = getathcap(vap, IEEE80211_F_TURBOP);
|
||||
break;
|
||||
case IEEE80211_IOC_BGSCAN:
|
||||
ireq->i_val = (vap->iv_flags & IEEE80211_F_BGSCAN) != 0;
|
||||
break;
|
||||
@ -2889,24 +2868,6 @@ ieee80211_ioctl_set80211(struct ieee80211vap *vap, u_long cmd, struct ieee80211r
|
||||
if (isvap11g(vap))
|
||||
error = ENETRESET;
|
||||
break;
|
||||
case IEEE80211_IOC_FF:
|
||||
if (ireq->i_val) {
|
||||
if ((vap->iv_caps & IEEE80211_C_FF) == 0)
|
||||
return EOPNOTSUPP;
|
||||
vap->iv_flags |= IEEE80211_F_FF;
|
||||
} else
|
||||
vap->iv_flags &= ~IEEE80211_F_FF;
|
||||
error = ERESTART;
|
||||
break;
|
||||
case IEEE80211_IOC_TURBOP:
|
||||
if (ireq->i_val) {
|
||||
if ((vap->iv_caps & IEEE80211_C_TURBOP) == 0)
|
||||
return EOPNOTSUPP;
|
||||
vap->iv_flags |= IEEE80211_F_TURBOP;
|
||||
} else
|
||||
vap->iv_flags &= ~IEEE80211_F_TURBOP;
|
||||
error = ENETRESET;
|
||||
break;
|
||||
case IEEE80211_IOC_BGSCAN:
|
||||
if (ireq->i_val) {
|
||||
if ((vap->iv_caps & IEEE80211_C_BGSCAN) == 0)
|
||||
|
@ -43,6 +43,9 @@ __FBSDID("$FreeBSD$");
|
||||
|
||||
#include <net80211/ieee80211_var.h>
|
||||
#include <net80211/ieee80211_input.h>
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
#include <net80211/ieee80211_superg.h>
|
||||
#endif
|
||||
#ifdef IEEE80211_SUPPORT_TDMA
|
||||
#include <net80211/ieee80211_tdma.h>
|
||||
#endif
|
||||
@ -754,8 +757,10 @@ ieee80211_sta_join(struct ieee80211vap *vap, struct ieee80211_channel *chan,
|
||||
|
||||
if (ieee80211_ies_init(&ni->ni_ies, se->se_ies.data, se->se_ies.len)) {
|
||||
ieee80211_ies_expand(&ni->ni_ies);
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
if (ni->ni_ies.ath_ie != NULL)
|
||||
ieee80211_parse_ath(ni, ni->ni_ies.ath_ie);
|
||||
#endif
|
||||
if (ni->ni_ies.htcap_ie != NULL)
|
||||
ieee80211_parse_htcap(ni, ni->ni_ies.htcap_ie);
|
||||
if (ni->ni_ies.htinfo_ie != NULL)
|
||||
@ -875,8 +880,10 @@ ieee80211_ies_expand(struct ieee80211_ies *ies)
|
||||
ies->wpa_ie = ie;
|
||||
else if (iswmeoui(ie))
|
||||
ies->wme_ie = ie;
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
else if (isatherosoui(ie))
|
||||
ies->ath_ie = ie;
|
||||
#endif
|
||||
#ifdef IEEE80211_SUPPORT_TDMA
|
||||
else if (istdmaoui(ie))
|
||||
ies->tdma_ie = ie;
|
||||
@ -1173,8 +1180,10 @@ ieee80211_node_create_wds(struct ieee80211vap *vap,
|
||||
*/
|
||||
if (vap->iv_flags & IEEE80211_F_WME)
|
||||
ni->ni_flags |= IEEE80211_NODE_QOS;
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
if (vap->iv_flags & IEEE80211_F_FF)
|
||||
ni->ni_flags |= IEEE80211_NODE_FF;
|
||||
#endif
|
||||
if ((ic->ic_htcaps & IEEE80211_HTC_HT) &&
|
||||
(vap->iv_flags_ext & IEEE80211_FEXT_HT)) {
|
||||
/*
|
||||
@ -1331,8 +1340,10 @@ ieee80211_fakeup_adhoc_node(struct ieee80211vap *vap,
|
||||
*/
|
||||
if (vap->iv_flags & IEEE80211_F_WME)
|
||||
ni->ni_flags |= IEEE80211_NODE_QOS;
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
if (vap->iv_flags & IEEE80211_F_FF)
|
||||
ni->ni_flags |= IEEE80211_NODE_FF;
|
||||
#endif
|
||||
}
|
||||
node_setuptxparms(ni);
|
||||
if (ic->ic_newassoc != NULL)
|
||||
@ -1366,8 +1377,10 @@ ieee80211_init_neighbor(struct ieee80211_node *ni,
|
||||
ni->ni_flags |= IEEE80211_NODE_QOS;
|
||||
else
|
||||
ni->ni_flags &= ~IEEE80211_NODE_QOS;
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
if (ni->ni_ies.ath_ie != NULL)
|
||||
ieee80211_parse_ath(ni, ni->ni_ies.ath_ie);
|
||||
#endif
|
||||
}
|
||||
|
||||
/* NB: must be after ni_chan is setup */
|
||||
|
@ -47,6 +47,9 @@ __FBSDID("$FreeBSD$");
|
||||
|
||||
#include <net80211/ieee80211_var.h>
|
||||
#include <net80211/ieee80211_regdomain.h>
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
#include <net80211/ieee80211_superg.h>
|
||||
#endif
|
||||
#ifdef IEEE80211_SUPPORT_TDMA
|
||||
#include <net80211/ieee80211_tdma.h>
|
||||
#endif
|
||||
@ -62,9 +65,6 @@ __FBSDID("$FreeBSD$");
|
||||
#define ETHER_HEADER_COPY(dst, src) \
|
||||
memcpy(dst, src, sizeof(struct ether_header))
|
||||
|
||||
static struct mbuf *ieee80211_encap_fastframe(struct ieee80211vap *,
|
||||
struct mbuf *m1, const struct ether_header *eh1,
|
||||
struct mbuf *m2, const struct ether_header *eh2);
|
||||
static int ieee80211_fragment(struct ieee80211vap *, struct mbuf *,
|
||||
u_int hdrsize, u_int ciphdrsize, u_int mtu);
|
||||
static void ieee80211_tx_mgt_cb(struct ieee80211_node *, void *, int);
|
||||
@ -731,7 +731,7 @@ ieee80211_classify(struct ieee80211_node *ni, struct mbuf *m)
|
||||
* Drivers and cipher modules assume we have done the necessary work
|
||||
* and fail rudely if they don't find the space they need.
|
||||
*/
|
||||
static struct mbuf *
|
||||
struct mbuf *
|
||||
ieee80211_mbuf_adjust(struct ieee80211vap *vap, int hdrsize,
|
||||
struct ieee80211_key *key, struct mbuf *m)
|
||||
{
|
||||
@ -861,7 +861,7 @@ ieee80211_encap(struct ieee80211_node *ni, struct mbuf *m)
|
||||
struct ieee80211_frame *wh;
|
||||
struct ieee80211_key *key;
|
||||
struct llc *llc;
|
||||
int hdrsize, hdrspace, datalen, addqos, txfrag, isff, is4addr;
|
||||
int hdrsize, hdrspace, datalen, addqos, txfrag, is4addr;
|
||||
|
||||
/*
|
||||
* Copy existing Ethernet header to a safe place. The
|
||||
@ -935,56 +935,7 @@ ieee80211_encap(struct ieee80211_node *ni, struct mbuf *m)
|
||||
else
|
||||
hdrspace = hdrsize;
|
||||
|
||||
if ((isff = m->m_flags & M_FF) != 0) {
|
||||
struct mbuf *m2;
|
||||
struct ether_header eh2;
|
||||
|
||||
/*
|
||||
* Fast frame encapsulation. There must be two packets
|
||||
* chained with m_nextpkt. We do header adjustment for
|
||||
* each, add the tunnel encapsulation, and then concatenate
|
||||
* the mbuf chains to form a single frame for transmission.
|
||||
*/
|
||||
m2 = m->m_nextpkt;
|
||||
if (m2 == NULL) {
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: only one frame\n", __func__);
|
||||
goto bad;
|
||||
}
|
||||
m->m_nextpkt = NULL;
|
||||
/*
|
||||
* Include fast frame headers in adjusting header
|
||||
* layout; this allocates space according to what
|
||||
* ieee80211_encap_fastframe will do.
|
||||
*/
|
||||
m = ieee80211_mbuf_adjust(vap,
|
||||
hdrspace + sizeof(struct llc) + sizeof(uint32_t) + 2 +
|
||||
sizeof(struct ether_header),
|
||||
key, m);
|
||||
if (m == NULL) {
|
||||
/* NB: ieee80211_mbuf_adjust handles msgs+statistics */
|
||||
m_freem(m2);
|
||||
goto bad;
|
||||
}
|
||||
/*
|
||||
* Copy second frame's Ethernet header out of line
|
||||
* and adjust for encapsulation headers. Note that
|
||||
* we make room for padding in case there isn't room
|
||||
* at the end of first frame.
|
||||
*/
|
||||
KASSERT(m2->m_len >= sizeof(eh2), ("no ethernet header!"));
|
||||
ETHER_HEADER_COPY(&eh2, mtod(m2, caddr_t));
|
||||
m2 = ieee80211_mbuf_adjust(vap,
|
||||
ATH_FF_MAX_HDR_PAD + sizeof(struct ether_header),
|
||||
NULL, m2);
|
||||
if (m2 == NULL) {
|
||||
/* NB: ieee80211_mbuf_adjust handles msgs+statistics */
|
||||
goto bad;
|
||||
}
|
||||
m = ieee80211_encap_fastframe(vap, m, &eh, m2, &eh2);
|
||||
if (m == NULL)
|
||||
goto bad;
|
||||
} else {
|
||||
if (__predict_true((m->m_flags & M_FF) == 0)) {
|
||||
/*
|
||||
* Normal frame.
|
||||
*/
|
||||
@ -1002,6 +953,12 @@ ieee80211_encap(struct ieee80211_node *ni, struct mbuf *m)
|
||||
llc->llc_snap.org_code[1] = 0;
|
||||
llc->llc_snap.org_code[2] = 0;
|
||||
llc->llc_snap.ether_type = eh.ether_type;
|
||||
} else {
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
m = ieee80211_ff_encap(vap, m, hdrspace, key);
|
||||
if (m == NULL)
|
||||
#endif
|
||||
goto bad;
|
||||
}
|
||||
datalen = m->m_pkthdr.len; /* NB: w/o 802.11 header */
|
||||
|
||||
@ -1047,7 +1004,7 @@ ieee80211_encap(struct ieee80211_node *ni, struct mbuf *m)
|
||||
case IEEE80211_M_WDS: /* NB: is4addr should always be true */
|
||||
goto bad;
|
||||
}
|
||||
if (m->m_flags & M_MORE_DATA)
|
||||
if (m->m_flags & M_MORE_DATA)
|
||||
wh->i_fc[1] |= IEEE80211_FC1_MORE_DATA;
|
||||
if (addqos) {
|
||||
uint8_t *qos;
|
||||
@ -1128,7 +1085,7 @@ ieee80211_encap(struct ieee80211_node *ni, struct mbuf *m)
|
||||
txfrag = (m->m_pkthdr.len > vap->iv_fragthreshold &&
|
||||
!IEEE80211_IS_MULTICAST(wh->i_addr1) &&
|
||||
(vap->iv_caps & IEEE80211_C_TXFRAG) &&
|
||||
!isff); /* NB: don't fragment ff's */
|
||||
(m->m_flags & M_FF) == 0); /* NB: don't fragment ff's */
|
||||
if (key != NULL) {
|
||||
/*
|
||||
* IEEE 802.1X: send EAPOL frames always in the clear.
|
||||
@ -1174,135 +1131,6 @@ ieee80211_encap(struct ieee80211_node *ni, struct mbuf *m)
|
||||
#undef WH4
|
||||
}
|
||||
|
||||
/*
|
||||
* Do Ethernet-LLC encapsulation for each payload in a fast frame
|
||||
* tunnel encapsulation. The frame is assumed to have an Ethernet
|
||||
* header at the front that must be stripped before prepending the
|
||||
* LLC followed by the Ethernet header passed in (with an Ethernet
|
||||
* type that specifies the payload size).
|
||||
*/
|
||||
static struct mbuf *
|
||||
ieee80211_encap1(struct ieee80211vap *vap, struct mbuf *m,
|
||||
const struct ether_header *eh)
|
||||
{
|
||||
struct llc *llc;
|
||||
uint16_t payload;
|
||||
|
||||
/* XXX optimize by combining m_adj+M_PREPEND */
|
||||
m_adj(m, sizeof(struct ether_header) - sizeof(struct llc));
|
||||
llc = mtod(m, struct llc *);
|
||||
llc->llc_dsap = llc->llc_ssap = LLC_SNAP_LSAP;
|
||||
llc->llc_control = LLC_UI;
|
||||
llc->llc_snap.org_code[0] = 0;
|
||||
llc->llc_snap.org_code[1] = 0;
|
||||
llc->llc_snap.org_code[2] = 0;
|
||||
llc->llc_snap.ether_type = eh->ether_type;
|
||||
payload = m->m_pkthdr.len; /* NB: w/o Ethernet header */
|
||||
|
||||
M_PREPEND(m, sizeof(struct ether_header), M_DONTWAIT);
|
||||
if (m == NULL) { /* XXX cannot happen */
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: no space for ether_header\n", __func__);
|
||||
vap->iv_stats.is_tx_nobuf++;
|
||||
return NULL;
|
||||
}
|
||||
ETHER_HEADER_COPY(mtod(m, void *), eh);
|
||||
mtod(m, struct ether_header *)->ether_type = htons(payload);
|
||||
return m;
|
||||
}
|
||||
|
||||
/*
|
||||
* Do fast frame tunnel encapsulation. The two frames and
|
||||
* Ethernet headers are supplied. The caller is assumed to
|
||||
* have arrange for space in the mbuf chains for encapsulating
|
||||
* headers (to avoid major mbuf fragmentation).
|
||||
*
|
||||
* The encapsulated frame is returned or NULL if there is a
|
||||
* problem (should not happen).
|
||||
*/
|
||||
static struct mbuf *
|
||||
ieee80211_encap_fastframe(struct ieee80211vap *vap,
|
||||
struct mbuf *m1, const struct ether_header *eh1,
|
||||
struct mbuf *m2, const struct ether_header *eh2)
|
||||
{
|
||||
struct llc *llc;
|
||||
struct mbuf *m;
|
||||
int pad;
|
||||
|
||||
/*
|
||||
* First, each frame gets a standard encapsulation.
|
||||
*/
|
||||
m1 = ieee80211_encap1(vap, m1, eh1);
|
||||
if (m1 == NULL) {
|
||||
m_freem(m2);
|
||||
return NULL;
|
||||
}
|
||||
m2 = ieee80211_encap1(vap, m2, eh2);
|
||||
if (m2 == NULL) {
|
||||
m_freem(m1);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/*
|
||||
* Pad leading frame to a 4-byte boundary. If there
|
||||
* is space at the end of the first frame, put it
|
||||
* there; otherwise prepend to the front of the second
|
||||
* frame. We know doing the second will always work
|
||||
* because we reserve space above. We prefer appending
|
||||
* as this typically has better DMA alignment properties.
|
||||
*/
|
||||
for (m = m1; m->m_next != NULL; m = m->m_next)
|
||||
;
|
||||
pad = roundup2(m1->m_pkthdr.len, 4) - m1->m_pkthdr.len;
|
||||
if (pad) {
|
||||
if (M_TRAILINGSPACE(m) < pad) { /* prepend to second */
|
||||
m2->m_data -= pad;
|
||||
m2->m_len += pad;
|
||||
m2->m_pkthdr.len += pad;
|
||||
} else { /* append to first */
|
||||
m->m_len += pad;
|
||||
m1->m_pkthdr.len += pad;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Now, stick 'em together and prepend the tunnel headers;
|
||||
* first the Atheros tunnel header (all zero for now) and
|
||||
* then a special fast frame LLC.
|
||||
*
|
||||
* XXX optimize by prepending together
|
||||
*/
|
||||
m->m_next = m2; /* NB: last mbuf from above */
|
||||
m1->m_pkthdr.len += m2->m_pkthdr.len;
|
||||
M_PREPEND(m1, sizeof(uint32_t)+2, M_DONTWAIT);
|
||||
if (m1 == NULL) { /* XXX cannot happen */
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: no space for tunnel header\n", __func__);
|
||||
vap->iv_stats.is_tx_nobuf++;
|
||||
return NULL;
|
||||
}
|
||||
memset(mtod(m1, void *), 0, sizeof(uint32_t)+2);
|
||||
|
||||
M_PREPEND(m1, sizeof(struct llc), M_DONTWAIT);
|
||||
if (m1 == NULL) { /* XXX cannot happen */
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: no space for llc header\n", __func__);
|
||||
vap->iv_stats.is_tx_nobuf++;
|
||||
return NULL;
|
||||
}
|
||||
llc = mtod(m1, struct llc *);
|
||||
llc->llc_dsap = llc->llc_ssap = LLC_SNAP_LSAP;
|
||||
llc->llc_control = LLC_UI;
|
||||
llc->llc_snap.org_code[0] = ATH_FF_SNAP_ORGCODE_0;
|
||||
llc->llc_snap.org_code[1] = ATH_FF_SNAP_ORGCODE_1;
|
||||
llc->llc_snap.org_code[2] = ATH_FF_SNAP_ORGCODE_2;
|
||||
llc->llc_snap.ether_type = htons(ATH_FF_ETH_TYPE);
|
||||
|
||||
vap->iv_stats.is_ff_encap++;
|
||||
|
||||
return m1;
|
||||
}
|
||||
|
||||
/*
|
||||
* Fragment the frame according to the specified mtu.
|
||||
* The size of the 802.11 header (w/o padding) is provided
|
||||
@ -1568,31 +1396,6 @@ ieee80211_add_wme_param(uint8_t *frm, struct ieee80211_wme_state *wme)
|
||||
}
|
||||
#undef WME_OUI_BYTES
|
||||
|
||||
#define ATH_OUI_BYTES 0x00, 0x03, 0x7f
|
||||
/*
|
||||
* Add a WME information element to a frame.
|
||||
*/
|
||||
static uint8_t *
|
||||
ieee80211_add_ath(uint8_t *frm, uint8_t caps, uint16_t defkeyix)
|
||||
{
|
||||
static const struct ieee80211_ath_ie info = {
|
||||
.ath_id = IEEE80211_ELEMID_VENDOR,
|
||||
.ath_len = sizeof(struct ieee80211_ath_ie) - 2,
|
||||
.ath_oui = { ATH_OUI_BYTES },
|
||||
.ath_oui_type = ATH_OUI_TYPE,
|
||||
.ath_oui_subtype= ATH_OUI_SUBTYPE,
|
||||
.ath_version = ATH_OUI_VERSION,
|
||||
};
|
||||
struct ieee80211_ath_ie *ath = (struct ieee80211_ath_ie *) frm;
|
||||
|
||||
memcpy(frm, &info, sizeof(info));
|
||||
ath->ath_capability = caps;
|
||||
ath->ath_defkeyix[0] = (defkeyix & 0xff);
|
||||
ath->ath_defkeyix[1] = ((defkeyix >> 8) & 0xff);
|
||||
return frm + sizeof(info);
|
||||
}
|
||||
#undef ATH_OUI_BYTES
|
||||
|
||||
/*
|
||||
* Add an 11h Power Constraint element to a frame.
|
||||
*/
|
||||
@ -1977,7 +1780,9 @@ ieee80211_send_mgmt(struct ieee80211_node *ni, int type, int arg)
|
||||
+ sizeof(struct ieee80211_wme_info)
|
||||
+ sizeof(struct ieee80211_ie_htcap)
|
||||
+ 4 + sizeof(struct ieee80211_ie_htcap)
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
+ sizeof(struct ieee80211_ath_ie)
|
||||
#endif
|
||||
+ (vap->iv_appie_wpa != NULL ?
|
||||
vap->iv_appie_wpa->ie_len : 0)
|
||||
+ (vap->iv_appie_assocreq != NULL ?
|
||||
@ -2046,6 +1851,7 @@ ieee80211_send_mgmt(struct ieee80211_node *ni, int type, int arg)
|
||||
ni->ni_ies.htcap_ie != NULL &&
|
||||
ni->ni_ies.htcap_ie[0] == IEEE80211_ELEMID_VENDOR)
|
||||
frm = ieee80211_add_htcap_vendor(frm, ni);
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
if (IEEE80211_ATH_CAP(vap, ni, IEEE80211_F_ATHEROS))
|
||||
frm = ieee80211_add_ath(frm,
|
||||
IEEE80211_ATH_CAP(vap, ni, IEEE80211_F_ATHEROS),
|
||||
@ -2053,6 +1859,7 @@ ieee80211_send_mgmt(struct ieee80211_node *ni, int type, int arg)
|
||||
ni->ni_authmode != IEEE80211_AUTH_8021X &&
|
||||
vap->iv_def_txkey != IEEE80211_KEYIX_NONE ?
|
||||
vap->iv_def_txkey : 0x7fff);
|
||||
#endif /* IEEE80211_SUPPORT_SUPERG */
|
||||
if (vap->iv_appie_assocreq != NULL)
|
||||
frm = add_appie(frm, vap->iv_appie_assocreq);
|
||||
m->m_pkthdr.len = m->m_len = frm - mtod(m, uint8_t *);
|
||||
@ -2088,7 +1895,9 @@ ieee80211_send_mgmt(struct ieee80211_node *ni, int type, int arg)
|
||||
+ sizeof(struct ieee80211_ie_htcap) + 4
|
||||
+ sizeof(struct ieee80211_ie_htinfo) + 4
|
||||
+ sizeof(struct ieee80211_wme_param)
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
+ sizeof(struct ieee80211_ath_ie)
|
||||
#endif
|
||||
+ (vap->iv_appie_assocresp != NULL ?
|
||||
vap->iv_appie_assocresp->ie_len : 0)
|
||||
);
|
||||
@ -2123,10 +1932,12 @@ ieee80211_send_mgmt(struct ieee80211_node *ni, int type, int arg)
|
||||
frm = ieee80211_add_htcap_vendor(frm, ni);
|
||||
frm = ieee80211_add_htinfo_vendor(frm, ni);
|
||||
}
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
if (IEEE80211_ATH_CAP(vap, ni, IEEE80211_F_ATHEROS))
|
||||
frm = ieee80211_add_ath(frm,
|
||||
IEEE80211_ATH_CAP(vap, ni, IEEE80211_F_ATHEROS),
|
||||
ni->ni_ath_defkeyix);
|
||||
#endif /* IEEE80211_SUPPORT_SUPERG */
|
||||
if (vap->iv_appie_assocresp != NULL)
|
||||
frm = add_appie(frm, vap->iv_appie_assocresp);
|
||||
m->m_pkthdr.len = m->m_len = frm - mtod(m, uint8_t *);
|
||||
@ -2227,7 +2038,9 @@ ieee80211_alloc_proberesp(struct ieee80211_node *bss, int legacy)
|
||||
+ sizeof(struct ieee80211_wme_param)
|
||||
+ 4 + sizeof(struct ieee80211_ie_htcap)
|
||||
+ 4 + sizeof(struct ieee80211_ie_htinfo)
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
+ sizeof(struct ieee80211_ath_ie)
|
||||
#endif
|
||||
+ (vap->iv_appie_proberesp != NULL ?
|
||||
vap->iv_appie_proberesp->ie_len : 0)
|
||||
);
|
||||
@ -2310,9 +2123,11 @@ ieee80211_alloc_proberesp(struct ieee80211_node *bss, int legacy)
|
||||
frm = ieee80211_add_htcap_vendor(frm, bss);
|
||||
frm = ieee80211_add_htinfo_vendor(frm, bss);
|
||||
}
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
if (bss->ni_ies.ath_ie != NULL && legacy != IEEE80211_SEND_LEGACY_11B)
|
||||
frm = ieee80211_add_ath(frm, bss->ni_ath_flags,
|
||||
bss->ni_ath_defkeyix);
|
||||
#endif
|
||||
if (vap->iv_appie_proberesp != NULL)
|
||||
frm = add_appie(frm, vap->iv_appie_proberesp);
|
||||
m->m_pkthdr.len = m->m_len = frm - mtod(m, uint8_t *);
|
||||
|
@ -1316,39 +1316,6 @@ ieee80211_resume_all(struct ieee80211com *ic)
|
||||
IEEE80211_UNLOCK(ic);
|
||||
}
|
||||
|
||||
/*
|
||||
* Switch between turbo and non-turbo operating modes.
|
||||
* Use the specified channel flags to locate the new
|
||||
* channel, update 802.11 state, and then call back into
|
||||
* the driver to effect the change.
|
||||
*/
|
||||
void
|
||||
ieee80211_dturbo_switch(struct ieee80211vap *vap, int newflags)
|
||||
{
|
||||
struct ieee80211com *ic = vap->iv_ic;
|
||||
struct ieee80211_channel *chan;
|
||||
|
||||
chan = ieee80211_find_channel(ic, ic->ic_bsschan->ic_freq, newflags);
|
||||
if (chan == NULL) { /* XXX should not happen */
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: no channel with freq %u flags 0x%x\n",
|
||||
__func__, ic->ic_bsschan->ic_freq, newflags);
|
||||
return;
|
||||
}
|
||||
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: %s -> %s (freq %u flags 0x%x)\n", __func__,
|
||||
ieee80211_phymode_name[ieee80211_chan2mode(ic->ic_bsschan)],
|
||||
ieee80211_phymode_name[ieee80211_chan2mode(chan)],
|
||||
chan->ic_freq, chan->ic_flags);
|
||||
|
||||
ic->ic_bsschan = chan;
|
||||
ic->ic_prevchan = ic->ic_curchan;
|
||||
ic->ic_curchan = chan;
|
||||
ic->ic_set_channel(ic);
|
||||
/* NB: do not need to reset ERP state 'cuz we're in sta mode */
|
||||
}
|
||||
|
||||
void
|
||||
ieee80211_beacon_miss(struct ieee80211com *ic)
|
||||
{
|
||||
|
@ -74,6 +74,8 @@ int ieee80211_output(struct ifnet *, struct mbuf *,
|
||||
void ieee80211_start(struct ifnet *);
|
||||
int ieee80211_send_nulldata(struct ieee80211_node *);
|
||||
int ieee80211_classify(struct ieee80211_node *, struct mbuf *m);
|
||||
struct mbuf *ieee80211_mbuf_adjust(struct ieee80211vap *, int,
|
||||
struct ieee80211_key *, struct mbuf *);
|
||||
struct mbuf *ieee80211_encap(struct ieee80211_node *, struct mbuf *);
|
||||
int ieee80211_send_mgmt(struct ieee80211_node *, int, int);
|
||||
struct ieee80211_appie;
|
||||
|
@ -57,6 +57,9 @@ __FBSDID("$FreeBSD$");
|
||||
#include <net80211/ieee80211_var.h>
|
||||
#include <net80211/ieee80211_sta.h>
|
||||
#include <net80211/ieee80211_input.h>
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
#include <net80211/ieee80211_superg.h>
|
||||
#endif
|
||||
|
||||
#define IEEE80211_RATE2MBS(r) (((r) & IEEE80211_RATE_VAL) / 2)
|
||||
|
||||
@ -129,6 +132,7 @@ sta_beacon_miss(struct ieee80211vap *vap)
|
||||
vap->iv_bmiss_count = 0;
|
||||
vap->iv_stats.is_beacon_miss++;
|
||||
if (vap->iv_roaming == IEEE80211_ROAMING_AUTO) {
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
/*
|
||||
* If we receive a beacon miss interrupt when using
|
||||
* dynamic turbo, attempt to switch modes before
|
||||
@ -137,6 +141,7 @@ sta_beacon_miss(struct ieee80211vap *vap)
|
||||
if (IEEE80211_ATH_CAP(vap, vap->iv_bss, IEEE80211_NODE_TURBOP))
|
||||
ieee80211_dturbo_switch(vap,
|
||||
ic->ic_bsschan->ic_flags ^ IEEE80211_CHAN_TURBO);
|
||||
#endif
|
||||
/*
|
||||
* Try to reassociate before scanning for a new ap.
|
||||
*/
|
||||
@ -795,32 +800,13 @@ sta_input(struct ieee80211_node *ni, struct mbuf *m,
|
||||
m = ieee80211_decap_amsdu(ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
} else if (IEEE80211_ATH_CAP(vap, ni, IEEE80211_NODE_FF) &&
|
||||
#define FF_LLC_SIZE (sizeof(struct ether_header) + sizeof(struct llc))
|
||||
m->m_pkthdr.len >= 3*FF_LLC_SIZE) {
|
||||
struct llc *llc;
|
||||
|
||||
/*
|
||||
* Check for fast-frame tunnel encapsulation.
|
||||
*/
|
||||
if (m->m_len < FF_LLC_SIZE &&
|
||||
(m = m_pullup(m, FF_LLC_SIZE)) == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame",
|
||||
"%s", "m_pullup(llc) failed");
|
||||
vap->iv_stats.is_rx_tooshort++;
|
||||
} else {
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
m = ieee80211_decap_fastframe(vap, ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
}
|
||||
llc = (struct llc *)(mtod(m, uint8_t *) +
|
||||
sizeof(struct ether_header));
|
||||
if (llc->llc_snap.ether_type == htons(ATH_FF_ETH_TYPE)) {
|
||||
m_adj(m, FF_LLC_SIZE);
|
||||
m = ieee80211_decap_fastframe(ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
#undef FF_LLC_SIZE
|
||||
ieee80211_deliver_data(vap, ni, m);
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
|
||||
@ -1094,51 +1080,6 @@ ieee80211_parse_wmeparams(struct ieee80211vap *vap, uint8_t *frm,
|
||||
#undef MS
|
||||
}
|
||||
|
||||
static int
|
||||
ieee80211_parse_athparams(struct ieee80211_node *ni, uint8_t *frm,
|
||||
const struct ieee80211_frame *wh)
|
||||
{
|
||||
struct ieee80211vap *vap = ni->ni_vap;
|
||||
const struct ieee80211_ath_ie *ath;
|
||||
u_int len = frm[1];
|
||||
int capschanged;
|
||||
uint16_t defkeyix;
|
||||
|
||||
if (len < sizeof(struct ieee80211_ath_ie)-2) {
|
||||
IEEE80211_DISCARD_IE(vap,
|
||||
IEEE80211_MSG_ELEMID | IEEE80211_MSG_SUPERG,
|
||||
wh, "Atheros", "too short, len %u", len);
|
||||
return -1;
|
||||
}
|
||||
ath = (const struct ieee80211_ath_ie *)frm;
|
||||
capschanged = (ni->ni_ath_flags != ath->ath_capability);
|
||||
defkeyix = LE_READ_2(ath->ath_defkeyix);
|
||||
if (capschanged || defkeyix != ni->ni_ath_defkeyix) {
|
||||
ni->ni_ath_flags = ath->ath_capability;
|
||||
ni->ni_ath_defkeyix = defkeyix;
|
||||
IEEE80211_NOTE(vap, IEEE80211_MSG_SUPERG, ni,
|
||||
"ath ie change: new caps 0x%x defkeyix 0x%x",
|
||||
ni->ni_ath_flags, ni->ni_ath_defkeyix);
|
||||
}
|
||||
if (IEEE80211_ATH_CAP(vap, ni, ATHEROS_CAP_TURBO_PRIME)) {
|
||||
uint16_t curflags, newflags;
|
||||
|
||||
/*
|
||||
* Check for turbo mode switch. Calculate flags
|
||||
* for the new mode and effect the switch.
|
||||
*/
|
||||
newflags = curflags = vap->iv_ic->ic_bsschan->ic_flags;
|
||||
/* NB: BOOST is not in ic_flags, so get it from the ie */
|
||||
if (ath->ath_capability & ATHEROS_CAP_BOOST)
|
||||
newflags |= IEEE80211_CHAN_TURBO;
|
||||
else
|
||||
newflags &= ~IEEE80211_CHAN_TURBO;
|
||||
if (newflags != curflags)
|
||||
ieee80211_dturbo_switch(vap, newflags);
|
||||
}
|
||||
return capschanged;
|
||||
}
|
||||
|
||||
/*
|
||||
* Return non-zero if a background scan may be continued:
|
||||
* o bg scan is active
|
||||
@ -1176,7 +1117,9 @@ startbgscan(struct ieee80211vap *vap)
|
||||
|
||||
return ((vap->iv_flags & IEEE80211_F_BGSCAN) &&
|
||||
(ic->ic_flags & IEEE80211_F_CSAPENDING) == 0 &&
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
!IEEE80211_IS_CHAN_DTURBO(ic->ic_curchan) &&
|
||||
#endif
|
||||
time_after(ticks, ic->ic_lastscan + vap->iv_bgscanintvl) &&
|
||||
time_after(ticks, ic->ic_lastdata + vap->iv_bgscanidle));
|
||||
}
|
||||
@ -1271,8 +1214,10 @@ sta_recv_mgmt(struct ieee80211_node *ni, struct mbuf *m0,
|
||||
(ni->ni_flags & IEEE80211_NODE_QOS) &&
|
||||
ieee80211_parse_wmeparams(vap, scan.wme, wh) > 0)
|
||||
ieee80211_wme_updateparams(vap);
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
if (scan.ath != NULL)
|
||||
ieee80211_parse_athparams(ni, scan.ath, wh);
|
||||
#endif
|
||||
if (scan.htcap != NULL && scan.htinfo != NULL &&
|
||||
(vap->iv_flags_ext & IEEE80211_FEXT_HT)) {
|
||||
ieee80211_ht_updateparams(ni,
|
||||
|
506
sys/net80211/ieee80211_superg.c
Normal file
506
sys/net80211/ieee80211_superg.c
Normal file
@ -0,0 +1,506 @@
|
||||
/*-
|
||||
* Copyright (c) 2002-2009 Sam Leffler, Errno Consulting
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions
|
||||
* are met:
|
||||
* 1. Redistributions of source code must retain the above copyright
|
||||
* notice, this list of conditions and the following disclaimer.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice, this list of conditions and the following disclaimer in the
|
||||
* documentation and/or other materials provided with the distribution.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
||||
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
||||
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
||||
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
||||
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#include <sys/cdefs.h>
|
||||
__FBSDID("$FreeBSD$");
|
||||
|
||||
#include "opt_wlan.h"
|
||||
|
||||
#include <sys/param.h>
|
||||
#include <sys/systm.h>
|
||||
#include <sys/mbuf.h>
|
||||
#include <sys/kernel.h>
|
||||
#include <sys/endian.h>
|
||||
|
||||
#include <sys/socket.h>
|
||||
|
||||
#include <net/bpf.h>
|
||||
#include <net/ethernet.h>
|
||||
#include <net/if.h>
|
||||
#include <net/if_llc.h>
|
||||
#include <net/if_media.h>
|
||||
|
||||
#include <net80211/ieee80211_var.h>
|
||||
#include <net80211/ieee80211_input.h>
|
||||
#include <net80211/ieee80211_phy.h>
|
||||
#include <net80211/ieee80211_superg.h>
|
||||
|
||||
#define ETHER_HEADER_COPY(dst, src) \
|
||||
memcpy(dst, src, sizeof(struct ether_header))
|
||||
|
||||
void
|
||||
ieee80211_superg_attach(struct ieee80211com *ic)
|
||||
{
|
||||
}
|
||||
|
||||
void
|
||||
ieee80211_superg_detach(struct ieee80211com *ic)
|
||||
{
|
||||
}
|
||||
|
||||
void
|
||||
ieee80211_superg_vattach(struct ieee80211vap *vap)
|
||||
{
|
||||
if (vap->iv_caps & IEEE80211_C_FF)
|
||||
vap->iv_flags |= IEEE80211_F_FF;
|
||||
if (vap->iv_caps & IEEE80211_C_TURBOP)
|
||||
vap->iv_flags |= IEEE80211_F_TURBOP;
|
||||
}
|
||||
|
||||
void
|
||||
ieee80211_superg_vdetach(struct ieee80211vap *vap)
|
||||
{
|
||||
}
|
||||
|
||||
#define ATH_OUI_BYTES 0x00, 0x03, 0x7f
|
||||
/*
|
||||
* Add a WME information element to a frame.
|
||||
*/
|
||||
uint8_t *
|
||||
ieee80211_add_ath(uint8_t *frm, uint8_t caps, uint16_t defkeyix)
|
||||
{
|
||||
static const struct ieee80211_ath_ie info = {
|
||||
.ath_id = IEEE80211_ELEMID_VENDOR,
|
||||
.ath_len = sizeof(struct ieee80211_ath_ie) - 2,
|
||||
.ath_oui = { ATH_OUI_BYTES },
|
||||
.ath_oui_type = ATH_OUI_TYPE,
|
||||
.ath_oui_subtype= ATH_OUI_SUBTYPE,
|
||||
.ath_version = ATH_OUI_VERSION,
|
||||
};
|
||||
struct ieee80211_ath_ie *ath = (struct ieee80211_ath_ie *) frm;
|
||||
|
||||
memcpy(frm, &info, sizeof(info));
|
||||
ath->ath_capability = caps;
|
||||
ath->ath_defkeyix[0] = (defkeyix & 0xff);
|
||||
ath->ath_defkeyix[1] = ((defkeyix >> 8) & 0xff);
|
||||
return frm + sizeof(info);
|
||||
}
|
||||
#undef ATH_OUI_BYTES
|
||||
|
||||
void
|
||||
ieee80211_parse_ath(struct ieee80211_node *ni, uint8_t *ie)
|
||||
{
|
||||
const struct ieee80211_ath_ie *ath =
|
||||
(const struct ieee80211_ath_ie *) ie;
|
||||
|
||||
ni->ni_ath_flags = ath->ath_capability;
|
||||
ni->ni_ath_defkeyix = LE_READ_2(&ath->ath_defkeyix);
|
||||
}
|
||||
|
||||
int
|
||||
ieee80211_parse_athparams(struct ieee80211_node *ni, uint8_t *frm,
|
||||
const struct ieee80211_frame *wh)
|
||||
{
|
||||
struct ieee80211vap *vap = ni->ni_vap;
|
||||
const struct ieee80211_ath_ie *ath;
|
||||
u_int len = frm[1];
|
||||
int capschanged;
|
||||
uint16_t defkeyix;
|
||||
|
||||
if (len < sizeof(struct ieee80211_ath_ie)-2) {
|
||||
IEEE80211_DISCARD_IE(vap,
|
||||
IEEE80211_MSG_ELEMID | IEEE80211_MSG_SUPERG,
|
||||
wh, "Atheros", "too short, len %u", len);
|
||||
return -1;
|
||||
}
|
||||
ath = (const struct ieee80211_ath_ie *)frm;
|
||||
capschanged = (ni->ni_ath_flags != ath->ath_capability);
|
||||
defkeyix = LE_READ_2(ath->ath_defkeyix);
|
||||
if (capschanged || defkeyix != ni->ni_ath_defkeyix) {
|
||||
ni->ni_ath_flags = ath->ath_capability;
|
||||
ni->ni_ath_defkeyix = defkeyix;
|
||||
IEEE80211_NOTE(vap, IEEE80211_MSG_SUPERG, ni,
|
||||
"ath ie change: new caps 0x%x defkeyix 0x%x",
|
||||
ni->ni_ath_flags, ni->ni_ath_defkeyix);
|
||||
}
|
||||
if (IEEE80211_ATH_CAP(vap, ni, ATHEROS_CAP_TURBO_PRIME)) {
|
||||
uint16_t curflags, newflags;
|
||||
|
||||
/*
|
||||
* Check for turbo mode switch. Calculate flags
|
||||
* for the new mode and effect the switch.
|
||||
*/
|
||||
newflags = curflags = vap->iv_ic->ic_bsschan->ic_flags;
|
||||
/* NB: BOOST is not in ic_flags, so get it from the ie */
|
||||
if (ath->ath_capability & ATHEROS_CAP_BOOST)
|
||||
newflags |= IEEE80211_CHAN_TURBO;
|
||||
else
|
||||
newflags &= ~IEEE80211_CHAN_TURBO;
|
||||
if (newflags != curflags)
|
||||
ieee80211_dturbo_switch(vap, newflags);
|
||||
}
|
||||
return capschanged;
|
||||
}
|
||||
|
||||
/*
|
||||
* Decap the encapsulated frame pair and dispatch the first
|
||||
* for delivery. The second frame is returned for delivery
|
||||
* via the normal path.
|
||||
*/
|
||||
struct mbuf *
|
||||
ieee80211_ff_decap(struct ieee80211_node *ni, struct mbuf *m)
|
||||
{
|
||||
#define FF_LLC_SIZE (sizeof(struct ether_header) + sizeof(struct llc))
|
||||
#define MS(x,f) (((x) & f) >> f##_S)
|
||||
struct ieee80211vap *vap = ni->ni_vap;
|
||||
struct llc *llc;
|
||||
uint32_t ath;
|
||||
struct mbuf *n;
|
||||
int framelen;
|
||||
|
||||
/* NB: we assume caller does this check for us */
|
||||
KASSERT(IEEE80211_ATH_CAP(vap, ni, IEEE80211_NODE_FF),
|
||||
("ff not negotiated"));
|
||||
/*
|
||||
* Check for fast-frame tunnel encapsulation.
|
||||
*/
|
||||
if (m->m_pkthdr.len < 3*FF_LLC_SIZE)
|
||||
return m;
|
||||
if (m->m_len < FF_LLC_SIZE &&
|
||||
(m = m_pullup(m, FF_LLC_SIZE)) == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame",
|
||||
"%s", "m_pullup(llc) failed");
|
||||
vap->iv_stats.is_rx_tooshort++;
|
||||
return NULL;
|
||||
}
|
||||
llc = (struct llc *)(mtod(m, uint8_t *) +
|
||||
sizeof(struct ether_header));
|
||||
if (llc->llc_snap.ether_type != htons(ATH_FF_ETH_TYPE))
|
||||
return m;
|
||||
m_adj(m, FF_LLC_SIZE);
|
||||
m_copydata(m, 0, sizeof(uint32_t), (caddr_t) &ath);
|
||||
if (MS(ath, ATH_FF_PROTO) != ATH_FF_PROTO_L2TUNNEL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame",
|
||||
"unsupport tunnel protocol, header 0x%x", ath);
|
||||
vap->iv_stats.is_ff_badhdr++;
|
||||
m_freem(m);
|
||||
return NULL;
|
||||
}
|
||||
/* NB: skip header and alignment padding */
|
||||
m_adj(m, roundup(sizeof(uint32_t) - 2, 4) + 2);
|
||||
|
||||
vap->iv_stats.is_ff_decap++;
|
||||
|
||||
/*
|
||||
* Decap the first frame, bust it apart from the
|
||||
* second and deliver; then decap the second frame
|
||||
* and return it to the caller for normal delivery.
|
||||
*/
|
||||
m = ieee80211_decap1(m, &framelen);
|
||||
if (m == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame", "%s", "first decap failed");
|
||||
vap->iv_stats.is_ff_tooshort++;
|
||||
return NULL;
|
||||
}
|
||||
n = m_split(m, framelen, M_NOWAIT);
|
||||
if (n == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame",
|
||||
"%s", "unable to split encapsulated frames");
|
||||
vap->iv_stats.is_ff_split++;
|
||||
m_freem(m); /* NB: must reclaim */
|
||||
return NULL;
|
||||
}
|
||||
/* XXX not right for WDS */
|
||||
vap->iv_deliver_data(vap, ni, m); /* 1st of pair */
|
||||
|
||||
/*
|
||||
* Decap second frame.
|
||||
*/
|
||||
m_adj(n, roundup2(framelen, 4) - framelen); /* padding */
|
||||
n = ieee80211_decap1(n, &framelen);
|
||||
if (n == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame", "%s", "second decap failed");
|
||||
vap->iv_stats.is_ff_tooshort++;
|
||||
}
|
||||
/* XXX verify framelen against mbuf contents */
|
||||
return n; /* 2nd delivered by caller */
|
||||
#undef MS
|
||||
#undef FF_LLC_SIZE
|
||||
}
|
||||
|
||||
/*
|
||||
* Do Ethernet-LLC encapsulation for each payload in a fast frame
|
||||
* tunnel encapsulation. The frame is assumed to have an Ethernet
|
||||
* header at the front that must be stripped before prepending the
|
||||
* LLC followed by the Ethernet header passed in (with an Ethernet
|
||||
* type that specifies the payload size).
|
||||
*/
|
||||
static struct mbuf *
|
||||
ff_encap1(struct ieee80211vap *vap, struct mbuf *m,
|
||||
const struct ether_header *eh)
|
||||
{
|
||||
struct llc *llc;
|
||||
uint16_t payload;
|
||||
|
||||
/* XXX optimize by combining m_adj+M_PREPEND */
|
||||
m_adj(m, sizeof(struct ether_header) - sizeof(struct llc));
|
||||
llc = mtod(m, struct llc *);
|
||||
llc->llc_dsap = llc->llc_ssap = LLC_SNAP_LSAP;
|
||||
llc->llc_control = LLC_UI;
|
||||
llc->llc_snap.org_code[0] = 0;
|
||||
llc->llc_snap.org_code[1] = 0;
|
||||
llc->llc_snap.org_code[2] = 0;
|
||||
llc->llc_snap.ether_type = eh->ether_type;
|
||||
payload = m->m_pkthdr.len; /* NB: w/o Ethernet header */
|
||||
|
||||
M_PREPEND(m, sizeof(struct ether_header), M_DONTWAIT);
|
||||
if (m == NULL) { /* XXX cannot happen */
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: no space for ether_header\n", __func__);
|
||||
vap->iv_stats.is_tx_nobuf++;
|
||||
return NULL;
|
||||
}
|
||||
ETHER_HEADER_COPY(mtod(m, void *), eh);
|
||||
mtod(m, struct ether_header *)->ether_type = htons(payload);
|
||||
return m;
|
||||
}
|
||||
|
||||
/*
|
||||
* Fast frame encapsulation. There must be two packets
|
||||
* chained with m_nextpkt. We do header adjustment for
|
||||
* each, add the tunnel encapsulation, and then concatenate
|
||||
* the mbuf chains to form a single frame for transmission.
|
||||
*/
|
||||
struct mbuf *
|
||||
ieee80211_ff_encap(struct ieee80211vap *vap, struct mbuf *m1, int hdrspace,
|
||||
struct ieee80211_key *key)
|
||||
{
|
||||
struct mbuf *m2;
|
||||
struct ether_header eh1, eh2;
|
||||
struct llc *llc;
|
||||
struct mbuf *m;
|
||||
int pad;
|
||||
|
||||
m2 = m1->m_nextpkt;
|
||||
if (m2 == NULL) {
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: only one frame\n", __func__);
|
||||
goto bad;
|
||||
}
|
||||
m1->m_nextpkt = NULL;
|
||||
/*
|
||||
* Include fast frame headers in adjusting header
|
||||
* layout; this allocates space according to what
|
||||
* ff_encap will do.
|
||||
*/
|
||||
m1 = ieee80211_mbuf_adjust(vap,
|
||||
hdrspace + sizeof(struct llc) + sizeof(uint32_t) + 2 +
|
||||
sizeof(struct ether_header),
|
||||
key, m1);
|
||||
if (m1 == NULL) {
|
||||
/* NB: ieee80211_mbuf_adjust handles msgs+statistics */
|
||||
m_freem(m2);
|
||||
goto bad;
|
||||
}
|
||||
|
||||
/*
|
||||
* Copy second frame's Ethernet header out of line
|
||||
* and adjust for encapsulation headers. Note that
|
||||
* we make room for padding in case there isn't room
|
||||
* at the end of first frame.
|
||||
*/
|
||||
KASSERT(m2->m_len >= sizeof(eh2), ("no ethernet header!"));
|
||||
ETHER_HEADER_COPY(&eh2, mtod(m2, caddr_t));
|
||||
m2 = ieee80211_mbuf_adjust(vap,
|
||||
ATH_FF_MAX_HDR_PAD + sizeof(struct ether_header),
|
||||
NULL, m2);
|
||||
if (m2 == NULL) {
|
||||
/* NB: ieee80211_mbuf_adjust handles msgs+statistics */
|
||||
goto bad;
|
||||
}
|
||||
|
||||
/*
|
||||
* Now do tunnel encapsulation. First, each
|
||||
* frame gets a standard encapsulation.
|
||||
*/
|
||||
m1 = ff_encap1(vap, m1, &eh1);
|
||||
if (m1 == NULL)
|
||||
goto bad;
|
||||
m2 = ff_encap1(vap, m2, &eh2);
|
||||
if (m2 == NULL)
|
||||
goto bad;
|
||||
|
||||
/*
|
||||
* Pad leading frame to a 4-byte boundary. If there
|
||||
* is space at the end of the first frame, put it
|
||||
* there; otherwise prepend to the front of the second
|
||||
* frame. We know doing the second will always work
|
||||
* because we reserve space above. We prefer appending
|
||||
* as this typically has better DMA alignment properties.
|
||||
*/
|
||||
for (m = m1; m->m_next != NULL; m = m->m_next)
|
||||
;
|
||||
pad = roundup2(m1->m_pkthdr.len, 4) - m1->m_pkthdr.len;
|
||||
if (pad) {
|
||||
if (M_TRAILINGSPACE(m) < pad) { /* prepend to second */
|
||||
m2->m_data -= pad;
|
||||
m2->m_len += pad;
|
||||
m2->m_pkthdr.len += pad;
|
||||
} else { /* append to first */
|
||||
m->m_len += pad;
|
||||
m1->m_pkthdr.len += pad;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Now, stick 'em together and prepend the tunnel headers;
|
||||
* first the Atheros tunnel header (all zero for now) and
|
||||
* then a special fast frame LLC.
|
||||
*
|
||||
* XXX optimize by prepending together
|
||||
*/
|
||||
m->m_next = m2; /* NB: last mbuf from above */
|
||||
m1->m_pkthdr.len += m2->m_pkthdr.len;
|
||||
M_PREPEND(m1, sizeof(uint32_t)+2, M_DONTWAIT);
|
||||
if (m1 == NULL) { /* XXX cannot happen */
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: no space for tunnel header\n", __func__);
|
||||
vap->iv_stats.is_tx_nobuf++;
|
||||
return NULL;
|
||||
}
|
||||
memset(mtod(m1, void *), 0, sizeof(uint32_t)+2);
|
||||
|
||||
M_PREPEND(m1, sizeof(struct llc), M_DONTWAIT);
|
||||
if (m1 == NULL) { /* XXX cannot happen */
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: no space for llc header\n", __func__);
|
||||
vap->iv_stats.is_tx_nobuf++;
|
||||
return NULL;
|
||||
}
|
||||
llc = mtod(m1, struct llc *);
|
||||
llc->llc_dsap = llc->llc_ssap = LLC_SNAP_LSAP;
|
||||
llc->llc_control = LLC_UI;
|
||||
llc->llc_snap.org_code[0] = ATH_FF_SNAP_ORGCODE_0;
|
||||
llc->llc_snap.org_code[1] = ATH_FF_SNAP_ORGCODE_1;
|
||||
llc->llc_snap.org_code[2] = ATH_FF_SNAP_ORGCODE_2;
|
||||
llc->llc_snap.ether_type = htons(ATH_FF_ETH_TYPE);
|
||||
|
||||
vap->iv_stats.is_ff_encap++;
|
||||
|
||||
return m1;
|
||||
bad:
|
||||
if (m1 != NULL)
|
||||
m_freem(m1);
|
||||
if (m2 != NULL)
|
||||
m_freem(m2);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/*
|
||||
* Switch between turbo and non-turbo operating modes.
|
||||
* Use the specified channel flags to locate the new
|
||||
* channel, update 802.11 state, and then call back into
|
||||
* the driver to effect the change.
|
||||
*/
|
||||
void
|
||||
ieee80211_dturbo_switch(struct ieee80211vap *vap, int newflags)
|
||||
{
|
||||
struct ieee80211com *ic = vap->iv_ic;
|
||||
struct ieee80211_channel *chan;
|
||||
|
||||
chan = ieee80211_find_channel(ic, ic->ic_bsschan->ic_freq, newflags);
|
||||
if (chan == NULL) { /* XXX should not happen */
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: no channel with freq %u flags 0x%x\n",
|
||||
__func__, ic->ic_bsschan->ic_freq, newflags);
|
||||
return;
|
||||
}
|
||||
|
||||
IEEE80211_DPRINTF(vap, IEEE80211_MSG_SUPERG,
|
||||
"%s: %s -> %s (freq %u flags 0x%x)\n", __func__,
|
||||
ieee80211_phymode_name[ieee80211_chan2mode(ic->ic_bsschan)],
|
||||
ieee80211_phymode_name[ieee80211_chan2mode(chan)],
|
||||
chan->ic_freq, chan->ic_flags);
|
||||
|
||||
ic->ic_bsschan = chan;
|
||||
ic->ic_prevchan = ic->ic_curchan;
|
||||
ic->ic_curchan = chan;
|
||||
ic->ic_set_channel(ic);
|
||||
/* NB: do not need to reset ERP state 'cuz we're in sta mode */
|
||||
}
|
||||
|
||||
/*
|
||||
* Return the current ``state'' of an Atheros capbility.
|
||||
* If associated in station mode report the negotiated
|
||||
* setting. Otherwise report the current setting.
|
||||
*/
|
||||
static int
|
||||
getathcap(struct ieee80211vap *vap, int cap)
|
||||
{
|
||||
if (vap->iv_opmode == IEEE80211_M_STA &&
|
||||
vap->iv_state == IEEE80211_S_RUN)
|
||||
return IEEE80211_ATH_CAP(vap, vap->iv_bss, cap) != 0;
|
||||
else
|
||||
return (vap->iv_flags & cap) != 0;
|
||||
}
|
||||
|
||||
static int
|
||||
superg_ioctl_get80211(struct ieee80211vap *vap, struct ieee80211req *ireq)
|
||||
{
|
||||
switch (ireq->i_type) {
|
||||
case IEEE80211_IOC_FF:
|
||||
ireq->i_val = getathcap(vap, IEEE80211_F_FF);
|
||||
break;
|
||||
case IEEE80211_IOC_TURBOP:
|
||||
ireq->i_val = getathcap(vap, IEEE80211_F_TURBOP);
|
||||
break;
|
||||
default:
|
||||
return ENOSYS;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
IEEE80211_IOCTL_GET(superg, superg_ioctl_get80211);
|
||||
|
||||
static int
|
||||
superg_ioctl_set80211(struct ieee80211vap *vap, struct ieee80211req *ireq)
|
||||
{
|
||||
switch (ireq->i_type) {
|
||||
case IEEE80211_IOC_FF:
|
||||
if (ireq->i_val) {
|
||||
if ((vap->iv_caps & IEEE80211_C_FF) == 0)
|
||||
return EOPNOTSUPP;
|
||||
vap->iv_flags |= IEEE80211_F_FF;
|
||||
} else
|
||||
vap->iv_flags &= ~IEEE80211_F_FF;
|
||||
return ERESTART;
|
||||
case IEEE80211_IOC_TURBOP:
|
||||
if (ireq->i_val) {
|
||||
if ((vap->iv_caps & IEEE80211_C_TURBOP) == 0)
|
||||
return EOPNOTSUPP;
|
||||
vap->iv_flags |= IEEE80211_F_TURBOP;
|
||||
} else
|
||||
vap->iv_flags &= ~IEEE80211_F_TURBOP;
|
||||
return ENETRESET;
|
||||
default:
|
||||
return ENOSYS;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
IEEE80211_IOCTL_SET(superg, superg_ioctl_set80211);
|
56
sys/net80211/ieee80211_superg.h
Normal file
56
sys/net80211/ieee80211_superg.h
Normal file
@ -0,0 +1,56 @@
|
||||
/*-
|
||||
* Copyright (c) 2009 Sam Leffler, Errno Consulting
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions
|
||||
* are met:
|
||||
* 1. Redistributions of source code must retain the above copyright
|
||||
* notice, this list of conditions and the following disclaimer.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice, this list of conditions and the following disclaimer in the
|
||||
* documentation and/or other materials provided with the distribution.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
||||
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
||||
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
||||
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
||||
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*
|
||||
* $FreeBSD$
|
||||
*/
|
||||
#ifndef _NET80211_IEEE80211_SUPERG_H_
|
||||
#define _NET80211_IEEE80211_SUPERG_H_
|
||||
|
||||
/*
|
||||
* Atheros' 802.11 SuperG protocol support.
|
||||
*/
|
||||
|
||||
void ieee80211_superg_attach(struct ieee80211com *);
|
||||
void ieee80211_superg_detach(struct ieee80211com *);
|
||||
void ieee80211_superg_vattach(struct ieee80211vap *);
|
||||
void ieee80211_superg_vdetach(struct ieee80211vap *);
|
||||
|
||||
uint8_t *ieee80211_add_ath(uint8_t *frm, uint8_t caps, uint16_t defkeyix);
|
||||
void ieee80211_parse_ath(struct ieee80211_node *, uint8_t *);
|
||||
int ieee80211_parse_athparams(struct ieee80211_node *, uint8_t *,
|
||||
const struct ieee80211_frame *);
|
||||
|
||||
struct mbuf *ieee80211_ff_encap(struct ieee80211vap *, struct mbuf *,
|
||||
int, struct ieee80211_key *);
|
||||
|
||||
struct mbuf *ieee80211_ff_decap(struct ieee80211_node *, struct mbuf *);
|
||||
|
||||
static __inline struct mbuf *
|
||||
ieee80211_decap_fastframe(struct ieee80211vap *vap, struct ieee80211_node *ni,
|
||||
struct mbuf *m)
|
||||
{
|
||||
return IEEE80211_ATH_CAP(vap, ni, IEEE80211_NODE_FF) ?
|
||||
ieee80211_ff_decap(ni, m) : m;
|
||||
}
|
||||
#endif /* _NET80211_IEEE80211_SUPERG_H_ */
|
@ -57,6 +57,9 @@ __FBSDID("$FreeBSD$");
|
||||
#include <net80211/ieee80211_var.h>
|
||||
#include <net80211/ieee80211_wds.h>
|
||||
#include <net80211/ieee80211_input.h>
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
#include <net80211/ieee80211_superg.h>
|
||||
#endif
|
||||
|
||||
static void wds_vattach(struct ieee80211vap *);
|
||||
static int wds_newstate(struct ieee80211vap *, enum ieee80211_state, int);
|
||||
@ -728,32 +731,13 @@ wds_input(struct ieee80211_node *ni, struct mbuf *m,
|
||||
m = ieee80211_decap_amsdu(ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
} else if (IEEE80211_ATH_CAP(vap, ni, IEEE80211_NODE_FF) &&
|
||||
#define FF_LLC_SIZE (sizeof(struct ether_header) + sizeof(struct llc))
|
||||
m->m_pkthdr.len >= 3*FF_LLC_SIZE) {
|
||||
struct llc *llc;
|
||||
|
||||
/*
|
||||
* Check for fast-frame tunnel encapsulation.
|
||||
*/
|
||||
if (m->m_len < FF_LLC_SIZE &&
|
||||
(m = m_pullup(m, FF_LLC_SIZE)) == NULL) {
|
||||
IEEE80211_DISCARD_MAC(vap, IEEE80211_MSG_ANY,
|
||||
ni->ni_macaddr, "fast-frame",
|
||||
"%s", "m_pullup(llc) failed");
|
||||
vap->iv_stats.is_rx_tooshort++;
|
||||
} else {
|
||||
#ifdef IEEE80211_SUPPORT_SUPERG
|
||||
m = ieee80211_decap_fastframe(vap, ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
}
|
||||
llc = (struct llc *)(mtod(m, uint8_t *) +
|
||||
sizeof(struct ether_header));
|
||||
if (llc->llc_snap.ether_type == htons(ATH_FF_ETH_TYPE)) {
|
||||
m_adj(m, FF_LLC_SIZE);
|
||||
m = ieee80211_decap_fastframe(ni, m);
|
||||
if (m == NULL)
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
#undef FF_LLC_SIZE
|
||||
ieee80211_deliver_data(vap, ni, m);
|
||||
return IEEE80211_FC0_TYPE_DATA;
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user