221 lines
5.9 KiB
C
Executable File
221 lines
5.9 KiB
C
Executable File
/*
|
|
* Broadcom Dongle Host Driver (DHD)
|
|
*
|
|
* Copyright (C) 1999-2018, 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.
|
|
*
|
|
* $Id: dhd_csi.c 606280 2015-12-15 05:28:25Z $
|
|
*/
|
|
#include <osl.h>
|
|
|
|
#include <bcmutils.h>
|
|
|
|
#include <bcmendian.h>
|
|
#include <linuxver.h>
|
|
#include <linux/list.h>
|
|
#include <linux/sort.h>
|
|
#include <dngl_stats.h>
|
|
#include <wlioctl.h>
|
|
|
|
#include <bcmevent.h>
|
|
#include <dhd.h>
|
|
#include <dhd_dbg.h>
|
|
#include <dhd_csi.h>
|
|
|
|
#define NULL_CHECK(p, s, err) \
|
|
do { \
|
|
if (!(p)) { \
|
|
printf("NULL POINTER (%s) : %s\n", __FUNCTION__, (s)); \
|
|
err = BCME_ERROR; \
|
|
return err; \
|
|
} \
|
|
} while (0)
|
|
|
|
#define TIMESPEC_TO_US(ts) (((uint64)(ts).tv_sec * USEC_PER_SEC) + \
|
|
(ts).tv_nsec / NSEC_PER_USEC)
|
|
|
|
#define NULL_ADDR "\x00\x00\x00\x00\x00\x00"
|
|
|
|
int
|
|
dhd_csi_event_handler(dhd_pub_t *dhd, wl_event_msg_t *event, void *event_data)
|
|
{
|
|
int ret = BCME_OK;
|
|
bool is_new = TRUE;
|
|
cfr_dump_data_t *p_event;
|
|
cfr_dump_list_t *ptr, *next, *new;
|
|
|
|
NULL_CHECK(dhd, "dhd is NULL", ret);
|
|
|
|
DHD_TRACE(("Enter %s\n", __FUNCTION__));
|
|
|
|
if (!event_data) {
|
|
DHD_ERROR(("%s: event_data is NULL\n", __FUNCTION__));
|
|
return -EINVAL;
|
|
}
|
|
p_event = (cfr_dump_data_t *)event_data;
|
|
|
|
/* check if this addr exist */
|
|
if (!list_empty(&dhd->csi_list)) {
|
|
list_for_each_entry_safe(ptr, next, &dhd->csi_list, list) {
|
|
if (bcmp(&ptr->entry.header.peer_macaddr, &p_event->header.peer_macaddr,
|
|
ETHER_ADDR_LEN) == 0) {
|
|
int pos = 0, dump_len = 0, remain = 0;
|
|
is_new = FALSE;
|
|
DHD_INFO(("CSI data exist\n"));
|
|
if (p_event->header.status == 0) {
|
|
bcopy(&p_event->header, &ptr->entry.header, sizeof(cfr_dump_header_t));
|
|
dump_len = p_event->header.cfr_dump_length;
|
|
if (dump_len < MAX_EVENT_SIZE) {
|
|
bcopy(&p_event->data, &ptr->entry.data, dump_len);
|
|
} else {
|
|
/* for big csi data */
|
|
uint8 *p = (uint8 *)&ptr->entry.data;
|
|
remain = p_event->header.remain_length;
|
|
if (remain) {
|
|
pos = dump_len - remain - MAX_EVENT_SIZE;
|
|
p += pos;
|
|
bcopy(&p_event->data, p, MAX_EVENT_SIZE);
|
|
}
|
|
/* copy rest of csi data */
|
|
else {
|
|
pos = dump_len - (dump_len % MAX_EVENT_SIZE);
|
|
p += pos;
|
|
bcopy(&p_event->data, p, (dump_len % MAX_EVENT_SIZE));
|
|
}
|
|
}
|
|
return BCME_OK;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (is_new) {
|
|
if (dhd->csi_count < MAX_CSI_NUM) {
|
|
new = (cfr_dump_list_t *)MALLOCZ(dhd->osh, sizeof(cfr_dump_list_t));
|
|
if (!new){
|
|
DHD_ERROR(("Malloc cfr dump list error\n"));
|
|
return BCME_NOMEM;
|
|
}
|
|
bcopy(&p_event->header, &new->entry.header, sizeof(cfr_dump_header_t));
|
|
DHD_INFO(("New entry data size %d\n", p_event->header.cfr_dump_length));
|
|
/* for big csi data */
|
|
if (p_event->header.remain_length) {
|
|
DHD_TRACE(("remain %d\n", p_event->header.remain_length));
|
|
bcopy(&p_event->data, &new->entry.data, MAX_EVENT_SIZE);
|
|
}
|
|
else
|
|
bcopy(&p_event->data, &new->entry.data, p_event->header.cfr_dump_length);
|
|
INIT_LIST_HEAD(&(new->list));
|
|
list_add_tail(&(new->list), &dhd->csi_list);
|
|
dhd->csi_count++;
|
|
}
|
|
else {
|
|
DHD_TRACE(("Over maximum CSI Number 8. SKIP it.\n"));
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int
|
|
dhd_csi_init(dhd_pub_t *dhd)
|
|
{
|
|
int err = BCME_OK;
|
|
|
|
NULL_CHECK(dhd, "dhd is NULL", err);
|
|
INIT_LIST_HEAD(&dhd->csi_list);
|
|
dhd->csi_count = 0;
|
|
|
|
return err;
|
|
}
|
|
|
|
int
|
|
dhd_csi_deinit(dhd_pub_t *dhd)
|
|
{
|
|
int err = BCME_OK;
|
|
cfr_dump_list_t *ptr, *next;
|
|
|
|
NULL_CHECK(dhd, "dhd is NULL", err);
|
|
|
|
if (!list_empty(&dhd->csi_list)) {
|
|
list_for_each_entry_safe(ptr, next, &dhd->csi_list, list) {
|
|
list_del(&ptr->list);
|
|
MFREE(dhd->osh, ptr, sizeof(cfr_dump_list_t));
|
|
}
|
|
}
|
|
return err;
|
|
}
|
|
|
|
void
|
|
dhd_csi_clean_list(dhd_pub_t *dhd)
|
|
{
|
|
cfr_dump_list_t *ptr, *next;
|
|
int num = 0;
|
|
|
|
if (!dhd) {
|
|
DHD_ERROR(("NULL POINTER: %s\n", __FUNCTION__));
|
|
return;
|
|
}
|
|
|
|
if (!list_empty(&dhd->csi_list)) {
|
|
list_for_each_entry_safe(ptr, next, &dhd->csi_list, list) {
|
|
if (0 == ptr->entry.header.remain_length) {
|
|
list_del(&ptr->list);
|
|
num++;
|
|
MFREE(dhd->osh, ptr, sizeof(cfr_dump_list_t));
|
|
}
|
|
}
|
|
}
|
|
dhd->csi_count = 0;
|
|
DHD_TRACE(("Clean up %d record\n", num));
|
|
}
|
|
|
|
int
|
|
dhd_csi_dump_list(dhd_pub_t *dhd, char *buf)
|
|
{
|
|
int ret = BCME_OK;
|
|
cfr_dump_list_t *ptr, *next;
|
|
uint8 * pbuf = buf;
|
|
int num = 0;
|
|
int length = 0;
|
|
|
|
NULL_CHECK(dhd, "dhd is NULL", ret);
|
|
|
|
/* check if this addr exist */
|
|
if (!list_empty(&dhd->csi_list)) {
|
|
list_for_each_entry_safe(ptr, next, &dhd->csi_list, list) {
|
|
if (ptr->entry.header.remain_length) {
|
|
DHD_ERROR(("data not ready %d\n", ptr->entry.header.remain_length));
|
|
continue;
|
|
}
|
|
bcopy(&ptr->entry.header, pbuf, sizeof(cfr_dump_header_t));
|
|
length += sizeof(cfr_dump_header_t);
|
|
pbuf += sizeof(cfr_dump_header_t);
|
|
DHD_TRACE(("Copy data size %d\n", ptr->entry.header.cfr_dump_length));
|
|
bcopy(&ptr->entry.data, pbuf, ptr->entry.header.cfr_dump_length);
|
|
length += ptr->entry.header.cfr_dump_length;
|
|
pbuf += ptr->entry.header.cfr_dump_length;
|
|
num++;
|
|
}
|
|
}
|
|
DHD_TRACE(("dump %d record %d bytes\n", num, length));
|
|
|
|
return length;
|
|
}
|
|
|