mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2024-12-28 11:18:45 +07:00
df19e777e0
brcmf_fweh_detach can be called while ifp is already NULL, due to init error. Fix NULL pointer access by checking ifp. Reviewed-by: Arend Van Spriel <arend@broadcom.com> Signed-off-by: Hante Meuleman <meuleman@broadcom.com> Signed-off-by: Franky Lin <frankyl@broadcom.com> Signed-off-by: John W. Linville <linville@tuxdriver.com>
497 lines
14 KiB
C
497 lines
14 KiB
C
/*
|
|
* Copyright (c) 2012 Broadcom Corporation
|
|
*
|
|
* Permission to use, copy, modify, and/or distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
|
|
* SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
|
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
#include <linux/netdevice.h>
|
|
|
|
#include "defs.h"
|
|
#include "brcmu_wifi.h"
|
|
#include "brcmu_utils.h"
|
|
|
|
#include "dhd.h"
|
|
#include "dhd_dbg.h"
|
|
#include "fweh.h"
|
|
#include "fwil.h"
|
|
|
|
/**
|
|
* struct brcm_ethhdr - broadcom specific ether header.
|
|
*
|
|
* @subtype: subtype for this packet.
|
|
* @length: TODO: length of appended data.
|
|
* @version: version indication.
|
|
* @oui: OUI of this packet.
|
|
* @usr_subtype: subtype for this OUI.
|
|
*/
|
|
struct brcm_ethhdr {
|
|
__be16 subtype;
|
|
__be16 length;
|
|
u8 version;
|
|
u8 oui[3];
|
|
__be16 usr_subtype;
|
|
} __packed;
|
|
|
|
struct brcmf_event_msg_be {
|
|
__be16 version;
|
|
__be16 flags;
|
|
__be32 event_type;
|
|
__be32 status;
|
|
__be32 reason;
|
|
__be32 auth_type;
|
|
__be32 datalen;
|
|
u8 addr[ETH_ALEN];
|
|
char ifname[IFNAMSIZ];
|
|
u8 ifidx;
|
|
u8 bsscfgidx;
|
|
} __packed;
|
|
|
|
/**
|
|
* struct brcmf_event - contents of broadcom event packet.
|
|
*
|
|
* @eth: standard ether header.
|
|
* @hdr: broadcom specific ether header.
|
|
* @msg: common part of the actual event message.
|
|
*/
|
|
struct brcmf_event {
|
|
struct ethhdr eth;
|
|
struct brcm_ethhdr hdr;
|
|
struct brcmf_event_msg_be msg;
|
|
} __packed;
|
|
|
|
/**
|
|
* struct brcmf_fweh_queue_item - event item on event queue.
|
|
*
|
|
* @q: list element for queuing.
|
|
* @code: event code.
|
|
* @ifidx: interface index related to this event.
|
|
* @ifaddr: ethernet address for interface.
|
|
* @emsg: common parameters of the firmware event message.
|
|
* @data: event specific data part of the firmware event.
|
|
*/
|
|
struct brcmf_fweh_queue_item {
|
|
struct list_head q;
|
|
enum brcmf_fweh_event_code code;
|
|
u8 ifidx;
|
|
u8 ifaddr[ETH_ALEN];
|
|
struct brcmf_event_msg_be emsg;
|
|
u8 data[0];
|
|
};
|
|
|
|
/**
|
|
* struct brcmf_fweh_event_name - code, name mapping entry.
|
|
*/
|
|
struct brcmf_fweh_event_name {
|
|
enum brcmf_fweh_event_code code;
|
|
const char *name;
|
|
};
|
|
|
|
#ifdef DEBUG
|
|
/* array for mapping code to event name */
|
|
static struct brcmf_fweh_event_name fweh_event_names[] = {
|
|
{ BRCMF_E_SET_SSID, "SET_SSID" },
|
|
{ BRCMF_E_JOIN, "JOIN" },
|
|
{ BRCMF_E_START, "START" },
|
|
{ BRCMF_E_AUTH, "AUTH" },
|
|
{ BRCMF_E_AUTH_IND, "AUTH_IND" },
|
|
{ BRCMF_E_DEAUTH, "DEAUTH" },
|
|
{ BRCMF_E_DEAUTH_IND, "DEAUTH_IND" },
|
|
{ BRCMF_E_ASSOC, "ASSOC" },
|
|
{ BRCMF_E_ASSOC_IND, "ASSOC_IND" },
|
|
{ BRCMF_E_REASSOC, "REASSOC" },
|
|
{ BRCMF_E_REASSOC_IND, "REASSOC_IND" },
|
|
{ BRCMF_E_DISASSOC, "DISASSOC" },
|
|
{ BRCMF_E_DISASSOC_IND, "DISASSOC_IND" },
|
|
{ BRCMF_E_QUIET_START, "START_QUIET" },
|
|
{ BRCMF_E_QUIET_END, "END_QUIET" },
|
|
{ BRCMF_E_BEACON_RX, "BEACON_RX" },
|
|
{ BRCMF_E_LINK, "LINK" },
|
|
{ BRCMF_E_MIC_ERROR, "MIC_ERROR" },
|
|
{ BRCMF_E_NDIS_LINK, "NDIS_LINK" },
|
|
{ BRCMF_E_ROAM, "ROAM" },
|
|
{ BRCMF_E_TXFAIL, "TXFAIL" },
|
|
{ BRCMF_E_PMKID_CACHE, "PMKID_CACHE" },
|
|
{ BRCMF_E_RETROGRADE_TSF, "RETROGRADE_TSF" },
|
|
{ BRCMF_E_PRUNE, "PRUNE" },
|
|
{ BRCMF_E_AUTOAUTH, "AUTOAUTH" },
|
|
{ BRCMF_E_EAPOL_MSG, "EAPOL_MSG" },
|
|
{ BRCMF_E_SCAN_COMPLETE, "SCAN_COMPLETE" },
|
|
{ BRCMF_E_ADDTS_IND, "ADDTS_IND" },
|
|
{ BRCMF_E_DELTS_IND, "DELTS_IND" },
|
|
{ BRCMF_E_BCNSENT_IND, "BCNSENT_IND" },
|
|
{ BRCMF_E_BCNRX_MSG, "BCNRX_MSG" },
|
|
{ BRCMF_E_BCNLOST_MSG, "BCNLOST_MSG" },
|
|
{ BRCMF_E_ROAM_PREP, "ROAM_PREP" },
|
|
{ BRCMF_E_PFN_NET_FOUND, "PNO_NET_FOUND" },
|
|
{ BRCMF_E_PFN_NET_LOST, "PNO_NET_LOST" },
|
|
{ BRCMF_E_RESET_COMPLETE, "RESET_COMPLETE" },
|
|
{ BRCMF_E_JOIN_START, "JOIN_START" },
|
|
{ BRCMF_E_ROAM_START, "ROAM_START" },
|
|
{ BRCMF_E_ASSOC_START, "ASSOC_START" },
|
|
{ BRCMF_E_IBSS_ASSOC, "IBSS_ASSOC" },
|
|
{ BRCMF_E_RADIO, "RADIO" },
|
|
{ BRCMF_E_PSM_WATCHDOG, "PSM_WATCHDOG" },
|
|
{ BRCMF_E_PROBREQ_MSG, "PROBREQ_MSG" },
|
|
{ BRCMF_E_SCAN_CONFIRM_IND, "SCAN_CONFIRM_IND" },
|
|
{ BRCMF_E_PSK_SUP, "PSK_SUP" },
|
|
{ BRCMF_E_COUNTRY_CODE_CHANGED, "COUNTRY_CODE_CHANGED" },
|
|
{ BRCMF_E_EXCEEDED_MEDIUM_TIME, "EXCEEDED_MEDIUM_TIME" },
|
|
{ BRCMF_E_ICV_ERROR, "ICV_ERROR" },
|
|
{ BRCMF_E_UNICAST_DECODE_ERROR, "UNICAST_DECODE_ERROR" },
|
|
{ BRCMF_E_MULTICAST_DECODE_ERROR, "MULTICAST_DECODE_ERROR" },
|
|
{ BRCMF_E_TRACE, "TRACE" },
|
|
{ BRCMF_E_IF, "IF" },
|
|
{ BRCMF_E_RSSI, "RSSI" },
|
|
{ BRCMF_E_PFN_SCAN_COMPLETE, "PFN_SCAN_COMPLETE" },
|
|
{ BRCMF_E_EXTLOG_MSG, "EXTLOG_MSG" },
|
|
{ BRCMF_E_ACTION_FRAME, "ACTION_FRAME" },
|
|
{ BRCMF_E_ACTION_FRAME_COMPLETE, "ACTION_FRAME_COMPLETE" },
|
|
{ BRCMF_E_PRE_ASSOC_IND, "PRE_ASSOC_IND" },
|
|
{ BRCMF_E_PRE_REASSOC_IND, "PRE_REASSOC_IND" },
|
|
{ BRCMF_E_CHANNEL_ADOPTED, "CHANNEL_ADOPTED" },
|
|
{ BRCMF_E_AP_STARTED, "AP_STARTED" },
|
|
{ BRCMF_E_DFS_AP_STOP, "DFS_AP_STOP" },
|
|
{ BRCMF_E_DFS_AP_RESUME, "DFS_AP_RESUME" },
|
|
{ BRCMF_E_ESCAN_RESULT, "ESCAN_RESULT" },
|
|
{ BRCMF_E_ACTION_FRAME_OFF_CHAN_COMPLETE, "ACTION_FRM_OFF_CHAN_CMPLT" },
|
|
{ BRCMF_E_DCS_REQUEST, "DCS_REQUEST" },
|
|
{ BRCMF_E_FIFO_CREDIT_MAP, "FIFO_CREDIT_MAP"}
|
|
};
|
|
|
|
/**
|
|
* brcmf_fweh_event_name() - returns name for given event code.
|
|
*
|
|
* @code: code to lookup.
|
|
*/
|
|
static const char *brcmf_fweh_event_name(enum brcmf_fweh_event_code code)
|
|
{
|
|
int i;
|
|
for (i = 0; i < ARRAY_SIZE(fweh_event_names); i++) {
|
|
if (fweh_event_names[i].code == code)
|
|
return fweh_event_names[i].name;
|
|
}
|
|
return "unknown";
|
|
}
|
|
#else
|
|
static const char *brcmf_fweh_event_name(enum brcmf_fweh_event_code code)
|
|
{
|
|
return "nodebug";
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* brcmf_fweh_queue_event() - create and queue event.
|
|
*
|
|
* @fweh: firmware event handling info.
|
|
* @event: event queue entry.
|
|
*/
|
|
static void brcmf_fweh_queue_event(struct brcmf_fweh_info *fweh,
|
|
struct brcmf_fweh_queue_item *event)
|
|
{
|
|
ulong flags;
|
|
|
|
spin_lock_irqsave(&fweh->evt_q_lock, flags);
|
|
list_add_tail(&event->q, &fweh->event_q);
|
|
spin_unlock_irqrestore(&fweh->evt_q_lock, flags);
|
|
schedule_work(&fweh->event_work);
|
|
}
|
|
|
|
/**
|
|
* brcmf_fweh_process_if_event() - handle IF event.
|
|
*
|
|
* @drvr: driver information object.
|
|
* @item: queue entry.
|
|
* @ifpp: interface object (may change upon ADD action).
|
|
*/
|
|
static int brcmf_fweh_process_if_event(struct brcmf_pub *drvr,
|
|
struct brcmf_fweh_queue_item *item,
|
|
struct brcmf_if **ifpp)
|
|
{
|
|
struct brcmf_event_msg_be *event = &item->emsg;
|
|
struct brcmf_if_event *ifevent = (struct brcmf_if_event *)item->data;
|
|
struct brcmf_if *ifp;
|
|
int err = 0;
|
|
|
|
brcmf_dbg(EVENT, "action: %u idx: %u bsscfg: %u flags: %u\n",
|
|
ifevent->action, ifevent->ifidx,
|
|
ifevent->bssidx, ifevent->flags);
|
|
|
|
if (ifevent->ifidx >= BRCMF_MAX_IFS) {
|
|
brcmf_dbg(ERROR, "invalid interface index: %u\n",
|
|
ifevent->ifidx);
|
|
return -EINVAL;
|
|
}
|
|
|
|
switch (ifevent->action) {
|
|
case BRCMF_E_IF_ADD:
|
|
brcmf_dbg(EVENT, "adding %s (%pM, %pM)\n", event->ifname,
|
|
event->addr, item->ifaddr);
|
|
ifp = brcmf_add_if(drvr, ifevent->ifidx, ifevent->bssidx,
|
|
event->ifname, item->ifaddr);
|
|
if (!IS_ERR(ifp)) {
|
|
*ifpp = ifp;
|
|
err = brcmf_net_attach(ifp);
|
|
} else {
|
|
err = PTR_ERR(ifp);
|
|
}
|
|
break;
|
|
case BRCMF_E_IF_DEL:
|
|
brcmf_del_if(drvr, ifevent->ifidx);
|
|
break;
|
|
case BRCMF_E_IF_CHANGE:
|
|
/* nothing to do here */
|
|
break;
|
|
default:
|
|
brcmf_dbg(ERROR, "unknown event action: %u\n", ifevent->action);
|
|
err = -EBADE;
|
|
break;
|
|
}
|
|
return err;
|
|
}
|
|
|
|
/**
|
|
* brcmf_fweh_dequeue_event() - get event from the queue.
|
|
*
|
|
* @fweh: firmware event handling info.
|
|
*/
|
|
static struct brcmf_fweh_queue_item *
|
|
brcmf_fweh_dequeue_event(struct brcmf_fweh_info *fweh)
|
|
{
|
|
struct brcmf_fweh_queue_item *event = NULL;
|
|
ulong flags;
|
|
|
|
spin_lock_irqsave(&fweh->evt_q_lock, flags);
|
|
if (!list_empty(&fweh->event_q)) {
|
|
event = list_first_entry(&fweh->event_q,
|
|
struct brcmf_fweh_queue_item, q);
|
|
list_del(&event->q);
|
|
}
|
|
spin_unlock_irqrestore(&fweh->evt_q_lock, flags);
|
|
|
|
return event;
|
|
}
|
|
|
|
/**
|
|
* brcmf_fweh_event_worker() - firmware event worker.
|
|
*
|
|
* @work: worker object.
|
|
*/
|
|
static void brcmf_fweh_event_worker(struct work_struct *work)
|
|
{
|
|
struct brcmf_pub *drvr;
|
|
struct brcmf_if *ifp;
|
|
struct brcmf_fweh_info *fweh;
|
|
struct brcmf_fweh_queue_item *event;
|
|
int err = 0;
|
|
struct brcmf_event_msg_be *emsg_be;
|
|
struct brcmf_event_msg emsg;
|
|
|
|
fweh = container_of(work, struct brcmf_fweh_info, event_work);
|
|
drvr = container_of(fweh, struct brcmf_pub, fweh);
|
|
|
|
while ((event = brcmf_fweh_dequeue_event(fweh))) {
|
|
ifp = drvr->iflist[event->ifidx];
|
|
|
|
brcmf_dbg(EVENT, "event %s (%u) ifidx %u bsscfg %u addr %pM:\n",
|
|
brcmf_fweh_event_name(event->code), event->code,
|
|
event->emsg.ifidx, event->emsg.bsscfgidx,
|
|
event->emsg.addr);
|
|
|
|
/* handle interface event */
|
|
if (event->code == BRCMF_E_IF) {
|
|
err = brcmf_fweh_process_if_event(drvr, event, &ifp);
|
|
if (err)
|
|
goto event_free;
|
|
}
|
|
|
|
/* convert event message */
|
|
emsg_be = &event->emsg;
|
|
emsg.version = be16_to_cpu(emsg_be->version);
|
|
emsg.flags = be16_to_cpu(emsg_be->flags);
|
|
emsg.event_code = event->code;
|
|
emsg.status = be32_to_cpu(emsg_be->status);
|
|
emsg.reason = be32_to_cpu(emsg_be->reason);
|
|
emsg.auth_type = be32_to_cpu(emsg_be->auth_type);
|
|
emsg.datalen = be32_to_cpu(emsg_be->datalen);
|
|
memcpy(emsg.addr, emsg_be->addr, ETH_ALEN);
|
|
memcpy(emsg.ifname, emsg_be->ifname, sizeof(emsg.ifname));
|
|
emsg.ifidx = emsg_be->ifidx;
|
|
emsg.bsscfgidx = emsg_be->bsscfgidx;
|
|
|
|
brcmf_dbg(EVENT, " version %u flags %u status %u reason %u\n",
|
|
emsg.version, emsg.flags, emsg.status, emsg.reason);
|
|
brcmf_dbg_hex_dump(BRCMF_EVENT_ON(), event->data,
|
|
min_t(u32, emsg.datalen, 64),
|
|
"appended:");
|
|
|
|
/* handle the event if valid interface and handler */
|
|
if (ifp->ndev && fweh->evt_handler[event->code])
|
|
err = fweh->evt_handler[event->code](ifp, &emsg,
|
|
event->data);
|
|
else
|
|
brcmf_dbg(ERROR, "unhandled event %d ignored\n",
|
|
event->code);
|
|
if (err) {
|
|
brcmf_dbg(ERROR, "event handler failed (%d)\n",
|
|
event->code);
|
|
err = 0;
|
|
}
|
|
event_free:
|
|
kfree(event);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* brcmf_fweh_attach() - initialize firmware event handling.
|
|
*
|
|
* @drvr: driver information object.
|
|
*/
|
|
void brcmf_fweh_attach(struct brcmf_pub *drvr)
|
|
{
|
|
struct brcmf_fweh_info *fweh = &drvr->fweh;
|
|
INIT_WORK(&fweh->event_work, brcmf_fweh_event_worker);
|
|
spin_lock_init(&fweh->evt_q_lock);
|
|
INIT_LIST_HEAD(&fweh->event_q);
|
|
}
|
|
|
|
/**
|
|
* brcmf_fweh_detach() - cleanup firmware event handling.
|
|
*
|
|
* @drvr: driver information object.
|
|
*/
|
|
void brcmf_fweh_detach(struct brcmf_pub *drvr)
|
|
{
|
|
struct brcmf_fweh_info *fweh = &drvr->fweh;
|
|
struct brcmf_if *ifp = drvr->iflist[0];
|
|
s8 eventmask[BRCMF_EVENTING_MASK_LEN];
|
|
|
|
if (ifp) {
|
|
/* clear all events */
|
|
memset(eventmask, 0, BRCMF_EVENTING_MASK_LEN);
|
|
(void)brcmf_fil_iovar_data_set(ifp, "event_msgs",
|
|
eventmask,
|
|
BRCMF_EVENTING_MASK_LEN);
|
|
}
|
|
/* cancel the worker */
|
|
cancel_work_sync(&fweh->event_work);
|
|
WARN_ON(!list_empty(&fweh->event_q));
|
|
memset(fweh->evt_handler, 0, sizeof(fweh->evt_handler));
|
|
}
|
|
|
|
/**
|
|
* brcmf_fweh_register() - register handler for given event code.
|
|
*
|
|
* @drvr: driver information object.
|
|
* @code: event code.
|
|
* @handler: handler for the given event code.
|
|
*/
|
|
int brcmf_fweh_register(struct brcmf_pub *drvr, enum brcmf_fweh_event_code code,
|
|
brcmf_fweh_handler_t handler)
|
|
{
|
|
if (drvr->fweh.evt_handler[code]) {
|
|
brcmf_dbg(ERROR, "event code %d already registered\n", code);
|
|
return -ENOSPC;
|
|
}
|
|
drvr->fweh.evt_handler[code] = handler;
|
|
brcmf_dbg(TRACE, "event handler registered for code %d\n", code);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* brcmf_fweh_unregister() - remove handler for given code.
|
|
*
|
|
* @drvr: driver information object.
|
|
* @code: event code.
|
|
*/
|
|
void brcmf_fweh_unregister(struct brcmf_pub *drvr,
|
|
enum brcmf_fweh_event_code code)
|
|
{
|
|
brcmf_dbg(TRACE, "event handler cleared for code %d\n", code);
|
|
drvr->fweh.evt_handler[code] = NULL;
|
|
}
|
|
|
|
/**
|
|
* brcmf_fweh_activate_events() - enables firmware events registered.
|
|
*
|
|
* @ifp: primary interface object.
|
|
*/
|
|
int brcmf_fweh_activate_events(struct brcmf_if *ifp)
|
|
{
|
|
int i, err;
|
|
s8 eventmask[BRCMF_EVENTING_MASK_LEN];
|
|
|
|
for (i = 0; i < BRCMF_E_LAST; i++) {
|
|
if (ifp->drvr->fweh.evt_handler[i]) {
|
|
brcmf_dbg(EVENT, "enable event %s\n",
|
|
brcmf_fweh_event_name(i));
|
|
setbit(eventmask, i);
|
|
}
|
|
}
|
|
|
|
/* want to handle IF event as well */
|
|
setbit(eventmask, BRCMF_E_IF);
|
|
|
|
err = brcmf_fil_iovar_data_set(ifp, "event_msgs",
|
|
eventmask, BRCMF_EVENTING_MASK_LEN);
|
|
if (err)
|
|
brcmf_dbg(ERROR, "Set event_msgs error (%d)\n", err);
|
|
|
|
return err;
|
|
}
|
|
|
|
/**
|
|
* brcmf_fweh_process_event() - process skb as firmware event.
|
|
*
|
|
* @drvr: driver information object.
|
|
* @event_packet: event packet to process.
|
|
* @ifidx: index of the firmware interface (may change).
|
|
*
|
|
* If the packet buffer contains a firmware event message it will
|
|
* dispatch the event to a registered handler (using worker).
|
|
*/
|
|
void brcmf_fweh_process_event(struct brcmf_pub *drvr,
|
|
struct brcmf_event *event_packet, u8 *ifidx)
|
|
{
|
|
enum brcmf_fweh_event_code code;
|
|
struct brcmf_fweh_info *fweh = &drvr->fweh;
|
|
struct brcmf_fweh_queue_item *event;
|
|
gfp_t alloc_flag = GFP_KERNEL;
|
|
void *data;
|
|
u32 datalen;
|
|
|
|
/* get event info */
|
|
code = get_unaligned_be32(&event_packet->msg.event_type);
|
|
datalen = get_unaligned_be32(&event_packet->msg.datalen);
|
|
*ifidx = event_packet->msg.ifidx;
|
|
data = &event_packet[1];
|
|
|
|
if (code != BRCMF_E_IF && !fweh->evt_handler[code]) {
|
|
brcmf_dbg(EVENT, "event ignored: code=%d\n", code);
|
|
brcmf_dbg_hex_dump(BRCMF_EVENT_ON(), data, datalen, "event:");
|
|
return;
|
|
}
|
|
|
|
if (in_interrupt())
|
|
alloc_flag = GFP_ATOMIC;
|
|
|
|
event = kzalloc(sizeof(*event) + datalen, alloc_flag);
|
|
event->code = code;
|
|
event->ifidx = *ifidx;
|
|
|
|
/* use memcpy to get aligned event message */
|
|
memcpy(&event->emsg, &event_packet->msg, sizeof(event->emsg));
|
|
memcpy(event->data, data, datalen);
|
|
memcpy(event->ifaddr, event_packet->eth.h_dest, ETH_ALEN);
|
|
|
|
brcmf_fweh_queue_event(fweh, event);
|
|
}
|