SmartAudio/lichee/linux-4.9/drivers/net/wireless/xradio/wlan/etf.c

886 lines
25 KiB
C
Raw Normal View History

2018-07-13 01:31:50 +00:00
/*
* ETF interfaces for XRadio drivers
*
* Copyright (c) 2013
* Xradio Technology Co., Ltd. <www.xradiotech.com>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/
#ifndef XRADIO_ETF_C
#define XRADIO_ETF_C
#include <linux/module.h>
#include <linux/init.h>
#include <linux/vmalloc.h>
#include <linux/netlink.h>
#include <net/sock.h>
#include <linux/kthread.h>
#include <linux/wait.h>
#include <linux/firmware.h>
#include "etf.h"
#include "xradio.h"
#include "sbus.h"
/********************* interfaces of adapter layer **********************/
static struct xradio_adapter adapter_priv;
#if (ETF_QUEUEMODE)
static void adapter_rec_handler(struct sk_buff *__skb)
{
int i = 0;
struct adapter_item *item;
etf_printk(XRADIO_DBG_TRC, "%s\n", __func__);
do {
spin_lock_bh(&adapter_priv.recv_lock);
if (likely(!list_empty(&adapter_priv.rx_pool))) {
item = list_first_entry(&adapter_priv.rx_pool, struct adapter_item, head);
item->skb = (void *)skb_get(__skb);
list_move_tail(&item->head, &adapter_priv.rx_queue);
spin_unlock_bh(&adapter_priv.recv_lock);
etf_printk(XRADIO_DBG_NIY, "%s recv msg to rx_queue!\n", __func__);
break;
} else if (++i > ETF_QUEUE_TIMEOUT) { /* about 2s.*/
spin_unlock_bh(&adapter_priv.recv_lock);
etf_printk(XRADIO_DBG_ERROR,
"%s driver is process timeout, drop msg!\n", __func__);
return;
} else {
etf_printk(XRADIO_DBG_NIY,
"%s rx_pool is empty, wait %d!ms\n", __func__, i);
}
spin_unlock_bh(&adapter_priv.recv_lock);
msleep(i);
} while (1);
if (atomic_add_return(1, &adapter_priv.rx_cnt) == 1)
up(&adapter_priv.proc_sem);
return;
}
#else
static void adapter_rec_handler(struct sk_buff *skb)
{
struct nlmsghdr *nlhdr = (struct nlmsghdr *)skb->data;
etf_printk(XRADIO_DBG_TRC, "%s\n", __func__);
if (nlhdr->nlmsg_len < sizeof(struct nlmsghdr)) {
etf_printk(XRADIO_DBG_ERROR, "Corrupt netlink msg!\n");
} else {
int len = nlhdr->nlmsg_len - NLMSG_LENGTH(0);
adapter_priv.send_pid = nlhdr->nlmsg_pid;
adapter_priv.msg_len = min(ADAPTER_RX_BUF_LEN, len);
memcpy(adapter_priv.msg_buf, NLMSG_DATA(nlhdr), adapter_priv.msg_len);
if (len != adapter_priv.msg_len)
etf_printk(XRADIO_DBG_WARN, "%s recv len(%d), proc len=%d!\n",
__func__, len, adapter_priv.msg_len);
else
etf_printk(XRADIO_DBG_NIY, "%s recv msg len(%d)!\n",
__func__, adapter_priv.msg_len);
if (adapter_priv.handler)
adapter_priv.handler(adapter_priv.msg_buf, adapter_priv.msg_len);
}
}
#endif
static int adapter_init_msgbuf(void)
{
#if (ETF_QUEUEMODE)
int i = 0;
#endif
etf_printk(XRADIO_DBG_NIY, "%s\n", __func__);
if (adapter_priv.msg_buf) {
etf_printk(XRADIO_DBG_ERROR, "%s msgbuf already init!\n", __func__);
return -1;
}
#if (ETF_QUEUEMODE)
atomic_set(&adapter_priv.rx_cnt, 0);
INIT_LIST_HEAD(&adapter_priv.rx_queue);
INIT_LIST_HEAD(&adapter_priv.rx_pool);
for (i = 0; i < ADAPTER_ITEM_MAX; i++)
list_add_tail(&adapter_priv.rx_items[i].head, &adapter_priv.rx_pool);
#endif
adapter_priv.msg_buf = xr_kzalloc(ADAPTER_RX_BUF_LEN, false);
if (!adapter_priv.msg_buf)
return -ENOMEM;
return 0;
}
static void adapter_deinit_msgbuf(void)
{
#if (ETF_QUEUEMODE)
struct adapter_item *item;
#endif
etf_printk(XRADIO_DBG_NIY, "%s\n", __func__);
if (adapter_priv.msg_buf) {
kfree(adapter_priv.msg_buf);
adapter_priv.msg_buf = NULL;
}
#if (ETF_QUEUEMODE)
spin_lock_bh(&adapter_priv.recv_lock);
list_for_each_entry(item, &adapter_priv.rx_queue, head) {
if (item->skb) {
dev_kfree_skb((struct sk_buff *)item->skb);
item->skb = NULL;
}
}
spin_unlock_bh(&adapter_priv.recv_lock);
#endif
}
#if (ETF_QUEUEMODE)
static int adapter_handle_rx_queue(void)
{
int len = 0;
struct nlmsghdr *nlhdr = NULL;
struct adapter_item *item;
bool bHandler = false;
while (1) {
spin_lock_bh(&adapter_priv.recv_lock);
if (!list_empty(&adapter_priv.rx_queue)) {
item = list_first_entry(&adapter_priv.rx_queue, struct adapter_item, head);
if (item->skb) {
struct sk_buff *skb = (struct sk_buff *)(item->skb);
nlhdr = (struct nlmsghdr *)skb->data;
if (nlhdr->nlmsg_len < sizeof(struct nlmsghdr)) {
etf_printk(XRADIO_DBG_ERROR, "Corrupt netlink msg!\n");
} else {
len = nlhdr->nlmsg_len - NLMSG_LENGTH(0);
adapter_priv.send_pid = nlhdr->nlmsg_pid;
adapter_priv.msg_len = min(ADAPTER_RX_BUF_LEN, len);
memcpy(adapter_priv.msg_buf, NLMSG_DATA(nlhdr), len);
bHandler = true;
if (len != adapter_priv.msg_len)
etf_printk(XRADIO_DBG_WARN, "%s recv len(%d), proc len=%d!\n",
__func__, len, adapter_priv.msg_len);
else
etf_printk(XRADIO_DBG_NIY, "%s recv msg len(%d)!\n",
__func__, adapter_priv.msg_len);
}
dev_kfree_skb(skb);
item->skb = NULL;
list_move_tail(&item->head, &adapter_priv.rx_pool);
} else {
etf_printk(XRADIO_DBG_ERROR, "%s skb is NULL!\n", __func__);
}
} else {
spin_unlock_bh(&adapter_priv.recv_lock);
break;
}
spin_unlock_bh(&adapter_priv.recv_lock);
if (bHandler && adapter_priv.handler) {
adapter_priv.handler(adapter_priv.msg_buf, adapter_priv.msg_len);
bHandler = false;
}
}
return 0;
}
static int adapter_proc(void *param)
{
int ret = 0;
int term = 0;
int recv = 0;
etf_printk(XRADIO_DBG_NIY, "%s\n", __func__);
for (;;) {
ret = down_interruptible(&adapter_priv.proc_sem);
recv = atomic_xchg(&adapter_priv.rx_cnt, 0);
term = kthread_should_stop();
etf_printk(XRADIO_DBG_NIY, "%s term=%d!\n", __func__, term);
if (term || adapter_priv.exit || ret < 0) {
break;
}
if (recv) {
adapter_handle_rx_queue();
}
}
etf_printk(XRADIO_DBG_NIY, "%s: exit!\n", __func__);
return 0;
}
static int adapter_start_thread(void)
{
int ret = 0;
struct sched_param param = {.sched_priority = 100 };
etf_printk(XRADIO_DBG_NIY, "%s\n", __func__);
adapter_priv.thread_tsk = NULL;
adapter_priv.exit = 0;
adapter_priv.thread_tsk = kthread_create(&adapter_proc, NULL, XRADIO_ADAPTER);
if (IS_ERR(adapter_priv.thread_tsk)) {
ret = PTR_ERR(adapter_priv.thread_tsk);
adapter_priv.thread_tsk = NULL;
} else {
sched_setscheduler(adapter_priv.thread_tsk, SCHED_NORMAL, &param);
wake_up_process(adapter_priv.thread_tsk);
}
return ret;
}
static void adapter_stop_thread(void)
{
etf_printk(XRADIO_DBG_NIY, "%s\n", __func__);
if (adapter_priv.thread_tsk != NULL) {
adapter_priv.exit = 1;
up(&adapter_priv.proc_sem);
kthread_stop(adapter_priv.thread_tsk);
adapter_priv.thread_tsk = NULL;
} else {
etf_printk(XRADIO_DBG_WARN, "%s: thread_tsk is NULL!\n", __func__);
}
}
#endif /* ETF_QUEUEMODE */
static int xradio_adapter_send(void *data, int len)
{
int ret = 0;
struct sk_buff *skb = NULL;
struct nlmsghdr *nlhdr = NULL;
u8 *payload = NULL;
if (0 == len || NULL == data)
return -EINVAL;
skb = xr_alloc_skb(NLMSG_SPACE(len + 1));
if (!skb) {
etf_printk(XRADIO_DBG_WARN, "%s:xr_alloc_skb failed!\n", __func__);
return -ENOMEM;
}
/* '\0' safe for strcpy, compat for old etf apk, may be removed in furture.*/
nlhdr = nlmsg_put(skb, 0, 0, 0, len + 1, 0); /* (len+1) is payload */
payload = (u8 *)NLMSG_DATA(nlhdr);
memcpy(payload, data, len);
payload[len] = '\0';
#if LINUX_VERSION_CODE < KERNEL_VERSION(3, 7, 0)
NETLINK_CB(skb).pid = 0;
#else
NETLINK_CB(skb).portid = 0;
#endif
NETLINK_CB(skb).dst_group = 0;
down(&adapter_priv.send_lock);
ret = netlink_unicast(adapter_priv.sock, skb, adapter_priv.send_pid, 0);
if (ret < 0)
etf_printk(XRADIO_DBG_ERROR, "%s:netlink_unicast failed(%d),pid=%d!\n",
__func__, ret, adapter_priv.send_pid);
else
etf_printk(XRADIO_DBG_NIY, "%s:netlink_unicast len(%d),pid=%d!\n",
__func__, ret, adapter_priv.send_pid);
up(&adapter_priv.send_lock);
return ret;
}
static int xradio_adapter_send_pkg(void *data1, int len1, void *data2, int len2)
{
int ret = 0;
struct sk_buff *skb = NULL;
struct nlmsghdr *nlhdr = NULL;
u8 *payload = NULL;
int total_len = len1 + len2;
if (len1 <= 0 || len2 < 0 || NULL == data1)
return -EINVAL;
skb = xr_alloc_skb(NLMSG_SPACE(total_len + 1));
if (!skb) {
etf_printk(XRADIO_DBG_WARN, "%s:xr_alloc_skb failed!\n", __func__);
return -ENOMEM;
}
/* '\0' safe for strcpy, compat for old etf apk, may be removed in furture.*/
nlhdr = nlmsg_put(skb, 0, 0, 0, total_len + 1, 0);
payload = (u8 *)NLMSG_DATA(nlhdr);
memcpy(payload, data1, len1);
if (data2 && len2)
memcpy((payload + len1), data2, len2);
payload[total_len] = '\0';
#if LINUX_VERSION_CODE < KERNEL_VERSION(3, 7, 0)
NETLINK_CB(skb).pid = 0;
#else
NETLINK_CB(skb).portid = 0;
#endif
NETLINK_CB(skb).dst_group = 0;
down(&adapter_priv.send_lock);
ret = netlink_unicast(adapter_priv.sock, skb, adapter_priv.send_pid, 0);
if (ret < 0)
etf_printk(XRADIO_DBG_ERROR, "%s:netlink_unicast failed(%d),pid=%d!\n",
__func__, ret, adapter_priv.send_pid);
else
etf_printk(XRADIO_DBG_NIY, "%s:netlink_unicast len(%d),pid=%d!\n",
__func__, ret, adapter_priv.send_pid);
up(&adapter_priv.send_lock);
return ret;
}
struct xradio_adapter *xradio_adapter_init(msg_proc func)
{
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(3, 6, 0))
struct netlink_kernel_cfg cfg = {
.input = adapter_rec_handler,
};
#endif
if (func == NULL) {
etf_printk(XRADIO_DBG_ERROR, "%s msg_proc is NULL\n", __func__);
return NULL;
}
adapter_priv.handler = func;
sema_init(&adapter_priv.send_lock, 1);
if (adapter_init_msgbuf()) {
etf_printk(XRADIO_DBG_ERROR, "%s adapter_init_msgbuf failed\n", __func__);
return NULL;
}
#if (ETF_QUEUEMODE)
sema_init(&adapter_priv.proc_sem, 0);
if (adapter_start_thread()) {
adapter_deinit_msgbuf();
etf_printk(XRADIO_DBG_ERROR, "%s adapter_start_thread failed\n", __func__);
return NULL;
}
#endif
#if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 6, 0))
adapter_priv.sock = netlink_kernel_create(&init_net, NL_FOR_XRADIO, 0,
adapter_rec_handler, NULL, THIS_MODULE);
#elif (LINUX_VERSION_CODE < KERNEL_VERSION(3, 7, 0))
adapter_priv.sock = netlink_kernel_create(&init_net, NL_FOR_XRADIO,
THIS_MODULE, &cfg);
#else
adapter_priv.sock = netlink_kernel_create(&init_net, NL_FOR_XRADIO, &cfg);
#endif
if (adapter_priv.sock == NULL) {
#if (ETF_QUEUEMODE)
adapter_stop_thread();
#endif
adapter_deinit_msgbuf();
etf_printk(XRADIO_DBG_ERROR, "%s netlink_kernel_create failed\n", __func__);
return NULL;
}
return &adapter_priv;
}
void xradio_adapter_deinit(void)
{
if (adapter_priv.sock != NULL) {
netlink_kernel_release(adapter_priv.sock);
adapter_priv.sock = NULL;
}
#if (ETF_QUEUEMODE)
adapter_stop_thread();
#endif
adapter_deinit_msgbuf();
}
/********************* interfaces of etf core **********************/
static struct xradio_etf etf_priv;
static struct etf_api_context context_save;
int wsm_etf_cmd(struct xradio_common *hw_priv, struct wsm_hdr *arg);
int xradio_bh_suspend(struct xradio_common *hw_priv);
int xradio_bh_resume(struct xradio_common *hw_priv);
bool etf_is_connect(void)
{
return (bool)(etf_priv.etf_state != ETF_STAT_NULL);
}
void etf_set_core(void *core_priv)
{
etf_priv.core_priv = core_priv;
}
const char *etf_get_fwpath(void)
{
return (const char *)etf_priv.fw_path;
}
const char *etf_get_sddpath(void)
{
return (const char *)etf_priv.sdd_path;
}
static int etf_get_sdd_param(u8 *sdd, int sdd_len, u8 ies, void **data)
{
int ie_len = 0;
int parsedLength = 0;
struct xradio_sdd *pElement = NULL;
etf_printk(XRADIO_DBG_TRC, "%s\n", __func__);
/*parse SDD config.*/
pElement = (struct xradio_sdd *)sdd;
parsedLength += (FIELD_OFFSET(struct xradio_sdd, data) + \
pElement->length);
pElement = FIND_NEXT_ELT(pElement);
while (parsedLength < sdd_len) {
if (pElement->id == ies) {
*data = (void *)pElement->data;
ie_len = pElement->length;
}
parsedLength += (FIELD_OFFSET(struct xradio_sdd, data) + \
pElement->length);
pElement = FIND_NEXT_ELT(pElement);
}
xradio_dbg(XRADIO_DBG_MSG, "parse len=%d, ie_len=%d.\n",
parsedLength, ie_len);
return ie_len;
}
static int etf_driver_resp(int state, int result)
{
struct drv_resp drv_ret;
drv_ret.len = sizeof(drv_ret);
drv_ret.id = ETF_DRIVER_IND_ID;
drv_ret.state = state;
drv_ret.result = result;
return xradio_adapter_send(&drv_ret, drv_ret.len);
}
static int xradio_etf_connect(void)
{
int ret = BOOT_SUCCESS;
etf_printk(XRADIO_DBG_TRC, "%s\n", __func__);
if (etf_priv.etf_state == ETF_STAT_NULL) {
etf_priv.etf_state = ETF_STAT_CONNECTING;
ret = xradio_core_init();
if (ret) {
etf_printk(XRADIO_DBG_ERROR, "%s:xradio_core_init failed(%d).\n",
__func__, ret);
goto err;
}
etf_priv.etf_state = ETF_STAT_CONNECTED;
} else {
etf_printk(XRADIO_DBG_WARN, "%s etf is already connect.\n", __func__);
return ETF_ERR_CONNECTED;
}
etf_printk(XRADIO_DBG_ALWY, "%s:ETF connect success.\n", __func__);
return 0;
err:
etf_priv.etf_state = ETF_STAT_NULL;
return ret;
}
static void xradio_etf_disconnect(void)
{
etf_printk(XRADIO_DBG_TRC, "%s\n", __func__);
if (etf_priv.etf_state != ETF_STAT_NULL) {
xradio_core_deinit();
etf_priv.etf_state = ETF_STAT_NULL;
}
etf_printk(XRADIO_DBG_ALWY, "%s end.\n", __func__);
}
static int xradio_etf_proc(void *data, int len)
{
int ret = BOOT_SUCCESS;
struct wsm_hdr *hdr_rev = (struct wsm_hdr *)data;
struct drv_download *download;
int datalen;
etf_printk(XRADIO_DBG_TRC, "%s\n", __func__);
down(&etf_priv.etf_lock);
/* Don't process ETF cmd in wlan mode */
if (etf_priv.is_wlan) {
etf_printk(XRADIO_DBG_WARN, "%s is in wlan mode.\n", __func__);
ret = ETF_ERR_WLAN_MODE;
etf_driver_resp(BOOT_STATE_NULL, ret);
goto out;
}
/* Process msg from ETF API */
switch (MSG_ID(hdr_rev->id)) {
case ETF_SOFT_RESET_REQ_ID:
etf_printk(XRADIO_DBG_ALWY, "SOFT_RESET\n");
xradio_etf_disconnect();
etf_priv.fw_path = XR819_ETF_FIRMWARE;
etf_priv.sdd_path = XR819_SDD_FILE;
etf_priv.seq_send = 0;
etf_priv.seq_recv = 0;
memset(&context_save, 0, sizeof(context_save));
etf_driver_resp(BOOT_STATE_NULL, BOOT_SUCCESS);
break;
case ETF_GET_API_CONTEXT_ID: /* for ETF tools which cannot save context*/
{
struct etf_api_context_result *ctxt_ret = NULL;
struct xradio_common *hw_priv = etf_priv.core_priv;
ctxt_ret = (struct etf_api_context_result *)hdr_rev;
if (hw_priv) {
ctxt_ret->is_etf_fw_run = hw_priv->wsm_caps.firmwareReady;
memcpy(ctxt_ret->fw_label, hw_priv->wsm_caps.fw_label,
min(HI_SW_LABEL_MAX, WSM_FW_LABEL));
ctxt_ret->fw_api_ver = hw_priv->wsm_caps.firmwareApiVer;
ctxt_ret->mib_baseaddr = context_save.config[1];
} else {
ctxt_ret->is_etf_fw_run = 0;
memset(ctxt_ret->fw_label, 0, HI_SW_LABEL_MAX);
}
ctxt_ret->len = sizeof(*ctxt_ret);
ctxt_ret->id |= ETF_CNF_BASE;
ctxt_ret->result = 0;
xradio_adapter_send(ctxt_ret, sizeof(*ctxt_ret));
}
break;
case ETF_DOWNLOAD_ID:
etf_printk(XRADIO_DBG_NIY, "DOWNLOAD_FW, size=%d\n", hdr_rev->len);
download = (struct drv_download *)(hdr_rev + 1);
datalen = hdr_rev->len - sizeof(*download) - sizeof(*hdr_rev);
if (datalen <= 0) {
/*it means old version APK, TODO:remove it in future */
if (hdr_rev->len <= sizeof(*hdr_rev)) {
etf_driver_resp(BOOT_COMPLETE, BOOT_SUCCESS);
ret = xradio_etf_connect();
if (ret)
etf_driver_resp(BOOT_STATE_NULL, ret);
}
break;
}
if (download->flags & DOWNLOAD_F_PATH_ONLY) {
etf_printk(XRADIO_DBG_WARN, "fw path=%s\n", (char *)(download+1));
/*TODO: set etf_priv.fw_path*/
} else if (download->flags & DOWNLOAD_F_START) {
etf_printk(XRADIO_DBG_NIY, "fw first block\n");
/*TODO: open a tmp file to save firmware */
}
/*TODO: put data into a tmp file */
/*TODO: put data into a tmp file */
if ((download->flags & DOWNLOAD_F_END)) {
etf_printk(XRADIO_DBG_NIY, "fw last block\n");
etf_driver_resp(BOOT_COMPLETE, BOOT_SUCCESS);
/*TODO: save data tmp file and update etf_priv.fw_path*/
ret = xradio_etf_connect();
if (ret)
etf_driver_resp(BOOT_STATE_NULL, ret);
}
break;
case ETF_DOWNLOAD_SDD_ID:
etf_printk(XRADIO_DBG_NIY, "DOWNLOAD_SDD, size=%d\n", hdr_rev->len);
download = (struct drv_download *)(hdr_rev + 1);
datalen = hdr_rev->len - sizeof(*download) - sizeof(*hdr_rev);
if (datalen <= 0) {
/*it means old version APK, TODO:remove it in future */
if (hdr_rev->len <= sizeof(*hdr_rev)) {
etf_driver_resp(BOOT_SDD_COMPLETE, BOOT_SUCCESS);
}
break;
}
if (download->flags & DOWNLOAD_F_PATH_ONLY) {
etf_printk(XRADIO_DBG_WARN, "sdd path=%s\n", (char *)(download+1));
/*TODO: set etf_priv.fw_path*/
} else if ((download->flags & DOWNLOAD_F_START)) {
etf_printk(XRADIO_DBG_NIY, "sdd first block\n");
/*TODO: open a tmp file to save firmware */
}
/*TODO: put data into a tmp file */
if ((download->flags & DOWNLOAD_F_END)) {
etf_printk(XRADIO_DBG_NIY, "sdd last block\n");
etf_driver_resp(BOOT_SDD_COMPLETE, BOOT_SUCCESS);
/*TODO: save data tmp file and update etf_priv.sdd_path*/
}
break;
case ETF_CONNECT_ID:
etf_printk(XRADIO_DBG_ALWY, "ETF_CONNECT_ID!\n");
etf_driver_resp(BOOT_COMPLETE, BOOT_SUCCESS);
ret = xradio_etf_connect();
if (ret)
etf_driver_resp(BOOT_STATE_NULL, ret);
break;
case ETF_DISCONNECT_ID:
etf_printk(XRADIO_DBG_ALWY, "ETF_DISCONNECT_ID!\n");
xradio_etf_disconnect();
etf_driver_resp(BOOT_STATE_NULL, BOOT_SUCCESS);
break;
case ETF_RECONNECT_ID:
etf_printk(XRADIO_DBG_ALWY, "ETF_RECONNECT_ID!\n");
if (1/*TODO: is update files*/) {
xradio_etf_disconnect();
etf_driver_resp(BOOT_COMPLETE, BOOT_SUCCESS);
ret = xradio_etf_connect();
if (ret)
etf_driver_resp(BOOT_STATE_NULL, ret);
} else {
ret = BOOT_ERR_BAD_OP;
etf_driver_resp(BOOT_STATE_NULL, ret);
etf_printk(XRADIO_DBG_ERROR, "No download files before!\n");
}
break;
case ETF_GET_SDD_POWER_DEFT: /* get default tx power */
{
#ifdef USE_VFS_FIRMWARE
const struct xr_file *sdd = NULL;
#else
const struct firmware *sdd = NULL;
#endif
etf_printk(XRADIO_DBG_NIY, "%s ETF_GET_SDD_POWER_DEFT!\n", __func__);
#ifdef USE_VFS_FIRMWARE
sdd = xr_request_file(etf_get_sddpath());
#else
ret = request_firmware(&sdd, etf_get_sddpath(), NULL);
#endif
/* get sdd data */
if (likely(sdd) && likely(sdd->data)) {
int i;
u16 *outpower = (u16 *)(hdr_rev + 1);
void *ie_data = NULL;
int ie_len = etf_get_sdd_param((u8 *)sdd->data, sdd->size,
SDD_MAX_OUTPUT_POWER_2G4_ELT_ID, &ie_data);
if (ie_len > 0 && ie_data &&
ie_len < (ADAPTER_RX_BUF_LEN - sizeof(*hdr_rev))) {
memcpy(outpower, ie_data, ie_len);
for (i = 0; i < (ie_len>>1); i++)
outpower[i] -= 48; /*default pwr = max power - 48*/
}
hdr_rev->len = sizeof(*hdr_rev) + ie_len;
xradio_adapter_send(hdr_rev, hdr_rev->len);
#ifdef USE_VFS_FIRMWARE
xr_fileclose(sdd);
#else
release_firmware(sdd);
#endif
} else {
etf_printk(XRADIO_DBG_ERROR, "%s: can't load sdd file %s.\n",
__func__, etf_get_sddpath());
ret = -ENOENT;
hdr_rev->len = sizeof(*hdr_rev);
xradio_adapter_send(hdr_rev, sizeof(*hdr_rev));
}
}
break;
case ETF_GET_SDD_PARAM_ID:
{
struct xradio_common *hw_priv = etf_priv.core_priv;
struct get_sdd_param_req *sddreq = (struct get_sdd_param_req *)hdr_rev;
struct get_sdd_result *sddret = (struct get_sdd_result *)sddreq;
void *ie_data = NULL;
int ie_len = 0;
#ifdef USE_VFS_FIRMWARE
const struct xr_file *sdd = NULL;
#else
const struct firmware *sdd = NULL;
#endif
etf_printk(XRADIO_DBG_NIY, "%s ETF_GET_SDD_PARAM_ID!\n", __func__);
if (hw_priv && hw_priv->sdd) {
etf_printk(XRADIO_DBG_NIY, "%s use xradio_common sdd!\n", __func__);
sdd = hw_priv->sdd;
} else {
#ifdef USE_VFS_FIRMWARE
sdd = xr_request_file(etf_get_sddpath());
#else
ret = request_firmware(&sdd, etf_get_sddpath(), NULL);
#endif
}
hdr_rev->id = hdr_rev->id + ETF_CNF_BASE;
if (likely(sdd) && likely(sdd->data)) {
if (sddreq->flags & FLAG_GET_SDD_ALL) {
sddret->result = 0;
sddret->length = sdd->size;
xradio_adapter_send_pkg(sddret, sizeof(*sddret), (u8 *)sdd->data,
sddret->length);
} else {
ie_len = etf_get_sdd_param((u8 *)sdd->data, sdd->size,
sddreq->ies, &ie_data);
if (ie_data && ie_len > 0) {
sddret->result = 0;
sddret->length = ie_len;
hdr_rev->len = sizeof(*sddret) + sddret->length;
xradio_adapter_send_pkg(sddret, sizeof(*sddret), ie_data,
sddret->length);
} else {
sddret->result = -ENOMSG;
sddret->length = 0;
hdr_rev->len = sizeof(*sddret) + sddret->length;
xradio_adapter_send(sddret, sizeof(*sddret));
}
}
if (hw_priv && hw_priv->sdd) {
sdd = NULL;
} else {
#ifdef USE_VFS_FIRMWARE
xr_fileclose(sdd);
#else
release_firmware(sdd);
#endif
}
} else {
etf_printk(XRADIO_DBG_ERROR, "%s: can't load sdd file %s.\n",
__func__, etf_get_sddpath());
sddret->result = -ENOENT;
sddret->length = 0;
hdr_rev->len = sizeof(*sddret) + sddret->length;
xradio_adapter_send(sddret, sizeof(*sddret));
}
}
break;
default:
etf_printk(XRADIO_DBG_NIY, "%s: passed cmd=0x%04x, len=%d",
__func__, MSG_ID(hdr_rev->id), hdr_rev->len);
if (etf_priv.etf_state != ETF_STAT_CONNECTED) {
etf_printk(XRADIO_DBG_WARN, "%s etf Not connect.\n", __func__);
ret = ETF_ERR_NOT_CONNECT;
etf_driver_resp(BOOT_STATE_NULL, ret);
goto out;
}
if (SEQ_MASK(etf_priv.seq_send) != MSG_SEQ(hdr_rev->id)) {
etf_printk(XRADIO_DBG_NIY, "%s:seq err, drv_seq=%d, seq=%d\n",
__func__, etf_priv.seq_send, MSG_SEQ(hdr_rev->id));
etf_priv.seq_send = MSG_SEQ(hdr_rev->id);
}
etf_priv.seq_send++;
if (ETF_HWT_REQ == MSG_ID(hdr_rev->id)) {
/*TODO: HW test*/
etf_printk(XRADIO_DBG_ERROR, "%s HW test unsupport\n", __func__);
} else if (etf_priv.core_priv) {
if (ETF_DOWNLOAD_SDD == MSG_ID(hdr_rev->id) &&
hdr_rev->len <= sizeof(struct etf_sdd_req)) {
struct xradio_common *hw_priv = etf_priv.core_priv;
struct etf_sdd_req *sdd_req = (struct etf_sdd_req *)hdr_rev;
u32 sdd_cmd = sdd_req->sdd_cmd;
etf_printk(XRADIO_DBG_WARN, "%s add sdd data, cmd=0x%08x!\n",
__func__, sdd_cmd);
if (hw_priv->sdd) {
u8 *sdd_data = (u8 *)(hdr_rev + 1);
int sdd_len = hw_priv->sdd->size;
if (sdd_len + sizeof(*hdr_rev) + 4 < ADAPTER_RX_BUF_LEN) {
memcpy(sdd_data, hw_priv->sdd->data, sdd_len);
memcpy(sdd_data + sdd_len, &sdd_cmd, 4);
hdr_rev->len += sdd_len;
} else {
etf_printk(XRADIO_DBG_ERROR,
"ADAPTER_RX_BUF_LEN=%d, sdd_len=%d\n",
ADAPTER_RX_BUF_LEN, sdd_len);
}
} else
etf_printk(XRADIO_DBG_ERROR, "%s core_priv sdd is NULL\n",
__func__);
}
/* send to device*/
ret = wsm_etf_cmd(etf_priv.core_priv, (struct wsm_hdr *)data);
if (ret < 0) {
etf_driver_resp(BOOT_STATE_NULL, ret);
etf_printk(XRADIO_DBG_ERROR, "%s wsm_etf_cmd failed(%d)\n",
__func__, ret);
}
} else {
etf_printk(XRADIO_DBG_ERROR, "%s core_priv is NULL\n",
__func__);
}
break;
}
up(&etf_priv.etf_lock);
etf_printk(XRADIO_DBG_NIY, "%s success\n", __func__);
return 0;
out:
up(&etf_priv.etf_lock);
if (ret)
etf_printk(XRADIO_DBG_ERROR, "%s proc failed(%d)\n", __func__, ret);
return ret;
}
int xradio_etf_from_device(struct sk_buff **skb)
{
int ret = 0;
struct wsm_hdr *wsm = (struct wsm_hdr *)((*skb)->data);
etf_printk(XRADIO_DBG_NIY, "%s MsgId=0x%04x, len=%d\n", __func__,
wsm->id, wsm->len);
ret = xradio_adapter_send((void *)wsm, wsm->len);
if (ret < 0) {
etf_printk(XRADIO_DBG_ERROR, "%s xradio_adapter_send failed(%d).\n",
__func__, ret);
} else {
etf_priv.seq_recv++;
}
return 0;
}
void xradio_etf_save_context(void *buf, int len)
{
etf_printk(XRADIO_DBG_TRC, "%s\n", __func__);
if (len == sizeof(context_save)) {
etf_printk(XRADIO_DBG_NIY, "%s, len=%d\n", __func__, len);
memcpy(&context_save, buf, sizeof(context_save));
} else if (len > sizeof(context_save)) {
etf_printk(XRADIO_DBG_WARN, "%s, len=%d\n", __func__, len);
memcpy(&context_save, buf, sizeof(context_save));
} else {
etf_printk(XRADIO_DBG_ERROR, "%s,len too short=%d\n", __func__, len);
}
}
void xradio_etf_to_wlan(u32 change)
{
down(&etf_priv.etf_lock);
etf_priv.is_wlan = change;
if (change && etf_is_connect()) {
etf_printk(XRADIO_DBG_NIY, "%s change to wlan.\n", __func__);
xradio_etf_disconnect();
}
up(&etf_priv.etf_lock);
}
EXPORT_SYMBOL_GPL(xradio_etf_to_wlan);
int xradio_etf_suspend(void)
{
etf_printk(XRADIO_DBG_NIY, "%s\n", __func__);
if (down_trylock(&etf_priv.etf_lock)) {
etf_printk(XRADIO_DBG_WARN,
"%s etf_lock is locked\n", __func__);
return -EBUSY; /*etf is locked.*/
}
if (etf_is_connect()) {
etf_printk(XRADIO_DBG_WARN,
"%s etf is running, don't suspend!\n", __func__);
up(&etf_priv.etf_lock);
return -EBUSY; /*etf is running*/
}
up(&etf_priv.etf_lock);
return 0;
}
int xradio_etf_resume(void)
{
etf_printk(XRADIO_DBG_NIY, "%s\n", __func__);
return 0;
}
int xradio_etf_init(void)
{
int ret = 0;
etf_printk(XRADIO_DBG_TRC, "%s\n", __func__);
sema_init(&etf_priv.etf_lock, 1);
etf_priv.etf_state = ETF_STAT_NULL;
etf_priv.fw_path = XR819_ETF_FIRMWARE;
etf_priv.sdd_path = XR819_SDD_FILE;
etf_priv.adapter = xradio_adapter_init(&xradio_etf_proc);
return ret;
}
void xradio_etf_deinit(void)
{
etf_printk(XRADIO_DBG_TRC, "%s\n", __func__);
/*ensure in disconnect state when etf deinit*/
xradio_etf_to_wlan(1);
xradio_adapter_deinit();
memset(&etf_priv, 0, sizeof(etf_priv));
}
#endif /*XRADIO_ETF_C*/