1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * bcmevent read-only data shared by kernel or app layers
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Copyright (C) 2020, Broadcom.
5*4882a593Smuzhiyun *
6*4882a593Smuzhiyun * Unless you and Broadcom execute a separate written software license
7*4882a593Smuzhiyun * agreement governing use of this software, this software is licensed to you
8*4882a593Smuzhiyun * under the terms of the GNU General Public License version 2 (the "GPL"),
9*4882a593Smuzhiyun * available at http://www.broadcom.com/licenses/GPLv2.php, with the
10*4882a593Smuzhiyun * following added to such license:
11*4882a593Smuzhiyun *
12*4882a593Smuzhiyun * As a special exception, the copyright holders of this software give you
13*4882a593Smuzhiyun * permission to link this software with independent modules, and to copy and
14*4882a593Smuzhiyun * distribute the resulting executable under terms of your choice, provided that
15*4882a593Smuzhiyun * you also meet, for each linked independent module, the terms and conditions of
16*4882a593Smuzhiyun * the license of that module. An independent module is a module which is not
17*4882a593Smuzhiyun * derived from this software. The special exception does not apply to any
18*4882a593Smuzhiyun * modifications of the software.
19*4882a593Smuzhiyun *
20*4882a593Smuzhiyun *
21*4882a593Smuzhiyun * <<Broadcom-WL-IPTag/Dual:>>
22*4882a593Smuzhiyun */
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun #include <typedefs.h>
25*4882a593Smuzhiyun #include <bcmutils.h>
26*4882a593Smuzhiyun #include <bcmendian.h>
27*4882a593Smuzhiyun #include <ethernet.h>
28*4882a593Smuzhiyun #include <bcmeth.h>
29*4882a593Smuzhiyun #include <bcmevent.h>
30*4882a593Smuzhiyun #include <802.11.h>
31*4882a593Smuzhiyun
32*4882a593Smuzhiyun /* Table of event name strings for UIs and debugging dumps */
33*4882a593Smuzhiyun typedef struct {
34*4882a593Smuzhiyun uint event;
35*4882a593Smuzhiyun const char *name;
36*4882a593Smuzhiyun } bcmevent_name_str_t;
37*4882a593Smuzhiyun
38*4882a593Smuzhiyun /* Use the actual name for event tracing */
39*4882a593Smuzhiyun #define BCMEVENT_NAME(_event) {(_event), #_event}
40*4882a593Smuzhiyun
41*4882a593Smuzhiyun /* this becomes static data when all code is changed to use
42*4882a593Smuzhiyun * the bcmevent_get_name() API
43*4882a593Smuzhiyun */
44*4882a593Smuzhiyun static const bcmevent_name_str_t bcmevent_names[] = {
45*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_SET_SSID),
46*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_JOIN),
47*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_START),
48*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AUTH),
49*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AUTH_IND),
50*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DEAUTH),
51*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DEAUTH_IND),
52*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ASSOC),
53*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ASSOC_IND),
54*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_REASSOC),
55*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_REASSOC_IND),
56*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DISASSOC),
57*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DISASSOC_IND),
58*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_QUIET_START),
59*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_QUIET_END),
60*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_BEACON_RX),
61*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_LINK),
62*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_MIC_ERROR),
63*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_NDIS_LINK),
64*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ROAM),
65*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_TXFAIL),
66*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PMKID_CACHE),
67*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_RETROGRADE_TSF),
68*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PRUNE),
69*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AUTOAUTH),
70*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_EAPOL_MSG),
71*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_SCAN_COMPLETE),
72*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ADDTS_IND),
73*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DELTS_IND),
74*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_BCNSENT_IND),
75*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_BCNRX_MSG),
76*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_BCNLOST_MSG),
77*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ROAM_PREP),
78*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_NET_FOUND),
79*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_SCAN_ALLGONE),
80*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_NET_LOST),
81*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_JOIN_START),
82*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ROAM_START),
83*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ASSOC_START),
84*4882a593Smuzhiyun #ifdef EXT_STA
85*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_RESET_COMPLETE),
86*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_JOIN_START),
87*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ROAM_START),
88*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ASSOC_START),
89*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ASSOC_RECREATED),
90*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_SPEEDY_RECREATE_FAIL),
91*4882a593Smuzhiyun #endif /* EXT_STA */
92*4882a593Smuzhiyun #if defined(IBSS_PEER_DISCOVERY_EVENT)
93*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_IBSS_ASSOC),
94*4882a593Smuzhiyun #endif /* defined(IBSS_PEER_DISCOVERY_EVENT) */
95*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_RADIO),
96*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PSM_WATCHDOG),
97*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PROBREQ_MSG),
98*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_SCAN_CONFIRM_IND),
99*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PSK_SUP),
100*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_COUNTRY_CODE_CHANGED),
101*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_EXCEEDED_MEDIUM_TIME),
102*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ICV_ERROR),
103*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_UNICAST_DECODE_ERROR),
104*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_MULTICAST_DECODE_ERROR),
105*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_TRACE),
106*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_IF),
107*4882a593Smuzhiyun #ifdef WLP2P
108*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_P2P_DISC_LISTEN_COMPLETE),
109*4882a593Smuzhiyun #endif
110*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_RSSI),
111*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_SCAN_COMPLETE),
112*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ACTION_FRAME),
113*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ACTION_FRAME_RX),
114*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ACTION_FRAME_COMPLETE),
115*4882a593Smuzhiyun #if defined(NDIS)
116*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PRE_ASSOC_IND),
117*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PRE_REASSOC_IND),
118*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_CHANNEL_ADOPTED),
119*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AP_STARTED),
120*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DFS_AP_STOP),
121*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DFS_AP_RESUME),
122*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ASSOC_IND_NDIS),
123*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_REASSOC_IND_NDIS),
124*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ACTION_FRAME_RX_NDIS),
125*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AUTH_REQ),
126*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_IBSS_COALESCE),
127*4882a593Smuzhiyun #endif /* #if defined(NDIS) */
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun #ifdef BCMWAPI_WAI
130*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_WAI_STA_EVENT),
131*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_WAI_MSG),
132*4882a593Smuzhiyun #endif /* BCMWAPI_WAI */
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ESCAN_RESULT),
135*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ACTION_FRAME_OFF_CHAN_COMPLETE),
136*4882a593Smuzhiyun #ifdef WLP2P
137*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PROBRESP_MSG),
138*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_P2P_PROBREQ_MSG),
139*4882a593Smuzhiyun #endif
140*4882a593Smuzhiyun #ifdef PROP_TXSTATUS
141*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_FIFO_CREDIT_MAP),
142*4882a593Smuzhiyun #endif
143*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_WAKE_EVENT),
144*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DCS_REQUEST),
145*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_RM_COMPLETE),
146*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_OVERLAY_REQ),
147*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_CSA_COMPLETE_IND),
148*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_EXCESS_PM_WAKE_EVENT),
149*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_SCAN_NONE),
150*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_SCAN_ALLGONE),
151*4882a593Smuzhiyun #ifdef SOFTAP
152*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_GTK_PLUMBED),
153*4882a593Smuzhiyun #endif
154*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ASSOC_REQ_IE),
155*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ASSOC_RESP_IE),
156*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_BEACON_FRAME_RX),
157*4882a593Smuzhiyun #ifdef WLTDLS
158*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_TDLS_PEER_EVENT),
159*4882a593Smuzhiyun #endif /* WLTDLS */
160*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_NATIVE),
161*4882a593Smuzhiyun #ifdef WLPKTDLYSTAT
162*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PKTDELAY_IND),
163*4882a593Smuzhiyun #endif /* WLPKTDLYSTAT */
164*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_SERVICE_FOUND),
165*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_GAS_FRAGMENT_RX),
166*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_GAS_COMPLETE),
167*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_P2PO_ADD_DEVICE),
168*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_P2PO_DEL_DEVICE),
169*4882a593Smuzhiyun #ifdef WLWNM
170*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_WNM_STA_SLEEP),
171*4882a593Smuzhiyun #endif /* WLWNM */
172*4882a593Smuzhiyun #if defined(WL_PROXDETECT) || defined(RTT_SUPPORT)
173*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PROXD),
174*4882a593Smuzhiyun #endif
175*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_CCA_CHAN_QUAL),
176*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_BSSID),
177*4882a593Smuzhiyun #ifdef PROP_TXSTATUS
178*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_BCMC_CREDIT_SUPPORT),
179*4882a593Smuzhiyun #endif
180*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PSTA_PRIMARY_INTF_IND),
181*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_TXFAIL_THRESH),
182*4882a593Smuzhiyun #ifdef WLAIBSS
183*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AIBSS_TXFAIL),
184*4882a593Smuzhiyun #endif /* WLAIBSS */
185*4882a593Smuzhiyun #ifdef GSCAN_SUPPORT
186*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_GSCAN_FULL_RESULT),
187*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_SSID_EXT),
188*4882a593Smuzhiyun #endif /* GSCAN_SUPPORT */
189*4882a593Smuzhiyun #ifdef WLBSSLOAD_REPORT
190*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_BSS_LOAD),
191*4882a593Smuzhiyun #endif
192*4882a593Smuzhiyun #if defined(BT_WIFI_HANDOVER) || defined(WL_TBOW)
193*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_BT_WIFI_HANDOVER_REQ),
194*4882a593Smuzhiyun #endif
195*4882a593Smuzhiyun #ifdef WLFBT
196*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_FBT),
197*4882a593Smuzhiyun #endif /* WLFBT */
198*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AUTHORIZED),
199*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PROBREQ_MSG_RX),
200*4882a593Smuzhiyun
201*4882a593Smuzhiyun #ifdef WLAWDL
202*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AWDL_AW),
203*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AWDL_ROLE),
204*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AWDL_EVENT),
205*4882a593Smuzhiyun #endif /* WLAWDL */
206*4882a593Smuzhiyun
207*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_CSA_START_IND),
208*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_CSA_DONE_IND),
209*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_CSA_FAILURE_IND),
210*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_RMC_EVENT),
211*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DPSTA_INTF_IND),
212*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ALLOW_CREDIT_BORROW),
213*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_MSCH),
214*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ULP),
215*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_NAN),
216*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PKT_FILTER),
217*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DMA_TXFLUSH_COMPLETE),
218*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PSK_AUTH),
219*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_SDB_TRANSITION),
220*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_SCAN_BACKOFF),
221*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_BSSID_SCAN_BACKOFF),
222*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AGGR_EVENT),
223*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_TVPM_MITIGATION),
224*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_SCAN),
225*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_SLOTTED_BSS_PEER_OP),
226*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PHY_CAL),
227*4882a593Smuzhiyun #ifdef WL_NAN
228*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_NAN_CRITICAL),
229*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_NAN_NON_CRITICAL),
230*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_NAN),
231*4882a593Smuzhiyun #endif /* WL_NAN */
232*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_RPSNOA),
233*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_WA_LQM),
234*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_OBSS_DETECTION),
235*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_SC_CHAN_QUAL),
236*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_DYNSAR),
237*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_ROAM_CACHE_UPDATE),
238*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AP_BCN_DRIFT),
239*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_PFN_SCAN_ALLGONE_EXT),
240*4882a593Smuzhiyun #ifdef WL_CLIENT_SAE
241*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_AUTH_START),
242*4882a593Smuzhiyun #endif /* WL_CLIENT_SAE */
243*4882a593Smuzhiyun #ifdef WL_TWT
244*4882a593Smuzhiyun BCMEVENT_NAME(WLC_E_TWT),
245*4882a593Smuzhiyun #endif /* WL_TWT */
246*4882a593Smuzhiyun };
247*4882a593Smuzhiyun
bcmevent_get_name(uint event_type)248*4882a593Smuzhiyun const char *bcmevent_get_name(uint event_type)
249*4882a593Smuzhiyun {
250*4882a593Smuzhiyun /* note: first coded this as a static const but some
251*4882a593Smuzhiyun * ROMs already have something called event_name so
252*4882a593Smuzhiyun * changed it so we don't have a variable for the
253*4882a593Smuzhiyun * 'unknown string
254*4882a593Smuzhiyun */
255*4882a593Smuzhiyun const char *event_name = NULL;
256*4882a593Smuzhiyun
257*4882a593Smuzhiyun uint idx;
258*4882a593Smuzhiyun for (idx = 0; idx < (uint)ARRAYSIZE(bcmevent_names); idx++) {
259*4882a593Smuzhiyun
260*4882a593Smuzhiyun if (bcmevent_names[idx].event == event_type) {
261*4882a593Smuzhiyun event_name = bcmevent_names[idx].name;
262*4882a593Smuzhiyun break;
263*4882a593Smuzhiyun }
264*4882a593Smuzhiyun }
265*4882a593Smuzhiyun
266*4882a593Smuzhiyun /* if we find an event name in the array, return it.
267*4882a593Smuzhiyun * otherwise return unknown string.
268*4882a593Smuzhiyun */
269*4882a593Smuzhiyun return ((event_name) ? event_name : "Unknown Event");
270*4882a593Smuzhiyun }
271*4882a593Smuzhiyun
272*4882a593Smuzhiyun void
wl_event_to_host_order(wl_event_msg_t * evt)273*4882a593Smuzhiyun wl_event_to_host_order(wl_event_msg_t * evt)
274*4882a593Smuzhiyun {
275*4882a593Smuzhiyun /* Event struct members passed from dongle to host are stored in network
276*4882a593Smuzhiyun * byte order. Convert all members to host-order.
277*4882a593Smuzhiyun */
278*4882a593Smuzhiyun evt->event_type = ntoh32(evt->event_type);
279*4882a593Smuzhiyun evt->flags = ntoh16(evt->flags);
280*4882a593Smuzhiyun evt->status = ntoh32(evt->status);
281*4882a593Smuzhiyun evt->reason = ntoh32(evt->reason);
282*4882a593Smuzhiyun evt->auth_type = ntoh32(evt->auth_type);
283*4882a593Smuzhiyun evt->datalen = ntoh32(evt->datalen);
284*4882a593Smuzhiyun evt->version = ntoh16(evt->version);
285*4882a593Smuzhiyun }
286*4882a593Smuzhiyun
287*4882a593Smuzhiyun void
wl_event_to_network_order(wl_event_msg_t * evt)288*4882a593Smuzhiyun wl_event_to_network_order(wl_event_msg_t * evt)
289*4882a593Smuzhiyun {
290*4882a593Smuzhiyun /* Event struct members passed from dongle to host are stored in network
291*4882a593Smuzhiyun * byte order. Convert all members to host-order.
292*4882a593Smuzhiyun */
293*4882a593Smuzhiyun evt->event_type = hton32(evt->event_type);
294*4882a593Smuzhiyun evt->flags = hton16(evt->flags);
295*4882a593Smuzhiyun evt->status = hton32(evt->status);
296*4882a593Smuzhiyun evt->reason = hton32(evt->reason);
297*4882a593Smuzhiyun evt->auth_type = hton32(evt->auth_type);
298*4882a593Smuzhiyun evt->datalen = hton32(evt->datalen);
299*4882a593Smuzhiyun evt->version = hton16(evt->version);
300*4882a593Smuzhiyun }
301*4882a593Smuzhiyun
302*4882a593Smuzhiyun /*
303*4882a593Smuzhiyun * Validate if the event is proper and if valid copy event header to event.
304*4882a593Smuzhiyun * If proper event pointer is passed, to just validate, pass NULL to event.
305*4882a593Smuzhiyun *
306*4882a593Smuzhiyun * Return values are
307*4882a593Smuzhiyun * BCME_OK - It is a BRCM event or BRCM dongle event
308*4882a593Smuzhiyun * BCME_NOTFOUND - Not BRCM, not an event, may be okay
309*4882a593Smuzhiyun * BCME_BADLEN - Bad length, should not process, just drop
310*4882a593Smuzhiyun */
311*4882a593Smuzhiyun int
is_wlc_event_frame(void * pktdata,uint pktlen,uint16 exp_usr_subtype,bcm_event_msg_u_t * out_event)312*4882a593Smuzhiyun is_wlc_event_frame(void *pktdata, uint pktlen, uint16 exp_usr_subtype,
313*4882a593Smuzhiyun bcm_event_msg_u_t *out_event)
314*4882a593Smuzhiyun {
315*4882a593Smuzhiyun uint16 evlen = 0; /* length in bcmeth_hdr */
316*4882a593Smuzhiyun uint16 subtype;
317*4882a593Smuzhiyun uint16 usr_subtype;
318*4882a593Smuzhiyun bcm_event_t *bcm_event;
319*4882a593Smuzhiyun uint8 *pktend;
320*4882a593Smuzhiyun uint8 *evend;
321*4882a593Smuzhiyun int err = BCME_OK;
322*4882a593Smuzhiyun uint32 data_len = 0; /* data length in bcm_event */
323*4882a593Smuzhiyun
324*4882a593Smuzhiyun pktend = (uint8 *)pktdata + pktlen;
325*4882a593Smuzhiyun bcm_event = (bcm_event_t *)pktdata;
326*4882a593Smuzhiyun
327*4882a593Smuzhiyun /* only care about 16-bit subtype / length versions */
328*4882a593Smuzhiyun if ((uint8 *)&bcm_event->bcm_hdr < pktend) {
329*4882a593Smuzhiyun uint8 short_subtype = *(uint8 *)&bcm_event->bcm_hdr;
330*4882a593Smuzhiyun if (!(short_subtype & 0x80)) {
331*4882a593Smuzhiyun err = BCME_NOTFOUND;
332*4882a593Smuzhiyun goto done;
333*4882a593Smuzhiyun }
334*4882a593Smuzhiyun }
335*4882a593Smuzhiyun
336*4882a593Smuzhiyun /* must have both ether_header and bcmeth_hdr */
337*4882a593Smuzhiyun if (pktlen < OFFSETOF(bcm_event_t, event)) {
338*4882a593Smuzhiyun err = BCME_BADLEN;
339*4882a593Smuzhiyun goto done;
340*4882a593Smuzhiyun }
341*4882a593Smuzhiyun
342*4882a593Smuzhiyun /* check length in bcmeth_hdr */
343*4882a593Smuzhiyun
344*4882a593Smuzhiyun #ifdef BCMDONGLEHOST
345*4882a593Smuzhiyun /* temporary - header length not always set properly. When the below
346*4882a593Smuzhiyun * !BCMDONGLEHOST is in all branches that use trunk DHD, the code
347*4882a593Smuzhiyun * under BCMDONGLEHOST can be removed.
348*4882a593Smuzhiyun */
349*4882a593Smuzhiyun evlen = (uint16)(pktend - (uint8 *)&bcm_event->bcm_hdr.version);
350*4882a593Smuzhiyun #else
351*4882a593Smuzhiyun evlen = ntoh16_ua((void *)&bcm_event->bcm_hdr.length);
352*4882a593Smuzhiyun #endif /* BCMDONGLEHOST */
353*4882a593Smuzhiyun evend = (uint8 *)&bcm_event->bcm_hdr.version + evlen;
354*4882a593Smuzhiyun if (evend != pktend) {
355*4882a593Smuzhiyun err = BCME_BADLEN;
356*4882a593Smuzhiyun goto done;
357*4882a593Smuzhiyun }
358*4882a593Smuzhiyun
359*4882a593Smuzhiyun /* match on subtype, oui and usr subtype for BRCM events */
360*4882a593Smuzhiyun subtype = ntoh16_ua((void *)&bcm_event->bcm_hdr.subtype);
361*4882a593Smuzhiyun if (subtype != BCMILCP_SUBTYPE_VENDOR_LONG) {
362*4882a593Smuzhiyun err = BCME_NOTFOUND;
363*4882a593Smuzhiyun goto done;
364*4882a593Smuzhiyun }
365*4882a593Smuzhiyun
366*4882a593Smuzhiyun if (bcmp(BRCM_OUI, &bcm_event->bcm_hdr.oui[0], DOT11_OUI_LEN)) {
367*4882a593Smuzhiyun err = BCME_NOTFOUND;
368*4882a593Smuzhiyun goto done;
369*4882a593Smuzhiyun }
370*4882a593Smuzhiyun
371*4882a593Smuzhiyun /* if it is a bcm_event or bcm_dngl_event_t, validate it */
372*4882a593Smuzhiyun usr_subtype = ntoh16_ua((void *)&bcm_event->bcm_hdr.usr_subtype);
373*4882a593Smuzhiyun switch (usr_subtype) {
374*4882a593Smuzhiyun case BCMILCP_BCM_SUBTYPE_EVENT:
375*4882a593Smuzhiyun /* check that header length and pkt length are sufficient */
376*4882a593Smuzhiyun if ((pktlen < sizeof(bcm_event_t)) ||
377*4882a593Smuzhiyun (evend < ((uint8 *)bcm_event + sizeof(bcm_event_t)))) {
378*4882a593Smuzhiyun err = BCME_BADLEN;
379*4882a593Smuzhiyun goto done;
380*4882a593Smuzhiyun }
381*4882a593Smuzhiyun
382*4882a593Smuzhiyun /* ensure data length in event is not beyond the packet. */
383*4882a593Smuzhiyun data_len = ntoh32_ua((void *)&bcm_event->event.datalen);
384*4882a593Smuzhiyun if ((sizeof(bcm_event_t) + data_len +
385*4882a593Smuzhiyun BCMILCP_BCM_SUBTYPE_EVENT_DATA_PAD) != pktlen) {
386*4882a593Smuzhiyun err = BCME_BADLEN;
387*4882a593Smuzhiyun goto done;
388*4882a593Smuzhiyun }
389*4882a593Smuzhiyun
390*4882a593Smuzhiyun if (exp_usr_subtype && (exp_usr_subtype != usr_subtype)) {
391*4882a593Smuzhiyun err = BCME_NOTFOUND;
392*4882a593Smuzhiyun goto done;
393*4882a593Smuzhiyun }
394*4882a593Smuzhiyun
395*4882a593Smuzhiyun if (out_event) {
396*4882a593Smuzhiyun /* ensure BRCM event pkt aligned */
397*4882a593Smuzhiyun memcpy(&out_event->event, &bcm_event->event, sizeof(wl_event_msg_t));
398*4882a593Smuzhiyun }
399*4882a593Smuzhiyun
400*4882a593Smuzhiyun break;
401*4882a593Smuzhiyun
402*4882a593Smuzhiyun case BCMILCP_BCM_SUBTYPE_DNGLEVENT:
403*4882a593Smuzhiyun #if defined(HEALTH_CHECK) || defined(DNGL_EVENT_SUPPORT)
404*4882a593Smuzhiyun if ((pktlen < sizeof(bcm_dngl_event_t)) ||
405*4882a593Smuzhiyun (evend < ((uint8 *)bcm_event + sizeof(bcm_dngl_event_t)))) {
406*4882a593Smuzhiyun err = BCME_BADLEN;
407*4882a593Smuzhiyun goto done;
408*4882a593Smuzhiyun }
409*4882a593Smuzhiyun
410*4882a593Smuzhiyun /* ensure data length in event is not beyond the packet. */
411*4882a593Smuzhiyun data_len = ntoh16_ua((void *)&((bcm_dngl_event_t *)pktdata)->dngl_event.datalen);
412*4882a593Smuzhiyun if ((sizeof(bcm_dngl_event_t) + data_len +
413*4882a593Smuzhiyun BCMILCP_BCM_SUBTYPE_EVENT_DATA_PAD) != pktlen) {
414*4882a593Smuzhiyun err = BCME_BADLEN;
415*4882a593Smuzhiyun goto done;
416*4882a593Smuzhiyun }
417*4882a593Smuzhiyun
418*4882a593Smuzhiyun if (exp_usr_subtype && (exp_usr_subtype != usr_subtype)) {
419*4882a593Smuzhiyun err = BCME_NOTFOUND;
420*4882a593Smuzhiyun goto done;
421*4882a593Smuzhiyun }
422*4882a593Smuzhiyun
423*4882a593Smuzhiyun if (out_event) {
424*4882a593Smuzhiyun /* ensure BRCM dngl event pkt aligned */
425*4882a593Smuzhiyun memcpy(&out_event->dngl_event, &((bcm_dngl_event_t *)pktdata)->dngl_event,
426*4882a593Smuzhiyun sizeof(bcm_dngl_event_msg_t));
427*4882a593Smuzhiyun }
428*4882a593Smuzhiyun
429*4882a593Smuzhiyun break;
430*4882a593Smuzhiyun #else
431*4882a593Smuzhiyun err = BCME_UNSUPPORTED;
432*4882a593Smuzhiyun break;
433*4882a593Smuzhiyun #endif /* HEALTH_CHECK || DNGL_EVENT_SUPPORT */
434*4882a593Smuzhiyun
435*4882a593Smuzhiyun default:
436*4882a593Smuzhiyun err = BCME_NOTFOUND;
437*4882a593Smuzhiyun goto done;
438*4882a593Smuzhiyun }
439*4882a593Smuzhiyun
440*4882a593Smuzhiyun BCM_REFERENCE(data_len);
441*4882a593Smuzhiyun done:
442*4882a593Smuzhiyun return err;
443*4882a593Smuzhiyun }
444