mirror of
https://github.com/edk2-porting/linux-next.git
synced 2025-01-10 22:54:11 +08:00
871fc09f1c
CMD_MAC_CONTROL is currently sent async to the firmware, and is sent from the lbs_setup_firmware() path during device init. This means that device init can complete with commands pending, and the if_sdio driver will sometimes power down the device (after init) with this command still pending. This was causing an occasional spurious command timeout after init, leading to a device reset. Fix this by making CMD_MAC_CONTROL synchronous when called from the lbs_setup_firmware() path. Signed-off-by: Daniel Drake <dsd@laptop.org> Signed-off-by: Dan Williams <dcbw@redhat.com> Signed-off-by: John W. Linville <linville@tuxdriver.com>
141 lines
4.2 KiB
C
141 lines
4.2 KiB
C
/* Copyright (C) 2007, Red Hat, Inc. */
|
|
|
|
#ifndef _LBS_CMD_H_
|
|
#define _LBS_CMD_H_
|
|
|
|
#include <net/cfg80211.h>
|
|
|
|
#include "host.h"
|
|
#include "dev.h"
|
|
|
|
|
|
/* Command & response transfer between host and card */
|
|
|
|
struct cmd_ctrl_node {
|
|
struct list_head list;
|
|
int result;
|
|
/* command response */
|
|
int (*callback)(struct lbs_private *,
|
|
unsigned long,
|
|
struct cmd_header *);
|
|
unsigned long callback_arg;
|
|
/* command data */
|
|
struct cmd_header *cmdbuf;
|
|
/* wait queue */
|
|
u16 cmdwaitqwoken;
|
|
wait_queue_head_t cmdwait_q;
|
|
};
|
|
|
|
|
|
/* lbs_cmd() infers the size of the buffer to copy data back into, from
|
|
the size of the target of the pointer. Since the command to be sent
|
|
may often be smaller, that size is set in cmd->size by the caller.*/
|
|
#define lbs_cmd(priv, cmdnr, cmd, cb, cb_arg) ({ \
|
|
uint16_t __sz = le16_to_cpu((cmd)->hdr.size); \
|
|
(cmd)->hdr.size = cpu_to_le16(sizeof(*(cmd))); \
|
|
__lbs_cmd(priv, cmdnr, &(cmd)->hdr, __sz, cb, cb_arg); \
|
|
})
|
|
|
|
#define lbs_cmd_with_response(priv, cmdnr, cmd) \
|
|
lbs_cmd(priv, cmdnr, cmd, lbs_cmd_copyback, (unsigned long) (cmd))
|
|
|
|
void lbs_cmd_async(struct lbs_private *priv, uint16_t command,
|
|
struct cmd_header *in_cmd, int in_cmd_size);
|
|
|
|
int __lbs_cmd(struct lbs_private *priv, uint16_t command,
|
|
struct cmd_header *in_cmd, int in_cmd_size,
|
|
int (*callback)(struct lbs_private *, unsigned long, struct cmd_header *),
|
|
unsigned long callback_arg);
|
|
|
|
struct cmd_ctrl_node *__lbs_cmd_async(struct lbs_private *priv,
|
|
uint16_t command, struct cmd_header *in_cmd, int in_cmd_size,
|
|
int (*callback)(struct lbs_private *, unsigned long, struct cmd_header *),
|
|
unsigned long callback_arg);
|
|
|
|
int lbs_cmd_copyback(struct lbs_private *priv, unsigned long extra,
|
|
struct cmd_header *resp);
|
|
|
|
int lbs_allocate_cmd_buffer(struct lbs_private *priv);
|
|
int lbs_free_cmd_buffer(struct lbs_private *priv);
|
|
|
|
int lbs_execute_next_command(struct lbs_private *priv);
|
|
void __lbs_complete_command(struct lbs_private *priv, struct cmd_ctrl_node *cmd,
|
|
int result);
|
|
void lbs_complete_command(struct lbs_private *priv, struct cmd_ctrl_node *cmd,
|
|
int result);
|
|
int lbs_process_command_response(struct lbs_private *priv, u8 *data, u32 len);
|
|
|
|
|
|
/* From cmdresp.c */
|
|
|
|
void lbs_mac_event_disconnected(struct lbs_private *priv);
|
|
|
|
|
|
|
|
/* Events */
|
|
|
|
int lbs_process_event(struct lbs_private *priv, u32 event);
|
|
|
|
|
|
/* Actual commands */
|
|
|
|
int lbs_update_hw_spec(struct lbs_private *priv);
|
|
|
|
int lbs_set_channel(struct lbs_private *priv, u8 channel);
|
|
|
|
int lbs_update_channel(struct lbs_private *priv);
|
|
|
|
int lbs_host_sleep_cfg(struct lbs_private *priv, uint32_t criteria,
|
|
struct wol_config *p_wol_config);
|
|
|
|
int lbs_cmd_802_11_sleep_params(struct lbs_private *priv, uint16_t cmd_action,
|
|
struct sleep_params *sp);
|
|
|
|
void lbs_ps_confirm_sleep(struct lbs_private *priv);
|
|
|
|
int lbs_set_radio(struct lbs_private *priv, u8 preamble, u8 radio_on);
|
|
|
|
void lbs_set_mac_control(struct lbs_private *priv);
|
|
int lbs_set_mac_control_sync(struct lbs_private *priv);
|
|
|
|
int lbs_get_tx_power(struct lbs_private *priv, s16 *curlevel, s16 *minlevel,
|
|
s16 *maxlevel);
|
|
|
|
int lbs_set_snmp_mib(struct lbs_private *priv, u32 oid, u16 val);
|
|
|
|
int lbs_get_snmp_mib(struct lbs_private *priv, u32 oid, u16 *out_val);
|
|
|
|
|
|
/* Commands only used in wext.c, assoc. and scan.c */
|
|
|
|
int lbs_set_power_adapt_cfg(struct lbs_private *priv, int enable, int8_t p0,
|
|
int8_t p1, int8_t p2);
|
|
|
|
int lbs_set_tpc_cfg(struct lbs_private *priv, int enable, int8_t p0, int8_t p1,
|
|
int8_t p2, int usesnr);
|
|
|
|
int lbs_set_data_rate(struct lbs_private *priv, u8 rate);
|
|
|
|
int lbs_cmd_802_11_rate_adapt_rateset(struct lbs_private *priv,
|
|
uint16_t cmd_action);
|
|
|
|
int lbs_set_tx_power(struct lbs_private *priv, s16 dbm);
|
|
|
|
int lbs_set_deep_sleep(struct lbs_private *priv, int deep_sleep);
|
|
|
|
int lbs_set_host_sleep(struct lbs_private *priv, int host_sleep);
|
|
|
|
int lbs_set_monitor_mode(struct lbs_private *priv, int enable);
|
|
|
|
int lbs_get_rssi(struct lbs_private *priv, s8 *snr, s8 *nf);
|
|
|
|
int lbs_set_11d_domain_info(struct lbs_private *priv);
|
|
|
|
int lbs_get_reg(struct lbs_private *priv, u16 reg, u16 offset, u32 *value);
|
|
|
|
int lbs_set_reg(struct lbs_private *priv, u16 reg, u16 offset, u32 value);
|
|
|
|
int lbs_set_ps_mode(struct lbs_private *priv, u16 cmd_action, bool block);
|
|
|
|
#endif /* _LBS_CMD_H */
|