xref: /freebsd/sys/net80211/ieee80211_ht.h (revision e1c4c8dd8d2d10b6104f06856a77bd5b4813a801)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright (c) 2007-2008 Sam Leffler, Errno Consulting
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer.
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26  */
27 #ifndef _NET80211_IEEE80211_HT_H_
28 #define _NET80211_IEEE80211_HT_H_
29 
30 /*
31  * 802.11n protocol implementation definitions.
32  */
33 
34 #include <sys/mbuf.h>
35 
36 #define	IEEE80211_AGGR_BAWMAX	64	/* max block ack window size */
37 /* threshold for aging overlapping non-HT bss */
38 #define	IEEE80211_NONHT_PRESENT_AGE	msecs_to_ticks(60*1000)
39 
40 struct ieee80211_tx_ampdu {
41 	struct ieee80211_node *txa_ni;	/* back pointer */
42 	u_short		txa_flags;
43 #define	IEEE80211_AGGR_IMMEDIATE	0x0001	/* BA policy */
44 #define	IEEE80211_AGGR_XCHGPEND		0x0002	/* ADDBA response pending */
45 #define	IEEE80211_AGGR_RUNNING		0x0004	/* ADDBA response received */
46 #define	IEEE80211_AGGR_SETUP		0x0008	/* deferred state setup */
47 #define	IEEE80211_AGGR_NAK		0x0010	/* peer NAK'd ADDBA request */
48 #define	IEEE80211_AGGR_BARPEND		0x0020	/* BAR response pending */
49 #define	IEEE80211_AGGR_WAITRX		0x0040	/* Wait for first RX frame to define BAW */
50 #define	IEEE80211_AGGR_AMSDU		0x0080	/* A-MSDU in A-MPDU TX allowed */
51 	uint8_t		txa_tid;
52 	uint8_t		txa_token;	/* dialog token */
53 	int		txa_lastsample;	/* ticks @ last traffic sample */
54 	int		txa_pkts;	/* packets over last sample interval */
55 	int		txa_avgpps;	/* filtered traffic over window */
56 	int		txa_qbytes;	/* data queued (bytes) */
57 	short		txa_qframes;	/* data queued (frames) */
58 	ieee80211_seq	txa_start;	/* BA window left edge */
59 	ieee80211_seq	txa_seqpending;	/* new txa_start pending BAR response */
60 	uint16_t	txa_wnd;	/* BA window size */
61 	uint8_t		txa_attempts;	/* # ADDBA/BAR requests w/o a response*/
62 	int		txa_nextrequest;/* soonest to make next request */
63 	struct callout	txa_timer;
64 	void		*txa_private;	/* driver-private storage */
65 	uint64_t	txa_pad[4];
66 };
67 
68 /* return non-zero if AMPDU tx for the TID is running */
69 #define	IEEE80211_AMPDU_RUNNING(tap) \
70 	(((tap)->txa_flags & IEEE80211_AGGR_RUNNING) != 0)
71 
72 /*
73  * Return non-zero if AMPDU tx for the TID is running and we can do
74  * A-MSDU in A-MPDU
75  */
76 #define	IEEE80211_AMPDU_RUNNING_AMSDU(tap) \
77 	(((tap)->txa_flags & (IEEE80211_AGGR_RUNNING | IEEE80211_AGGR_AMSDU)) \
78 	    == (IEEE80211_AGGR_RUNNING | IEEE80211_AGGR_AMSDU))
79 
80 /* return non-zero if AMPDU tx for the TID was NACKed */
81 #define	IEEE80211_AMPDU_NACKED(tap)\
82 	(!! ((tap)->txa_flags & IEEE80211_AGGR_NAK))
83 
84 /* return non-zero if AMPDU tx for the TID is running or started */
85 #define	IEEE80211_AMPDU_REQUESTED(tap) \
86 	(((tap)->txa_flags & \
87 	 (IEEE80211_AGGR_RUNNING|IEEE80211_AGGR_XCHGPEND|IEEE80211_AGGR_NAK)) != 0)
88 
89 #define	IEEE80211_AGGR_BITS \
90 	"\20\1IMMEDIATE\2XCHGPEND\3RUNNING\4SETUP\5NAK"
91 
92 /*
93  * Traffic estimator support.  We estimate packets/sec for
94  * each AC that is setup for AMPDU or will potentially be
95  * setup for AMPDU.  The traffic rate can be used to decide
96  * when AMPDU should be setup (according to a threshold)
97  * and is available for drivers to do things like cache
98  * eviction when only a limited number of BA streams are
99  * available and more streams are requested than available.
100  */
101 
102 static __inline void
103 ieee80211_txampdu_init_pps(struct ieee80211_tx_ampdu *tap)
104 {
105 	/*
106 	 * Reset packet estimate.
107 	 */
108 	tap->txa_lastsample = ticks;
109 	tap->txa_avgpps = 0;
110 }
111 
112 static __inline void
113 ieee80211_txampdu_update_pps(struct ieee80211_tx_ampdu *tap)
114 {
115 
116 	/* NB: scale factor of 2 was picked heuristically */
117 	tap->txa_avgpps = ((tap->txa_avgpps << 2) -
118 	     tap->txa_avgpps + tap->txa_pkts) >> 2;
119 }
120 
121 /*
122  * Count a packet towards the pps estimate.
123  */
124 static __inline void
125 ieee80211_txampdu_count_packet(struct ieee80211_tx_ampdu *tap)
126 {
127 
128 	/* XXX bound loop/do more crude estimate? */
129 	while (ticks - tap->txa_lastsample >= hz) {
130 		ieee80211_txampdu_update_pps(tap);
131 		/* reset to start new sample interval */
132 		tap->txa_pkts = 0;
133 		if (tap->txa_avgpps == 0) {
134 			tap->txa_lastsample = ticks;
135 			break;
136 		}
137 		tap->txa_lastsample += hz;
138 	}
139 	tap->txa_pkts++;
140 }
141 
142 /*
143  * Get the current pps estimate.  If the average is out of
144  * date due to lack of traffic then we decay the estimate
145  * to account for the idle time.
146  */
147 static __inline int
148 ieee80211_txampdu_getpps(struct ieee80211_tx_ampdu *tap)
149 {
150 	/* XXX bound loop/do more crude estimate? */
151 	while (ticks - tap->txa_lastsample >= hz) {
152 		ieee80211_txampdu_update_pps(tap);
153 		tap->txa_pkts = 0;
154 		if (tap->txa_avgpps == 0) {
155 			tap->txa_lastsample = ticks;
156 			break;
157 		}
158 		tap->txa_lastsample += hz;
159 	}
160 	return tap->txa_avgpps;
161 }
162 
163 struct ieee80211_rx_ampdu {
164 	int		rxa_flags;
165 	int		rxa_qbytes;	/* data queued (bytes) */
166 	short		rxa_qframes;	/* data queued (frames) */
167 	ieee80211_seq	rxa_seqstart;
168 	ieee80211_seq	rxa_start;	/* start of current BA window */
169 	uint16_t	rxa_wnd;	/* BA window size */
170 	int		rxa_age;	/* age of oldest frame in window */
171 	int		rxa_nframes;	/* frames since ADDBA */
172 	struct mbufq rxa_mq[IEEE80211_AGGR_BAWMAX];
173 	void		*rxa_private;
174 	uint64_t	rxa_pad[3];
175 };
176 
177 void	ieee80211_ht_attach(struct ieee80211com *);
178 void	ieee80211_ht_detach(struct ieee80211com *);
179 void	ieee80211_ht_vattach(struct ieee80211vap *);
180 void	ieee80211_ht_vdetach(struct ieee80211vap *);
181 
182 void	ieee80211_ht_announce(struct ieee80211com *);
183 
184 struct ieee80211_mcs_rates {
185 	uint16_t	ht20_rate_800ns;
186 	uint16_t	ht20_rate_400ns;
187 	uint16_t	ht40_rate_800ns;
188 	uint16_t	ht40_rate_400ns;
189 };
190 extern const struct ieee80211_mcs_rates ieee80211_htrates[];
191 void	ieee80211_init_suphtrates(struct ieee80211com *);
192 
193 struct ieee80211_node;
194 int	ieee80211_setup_htrates(struct ieee80211_node *,
195 		const uint8_t *htcap, int flags);
196 void	ieee80211_setup_basic_htrates(struct ieee80211_node *,
197 		const uint8_t *htinfo);
198 struct mbuf *ieee80211_decap_amsdu(struct ieee80211_node *, struct mbuf *);
199 int	ieee80211_ampdu_reorder(struct ieee80211_node *, struct mbuf *,
200 	    const struct ieee80211_rx_stats *);
201 void	ieee80211_recv_bar(struct ieee80211_node *, struct mbuf *);
202 void	ieee80211_ht_node_init(struct ieee80211_node *);
203 void	ieee80211_ht_node_cleanup(struct ieee80211_node *);
204 void	ieee80211_ht_node_age(struct ieee80211_node *);
205 
206 struct ieee80211_channel *ieee80211_ht_adjust_channel(struct ieee80211com *,
207 		struct ieee80211_channel *, int);
208 void	ieee80211_ht_wds_init(struct ieee80211_node *);
209 void	ieee80211_ht_node_join(struct ieee80211_node *);
210 void	ieee80211_ht_node_leave(struct ieee80211_node *);
211 void	ieee80211_htprot_update(struct ieee80211vap *, int protmode);
212 void	ieee80211_ht_timeout(struct ieee80211vap *);
213 void	ieee80211_parse_htcap(struct ieee80211_node *, const uint8_t *);
214 void	ieee80211_parse_htinfo(struct ieee80211_node *, const uint8_t *);
215 void	ieee80211_ht_updateparams(struct ieee80211_node *, const uint8_t *,
216 		const uint8_t *);
217 int	ieee80211_ht_updateparams_final(struct ieee80211_node *,
218 	    const uint8_t *, const uint8_t *);
219 void	ieee80211_ht_updatehtcap(struct ieee80211_node *, const uint8_t *);
220 void	ieee80211_ht_updatehtcap_final(struct ieee80211_node *);
221 int	ieee80211_ampdu_request(struct ieee80211_node *,
222 		struct ieee80211_tx_ampdu *);
223 void	ieee80211_ampdu_stop(struct ieee80211_node *,
224 		struct ieee80211_tx_ampdu *, int);
225 int	ieee80211_send_bar(struct ieee80211_node *, struct ieee80211_tx_ampdu *,
226 		ieee80211_seq);
227 uint8_t	*ieee80211_add_htcap(uint8_t *, struct ieee80211_node *);
228 uint8_t	*ieee80211_add_htcap_ch(uint8_t *, struct ieee80211vap *,
229 	    struct ieee80211_channel *);
230 uint8_t	*ieee80211_add_htcap_vendor(uint8_t *, struct ieee80211_node *);
231 uint8_t	*ieee80211_add_htinfo(uint8_t *, struct ieee80211_node *);
232 uint8_t	*ieee80211_add_htinfo_vendor(uint8_t *, struct ieee80211_node *);
233 struct ieee80211_beacon_offsets;
234 void	ieee80211_ht_update_beacon(struct ieee80211vap *,
235 		struct ieee80211_beacon_offsets *);
236 int	ieee80211_ampdu_rx_start_ext(struct ieee80211_node *ni, int tid,
237 	    int seq, int baw);
238 void	ieee80211_ampdu_rx_stop_ext(struct ieee80211_node *ni, int tid);
239 int	ieee80211_ampdu_tx_request_ext(struct ieee80211_node *ni, int tid);
240 int	ieee80211_ampdu_tx_request_active_ext(struct ieee80211_node *ni,
241 	    int tid, int status);
242 void	ieee80211_htinfo_notify(struct ieee80211vap *vap);
243 
244 #endif /* _NET80211_IEEE80211_HT_H_ */
245