xref: /openbmc/linux/drivers/net/wireless/intersil/hostap/hostap_ap.h (revision 498495dba268b20e8eadd7fe93c140c68b6cc9d2)
1*b2441318SGreg Kroah-Hartman /* SPDX-License-Identifier: GPL-2.0 */
2eb4f98d5SKalle Valo #ifndef HOSTAP_AP_H
3eb4f98d5SKalle Valo #define HOSTAP_AP_H
4eb4f98d5SKalle Valo 
5eb4f98d5SKalle Valo #include "hostap_80211.h"
6eb4f98d5SKalle Valo 
7eb4f98d5SKalle Valo /* AP data structures for STAs */
8eb4f98d5SKalle Valo 
9eb4f98d5SKalle Valo /* maximum number of frames to buffer per STA */
10eb4f98d5SKalle Valo #define STA_MAX_TX_BUFFER 32
11eb4f98d5SKalle Valo 
12eb4f98d5SKalle Valo /* STA flags */
13eb4f98d5SKalle Valo #define WLAN_STA_AUTH BIT(0)
14eb4f98d5SKalle Valo #define WLAN_STA_ASSOC BIT(1)
15eb4f98d5SKalle Valo #define WLAN_STA_PS BIT(2)
16eb4f98d5SKalle Valo #define WLAN_STA_TIM BIT(3) /* TIM bit is on for PS stations */
17eb4f98d5SKalle Valo #define WLAN_STA_PERM BIT(4) /* permanent; do not remove entry on expiration */
18eb4f98d5SKalle Valo #define WLAN_STA_AUTHORIZED BIT(5) /* If 802.1X is used, this flag is
19eb4f98d5SKalle Valo 				    * controlling whether STA is authorized to
20eb4f98d5SKalle Valo 				    * send and receive non-IEEE 802.1X frames
21eb4f98d5SKalle Valo 				    */
22eb4f98d5SKalle Valo #define WLAN_STA_PENDING_POLL BIT(6) /* pending activity poll not ACKed */
23eb4f98d5SKalle Valo 
24eb4f98d5SKalle Valo #define WLAN_RATE_1M BIT(0)
25eb4f98d5SKalle Valo #define WLAN_RATE_2M BIT(1)
26eb4f98d5SKalle Valo #define WLAN_RATE_5M5 BIT(2)
27eb4f98d5SKalle Valo #define WLAN_RATE_11M BIT(3)
28eb4f98d5SKalle Valo #define WLAN_RATE_COUNT 4
29eb4f98d5SKalle Valo 
30eb4f98d5SKalle Valo /* Maximum size of Supported Rates info element. IEEE 802.11 has a limit of 8,
31eb4f98d5SKalle Valo  * but some pre-standard IEEE 802.11g products use longer elements. */
32eb4f98d5SKalle Valo #define WLAN_SUPP_RATES_MAX 32
33eb4f98d5SKalle Valo 
34eb4f98d5SKalle Valo /* Try to increase TX rate after # successfully sent consecutive packets */
35eb4f98d5SKalle Valo #define WLAN_RATE_UPDATE_COUNT 50
36eb4f98d5SKalle Valo 
37eb4f98d5SKalle Valo /* Decrease TX rate after # consecutive dropped packets */
38eb4f98d5SKalle Valo #define WLAN_RATE_DECREASE_THRESHOLD 2
39eb4f98d5SKalle Valo 
40eb4f98d5SKalle Valo struct sta_info {
41eb4f98d5SKalle Valo 	struct list_head list;
42eb4f98d5SKalle Valo 	struct sta_info *hnext; /* next entry in hash table list */
43eb4f98d5SKalle Valo 	atomic_t users; /* number of users (do not remove if > 0) */
44eb4f98d5SKalle Valo 	struct proc_dir_entry *proc;
45eb4f98d5SKalle Valo 
46eb4f98d5SKalle Valo 	u8 addr[6];
47eb4f98d5SKalle Valo 	u16 aid; /* STA's unique AID (1 .. 2007) or 0 if not yet assigned */
48eb4f98d5SKalle Valo 	u32 flags;
49eb4f98d5SKalle Valo 	u16 capability;
50eb4f98d5SKalle Valo 	u16 listen_interval; /* or beacon_int for APs */
51eb4f98d5SKalle Valo 	u8 supported_rates[WLAN_SUPP_RATES_MAX];
52eb4f98d5SKalle Valo 
53eb4f98d5SKalle Valo 	unsigned long last_auth;
54eb4f98d5SKalle Valo 	unsigned long last_assoc;
55eb4f98d5SKalle Valo 	unsigned long last_rx;
56eb4f98d5SKalle Valo 	unsigned long last_tx;
57eb4f98d5SKalle Valo 	unsigned long rx_packets, tx_packets;
58eb4f98d5SKalle Valo 	unsigned long rx_bytes, tx_bytes;
59eb4f98d5SKalle Valo 	struct sk_buff_head tx_buf;
60eb4f98d5SKalle Valo 	/* FIX: timeout buffers with an expiry time somehow derived from
61eb4f98d5SKalle Valo 	 * listen_interval */
62eb4f98d5SKalle Valo 
63eb4f98d5SKalle Valo 	s8 last_rx_silence; /* Noise in dBm */
64eb4f98d5SKalle Valo 	s8 last_rx_signal; /* Signal strength in dBm */
65eb4f98d5SKalle Valo 	u8 last_rx_rate; /* TX rate in 0.1 Mbps */
66eb4f98d5SKalle Valo 	u8 last_rx_updated; /* IWSPY's struct iw_quality::updated */
67eb4f98d5SKalle Valo 
68eb4f98d5SKalle Valo 	u8 tx_supp_rates; /* bit field of supported TX rates */
69eb4f98d5SKalle Valo 	u8 tx_rate; /* current TX rate (in 0.1 Mbps) */
70eb4f98d5SKalle Valo 	u8 tx_rate_idx; /* current TX rate (WLAN_RATE_*) */
71eb4f98d5SKalle Valo 	u8 tx_max_rate; /* max TX rate (WLAN_RATE_*) */
72eb4f98d5SKalle Valo 	u32 tx_count[WLAN_RATE_COUNT]; /* number of frames sent (per rate) */
73eb4f98d5SKalle Valo 	u32 rx_count[WLAN_RATE_COUNT]; /* number of frames received (per rate)
74eb4f98d5SKalle Valo 					*/
75eb4f98d5SKalle Valo 	u32 tx_since_last_failure;
76eb4f98d5SKalle Valo 	u32 tx_consecutive_exc;
77eb4f98d5SKalle Valo 
78eb4f98d5SKalle Valo 	struct lib80211_crypt_data *crypt;
79eb4f98d5SKalle Valo 
80eb4f98d5SKalle Valo 	int ap; /* whether this station is an AP */
81eb4f98d5SKalle Valo 
82eb4f98d5SKalle Valo 	local_info_t *local;
83eb4f98d5SKalle Valo 
84eb4f98d5SKalle Valo #ifndef PRISM2_NO_KERNEL_IEEE80211_MGMT
85eb4f98d5SKalle Valo 	union {
86eb4f98d5SKalle Valo 		struct {
87eb4f98d5SKalle Valo 			char *challenge; /* shared key authentication
88eb4f98d5SKalle Valo 					  * challenge */
89eb4f98d5SKalle Valo 		} sta;
90eb4f98d5SKalle Valo 		struct {
91eb4f98d5SKalle Valo 			int ssid_len;
92eb4f98d5SKalle Valo 			unsigned char ssid[MAX_SSID_LEN + 1]; /* AP's ssid */
93eb4f98d5SKalle Valo 			int channel;
94eb4f98d5SKalle Valo 			unsigned long last_beacon; /* last RX beacon time */
95eb4f98d5SKalle Valo 		} ap;
96eb4f98d5SKalle Valo 	} u;
97eb4f98d5SKalle Valo 
98eb4f98d5SKalle Valo 	struct timer_list timer;
99eb4f98d5SKalle Valo 	enum { STA_NULLFUNC = 0, STA_DISASSOC, STA_DEAUTH } timeout_next;
100eb4f98d5SKalle Valo #endif /* PRISM2_NO_KERNEL_IEEE80211_MGMT */
101eb4f98d5SKalle Valo };
102eb4f98d5SKalle Valo 
103eb4f98d5SKalle Valo 
104eb4f98d5SKalle Valo #define MAX_STA_COUNT 1024
105eb4f98d5SKalle Valo 
106eb4f98d5SKalle Valo /* Maximum number of AIDs to use for STAs; must be 2007 or lower
107eb4f98d5SKalle Valo  * (8802.11 limitation) */
108eb4f98d5SKalle Valo #define MAX_AID_TABLE_SIZE 128
109eb4f98d5SKalle Valo 
110eb4f98d5SKalle Valo #define STA_HASH_SIZE 256
111eb4f98d5SKalle Valo #define STA_HASH(sta) (sta[5])
112eb4f98d5SKalle Valo 
113eb4f98d5SKalle Valo 
114eb4f98d5SKalle Valo /* Default value for maximum station inactivity. After AP_MAX_INACTIVITY_SEC
115eb4f98d5SKalle Valo  * has passed since last received frame from the station, a nullfunc data
116eb4f98d5SKalle Valo  * frame is sent to the station. If this frame is not acknowledged and no other
117eb4f98d5SKalle Valo  * frames have been received, the station will be disassociated after
118eb4f98d5SKalle Valo  * AP_DISASSOC_DELAY. Similarly, a the station will be deauthenticated after
119eb4f98d5SKalle Valo  * AP_DEAUTH_DELAY. AP_TIMEOUT_RESOLUTION is the resolution that is used with
120eb4f98d5SKalle Valo  * max inactivity timer. */
121eb4f98d5SKalle Valo #define AP_MAX_INACTIVITY_SEC (5 * 60)
122eb4f98d5SKalle Valo #define AP_DISASSOC_DELAY (HZ)
123eb4f98d5SKalle Valo #define AP_DEAUTH_DELAY (HZ)
124eb4f98d5SKalle Valo 
125eb4f98d5SKalle Valo /* ap_policy: whether to accept frames to/from other APs/IBSS */
126eb4f98d5SKalle Valo typedef enum {
127eb4f98d5SKalle Valo 	AP_OTHER_AP_SKIP_ALL = 0,
128eb4f98d5SKalle Valo 	AP_OTHER_AP_SAME_SSID = 1,
129eb4f98d5SKalle Valo 	AP_OTHER_AP_ALL = 2,
130eb4f98d5SKalle Valo 	AP_OTHER_AP_EVEN_IBSS = 3
131eb4f98d5SKalle Valo } ap_policy_enum;
132eb4f98d5SKalle Valo 
133eb4f98d5SKalle Valo #define PRISM2_AUTH_OPEN BIT(0)
134eb4f98d5SKalle Valo #define PRISM2_AUTH_SHARED_KEY BIT(1)
135eb4f98d5SKalle Valo 
136eb4f98d5SKalle Valo 
137eb4f98d5SKalle Valo /* MAC address-based restrictions */
138eb4f98d5SKalle Valo struct mac_entry {
139eb4f98d5SKalle Valo 	struct list_head list;
140eb4f98d5SKalle Valo 	u8 addr[6];
141eb4f98d5SKalle Valo };
142eb4f98d5SKalle Valo 
143eb4f98d5SKalle Valo struct mac_restrictions {
144eb4f98d5SKalle Valo 	enum { MAC_POLICY_OPEN = 0, MAC_POLICY_ALLOW, MAC_POLICY_DENY } policy;
145eb4f98d5SKalle Valo 	unsigned int entries;
146eb4f98d5SKalle Valo 	struct list_head mac_list;
147eb4f98d5SKalle Valo 	spinlock_t lock;
148eb4f98d5SKalle Valo };
149eb4f98d5SKalle Valo 
150eb4f98d5SKalle Valo 
151eb4f98d5SKalle Valo struct add_sta_proc_data {
152eb4f98d5SKalle Valo 	u8 addr[ETH_ALEN];
153eb4f98d5SKalle Valo 	struct add_sta_proc_data *next;
154eb4f98d5SKalle Valo };
155eb4f98d5SKalle Valo 
156eb4f98d5SKalle Valo 
157eb4f98d5SKalle Valo typedef enum { WDS_ADD, WDS_DEL } wds_oper_type;
158eb4f98d5SKalle Valo struct wds_oper_data {
159eb4f98d5SKalle Valo 	wds_oper_type type;
160eb4f98d5SKalle Valo 	u8 addr[ETH_ALEN];
161eb4f98d5SKalle Valo 	struct wds_oper_data *next;
162eb4f98d5SKalle Valo };
163eb4f98d5SKalle Valo 
164eb4f98d5SKalle Valo 
165eb4f98d5SKalle Valo struct ap_data {
166eb4f98d5SKalle Valo 	int initialized; /* whether ap_data has been initialized */
167eb4f98d5SKalle Valo 	local_info_t *local;
168eb4f98d5SKalle Valo 	int bridge_packets; /* send packet to associated STAs directly to the
169eb4f98d5SKalle Valo 			     * wireless media instead of higher layers in the
170eb4f98d5SKalle Valo 			     * kernel */
171eb4f98d5SKalle Valo 	unsigned int bridged_unicast; /* number of unicast frames bridged on
172eb4f98d5SKalle Valo 				       * wireless media */
173eb4f98d5SKalle Valo 	unsigned int bridged_multicast; /* number of non-unicast frames
174eb4f98d5SKalle Valo 					 * bridged on wireless media */
175eb4f98d5SKalle Valo 	unsigned int tx_drop_nonassoc; /* number of unicast TX packets dropped
176eb4f98d5SKalle Valo 					* because they were to an address that
177eb4f98d5SKalle Valo 					* was not associated */
178eb4f98d5SKalle Valo 	int nullfunc_ack; /* use workaround for nullfunc frame ACKs */
179eb4f98d5SKalle Valo 
180eb4f98d5SKalle Valo 	spinlock_t sta_table_lock;
181eb4f98d5SKalle Valo 	int num_sta; /* number of entries in sta_list */
182eb4f98d5SKalle Valo 	struct list_head sta_list; /* STA info list head */
183eb4f98d5SKalle Valo 	struct sta_info *sta_hash[STA_HASH_SIZE];
184eb4f98d5SKalle Valo 
185eb4f98d5SKalle Valo 	struct proc_dir_entry *proc;
186eb4f98d5SKalle Valo 
187eb4f98d5SKalle Valo 	ap_policy_enum ap_policy;
188eb4f98d5SKalle Valo 	unsigned int max_inactivity;
189eb4f98d5SKalle Valo 	int autom_ap_wds;
190eb4f98d5SKalle Valo 
191eb4f98d5SKalle Valo 	struct mac_restrictions mac_restrictions; /* MAC-based auth */
192eb4f98d5SKalle Valo 	int last_tx_rate;
193eb4f98d5SKalle Valo 
194eb4f98d5SKalle Valo 	struct work_struct add_sta_proc_queue;
195eb4f98d5SKalle Valo 	struct add_sta_proc_data *add_sta_proc_entries;
196eb4f98d5SKalle Valo 
197eb4f98d5SKalle Valo 	struct work_struct wds_oper_queue;
198eb4f98d5SKalle Valo 	struct wds_oper_data *wds_oper_entries;
199eb4f98d5SKalle Valo 
200eb4f98d5SKalle Valo 	u16 tx_callback_idx;
201eb4f98d5SKalle Valo 
202eb4f98d5SKalle Valo #ifndef PRISM2_NO_KERNEL_IEEE80211_MGMT
203eb4f98d5SKalle Valo 	/* pointers to STA info; based on allocated AID or NULL if AID free
204eb4f98d5SKalle Valo 	 * AID is in the range 1-2007, so sta_aid[0] corresponders to AID 1
205eb4f98d5SKalle Valo 	 * and so on
206eb4f98d5SKalle Valo 	 */
207eb4f98d5SKalle Valo 	struct sta_info *sta_aid[MAX_AID_TABLE_SIZE];
208eb4f98d5SKalle Valo 
209eb4f98d5SKalle Valo 	u16 tx_callback_auth, tx_callback_assoc, tx_callback_poll;
210eb4f98d5SKalle Valo 
211eb4f98d5SKalle Valo 	/* WEP operations for generating challenges to be used with shared key
212eb4f98d5SKalle Valo 	 * authentication */
213eb4f98d5SKalle Valo 	struct lib80211_crypto_ops *crypt;
214eb4f98d5SKalle Valo 	void *crypt_priv;
215eb4f98d5SKalle Valo #endif /* PRISM2_NO_KERNEL_IEEE80211_MGMT */
216eb4f98d5SKalle Valo };
217eb4f98d5SKalle Valo 
218eb4f98d5SKalle Valo 
219eb4f98d5SKalle Valo void hostap_rx(struct net_device *dev, struct sk_buff *skb,
220eb4f98d5SKalle Valo 	       struct hostap_80211_rx_status *rx_stats);
221eb4f98d5SKalle Valo void hostap_init_data(local_info_t *local);
222eb4f98d5SKalle Valo void hostap_init_ap_proc(local_info_t *local);
223eb4f98d5SKalle Valo void hostap_free_data(struct ap_data *ap);
224eb4f98d5SKalle Valo void hostap_check_sta_fw_version(struct ap_data *ap, int sta_fw_ver);
225eb4f98d5SKalle Valo 
226eb4f98d5SKalle Valo typedef enum {
227eb4f98d5SKalle Valo 	AP_TX_CONTINUE, AP_TX_DROP, AP_TX_RETRY, AP_TX_BUFFERED,
228eb4f98d5SKalle Valo 	AP_TX_CONTINUE_NOT_AUTHORIZED
229eb4f98d5SKalle Valo } ap_tx_ret;
230eb4f98d5SKalle Valo struct hostap_tx_data {
231eb4f98d5SKalle Valo 	struct sk_buff *skb;
232eb4f98d5SKalle Valo 	int host_encrypt;
233eb4f98d5SKalle Valo 	struct lib80211_crypt_data *crypt;
234eb4f98d5SKalle Valo 	void *sta_ptr;
235eb4f98d5SKalle Valo };
236eb4f98d5SKalle Valo ap_tx_ret hostap_handle_sta_tx(local_info_t *local, struct hostap_tx_data *tx);
237eb4f98d5SKalle Valo void hostap_handle_sta_release(void *ptr);
238eb4f98d5SKalle Valo void hostap_handle_sta_tx_exc(local_info_t *local, struct sk_buff *skb);
239eb4f98d5SKalle Valo int hostap_update_sta_ps(local_info_t *local, struct ieee80211_hdr *hdr);
240eb4f98d5SKalle Valo typedef enum {
241eb4f98d5SKalle Valo 	AP_RX_CONTINUE, AP_RX_DROP, AP_RX_EXIT, AP_RX_CONTINUE_NOT_AUTHORIZED
242eb4f98d5SKalle Valo } ap_rx_ret;
243eb4f98d5SKalle Valo ap_rx_ret hostap_handle_sta_rx(local_info_t *local, struct net_device *dev,
244eb4f98d5SKalle Valo 			       struct sk_buff *skb,
245eb4f98d5SKalle Valo 			       struct hostap_80211_rx_status *rx_stats,
246eb4f98d5SKalle Valo 			       int wds);
247eb4f98d5SKalle Valo int hostap_handle_sta_crypto(local_info_t *local, struct ieee80211_hdr *hdr,
248eb4f98d5SKalle Valo 			     struct lib80211_crypt_data **crypt,
249eb4f98d5SKalle Valo 			     void **sta_ptr);
250eb4f98d5SKalle Valo int hostap_is_sta_assoc(struct ap_data *ap, u8 *sta_addr);
251eb4f98d5SKalle Valo int hostap_is_sta_authorized(struct ap_data *ap, u8 *sta_addr);
252eb4f98d5SKalle Valo int hostap_add_sta(struct ap_data *ap, u8 *sta_addr);
253eb4f98d5SKalle Valo int hostap_update_rx_stats(struct ap_data *ap, struct ieee80211_hdr *hdr,
254eb4f98d5SKalle Valo 			   struct hostap_80211_rx_status *rx_stats);
255eb4f98d5SKalle Valo void hostap_update_rates(local_info_t *local);
256eb4f98d5SKalle Valo void hostap_add_wds_links(local_info_t *local);
257eb4f98d5SKalle Valo void hostap_wds_link_oper(local_info_t *local, u8 *addr, wds_oper_type type);
258eb4f98d5SKalle Valo 
259eb4f98d5SKalle Valo #ifndef PRISM2_NO_KERNEL_IEEE80211_MGMT
260eb4f98d5SKalle Valo void hostap_deauth_all_stas(struct net_device *dev, struct ap_data *ap,
261eb4f98d5SKalle Valo 			    int resend);
262eb4f98d5SKalle Valo #endif /* PRISM2_NO_KERNEL_IEEE80211_MGMT */
263eb4f98d5SKalle Valo 
264eb4f98d5SKalle Valo #endif /* HOSTAP_AP_H */
265