Compare commits
29 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| a70e882f3d | |||
| 445c619da8 | |||
| 8fd20308ed | |||
| b9e340c78c | |||
| 0f185172b0 | |||
| d9e6361cf0 | |||
| 2fb72f06e5 | |||
| 737f28e29c | |||
| 06fab77745 | |||
| 91640bd96d | |||
| 3942404ae1 | |||
| dc1505f5ba | |||
| 7a65dc374c | |||
| 40aec44a6e | |||
| 22b799f5a2 | |||
| 179c2e0bf8 | |||
| 844e2245a1 | |||
| 1d57b6b342 | |||
| 894c502cd5 | |||
| c3d28aea46 | |||
| 0768e11da6 | |||
| 0c1f98df59 | |||
| 2f9b4c719f | |||
| 789ab98e4c | |||
| cd5f85e104 | |||
| e8550e55fc | |||
| 40a0a1a0c7 | |||
| 13dd191def | |||
| 4a1bbc7444 |
+3
-15
@@ -2,7 +2,7 @@ KERN_DIR = /lib/modules/$(KERNELRELEASE)/build
|
||||
# feature option
|
||||
BES2600 ?= m
|
||||
|
||||
CONFIG_BES2600_TESTMODE ?= n
|
||||
CONFIG_BES2600_TESTMODE ?= y
|
||||
|
||||
CONFIG_BES2600_ENABLE_DEVEL_LOGS ?= n
|
||||
|
||||
@@ -28,7 +28,6 @@ CONFIG_BES2600_WIFI_BOOT_ON ?= y
|
||||
CONFIG_BES2600_BT_BOOT_ON ?= n
|
||||
|
||||
BES2600_GPIO_WAKEUP_AP ?= n
|
||||
BES2600_WRITE_DPD_TO_FILE ?= n
|
||||
BES2600_TX_MORE_RETRY ?= n
|
||||
|
||||
# bes evb
|
||||
@@ -65,8 +64,8 @@ BES2600_DRV_VERSION := bes2600_0.3.5_2024.0116
|
||||
|
||||
ifeq ($(CONFIG_BES2600_CALIB_FROM_LINUX),y)
|
||||
FACTORY_CRC_CHECK ?= n
|
||||
STANDARD_FACTORY_EFUSE_FLAG ?= y
|
||||
FACTORY_PATH ?= /lib/firmware/bes2600_factory.txt
|
||||
STANDARD_FACTORY_EFUSE_FLAG ?= n
|
||||
FACTORY_PATH ?= bes2600/bes2600_factory.txt
|
||||
endif
|
||||
|
||||
# basic function
|
||||
@@ -93,12 +92,6 @@ ccflags-y += -DBES_UNIFIED_PM
|
||||
ccflags-y += -DBES_SDIO_OPTIMIZED_LEN
|
||||
ccflags-y += -DBES2600_HOST_TIMESTAMP_DEBUG
|
||||
|
||||
ifeq ($(BES2600_WRITE_DPD_TO_FILE),y)
|
||||
BES2600_DPD_PATH ?= /data/cfg/bes2600_dpd.bin
|
||||
BES2600_DEFAULT_DPD_PATH ?= /lib/firmware/bes2600_dpd.bin
|
||||
BES2600_DPD_GOLDEN_PATH ?= /data/cfg/bes2600_dpd_golden.bin
|
||||
endif
|
||||
|
||||
ifeq ($(BES2600_DUMP_FW_DPD_LOG),y)
|
||||
BES2600_DPD_LOG_PATH ?= /data/applog/bes2600_dpd_log.log
|
||||
endif
|
||||
@@ -135,9 +128,6 @@ ccflags-y += $(call boolen_flag,BSS_LOSS_CHECK,y)
|
||||
ccflags-y += $(call string_flag,BES2600_LOAD_FW_TOOL_PATH)
|
||||
ccflags-y += $(call string_flag,BES2600_LOAD_FW_TOOL_DEVICE)
|
||||
ccflags-y += $(call string_flag,BES2600_DRV_VERSION)
|
||||
ccflags-y += $(call string_flag,BES2600_DPD_PATH)
|
||||
ccflags-y += $(call string_flag,BES2600_DEFAULT_DPD_PATH)
|
||||
ccflags-y += $(call string_flag,BES2600_DPD_GOLDEN_PATH)
|
||||
|
||||
ccflags-y += $(call boolen_flag,BES2600_INDEPENDENT_EVB,y)
|
||||
ccflags-y += $(call boolen_flag,BES2600_INTEGRATED_MODULE_V1,y)
|
||||
@@ -159,8 +149,6 @@ ccflags-y += $(call boolen_flag,FACTORY_SAVE_MULTI_PATH,y)
|
||||
ccflags-y += $(call boolen_flag,FACTORY_CRC_CHECK,y)
|
||||
|
||||
ccflags-y += $(call boolen_flag,BES2600_GPIO_WAKEUP_AP,y)
|
||||
ccflags-y += $(call boolen_flag,BES2600_WRITE_DPD_TO_FILE,y)
|
||||
|
||||
ccflags-y += $(call boolen_flag,BES2600_DUMP_FW_DPD_LOG,y)
|
||||
ccflags-y += $(call string_flag,BES2600_DPD_LOG_PATH)
|
||||
|
||||
|
||||
+43
-65
@@ -9,6 +9,7 @@
|
||||
#include <linux/module.h>
|
||||
#include <linux/sched.h>
|
||||
#include <linux/fs.h>
|
||||
#include <linux/firmware.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/mutex.h>
|
||||
#include <linux/crc32.h>
|
||||
@@ -27,6 +28,18 @@
|
||||
|
||||
static DEFINE_MUTEX(factory_lock);
|
||||
|
||||
/*
|
||||
* struct device * for request_firmware() context. Set once at SDIO
|
||||
* probe via bes2600_factory_set_dev(). NULL is tolerated (falls back
|
||||
* to the udev-less firmware-class path) but loses per-device logging.
|
||||
*/
|
||||
static struct device *bes2600_factory_dev;
|
||||
|
||||
void bes2600_factory_set_dev(struct device *dev)
|
||||
{
|
||||
bes2600_factory_dev = dev;
|
||||
}
|
||||
|
||||
/*
|
||||
* It is only used for temporary storage.
|
||||
* Every time get the factory, it will read from the
|
||||
@@ -134,66 +147,32 @@ static int bes2600_factory_crc_check(struct factory_t *factory_data)
|
||||
*/
|
||||
static int factory_section_read_file(char *path, void *buffer)
|
||||
{
|
||||
int ret = 0;
|
||||
struct file *fp;
|
||||
const struct firmware *fw;
|
||||
int ret;
|
||||
|
||||
if (!path || !buffer) {
|
||||
bes_err("%s NULL pointer err\n", __func__);
|
||||
return -1;
|
||||
}
|
||||
|
||||
bes_devel("reading %s \n", path);
|
||||
bes_devel("requesting firmware-class %s\n", path);
|
||||
|
||||
fp = filp_open(path, O_RDONLY, 0); //S_IRUSR
|
||||
if (IS_ERR(fp)) {
|
||||
bes_devel("BES2600 : can't open %s\n",path);
|
||||
ret = request_firmware(&fw, path, bes2600_factory_dev);
|
||||
if (ret) {
|
||||
bes_devel("BES2600: request_firmware(%s) failed: %d\n", path, ret);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (fp->f_inode->i_size <= 0 || fp->f_inode->i_size > FACTORY_MAX_SIZE) {
|
||||
bes_err( "bes2600_factory.txt size check failed, read_size: %lld max_size: %d\n",
|
||||
fp->f_inode->i_size, FACTORY_MAX_SIZE);
|
||||
filp_close(fp, NULL);
|
||||
if (fw->size == 0 || fw->size > FACTORY_MAX_SIZE) {
|
||||
bes_err("bes2600_factory.txt size check failed, read_size: %zu max_size: %d\n",
|
||||
fw->size, FACTORY_MAX_SIZE);
|
||||
release_firmware(fw);
|
||||
return -1;
|
||||
}
|
||||
|
||||
ret = kernel_read(fp, buffer, fp->f_inode->i_size, &fp->f_pos);
|
||||
|
||||
filp_close(fp, NULL);
|
||||
|
||||
if (ret != fp->f_inode->i_size) {
|
||||
bes_err("bes2600_factory.txt read fail\n");
|
||||
ret = -1;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/**
|
||||
* factory_section_write_file - Write data of specified length to file
|
||||
* @path: path of the file
|
||||
* @buffer: storage of write data
|
||||
* @size: length of data to write
|
||||
*
|
||||
* Return: length on success, negative error code otherwise.
|
||||
*/
|
||||
static int factory_section_write_file(char *path, void *buffer, int size)
|
||||
{
|
||||
int ret = 0;
|
||||
struct file *fp;
|
||||
|
||||
bes_devel("writing %s \n", path);
|
||||
|
||||
fp = filp_open(path, O_TRUNC | O_CREAT | O_RDWR, S_IRUSR);
|
||||
if (IS_ERR(fp)) {
|
||||
bes_devel("BES2600 : can't open %s\n",path);
|
||||
return -1;
|
||||
}
|
||||
|
||||
ret = kernel_write(fp, buffer, size, &fp->f_pos);
|
||||
|
||||
filp_close(fp,NULL);
|
||||
|
||||
memcpy(buffer, fw->data, fw->size);
|
||||
ret = (int)fw->size;
|
||||
release_firmware(fw);
|
||||
return ret;
|
||||
}
|
||||
|
||||
@@ -888,9 +867,22 @@ static inline int factory_build(uint8_t *dest_buf, struct factory_t *factory)
|
||||
#endif
|
||||
}
|
||||
|
||||
/*
|
||||
* Rebuild the serialised calibration blob in file_buffer from the live
|
||||
* in-memory factory_save_p. Previously this function also persisted the
|
||||
* blob back to FACTORY_PATH via filp_open(O_CREAT) + kernel_write(); that
|
||||
* is not acceptable in mainline, so the persistence step has been removed.
|
||||
*
|
||||
* The in-memory factory_save_p remains authoritative for the duration of
|
||||
* the session; on the next probe the firmware-class file is read back
|
||||
* read-only via request_firmware(). If cross-reboot persistence of runtime
|
||||
* calibration updates becomes a requirement, the expected route is a
|
||||
* userspace-facing dump interface (debugfs read-only blob, or nl80211
|
||||
* vendor command) that lets userspace read the serialised form and store
|
||||
* it under its own privileges.
|
||||
*/
|
||||
static int bes2600_wifi_cali_table_save(u8 *file_buffer, struct factory_t *factory_save_p)
|
||||
{
|
||||
int ret = 0;
|
||||
int w_size;
|
||||
u32 crc_len = sizeof(factory_data_t);
|
||||
#ifndef STANDARD_FACTORY_EFUSE_FLAG
|
||||
@@ -899,13 +891,11 @@ static int bes2600_wifi_cali_table_save(u8 *file_buffer, struct factory_t *facto
|
||||
|
||||
bes_devel("enter %s\n", __func__);
|
||||
|
||||
if (!file_buffer) {
|
||||
if (!file_buffer)
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
if (!factory_save_p) {
|
||||
if (!factory_save_p)
|
||||
return -ENOENT;
|
||||
}
|
||||
|
||||
/* All initialized to space */
|
||||
memset(file_buffer, 32, FACTORY_MAX_SIZE);
|
||||
@@ -917,22 +907,10 @@ static int bes2600_wifi_cali_table_save(u8 *file_buffer, struct factory_t *facto
|
||||
w_size = factory_build(file_buffer, factory_save_p);
|
||||
|
||||
if (w_size < 0 || w_size > FACTORY_MAX_SIZE) {
|
||||
bes_err("%s: build failed! ret = %d.", __func__, ret);
|
||||
bes_err("%s: build failed! w_size = %d.", __func__, w_size);
|
||||
return -ETXTBSY;
|
||||
}
|
||||
|
||||
#ifdef FACTORY_SAVE_MULTI_PATH
|
||||
/* avoid trailing characters '\0' */
|
||||
file_buffer[w_size] = 32;
|
||||
ret = factory_section_write_file(FACTORY_PATH, file_buffer, FACTORY_MAX_SIZE);
|
||||
#else
|
||||
ret = factory_section_write_file(FACTORY_PATH, file_buffer, w_size);
|
||||
#endif
|
||||
if(ret < 0) {
|
||||
bes_err("%s: write failed! ret = %d.", __func__, ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
@@ -196,6 +196,9 @@ enum factory_cali_status {
|
||||
/* just calibrate 11n, other protocols are automatically mapped */
|
||||
#define WIFI_RF_11N_MODE 0x15
|
||||
|
||||
/* set the struct device * used for request_firmware() context */
|
||||
void bes2600_factory_set_dev(struct device *dev);
|
||||
|
||||
/* read wifi & bt factory cali value*/
|
||||
u8* bes2600_get_factory_cali_data(u8 *file_buffer, u32 *data_len, char *path);
|
||||
void factory_little_endian_cvrt(u8 *data);
|
||||
|
||||
@@ -33,6 +33,7 @@
|
||||
#include "bh.h"
|
||||
#include "sbus.h"
|
||||
#include "bes2600_plat.h"
|
||||
#include "bes2600_factory.h"
|
||||
#include "hwio.h"
|
||||
#include "bes_chardev.h"
|
||||
#include "bes_log.h"
|
||||
@@ -1937,6 +1938,9 @@ static int bes2600_sdio_probe(struct sdio_func *func,
|
||||
if (ret)
|
||||
goto err;
|
||||
|
||||
/* wire struct device into factory.c for request_firmware() context */
|
||||
bes2600_factory_set_dev(dev);
|
||||
|
||||
self->pdata = bes2600_get_platform_data();
|
||||
self->func = func;
|
||||
self->dev = &func->dev;
|
||||
|
||||
+14
-690
@@ -40,12 +40,6 @@ enum bus_probe_state {
|
||||
};
|
||||
|
||||
struct bes_cdev {
|
||||
struct cdev cdev;
|
||||
dev_t dev_id;
|
||||
int major;
|
||||
int minor;
|
||||
struct class *class;
|
||||
struct device *device;
|
||||
atomic_t num_proc;
|
||||
wait_queue_head_t open_wq;
|
||||
spinlock_t status_lock;
|
||||
@@ -66,9 +60,6 @@ struct bes_cdev {
|
||||
struct delayed_work probe_timeout_work;
|
||||
enum bus_probe_state bus_probe;
|
||||
struct work_struct wifi_force_close_work;
|
||||
#ifdef BES2600_WRITE_DPD_TO_FILE
|
||||
int no_dpd;
|
||||
#endif
|
||||
enum pend_read_op read_flag;
|
||||
enum wakeup_event wakeup_by_event; /* used to filter unwanted event wakeup reason report */
|
||||
u16 wakeup_state; /* for userspace check wakeup reason */
|
||||
@@ -88,9 +79,6 @@ struct bes2600_op_map {
|
||||
|
||||
static struct bes_cdev bes2600_cdev;
|
||||
module_param_named(fw_type, bes2600_cdev.fw_type, int, 0644);
|
||||
#ifdef BES2600_WRITE_DPD_TO_FILE
|
||||
module_param_named(no_dpd, bes2600_cdev.no_dpd, int, 0644);
|
||||
#endif
|
||||
|
||||
extern int bes2600_register_net_dev(struct sbus_priv *bus_priv);
|
||||
extern int bes2600_unregister_net_dev(struct sbus_priv *bus_priv);
|
||||
@@ -226,11 +214,11 @@ static int bes2600_switch_bt(bool on)
|
||||
/* check if there is a error when bootup */
|
||||
ret = (status <= 0 || bes2600_chrdev_is_bus_error()) ? -1 : 0;
|
||||
} else {
|
||||
bes_devel("bes2600 activate bt.\n");
|
||||
bes_info("enable BT\n");
|
||||
ret = bes2600_chrdev_switch_subsys(GPIO_WAKE_FLAG_BT_ON, SUBSYSTEM_BT, true);
|
||||
}
|
||||
} else {
|
||||
bes_devel("bes2600 deactivate bt.\n");
|
||||
bes_info("disable BT\n");
|
||||
bes2600_chrdev_switch_subsys(GPIO_WAKE_FLAG_BT_OFF, SUBSYSTEM_BT, false);
|
||||
}
|
||||
|
||||
@@ -247,391 +235,47 @@ static int bes2600_switch_bt(bool on)
|
||||
}
|
||||
|
||||
/*
|
||||
* This is a global function so we don't have to make many changes to
|
||||
* the driver.
|
||||
* Re-added for danctnix's bes2600_btuart.c (a danctnix-only file) which
|
||||
* relies on the chardev utility API for BT power switching and bus-error
|
||||
* checks. The userspace /dev/bes2600 chardev itself is removed by the
|
||||
* remove-chardev-user-interface series; these in-kernel helpers stay.
|
||||
*
|
||||
* @wifi: 1 to turn on, 0 to turn off. Otherwise, leave unchanged
|
||||
* @bt: 1 to turn on, 0 to turn off. Otherwise, leave unchanged
|
||||
* @bt: 1 to turn on, 0 to turn off. Otherwise, leave unchanged
|
||||
*/
|
||||
int bes2600_chrdev_switch_subsys_glb(int wifi, int bt)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
switch (wifi) {
|
||||
case 0:
|
||||
ret = bes2600_switch_wifi(false);
|
||||
break;
|
||||
case 1:
|
||||
ret = bes2600_switch_wifi(true);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
case 0: ret = bes2600_switch_wifi(false); break;
|
||||
case 1: ret = bes2600_switch_wifi(true); break;
|
||||
default: break;
|
||||
}
|
||||
|
||||
if (ret)
|
||||
goto result;
|
||||
return ret;
|
||||
|
||||
switch (bt) {
|
||||
case 0:
|
||||
ret = bes2600_switch_bt(false);
|
||||
break;
|
||||
case 1:
|
||||
ret = bes2600_switch_bt(true);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
case 0: ret = bes2600_switch_bt(false); break;
|
||||
case 1: ret = bes2600_switch_bt(true); break;
|
||||
default: break;
|
||||
}
|
||||
|
||||
result:
|
||||
return ret;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(bes2600_chrdev_switch_subsys_glb);
|
||||
|
||||
static int bes2600_get_cmd_and_ifname(const char *str, char **result)
|
||||
{
|
||||
int cmd_len = 0;
|
||||
int ifname_len = 0;
|
||||
char *sp = NULL;
|
||||
char *tmp_ptr = NULL;
|
||||
char *cmd_ptr = NULL;
|
||||
|
||||
/* check if input arguments is valid */
|
||||
if (!str || strncmp(str, "ifname:", 7) != 0)
|
||||
return -1;
|
||||
|
||||
sp = strchr(str, ' ');
|
||||
if (strncmp(sp + 1, "cmd:", 4) != 0)
|
||||
return -1;
|
||||
|
||||
/* extract interface name */
|
||||
ifname_len = sp - str - 7;
|
||||
tmp_ptr = kmalloc(ifname_len + 1, GFP_KERNEL);
|
||||
if (!tmp_ptr) {
|
||||
return -2;
|
||||
}
|
||||
|
||||
strncpy(tmp_ptr, str+7, ifname_len);
|
||||
tmp_ptr[ifname_len] = '\0';
|
||||
result[0] = tmp_ptr;
|
||||
|
||||
/* get command length */
|
||||
cmd_ptr = strstr(str, "cmd:");
|
||||
cmd_ptr += 4;
|
||||
sp = strchr(cmd_ptr, ' ');
|
||||
if (!sp) { /* the command don't have any parameter */
|
||||
cmd_len = strlen(cmd_ptr);
|
||||
if (cmd_ptr[cmd_len - 1] == '\n')
|
||||
--cmd_len;
|
||||
} else { /* the command have one or more parameter */
|
||||
cmd_len = sp - cmd_ptr;
|
||||
}
|
||||
|
||||
/* copy command to out buffer */
|
||||
tmp_ptr = kmalloc( cmd_len + 1, GFP_KERNEL);
|
||||
if (!tmp_ptr) {
|
||||
kfree(result[0]);
|
||||
result[0] = NULL;
|
||||
return -3;
|
||||
}
|
||||
|
||||
strncpy(tmp_ptr, cmd_ptr, cmd_len);
|
||||
tmp_ptr[cmd_len] = '\0';
|
||||
result[1] = tmp_ptr;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void bes2600_recyle_cmd_and_ifname_mem(char **info)
|
||||
{
|
||||
if (info[0]) {
|
||||
kfree(info[0]);
|
||||
info[0] = NULL;
|
||||
}
|
||||
|
||||
if (info[1]) {
|
||||
kfree(info[1]);
|
||||
info[1] = NULL;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
static int bes2600_op_default_handler(const char *str)
|
||||
{
|
||||
char *info[2] = {0};
|
||||
|
||||
if (bes2600_get_cmd_and_ifname(str, info) == 0) {
|
||||
bes_devel("cmd(%s) on %s not handled\n", info[1], info[0]);
|
||||
} else {
|
||||
bes_err("%s get command fail, the origin string is %s\n", __func__, str);
|
||||
}
|
||||
|
||||
bes2600_recyle_cmd_and_ifname_mem(info);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bes2600_op_wifi_bt_on_off(const char *str)
|
||||
{
|
||||
char *info[2] = {0};
|
||||
int ret = 0;
|
||||
enum wait_state wait_state;
|
||||
enum bus_probe_state probe_state;
|
||||
unsigned long status = 0;
|
||||
|
||||
spin_lock(&bes2600_cdev.status_lock);
|
||||
probe_state = bes2600_cdev.bus_probe;
|
||||
wait_state = bes2600_cdev.wait_state;
|
||||
spin_unlock(&bes2600_cdev.status_lock);
|
||||
|
||||
/* only work for wifi signal mode */
|
||||
if (bes2600_cdev.fw_type != BES2600_FW_TYPE_WIFI_SIGNAL)
|
||||
return -EFAULT;
|
||||
|
||||
/* wait bus probe operation end */
|
||||
if (probe_state == BES2600_BUS_PROBE_START) {
|
||||
bes_devel("wait bus probe operation end\n");
|
||||
status = wait_event_timeout(bes2600_cdev.probe_done_wq,
|
||||
(bes2600_cdev.bus_probe > BES2600_BUS_PROBE_START),
|
||||
HZ);
|
||||
WARN_ON(status <= 0);
|
||||
}
|
||||
|
||||
/* must wait previous operation end in critical section */
|
||||
if (wait_state != BES2600_BOOT_WAIT_NONE) {
|
||||
bes_devel("wait previous operation end\n");
|
||||
status = wait_event_timeout(bes2600_cdev.probe_done_wq,
|
||||
(bes2600_cdev.wait_state == BES2600_BOOT_WAIT_NONE),
|
||||
HZ * 8);
|
||||
WARN_ON(status <= 0);
|
||||
}
|
||||
|
||||
/* if dpd calibration is doing, modify wifi and bt state directly */
|
||||
spin_lock(&bes2600_cdev.status_lock);
|
||||
if (bes2600_cdev.bus_probe == BES2600_BUS_PROBE_OK && !bes2600_cdev.dpd_calied) {
|
||||
if (bes2600_get_cmd_and_ifname(str, info) == 0) {
|
||||
if (strncmp(info[1], "WIFI_ON", 7) == 0) {
|
||||
bes2600_cdev.wifi_opened = true;
|
||||
} else if (strncmp(info[1], "WIFI_OFF", 8) == 0) {
|
||||
bes2600_cdev.wifi_opened = false;
|
||||
} else if (strncmp(info[1], "BT_ON", 5) == 0) {
|
||||
bes2600_cdev.bt_opened = true;
|
||||
bes2600_cdev.bton_pending = true;
|
||||
} else if (strncmp(info[1], "BT_OFF", 6) == 0) {
|
||||
bes2600_cdev.bt_opened = false;
|
||||
bes2600_cdev.bton_pending = false;
|
||||
}
|
||||
}
|
||||
bes2600_recyle_cmd_and_ifname_mem(info);
|
||||
spin_unlock(&bes2600_cdev.status_lock);
|
||||
|
||||
/* wait probe done event */
|
||||
status = wait_event_timeout(bes2600_cdev.probe_done_wq,
|
||||
bes2600_bootup_end(), HZ * 8);
|
||||
WARN_ON(status <= 0);
|
||||
|
||||
return (status <= 0 || bes2600_chrdev_is_bus_error()) ? -EFAULT : 0;
|
||||
}
|
||||
spin_unlock(&bes2600_cdev.status_lock);
|
||||
|
||||
/* process wifi/bt on/off operation */
|
||||
if (bes2600_get_cmd_and_ifname(str, info) == 0) {
|
||||
if (strncmp(info[1], "WIFI_ON", 7) == 0) {
|
||||
ret = bes2600_switch_wifi(1);
|
||||
} else if (strncmp(info[1], "WIFI_OFF", 8) == 0) {
|
||||
ret = bes2600_switch_wifi(0);
|
||||
} else if (strncmp(info[1], "BT_ON", 5) == 0) {
|
||||
ret = bes2600_switch_bt(1);
|
||||
} else if (strncmp(info[1], "BT_OFF", 6) == 0) {
|
||||
ret = bes2600_switch_bt(0);
|
||||
}
|
||||
}
|
||||
|
||||
if (!ret && bes2600_chrdev_check_system_close())
|
||||
ret = bes2600_chrdev_do_system_close(bes2600_cdev.sbus_ops,
|
||||
bes2600_cdev.sbus_priv);
|
||||
|
||||
bes2600_recyle_cmd_and_ifname_mem(info);
|
||||
|
||||
return ret ;
|
||||
}
|
||||
|
||||
|
||||
static int bes2600_op_change_fw_type(const char *str)
|
||||
{
|
||||
int ret = 0;
|
||||
int temp = 0;
|
||||
long status = 0;
|
||||
char *cmd_ptr = NULL;
|
||||
char fw_type[5] = {0};
|
||||
bool sys_closed = bes2600_chrdev_check_system_close();
|
||||
|
||||
bes_devel("%s is called, arg:%s\n", __func__, str);
|
||||
|
||||
if (!bes2600_cdev.sbus_ops->power_switch && !bes2600_cdev.sbus_ops->reboot)
|
||||
return -EPERM;
|
||||
|
||||
/* check if user input is valid */
|
||||
cmd_ptr = strstr(str, "CHANGE_FW_TYPE ");
|
||||
if (strlen(str) < 16 || !cmd_ptr) {
|
||||
bes_err("the format of \"%s\" is error\n", str);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* convert fw_type from string to int */
|
||||
strncpy(fw_type, cmd_ptr + 14, 4);
|
||||
fw_type[0] = '+';
|
||||
ret = kstrtoint(fw_type, 10, &temp);
|
||||
if (ret < 0) {
|
||||
bes_err("%s parse error\n", __func__);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* no need to realod firmware if new fw_type is equal to the old */
|
||||
if (temp == bes2600_cdev.fw_type ) {
|
||||
bes_devel("fw type is equal\n");
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* close wifi net device */
|
||||
if (bes2600_cdev.sbus_priv
|
||||
&& bes2600_is_net_dev_created(bes2600_cdev.sbus_priv)) {
|
||||
bes2600_unregister_net_dev(bes2600_cdev.sbus_priv);
|
||||
}
|
||||
|
||||
/* update firmware type */
|
||||
bes2600_cdev.fw_type = temp;
|
||||
bes2600_chrdev_update_signal_mode();
|
||||
|
||||
if (!sys_closed) {
|
||||
/* close device to call disconnect function */
|
||||
if (bes2600_cdev.sbus_ops->power_switch)
|
||||
bes2600_cdev.sbus_ops->power_switch(bes2600_cdev.sbus_priv, 0);
|
||||
else if (bes2600_cdev.sbus_ops->reboot)
|
||||
bes2600_cdev.sbus_ops->reboot(bes2600_cdev.sbus_priv);
|
||||
}
|
||||
|
||||
if (bes2600_cdev.sbus_ops->reboot)
|
||||
bes2600_chrdev_start_bus_probe();
|
||||
|
||||
/* wait disconnect event */
|
||||
status = wait_event_timeout(bes2600_cdev.probe_done_wq, (bes2600_cdev.sbus_priv == NULL), HZ * 10);
|
||||
WARN_ON(status <= 0);
|
||||
|
||||
if (bes2600_cdev.dpd_calied
|
||||
&& bes2600_chrdev_check_system_close()) {
|
||||
bes_devel("no need to reload firmware\n");
|
||||
return 0;
|
||||
}
|
||||
|
||||
bes_devel("reload firmware...\n");
|
||||
/* power on device to call probe function */
|
||||
if (bes2600_cdev.sbus_ops->power_switch)
|
||||
bes2600_cdev.sbus_ops->power_switch(NULL, 1);
|
||||
|
||||
/* wait probe done event */
|
||||
status = wait_event_timeout(bes2600_cdev.probe_done_wq,
|
||||
bes2600_bootup_end(), HZ * 10);
|
||||
WARN_ON(status <= 0);
|
||||
|
||||
ret = (status <= 0 || bes2600_chrdev_is_bus_error()) ? -1 : 0;
|
||||
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int bes2600_op_bt_wakeup(const char *str)
|
||||
{
|
||||
int ret = 0;
|
||||
unsigned long status = 0;
|
||||
|
||||
spin_lock(&bes2600_cdev.status_lock);
|
||||
if (!bes2600_cdev.bt_opened) {
|
||||
spin_unlock(&bes2600_cdev.status_lock);
|
||||
return -EFAULT;
|
||||
}
|
||||
spin_unlock(&bes2600_cdev.status_lock);
|
||||
|
||||
/* wait probe done event */
|
||||
status = wait_event_timeout(bes2600_cdev.probe_done_wq,
|
||||
bes2600_bootup_end(), HZ * 8);
|
||||
if (status <= 0 || bes2600_chrdev_is_bus_error())
|
||||
return -EFAULT;
|
||||
|
||||
bes_devel("bes2600 wakeup bt.\n");
|
||||
ret = bes2600_chrdev_switch_subsys(GPIO_WAKE_FLAG_BT_LP_ON, SUBSYSTEM_BT_LP, true);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int bes2600_op_bt_sleep(const char *str)
|
||||
{
|
||||
int ret = 0;
|
||||
unsigned long status = 0;
|
||||
|
||||
spin_lock(&bes2600_cdev.status_lock);
|
||||
if (!bes2600_cdev.bt_opened) {
|
||||
spin_unlock(&bes2600_cdev.status_lock);
|
||||
return -EFAULT;
|
||||
}
|
||||
spin_unlock(&bes2600_cdev.status_lock);
|
||||
|
||||
/* wait probe done event */
|
||||
status = wait_event_timeout(bes2600_cdev.probe_done_wq,
|
||||
bes2600_bootup_end(), HZ * 8);
|
||||
if (status <= 0 || bes2600_chrdev_is_bus_error())
|
||||
return -EFAULT;
|
||||
|
||||
bes_devel("bes2600 allow bt sleep.\n");
|
||||
ret = bes2600_chrdev_switch_subsys(GPIO_WAKE_FLAG_BT_LP_OFF, SUBSYSTEM_BT_LP, false);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int bes2600_op_set_wakeup_read_flag(const char *str)
|
||||
{
|
||||
bes_devel("%s is called, arg:%s\n", __func__, str);
|
||||
spin_lock(&bes2600_cdev.status_lock);
|
||||
bes2600_cdev.read_flag = BES_CDEV_READ_WAKEUP_STATE;
|
||||
spin_unlock(&bes2600_cdev.status_lock);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef FW_DOWNLOAD_UART_DAEMON
|
||||
int bes2600_load_uevent(char *env[])
|
||||
{
|
||||
return kobject_uevent_env(&bes2600_cdev.device->kobj, KOBJ_CHANGE, env);
|
||||
}
|
||||
#endif
|
||||
|
||||
static struct bes2600_op_map bes2600_op_map_tab[] ={
|
||||
/*op op_len handler */
|
||||
{"P2P_SET_NOA", 11, bes2600_op_default_handler},
|
||||
{"P2P_SET_PS", 10, bes2600_op_default_handler},
|
||||
{"SET_AP_WPS_P2P_IE", 17, bes2600_op_default_handler},
|
||||
{"LINKSPEED", 9, bes2600_op_default_handler},
|
||||
{"RSSI", 4, bes2600_op_default_handler},
|
||||
{"GETBAND", 7, bes2600_op_default_handler},
|
||||
{"WLS_BATCHING", 12, bes2600_op_default_handler},
|
||||
{"MACADDR", 7, bes2600_op_default_handler},
|
||||
{"RXFILTER-START", 14, bes2600_op_default_handler},
|
||||
{"RXFILTER-STOP", 13, bes2600_op_default_handler},
|
||||
{"RXFILTER-ADD", 12, bes2600_op_default_handler},
|
||||
{"RXFILTER-REMOVE", 15, bes2600_op_default_handler},
|
||||
{"BTCOEXMODE", 10, bes2600_op_default_handler},
|
||||
{"BTCOEXSCAN-START", 16, bes2600_op_default_handler},
|
||||
{"BTCOEXSCAN-STOP", 15, bes2600_op_default_handler},
|
||||
{"SETSUSPENDMODE", 14, bes2600_op_default_handler},
|
||||
{"COUNTRY", 7, bes2600_op_default_handler},
|
||||
{"WIFI_ON", 7, bes2600_op_wifi_bt_on_off},
|
||||
{"WIFI_OFF", 8, bes2600_op_wifi_bt_on_off},
|
||||
{"BT_ON", 5, bes2600_op_wifi_bt_on_off},
|
||||
{"BT_OFF", 6, bes2600_op_wifi_bt_on_off},
|
||||
{"CHANGE_FW_TYPE", 14, bes2600_op_change_fw_type},
|
||||
{"BT_WAKEUP", 9, bes2600_op_bt_wakeup},
|
||||
{"BT_SLEEP", 8, bes2600_op_bt_sleep},
|
||||
{"WAKEUP_STATE", 12, bes2600_op_set_wakeup_read_flag},
|
||||
};
|
||||
|
||||
static int bes2600_chrdev_check_system_close_internal(void)
|
||||
{
|
||||
@@ -641,255 +285,13 @@ static int bes2600_chrdev_check_system_close_internal(void)
|
||||
&& (bes2600_cdev.wifi_opened == false);
|
||||
}
|
||||
|
||||
static int bes2600_chrdev_open(struct inode *inode, struct file *filp)
|
||||
{
|
||||
if (atomic_read(&bes2600_cdev.num_proc) > 0) {
|
||||
wait_event_timeout(bes2600_cdev.open_wq,
|
||||
(atomic_read(&bes2600_cdev.num_proc) == 0),
|
||||
MAX_SCHEDULE_TIMEOUT);
|
||||
}
|
||||
|
||||
bes_devel("bes2600 char device is opened\n");
|
||||
atomic_inc(&bes2600_cdev.num_proc);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static ssize_t bes2600_chrdev_read(struct file *file, char __user *user_buf,
|
||||
size_t count, loff_t *ppos)
|
||||
{
|
||||
char buf[64] = {0};
|
||||
unsigned int len;
|
||||
long status = 0;
|
||||
|
||||
switch (bes2600_cdev.read_flag) {
|
||||
case BES_CDEV_READ_WAKEUP_STATE:
|
||||
if (bes2600_chrdev_wakeup_by_event_get() > WAKEUP_EVENT_NONE) {
|
||||
status = wait_event_timeout(bes2600_cdev.wakeup_reason_wq,
|
||||
bes2600_chrdev_wakeup_by_event_get() == WAKEUP_EVENT_NONE, HZ * 2);
|
||||
WARN_ON(status <= 0);
|
||||
}
|
||||
len = sprintf(buf, "wakeup_reason: %u, src_port: %u\n",
|
||||
bes2600_cdev.wakeup_state, bes2600_cdev.src_port);
|
||||
break;
|
||||
default:
|
||||
len = sprintf(buf, "dpd_calied:%d wifi_opened:%d bt_opened:%d fw_type:%d\n",
|
||||
bes2600_cdev.dpd_calied,
|
||||
bes2600_cdev.wifi_opened,
|
||||
bes2600_cdev.bt_opened,
|
||||
bes2600_cdev.fw_type);
|
||||
break;
|
||||
}
|
||||
|
||||
len = sizeof(buf);
|
||||
/* reset read flag */
|
||||
spin_lock(&bes2600_cdev.status_lock);
|
||||
bes2600_cdev.read_flag = BES_CDEV_READ_NUM_MAX;
|
||||
spin_unlock(&bes2600_cdev.status_lock);
|
||||
|
||||
return simple_read_from_buffer(user_buf, count, ppos, buf, len);
|
||||
}
|
||||
|
||||
static ssize_t bes2600_chrdev_write(struct file *file,
|
||||
const char __user *user_buf, size_t count, loff_t *ppos)
|
||||
{
|
||||
int i = 0;
|
||||
int cmd_num = ARRAY_SIZE(bes2600_op_map_tab);
|
||||
int cmd_len = 0;
|
||||
int ret = 0;
|
||||
char *info[2] = {0};
|
||||
char *buf = NULL;
|
||||
|
||||
/* copy content from user space to kernel */
|
||||
/* message format:"ifname:wlanx cmd:xxx arg1 arg2 ..." */
|
||||
buf = kmalloc(count + 1, GFP_KERNEL);
|
||||
if (copy_from_user(buf, user_buf, count))
|
||||
return -EFAULT;
|
||||
|
||||
/* add terminal character */
|
||||
buf[count] = '\0';
|
||||
|
||||
/* extract comand and interface */
|
||||
if (bes2600_get_cmd_and_ifname(buf, info) != 0) {
|
||||
bes_err("%s get command fail, the origin string is %s\n", __func__, buf);
|
||||
kfree(buf);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* match operation item and execure its handler */
|
||||
cmd_len = strlen(info[1]);
|
||||
for (i = 0; i < cmd_num; i++) {
|
||||
if (cmd_len < bes2600_op_map_tab[i].op_len)
|
||||
continue;
|
||||
|
||||
if (strncasecmp(info[1], bes2600_op_map_tab[i].op, bes2600_op_map_tab[i].op_len) == 0) {
|
||||
ret = bes2600_op_map_tab[i].handler(buf);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
/* operation item mismatch */
|
||||
if (i == cmd_num) {
|
||||
bes_err("cmd(%s) mismatch\n", info[1]);
|
||||
}
|
||||
|
||||
bes2600_recyle_cmd_and_ifname_mem(info);
|
||||
kfree(buf);
|
||||
|
||||
return (ret == 0) ? count : ret;
|
||||
}
|
||||
|
||||
static int bes2600_chrdev_release (struct inode *inode, struct file *file)
|
||||
{
|
||||
if (atomic_dec_and_test(&bes2600_cdev.num_proc)) {
|
||||
wake_up(&bes2600_cdev.open_wq);
|
||||
}
|
||||
|
||||
bes_devel("bes2600 char device is closed\n");
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct file_operations bes2600_chardev_fops =
|
||||
{
|
||||
.owner = THIS_MODULE,
|
||||
.open = bes2600_chrdev_open,
|
||||
.read = bes2600_chrdev_read,
|
||||
.write = bes2600_chrdev_write,
|
||||
.release = bes2600_chrdev_release,
|
||||
};
|
||||
|
||||
#ifdef BES2600_WRITE_DPD_TO_FILE
|
||||
static int bes2600_chrdev_write_dpd_data_to_file(const char *path, void *buffer, int size)
|
||||
{
|
||||
int ret = 0;
|
||||
struct file *fp;
|
||||
|
||||
if (buffer == NULL || size == 0)
|
||||
return 0;
|
||||
|
||||
fp = filp_open(path, O_TRUNC | O_CREAT | O_RDWR, S_IRUSR);
|
||||
if (IS_ERR(fp)) {
|
||||
bes_err("BES2600 : can't open %s\n",path);
|
||||
return -1;
|
||||
}
|
||||
|
||||
ret = kernel_write(fp, buffer, size, &fp->f_pos);
|
||||
if (ret < 0)
|
||||
bes_err("write dpd to file failed\n");
|
||||
|
||||
filp_close(fp,NULL);
|
||||
|
||||
bes_devel("write dpd to %s\n", path);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static bool bes2600_chrdev_dpd_is_vaild(u8 *dpd_data)
|
||||
{
|
||||
u32 cal_crc = 0;
|
||||
u32 dpd_crc = le32_to_cpup((__le32 *)(dpd_data));
|
||||
u32 dpd_ver = le32_to_cpup((__le32 *)(dpd_data + DPD_VERSION_OFFSET));
|
||||
|
||||
/* check version */
|
||||
if (dpd_ver < DPD_CUR_VERSION)
|
||||
return false;
|
||||
|
||||
cal_crc ^= 0xffffffffL;
|
||||
cal_crc = crc32_le(cal_crc, dpd_data + 4, DPD_BIN_SIZE - 4);
|
||||
cal_crc ^= 0xffffffffL;
|
||||
|
||||
/* check if the dpd data is valid */
|
||||
if (cal_crc != dpd_crc) {
|
||||
bes_err(
|
||||
"bes2600 dpd data from file check failed, calc_crc:0x%08x dpd_crc: 0x%08x\n",
|
||||
cal_crc, dpd_crc);
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static int bes2600_chrdev_read_and_check_dpd_data(const char *file, u8 **data, u32 *len)
|
||||
{
|
||||
int ret = 0;
|
||||
u8* read_data = NULL;
|
||||
struct file *fp;
|
||||
|
||||
/* open file */
|
||||
fp = filp_open(file, O_RDONLY, 0);//S_IRUSR
|
||||
if (IS_ERR(fp)) {
|
||||
bes_devel("BES2600 : can't open %s\n",file);
|
||||
return -1;
|
||||
}
|
||||
|
||||
#ifdef BES2600_WRITE_DPD_TO_FILE
|
||||
if (fp->f_inode->i_size != DPD_BIN_FILE_SIZE) {
|
||||
bes_err(
|
||||
"bes2600 dpd data file size check failed, read_size: %lld file_size: %d\n",
|
||||
fp->f_inode->i_size, DPD_BIN_FILE_SIZE);
|
||||
filp_close(fp, NULL);
|
||||
return -1;
|
||||
}
|
||||
#endif
|
||||
|
||||
/* allocate memory for storing reading data */
|
||||
read_data = kmalloc(fp->f_inode->i_size, GFP_KERNEL);
|
||||
if (read_data == NULL) {
|
||||
bes_devel("%s alloc mem fail\n", __func__);
|
||||
goto err1;
|
||||
}
|
||||
|
||||
/* read data from file */
|
||||
ret = kernel_read(fp, read_data, fp->f_inode->i_size, &fp->f_pos);
|
||||
if (ret < DPD_BIN_SIZE) {
|
||||
bes_err("%s read fail, ret=%d\n", __func__, ret);
|
||||
goto err2;
|
||||
}
|
||||
|
||||
/* check dpd version and crc */
|
||||
if (!bes2600_chrdev_dpd_is_vaild(read_data))
|
||||
goto err2;
|
||||
|
||||
/* close file */
|
||||
filp_close(fp, NULL);
|
||||
|
||||
/* copy data to external */
|
||||
*data = read_data;
|
||||
*len = DPD_BIN_SIZE;;
|
||||
|
||||
/* output debug information */
|
||||
bes_devel("read dpd data from %s\n", file);
|
||||
|
||||
return 0;
|
||||
|
||||
err2:
|
||||
kfree(read_data);
|
||||
err1:
|
||||
filp_close(fp, NULL);
|
||||
*data = NULL;
|
||||
*len = 0;
|
||||
return -1;
|
||||
}
|
||||
#endif
|
||||
|
||||
const u8* bes2600_chrdev_get_dpd_data(u32 *len)
|
||||
{
|
||||
#ifdef BES2600_WRITE_DPD_TO_FILE
|
||||
if (!bes2600_cdev.dpd_calied && bes2600_cdev.no_dpd) {
|
||||
/* read dpd data from file that stores factory dpd calibration data */
|
||||
if ((bes2600_chrdev_read_and_check_dpd_data(BES2600_DPD_GOLDEN_PATH,
|
||||
&bes2600_cdev.dpd_data, &bes2600_cdev.dpd_len) < 0) &&
|
||||
(bes2600_chrdev_read_and_check_dpd_data(BES2600_DEFAULT_DPD_PATH,
|
||||
&bes2600_cdev.dpd_data, &bes2600_cdev.dpd_len) < 0)) {
|
||||
bes_err("%s read dpd data fail\n", __func__);
|
||||
return NULL;
|
||||
} else {
|
||||
bes2600_cdev.dpd_calied = true;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
if (!bes2600_cdev.dpd_calied)
|
||||
return NULL;
|
||||
if (len)
|
||||
@@ -950,14 +352,6 @@ int bes2600_chrdev_update_dpd_data(void)
|
||||
}
|
||||
spin_unlock(&bes2600_cdev.status_lock);
|
||||
|
||||
#ifdef BES2600_WRITE_DPD_TO_FILE
|
||||
/* write dpd data to file */
|
||||
memset(bes2600_cdev.dpd_data + DPD_BIN_SIZE, 0, DPD_BIN_FILE_SIZE - DPD_BIN_SIZE);
|
||||
bes2600_chrdev_write_dpd_data_to_file(BES2600_DPD_PATH,
|
||||
bes2600_cdev.dpd_data, DPD_BIN_FILE_SIZE);
|
||||
#endif
|
||||
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -1116,7 +510,6 @@ int bes2600_chrdev_do_bus_reset(const struct sbus_ops *sbus_ops, struct sbus_pri
|
||||
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(bes2600_chrdev_do_bus_reset);
|
||||
|
||||
/*
|
||||
* Trigger bes2600_chrdev_do_bus_reset() against the file-global
|
||||
@@ -1129,7 +522,6 @@ int bes2600_chrdev_trigger_bus_reset(void)
|
||||
return bes2600_chrdev_do_bus_reset(bes2600_cdev.sbus_ops,
|
||||
bes2600_cdev.sbus_priv);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(bes2600_chrdev_trigger_bus_reset);
|
||||
|
||||
bool bes2600_chrdev_is_wifi_opened(void)
|
||||
{
|
||||
@@ -1179,7 +571,6 @@ void bes2600_chrdev_wakeup_bt(void)
|
||||
bes_err("Wakeup BT fail in resume\n");
|
||||
}
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(bes2600_chrdev_wakeup_bt);
|
||||
|
||||
int bes2600_chrdev_get_fw_type(void)
|
||||
{
|
||||
@@ -1220,12 +611,6 @@ void bes2600_chrdev_update_signal_mode(void)
|
||||
|
||||
static void bes2600_chrdev_wifi_force_close_work(struct work_struct *work)
|
||||
{
|
||||
char wifi_state[15];
|
||||
char bt_state[15];
|
||||
char fw_type[15];
|
||||
char *env[] = { wifi_state, bt_state, fw_type, NULL };
|
||||
int ret;
|
||||
|
||||
if (bes2600_chrdev_is_wifi_opened()) {
|
||||
bes_devel("system exeception, force wifi down\n");
|
||||
|
||||
@@ -1255,14 +640,6 @@ static void bes2600_chrdev_wifi_force_close_work(struct work_struct *work)
|
||||
bes2600_chrdev_do_system_close(bes2600_cdev.sbus_ops,
|
||||
bes2600_cdev.sbus_priv);
|
||||
}
|
||||
|
||||
/* notify userspace */
|
||||
snprintf(wifi_state, sizeof(wifi_state), "WIFI_OPENED=%d", bes2600_cdev.wifi_opened);
|
||||
snprintf(bt_state, sizeof(bt_state), "BT_OPENED=%d", bes2600_cdev.bt_opened);
|
||||
snprintf(fw_type, sizeof(fw_type), "FW_TYPE=%d", bes2600_cdev.fw_type);
|
||||
ret = kobject_uevent_env(&bes2600_cdev.device->kobj, KOBJ_CHANGE, env);
|
||||
if (!ret)
|
||||
bes_err("bes2600 notify userspace failed\n");
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1356,46 +733,6 @@ int bes2600_chrdev_wakeup_by_event_get(void)
|
||||
|
||||
int bes2600_chrdev_init(struct sbus_ops *ops)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
/* allocate devide id */
|
||||
ret = alloc_chrdev_region(&bes2600_cdev.dev_id, 0, 1, "bes2600_chrdev");
|
||||
if (ret < 0){
|
||||
bes_err("bes2600 alloc device id fail\n");
|
||||
ret = -EFAULT;
|
||||
goto fail;
|
||||
}
|
||||
|
||||
/* extract major and minor device id */
|
||||
bes2600_cdev.major = MAJOR(bes2600_cdev.dev_id);
|
||||
bes2600_cdev.minor = MINOR(bes2600_cdev.dev_id);
|
||||
|
||||
/* add char device and bind operation function */
|
||||
bes2600_cdev.cdev.owner = THIS_MODULE;
|
||||
cdev_init(&bes2600_cdev.cdev, &bes2600_chardev_fops);
|
||||
ret = cdev_add(&bes2600_cdev.cdev, bes2600_cdev.dev_id, 1);
|
||||
if (ret < 0){
|
||||
bes_err("bes2600 char device add fail\n");
|
||||
ret = -EFAULT;
|
||||
goto fail1;
|
||||
}
|
||||
|
||||
/* create class for creating device node */
|
||||
bes2600_cdev.class = class_create("bes2600_chrdev");
|
||||
if (IS_ERR(bes2600_cdev.class)){
|
||||
bes_err("bes2600 char device add fail\n");
|
||||
ret = -EFAULT;
|
||||
goto fail2;
|
||||
}
|
||||
|
||||
/* get char device pointer */
|
||||
bes2600_cdev.device = device_create(bes2600_cdev.class, NULL, bes2600_cdev.dev_id, NULL, "bes2600");
|
||||
if (IS_ERR(bes2600_cdev.device)){
|
||||
bes_err("bes2600 char device create fail\n");
|
||||
ret = -EFAULT;
|
||||
goto fail3;
|
||||
}
|
||||
|
||||
/* initialise global variable */
|
||||
atomic_set(&bes2600_cdev.num_proc, 0);
|
||||
init_waitqueue_head(&bes2600_cdev.open_wq);
|
||||
@@ -1427,15 +764,6 @@ int bes2600_chrdev_init(struct sbus_ops *ops)
|
||||
bes_devel("%s done\n", __func__);
|
||||
|
||||
return 0;
|
||||
|
||||
fail3:
|
||||
class_destroy(bes2600_cdev.class);
|
||||
fail2:
|
||||
cdev_del(&bes2600_cdev.cdev);
|
||||
fail1:
|
||||
unregister_chrdev_region(bes2600_cdev.dev_id, 1);
|
||||
fail:
|
||||
return ret;
|
||||
}
|
||||
|
||||
void bes2600_chrdev_free(void)
|
||||
@@ -1445,9 +773,5 @@ void bes2600_chrdev_free(void)
|
||||
bes2600_free_dpd_log_buffer();
|
||||
#endif
|
||||
bes2600_chrdev_free_dpd_data();
|
||||
cdev_del(&bes2600_cdev.cdev);
|
||||
unregister_chrdev_region(bes2600_cdev.dev_id, 1);
|
||||
device_destroy(bes2600_cdev.class, bes2600_cdev.dev_id);
|
||||
class_destroy(bes2600_cdev.class);
|
||||
bes_devel("%s done\n", __func__);
|
||||
}
|
||||
|
||||
@@ -122,8 +122,6 @@ int bes_host_slave_sync(struct bes2600_common *hw_priv)
|
||||
}
|
||||
*/
|
||||
|
||||
//#define DATA_DUMP_OBSERVE
|
||||
|
||||
static int bes_firmware_download_write_reg(struct platform_fw_t *fw_data, u32 addr, u32 val)
|
||||
{
|
||||
u8 frame_num = 0;
|
||||
@@ -465,14 +463,6 @@ static int bes_firmware_download(struct platform_fw_t *fw_data, const char *fw_n
|
||||
|
||||
const struct firmware *fw_bin;
|
||||
|
||||
#ifdef DATA_DUMP_OBSERVE
|
||||
char *observe;
|
||||
size_t observe_len;
|
||||
loff_t observe_off = 0;
|
||||
mm_segment_t old_fs;
|
||||
struct file *observe_file = NULL;
|
||||
#endif
|
||||
|
||||
struct fw_msg_hdr_t header;
|
||||
struct fw_info_t fw_info;
|
||||
struct download_fw_t download_addr;
|
||||
@@ -580,14 +570,6 @@ retry:
|
||||
}
|
||||
download_addr.addr = fw_info.addr;
|
||||
|
||||
#ifdef DATA_DUMP_OBSERVE
|
||||
observe_file = filp_open("/lib/firmware/bes2002_fw_write.bin", O_CREAT | O_RDWR, 0);
|
||||
if (IS_ERR(observe_file)) {
|
||||
bes_err("create data_dump file err:%ld\n", IS_ERR(observe_file));
|
||||
observe_file = NULL;
|
||||
}
|
||||
#endif
|
||||
|
||||
while (code_length) {
|
||||
|
||||
#if 1
|
||||
@@ -637,17 +619,6 @@ retry:
|
||||
//mdelay(5000);
|
||||
bes_devel("tx_download_firmware_data:%x %d\n", download_addr.addr, length);
|
||||
|
||||
#ifdef DATA_DUMP_OBSERVE
|
||||
if (observe_file) {
|
||||
observe = (char *)(long_buf + sizeof(struct fw_msg_hdr_t) + sizeof(struct download_fw_t));
|
||||
observe_len = length - sizeof(struct fw_msg_hdr_t) - sizeof(struct download_fw_t);
|
||||
old_fs = get_fs();
|
||||
set_fs(KERNEL_DS);
|
||||
vfs_write(observe_file, observe, observe_len, &observe_off);
|
||||
set_fs(old_fs);
|
||||
}
|
||||
#endif
|
||||
|
||||
ret = bes2600_data_write(long_buf, length > 512 ? length : 512);
|
||||
if (ret) {
|
||||
bes_err("tx download fw data err:%d\n", ret);
|
||||
@@ -829,11 +800,6 @@ retry:
|
||||
|
||||
err2:
|
||||
kfree(long_buf);
|
||||
#ifdef DATA_DUMP_OBSERVE
|
||||
if (observe_file) {
|
||||
filp_close(observe_file, NULL);
|
||||
}
|
||||
#endif
|
||||
err1:
|
||||
kfree(short_buf);
|
||||
release_firmware(fw_bin);
|
||||
|
||||
@@ -15,3 +15,26 @@ extern struct device *global_dev;
|
||||
#define bes_info(fmt, ...) dev_info(global_dev, fmt, ##__VA_ARGS__)
|
||||
#define bes_warn(fmt, ...) dev_warn(global_dev, fmt, ##__VA_ARGS__)
|
||||
#define bes_err(fmt, ...) dev_err(global_dev, fmt, ##__VA_ARGS__)
|
||||
|
||||
/*
|
||||
* Legacy debug-subsystem-tagged log macros. The per-subsystem filtering
|
||||
* was never implemented in-tree; these shims let code paths gated by
|
||||
* CONFIG_BES2600_TESTMODE / CONFIG_BES2600_ITP / BES2600_DETECTION_LOGIC
|
||||
* build when their conditions are enabled. The first argument is
|
||||
* currently unused; pick one of the BES2600_DBG_* constants below for
|
||||
* documentation.
|
||||
*/
|
||||
#define BES2600_DBG_SBUS 0
|
||||
#define BES2600_DBG_DOWNLOAD 0
|
||||
#define BES2600_DBG_ITP 0
|
||||
#define BES2600_DBG_TEST_MODE 0
|
||||
|
||||
#define bes2600_info(_dbg, fmt, ...) bes_info(fmt, ##__VA_ARGS__)
|
||||
#define bes2600_err(_dbg, fmt, ...) bes_err(fmt, ##__VA_ARGS__)
|
||||
#define bes2600_warn(_dbg, fmt, ...) bes_warn(fmt, ##__VA_ARGS__)
|
||||
#define bes2600_dbg(_dbg, fmt, ...) bes_devel(fmt, ##__VA_ARGS__)
|
||||
#define bes2600_err_with_cond(_cond, _dbg, fmt, ...) \
|
||||
do { \
|
||||
if (_cond) \
|
||||
bes_err(fmt, ##__VA_ARGS__); \
|
||||
} while (0)
|
||||
|
||||
@@ -586,6 +586,8 @@ static int bes2600_pwr_enter_lp_mode(struct bes2600_common *hw_priv)
|
||||
if (ret) {
|
||||
atomic_set(&hw_priv->bes_power.pm_set_in_process, 0);
|
||||
bes_err("%s, set operation mode fail\n", __func__);
|
||||
timeouts++;
|
||||
continue;
|
||||
}
|
||||
|
||||
/* wait power save mode changed indication */
|
||||
|
||||
+558
-20
@@ -317,6 +317,83 @@ int wsm_release_buffer_to_fw(struct bes2600_vif *priv, int count)
|
||||
}
|
||||
#endif
|
||||
|
||||
#if 0
|
||||
static struct sk_buff *bes2600_get_skb(struct bes2600_common *hw_priv, size_t len)
|
||||
{
|
||||
struct sk_buff *skb;
|
||||
size_t alloc_len = (len > SDIO_BLOCK_SIZE) ? len : SDIO_BLOCK_SIZE;
|
||||
|
||||
if (len > SDIO_BLOCK_SIZE || !hw_priv->skb_cache) {
|
||||
skb = dev_alloc_skb(alloc_len
|
||||
+ WSM_TX_EXTRA_HEADROOM
|
||||
+ 8 /* TKIP IV */
|
||||
+ 12 /* TKIP ICV + MIC */
|
||||
- 2 /* Piggyback */);
|
||||
/* In AP mode RXed SKB can be looped back as a broadcast.
|
||||
* Here we reserve enough space for headers. */
|
||||
skb_reserve(skb, WSM_TX_EXTRA_HEADROOM
|
||||
+ 8 /* TKIP IV */
|
||||
- WSM_RX_EXTRA_HEADROOM);
|
||||
} else {
|
||||
skb = hw_priv->skb_cache;
|
||||
hw_priv->skb_cache = NULL;
|
||||
}
|
||||
return skb;
|
||||
}
|
||||
|
||||
static void bes2600_put_skb(struct bes2600_common *hw_priv, struct sk_buff *skb)
|
||||
{
|
||||
if (hw_priv->skb_cache)
|
||||
dev_kfree_skb(skb);
|
||||
else
|
||||
hw_priv->skb_cache = skb;
|
||||
}
|
||||
|
||||
static int bes2600_bh_read_ctrl_reg(struct bes2600_common *hw_priv,
|
||||
u16 *ctrl_reg)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = bes2600_reg_read_16(hw_priv,
|
||||
ST90TDS_CONTROL_REG_ID, ctrl_reg);
|
||||
if (ret) {
|
||||
ret = bes2600_reg_read_16(hw_priv,
|
||||
ST90TDS_CONTROL_REG_ID, ctrl_reg);
|
||||
if (ret)
|
||||
bes_err("[BH] Failed to read control register.\n");
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int bes2600_device_wakeup(struct bes2600_common *hw_priv)
|
||||
{
|
||||
u16 ctrl_reg;
|
||||
int ret;
|
||||
|
||||
bes_devel("[BH] Device wakeup.\n");
|
||||
|
||||
/* To force the device to be always-on, the host sets WLAN_UP to 1 */
|
||||
ret = bes2600_reg_write_16(hw_priv, ST90TDS_CONTROL_REG_ID,
|
||||
ST90TDS_CONT_WUP_BIT);
|
||||
if (WARN_ON(ret))
|
||||
return ret;
|
||||
|
||||
ret = bes2600_bh_read_ctrl_reg(hw_priv, &ctrl_reg);
|
||||
if (WARN_ON(ret))
|
||||
return ret;
|
||||
|
||||
/* If the device returns WLAN_RDY as 1, the device is active and will
|
||||
* remain active. */
|
||||
if (ctrl_reg & ST90TDS_CONT_RDY_BIT) {
|
||||
bes_devel("[BH] Device awake.\n");
|
||||
return 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
/* Must be called from BH thraed. */
|
||||
void bes2600_enable_powersave(struct bes2600_vif *priv,
|
||||
@@ -326,6 +403,475 @@ void bes2600_enable_powersave(struct bes2600_vif *priv,
|
||||
priv->powersave_enabled = enable;
|
||||
}
|
||||
|
||||
#if 0
|
||||
#define INTERRUPT_WORKAROUND
|
||||
static int bes2600_bh(void *arg)
|
||||
{
|
||||
struct bes2600_common *hw_priv = arg;
|
||||
struct bes2600_vif *priv = NULL;
|
||||
struct sk_buff *skb_rx = NULL;
|
||||
size_t read_len = 0;
|
||||
int rx, tx, term, suspend;
|
||||
struct wsm_hdr *wsm;
|
||||
size_t wsm_len;
|
||||
int wsm_id;
|
||||
u8 wsm_seq;
|
||||
int rx_resync = 1;
|
||||
u16 ctrl_reg = 0;
|
||||
int tx_allowed;
|
||||
int pending_tx = 0;
|
||||
int tx_burst;
|
||||
int rx_burst = 0;
|
||||
long status;
|
||||
#if defined(CONFIG_BES2600_WSM_DUMPS)
|
||||
size_t wsm_dump_max = -1;
|
||||
#endif
|
||||
u32 dummy;
|
||||
bool powersave_enabled;
|
||||
int i;
|
||||
int vif_selected;
|
||||
|
||||
for (;;) {
|
||||
powersave_enabled = 1;
|
||||
spin_lock(&hw_priv->vif_list_lock);
|
||||
bes2600_for_each_vif(hw_priv, priv, i) {
|
||||
#ifdef P2P_MULTIVIF
|
||||
if ((i = (CW12XX_MAX_VIFS - 1)) || !priv)
|
||||
#else
|
||||
if (!priv)
|
||||
#endif
|
||||
continue;
|
||||
powersave_enabled &= !!priv->powersave_enabled;
|
||||
}
|
||||
spin_unlock(&hw_priv->vif_list_lock);
|
||||
if (!hw_priv->hw_bufs_used
|
||||
&& powersave_enabled
|
||||
&& !hw_priv->device_can_sleep
|
||||
&& !atomic_read(&hw_priv->recent_scan)) {
|
||||
status = HZ/8;
|
||||
bes_devel("[BH] No Device wakedown.\n");
|
||||
#ifndef FPGA_SETUP
|
||||
WARN_ON(bes2600_reg_write_16(hw_priv,
|
||||
ST90TDS_CONTROL_REG_ID, 0));
|
||||
hw_priv->device_can_sleep = true;
|
||||
#endif
|
||||
} else if (hw_priv->hw_bufs_used)
|
||||
/* Interrupt loss detection */
|
||||
status = HZ/8;
|
||||
else
|
||||
status = HZ/8;
|
||||
|
||||
/* Dummy Read for SDIO retry mechanism*/
|
||||
if (((atomic_read(&hw_priv->bh_rx) == 0) &&
|
||||
(atomic_read(&hw_priv->bh_tx) == 0)))
|
||||
bes2600_reg_read(hw_priv, ST90TDS_CONFIG_REG_ID,
|
||||
&dummy, sizeof(dummy));
|
||||
#if defined(CONFIG_BES2600_WSM_DUMPS_SHORT)
|
||||
wsm_dump_max = hw_priv->wsm_dump_max_size;
|
||||
#endif /* CONFIG_BES2600_WSM_DUMPS_SHORT */
|
||||
|
||||
#ifdef INTERRUPT_WORKAROUND
|
||||
/* If a packet has already been txed to the device then read the
|
||||
control register for a probable interrupt miss before going
|
||||
further to wait for interrupt; if the read length is non-zero
|
||||
then it means there is some data to be received */
|
||||
if (hw_priv->hw_bufs_used) {
|
||||
bes2600_bh_read_ctrl_reg(hw_priv, &ctrl_reg);
|
||||
if(ctrl_reg & ST90TDS_CONT_NEXT_LEN_MASK)
|
||||
{
|
||||
rx = 1;
|
||||
goto test;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
status = wait_event_interruptible_timeout(hw_priv->bh_wq, ({
|
||||
rx = atomic_xchg(&hw_priv->bh_rx, 0);
|
||||
tx = atomic_xchg(&hw_priv->bh_tx, 0);
|
||||
term = atomic_xchg(&hw_priv->bh_term, 0);
|
||||
suspend = pending_tx ?
|
||||
0 : atomic_read(&hw_priv->bh_suspend);
|
||||
(rx || tx || term || suspend || hw_priv->bh_error);
|
||||
}), status);
|
||||
|
||||
if (status < 0 || term || hw_priv->bh_error)
|
||||
break;
|
||||
|
||||
#ifdef INTERRUPT_WORKAROUND
|
||||
if (!status) {
|
||||
bes2600_bh_read_ctrl_reg(hw_priv, &ctrl_reg);
|
||||
if(ctrl_reg & ST90TDS_CONT_NEXT_LEN_MASK)
|
||||
{
|
||||
bes_err("MISS 1\n");
|
||||
rx = 1;
|
||||
goto test;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
if (!status && hw_priv->hw_bufs_used) {
|
||||
unsigned long timestamp = jiffies;
|
||||
long timeout;
|
||||
bool pending = false;
|
||||
int i;
|
||||
|
||||
wiphy_warn(hw_priv->hw->wiphy, "Missed interrupt?\n");
|
||||
rx = 1;
|
||||
|
||||
/* Get a timestamp of "oldest" frame */
|
||||
for (i = 0; i < 4; ++i)
|
||||
pending |= bes2600_queue_get_xmit_timestamp(
|
||||
&hw_priv->tx_queue[i],
|
||||
×tamp, -1,
|
||||
hw_priv->pending_frame_id);
|
||||
|
||||
/* Check if frame transmission is timed out.
|
||||
* Add an extra second with respect to possible
|
||||
* interrupt loss. */
|
||||
timeout = timestamp +
|
||||
WSM_CMD_LAST_CHANCE_TIMEOUT +
|
||||
1 * HZ -
|
||||
jiffies;
|
||||
|
||||
/* And terminate BH tread if the frame is "stuck" */
|
||||
if (pending && timeout < 0) {
|
||||
//wiphy_warn(priv->hw->wiphy,
|
||||
// "Timeout waiting for TX confirm.\n");
|
||||
bes_devel("bes2600_bh: Timeout waiting for TX confirm.\n");
|
||||
break;
|
||||
}
|
||||
|
||||
#if defined(CONFIG_BES2600_DUMP_ON_ERROR)
|
||||
BUG_ON(1);
|
||||
#endif /* CONFIG_BES2600_DUMP_ON_ERROR */
|
||||
} else if (!status) {
|
||||
if (!hw_priv->device_can_sleep
|
||||
&& !atomic_read(&hw_priv->recent_scan)) {
|
||||
bes_devel("[BH] Device wakedown. Timeout.\n");
|
||||
#ifndef FPGA_SETUP
|
||||
WARN_ON(bes2600_reg_write_16(hw_priv,
|
||||
ST90TDS_CONTROL_REG_ID, 0));
|
||||
hw_priv->device_can_sleep = true;
|
||||
#endif
|
||||
}
|
||||
continue;
|
||||
} else if (suspend) {
|
||||
bes_devel("[BH] Device suspend.\n");
|
||||
powersave_enabled = 1;
|
||||
spin_lock(&hw_priv->vif_list_lock);
|
||||
bes2600_for_each_vif(hw_priv, priv, i) {
|
||||
#ifdef P2P_MULTIVIF
|
||||
if ((i = (CW12XX_MAX_VIFS - 1)) || !priv)
|
||||
#else
|
||||
if (!priv)
|
||||
#endif
|
||||
continue;
|
||||
powersave_enabled &= !!priv->powersave_enabled;
|
||||
}
|
||||
spin_unlock(&hw_priv->vif_list_lock);
|
||||
if (powersave_enabled) {
|
||||
bes_devel("[BH] No Device wakedown. Suspend.\n");
|
||||
#ifndef FPGA_SETUP
|
||||
WARN_ON(bes2600_reg_write_16(hw_priv,
|
||||
ST90TDS_CONTROL_REG_ID, 0));
|
||||
hw_priv->device_can_sleep = true;
|
||||
#endif
|
||||
}
|
||||
|
||||
atomic_set(&hw_priv->bh_suspend, BES2600_BH_SUSPENDED);
|
||||
wake_up(&hw_priv->bh_evt_wq);
|
||||
status = wait_event_interruptible(hw_priv->bh_wq,
|
||||
BES2600_BH_RESUME == atomic_read(
|
||||
&hw_priv->bh_suspend));
|
||||
if (status < 0) {
|
||||
wiphy_err(hw_priv->hw->wiphy,
|
||||
"%s: Failed to wait for resume: %ld.\n",
|
||||
__func__, status);
|
||||
break;
|
||||
}
|
||||
bes_devel("[BH] Device resume.\n");
|
||||
atomic_set(&hw_priv->bh_suspend, BES2600_BH_RESUMED);
|
||||
wake_up(&hw_priv->bh_evt_wq);
|
||||
atomic_inc(&hw_priv->bh_rx);
|
||||
continue;
|
||||
}
|
||||
|
||||
test:
|
||||
tx += pending_tx;
|
||||
pending_tx = 0;
|
||||
|
||||
if (rx) {
|
||||
size_t alloc_len;
|
||||
u8 *data;
|
||||
|
||||
#ifdef INTERRUPT_WORKAROUND
|
||||
if(!(ctrl_reg & ST90TDS_CONT_NEXT_LEN_MASK))
|
||||
#endif
|
||||
if (WARN_ON(bes2600_bh_read_ctrl_reg(
|
||||
hw_priv, &ctrl_reg)))
|
||||
break;
|
||||
rx:
|
||||
read_len = (ctrl_reg & ST90TDS_CONT_NEXT_LEN_MASK) * 2;
|
||||
if (!read_len) {
|
||||
rx_burst = 0;
|
||||
goto tx;
|
||||
}
|
||||
|
||||
if (WARN_ON((read_len < sizeof(struct wsm_hdr)) ||
|
||||
(read_len > EFFECTIVE_BUF_SIZE))) {
|
||||
bes_devel("Invalid read len: %d", read_len);
|
||||
break;
|
||||
}
|
||||
|
||||
/* Add SIZE of PIGGYBACK reg (CONTROL Reg)
|
||||
* to the NEXT Message length + 2 Bytes for SKB */
|
||||
read_len = read_len + 2;
|
||||
|
||||
#if defined(CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES)
|
||||
alloc_len = hw_priv->sbus_ops->align_size(
|
||||
hw_priv->sbus_priv, read_len);
|
||||
#else /* CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES */
|
||||
/* Platform's SDIO workaround */
|
||||
alloc_len = read_len & ~(SDIO_BLOCK_SIZE - 1);
|
||||
if (read_len & (SDIO_BLOCK_SIZE - 1))
|
||||
alloc_len += SDIO_BLOCK_SIZE;
|
||||
#endif /* CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES */
|
||||
|
||||
/* Check if not exceeding BES2600 capabilities */
|
||||
if (WARN_ON_ONCE(alloc_len > EFFECTIVE_BUF_SIZE))
|
||||
bes_devel("Read aligned len: %d\n", alloc_len);
|
||||
|
||||
skb_rx = bes2600_get_skb(hw_priv, alloc_len);
|
||||
if (WARN_ON(!skb_rx))
|
||||
break;
|
||||
|
||||
skb_trim(skb_rx, 0);
|
||||
skb_put(skb_rx, read_len);
|
||||
data = skb_rx->data;
|
||||
if (WARN_ON(!data))
|
||||
break;
|
||||
|
||||
if (WARN_ON(bes2600_data_read(hw_priv, data, alloc_len)))
|
||||
break;
|
||||
|
||||
/* Piggyback */
|
||||
ctrl_reg = __le16_to_cpu(
|
||||
((__le16 *)data)[alloc_len / 2 - 1]);
|
||||
|
||||
wsm = (struct wsm_hdr *)data;
|
||||
wsm_len = __le32_to_cpu(wsm->len);
|
||||
if (WARN_ON(wsm_len > read_len))
|
||||
break;
|
||||
|
||||
#if defined(CONFIG_BES2600_WSM_DUMPS)
|
||||
if (unlikely(hw_priv->wsm_enable_wsm_dumps)) {
|
||||
u16 msgid, ifid;
|
||||
u16 *p = (u16 *)data;
|
||||
msgid = (*(p + 1)) & 0xC3F;
|
||||
ifid = (*(p + 1)) >> 6;
|
||||
ifid &= 0xF;
|
||||
bes_devel("[DUMP] <<< msgid 0x%.4X ifid %d len %d\n", msgid, ifid, *p);
|
||||
print_hex_dump(KERN_DEBUG, "<-- ", DUMP_PREFIX_NONE, data, min(wsm_len, wsm_dump_max));
|
||||
}
|
||||
#endif /* CONFIG_BES2600_WSM_DUMPS */
|
||||
|
||||
wsm_id = __le32_to_cpu(wsm->id) & 0xFFF;
|
||||
wsm_seq = (__le32_to_cpu(wsm->id) >> 13) & 7;
|
||||
|
||||
skb_trim(skb_rx, wsm_len);
|
||||
|
||||
if (unlikely(wsm_id == 0x0800)) {
|
||||
wsm_handle_exception(hw_priv,
|
||||
&data[sizeof(*wsm)],
|
||||
wsm_len - sizeof(*wsm));
|
||||
break;
|
||||
} else if (unlikely(!rx_resync)) {
|
||||
if (WARN_ON(wsm_seq != hw_priv->wsm_rx_seq)) {
|
||||
#if defined(CONFIG_BES2600_DUMP_ON_ERROR)
|
||||
BUG_ON(1);
|
||||
#endif /* CONFIG_BES2600_DUMP_ON_ERROR */
|
||||
break;
|
||||
}
|
||||
}
|
||||
hw_priv->wsm_rx_seq = (wsm_seq + 1) & 7;
|
||||
rx_resync = 0;
|
||||
|
||||
if (wsm_id & 0x0400) {
|
||||
int rc = wsm_release_tx_buffer(hw_priv, 1);
|
||||
if (WARN_ON(rc < 0))
|
||||
break;
|
||||
else if (rc > 0)
|
||||
tx = 1;
|
||||
}
|
||||
|
||||
/* bes2600_wsm_rx takes care on SKB livetime */
|
||||
if (WARN_ON(wsm_handle_rx(hw_priv, wsm_id, wsm,
|
||||
&skb_rx)))
|
||||
break;
|
||||
|
||||
if (skb_rx) {
|
||||
bes2600_put_skb(hw_priv, skb_rx);
|
||||
skb_rx = NULL;
|
||||
}
|
||||
|
||||
read_len = 0;
|
||||
|
||||
if (rx_burst) {
|
||||
bes2600_debug_rx_burst(hw_priv);
|
||||
--rx_burst;
|
||||
goto rx;
|
||||
}
|
||||
}
|
||||
|
||||
tx:
|
||||
BUG_ON(hw_priv->hw_bufs_used > hw_priv->wsm_caps.numInpChBufs);
|
||||
tx_burst = hw_priv->wsm_caps.numInpChBufs -
|
||||
hw_priv->hw_bufs_used;
|
||||
tx_allowed = tx_burst > 0;
|
||||
if (tx && tx_allowed) {
|
||||
size_t tx_len;
|
||||
u8 *data;
|
||||
int ret;
|
||||
|
||||
if (hw_priv->device_can_sleep) {
|
||||
ret = bes2600_device_wakeup(hw_priv);
|
||||
if (WARN_ON(ret < 0))
|
||||
break;
|
||||
else if (ret)
|
||||
hw_priv->device_can_sleep = false;
|
||||
else {
|
||||
/* Wait for "awake" interrupt */
|
||||
pending_tx = tx;
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
wsm_alloc_tx_buffer(hw_priv);
|
||||
ret = wsm_get_tx(hw_priv, &data, &tx_len, &tx_burst,
|
||||
&vif_selected);
|
||||
if (ret <= 0) {
|
||||
wsm_release_tx_buffer(hw_priv, 1);
|
||||
if (WARN_ON(ret < 0))
|
||||
break;
|
||||
} else {
|
||||
wsm = (struct wsm_hdr *)data;
|
||||
BUG_ON(tx_len < sizeof(*wsm));
|
||||
BUG_ON(__le32_to_cpu(wsm->len) != tx_len);
|
||||
|
||||
#if 0 /* count is not implemented */
|
||||
if (ret > 1)
|
||||
atomic_inc(&hw_priv->bh_tx);
|
||||
#else
|
||||
atomic_inc(&hw_priv->bh_tx);
|
||||
#endif
|
||||
|
||||
#if defined(CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES)
|
||||
if (tx_len <= 8)
|
||||
tx_len = 16;
|
||||
tx_len = hw_priv->sbus_ops->align_size(
|
||||
hw_priv->sbus_priv, tx_len);
|
||||
#else /* CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES */
|
||||
/* HACK!!! Platform limitation.
|
||||
* It is also supported by upper layer:
|
||||
* there is always enough space at the
|
||||
* end of the buffer. */
|
||||
if (tx_len & (SDIO_BLOCK_SIZE - 1)) {
|
||||
tx_len &= ~(SDIO_BLOCK_SIZE - 1);
|
||||
tx_len += SDIO_BLOCK_SIZE;
|
||||
}
|
||||
#endif /* CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES */
|
||||
|
||||
/* Check if not exceeding BES2600
|
||||
capabilities */
|
||||
if (WARN_ON_ONCE(tx_len > EFFECTIVE_BUF_SIZE))
|
||||
bes_devel("Write aligned len: %d\n", tx_len);
|
||||
|
||||
wsm->id &= __cpu_to_le32(
|
||||
~WSM_TX_SEQ(WSM_TX_SEQ_MAX));
|
||||
wsm->id |= cpu_to_le32(WSM_TX_SEQ(
|
||||
hw_priv->wsm_tx_seq));
|
||||
|
||||
if (WARN_ON(bes2600_data_write(hw_priv,
|
||||
data, tx_len))) {
|
||||
wsm_release_tx_buffer(hw_priv, 1);
|
||||
break;
|
||||
}
|
||||
|
||||
if (vif_selected != -1) {
|
||||
hw_priv->hw_bufs_used_vif[
|
||||
vif_selected]++;
|
||||
}
|
||||
|
||||
#if defined(CONFIG_BES2600_WSM_DUMPS)
|
||||
if (unlikely(hw_priv->wsm_enable_wsm_dumps)) {
|
||||
u16 msgid, ifid;
|
||||
u16 *p = (u16 *)data;
|
||||
msgid = (*(p + 1)) & 0x3F;
|
||||
ifid = (*(p + 1)) >> 6;
|
||||
ifid &= 0xF;
|
||||
if (msgid == 0x0006)
|
||||
bes_devel("[DUMP] >>> msgid 0x%.4X ifid %d len %d MIB 0x%.4X\n", msgid, ifid, *p, *(p + 2));
|
||||
else
|
||||
bes_devel("[DUMP] >>> msgid 0x%.4X ifid %d len %d\n", msgid, ifid, *p);
|
||||
print_hex_dump(KERN_DEBUG, "--> ", DUMP_PREFIX_NONE, data, min(__le32_to_cpu(wsm->len), wsm_dump_max));
|
||||
}
|
||||
#endif /* CONFIG_BES2600_WSM_DUMPS */
|
||||
|
||||
wsm_txed(hw_priv, data);
|
||||
hw_priv->wsm_tx_seq = (hw_priv->wsm_tx_seq + 1)
|
||||
& WSM_TX_SEQ_MAX;
|
||||
|
||||
if (tx_burst > 1) {
|
||||
bes2600_debug_tx_burst(hw_priv);
|
||||
++rx_burst;
|
||||
goto tx;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (ctrl_reg & ST90TDS_CONT_NEXT_LEN_MASK)
|
||||
goto rx;
|
||||
}
|
||||
|
||||
if (skb_rx) {
|
||||
bes2600_put_skb(hw_priv, skb_rx);
|
||||
skb_rx = NULL;
|
||||
}
|
||||
|
||||
|
||||
if (!term) {
|
||||
bes_devel("[BH] Fatal error, exitting.\n");
|
||||
#if defined(CONFIG_BES2600_DUMP_ON_ERROR)
|
||||
BUG_ON(1);
|
||||
#endif /* CONFIG_BES2600_DUMP_ON_ERROR */
|
||||
hw_priv->bh_error = 1;
|
||||
#if defined(CONFIG_BES2600_USE_STE_EXTENSIONS)
|
||||
spin_lock(&hw_priv->vif_list_lock);
|
||||
bes2600_for_each_vif(hw_priv, priv, i) {
|
||||
if (!priv)
|
||||
continue;
|
||||
ieee80211_driver_hang_notify(priv->vif, GFP_KERNEL);
|
||||
}
|
||||
spin_unlock(&hw_priv->vif_list_lock);
|
||||
bes2600_pm_stay_awake(&hw_priv->pm_state, 3*HZ);
|
||||
#endif
|
||||
/* TODO: schedule_work(recovery) */
|
||||
#ifndef HAS_PUT_TASK_STRUCT
|
||||
/* The only reason of having this stupid code here is
|
||||
* that __put_task_struct is not exported by kernel. */
|
||||
for (;;) {
|
||||
int status = wait_event_interruptible(hw_priv->bh_wq, ({
|
||||
term = atomic_xchg(&hw_priv->bh_term, 0);
|
||||
(term);
|
||||
}));
|
||||
|
||||
if (status || term)
|
||||
break;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
#else
|
||||
|
||||
extern int bes2600_bh_read_ctrl_reg(struct bes2600_common *priv, u32 *ctrl_reg);
|
||||
|
||||
@@ -1053,15 +1599,7 @@ static int bes2600_bh(void *arg)
|
||||
|
||||
tx = 0;
|
||||
|
||||
/*
|
||||
* Patch H: BUG_ON -> WARN_ON_ONCE in the steady-state
|
||||
* hot path. The original BUG_ON ran every bh-loop
|
||||
* iteration; tripping it on a bookkeeping bug locks
|
||||
* the kernel up during normal operation, which is
|
||||
* the wrong response. WARN_ON_ONCE surfaces the
|
||||
* issue without taking the system down.
|
||||
*/
|
||||
WARN_ON_ONCE(hw_priv->hw_bufs_used > hw_priv->wsm_caps.numInpChBufs);
|
||||
BUG_ON(hw_priv->hw_bufs_used > hw_priv->wsm_caps.numInpChBufs);
|
||||
tx_burst = hw_priv->wsm_caps.numInpChBufs - hw_priv->hw_bufs_used;
|
||||
tx_allowed = tx_burst > 0;
|
||||
|
||||
@@ -1105,19 +1643,18 @@ static int bes2600_bh(void *arg)
|
||||
goto tx;
|
||||
|
||||
done:
|
||||
/*
|
||||
* Patch H: dropped the dead `__bes2600_irq_enable(1)` /
|
||||
* `asm volatile("nop")` placeholder that used to sit here.
|
||||
* `__bes2600_irq_enable()` is a stub that returns 0 on
|
||||
* bes2600 silicon — the IRQ is managed by sdio_claim_irq
|
||||
* and chip-side firmware, not by a driver-side enable bit.
|
||||
* (cw1200 inherited the function from a different chip
|
||||
* shape; bes2600 kept the stub but the call sites are
|
||||
* meaningless.)
|
||||
*/
|
||||
;
|
||||
/* Re-enable device interrupts */
|
||||
//hw_priv->sbus_ops->lock(hw_priv->sbus_priv);
|
||||
//__bes2600_irq_enable(1);
|
||||
//hw_priv->sbus_ops->unlock(hw_priv->sbus_priv);
|
||||
asm volatile ("nop");
|
||||
}
|
||||
|
||||
/* Explicitly disable device interrupts */
|
||||
hw_priv->sbus_ops->lock(hw_priv->sbus_priv);
|
||||
__bes2600_irq_enable(0);
|
||||
hw_priv->sbus_ops->unlock(hw_priv->sbus_priv);
|
||||
|
||||
if (!term) {
|
||||
bes_err("[BH] Fatal error, exiting.\n");
|
||||
sdio_work_debug(hw_priv->sbus_priv);
|
||||
@@ -1126,3 +1663,4 @@ static int bes2600_bh(void *arg)
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
+4
-7
@@ -324,10 +324,7 @@ out:
|
||||
}
|
||||
#endif
|
||||
|
||||
/*
|
||||
* Patch H: __bes2600_irq_enable stub removed. It was a no-op
|
||||
* (always returned 0) inherited from cw1200 where the analogous
|
||||
* function manipulates the chip's IRQ-enable register. bes2600
|
||||
* silicon manages SDIO IRQ via sdio_claim_irq and chip-side
|
||||
* firmware — there is no driver-side enable register to write.
|
||||
*/
|
||||
int __bes2600_irq_enable(int enable)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -805,41 +805,6 @@ void bes2600_core_release(struct bes2600_common *self)
|
||||
return;
|
||||
}
|
||||
|
||||
#if (GET_MAC_ADDR_METHOD == 2) || (GET_MAC_ADDR_METHOD == 3) /* To use macaddr and ps mode of customers */
|
||||
int access_file(char *path, char *buffer, int size, int isRead)
|
||||
{
|
||||
int ret=0;
|
||||
struct file *fp;
|
||||
mm_segment_t old_fs = get_fs();
|
||||
|
||||
if(isRead)
|
||||
fp = filp_open(path,O_RDONLY,S_IRUSR);
|
||||
else
|
||||
fp = filp_open(path,O_CREAT|O_WRONLY,S_IRUSR);
|
||||
|
||||
if (IS_ERR(fp)) {
|
||||
bes_err("BES2600 : can't open %s\n", path);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (isRead) {
|
||||
fp->f_pos = 0;
|
||||
set_fs(KERNEL_DS);
|
||||
ret = vfs_read(fp,buffer,size,&fp->f_pos);
|
||||
set_fs(old_fs);
|
||||
} else {
|
||||
fp->f_pos = 0;
|
||||
set_fs(KERNEL_DS);
|
||||
ret = vfs_write(fp,buffer,size,&fp->f_pos);
|
||||
set_fs(old_fs);
|
||||
}
|
||||
filp_close(fp,NULL);
|
||||
|
||||
bes_info("BES2600 : access_file return code(%d)\n", ret);
|
||||
return ret;
|
||||
}
|
||||
#endif
|
||||
|
||||
int bes2600_wifi_start(struct bes2600_common *hw_priv)
|
||||
{
|
||||
int ret = 0, if_id;
|
||||
|
||||
+3
-3
@@ -827,19 +827,19 @@ int bes2600_queue_get_skb(struct bes2600_queue *queue, u32 packetID,
|
||||
bes2600_queue_parse_id(packetID, &queue_generation, &queue_id,
|
||||
&item_generation, &item_id, &if_id, &link_id);
|
||||
|
||||
spin_lock_bh(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
spin_lock(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
if (!list_empty(&queue->stats->hw_priv->tx_loop.pending_record_list)) {
|
||||
list_for_each_entry_safe(record_item, temp_record_item, &queue->stats->hw_priv->tx_loop.pending_record_list, head) {
|
||||
if (record_item->packetID == packetID) {
|
||||
list_del(&record_item->head);
|
||||
dev_kfree_skb(record_item->skb);
|
||||
kfree(record_item);
|
||||
spin_unlock_bh(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
spin_unlock(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
}
|
||||
spin_unlock_bh(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
spin_unlock(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
|
||||
item = &queue->pool[item_id];
|
||||
|
||||
|
||||
+2
-1
@@ -95,6 +95,7 @@ struct sbus_ops {
|
||||
|
||||
void bes2600_irq_handler(struct bes2600_common *priv);
|
||||
|
||||
/* Patch H: __bes2600_irq_enable removed (was a stub). */
|
||||
/* This MUST be wrapped with hwbus_ops->lock/unlock! */
|
||||
int __bes2600_irq_enable(int enable);
|
||||
|
||||
#endif /* BES2600_SBUS_H */
|
||||
|
||||
+3
-3
@@ -3749,7 +3749,7 @@ static int bes2600_set_power_save(struct ieee80211_hw *hw,
|
||||
*
|
||||
* Returns: 0 on success or non zero value on failure
|
||||
*/
|
||||
int bes2600_start_stop_tsm(struct ieee80211_hw *hw, void *data)
|
||||
static int bes2600_start_stop_tsm(struct ieee80211_hw *hw, void *data)
|
||||
{
|
||||
struct bes_msg_start_stop_tsm *start_stop_tsm =
|
||||
(struct bes_msg_start_stop_tsm *) data;
|
||||
@@ -3779,7 +3779,7 @@ int bes2600_start_stop_tsm(struct ieee80211_hw *hw, void *data)
|
||||
*
|
||||
* Returns: TSM parameters collected
|
||||
*/
|
||||
int bes2600_get_tsm_params(struct ieee80211_hw *hw)
|
||||
static int bes2600_get_tsm_params(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct bes2600_common *hw_priv = hw->priv;
|
||||
struct bes_tsm_stats tsm_stats;
|
||||
@@ -3819,7 +3819,7 @@ int bes2600_get_tsm_params(struct ieee80211_hw *hw)
|
||||
*
|
||||
* Returns: Returns the last measured roam delay
|
||||
*/
|
||||
int bes2600_get_roam_delay(struct ieee80211_hw *hw)
|
||||
static int bes2600_get_roam_delay(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct bes2600_common *hw_priv = hw->priv;
|
||||
u16 roam_delay = hw_priv->tsm_info.roam_delay / 1000;
|
||||
|
||||
+2
-2
@@ -109,9 +109,9 @@ void bes2600_tx_loop_set_enable(struct bes2600_common *hw_priv, bool need_warn)
|
||||
bes2600_queue_iterate_pending_packet(&hw_priv->tx_queue[i],
|
||||
bes2600_tx_loop_item_pending_item);
|
||||
}
|
||||
spin_lock_bh(&hw_priv->tx_loop.pending_record_lock);
|
||||
spin_lock(&hw_priv->tx_loop.pending_record_lock);
|
||||
bes2600_queue_iterate_record_pending_packet(hw_priv, bes2600_tx_loop_item_pending_item);
|
||||
spin_unlock_bh(&hw_priv->tx_loop.pending_record_lock);
|
||||
spin_unlock(&hw_priv->tx_loop.pending_record_lock);
|
||||
|
||||
if (atomic_read(&hw_priv->bh_rx) > 0)
|
||||
wake_up(&hw_priv->bh_wq);
|
||||
|
||||
@@ -2232,7 +2232,5 @@ int wsm_cpu_usage_cmd(struct bes2600_common *hw_priv);
|
||||
|
||||
int wsm_wifi_status_cmd(struct bes2600_common *hw_priv, uint32_t status);
|
||||
|
||||
#if defined(STANDARD_FACTORY_EFUSE_FLAG)
|
||||
int wsm_save_factory_txt_to_mcu(struct bes2600_common *hw_priv, const u8 *data, int if_id, enum bes2600_rf_cmd_type cmd_type);
|
||||
#endif
|
||||
#endif /* BES2600_HWIO_H_INCLUDED */
|
||||
|
||||
Vendored
+1
-2
@@ -18,8 +18,7 @@ License: LGPL-2.1
|
||||
License for more details.
|
||||
.
|
||||
You should have received a copy of the GNU Lesser General Public License
|
||||
along with this library; if not, write to the Free Software Foundation, Inc.,
|
||||
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
along with this library; if not, see <https://www.gnu.org/licenses/>.
|
||||
.
|
||||
On Debian systems, the full text of the GNU Lesser General Public License
|
||||
version 2.1 can be found in the file "/usr/share/common-licenses/LGPL-2.1".
|
||||
|
||||
Reference in New Issue
Block a user