1*4882a593Smuzhiyun /* SPDX-License-Identifier: GPL-2.0-only */
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * PS3 gelic network driver.
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Copyright (C) 2007 Sony Computer Entertainment Inc.
6*4882a593Smuzhiyun * Copyright 2007 Sony Corporation
7*4882a593Smuzhiyun */
8*4882a593Smuzhiyun #ifndef _GELIC_WIRELESS_H
9*4882a593Smuzhiyun #define _GELIC_WIRELESS_H
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun #include <linux/wireless.h>
12*4882a593Smuzhiyun #include <net/iw_handler.h>
13*4882a593Smuzhiyun
14*4882a593Smuzhiyun
15*4882a593Smuzhiyun /* return value from GELIC_LV1_GET_WLAN_EVENT netcontrol */
16*4882a593Smuzhiyun enum gelic_lv1_wl_event {
17*4882a593Smuzhiyun GELIC_LV1_WL_EVENT_DEVICE_READY = 0x01, /* Eurus ready */
18*4882a593Smuzhiyun GELIC_LV1_WL_EVENT_SCAN_COMPLETED = 0x02, /* Scan has completed */
19*4882a593Smuzhiyun GELIC_LV1_WL_EVENT_DEAUTH = 0x04, /* Deauthed by the AP */
20*4882a593Smuzhiyun GELIC_LV1_WL_EVENT_BEACON_LOST = 0x08, /* Beacon lost detected */
21*4882a593Smuzhiyun GELIC_LV1_WL_EVENT_CONNECTED = 0x10, /* Connected to AP */
22*4882a593Smuzhiyun GELIC_LV1_WL_EVENT_WPA_CONNECTED = 0x20, /* WPA connection */
23*4882a593Smuzhiyun GELIC_LV1_WL_EVENT_WPA_ERROR = 0x40, /* MIC error */
24*4882a593Smuzhiyun };
25*4882a593Smuzhiyun
26*4882a593Smuzhiyun /* arguments for GELIC_LV1_POST_WLAN_COMMAND netcontrol */
27*4882a593Smuzhiyun enum gelic_eurus_command {
28*4882a593Smuzhiyun GELIC_EURUS_CMD_ASSOC = 1, /* association start */
29*4882a593Smuzhiyun GELIC_EURUS_CMD_DISASSOC = 2, /* disassociate */
30*4882a593Smuzhiyun GELIC_EURUS_CMD_START_SCAN = 3, /* scan start */
31*4882a593Smuzhiyun GELIC_EURUS_CMD_GET_SCAN = 4, /* get scan result */
32*4882a593Smuzhiyun GELIC_EURUS_CMD_SET_COMMON_CFG = 5, /* set common config */
33*4882a593Smuzhiyun GELIC_EURUS_CMD_GET_COMMON_CFG = 6, /* set common config */
34*4882a593Smuzhiyun GELIC_EURUS_CMD_SET_WEP_CFG = 7, /* set WEP config */
35*4882a593Smuzhiyun GELIC_EURUS_CMD_GET_WEP_CFG = 8, /* get WEP config */
36*4882a593Smuzhiyun GELIC_EURUS_CMD_SET_WPA_CFG = 9, /* set WPA config */
37*4882a593Smuzhiyun GELIC_EURUS_CMD_GET_WPA_CFG = 10, /* get WPA config */
38*4882a593Smuzhiyun GELIC_EURUS_CMD_GET_RSSI_CFG = 11, /* get RSSI info. */
39*4882a593Smuzhiyun GELIC_EURUS_CMD_MAX_INDEX
40*4882a593Smuzhiyun };
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun /* for GELIC_EURUS_CMD_COMMON_CFG */
43*4882a593Smuzhiyun enum gelic_eurus_bss_type {
44*4882a593Smuzhiyun GELIC_EURUS_BSS_INFRA = 0,
45*4882a593Smuzhiyun GELIC_EURUS_BSS_ADHOC = 1, /* not supported */
46*4882a593Smuzhiyun };
47*4882a593Smuzhiyun
48*4882a593Smuzhiyun enum gelic_eurus_auth_method {
49*4882a593Smuzhiyun GELIC_EURUS_AUTH_OPEN = 0, /* FIXME: WLAN_AUTH_OPEN */
50*4882a593Smuzhiyun GELIC_EURUS_AUTH_SHARED = 1, /* not supported */
51*4882a593Smuzhiyun };
52*4882a593Smuzhiyun
53*4882a593Smuzhiyun enum gelic_eurus_opmode {
54*4882a593Smuzhiyun GELIC_EURUS_OPMODE_11BG = 0, /* 802.11b/g */
55*4882a593Smuzhiyun GELIC_EURUS_OPMODE_11B = 1, /* 802.11b only */
56*4882a593Smuzhiyun GELIC_EURUS_OPMODE_11G = 2, /* 802.11g only */
57*4882a593Smuzhiyun };
58*4882a593Smuzhiyun
59*4882a593Smuzhiyun struct gelic_eurus_common_cfg {
60*4882a593Smuzhiyun /* all fields are big endian */
61*4882a593Smuzhiyun u16 scan_index;
62*4882a593Smuzhiyun u16 bss_type; /* infra or adhoc */
63*4882a593Smuzhiyun u16 auth_method; /* shared key or open */
64*4882a593Smuzhiyun u16 op_mode; /* B/G */
65*4882a593Smuzhiyun } __packed;
66*4882a593Smuzhiyun
67*4882a593Smuzhiyun
68*4882a593Smuzhiyun /* for GELIC_EURUS_CMD_WEP_CFG */
69*4882a593Smuzhiyun enum gelic_eurus_wep_security {
70*4882a593Smuzhiyun GELIC_EURUS_WEP_SEC_NONE = 0,
71*4882a593Smuzhiyun GELIC_EURUS_WEP_SEC_40BIT = 1,
72*4882a593Smuzhiyun GELIC_EURUS_WEP_SEC_104BIT = 2,
73*4882a593Smuzhiyun };
74*4882a593Smuzhiyun
75*4882a593Smuzhiyun struct gelic_eurus_wep_cfg {
76*4882a593Smuzhiyun /* all fields are big endian */
77*4882a593Smuzhiyun u16 security;
78*4882a593Smuzhiyun u8 key[4][16];
79*4882a593Smuzhiyun } __packed;
80*4882a593Smuzhiyun
81*4882a593Smuzhiyun /* for GELIC_EURUS_CMD_WPA_CFG */
82*4882a593Smuzhiyun enum gelic_eurus_wpa_security {
83*4882a593Smuzhiyun GELIC_EURUS_WPA_SEC_NONE = 0x0000,
84*4882a593Smuzhiyun /* group=TKIP, pairwise=TKIP */
85*4882a593Smuzhiyun GELIC_EURUS_WPA_SEC_WPA_TKIP_TKIP = 0x0001,
86*4882a593Smuzhiyun /* group=AES, pairwise=AES */
87*4882a593Smuzhiyun GELIC_EURUS_WPA_SEC_WPA_AES_AES = 0x0002,
88*4882a593Smuzhiyun /* group=TKIP, pairwise=TKIP */
89*4882a593Smuzhiyun GELIC_EURUS_WPA_SEC_WPA2_TKIP_TKIP = 0x0004,
90*4882a593Smuzhiyun /* group=AES, pairwise=AES */
91*4882a593Smuzhiyun GELIC_EURUS_WPA_SEC_WPA2_AES_AES = 0x0008,
92*4882a593Smuzhiyun /* group=TKIP, pairwise=AES */
93*4882a593Smuzhiyun GELIC_EURUS_WPA_SEC_WPA_TKIP_AES = 0x0010,
94*4882a593Smuzhiyun /* group=TKIP, pairwise=AES */
95*4882a593Smuzhiyun GELIC_EURUS_WPA_SEC_WPA2_TKIP_AES = 0x0020,
96*4882a593Smuzhiyun };
97*4882a593Smuzhiyun
98*4882a593Smuzhiyun enum gelic_eurus_wpa_psk_type {
99*4882a593Smuzhiyun GELIC_EURUS_WPA_PSK_PASSPHRASE = 0, /* passphrase string */
100*4882a593Smuzhiyun GELIC_EURUS_WPA_PSK_BIN = 1, /* 32 bytes binary key */
101*4882a593Smuzhiyun };
102*4882a593Smuzhiyun
103*4882a593Smuzhiyun #define GELIC_WL_EURUS_PSK_MAX_LEN 64
104*4882a593Smuzhiyun #define WPA_PSK_LEN 32 /* WPA spec says 256bit */
105*4882a593Smuzhiyun
106*4882a593Smuzhiyun struct gelic_eurus_wpa_cfg {
107*4882a593Smuzhiyun /* all fields are big endian */
108*4882a593Smuzhiyun u16 security;
109*4882a593Smuzhiyun u16 psk_type; /* psk key encoding type */
110*4882a593Smuzhiyun u8 psk[GELIC_WL_EURUS_PSK_MAX_LEN]; /* psk key; hex or passphrase */
111*4882a593Smuzhiyun } __packed;
112*4882a593Smuzhiyun
113*4882a593Smuzhiyun /* for GELIC_EURUS_CMD_{START,GET}_SCAN */
114*4882a593Smuzhiyun enum gelic_eurus_scan_capability {
115*4882a593Smuzhiyun GELIC_EURUS_SCAN_CAP_ADHOC = 0x0000,
116*4882a593Smuzhiyun GELIC_EURUS_SCAN_CAP_INFRA = 0x0001,
117*4882a593Smuzhiyun GELIC_EURUS_SCAN_CAP_MASK = 0x0001,
118*4882a593Smuzhiyun };
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun enum gelic_eurus_scan_sec_type {
121*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_NONE = 0x0000,
122*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WEP = 0x0100,
123*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WPA = 0x0200,
124*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WPA2 = 0x0400,
125*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_MASK = 0x0f00,
126*4882a593Smuzhiyun };
127*4882a593Smuzhiyun
128*4882a593Smuzhiyun enum gelic_eurus_scan_sec_wep_type {
129*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WEP_UNKNOWN = 0x0000,
130*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WEP_40 = 0x0001,
131*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WEP_104 = 0x0002,
132*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WEP_MASK = 0x0003,
133*4882a593Smuzhiyun };
134*4882a593Smuzhiyun
135*4882a593Smuzhiyun enum gelic_eurus_scan_sec_wpa_type {
136*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WPA_UNKNOWN = 0x0000,
137*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WPA_TKIP = 0x0001,
138*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WPA_AES = 0x0002,
139*4882a593Smuzhiyun GELIC_EURUS_SCAN_SEC_WPA_MASK = 0x0003,
140*4882a593Smuzhiyun };
141*4882a593Smuzhiyun
142*4882a593Smuzhiyun /*
143*4882a593Smuzhiyun * hw BSS information structure returned from GELIC_EURUS_CMD_GET_SCAN
144*4882a593Smuzhiyun */
145*4882a593Smuzhiyun struct gelic_eurus_scan_info {
146*4882a593Smuzhiyun /* all fields are big endian */
147*4882a593Smuzhiyun __be16 size;
148*4882a593Smuzhiyun __be16 rssi; /* percentage */
149*4882a593Smuzhiyun __be16 channel; /* channel number */
150*4882a593Smuzhiyun __be16 beacon_period; /* FIXME: in msec unit */
151*4882a593Smuzhiyun __be16 capability;
152*4882a593Smuzhiyun __be16 security;
153*4882a593Smuzhiyun u8 bssid[8]; /* last ETH_ALEN are valid. bssid[0],[1] are unused */
154*4882a593Smuzhiyun u8 essid[32]; /* IW_ESSID_MAX_SIZE */
155*4882a593Smuzhiyun u8 rate[16]; /* first 12 are valid */
156*4882a593Smuzhiyun u8 ext_rate[16]; /* first 16 are valid */
157*4882a593Smuzhiyun __be32 reserved1;
158*4882a593Smuzhiyun __be32 reserved2;
159*4882a593Smuzhiyun __be32 reserved3;
160*4882a593Smuzhiyun __be32 reserved4;
161*4882a593Smuzhiyun u8 elements[]; /* ie */
162*4882a593Smuzhiyun } __packed;
163*4882a593Smuzhiyun
164*4882a593Smuzhiyun /* the hypervisor returns bbs up to 16 */
165*4882a593Smuzhiyun #define GELIC_EURUS_MAX_SCAN (16)
166*4882a593Smuzhiyun struct gelic_wl_scan_info {
167*4882a593Smuzhiyun struct list_head list;
168*4882a593Smuzhiyun struct gelic_eurus_scan_info *hwinfo;
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun int valid; /* set 1 if this entry was in latest scanned list
171*4882a593Smuzhiyun * from Eurus */
172*4882a593Smuzhiyun unsigned int eurus_index; /* index in the Eurus list */
173*4882a593Smuzhiyun unsigned long last_scanned; /* acquired time */
174*4882a593Smuzhiyun
175*4882a593Smuzhiyun unsigned int rate_len;
176*4882a593Smuzhiyun unsigned int rate_ext_len;
177*4882a593Smuzhiyun unsigned int essid_len;
178*4882a593Smuzhiyun };
179*4882a593Smuzhiyun
180*4882a593Smuzhiyun /* for GELIC_EURUS_CMD_GET_RSSI */
181*4882a593Smuzhiyun struct gelic_eurus_rssi_info {
182*4882a593Smuzhiyun /* big endian */
183*4882a593Smuzhiyun __be16 rssi;
184*4882a593Smuzhiyun } __packed;
185*4882a593Smuzhiyun
186*4882a593Smuzhiyun
187*4882a593Smuzhiyun /* for 'stat' member of gelic_wl_info */
188*4882a593Smuzhiyun enum gelic_wl_info_status_bit {
189*4882a593Smuzhiyun GELIC_WL_STAT_CONFIGURED,
190*4882a593Smuzhiyun GELIC_WL_STAT_CH_INFO, /* ch info acquired */
191*4882a593Smuzhiyun GELIC_WL_STAT_ESSID_SET, /* ESSID specified by userspace */
192*4882a593Smuzhiyun GELIC_WL_STAT_BSSID_SET, /* BSSID specified by userspace */
193*4882a593Smuzhiyun GELIC_WL_STAT_WPA_PSK_SET, /* PMK specified by userspace */
194*4882a593Smuzhiyun GELIC_WL_STAT_WPA_LEVEL_SET, /* WEP or WPA[2] selected */
195*4882a593Smuzhiyun };
196*4882a593Smuzhiyun
197*4882a593Smuzhiyun /* for 'scan_stat' member of gelic_wl_info */
198*4882a593Smuzhiyun enum gelic_wl_scan_state {
199*4882a593Smuzhiyun /* just initialized or get last scan result failed */
200*4882a593Smuzhiyun GELIC_WL_SCAN_STAT_INIT,
201*4882a593Smuzhiyun /* scan request issued, accepted or chip is scanning */
202*4882a593Smuzhiyun GELIC_WL_SCAN_STAT_SCANNING,
203*4882a593Smuzhiyun /* scan results retrieved */
204*4882a593Smuzhiyun GELIC_WL_SCAN_STAT_GOT_LIST,
205*4882a593Smuzhiyun };
206*4882a593Smuzhiyun
207*4882a593Smuzhiyun /* for 'cipher_method' */
208*4882a593Smuzhiyun enum gelic_wl_cipher_method {
209*4882a593Smuzhiyun GELIC_WL_CIPHER_NONE,
210*4882a593Smuzhiyun GELIC_WL_CIPHER_WEP,
211*4882a593Smuzhiyun GELIC_WL_CIPHER_TKIP,
212*4882a593Smuzhiyun GELIC_WL_CIPHER_AES,
213*4882a593Smuzhiyun };
214*4882a593Smuzhiyun
215*4882a593Smuzhiyun /* for 'wpa_level' */
216*4882a593Smuzhiyun enum gelic_wl_wpa_level {
217*4882a593Smuzhiyun GELIC_WL_WPA_LEVEL_NONE,
218*4882a593Smuzhiyun GELIC_WL_WPA_LEVEL_WPA,
219*4882a593Smuzhiyun GELIC_WL_WPA_LEVEL_WPA2,
220*4882a593Smuzhiyun };
221*4882a593Smuzhiyun
222*4882a593Smuzhiyun /* for 'assoc_stat' */
223*4882a593Smuzhiyun enum gelic_wl_assoc_state {
224*4882a593Smuzhiyun GELIC_WL_ASSOC_STAT_DISCONN,
225*4882a593Smuzhiyun GELIC_WL_ASSOC_STAT_ASSOCIATING,
226*4882a593Smuzhiyun GELIC_WL_ASSOC_STAT_ASSOCIATED,
227*4882a593Smuzhiyun };
228*4882a593Smuzhiyun /* part of private data alloc_etherdev() allocated */
229*4882a593Smuzhiyun #define GELIC_WEP_KEYS 4
230*4882a593Smuzhiyun struct gelic_wl_info {
231*4882a593Smuzhiyun /* bss list */
232*4882a593Smuzhiyun struct mutex scan_lock;
233*4882a593Smuzhiyun struct list_head network_list;
234*4882a593Smuzhiyun struct list_head network_free_list;
235*4882a593Smuzhiyun struct gelic_wl_scan_info *networks;
236*4882a593Smuzhiyun
237*4882a593Smuzhiyun unsigned long scan_age; /* last scanned time */
238*4882a593Smuzhiyun enum gelic_wl_scan_state scan_stat;
239*4882a593Smuzhiyun struct completion scan_done;
240*4882a593Smuzhiyun
241*4882a593Smuzhiyun /* eurus command queue */
242*4882a593Smuzhiyun struct workqueue_struct *eurus_cmd_queue;
243*4882a593Smuzhiyun struct completion cmd_done_intr;
244*4882a593Smuzhiyun
245*4882a593Smuzhiyun /* eurus event handling */
246*4882a593Smuzhiyun struct workqueue_struct *event_queue;
247*4882a593Smuzhiyun struct delayed_work event_work;
248*4882a593Smuzhiyun
249*4882a593Smuzhiyun /* wl status bits */
250*4882a593Smuzhiyun unsigned long stat;
251*4882a593Smuzhiyun enum gelic_eurus_auth_method auth_method; /* open/shared */
252*4882a593Smuzhiyun enum gelic_wl_cipher_method group_cipher_method;
253*4882a593Smuzhiyun enum gelic_wl_cipher_method pairwise_cipher_method;
254*4882a593Smuzhiyun enum gelic_wl_wpa_level wpa_level; /* wpa/wpa2 */
255*4882a593Smuzhiyun
256*4882a593Smuzhiyun /* association handling */
257*4882a593Smuzhiyun struct mutex assoc_stat_lock;
258*4882a593Smuzhiyun struct delayed_work assoc_work;
259*4882a593Smuzhiyun enum gelic_wl_assoc_state assoc_stat;
260*4882a593Smuzhiyun struct completion assoc_done;
261*4882a593Smuzhiyun
262*4882a593Smuzhiyun spinlock_t lock;
263*4882a593Smuzhiyun u16 ch_info; /* available channels. bit0 = ch1 */
264*4882a593Smuzhiyun /* WEP keys */
265*4882a593Smuzhiyun u8 key[GELIC_WEP_KEYS][IW_ENCODING_TOKEN_MAX];
266*4882a593Smuzhiyun unsigned long key_enabled;
267*4882a593Smuzhiyun unsigned int key_len[GELIC_WEP_KEYS];
268*4882a593Smuzhiyun unsigned int current_key;
269*4882a593Smuzhiyun /* WWPA PSK */
270*4882a593Smuzhiyun u8 psk[GELIC_WL_EURUS_PSK_MAX_LEN];
271*4882a593Smuzhiyun enum gelic_eurus_wpa_psk_type psk_type;
272*4882a593Smuzhiyun unsigned int psk_len;
273*4882a593Smuzhiyun
274*4882a593Smuzhiyun u8 essid[IW_ESSID_MAX_SIZE];
275*4882a593Smuzhiyun u8 bssid[ETH_ALEN]; /* userland requested */
276*4882a593Smuzhiyun u8 active_bssid[ETH_ALEN]; /* associated bssid */
277*4882a593Smuzhiyun unsigned int essid_len;
278*4882a593Smuzhiyun
279*4882a593Smuzhiyun struct iw_public_data wireless_data;
280*4882a593Smuzhiyun struct iw_statistics iwstat;
281*4882a593Smuzhiyun };
282*4882a593Smuzhiyun
283*4882a593Smuzhiyun #define GELIC_WL_BSS_MAX_ENT 32
284*4882a593Smuzhiyun #define GELIC_WL_ASSOC_RETRY 50
wl_port(struct gelic_wl_info * wl)285*4882a593Smuzhiyun static inline struct gelic_port *wl_port(struct gelic_wl_info *wl)
286*4882a593Smuzhiyun {
287*4882a593Smuzhiyun return container_of((void *)wl, struct gelic_port, priv);
288*4882a593Smuzhiyun }
port_wl(struct gelic_port * port)289*4882a593Smuzhiyun static inline struct gelic_wl_info *port_wl(struct gelic_port *port)
290*4882a593Smuzhiyun {
291*4882a593Smuzhiyun return port_priv(port);
292*4882a593Smuzhiyun }
293*4882a593Smuzhiyun
294*4882a593Smuzhiyun struct gelic_eurus_cmd {
295*4882a593Smuzhiyun struct work_struct work;
296*4882a593Smuzhiyun struct gelic_wl_info *wl;
297*4882a593Smuzhiyun unsigned int cmd; /* command code */
298*4882a593Smuzhiyun u64 tag;
299*4882a593Smuzhiyun u64 size;
300*4882a593Smuzhiyun void *buffer;
301*4882a593Smuzhiyun unsigned int buf_size;
302*4882a593Smuzhiyun struct completion done;
303*4882a593Smuzhiyun int status;
304*4882a593Smuzhiyun u64 cmd_status;
305*4882a593Smuzhiyun };
306*4882a593Smuzhiyun
307*4882a593Smuzhiyun /* private ioctls to pass PSK */
308*4882a593Smuzhiyun #define GELIC_WL_PRIV_SET_PSK (SIOCIWFIRSTPRIV + 0)
309*4882a593Smuzhiyun #define GELIC_WL_PRIV_GET_PSK (SIOCIWFIRSTPRIV + 1)
310*4882a593Smuzhiyun
311*4882a593Smuzhiyun int gelic_wl_driver_probe(struct gelic_card *card);
312*4882a593Smuzhiyun int gelic_wl_driver_remove(struct gelic_card *card);
313*4882a593Smuzhiyun void gelic_wl_interrupt(struct net_device *netdev, u64 status);
314*4882a593Smuzhiyun #endif /* _GELIC_WIRELESS_H */
315