|  | /* | 
|  | * bcmevent read-only data shared by kernel or app layers | 
|  | * | 
|  | * Copyright (C) 1999-2019, Broadcom. | 
|  | * | 
|  | *      Unless you and Broadcom execute a separate written software license | 
|  | * agreement governing use of this software, this software is licensed to you | 
|  | * under the terms of the GNU General Public License version 2 (the "GPL"), | 
|  | * available at http://www.broadcom.com/licenses/GPLv2.php, with the | 
|  | * following added to such license: | 
|  | * | 
|  | *      As a special exception, the copyright holders of this software give you | 
|  | * permission to link this software with independent modules, and to copy and | 
|  | * distribute the resulting executable under terms of your choice, provided that | 
|  | * you also meet, for each linked independent module, the terms and conditions of | 
|  | * the license of that module.  An independent module is a module which is not | 
|  | * derived from this software.  The special exception does not apply to any | 
|  | * modifications of the software. | 
|  | * | 
|  | *      Notwithstanding the above, under no circumstances may you combine this | 
|  | * software in any way with any other Broadcom software provided under a license | 
|  | * other than the GPL, without Broadcom's express prior written consent. | 
|  | * | 
|  | * | 
|  | * <<Broadcom-WL-IPTag/Open:>> | 
|  | * | 
|  | * $Id: bcmevent.c 807989 2019-03-05 07:57:42Z $ | 
|  | */ | 
|  |  | 
|  | #include <typedefs.h> | 
|  | #include <bcmutils.h> | 
|  | #include <bcmendian.h> | 
|  | #include <ethernet.h> | 
|  | #include <bcmeth.h> | 
|  | #include <bcmevent.h> | 
|  | #include <802.11.h> | 
|  |  | 
|  | /* Table of event name strings for UIs and debugging dumps */ | 
|  | typedef struct { | 
|  | uint event; | 
|  | const char *name; | 
|  | } bcmevent_name_str_t; | 
|  |  | 
|  | /* Use the actual name for event tracing */ | 
|  | #define BCMEVENT_NAME(_event) {(_event), #_event} | 
|  |  | 
|  | static const bcmevent_name_str_t bcmevent_names[] = { | 
|  | BCMEVENT_NAME(WLC_E_SET_SSID), | 
|  | BCMEVENT_NAME(WLC_E_JOIN), | 
|  | BCMEVENT_NAME(WLC_E_START), | 
|  | BCMEVENT_NAME(WLC_E_AUTH), | 
|  | BCMEVENT_NAME(WLC_E_AUTH_IND), | 
|  | BCMEVENT_NAME(WLC_E_DEAUTH), | 
|  | BCMEVENT_NAME(WLC_E_DEAUTH_IND), | 
|  | BCMEVENT_NAME(WLC_E_ASSOC), | 
|  | BCMEVENT_NAME(WLC_E_ASSOC_IND), | 
|  | BCMEVENT_NAME(WLC_E_REASSOC), | 
|  | BCMEVENT_NAME(WLC_E_REASSOC_IND), | 
|  | BCMEVENT_NAME(WLC_E_DISASSOC), | 
|  | BCMEVENT_NAME(WLC_E_DISASSOC_IND), | 
|  | BCMEVENT_NAME(WLC_E_QUIET_START), | 
|  | BCMEVENT_NAME(WLC_E_QUIET_END), | 
|  | BCMEVENT_NAME(WLC_E_BEACON_RX), | 
|  | BCMEVENT_NAME(WLC_E_LINK), | 
|  | BCMEVENT_NAME(WLC_E_MIC_ERROR), | 
|  | BCMEVENT_NAME(WLC_E_NDIS_LINK), | 
|  | BCMEVENT_NAME(WLC_E_ROAM), | 
|  | BCMEVENT_NAME(WLC_E_TXFAIL), | 
|  | BCMEVENT_NAME(WLC_E_PMKID_CACHE), | 
|  | BCMEVENT_NAME(WLC_E_RETROGRADE_TSF), | 
|  | BCMEVENT_NAME(WLC_E_PRUNE), | 
|  | BCMEVENT_NAME(WLC_E_AUTOAUTH), | 
|  | BCMEVENT_NAME(WLC_E_EAPOL_MSG), | 
|  | BCMEVENT_NAME(WLC_E_SCAN_COMPLETE), | 
|  | BCMEVENT_NAME(WLC_E_ADDTS_IND), | 
|  | BCMEVENT_NAME(WLC_E_DELTS_IND), | 
|  | BCMEVENT_NAME(WLC_E_BCNSENT_IND), | 
|  | BCMEVENT_NAME(WLC_E_BCNRX_MSG), | 
|  | BCMEVENT_NAME(WLC_E_BCNLOST_MSG), | 
|  | BCMEVENT_NAME(WLC_E_ROAM_PREP), | 
|  | BCMEVENT_NAME(WLC_E_PFN_NET_FOUND), | 
|  | BCMEVENT_NAME(WLC_E_PFN_SCAN_ALLGONE), | 
|  | BCMEVENT_NAME(WLC_E_PFN_NET_LOST), | 
|  | BCMEVENT_NAME(WLC_E_JOIN_START), | 
|  | BCMEVENT_NAME(WLC_E_ROAM_START), | 
|  | BCMEVENT_NAME(WLC_E_ASSOC_START), | 
|  | #if defined(IBSS_PEER_DISCOVERY_EVENT) | 
|  | BCMEVENT_NAME(WLC_E_IBSS_ASSOC), | 
|  | #endif /* defined(IBSS_PEER_DISCOVERY_EVENT) */ | 
|  | BCMEVENT_NAME(WLC_E_RADIO), | 
|  | BCMEVENT_NAME(WLC_E_PSM_WATCHDOG), | 
|  | BCMEVENT_NAME(WLC_E_PROBREQ_MSG), | 
|  | BCMEVENT_NAME(WLC_E_SCAN_CONFIRM_IND), | 
|  | BCMEVENT_NAME(WLC_E_PSK_SUP), | 
|  | BCMEVENT_NAME(WLC_E_COUNTRY_CODE_CHANGED), | 
|  | BCMEVENT_NAME(WLC_E_EXCEEDED_MEDIUM_TIME), | 
|  | BCMEVENT_NAME(WLC_E_ICV_ERROR), | 
|  | BCMEVENT_NAME(WLC_E_UNICAST_DECODE_ERROR), | 
|  | BCMEVENT_NAME(WLC_E_MULTICAST_DECODE_ERROR), | 
|  | BCMEVENT_NAME(WLC_E_TRACE), | 
|  | BCMEVENT_NAME(WLC_E_IF), | 
|  | #ifdef WLP2P | 
|  | BCMEVENT_NAME(WLC_E_P2P_DISC_LISTEN_COMPLETE), | 
|  | #endif // endif | 
|  | BCMEVENT_NAME(WLC_E_RSSI), | 
|  | BCMEVENT_NAME(WLC_E_PFN_SCAN_COMPLETE), | 
|  | BCMEVENT_NAME(WLC_E_ACTION_FRAME), | 
|  | BCMEVENT_NAME(WLC_E_ACTION_FRAME_RX), | 
|  | BCMEVENT_NAME(WLC_E_ACTION_FRAME_COMPLETE), | 
|  | #ifdef BCMWAPI_WAI | 
|  | BCMEVENT_NAME(WLC_E_WAI_STA_EVENT), | 
|  | BCMEVENT_NAME(WLC_E_WAI_MSG), | 
|  | #endif /* BCMWAPI_WAI */ | 
|  | BCMEVENT_NAME(WLC_E_ESCAN_RESULT), | 
|  | BCMEVENT_NAME(WLC_E_ACTION_FRAME_OFF_CHAN_COMPLETE), | 
|  | #ifdef WLP2P | 
|  | BCMEVENT_NAME(WLC_E_PROBRESP_MSG), | 
|  | BCMEVENT_NAME(WLC_E_P2P_PROBREQ_MSG), | 
|  | #endif // endif | 
|  | #ifdef PROP_TXSTATUS | 
|  | BCMEVENT_NAME(WLC_E_FIFO_CREDIT_MAP), | 
|  | #endif // endif | 
|  | BCMEVENT_NAME(WLC_E_WAKE_EVENT), | 
|  | BCMEVENT_NAME(WLC_E_DCS_REQUEST), | 
|  | BCMEVENT_NAME(WLC_E_RM_COMPLETE), | 
|  | BCMEVENT_NAME(WLC_E_OVERLAY_REQ), | 
|  | BCMEVENT_NAME(WLC_E_CSA_COMPLETE_IND), | 
|  | BCMEVENT_NAME(WLC_E_EXCESS_PM_WAKE_EVENT), | 
|  | BCMEVENT_NAME(WLC_E_PFN_SCAN_NONE), | 
|  | BCMEVENT_NAME(WLC_E_PFN_SCAN_ALLGONE), | 
|  | #ifdef SOFTAP | 
|  | BCMEVENT_NAME(WLC_E_GTK_PLUMBED), | 
|  | #endif // endif | 
|  | BCMEVENT_NAME(WLC_E_ASSOC_REQ_IE), | 
|  | BCMEVENT_NAME(WLC_E_ASSOC_RESP_IE), | 
|  | BCMEVENT_NAME(WLC_E_BEACON_FRAME_RX), | 
|  | #ifdef WLTDLS | 
|  | BCMEVENT_NAME(WLC_E_TDLS_PEER_EVENT), | 
|  | #endif /* WLTDLS */ | 
|  | BCMEVENT_NAME(WLC_E_NATIVE), | 
|  | #ifdef WLPKTDLYSTAT | 
|  | BCMEVENT_NAME(WLC_E_PKTDELAY_IND), | 
|  | #endif /* WLPKTDLYSTAT */ | 
|  | BCMEVENT_NAME(WLC_E_SERVICE_FOUND), | 
|  | BCMEVENT_NAME(WLC_E_GAS_FRAGMENT_RX), | 
|  | BCMEVENT_NAME(WLC_E_GAS_COMPLETE), | 
|  | BCMEVENT_NAME(WLC_E_P2PO_ADD_DEVICE), | 
|  | BCMEVENT_NAME(WLC_E_P2PO_DEL_DEVICE), | 
|  | #ifdef WLWNM | 
|  | BCMEVENT_NAME(WLC_E_WNM_STA_SLEEP), | 
|  | #endif /* WLWNM */ | 
|  | #if defined(WL_PROXDETECT) || defined(RTT_SUPPORT) | 
|  | BCMEVENT_NAME(WLC_E_PROXD), | 
|  | #endif // endif | 
|  | BCMEVENT_NAME(WLC_E_CCA_CHAN_QUAL), | 
|  | BCMEVENT_NAME(WLC_E_BSSID), | 
|  | #ifdef PROP_TXSTATUS | 
|  | BCMEVENT_NAME(WLC_E_BCMC_CREDIT_SUPPORT), | 
|  | #endif // endif | 
|  | BCMEVENT_NAME(WLC_E_PSTA_PRIMARY_INTF_IND), | 
|  | BCMEVENT_NAME(WLC_E_TXFAIL_THRESH), | 
|  | #ifdef GSCAN_SUPPORT | 
|  | BCMEVENT_NAME(WLC_E_PFN_GSCAN_FULL_RESULT), | 
|  | BCMEVENT_NAME(WLC_E_PFN_SSID_EXT), | 
|  | #endif /* GSCAN_SUPPORT */ | 
|  | #ifdef WLBSSLOAD_REPORT | 
|  | BCMEVENT_NAME(WLC_E_BSS_LOAD), | 
|  | #endif // endif | 
|  | #if defined(BT_WIFI_HANDOVER) || defined(WL_TBOW) | 
|  | BCMEVENT_NAME(WLC_E_BT_WIFI_HANDOVER_REQ), | 
|  | #endif // endif | 
|  | #ifdef WLFBT | 
|  | BCMEVENT_NAME(WLC_E_FBT), | 
|  | #endif /* WLFBT */ | 
|  | BCMEVENT_NAME(WLC_E_AUTHORIZED), | 
|  | BCMEVENT_NAME(WLC_E_PROBREQ_MSG_RX), | 
|  | BCMEVENT_NAME(WLC_E_CSA_START_IND), | 
|  | BCMEVENT_NAME(WLC_E_CSA_DONE_IND), | 
|  | BCMEVENT_NAME(WLC_E_CSA_FAILURE_IND), | 
|  | BCMEVENT_NAME(WLC_E_RMC_EVENT), | 
|  | BCMEVENT_NAME(WLC_E_DPSTA_INTF_IND), | 
|  | BCMEVENT_NAME(WLC_E_ALLOW_CREDIT_BORROW), | 
|  | BCMEVENT_NAME(WLC_E_MSCH), | 
|  | BCMEVENT_NAME(WLC_E_ULP), | 
|  | BCMEVENT_NAME(WLC_E_NAN), | 
|  | BCMEVENT_NAME(WLC_E_PKT_FILTER), | 
|  | BCMEVENT_NAME(WLC_E_DMA_TXFLUSH_COMPLETE), | 
|  | BCMEVENT_NAME(WLC_E_PSK_AUTH), | 
|  | BCMEVENT_NAME(WLC_E_SDB_TRANSITION), | 
|  | BCMEVENT_NAME(WLC_E_PFN_SCAN_BACKOFF), | 
|  | BCMEVENT_NAME(WLC_E_PFN_BSSID_SCAN_BACKOFF), | 
|  | BCMEVENT_NAME(WLC_E_AGGR_EVENT), | 
|  | BCMEVENT_NAME(WLC_E_TVPM_MITIGATION), | 
|  | #ifdef WL_NAN | 
|  | BCMEVENT_NAME(WLC_E_NAN_CRITICAL), | 
|  | BCMEVENT_NAME(WLC_E_NAN_NON_CRITICAL), | 
|  | BCMEVENT_NAME(WLC_E_NAN), | 
|  | #endif /* WL_NAN */ | 
|  | BCMEVENT_NAME(WLC_E_RPSNOA), | 
|  | BCMEVENT_NAME(WLC_E_PHY_CAL), | 
|  | BCMEVENT_NAME(WLC_E_WA_LQM), | 
|  | }; | 
|  |  | 
|  | const char *bcmevent_get_name(uint event_type) | 
|  | { | 
|  | /* note:  first coded this as a static const but some | 
|  | * ROMs already have something called event_name so | 
|  | * changed it so we don't have a variable for the | 
|  | * 'unknown string | 
|  | */ | 
|  | const char *event_name = NULL; | 
|  |  | 
|  | uint idx; | 
|  | for (idx = 0; idx < (uint)ARRAYSIZE(bcmevent_names); idx++) { | 
|  |  | 
|  | if (bcmevent_names[idx].event == event_type) { | 
|  | event_name = bcmevent_names[idx].name; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* if we find an event name in the array, return it. | 
|  | * otherwise return unknown string. | 
|  | */ | 
|  | return ((event_name) ? event_name : "Unknown Event"); | 
|  | } | 
|  |  | 
|  | void | 
|  | wl_event_to_host_order(wl_event_msg_t * evt) | 
|  | { | 
|  | /* Event struct members passed from dongle to host are stored in network | 
|  | * byte order. Convert all members to host-order. | 
|  | */ | 
|  | evt->event_type = ntoh32(evt->event_type); | 
|  | evt->flags = ntoh16(evt->flags); | 
|  | evt->status = ntoh32(evt->status); | 
|  | evt->reason = ntoh32(evt->reason); | 
|  | evt->auth_type = ntoh32(evt->auth_type); | 
|  | evt->datalen = ntoh32(evt->datalen); | 
|  | evt->version = ntoh16(evt->version); | 
|  | } | 
|  |  | 
|  | void | 
|  | wl_event_to_network_order(wl_event_msg_t * evt) | 
|  | { | 
|  | /* Event struct members passed from dongle to host are stored in network | 
|  | * byte order. Convert all members to host-order. | 
|  | */ | 
|  | evt->event_type = hton32(evt->event_type); | 
|  | evt->flags = hton16(evt->flags); | 
|  | evt->status = hton32(evt->status); | 
|  | evt->reason = hton32(evt->reason); | 
|  | evt->auth_type = hton32(evt->auth_type); | 
|  | evt->datalen = hton32(evt->datalen); | 
|  | evt->version = hton16(evt->version); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Validate if the event is proper and if valid copy event header to event. | 
|  | * If proper event pointer is passed, to just validate, pass NULL to event. | 
|  | * | 
|  | * Return values are | 
|  | *	BCME_OK - It is a BRCM event or BRCM dongle event | 
|  | *	BCME_NOTFOUND - Not BRCM, not an event, may be okay | 
|  | *	BCME_BADLEN - Bad length, should not process, just drop | 
|  | */ | 
|  | int | 
|  | is_wlc_event_frame(void *pktdata, uint pktlen, uint16 exp_usr_subtype, | 
|  | bcm_event_msg_u_t *out_event) | 
|  | { | 
|  | uint16 evlen = 0;	/* length in bcmeth_hdr */ | 
|  | uint16 subtype; | 
|  | uint16 usr_subtype; | 
|  | bcm_event_t *bcm_event; | 
|  | uint8 *pktend; | 
|  | uint8 *evend; | 
|  | int err = BCME_OK; | 
|  | uint32 data_len = 0; /* data length in bcm_event */ | 
|  |  | 
|  | pktend = (uint8 *)pktdata + pktlen; | 
|  | bcm_event = (bcm_event_t *)pktdata; | 
|  |  | 
|  | /* only care about 16-bit subtype / length versions */ | 
|  | if ((uint8 *)&bcm_event->bcm_hdr < pktend) { | 
|  | uint8 short_subtype = *(uint8 *)&bcm_event->bcm_hdr; | 
|  | if (!(short_subtype & 0x80)) { | 
|  | err = BCME_NOTFOUND; | 
|  | goto done; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* must have both ether_header and bcmeth_hdr */ | 
|  | if (pktlen < OFFSETOF(bcm_event_t, event)) { | 
|  | err = BCME_BADLEN; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | /* check length in bcmeth_hdr */ | 
|  |  | 
|  | /* temporary - header length not always set properly. When the below | 
|  | * !BCMDONGLEHOST is in all branches that use trunk DHD, the code | 
|  | * under BCMDONGLEHOST can be removed. | 
|  | */ | 
|  | evlen = (uint16)(pktend - (uint8 *)&bcm_event->bcm_hdr.version); | 
|  | evend = (uint8 *)&bcm_event->bcm_hdr.version + evlen; | 
|  | if (evend != pktend) { | 
|  | err = BCME_BADLEN; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | /* match on subtype, oui and usr subtype for BRCM events */ | 
|  | subtype = ntoh16_ua((void *)&bcm_event->bcm_hdr.subtype); | 
|  | if (subtype != BCMILCP_SUBTYPE_VENDOR_LONG) { | 
|  | err = BCME_NOTFOUND; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | if (bcmp(BRCM_OUI, &bcm_event->bcm_hdr.oui[0], DOT11_OUI_LEN)) { | 
|  | err = BCME_NOTFOUND; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | /* if it is a bcm_event or bcm_dngl_event_t, validate it */ | 
|  | usr_subtype = ntoh16_ua((void *)&bcm_event->bcm_hdr.usr_subtype); | 
|  | switch (usr_subtype) { | 
|  | case BCMILCP_BCM_SUBTYPE_EVENT: | 
|  | /* check that header length and pkt length are sufficient */ | 
|  | if ((pktlen < sizeof(bcm_event_t)) || | 
|  | (evend < ((uint8 *)bcm_event + sizeof(bcm_event_t)))) { | 
|  | err = BCME_BADLEN; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | /* ensure data length in event is not beyond the packet. */ | 
|  | data_len = ntoh32_ua((void *)&bcm_event->event.datalen); | 
|  | if ((sizeof(bcm_event_t) + data_len + | 
|  | BCMILCP_BCM_SUBTYPE_EVENT_DATA_PAD) != pktlen) { | 
|  | err = BCME_BADLEN; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | if (exp_usr_subtype && (exp_usr_subtype != usr_subtype)) { | 
|  | err = BCME_NOTFOUND; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | if (out_event) { | 
|  | /* ensure BRCM event pkt aligned */ | 
|  | memcpy(&out_event->event, &bcm_event->event, sizeof(wl_event_msg_t)); | 
|  | } | 
|  |  | 
|  | break; | 
|  |  | 
|  | case BCMILCP_BCM_SUBTYPE_DNGLEVENT: | 
|  | #if defined(DNGL_EVENT_SUPPORT) | 
|  | if ((pktlen < sizeof(bcm_dngl_event_t)) || | 
|  | (evend < ((uint8 *)bcm_event + sizeof(bcm_dngl_event_t)))) { | 
|  | err = BCME_BADLEN; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | /* ensure data length in event is not beyond the packet. */ | 
|  | data_len = ntoh16_ua((void *)&((bcm_dngl_event_t *)pktdata)->dngl_event.datalen); | 
|  | if ((sizeof(bcm_dngl_event_t) + data_len + | 
|  | BCMILCP_BCM_SUBTYPE_EVENT_DATA_PAD) != pktlen) { | 
|  | err = BCME_BADLEN; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | if (exp_usr_subtype && (exp_usr_subtype != usr_subtype)) { | 
|  | err = BCME_NOTFOUND; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | if (out_event) { | 
|  | /* ensure BRCM dngl event pkt aligned */ | 
|  | memcpy(&out_event->dngl_event, &((bcm_dngl_event_t *)pktdata)->dngl_event, | 
|  | sizeof(bcm_dngl_event_msg_t)); | 
|  | } | 
|  |  | 
|  | break; | 
|  | #else | 
|  | err = BCME_UNSUPPORTED; | 
|  | break; | 
|  | #endif // endif | 
|  |  | 
|  | default: | 
|  | err = BCME_NOTFOUND; | 
|  | goto done; | 
|  | } | 
|  |  | 
|  | BCM_REFERENCE(data_len); | 
|  | done: | 
|  | return err; | 
|  | } |