mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2024-12-28 11:18:45 +07:00
wireless-drivers-next patches for v5.8
Third set of patches for v5.8. Final new features before the merge window (most likely) opens, noteworthy here is adding WPA3 support to old drivers rt2800, b43 and b43_legacy. Major changes: ath10k * SDIO and SNOC busses are not experimental anymore ath9k * allow receive of broadcast Action frames ath9k_htc * allow receive of broadcast Action frames rt2800 * enable WPA3 support out of box b43 * enable WPA3 support b43_legacy * enable WPA3 support mwifiex * advertise max number of clients to user space mt76 * mt7663: add remain-on-channel support * mt7915: add spatial reuse support * add support for mt7611n hardware iwlwifi * add ACPI DSM support * support enabling 5.2GHz bands in Indonesia via ACPI * bump FW API version to 56 -----BEGIN PGP SIGNATURE----- Version: GnuPG v1 iQEcBAABAgAGBQJe0nFMAAoJEG4XJFUm622bf04H/1IqM/9Wpt6qMSa+adBz5ij7 yNwWCD2nXAgyirafRIot6uoOY7ePfutwwUXD5IatwPFIEQVZP/uoRDqBPqsbz85J w5Izvpt15eG5uSDuy5rLUFucOz3XJZQkOGWRo/GKGSLjm2uPGGXAVul+fWX+i05J xfH1GvLys//An/X6AIx6kWqyJYqUfmxXj7pLkiOYFlt8JNS49gDaA1Emvv7/xv+Q hyBswKjNrEeT5K4dqYjQ/hp9/ukX8JQtpVBRF7rth5T8gxzJQmOKiP86L8YH5SyI r3U3ok1aUIuG6xhsTV/cWkqQeD6YsO9M1dTO1WQQLQyZgwl/esbjw/4BVNMGgVk= =9pX4 -----END PGP SIGNATURE----- Merge tag 'wireless-drivers-next-2020-05-30' of git://git.kernel.org/pub/scm/linux/kernel/git/kvalo/wireless-drivers-next Kalle Valo says: ==================== wireless-drivers-next patches for v5.8 Third set of patches for v5.8. Final new features before the merge window (most likely) opens, noteworthy here is adding WPA3 support to old drivers rt2800, b43 and b43_legacy. Major changes: ath10k * SDIO and SNOC busses are not experimental anymore ath9k * allow receive of broadcast Action frames ath9k_htc * allow receive of broadcast Action frames rt2800 * enable WPA3 support out of box b43 * enable WPA3 support b43_legacy * enable WPA3 support mwifiex * advertise max number of clients to user space mt76 * mt7663: add remain-on-channel support * mt7915: add spatial reuse support * add support for mt7611n hardware iwlwifi * add ACPI DSM support * support enabling 5.2GHz bands in Indonesia via ACPI * bump FW API version to 56 ==================== Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
commit
d7ad1415bd
@ -14066,7 +14066,7 @@ F: drivers/net/wireless/ath/wcn36xx/
|
||||
|
||||
QUANTENNA QTNFMAC WIRELESS DRIVER
|
||||
M: Igor Mitsyanko <imitsyanko@quantenna.com>
|
||||
M: Sergey Matyukevich <smatyukevich@quantenna.com>
|
||||
R: Sergey Matyukevich <geomatsi@gmail.com>
|
||||
L: linux-wireless@vger.kernel.org
|
||||
S: Maintained
|
||||
F: drivers/net/wireless/quantenna
|
||||
|
@ -28,11 +28,10 @@ config ATH10K_AHB
|
||||
This module adds support for AHB bus
|
||||
|
||||
config ATH10K_SDIO
|
||||
tristate "Atheros ath10k SDIO support (EXPERIMENTAL)"
|
||||
tristate "Atheros ath10k SDIO support"
|
||||
depends on ATH10K && MMC
|
||||
---help---
|
||||
This module adds experimental support for SDIO/MMC bus. Currently
|
||||
work in progress and will not fully work.
|
||||
This module adds support for SDIO/MMC bus.
|
||||
|
||||
config ATH10K_USB
|
||||
tristate "Atheros ath10k USB support (EXPERIMENTAL)"
|
||||
@ -42,7 +41,7 @@ config ATH10K_USB
|
||||
work in progress and will not fully work.
|
||||
|
||||
config ATH10K_SNOC
|
||||
tristate "Qualcomm ath10k SNOC support (EXPERIMENTAL)"
|
||||
tristate "Qualcomm ath10k SNOC support"
|
||||
depends on ATH10K
|
||||
depends on ARCH_QCOM || COMPILE_TEST
|
||||
select QCOM_QMI_HELPERS
|
||||
|
@ -110,7 +110,7 @@ struct ath10k_ce_ring {
|
||||
struct ce_desc_64 *shadow_base;
|
||||
|
||||
/* keep last */
|
||||
void *per_transfer_context[0];
|
||||
void *per_transfer_context[];
|
||||
};
|
||||
|
||||
struct ath10k_ce_pipe {
|
||||
|
@ -1262,7 +1262,7 @@ struct ath10k {
|
||||
int coex_gpio_pin;
|
||||
|
||||
/* must be last */
|
||||
u8 drv_priv[0] __aligned(sizeof(void *));
|
||||
u8 drv_priv[] __aligned(sizeof(void *));
|
||||
};
|
||||
|
||||
static inline bool ath10k_peer_stats_enabled(struct ath10k *ar)
|
||||
|
@ -88,7 +88,7 @@ struct ath10k_dump_file_data {
|
||||
u8 unused[128];
|
||||
|
||||
/* struct ath10k_tlv_dump_data + more */
|
||||
u8 data[0];
|
||||
u8 data[];
|
||||
} __packed;
|
||||
|
||||
struct ath10k_dump_ram_data_hdr {
|
||||
@ -100,7 +100,7 @@ struct ath10k_dump_ram_data_hdr {
|
||||
/* length of payload data, not including this header */
|
||||
__le32 length;
|
||||
|
||||
u8 data[0];
|
||||
u8 data[];
|
||||
};
|
||||
|
||||
/* magic number to fill the holes not copied due to sections in regions */
|
||||
|
@ -65,7 +65,7 @@ struct ath10k_pktlog_hdr {
|
||||
__le16 log_type; /* Type of log information foll this header */
|
||||
__le16 size; /* Size of variable length log information in bytes */
|
||||
__le32 timestamp;
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
/* FIXME: How to calculate the buffer size sanely? */
|
||||
|
@ -289,12 +289,12 @@ struct htt_rx_ring_setup_hdr {
|
||||
|
||||
struct htt_rx_ring_setup_32 {
|
||||
struct htt_rx_ring_setup_hdr hdr;
|
||||
struct htt_rx_ring_setup_ring32 rings[0];
|
||||
struct htt_rx_ring_setup_ring32 rings[];
|
||||
} __packed;
|
||||
|
||||
struct htt_rx_ring_setup_64 {
|
||||
struct htt_rx_ring_setup_hdr hdr;
|
||||
struct htt_rx_ring_setup_ring64 rings[0];
|
||||
struct htt_rx_ring_setup_ring64 rings[];
|
||||
} __packed;
|
||||
|
||||
/*
|
||||
@ -732,7 +732,7 @@ struct htt_rx_indication {
|
||||
* %mpdu_ranges starts after &%prefix + roundup(%fw_rx_desc_bytes, 4)
|
||||
* and has %num_mpdu_ranges elements.
|
||||
*/
|
||||
struct htt_rx_indication_mpdu_range mpdu_ranges[0];
|
||||
struct htt_rx_indication_mpdu_range mpdu_ranges[];
|
||||
} __packed;
|
||||
|
||||
/* High latency version of the RX indication */
|
||||
@ -741,7 +741,7 @@ struct htt_rx_indication_hl {
|
||||
struct htt_rx_indication_ppdu ppdu;
|
||||
struct htt_rx_indication_prefix prefix;
|
||||
struct fw_rx_desc_hl fw_desc;
|
||||
struct htt_rx_indication_mpdu_range mpdu_ranges[0];
|
||||
struct htt_rx_indication_mpdu_range mpdu_ranges[];
|
||||
} __packed;
|
||||
|
||||
struct htt_hl_rx_desc {
|
||||
@ -908,7 +908,7 @@ struct htt_append_retries {
|
||||
struct htt_data_tx_completion_ext {
|
||||
struct htt_append_retries a_retries;
|
||||
__le32 t_stamp;
|
||||
__le16 msdus_rssi[0];
|
||||
__le16 msdus_rssi[];
|
||||
} __packed;
|
||||
|
||||
/**
|
||||
@ -992,7 +992,7 @@ struct htt_data_tx_completion {
|
||||
} __packed;
|
||||
u8 num_msdus;
|
||||
u8 flags2; /* HTT_TX_CMPL_FLAG_DATA_RSSI */
|
||||
__le16 msdus[0]; /* variable length based on %num_msdus */
|
||||
__le16 msdus[]; /* variable length based on %num_msdus */
|
||||
} __packed;
|
||||
|
||||
#define HTT_TX_PPDU_DUR_INFO0_PEER_ID_MASK GENMASK(15, 0)
|
||||
@ -1007,7 +1007,7 @@ struct htt_data_tx_ppdu_dur {
|
||||
|
||||
struct htt_data_tx_compl_ppdu_dur {
|
||||
__le32 info0; /* HTT_TX_COMPL_PPDU_DUR_INFO0_ */
|
||||
struct htt_data_tx_ppdu_dur ppdu_dur[0];
|
||||
struct htt_data_tx_ppdu_dur ppdu_dur[];
|
||||
} __packed;
|
||||
|
||||
struct htt_tx_compl_ind_base {
|
||||
@ -1033,7 +1033,7 @@ struct htt_rc_update {
|
||||
u8 addr[6];
|
||||
u8 num_elems;
|
||||
u8 rsvd0;
|
||||
struct htt_rc_tx_done_params params[0]; /* variable length %num_elems */
|
||||
struct htt_rc_tx_done_params params[]; /* variable length %num_elems */
|
||||
} __packed;
|
||||
|
||||
/* see htt_rx_indication for similar fields and descriptions */
|
||||
@ -1050,7 +1050,7 @@ struct htt_rx_fragment_indication {
|
||||
__le16 fw_rx_desc_bytes;
|
||||
__le16 rsvd0;
|
||||
|
||||
u8 fw_msdu_rx_desc[0];
|
||||
u8 fw_msdu_rx_desc[];
|
||||
} __packed;
|
||||
|
||||
#define ATH10K_IEEE80211_EXTIV BIT(5)
|
||||
@ -1075,7 +1075,7 @@ struct htt_rx_pn_ind {
|
||||
u8 seqno_end;
|
||||
u8 pn_ie_count;
|
||||
u8 reserved;
|
||||
u8 pn_ies[0];
|
||||
u8 pn_ies[];
|
||||
} __packed;
|
||||
|
||||
struct htt_rx_offload_msdu {
|
||||
@ -1084,7 +1084,7 @@ struct htt_rx_offload_msdu {
|
||||
u8 vdev_id;
|
||||
u8 tid;
|
||||
u8 fw_desc;
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
struct htt_rx_offload_ind {
|
||||
@ -1167,7 +1167,7 @@ struct htt_rx_test {
|
||||
* a) num_ints * sizeof(__le32)
|
||||
* b) num_chars * sizeof(u8) aligned to 4bytes
|
||||
*/
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
static inline __le32 *htt_rx_test_get_ints(struct htt_rx_test *rx_test)
|
||||
@ -1201,7 +1201,7 @@ static inline u8 *htt_rx_test_get_chars(struct htt_rx_test *rx_test)
|
||||
*/
|
||||
struct htt_pktlog_msg {
|
||||
u8 pad[3];
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
struct htt_dbg_stats_rx_reorder_stats {
|
||||
@ -1490,7 +1490,7 @@ struct htt_stats_conf_item {
|
||||
} __packed;
|
||||
u8 pad;
|
||||
__le16 length;
|
||||
u8 payload[0]; /* roundup(length, 4) long */
|
||||
u8 payload[]; /* roundup(length, 4) long */
|
||||
} __packed;
|
||||
|
||||
struct htt_stats_conf {
|
||||
@ -1499,7 +1499,7 @@ struct htt_stats_conf {
|
||||
__le32 cookie_msb;
|
||||
|
||||
/* each item has variable length! */
|
||||
struct htt_stats_conf_item items[0];
|
||||
struct htt_stats_conf_item items[];
|
||||
} __packed;
|
||||
|
||||
static inline struct htt_stats_conf_item *htt_stats_conf_next_item(
|
||||
@ -1673,8 +1673,8 @@ struct htt_tx_fetch_ind {
|
||||
__le32 token;
|
||||
__le16 num_resp_ids;
|
||||
__le16 num_records;
|
||||
struct htt_tx_fetch_record records[0];
|
||||
__le32 resp_ids[0]; /* ath10k_htt_get_tx_fetch_ind_resp_ids() */
|
||||
struct htt_tx_fetch_record records[];
|
||||
} __packed;
|
||||
|
||||
static inline void *
|
||||
@ -1689,13 +1689,13 @@ struct htt_tx_fetch_resp {
|
||||
__le16 fetch_seq_num;
|
||||
__le16 num_records;
|
||||
__le32 token;
|
||||
struct htt_tx_fetch_record records[0];
|
||||
struct htt_tx_fetch_record records[];
|
||||
} __packed;
|
||||
|
||||
struct htt_tx_fetch_confirm {
|
||||
u8 pad0;
|
||||
__le16 num_resp_ids;
|
||||
__le32 resp_ids[0];
|
||||
__le32 resp_ids[];
|
||||
} __packed;
|
||||
|
||||
enum htt_tx_mode_switch_mode {
|
||||
@ -1727,7 +1727,7 @@ struct htt_tx_mode_switch_ind {
|
||||
__le16 info0; /* HTT_TX_MODE_SWITCH_IND_INFO0_ */
|
||||
__le16 info1; /* HTT_TX_MODE_SWITCH_IND_INFO1_ */
|
||||
u8 pad1[2];
|
||||
struct htt_tx_mode_switch_record records[0];
|
||||
struct htt_tx_mode_switch_record records[];
|
||||
} __packed;
|
||||
|
||||
struct htt_channel_change {
|
||||
@ -1757,7 +1757,7 @@ struct htt_peer_tx_stats {
|
||||
u8 num_ppdu;
|
||||
u8 ppdu_len;
|
||||
u8 version;
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
#define ATH10K_10_2_TX_STATS_OFFSET 136
|
||||
@ -2206,7 +2206,7 @@ struct htt_rx_desc {
|
||||
struct rx_ppdu_end ppdu_end;
|
||||
} __packed;
|
||||
u8 rx_hdr_status[RX_HTT_HDR_STATUS_LEN];
|
||||
u8 msdu_payload[0];
|
||||
u8 msdu_payload[];
|
||||
};
|
||||
|
||||
#define HTT_RX_DESC_HL_INFO_SEQ_NUM_MASK 0x00000fff
|
||||
|
@ -165,7 +165,7 @@ enum qca9377_chip_id_rev {
|
||||
struct ath10k_fw_ie {
|
||||
__le32 id;
|
||||
__le32 len;
|
||||
u8 data[0];
|
||||
u8 data[];
|
||||
};
|
||||
|
||||
enum ath10k_fw_ie_type {
|
||||
|
@ -3967,6 +3967,9 @@ void ath10k_mgmt_over_wmi_tx_work(struct work_struct *work)
|
||||
if (ret) {
|
||||
ath10k_warn(ar, "failed to transmit management frame by ref via WMI: %d\n",
|
||||
ret);
|
||||
/* remove this msdu from idr tracking */
|
||||
ath10k_wmi_cleanup_mgmt_tx_send(ar, skb);
|
||||
|
||||
dma_unmap_single(ar->dev, paddr, skb->len,
|
||||
DMA_TO_DEVICE);
|
||||
ieee80211_free_txskb(ar->hw, skb);
|
||||
|
@ -178,15 +178,16 @@ struct ath10k_pci {
|
||||
*/
|
||||
u32 (*targ_cpu_to_ce_addr)(struct ath10k *ar, u32 addr);
|
||||
|
||||
struct ce_attr *attr;
|
||||
struct ce_pipe_config *pipe_config;
|
||||
struct ce_service_to_pipe *serv_to_pipe;
|
||||
|
||||
/* Keep this entry in the last, memory for struct ath10k_ahb is
|
||||
* allocated (ahb support enabled case) in the continuation of
|
||||
* this struct.
|
||||
*/
|
||||
struct ath10k_ahb ahb[0];
|
||||
struct ath10k_ahb ahb[];
|
||||
|
||||
struct ce_attr *attr;
|
||||
struct ce_pipe_config *pipe_config;
|
||||
struct ce_service_to_pipe *serv_to_pipe;
|
||||
};
|
||||
|
||||
static inline struct ath10k_pci *ath10k_pci_priv(struct ath10k *ar)
|
||||
|
@ -961,7 +961,16 @@ static void ath10k_qmi_del_server(struct qmi_handle *qmi_hdl,
|
||||
container_of(qmi_hdl, struct ath10k_qmi, qmi_hdl);
|
||||
|
||||
qmi->fw_ready = false;
|
||||
ath10k_qmi_driver_event_post(qmi, ATH10K_QMI_EVENT_SERVER_EXIT, NULL);
|
||||
|
||||
/*
|
||||
* The del_server event is to be processed only if coming from
|
||||
* the qmi server. The qmi infrastructure sends del_server, when
|
||||
* any client releases the qmi handle. In this case do not process
|
||||
* this del_server event.
|
||||
*/
|
||||
if (qmi->state == ATH10K_QMI_STATE_INIT_DONE)
|
||||
ath10k_qmi_driver_event_post(qmi, ATH10K_QMI_EVENT_SERVER_EXIT,
|
||||
NULL);
|
||||
}
|
||||
|
||||
static struct qmi_ops ath10k_qmi_ops = {
|
||||
@ -1046,6 +1055,7 @@ int ath10k_qmi_init(struct ath10k *ar, u32 msa_size)
|
||||
if (ret)
|
||||
goto err_qmi_lookup;
|
||||
|
||||
qmi->state = ATH10K_QMI_STATE_INIT_DONE;
|
||||
return 0;
|
||||
|
||||
err_qmi_lookup:
|
||||
@ -1064,6 +1074,7 @@ int ath10k_qmi_deinit(struct ath10k *ar)
|
||||
struct ath10k_snoc *ar_snoc = ath10k_snoc_priv(ar);
|
||||
struct ath10k_qmi *qmi = ar_snoc->qmi;
|
||||
|
||||
qmi->state = ATH10K_QMI_STATE_DEINIT;
|
||||
qmi_handle_release(&qmi->qmi_hdl);
|
||||
cancel_work_sync(&qmi->event_work);
|
||||
destroy_workqueue(qmi->event_wq);
|
||||
|
@ -83,6 +83,11 @@ struct ath10k_qmi_driver_event {
|
||||
void *data;
|
||||
};
|
||||
|
||||
enum ath10k_qmi_state {
|
||||
ATH10K_QMI_STATE_INIT_DONE,
|
||||
ATH10K_QMI_STATE_DEINIT,
|
||||
};
|
||||
|
||||
struct ath10k_qmi {
|
||||
struct ath10k *ar;
|
||||
struct qmi_handle qmi_hdl;
|
||||
@ -102,6 +107,7 @@ struct ath10k_qmi {
|
||||
char fw_build_timestamp[MAX_TIMESTAMP_LEN + 1];
|
||||
struct ath10k_qmi_cal_data cal_data[MAX_NUM_CAL_V01];
|
||||
bool msa_fixed_perm;
|
||||
enum ath10k_qmi_state state;
|
||||
};
|
||||
|
||||
int ath10k_qmi_wlan_enable(struct ath10k *ar,
|
||||
@ -109,7 +115,6 @@ int ath10k_qmi_wlan_enable(struct ath10k *ar,
|
||||
enum wlfw_driver_mode_enum_v01 mode,
|
||||
const char *version);
|
||||
int ath10k_qmi_wlan_disable(struct ath10k *ar);
|
||||
int ath10k_qmi_register_service_notifier(struct notifier_block *nb);
|
||||
int ath10k_qmi_init(struct ath10k *ar, u32 msa_size);
|
||||
int ath10k_qmi_deinit(struct ath10k *ar);
|
||||
int ath10k_qmi_set_fw_log_mode(struct ath10k *ar, u8 fw_log_mode);
|
||||
|
@ -140,6 +140,7 @@ struct wmi_ops {
|
||||
struct sk_buff *(*gen_mgmt_tx_send)(struct ath10k *ar,
|
||||
struct sk_buff *skb,
|
||||
dma_addr_t paddr);
|
||||
int (*cleanup_mgmt_tx_send)(struct ath10k *ar, struct sk_buff *msdu);
|
||||
struct sk_buff *(*gen_dbglog_cfg)(struct ath10k *ar, u64 module_enable,
|
||||
u32 log_level);
|
||||
struct sk_buff *(*gen_pktlog_enable)(struct ath10k *ar, u32 filter);
|
||||
@ -448,6 +449,15 @@ ath10k_wmi_get_txbf_conf_scheme(struct ath10k *ar)
|
||||
return ar->wmi.ops->get_txbf_conf_scheme(ar);
|
||||
}
|
||||
|
||||
static inline int
|
||||
ath10k_wmi_cleanup_mgmt_tx_send(struct ath10k *ar, struct sk_buff *msdu)
|
||||
{
|
||||
if (!ar->wmi.ops->cleanup_mgmt_tx_send)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
return ar->wmi.ops->cleanup_mgmt_tx_send(ar, msdu);
|
||||
}
|
||||
|
||||
static inline int
|
||||
ath10k_wmi_mgmt_tx_send(struct ath10k *ar, struct sk_buff *msdu,
|
||||
dma_addr_t paddr)
|
||||
|
@ -3010,11 +3010,23 @@ ath10k_wmi_tlv_op_gen_request_peer_stats_info(struct ath10k *ar,
|
||||
if (type == WMI_REQUEST_ONE_PEER_STATS_INFO)
|
||||
ether_addr_copy(cmd->peer_macaddr.addr, addr);
|
||||
|
||||
cmd->reset_after_request = reset;
|
||||
cmd->reset_after_request = __cpu_to_le32(reset);
|
||||
ath10k_dbg(ar, ATH10K_DBG_WMI, "wmi tlv request peer stats info\n");
|
||||
return skb;
|
||||
}
|
||||
|
||||
static int
|
||||
ath10k_wmi_tlv_op_cleanup_mgmt_tx_send(struct ath10k *ar,
|
||||
struct sk_buff *msdu)
|
||||
{
|
||||
struct ath10k_skb_cb *cb = ATH10K_SKB_CB(msdu);
|
||||
struct ath10k_wmi *wmi = &ar->wmi;
|
||||
|
||||
idr_remove(&wmi->mgmt_pending_tx, cb->msdu_id);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
ath10k_wmi_mgmt_tx_alloc_msdu_id(struct ath10k *ar, struct sk_buff *skb,
|
||||
dma_addr_t paddr)
|
||||
@ -3089,6 +3101,8 @@ ath10k_wmi_tlv_op_gen_mgmt_tx_send(struct ath10k *ar, struct sk_buff *msdu,
|
||||
if (desc_id < 0)
|
||||
goto err_free_skb;
|
||||
|
||||
cb->msdu_id = desc_id;
|
||||
|
||||
ptr = (void *)skb->data;
|
||||
tlv = ptr;
|
||||
tlv->tag = __cpu_to_le16(WMI_TLV_TAG_STRUCT_MGMT_TX_CMD);
|
||||
@ -4540,6 +4554,7 @@ static const struct wmi_ops wmi_tlv_ops = {
|
||||
.gen_force_fw_hang = ath10k_wmi_tlv_op_gen_force_fw_hang,
|
||||
/* .gen_mgmt_tx = not implemented; HTT is used */
|
||||
.gen_mgmt_tx_send = ath10k_wmi_tlv_op_gen_mgmt_tx_send,
|
||||
.cleanup_mgmt_tx_send = ath10k_wmi_tlv_op_cleanup_mgmt_tx_send,
|
||||
.gen_dbglog_cfg = ath10k_wmi_tlv_op_gen_dbglog_cfg,
|
||||
.gen_pktlog_enable = ath10k_wmi_tlv_op_gen_pktlog_enable,
|
||||
.gen_pktlog_disable = ath10k_wmi_tlv_op_gen_pktlog_disable,
|
||||
|
@ -1637,7 +1637,7 @@ wmi_tlv_svc_map_ext(const __le32 *in, unsigned long *out, size_t len)
|
||||
struct wmi_tlv {
|
||||
__le16 len;
|
||||
__le16 tag;
|
||||
u8 value[0];
|
||||
u8 value[];
|
||||
} __packed;
|
||||
|
||||
struct ath10k_mgmt_tx_pkt_addr {
|
||||
@ -2037,7 +2037,7 @@ struct wmi_tlv_bcn_tx_status_ev {
|
||||
struct wmi_tlv_bcn_prb_info {
|
||||
__le32 caps;
|
||||
__le32 erp;
|
||||
u8 ies[0];
|
||||
u8 ies[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_tlv_bcn_tmpl_cmd {
|
||||
@ -2068,7 +2068,7 @@ struct wmi_tlv_diag_item {
|
||||
__le16 len;
|
||||
__le32 timestamp;
|
||||
__le32 code;
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_tlv_diag_data_ev {
|
||||
|
@ -2292,7 +2292,7 @@ struct wmi_service_ready_event {
|
||||
* where FW can access this memory directly (or) by DMA.
|
||||
*/
|
||||
__le32 num_mem_reqs;
|
||||
struct wlan_host_mem_req mem_reqs[0];
|
||||
struct wlan_host_mem_req mem_reqs[];
|
||||
} __packed;
|
||||
|
||||
/* This is the definition from 10.X firmware branch */
|
||||
@ -2331,7 +2331,7 @@ struct wmi_10x_service_ready_event {
|
||||
*/
|
||||
__le32 num_mem_reqs;
|
||||
|
||||
struct wlan_host_mem_req mem_reqs[0];
|
||||
struct wlan_host_mem_req mem_reqs[];
|
||||
} __packed;
|
||||
|
||||
#define WMI_SERVICE_READY_TIMEOUT_HZ (5 * HZ)
|
||||
@ -3086,19 +3086,19 @@ struct wmi_chan_list_entry {
|
||||
struct wmi_chan_list {
|
||||
__le32 tag; /* WMI_CHAN_LIST_TAG */
|
||||
__le32 num_chan;
|
||||
struct wmi_chan_list_entry channel_list[0];
|
||||
struct wmi_chan_list_entry channel_list[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_bssid_list {
|
||||
__le32 tag; /* WMI_BSSID_LIST_TAG */
|
||||
__le32 num_bssid;
|
||||
struct wmi_mac_addr bssid_list[0];
|
||||
struct wmi_mac_addr bssid_list[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_ie_data {
|
||||
__le32 tag; /* WMI_IE_TAG */
|
||||
__le32 ie_len;
|
||||
u8 ie_data[0];
|
||||
u8 ie_data[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_ssid {
|
||||
@ -3109,7 +3109,7 @@ struct wmi_ssid {
|
||||
struct wmi_ssid_list {
|
||||
__le32 tag; /* WMI_SSID_LIST_TAG */
|
||||
__le32 num_ssids;
|
||||
struct wmi_ssid ssids[0];
|
||||
struct wmi_ssid ssids[];
|
||||
} __packed;
|
||||
|
||||
/* prefix used by scan requestor ids on the host */
|
||||
@ -3311,7 +3311,7 @@ struct wmi_stop_scan_arg {
|
||||
|
||||
struct wmi_scan_chan_list_cmd {
|
||||
__le32 num_scan_chans;
|
||||
struct wmi_channel chan_info[0];
|
||||
struct wmi_channel chan_info[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_scan_chan_list_arg {
|
||||
@ -3395,12 +3395,12 @@ struct wmi_mgmt_rx_hdr_v2 {
|
||||
|
||||
struct wmi_mgmt_rx_event_v1 {
|
||||
struct wmi_mgmt_rx_hdr_v1 hdr;
|
||||
u8 buf[0];
|
||||
u8 buf[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_mgmt_rx_event_v2 {
|
||||
struct wmi_mgmt_rx_hdr_v2 hdr;
|
||||
u8 buf[0];
|
||||
u8 buf[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_10_4_mgmt_rx_hdr {
|
||||
@ -3415,7 +3415,7 @@ struct wmi_10_4_mgmt_rx_hdr {
|
||||
|
||||
struct wmi_10_4_mgmt_rx_event {
|
||||
struct wmi_10_4_mgmt_rx_hdr hdr;
|
||||
u8 buf[0];
|
||||
u8 buf[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_mgmt_rx_ext_info {
|
||||
@ -3455,14 +3455,14 @@ struct wmi_phyerr {
|
||||
__le32 rssi_chains[4];
|
||||
__le16 nf_chains[4];
|
||||
__le32 buf_len;
|
||||
u8 buf[0];
|
||||
u8 buf[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_phyerr_event {
|
||||
__le32 num_phyerrs;
|
||||
__le32 tsf_l32;
|
||||
__le32 tsf_u32;
|
||||
struct wmi_phyerr phyerrs[0];
|
||||
struct wmi_phyerr phyerrs[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_10_4_phyerr_event {
|
||||
@ -3479,7 +3479,7 @@ struct wmi_10_4_phyerr_event {
|
||||
__le32 phy_err_mask[2];
|
||||
__le32 tsf_timestamp;
|
||||
__le32 buf_len;
|
||||
u8 buf[0];
|
||||
u8 buf[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_radar_found_info {
|
||||
@ -3592,7 +3592,7 @@ struct wmi_mgmt_tx_hdr {
|
||||
|
||||
struct wmi_mgmt_tx_cmd {
|
||||
struct wmi_mgmt_tx_hdr hdr;
|
||||
u8 buf[0];
|
||||
u8 buf[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_echo_event {
|
||||
@ -4628,7 +4628,7 @@ struct wmi_stats_event {
|
||||
* By having a zero sized array, the pointer to data area
|
||||
* becomes available without increasing the struct size
|
||||
*/
|
||||
u8 data[0];
|
||||
u8 data[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_10_2_stats_event {
|
||||
@ -4638,7 +4638,7 @@ struct wmi_10_2_stats_event {
|
||||
__le32 num_vdev_stats;
|
||||
__le32 num_peer_stats;
|
||||
__le32 num_bcnflt_stats;
|
||||
u8 data[0];
|
||||
u8 data[];
|
||||
} __packed;
|
||||
|
||||
/*
|
||||
@ -5033,7 +5033,7 @@ struct wmi_vdev_install_key_cmd {
|
||||
__le32 key_rxmic_len;
|
||||
|
||||
/* contains key followed by tx mic followed by rx mic */
|
||||
u8 key_data[0];
|
||||
u8 key_data[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_vdev_install_key_arg {
|
||||
@ -5703,7 +5703,7 @@ struct wmi_bcn_tx_hdr {
|
||||
|
||||
struct wmi_bcn_tx_cmd {
|
||||
struct wmi_bcn_tx_hdr hdr;
|
||||
u8 *bcn[0];
|
||||
u8 *bcn[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_bcn_tx_arg {
|
||||
@ -6120,7 +6120,7 @@ struct wmi_bcn_info {
|
||||
|
||||
struct wmi_host_swba_event {
|
||||
__le32 vdev_map;
|
||||
struct wmi_bcn_info bcn_info[0];
|
||||
struct wmi_bcn_info bcn_info[];
|
||||
} __packed;
|
||||
|
||||
struct wmi_10_2_4_bcn_info {
|
||||
@ -6130,7 +6130,7 @@ struct wmi_10_2_4_bcn_info {
|
||||
|
||||
struct wmi_10_2_4_host_swba_event {
|
||||
__le32 vdev_map;
|
||||
struct wmi_10_2_4_bcn_info bcn_info[0];
|
||||
struct wmi_10_2_4_bcn_info bcn_info[];
|
||||
} __packed;
|
||||
|
||||
/* 16 words = 512 client + 1 word = for guard */
|
||||
@ -6171,7 +6171,7 @@ struct wmi_10_4_bcn_info {
|
||||
|
||||
struct wmi_10_4_host_swba_event {
|
||||
__le32 vdev_map;
|
||||
struct wmi_10_4_bcn_info bcn_info[0];
|
||||
struct wmi_10_4_bcn_info bcn_info[];
|
||||
} __packed;
|
||||
|
||||
#define WMI_MAX_AP_VDEV 16
|
||||
|
@ -10,6 +10,7 @@
|
||||
#include <linux/dma-mapping.h>
|
||||
#include "ahb.h"
|
||||
#include "debug.h"
|
||||
#include "hif.h"
|
||||
#include <linux/remoteproc.h>
|
||||
|
||||
static const struct of_device_id ath11k_ahb_of_match[] = {
|
||||
@ -434,6 +435,16 @@ enum ext_irq_num {
|
||||
tcl2host_status_ring,
|
||||
};
|
||||
|
||||
static inline u32 ath11k_ahb_read32(struct ath11k_base *ab, u32 offset)
|
||||
{
|
||||
return ioread32(ab->mem + offset);
|
||||
}
|
||||
|
||||
static inline void ath11k_ahb_write32(struct ath11k_base *ab, u32 offset, u32 value)
|
||||
{
|
||||
iowrite32(value, ab->mem + offset);
|
||||
}
|
||||
|
||||
static void ath11k_ahb_kill_tasklets(struct ath11k_base *ab)
|
||||
{
|
||||
int i;
|
||||
@ -575,7 +586,7 @@ static void ath11k_ahb_ce_irqs_disable(struct ath11k_base *ab)
|
||||
}
|
||||
}
|
||||
|
||||
int ath11k_ahb_start(struct ath11k_base *ab)
|
||||
static int ath11k_ahb_start(struct ath11k_base *ab)
|
||||
{
|
||||
ath11k_ahb_ce_irqs_enable(ab);
|
||||
ath11k_ce_rx_post_buf(ab);
|
||||
@ -583,7 +594,7 @@ int ath11k_ahb_start(struct ath11k_base *ab)
|
||||
return 0;
|
||||
}
|
||||
|
||||
void ath11k_ahb_ext_irq_enable(struct ath11k_base *ab)
|
||||
static void ath11k_ahb_ext_irq_enable(struct ath11k_base *ab)
|
||||
{
|
||||
int i;
|
||||
|
||||
@ -595,13 +606,13 @@ void ath11k_ahb_ext_irq_enable(struct ath11k_base *ab)
|
||||
}
|
||||
}
|
||||
|
||||
void ath11k_ahb_ext_irq_disable(struct ath11k_base *ab)
|
||||
static void ath11k_ahb_ext_irq_disable(struct ath11k_base *ab)
|
||||
{
|
||||
__ath11k_ahb_ext_irq_disable(ab);
|
||||
ath11k_ahb_sync_ext_irqs(ab);
|
||||
}
|
||||
|
||||
void ath11k_ahb_stop(struct ath11k_base *ab)
|
||||
static void ath11k_ahb_stop(struct ath11k_base *ab)
|
||||
{
|
||||
if (!test_bit(ATH11K_FLAG_CRASH_FLUSH, &ab->dev_flags))
|
||||
ath11k_ahb_ce_irqs_disable(ab);
|
||||
@ -611,7 +622,7 @@ void ath11k_ahb_stop(struct ath11k_base *ab)
|
||||
ath11k_ce_cleanup_pipes(ab);
|
||||
}
|
||||
|
||||
int ath11k_ahb_power_up(struct ath11k_base *ab)
|
||||
static int ath11k_ahb_power_up(struct ath11k_base *ab)
|
||||
{
|
||||
int ret;
|
||||
|
||||
@ -622,7 +633,7 @@ int ath11k_ahb_power_up(struct ath11k_base *ab)
|
||||
return ret;
|
||||
}
|
||||
|
||||
void ath11k_ahb_power_down(struct ath11k_base *ab)
|
||||
static void ath11k_ahb_power_down(struct ath11k_base *ab)
|
||||
{
|
||||
rproc_shutdown(ab->tgt_rproc);
|
||||
}
|
||||
@ -834,8 +845,8 @@ static int ath11k_ahb_config_irq(struct ath11k_base *ab)
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ath11k_ahb_map_service_to_pipe(struct ath11k_base *ab, u16 service_id,
|
||||
u8 *ul_pipe, u8 *dl_pipe)
|
||||
static int ath11k_ahb_map_service_to_pipe(struct ath11k_base *ab, u16 service_id,
|
||||
u8 *ul_pipe, u8 *dl_pipe)
|
||||
{
|
||||
const struct service_to_pipe *entry;
|
||||
bool ul_set = false, dl_set = false;
|
||||
@ -877,6 +888,18 @@ int ath11k_ahb_map_service_to_pipe(struct ath11k_base *ab, u16 service_id,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct ath11k_hif_ops ath11k_ahb_hif_ops = {
|
||||
.start = ath11k_ahb_start,
|
||||
.stop = ath11k_ahb_stop,
|
||||
.read32 = ath11k_ahb_read32,
|
||||
.write32 = ath11k_ahb_write32,
|
||||
.irq_enable = ath11k_ahb_ext_irq_enable,
|
||||
.irq_disable = ath11k_ahb_ext_irq_disable,
|
||||
.map_service_to_pipe = ath11k_ahb_map_service_to_pipe,
|
||||
.power_down = ath11k_ahb_power_down,
|
||||
.power_up = ath11k_ahb_power_up,
|
||||
};
|
||||
|
||||
static int ath11k_ahb_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct ath11k_base *ab;
|
||||
@ -891,13 +914,7 @@ static int ath11k_ahb_probe(struct platform_device *pdev)
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
mem_res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||
if (!mem_res) {
|
||||
dev_err(&pdev->dev, "failed to get IO memory resource\n");
|
||||
return -ENXIO;
|
||||
}
|
||||
|
||||
mem = devm_ioremap_resource(&pdev->dev, mem_res);
|
||||
mem = devm_platform_get_and_ioremap_resource(pdev, 0, &mem_res);
|
||||
if (IS_ERR(mem)) {
|
||||
dev_err(&pdev->dev, "ioremap error\n");
|
||||
return PTR_ERR(mem);
|
||||
@ -909,12 +926,13 @@ static int ath11k_ahb_probe(struct platform_device *pdev)
|
||||
return ret;
|
||||
}
|
||||
|
||||
ab = ath11k_core_alloc(&pdev->dev);
|
||||
ab = ath11k_core_alloc(&pdev->dev, 0, ATH11K_BUS_AHB);
|
||||
if (!ab) {
|
||||
dev_err(&pdev->dev, "failed to allocate ath11k base\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
ab->hif.ops = &ath11k_ahb_hif_ops;
|
||||
ab->pdev = pdev;
|
||||
ab->hw_rev = (enum ath11k_hw_rev)of_id->data;
|
||||
ab->mem = mem;
|
||||
@ -993,12 +1011,17 @@ static struct platform_driver ath11k_ahb_driver = {
|
||||
.remove = ath11k_ahb_remove,
|
||||
};
|
||||
|
||||
int ath11k_ahb_init(void)
|
||||
static int ath11k_ahb_init(void)
|
||||
{
|
||||
return platform_driver_register(&ath11k_ahb_driver);
|
||||
}
|
||||
module_init(ath11k_ahb_init);
|
||||
|
||||
void ath11k_ahb_exit(void)
|
||||
static void ath11k_ahb_exit(void)
|
||||
{
|
||||
platform_driver_unregister(&ath11k_ahb_driver);
|
||||
}
|
||||
module_exit(ath11k_ahb_exit);
|
||||
|
||||
MODULE_DESCRIPTION("Driver support for Qualcomm Technologies 802.11ax wireless chip");
|
||||
MODULE_LICENSE("Dual BSD/GPL");
|
||||
|
@ -10,26 +10,4 @@
|
||||
#define ATH11K_AHB_RECOVERY_TIMEOUT (3 * HZ)
|
||||
struct ath11k_base;
|
||||
|
||||
static inline u32 ath11k_ahb_read32(struct ath11k_base *ab, u32 offset)
|
||||
{
|
||||
return ioread32(ab->mem + offset);
|
||||
}
|
||||
|
||||
static inline void ath11k_ahb_write32(struct ath11k_base *ab, u32 offset, u32 value)
|
||||
{
|
||||
iowrite32(value, ab->mem + offset);
|
||||
}
|
||||
|
||||
void ath11k_ahb_ext_irq_enable(struct ath11k_base *ab);
|
||||
void ath11k_ahb_ext_irq_disable(struct ath11k_base *ab);
|
||||
int ath11k_ahb_start(struct ath11k_base *ab);
|
||||
void ath11k_ahb_stop(struct ath11k_base *ab);
|
||||
int ath11k_ahb_power_up(struct ath11k_base *ab);
|
||||
void ath11k_ahb_power_down(struct ath11k_base *ab);
|
||||
int ath11k_ahb_map_service_to_pipe(struct ath11k_base *ab, u16 service_id,
|
||||
u8 *ul_pipe, u8 *dl_pipe);
|
||||
|
||||
int ath11k_ahb_init(void);
|
||||
void ath11k_ahb_exit(void);
|
||||
|
||||
#endif
|
||||
|
@ -7,11 +7,11 @@
|
||||
#include <linux/slab.h>
|
||||
#include <linux/remoteproc.h>
|
||||
#include <linux/firmware.h>
|
||||
#include "ahb.h"
|
||||
#include "core.h"
|
||||
#include "dp_tx.h"
|
||||
#include "dp_rx.h"
|
||||
#include "debug.h"
|
||||
#include "hif.h"
|
||||
|
||||
unsigned int ath11k_debug_mask;
|
||||
module_param_named(debug_mask, ath11k_debug_mask, uint, 0644);
|
||||
@ -41,6 +41,7 @@ u8 ath11k_core_get_hw_mac_id(struct ath11k_base *ab, int pdev_idx)
|
||||
return ATH11K_INVALID_HW_MAC_ID;
|
||||
}
|
||||
}
|
||||
EXPORT_SYMBOL(ath11k_core_get_hw_mac_id);
|
||||
|
||||
static int ath11k_core_create_board_name(struct ath11k_base *ab, char *name,
|
||||
size_t name_len)
|
||||
@ -324,7 +325,7 @@ static void ath11k_core_stop(struct ath11k_base *ab)
|
||||
{
|
||||
if (!test_bit(ATH11K_FLAG_CRASH_FLUSH, &ab->dev_flags))
|
||||
ath11k_qmi_firmware_stop(ab);
|
||||
ath11k_ahb_stop(ab);
|
||||
ath11k_hif_stop(ab);
|
||||
ath11k_wmi_detach(ab);
|
||||
ath11k_dp_pdev_reo_cleanup(ab);
|
||||
|
||||
@ -347,7 +348,7 @@ static int ath11k_core_soc_create(struct ath11k_base *ab)
|
||||
goto err_qmi_deinit;
|
||||
}
|
||||
|
||||
ret = ath11k_ahb_power_up(ab);
|
||||
ret = ath11k_hif_power_up(ab);
|
||||
if (ret) {
|
||||
ath11k_err(ab, "failed to power up :%d\n", ret);
|
||||
goto err_debugfs_reg;
|
||||
@ -415,7 +416,7 @@ static void ath11k_core_pdev_destroy(struct ath11k_base *ab)
|
||||
{
|
||||
ath11k_thermal_unregister(ab);
|
||||
ath11k_mac_unregister(ab);
|
||||
ath11k_ahb_ext_irq_disable(ab);
|
||||
ath11k_hif_irq_disable(ab);
|
||||
ath11k_dp_pdev_free(ab);
|
||||
ath11k_debug_pdev_destroy(ab);
|
||||
}
|
||||
@ -443,7 +444,7 @@ static int ath11k_core_start(struct ath11k_base *ab,
|
||||
goto err_wmi_detach;
|
||||
}
|
||||
|
||||
ret = ath11k_ahb_start(ab);
|
||||
ret = ath11k_hif_start(ab);
|
||||
if (ret) {
|
||||
ath11k_err(ab, "failed to start HIF: %d\n", ret);
|
||||
goto err_wmi_detach;
|
||||
@ -522,7 +523,7 @@ static int ath11k_core_start(struct ath11k_base *ab,
|
||||
err_mac_destroy:
|
||||
ath11k_mac_destroy(ab);
|
||||
err_hif_stop:
|
||||
ath11k_ahb_stop(ab);
|
||||
ath11k_hif_stop(ab);
|
||||
err_wmi_detach:
|
||||
ath11k_wmi_detach(ab);
|
||||
err_firmware_stop:
|
||||
@ -559,7 +560,7 @@ int ath11k_core_qmi_firmware_ready(struct ath11k_base *ab)
|
||||
ath11k_err(ab, "failed to create pdev core: %d\n", ret);
|
||||
goto err_core_stop;
|
||||
}
|
||||
ath11k_ahb_ext_irq_enable(ab);
|
||||
ath11k_hif_irq_enable(ab);
|
||||
mutex_unlock(&ab->core_lock);
|
||||
|
||||
return 0;
|
||||
@ -579,9 +580,9 @@ static int ath11k_core_reconfigure_on_crash(struct ath11k_base *ab)
|
||||
|
||||
mutex_lock(&ab->core_lock);
|
||||
ath11k_thermal_unregister(ab);
|
||||
ath11k_ahb_ext_irq_disable(ab);
|
||||
ath11k_hif_irq_disable(ab);
|
||||
ath11k_dp_pdev_free(ab);
|
||||
ath11k_ahb_stop(ab);
|
||||
ath11k_hif_stop(ab);
|
||||
ath11k_wmi_detach(ab);
|
||||
ath11k_dp_pdev_reo_cleanup(ab);
|
||||
mutex_unlock(&ab->core_lock);
|
||||
@ -744,7 +745,7 @@ void ath11k_core_deinit(struct ath11k_base *ab)
|
||||
|
||||
mutex_unlock(&ab->core_lock);
|
||||
|
||||
ath11k_ahb_power_down(ab);
|
||||
ath11k_hif_power_down(ab);
|
||||
ath11k_mac_destroy(ab);
|
||||
ath11k_core_soc_destroy(ab);
|
||||
}
|
||||
@ -754,11 +755,12 @@ void ath11k_core_free(struct ath11k_base *ab)
|
||||
kfree(ab);
|
||||
}
|
||||
|
||||
struct ath11k_base *ath11k_core_alloc(struct device *dev)
|
||||
struct ath11k_base *ath11k_core_alloc(struct device *dev, size_t priv_size,
|
||||
enum ath11k_bus bus)
|
||||
{
|
||||
struct ath11k_base *ab;
|
||||
|
||||
ab = kzalloc(sizeof(*ab), GFP_KERNEL);
|
||||
ab = kzalloc(sizeof(*ab) + priv_size, GFP_KERNEL);
|
||||
if (!ab)
|
||||
return NULL;
|
||||
|
||||
@ -784,24 +786,3 @@ struct ath11k_base *ath11k_core_alloc(struct device *dev)
|
||||
kfree(ab);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static int __init ath11k_init(void)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = ath11k_ahb_init();
|
||||
if (ret)
|
||||
printk(KERN_ERR "failed to register ath11k ahb driver: %d\n",
|
||||
ret);
|
||||
return ret;
|
||||
}
|
||||
module_init(ath11k_init);
|
||||
|
||||
static void __exit ath11k_exit(void)
|
||||
{
|
||||
ath11k_ahb_exit();
|
||||
}
|
||||
module_exit(ath11k_exit);
|
||||
|
||||
MODULE_DESCRIPTION("Driver support for Qualcomm Technologies 802.11ax wireless chip");
|
||||
MODULE_LICENSE("Dual BSD/GPL");
|
||||
|
@ -607,7 +607,9 @@ struct ath11k_base {
|
||||
void __iomem *mem;
|
||||
unsigned long mem_len;
|
||||
|
||||
const struct ath11k_hif_ops *hif_ops;
|
||||
struct {
|
||||
const struct ath11k_hif_ops *ops;
|
||||
} hif;
|
||||
|
||||
struct ath11k_ce ce;
|
||||
struct timer_list rx_replenish_retry;
|
||||
@ -665,6 +667,9 @@ struct ath11k_base {
|
||||
|
||||
/* Round robbin based TCL ring selector */
|
||||
atomic_t tcl_ring_selector;
|
||||
|
||||
/* must be last */
|
||||
u8 drv_priv[0] __aligned(sizeof(void *));
|
||||
};
|
||||
|
||||
struct ath11k_fw_stats_pdev {
|
||||
@ -801,7 +806,8 @@ struct ath11k_peer *ath11k_peer_find_by_id(struct ath11k_base *ab, int peer_id);
|
||||
int ath11k_core_qmi_firmware_ready(struct ath11k_base *ab);
|
||||
int ath11k_core_init(struct ath11k_base *ath11k);
|
||||
void ath11k_core_deinit(struct ath11k_base *ath11k);
|
||||
struct ath11k_base *ath11k_core_alloc(struct device *dev);
|
||||
struct ath11k_base *ath11k_core_alloc(struct device *dev, size_t priv_size,
|
||||
enum ath11k_bus bus);
|
||||
void ath11k_core_free(struct ath11k_base *ath11k);
|
||||
int ath11k_core_fetch_bdf(struct ath11k_base *ath11k,
|
||||
struct ath11k_board_data *bd);
|
||||
|
@ -4306,6 +4306,7 @@ void ath11k_dbg_htt_ext_stats_handler(struct ath11k_base *ab,
|
||||
u32 len;
|
||||
u64 cookie;
|
||||
int ret;
|
||||
bool send_completion = false;
|
||||
u8 pdev_id;
|
||||
|
||||
msg = (struct ath11k_htt_extd_stats_msg *)skb->data;
|
||||
@ -4330,11 +4331,11 @@ void ath11k_dbg_htt_ext_stats_handler(struct ath11k_base *ab,
|
||||
return;
|
||||
|
||||
spin_lock_bh(&ar->debug.htt_stats.lock);
|
||||
if (stats_req->done) {
|
||||
spin_unlock_bh(&ar->debug.htt_stats.lock);
|
||||
return;
|
||||
}
|
||||
stats_req->done = true;
|
||||
|
||||
stats_req->done = FIELD_GET(HTT_T2H_EXT_STATS_INFO1_DONE, msg->info1);
|
||||
if (stats_req->done)
|
||||
send_completion = true;
|
||||
|
||||
spin_unlock_bh(&ar->debug.htt_stats.lock);
|
||||
|
||||
len = FIELD_GET(HTT_T2H_EXT_STATS_INFO1_LENGTH, msg->info1);
|
||||
@ -4344,7 +4345,8 @@ void ath11k_dbg_htt_ext_stats_handler(struct ath11k_base *ab,
|
||||
if (ret)
|
||||
ath11k_warn(ab, "Failed to parse tlv %d\n", ret);
|
||||
|
||||
complete(&stats_req->cmpln);
|
||||
if (send_completion)
|
||||
complete(&stats_req->cmpln);
|
||||
}
|
||||
|
||||
static ssize_t ath11k_read_htt_stats_type(struct file *file,
|
||||
@ -4497,28 +4499,54 @@ static int ath11k_open_htt_stats(struct inode *inode, struct file *file)
|
||||
if (type == ATH11K_DBG_HTT_EXT_STATS_RESET)
|
||||
return -EPERM;
|
||||
|
||||
stats_req = vzalloc(sizeof(*stats_req) + ATH11K_HTT_STATS_BUF_SIZE);
|
||||
if (!stats_req)
|
||||
return -ENOMEM;
|
||||
|
||||
mutex_lock(&ar->conf_mutex);
|
||||
|
||||
if (ar->state != ATH11K_STATE_ON) {
|
||||
ret = -ENETDOWN;
|
||||
goto err_unlock;
|
||||
}
|
||||
|
||||
if (ar->debug.htt_stats.stats_req) {
|
||||
ret = -EAGAIN;
|
||||
goto err_unlock;
|
||||
}
|
||||
|
||||
stats_req = vzalloc(sizeof(*stats_req) + ATH11K_HTT_STATS_BUF_SIZE);
|
||||
if (!stats_req) {
|
||||
ret = -ENOMEM;
|
||||
goto err_unlock;
|
||||
}
|
||||
|
||||
ar->debug.htt_stats.stats_req = stats_req;
|
||||
stats_req->type = type;
|
||||
|
||||
ret = ath11k_dbg_htt_stats_req(ar);
|
||||
mutex_unlock(&ar->conf_mutex);
|
||||
if (ret < 0)
|
||||
goto out;
|
||||
|
||||
file->private_data = stats_req;
|
||||
|
||||
mutex_unlock(&ar->conf_mutex);
|
||||
|
||||
return 0;
|
||||
out:
|
||||
vfree(stats_req);
|
||||
ar->debug.htt_stats.stats_req = NULL;
|
||||
err_unlock:
|
||||
mutex_unlock(&ar->conf_mutex);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int ath11k_release_htt_stats(struct inode *inode, struct file *file)
|
||||
{
|
||||
struct ath11k *ar = inode->i_private;
|
||||
|
||||
mutex_lock(&ar->conf_mutex);
|
||||
vfree(file->private_data);
|
||||
ar->debug.htt_stats.stats_req = NULL;
|
||||
mutex_unlock(&ar->conf_mutex);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -701,6 +701,7 @@ int ath11k_dp_service_srng(struct ath11k_base *ab,
|
||||
done:
|
||||
return tot_work_done;
|
||||
}
|
||||
EXPORT_SYMBOL(ath11k_dp_service_srng);
|
||||
|
||||
void ath11k_dp_pdev_free(struct ath11k_base *ab)
|
||||
{
|
||||
|
@ -1517,6 +1517,7 @@ struct htt_ext_stats_cfg_params {
|
||||
* 4 bytes.
|
||||
*/
|
||||
|
||||
#define HTT_T2H_EXT_STATS_INFO1_DONE BIT(11)
|
||||
#define HTT_T2H_EXT_STATS_INFO1_LENGTH GENMASK(31, 16)
|
||||
|
||||
struct ath11k_htt_extd_stats_msg {
|
||||
|
@ -2728,7 +2728,7 @@ static int ath11k_dp_rx_reap_mon_status_ring(struct ath11k_base *ab, int mac_id,
|
||||
ath11k_warn(ab, "rx monitor status with invalid buf_id %d\n",
|
||||
buf_id);
|
||||
spin_unlock_bh(&rx_ring->idr_lock);
|
||||
continue;
|
||||
goto move_next;
|
||||
}
|
||||
|
||||
idr_remove(&rx_ring->bufs_idr, buf_id);
|
||||
@ -2747,13 +2747,16 @@ static int ath11k_dp_rx_reap_mon_status_ring(struct ath11k_base *ab, int mac_id,
|
||||
tlv = (struct hal_tlv_hdr *)skb->data;
|
||||
if (FIELD_GET(HAL_TLV_HDR_TAG, tlv->tl) !=
|
||||
HAL_RX_STATUS_BUFFER_DONE) {
|
||||
ath11k_hal_srng_src_get_next_entry(ab, srng);
|
||||
continue;
|
||||
ath11k_warn(ab, "mon status DONE not set %lx\n",
|
||||
FIELD_GET(HAL_TLV_HDR_TAG,
|
||||
tlv->tl));
|
||||
dev_kfree_skb_any(skb);
|
||||
goto move_next;
|
||||
}
|
||||
|
||||
__skb_queue_tail(skb_list, skb);
|
||||
}
|
||||
|
||||
move_next:
|
||||
skb = ath11k_dp_rx_alloc_mon_status_buf(ab, rx_ring,
|
||||
&buf_id, GFP_ATOMIC);
|
||||
|
||||
|
@ -3,10 +3,10 @@
|
||||
* Copyright (c) 2018-2019 The Linux Foundation. All rights reserved.
|
||||
*/
|
||||
#include <linux/dma-mapping.h>
|
||||
#include "ahb.h"
|
||||
#include "hal_tx.h"
|
||||
#include "debug.h"
|
||||
#include "hal_desc.h"
|
||||
#include "hif.h"
|
||||
|
||||
static const struct hal_srng_config hw_srng_config[] = {
|
||||
/* TODO: max_rings can populated by querying HW capabilities */
|
||||
@ -351,11 +351,12 @@ static void ath11k_hal_ce_dst_setup(struct ath11k_base *ab,
|
||||
addr = HAL_CE_DST_RING_CTRL +
|
||||
srng_config->reg_start[HAL_SRNG_REG_GRP_R0] +
|
||||
ring_num * srng_config->reg_size[HAL_SRNG_REG_GRP_R0];
|
||||
val = ath11k_ahb_read32(ab, addr);
|
||||
|
||||
val = ath11k_hif_read32(ab, addr);
|
||||
val &= ~HAL_CE_DST_R0_DEST_CTRL_MAX_LEN;
|
||||
val |= FIELD_PREP(HAL_CE_DST_R0_DEST_CTRL_MAX_LEN,
|
||||
srng->u.dst_ring.max_buffer_length);
|
||||
ath11k_ahb_write32(ab, addr, val);
|
||||
ath11k_hif_write32(ab, addr, val);
|
||||
}
|
||||
|
||||
static void ath11k_hal_srng_dst_hw_init(struct ath11k_base *ab,
|
||||
@ -369,34 +370,34 @@ static void ath11k_hal_srng_dst_hw_init(struct ath11k_base *ab,
|
||||
reg_base = srng->hwreg_base[HAL_SRNG_REG_GRP_R0];
|
||||
|
||||
if (srng->flags & HAL_SRNG_FLAGS_MSI_INTR) {
|
||||
ath11k_ahb_write32(ab, reg_base +
|
||||
HAL_REO1_RING_MSI1_BASE_LSB_OFFSET,
|
||||
ath11k_hif_write32(ab, reg_base +
|
||||
HAL_REO1_RING_MSI1_BASE_LSB_OFFSET,
|
||||
(u32)srng->msi_addr);
|
||||
|
||||
val = FIELD_PREP(HAL_REO1_RING_MSI1_BASE_MSB_ADDR,
|
||||
((u64)srng->msi_addr >>
|
||||
HAL_ADDR_MSB_REG_SHIFT)) |
|
||||
HAL_REO1_RING_MSI1_BASE_MSB_MSI1_ENABLE;
|
||||
ath11k_ahb_write32(ab, reg_base +
|
||||
ath11k_hif_write32(ab, reg_base +
|
||||
HAL_REO1_RING_MSI1_BASE_MSB_OFFSET, val);
|
||||
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
reg_base + HAL_REO1_RING_MSI1_DATA_OFFSET,
|
||||
srng->msi_data);
|
||||
}
|
||||
|
||||
ath11k_ahb_write32(ab, reg_base, (u32)srng->ring_base_paddr);
|
||||
ath11k_hif_write32(ab, reg_base, (u32)srng->ring_base_paddr);
|
||||
|
||||
val = FIELD_PREP(HAL_REO1_RING_BASE_MSB_RING_BASE_ADDR_MSB,
|
||||
((u64)srng->ring_base_paddr >>
|
||||
HAL_ADDR_MSB_REG_SHIFT)) |
|
||||
FIELD_PREP(HAL_REO1_RING_BASE_MSB_RING_SIZE,
|
||||
(srng->entry_size * srng->num_entries));
|
||||
ath11k_ahb_write32(ab, reg_base + HAL_REO1_RING_BASE_MSB_OFFSET, val);
|
||||
ath11k_hif_write32(ab, reg_base + HAL_REO1_RING_BASE_MSB_OFFSET, val);
|
||||
|
||||
val = FIELD_PREP(HAL_REO1_RING_ID_RING_ID, srng->ring_id) |
|
||||
FIELD_PREP(HAL_REO1_RING_ID_ENTRY_SIZE, srng->entry_size);
|
||||
ath11k_ahb_write32(ab, reg_base + HAL_REO1_RING_ID_OFFSET, val);
|
||||
ath11k_hif_write32(ab, reg_base + HAL_REO1_RING_ID_OFFSET, val);
|
||||
|
||||
/* interrupt setup */
|
||||
val = FIELD_PREP(HAL_REO1_RING_PRDR_INT_SETUP_INTR_TMR_THOLD,
|
||||
@ -406,22 +407,22 @@ static void ath11k_hal_srng_dst_hw_init(struct ath11k_base *ab,
|
||||
(srng->intr_batch_cntr_thres_entries *
|
||||
srng->entry_size));
|
||||
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
reg_base + HAL_REO1_RING_PRODUCER_INT_SETUP_OFFSET,
|
||||
val);
|
||||
|
||||
hp_addr = hal->rdp.paddr +
|
||||
((unsigned long)srng->u.dst_ring.hp_addr -
|
||||
(unsigned long)hal->rdp.vaddr);
|
||||
ath11k_ahb_write32(ab, reg_base + HAL_REO1_RING_HP_ADDR_LSB_OFFSET,
|
||||
ath11k_hif_write32(ab, reg_base + HAL_REO1_RING_HP_ADDR_LSB_OFFSET,
|
||||
hp_addr & HAL_ADDR_LSB_REG_MASK);
|
||||
ath11k_ahb_write32(ab, reg_base + HAL_REO1_RING_HP_ADDR_MSB_OFFSET,
|
||||
ath11k_hif_write32(ab, reg_base + HAL_REO1_RING_HP_ADDR_MSB_OFFSET,
|
||||
hp_addr >> HAL_ADDR_MSB_REG_SHIFT);
|
||||
|
||||
/* Initialize head and tail pointers to indicate ring is empty */
|
||||
reg_base = srng->hwreg_base[HAL_SRNG_REG_GRP_R2];
|
||||
ath11k_ahb_write32(ab, reg_base, 0);
|
||||
ath11k_ahb_write32(ab, reg_base + HAL_REO1_RING_TP_OFFSET, 0);
|
||||
ath11k_hif_write32(ab, reg_base, 0);
|
||||
ath11k_hif_write32(ab, reg_base + HAL_REO1_RING_TP_OFFSET, 0);
|
||||
*srng->u.dst_ring.hp_addr = 0;
|
||||
|
||||
reg_base = srng->hwreg_base[HAL_SRNG_REG_GRP_R0];
|
||||
@ -434,7 +435,7 @@ static void ath11k_hal_srng_dst_hw_init(struct ath11k_base *ab,
|
||||
val |= HAL_REO1_RING_MISC_MSI_SWAP;
|
||||
val |= HAL_REO1_RING_MISC_SRNG_ENABLE;
|
||||
|
||||
ath11k_ahb_write32(ab, reg_base + HAL_REO1_RING_MISC_OFFSET, val);
|
||||
ath11k_hif_write32(ab, reg_base + HAL_REO1_RING_MISC_OFFSET, val);
|
||||
}
|
||||
|
||||
static void ath11k_hal_srng_src_hw_init(struct ath11k_base *ab,
|
||||
@ -448,34 +449,34 @@ static void ath11k_hal_srng_src_hw_init(struct ath11k_base *ab,
|
||||
reg_base = srng->hwreg_base[HAL_SRNG_REG_GRP_R0];
|
||||
|
||||
if (srng->flags & HAL_SRNG_FLAGS_MSI_INTR) {
|
||||
ath11k_ahb_write32(ab, reg_base +
|
||||
HAL_TCL1_RING_MSI1_BASE_LSB_OFFSET,
|
||||
ath11k_hif_write32(ab, reg_base +
|
||||
HAL_TCL1_RING_MSI1_BASE_LSB_OFFSET,
|
||||
(u32)srng->msi_addr);
|
||||
|
||||
val = FIELD_PREP(HAL_TCL1_RING_MSI1_BASE_MSB_ADDR,
|
||||
((u64)srng->msi_addr >>
|
||||
HAL_ADDR_MSB_REG_SHIFT)) |
|
||||
HAL_TCL1_RING_MSI1_BASE_MSB_MSI1_ENABLE;
|
||||
ath11k_ahb_write32(ab, reg_base +
|
||||
ath11k_hif_write32(ab, reg_base +
|
||||
HAL_TCL1_RING_MSI1_BASE_MSB_OFFSET,
|
||||
val);
|
||||
|
||||
ath11k_ahb_write32(ab, reg_base +
|
||||
ath11k_hif_write32(ab, reg_base +
|
||||
HAL_TCL1_RING_MSI1_DATA_OFFSET,
|
||||
srng->msi_data);
|
||||
}
|
||||
|
||||
ath11k_ahb_write32(ab, reg_base, (u32)srng->ring_base_paddr);
|
||||
ath11k_hif_write32(ab, reg_base, (u32)srng->ring_base_paddr);
|
||||
|
||||
val = FIELD_PREP(HAL_TCL1_RING_BASE_MSB_RING_BASE_ADDR_MSB,
|
||||
((u64)srng->ring_base_paddr >>
|
||||
HAL_ADDR_MSB_REG_SHIFT)) |
|
||||
FIELD_PREP(HAL_TCL1_RING_BASE_MSB_RING_SIZE,
|
||||
(srng->entry_size * srng->num_entries));
|
||||
ath11k_ahb_write32(ab, reg_base + HAL_TCL1_RING_BASE_MSB_OFFSET, val);
|
||||
ath11k_hif_write32(ab, reg_base + HAL_TCL1_RING_BASE_MSB_OFFSET, val);
|
||||
|
||||
val = FIELD_PREP(HAL_REO1_RING_ID_ENTRY_SIZE, srng->entry_size);
|
||||
ath11k_ahb_write32(ab, reg_base + HAL_TCL1_RING_ID_OFFSET, val);
|
||||
ath11k_hif_write32(ab, reg_base + HAL_TCL1_RING_ID_OFFSET, val);
|
||||
|
||||
/* interrupt setup */
|
||||
/* NOTE: IPQ8074 v2 requires the interrupt timer threshold in the
|
||||
@ -488,7 +489,7 @@ static void ath11k_hal_srng_src_hw_init(struct ath11k_base *ab,
|
||||
(srng->intr_batch_cntr_thres_entries *
|
||||
srng->entry_size));
|
||||
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
reg_base + HAL_TCL1_RING_CONSR_INT_SETUP_IX0_OFFSET,
|
||||
val);
|
||||
|
||||
@ -497,7 +498,7 @@ static void ath11k_hal_srng_src_hw_init(struct ath11k_base *ab,
|
||||
val |= FIELD_PREP(HAL_TCL1_RING_CONSR_INT_SETUP_IX1_LOW_THOLD,
|
||||
srng->u.src_ring.low_threshold);
|
||||
}
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
reg_base + HAL_TCL1_RING_CONSR_INT_SETUP_IX1_OFFSET,
|
||||
val);
|
||||
|
||||
@ -505,18 +506,18 @@ static void ath11k_hal_srng_src_hw_init(struct ath11k_base *ab,
|
||||
tp_addr = hal->rdp.paddr +
|
||||
((unsigned long)srng->u.src_ring.tp_addr -
|
||||
(unsigned long)hal->rdp.vaddr);
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
reg_base + HAL_TCL1_RING_TP_ADDR_LSB_OFFSET,
|
||||
tp_addr & HAL_ADDR_LSB_REG_MASK);
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
reg_base + HAL_TCL1_RING_TP_ADDR_MSB_OFFSET,
|
||||
tp_addr >> HAL_ADDR_MSB_REG_SHIFT);
|
||||
}
|
||||
|
||||
/* Initialize head and tail pointers to indicate ring is empty */
|
||||
reg_base = srng->hwreg_base[HAL_SRNG_REG_GRP_R2];
|
||||
ath11k_ahb_write32(ab, reg_base, 0);
|
||||
ath11k_ahb_write32(ab, reg_base + HAL_TCL1_RING_TP_OFFSET, 0);
|
||||
ath11k_hif_write32(ab, reg_base, 0);
|
||||
ath11k_hif_write32(ab, reg_base + HAL_TCL1_RING_TP_OFFSET, 0);
|
||||
*srng->u.src_ring.tp_addr = 0;
|
||||
|
||||
reg_base = srng->hwreg_base[HAL_SRNG_REG_GRP_R0];
|
||||
@ -533,7 +534,7 @@ static void ath11k_hal_srng_src_hw_init(struct ath11k_base *ab,
|
||||
|
||||
val |= HAL_TCL1_RING_MISC_SRNG_ENABLE;
|
||||
|
||||
ath11k_ahb_write32(ab, reg_base + HAL_TCL1_RING_MISC_OFFSET, val);
|
||||
ath11k_hif_write32(ab, reg_base + HAL_TCL1_RING_MISC_OFFSET, val);
|
||||
}
|
||||
|
||||
static void ath11k_hal_srng_hw_init(struct ath11k_base *ab,
|
||||
@ -889,13 +890,13 @@ void ath11k_hal_srng_access_end(struct ath11k_base *ab, struct hal_srng *srng)
|
||||
if (srng->ring_dir == HAL_SRNG_DIR_SRC) {
|
||||
srng->u.src_ring.last_tp =
|
||||
*(volatile u32 *)srng->u.src_ring.tp_addr;
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
(unsigned long)srng->u.src_ring.hp_addr -
|
||||
(unsigned long)ab->mem,
|
||||
srng->u.src_ring.hp);
|
||||
} else {
|
||||
srng->u.dst_ring.last_hp = *srng->u.dst_ring.hp_addr;
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
(unsigned long)srng->u.dst_ring.tp_addr -
|
||||
(unsigned long)ab->mem,
|
||||
srng->u.dst_ring.tp);
|
||||
@ -929,20 +930,20 @@ void ath11k_hal_setup_link_idle_list(struct ath11k_base *ab,
|
||||
HAL_WBM_IDLE_SCATTER_BUF_SIZE;
|
||||
}
|
||||
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG + HAL_WBM_R0_IDLE_LIST_CONTROL_ADDR,
|
||||
FIELD_PREP(HAL_WBM_SCATTER_BUFFER_SIZE, reg_scatter_buf_sz) |
|
||||
FIELD_PREP(HAL_WBM_LINK_DESC_IDLE_LIST_MODE, 0x1));
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG + HAL_WBM_R0_IDLE_LIST_SIZE_ADDR,
|
||||
FIELD_PREP(HAL_WBM_SCATTER_RING_SIZE_OF_IDLE_LINK_DESC_LIST,
|
||||
reg_scatter_buf_sz * nsbufs));
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG +
|
||||
HAL_WBM_SCATTERED_RING_BASE_LSB,
|
||||
FIELD_PREP(BUFFER_ADDR_INFO0_ADDR,
|
||||
sbuf[0].paddr & HAL_ADDR_LSB_REG_MASK));
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG +
|
||||
HAL_WBM_SCATTERED_RING_BASE_MSB,
|
||||
FIELD_PREP(
|
||||
@ -953,12 +954,12 @@ void ath11k_hal_setup_link_idle_list(struct ath11k_base *ab,
|
||||
BASE_ADDR_MATCH_TAG_VAL));
|
||||
|
||||
/* Setup head and tail pointers for the idle list */
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG +
|
||||
HAL_WBM_SCATTERED_DESC_PTR_HEAD_INFO_IX0,
|
||||
FIELD_PREP(BUFFER_ADDR_INFO0_ADDR,
|
||||
sbuf[nsbufs - 1].paddr));
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG +
|
||||
HAL_WBM_SCATTERED_DESC_PTR_HEAD_INFO_IX1,
|
||||
FIELD_PREP(
|
||||
@ -967,18 +968,18 @@ void ath11k_hal_setup_link_idle_list(struct ath11k_base *ab,
|
||||
HAL_ADDR_MSB_REG_SHIFT)) |
|
||||
FIELD_PREP(HAL_WBM_SCATTERED_DESC_HEAD_P_OFFSET_IX1,
|
||||
(end_offset >> 2)));
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG +
|
||||
HAL_WBM_SCATTERED_DESC_PTR_HEAD_INFO_IX0,
|
||||
FIELD_PREP(BUFFER_ADDR_INFO0_ADDR,
|
||||
sbuf[0].paddr));
|
||||
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG +
|
||||
HAL_WBM_SCATTERED_DESC_PTR_TAIL_INFO_IX0,
|
||||
FIELD_PREP(BUFFER_ADDR_INFO0_ADDR,
|
||||
sbuf[0].paddr));
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG +
|
||||
HAL_WBM_SCATTERED_DESC_PTR_TAIL_INFO_IX1,
|
||||
FIELD_PREP(
|
||||
@ -986,13 +987,13 @@ void ath11k_hal_setup_link_idle_list(struct ath11k_base *ab,
|
||||
((u64)sbuf[0].paddr >> HAL_ADDR_MSB_REG_SHIFT)) |
|
||||
FIELD_PREP(HAL_WBM_SCATTERED_DESC_TAIL_P_OFFSET_IX1,
|
||||
0));
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG +
|
||||
HAL_WBM_SCATTERED_DESC_PTR_HP_ADDR,
|
||||
2 * tot_link_desc);
|
||||
|
||||
/* Enable the SRNG */
|
||||
ath11k_ahb_write32(ab,
|
||||
ath11k_hif_write32(ab,
|
||||
HAL_SEQ_WCSS_UMAC_WBM_REG +
|
||||
HAL_WBM_IDLE_LINK_RING_MISC_ADDR, 0x40);
|
||||
}
|
||||
|
@ -2,6 +2,8 @@
|
||||
/*
|
||||
* Copyright (c) 2018-2019 The Linux Foundation. All rights reserved.
|
||||
*/
|
||||
#include "core.h"
|
||||
|
||||
#ifndef ATH11K_HAL_DESC_H
|
||||
#define ATH11K_HAL_DESC_H
|
||||
|
||||
|
@ -3,12 +3,12 @@
|
||||
* Copyright (c) 2018-2019 The Linux Foundation. All rights reserved.
|
||||
*/
|
||||
|
||||
#include "ahb.h"
|
||||
#include "debug.h"
|
||||
#include "hal.h"
|
||||
#include "hal_tx.h"
|
||||
#include "hal_rx.h"
|
||||
#include "hal_desc.h"
|
||||
#include "hif.h"
|
||||
|
||||
static void ath11k_hal_reo_set_desc_hdr(struct hal_desc_header *hdr,
|
||||
u8 owner, u8 buffer_type, u32 magic)
|
||||
@ -804,34 +804,34 @@ void ath11k_hal_reo_hw_setup(struct ath11k_base *ab, u32 ring_hash_map)
|
||||
u32 reo_base = HAL_SEQ_WCSS_UMAC_REO_REG;
|
||||
u32 val;
|
||||
|
||||
val = ath11k_ahb_read32(ab, reo_base + HAL_REO1_GEN_ENABLE);
|
||||
val = ath11k_hif_read32(ab, reo_base + HAL_REO1_GEN_ENABLE);
|
||||
|
||||
val &= ~HAL_REO1_GEN_ENABLE_FRAG_DST_RING;
|
||||
val |= FIELD_PREP(HAL_REO1_GEN_ENABLE_FRAG_DST_RING,
|
||||
HAL_SRNG_RING_ID_REO2SW1) |
|
||||
FIELD_PREP(HAL_REO1_GEN_ENABLE_AGING_LIST_ENABLE, 1) |
|
||||
FIELD_PREP(HAL_REO1_GEN_ENABLE_AGING_FLUSH_ENABLE, 1);
|
||||
ath11k_ahb_write32(ab, reo_base + HAL_REO1_GEN_ENABLE, val);
|
||||
ath11k_hif_write32(ab, reo_base + HAL_REO1_GEN_ENABLE, val);
|
||||
|
||||
ath11k_ahb_write32(ab, reo_base + HAL_REO1_AGING_THRESH_IX_0,
|
||||
ath11k_hif_write32(ab, reo_base + HAL_REO1_AGING_THRESH_IX_0,
|
||||
HAL_DEFAULT_REO_TIMEOUT_USEC);
|
||||
ath11k_ahb_write32(ab, reo_base + HAL_REO1_AGING_THRESH_IX_1,
|
||||
ath11k_hif_write32(ab, reo_base + HAL_REO1_AGING_THRESH_IX_1,
|
||||
HAL_DEFAULT_REO_TIMEOUT_USEC);
|
||||
ath11k_ahb_write32(ab, reo_base + HAL_REO1_AGING_THRESH_IX_2,
|
||||
ath11k_hif_write32(ab, reo_base + HAL_REO1_AGING_THRESH_IX_2,
|
||||
HAL_DEFAULT_REO_TIMEOUT_USEC);
|
||||
ath11k_ahb_write32(ab, reo_base + HAL_REO1_AGING_THRESH_IX_3,
|
||||
ath11k_hif_write32(ab, reo_base + HAL_REO1_AGING_THRESH_IX_3,
|
||||
HAL_DEFAULT_REO_TIMEOUT_USEC);
|
||||
|
||||
ath11k_ahb_write32(ab, reo_base + HAL_REO1_DEST_RING_CTRL_IX_0,
|
||||
ath11k_hif_write32(ab, reo_base + HAL_REO1_DEST_RING_CTRL_IX_0,
|
||||
FIELD_PREP(HAL_REO_DEST_RING_CTRL_HASH_RING_MAP,
|
||||
ring_hash_map));
|
||||
ath11k_ahb_write32(ab, reo_base + HAL_REO1_DEST_RING_CTRL_IX_1,
|
||||
ath11k_hif_write32(ab, reo_base + HAL_REO1_DEST_RING_CTRL_IX_1,
|
||||
FIELD_PREP(HAL_REO_DEST_RING_CTRL_HASH_RING_MAP,
|
||||
ring_hash_map));
|
||||
ath11k_ahb_write32(ab, reo_base + HAL_REO1_DEST_RING_CTRL_IX_2,
|
||||
ath11k_hif_write32(ab, reo_base + HAL_REO1_DEST_RING_CTRL_IX_2,
|
||||
FIELD_PREP(HAL_REO_DEST_RING_CTRL_HASH_RING_MAP,
|
||||
ring_hash_map));
|
||||
ath11k_ahb_write32(ab, reo_base + HAL_REO1_DEST_RING_CTRL_IX_3,
|
||||
ath11k_hif_write32(ab, reo_base + HAL_REO1_DEST_RING_CTRL_IX_3,
|
||||
FIELD_PREP(HAL_REO_DEST_RING_CTRL_HASH_RING_MAP,
|
||||
ring_hash_map));
|
||||
}
|
||||
|
@ -3,9 +3,10 @@
|
||||
* Copyright (c) 2018-2019 The Linux Foundation. All rights reserved.
|
||||
*/
|
||||
|
||||
#include "ahb.h"
|
||||
#include "hal_desc.h"
|
||||
#include "hal.h"
|
||||
#include "hal_tx.h"
|
||||
#include "hif.h"
|
||||
|
||||
#define DSCP_TID_MAP_TBL_ENTRY_SIZE 64
|
||||
|
||||
@ -83,11 +84,11 @@ void ath11k_hal_tx_set_dscp_tid_map(struct ath11k_base *ab, int id)
|
||||
u32 value;
|
||||
int cnt = 0;
|
||||
|
||||
ctrl_reg_val = ath11k_ahb_read32(ab, HAL_SEQ_WCSS_UMAC_TCL_REG +
|
||||
ctrl_reg_val = ath11k_hif_read32(ab, HAL_SEQ_WCSS_UMAC_TCL_REG +
|
||||
HAL_TCL1_RING_CMN_CTRL_REG);
|
||||
/* Enable read/write access */
|
||||
ctrl_reg_val |= HAL_TCL1_RING_CMN_CTRL_DSCP_TID_MAP_PROG_EN;
|
||||
ath11k_ahb_write32(ab, HAL_SEQ_WCSS_UMAC_TCL_REG +
|
||||
ath11k_hif_write32(ab, HAL_SEQ_WCSS_UMAC_TCL_REG +
|
||||
HAL_TCL1_RING_CMN_CTRL_REG, ctrl_reg_val);
|
||||
|
||||
addr = HAL_SEQ_WCSS_UMAC_TCL_REG + HAL_TCL1_RING_DSCP_TID_MAP +
|
||||
@ -118,15 +119,15 @@ void ath11k_hal_tx_set_dscp_tid_map(struct ath11k_base *ab, int id)
|
||||
}
|
||||
|
||||
for (i = 0; i < HAL_DSCP_TID_TBL_SIZE; i += 4) {
|
||||
ath11k_ahb_write32(ab, addr, *(u32 *)&hw_map_val[i]);
|
||||
ath11k_hif_write32(ab, addr, *(u32 *)&hw_map_val[i]);
|
||||
addr += 4;
|
||||
}
|
||||
|
||||
/* Disable read/write access */
|
||||
ctrl_reg_val = ath11k_ahb_read32(ab, HAL_SEQ_WCSS_UMAC_TCL_REG +
|
||||
ctrl_reg_val = ath11k_hif_read32(ab, HAL_SEQ_WCSS_UMAC_TCL_REG +
|
||||
HAL_TCL1_RING_CMN_CTRL_REG);
|
||||
ctrl_reg_val &= ~HAL_TCL1_RING_CMN_CTRL_DSCP_TID_MAP_PROG_EN;
|
||||
ath11k_ahb_write32(ab, HAL_SEQ_WCSS_UMAC_TCL_REG +
|
||||
ath11k_hif_write32(ab, HAL_SEQ_WCSS_UMAC_TCL_REG +
|
||||
HAL_TCL1_RING_CMN_CTRL_REG,
|
||||
ctrl_reg_val);
|
||||
}
|
||||
|
@ -7,6 +7,7 @@
|
||||
#define ATH11K_HAL_TX_H
|
||||
|
||||
#include "hal_desc.h"
|
||||
#include "core.h"
|
||||
|
||||
#define HAL_TX_ADDRX_EN 1
|
||||
#define HAL_TX_ADDRY_EN 2
|
||||
|
65
drivers/net/wireless/ath/ath11k/hif.h
Normal file
65
drivers/net/wireless/ath/ath11k/hif.h
Normal file
@ -0,0 +1,65 @@
|
||||
/* SPDX-License-Identifier: BSD-3-Clause-Clear */
|
||||
/*
|
||||
* Copyright (c) 2019-2020 The Linux Foundation. All rights reserved.
|
||||
*/
|
||||
|
||||
#include "core.h"
|
||||
|
||||
struct ath11k_hif_ops {
|
||||
u32 (*read32)(struct ath11k_base *sc, u32 address);
|
||||
void (*write32)(struct ath11k_base *sc, u32 address, u32 data);
|
||||
void (*irq_enable)(struct ath11k_base *sc);
|
||||
void (*irq_disable)(struct ath11k_base *sc);
|
||||
int (*start)(struct ath11k_base *sc);
|
||||
void (*stop)(struct ath11k_base *sc);
|
||||
int (*power_up)(struct ath11k_base *sc);
|
||||
void (*power_down)(struct ath11k_base *sc);
|
||||
int (*map_service_to_pipe)(struct ath11k_base *sc, u16 service_id,
|
||||
u8 *ul_pipe, u8 *dl_pipe);
|
||||
};
|
||||
|
||||
static inline int ath11k_hif_start(struct ath11k_base *sc)
|
||||
{
|
||||
return sc->hif.ops->start(sc);
|
||||
}
|
||||
|
||||
static inline void ath11k_hif_stop(struct ath11k_base *sc)
|
||||
{
|
||||
sc->hif.ops->stop(sc);
|
||||
}
|
||||
|
||||
static inline void ath11k_hif_irq_enable(struct ath11k_base *sc)
|
||||
{
|
||||
sc->hif.ops->irq_enable(sc);
|
||||
}
|
||||
|
||||
static inline void ath11k_hif_irq_disable(struct ath11k_base *sc)
|
||||
{
|
||||
sc->hif.ops->irq_disable(sc);
|
||||
}
|
||||
|
||||
static inline int ath11k_hif_power_up(struct ath11k_base *sc)
|
||||
{
|
||||
return sc->hif.ops->power_up(sc);
|
||||
}
|
||||
|
||||
static inline void ath11k_hif_power_down(struct ath11k_base *sc)
|
||||
{
|
||||
sc->hif.ops->power_down(sc);
|
||||
}
|
||||
|
||||
static inline u32 ath11k_hif_read32(struct ath11k_base *sc, u32 address)
|
||||
{
|
||||
return sc->hif.ops->read32(sc, address);
|
||||
}
|
||||
|
||||
static inline void ath11k_hif_write32(struct ath11k_base *sc, u32 address, u32 data)
|
||||
{
|
||||
sc->hif.ops->write32(sc, address, data);
|
||||
}
|
||||
|
||||
static inline int ath11k_hif_map_service_to_pipe(struct ath11k_base *sc, u16 service_id,
|
||||
u8 *ul_pipe, u8 *dl_pipe)
|
||||
{
|
||||
return sc->hif.ops->map_service_to_pipe(sc, service_id, ul_pipe, dl_pipe);
|
||||
}
|
@ -5,8 +5,8 @@
|
||||
#include <linux/skbuff.h>
|
||||
#include <linux/ctype.h>
|
||||
|
||||
#include "ahb.h"
|
||||
#include "debug.h"
|
||||
#include "hif.h"
|
||||
|
||||
struct sk_buff *ath11k_htc_alloc_skb(struct ath11k_base *ab, int size)
|
||||
{
|
||||
@ -672,7 +672,7 @@ int ath11k_htc_connect_service(struct ath11k_htc *htc,
|
||||
/* copy all the callbacks */
|
||||
ep->ep_ops = conn_req->ep_ops;
|
||||
|
||||
status = ath11k_ahb_map_service_to_pipe(htc->ab,
|
||||
status = ath11k_hif_map_service_to_pipe(htc->ab,
|
||||
ep->service_id,
|
||||
&ep->ul_pipe_id,
|
||||
&ep->dl_pipe_id);
|
||||
|
@ -99,6 +99,11 @@ enum ath11k_hw_rate_ofdm {
|
||||
ATH11K_HW_RATE_OFDM_9M,
|
||||
};
|
||||
|
||||
enum ath11k_bus {
|
||||
ATH11K_BUS_AHB,
|
||||
ATH11K_BUS_PCI,
|
||||
};
|
||||
|
||||
struct ath11k_hw_params {
|
||||
const char *name;
|
||||
struct {
|
||||
|
@ -3562,7 +3562,7 @@ static int ath11k_mac_copy_he_cap(struct ath11k *ar,
|
||||
memcpy(he_cap_elem->phy_cap_info, band_cap->he_cap_phy_info,
|
||||
sizeof(he_cap_elem->phy_cap_info));
|
||||
|
||||
he_cap_elem->mac_cap_info[1] |=
|
||||
he_cap_elem->mac_cap_info[1] &=
|
||||
IEEE80211_HE_MAC_CAP1_TF_MAC_PAD_DUR_MASK;
|
||||
he_cap_elem->phy_cap_info[4] &=
|
||||
~IEEE80211_HE_PHY_CAP4_BEAMFORMEE_MAX_STS_UNDER_80MHZ_MASK;
|
||||
@ -3578,6 +3578,8 @@ static int ath11k_mac_copy_he_cap(struct ath11k *ar,
|
||||
|
||||
switch (i) {
|
||||
case NL80211_IFTYPE_AP:
|
||||
he_cap_elem->phy_cap_info[3] &=
|
||||
~IEEE80211_HE_PHY_CAP3_DCM_MAX_CONST_TX_MASK;
|
||||
he_cap_elem->phy_cap_info[9] |=
|
||||
IEEE80211_HE_PHY_CAP9_RX_1024_QAM_LESS_THAN_242_TONE_RU;
|
||||
break;
|
||||
@ -3692,7 +3694,7 @@ static int __ath11k_set_antenna(struct ath11k *ar, u32 tx_ant, u32 rx_ant)
|
||||
int ath11k_mac_tx_mgmt_pending_free(int buf_id, void *skb, void *ctx)
|
||||
{
|
||||
struct sk_buff *msdu = skb;
|
||||
struct ieee80211_tx_info *info = IEEE80211_SKB_CB(msdu);
|
||||
struct ieee80211_tx_info *info;
|
||||
struct ath11k *ar = ctx;
|
||||
struct ath11k_base *ab = ar->ab;
|
||||
|
||||
|
@ -174,9 +174,12 @@ int ath11k_thermal_register(struct ath11k_base *sc)
|
||||
if (IS_ERR(cdev)) {
|
||||
ath11k_err(sc, "failed to setup thermal device result: %ld\n",
|
||||
PTR_ERR(cdev));
|
||||
return -EINVAL;
|
||||
ret = -EINVAL;
|
||||
goto err_thermal_destroy;
|
||||
}
|
||||
|
||||
ar->thermal.cdev = cdev;
|
||||
|
||||
ret = sysfs_create_link(&ar->hw->wiphy->dev.kobj, &cdev->device.kobj,
|
||||
"cooling_device");
|
||||
if (ret) {
|
||||
@ -184,7 +187,6 @@ int ath11k_thermal_register(struct ath11k_base *sc)
|
||||
goto err_thermal_destroy;
|
||||
}
|
||||
|
||||
ar->thermal.cdev = cdev;
|
||||
if (!IS_REACHABLE(CONFIG_HWMON))
|
||||
return 0;
|
||||
|
||||
|
@ -780,6 +780,8 @@ static void ath9k_set_hw_capab(struct ath9k_htc_priv *priv,
|
||||
SET_IEEE80211_PERM_ADDR(hw, common->macaddr);
|
||||
|
||||
wiphy_ext_feature_set(hw->wiphy, NL80211_EXT_FEATURE_CQM_RSSI_LIST);
|
||||
wiphy_ext_feature_set(hw->wiphy,
|
||||
NL80211_EXT_FEATURE_MULTICAST_REGISTRATIONS);
|
||||
}
|
||||
|
||||
static int ath9k_init_firmware_version(struct ath9k_htc_priv *priv)
|
||||
|
@ -1251,6 +1251,7 @@ static int ath9k_htc_config(struct ieee80211_hw *hw, u32 changed)
|
||||
FIF_OTHER_BSS | \
|
||||
FIF_BCN_PRBRESP_PROMISC | \
|
||||
FIF_PROBE_REQ | \
|
||||
FIF_MCAST_ACTION | \
|
||||
FIF_FCSFAIL)
|
||||
|
||||
static void ath9k_htc_configure_filter(struct ieee80211_hw *hw,
|
||||
|
@ -893,7 +893,8 @@ u32 ath9k_htc_calcrxfilter(struct ath9k_htc_priv *priv)
|
||||
if (priv->rxfilter & FIF_PSPOLL)
|
||||
rfilt |= ATH9K_RX_FILTER_PSPOLL;
|
||||
|
||||
if (priv->nvifs > 1 || priv->rxfilter & FIF_OTHER_BSS)
|
||||
if (priv->nvifs > 1 ||
|
||||
priv->rxfilter & (FIF_OTHER_BSS | FIF_MCAST_ACTION))
|
||||
rfilt |= ATH9K_RX_FILTER_MCAST_BCAST_ALL;
|
||||
|
||||
return rfilt;
|
||||
|
@ -1012,6 +1012,8 @@ static void ath9k_set_hw_capab(struct ath_softc *sc, struct ieee80211_hw *hw)
|
||||
|
||||
wiphy_ext_feature_set(hw->wiphy, NL80211_EXT_FEATURE_CQM_RSSI_LIST);
|
||||
wiphy_ext_feature_set(hw->wiphy, NL80211_EXT_FEATURE_AIRTIME_FAIRNESS);
|
||||
wiphy_ext_feature_set(hw->wiphy,
|
||||
NL80211_EXT_FEATURE_MULTICAST_REGISTRATIONS);
|
||||
}
|
||||
|
||||
int ath9k_init_device(u16 devid, struct ath_softc *sc,
|
||||
|
@ -1476,6 +1476,7 @@ static int ath9k_config(struct ieee80211_hw *hw, u32 changed)
|
||||
FIF_OTHER_BSS | \
|
||||
FIF_BCN_PRBRESP_PROMISC | \
|
||||
FIF_PROBE_REQ | \
|
||||
FIF_MCAST_ACTION | \
|
||||
FIF_FCSFAIL)
|
||||
|
||||
/* FIXME: sc->sc_full_reset ? */
|
||||
|
@ -413,7 +413,8 @@ u32 ath_calcrxfilter(struct ath_softc *sc)
|
||||
if (sc->cur_chandef.width != NL80211_CHAN_WIDTH_20_NOHT)
|
||||
rfilt |= ATH9K_RX_FILTER_COMP_BAR;
|
||||
|
||||
if (sc->cur_chan->nvifs > 1 || (sc->cur_chan->rxfilter & FIF_OTHER_BSS)) {
|
||||
if (sc->cur_chan->nvifs > 1 ||
|
||||
(sc->cur_chan->rxfilter & (FIF_OTHER_BSS | FIF_MCAST_ACTION))) {
|
||||
/* This is needed for older chips */
|
||||
if (sc->sc_ah->hw_version.macVersion <= AR_SREV_VERSION_9160)
|
||||
rfilt |= ATH9K_RX_FILTER_PROM;
|
||||
|
@ -127,7 +127,7 @@ struct carl9170_write_reg {
|
||||
struct carl9170_write_reg_byte {
|
||||
__le32 addr;
|
||||
__le32 count;
|
||||
u8 val[0];
|
||||
u8 val[];
|
||||
} __packed;
|
||||
|
||||
#define CARL9170FW_PHY_HT_ENABLE 0x4
|
||||
|
@ -851,7 +851,7 @@ struct ar9170_stream {
|
||||
__le16 length;
|
||||
__le16 tag;
|
||||
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed __aligned(4);
|
||||
#define AR9170_STREAM_LEN 4
|
||||
|
||||
|
@ -2240,7 +2240,7 @@ struct wcn36xx_hal_process_ptt_msg_req_msg {
|
||||
struct wcn36xx_hal_msg_header header;
|
||||
|
||||
/* Actual FTM Command body */
|
||||
u8 ptt_msg[0];
|
||||
u8 ptt_msg[];
|
||||
} __packed;
|
||||
|
||||
struct wcn36xx_hal_process_ptt_msg_rsp_msg {
|
||||
@ -2249,7 +2249,7 @@ struct wcn36xx_hal_process_ptt_msg_rsp_msg {
|
||||
/* FTM Command response status */
|
||||
u32 ptt_msg_resp_status;
|
||||
/* Actual FTM Command body */
|
||||
u8 ptt_msg[0];
|
||||
u8 ptt_msg[];
|
||||
} __packed;
|
||||
|
||||
struct update_edca_params_req_msg {
|
||||
|
@ -1339,7 +1339,7 @@ static int wcn36xx_probe(struct platform_device *pdev)
|
||||
if (addr && ret != ETH_ALEN) {
|
||||
wcn36xx_err("invalid local-mac-address\n");
|
||||
ret = -EINVAL;
|
||||
goto out_wq;
|
||||
goto out_destroy_ept;
|
||||
} else if (addr) {
|
||||
wcn36xx_info("mac address: %pM\n", addr);
|
||||
SET_IEEE80211_PERM_ADDR(wcn->hw, addr);
|
||||
@ -1347,7 +1347,7 @@ static int wcn36xx_probe(struct platform_device *pdev)
|
||||
|
||||
ret = wcn36xx_platform_get_resources(wcn, pdev);
|
||||
if (ret)
|
||||
goto out_wq;
|
||||
goto out_destroy_ept;
|
||||
|
||||
wcn36xx_init_ieee80211(wcn);
|
||||
ret = ieee80211_register_hw(wcn->hw);
|
||||
@ -1359,6 +1359,8 @@ static int wcn36xx_probe(struct platform_device *pdev)
|
||||
out_unmap:
|
||||
iounmap(wcn->ccu_base);
|
||||
iounmap(wcn->dxe_base);
|
||||
out_destroy_ept:
|
||||
rpmsg_destroy_ept(wcn->smd_channel);
|
||||
out_wq:
|
||||
ieee80211_free_hw(hw);
|
||||
out_err:
|
||||
|
@ -20,7 +20,7 @@ struct ftm_rsp_msg {
|
||||
u16 msg_id;
|
||||
u16 msg_body_length;
|
||||
u32 resp_status;
|
||||
u8 msg_response[0];
|
||||
u8 msg_response[];
|
||||
} __packed;
|
||||
|
||||
/* The request buffer of FTM which contains a byte of command and the request */
|
||||
|
@ -33,7 +33,7 @@ struct wil_fw_record_head {
|
||||
*/
|
||||
struct wil_fw_record_data { /* type == wil_fw_type_data */
|
||||
__le32 addr;
|
||||
__le32 data[0]; /* [data_size], see above */
|
||||
__le32 data[]; /* [data_size], see above */
|
||||
} __packed;
|
||||
|
||||
/* fill with constant @value, @size bytes starting from @addr */
|
||||
@ -61,7 +61,7 @@ struct wil_fw_record_capabilities { /* type == wil_fw_type_comment */
|
||||
/* identifies capabilities record */
|
||||
struct wil_fw_record_comment_hdr hdr;
|
||||
/* capabilities (variable size), see enum wmi_fw_capability */
|
||||
u8 capabilities[0];
|
||||
u8 capabilities[];
|
||||
} __packed;
|
||||
|
||||
/* FW VIF concurrency encoded inside a comment record
|
||||
@ -80,7 +80,7 @@ struct wil_fw_concurrency_combo {
|
||||
u8 n_diff_channels; /* total number of different channels allowed */
|
||||
u8 same_bi; /* for APs, 1 if all APs must have same BI */
|
||||
/* keep last - concurrency limits, variable size by n_limits */
|
||||
struct wil_fw_concurrency_limit limits[0];
|
||||
struct wil_fw_concurrency_limit limits[];
|
||||
} __packed;
|
||||
|
||||
struct wil_fw_record_concurrency { /* type == wil_fw_type_comment */
|
||||
@ -93,7 +93,7 @@ struct wil_fw_record_concurrency { /* type == wil_fw_type_comment */
|
||||
/* number of concurrency combinations that follow */
|
||||
__le16 n_combos;
|
||||
/* keep last - combinations, variable size by n_combos */
|
||||
struct wil_fw_concurrency_combo combos[0];
|
||||
struct wil_fw_concurrency_combo combos[];
|
||||
} __packed;
|
||||
|
||||
/* brd file info encoded inside a comment record */
|
||||
@ -108,7 +108,7 @@ struct wil_fw_record_brd_file { /* type == wil_fw_type_comment */
|
||||
/* identifies brd file record */
|
||||
struct wil_fw_record_comment_hdr hdr;
|
||||
__le32 version;
|
||||
struct brd_info brd_info[0];
|
||||
struct brd_info brd_info[];
|
||||
} __packed;
|
||||
|
||||
/* perform action
|
||||
@ -116,7 +116,7 @@ struct wil_fw_record_brd_file { /* type == wil_fw_type_comment */
|
||||
*/
|
||||
struct wil_fw_record_action { /* type == wil_fw_type_action */
|
||||
__le32 action; /* action to perform: reset, wait for fw ready etc. */
|
||||
__le32 data[0]; /* action specific, [data_size], see above */
|
||||
__le32 data[]; /* action specific, [data_size], see above */
|
||||
} __packed;
|
||||
|
||||
/* data block for struct wil_fw_record_direct_write */
|
||||
@ -179,7 +179,7 @@ struct wil_fw_record_gateway_data { /* type == wil_fw_type_gateway_data */
|
||||
#define WIL_FW_GW_CTL_BUSY BIT(29) /* gateway busy performing operation */
|
||||
#define WIL_FW_GW_CTL_RUN BIT(30) /* start gateway operation */
|
||||
__le32 command;
|
||||
struct wil_fw_data_gw data[0]; /* total size [data_size], see above */
|
||||
struct wil_fw_data_gw data[]; /* total size [data_size], see above */
|
||||
} __packed;
|
||||
|
||||
/* 4-dword gateway */
|
||||
@ -201,7 +201,7 @@ struct wil_fw_record_gateway_data4 { /* type == wil_fw_type_gateway_data4 */
|
||||
__le32 gateway_cmd_addr;
|
||||
__le32 gateway_ctrl_address; /* same logic as for 1-dword gw */
|
||||
__le32 command;
|
||||
struct wil_fw_data_gw4 data[0]; /* total size [data_size], see above */
|
||||
struct wil_fw_data_gw4 data[]; /* total size [data_size], see above */
|
||||
} __packed;
|
||||
|
||||
#endif /* __WIL_FW_H__ */
|
||||
|
@ -222,7 +222,7 @@ struct auth_no_hdr {
|
||||
__le16 auth_transaction;
|
||||
__le16 status_code;
|
||||
/* possibly followed by Challenge text */
|
||||
u8 variable[0];
|
||||
u8 variable[];
|
||||
} __packed;
|
||||
|
||||
u8 led_polarity = LED_POLARITY_LOW_ACTIVE;
|
||||
|
@ -474,7 +474,7 @@ struct wmi_start_scan_cmd {
|
||||
struct {
|
||||
u8 channel;
|
||||
u8 reserved;
|
||||
} channel_list[0];
|
||||
} channel_list[];
|
||||
} __packed;
|
||||
|
||||
#define WMI_MAX_PNO_SSID_NUM (16)
|
||||
@ -530,7 +530,7 @@ struct wmi_update_ft_ies_cmd {
|
||||
/* Length of the FT IEs */
|
||||
__le16 ie_len;
|
||||
u8 reserved[2];
|
||||
u8 ie_info[0];
|
||||
u8 ie_info[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_SET_PROBED_SSID_CMDID */
|
||||
@ -575,7 +575,7 @@ struct wmi_set_appie_cmd {
|
||||
u8 reserved;
|
||||
/* Length of the IE to be added to MGMT frame */
|
||||
__le16 ie_len;
|
||||
u8 ie_info[0];
|
||||
u8 ie_info[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_PXMT_RANGE_CFG_CMDID */
|
||||
@ -850,7 +850,7 @@ struct wmi_pcp_start_cmd {
|
||||
struct wmi_sw_tx_req_cmd {
|
||||
u8 dst_mac[WMI_MAC_LEN];
|
||||
__le16 len;
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_SW_TX_REQ_EXT_CMDID */
|
||||
@ -861,7 +861,7 @@ struct wmi_sw_tx_req_ext_cmd {
|
||||
/* Channel to use, 0xFF for currently active channel */
|
||||
u8 channel;
|
||||
u8 reserved[5];
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_VRING_SWITCH_TIMING_CONFIG_CMDID */
|
||||
@ -1423,7 +1423,7 @@ struct wmi_rf_xpm_write_cmd {
|
||||
u8 verify;
|
||||
u8 reserved1[3];
|
||||
/* actual size=num_bytes */
|
||||
u8 data_bytes[0];
|
||||
u8 data_bytes[];
|
||||
} __packed;
|
||||
|
||||
/* Possible modes for temperature measurement */
|
||||
@ -1572,7 +1572,7 @@ struct wmi_tof_session_start_cmd {
|
||||
u8 aoa_type;
|
||||
__le16 num_of_dest;
|
||||
u8 reserved[4];
|
||||
struct wmi_ftm_dest_info ftm_dest_info[0];
|
||||
struct wmi_ftm_dest_info ftm_dest_info[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_TOF_CFG_RESPONDER_CMDID */
|
||||
@ -1766,7 +1766,7 @@ struct wmi_internal_fw_ioctl_cmd {
|
||||
/* payload max size is WMI_MAX_IOCTL_PAYLOAD_SIZE
|
||||
* Must be the last member of the struct
|
||||
*/
|
||||
__le32 payload[0];
|
||||
__le32 payload[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_INTERNAL_FW_IOCTL_EVENTID */
|
||||
@ -1778,7 +1778,7 @@ struct wmi_internal_fw_ioctl_event {
|
||||
/* payload max size is WMI_MAX_IOCTL_REPLY_PAYLOAD_SIZE
|
||||
* Must be the last member of the struct
|
||||
*/
|
||||
__le32 payload[0];
|
||||
__le32 payload[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_INTERNAL_FW_EVENT_EVENTID */
|
||||
@ -1788,7 +1788,7 @@ struct wmi_internal_fw_event_event {
|
||||
/* payload max size is WMI_MAX_INTERNAL_EVENT_PAYLOAD_SIZE
|
||||
* Must be the last member of the struct
|
||||
*/
|
||||
__le32 payload[0];
|
||||
__le32 payload[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_SET_VRING_PRIORITY_WEIGHT_CMDID */
|
||||
@ -1818,7 +1818,7 @@ struct wmi_set_vring_priority_cmd {
|
||||
*/
|
||||
u8 num_of_vrings;
|
||||
u8 reserved[3];
|
||||
struct wmi_vring_priority vring_priority[0];
|
||||
struct wmi_vring_priority vring_priority[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_BF_CONTROL_CMDID - deprecated */
|
||||
@ -1910,7 +1910,7 @@ struct wmi_bf_control_ex_cmd {
|
||||
u8 each_mcs_cfg_size;
|
||||
u8 reserved1;
|
||||
/* Configuration for each MCS */
|
||||
struct wmi_bf_control_ex_mcs each_mcs_cfg[0];
|
||||
struct wmi_bf_control_ex_mcs each_mcs_cfg[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_LINK_STATS_CMD */
|
||||
@ -2192,7 +2192,7 @@ struct wmi_fw_ver_event {
|
||||
/* FW capabilities info
|
||||
* Must be the last member of the struct
|
||||
*/
|
||||
__le32 fw_capabilities[0];
|
||||
__le32 fw_capabilities[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_GET_RF_STATUS_EVENTID */
|
||||
@ -2270,7 +2270,7 @@ struct wmi_mac_addr_resp_event {
|
||||
struct wmi_eapol_rx_event {
|
||||
u8 src_mac[WMI_MAC_LEN];
|
||||
__le16 eapol_len;
|
||||
u8 eapol[0];
|
||||
u8 eapol[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_READY_EVENTID */
|
||||
@ -2343,7 +2343,7 @@ struct wmi_connect_event {
|
||||
u8 aid;
|
||||
u8 reserved2[2];
|
||||
/* not in use */
|
||||
u8 assoc_info[0];
|
||||
u8 assoc_info[];
|
||||
} __packed;
|
||||
|
||||
/* disconnect_reason */
|
||||
@ -2376,7 +2376,7 @@ struct wmi_disconnect_event {
|
||||
/* last assoc req may passed to host - not in used */
|
||||
u8 assoc_resp_len;
|
||||
/* last assoc req may passed to host - not in used */
|
||||
u8 assoc_info[0];
|
||||
u8 assoc_info[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_SCAN_COMPLETE_EVENTID */
|
||||
@ -2400,7 +2400,7 @@ struct wmi_ft_auth_status_event {
|
||||
u8 reserved[3];
|
||||
u8 mac_addr[WMI_MAC_LEN];
|
||||
__le16 ie_len;
|
||||
u8 ie_info[0];
|
||||
u8 ie_info[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_FT_REASSOC_STATUS_EVENTID */
|
||||
@ -2418,7 +2418,7 @@ struct wmi_ft_reassoc_status_event {
|
||||
__le16 reassoc_req_ie_len;
|
||||
__le16 reassoc_resp_ie_len;
|
||||
u8 reserved[4];
|
||||
u8 ie_info[0];
|
||||
u8 ie_info[];
|
||||
} __packed;
|
||||
|
||||
/* wmi_rx_mgmt_info */
|
||||
@ -2461,7 +2461,7 @@ struct wmi_stop_sched_scan_event {
|
||||
|
||||
struct wmi_sched_scan_result_event {
|
||||
struct wmi_rx_mgmt_info info;
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_ACS_PASSIVE_SCAN_COMPLETE_EVENT */
|
||||
@ -2492,7 +2492,7 @@ struct wmi_acs_passive_scan_complete_event {
|
||||
__le16 filled;
|
||||
u8 num_scanned_channels;
|
||||
u8 reserved;
|
||||
struct scan_acs_info scan_info_list[0];
|
||||
struct scan_acs_info scan_info_list[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_BA_STATUS_EVENTID */
|
||||
@ -2751,7 +2751,7 @@ struct wmi_rf_xpm_read_result_event {
|
||||
u8 status;
|
||||
u8 reserved[3];
|
||||
/* requested num_bytes of data */
|
||||
u8 data_bytes[0];
|
||||
u8 data_bytes[];
|
||||
} __packed;
|
||||
|
||||
/* EVENT: WMI_RF_XPM_WRITE_RESULT_EVENTID */
|
||||
@ -2769,7 +2769,7 @@ struct wmi_tx_mgmt_packet_event {
|
||||
/* WMI_RX_MGMT_PACKET_EVENTID */
|
||||
struct wmi_rx_mgmt_packet_event {
|
||||
struct wmi_rx_mgmt_info info;
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_ECHO_RSP_EVENTID */
|
||||
@ -2969,7 +2969,7 @@ struct wmi_rs_cfg_ex_cmd {
|
||||
u8 each_mcs_cfg_size;
|
||||
u8 reserved[3];
|
||||
/* Configuration for each MCS */
|
||||
struct wmi_rs_cfg_ex_mcs each_mcs_cfg[0];
|
||||
struct wmi_rs_cfg_ex_mcs each_mcs_cfg[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_RS_CFG_EX_EVENTID */
|
||||
@ -3178,7 +3178,7 @@ struct wmi_get_detailed_rs_res_ex_event {
|
||||
u8 each_mcs_results_size;
|
||||
u8 reserved1[3];
|
||||
/* Results for each MCS */
|
||||
struct wmi_rs_results_ex_mcs each_mcs_results[0];
|
||||
struct wmi_rs_results_ex_mcs each_mcs_results[];
|
||||
} __packed;
|
||||
|
||||
/* BRP antenna limit mode */
|
||||
@ -3320,7 +3320,7 @@ struct wmi_set_link_monitor_cmd {
|
||||
u8 rssi_hyst;
|
||||
u8 reserved[12];
|
||||
u8 rssi_thresholds_list_size;
|
||||
s8 rssi_thresholds_list[0];
|
||||
s8 rssi_thresholds_list[];
|
||||
} __packed;
|
||||
|
||||
/* wmi_link_monitor_event_type */
|
||||
@ -3637,7 +3637,7 @@ struct wmi_tof_ftm_per_dest_res_event {
|
||||
/* Measurments are from RFs, defined by the mask */
|
||||
__le32 meas_rf_mask;
|
||||
u8 reserved0[3];
|
||||
struct wmi_responder_ftm_res responder_ftm_res[0];
|
||||
struct wmi_responder_ftm_res responder_ftm_res[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_TOF_CFG_RESPONDER_EVENTID */
|
||||
@ -3669,7 +3669,7 @@ struct wmi_tof_channel_info_event {
|
||||
/* data report length */
|
||||
u8 len;
|
||||
/* data report payload */
|
||||
u8 report[0];
|
||||
u8 report[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_TOF_SET_TX_RX_OFFSET_EVENTID */
|
||||
@ -4085,7 +4085,7 @@ struct wmi_link_stats_event {
|
||||
u8 has_next;
|
||||
u8 reserved[5];
|
||||
/* a stream of wmi_link_stats_record_s */
|
||||
u8 payload[0];
|
||||
u8 payload[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_LINK_STATS_EVENT */
|
||||
@ -4094,7 +4094,7 @@ struct wmi_link_stats_record {
|
||||
u8 record_type_id;
|
||||
u8 reserved;
|
||||
__le16 record_size;
|
||||
u8 record[0];
|
||||
u8 record[];
|
||||
} __packed;
|
||||
|
||||
/* WMI_LINK_STATS_TYPE_BASIC */
|
||||
|
@ -798,7 +798,6 @@ static void tx_update_descriptor(struct atmel_private *priv, int is_bcast,
|
||||
|
||||
static netdev_tx_t start_tx(struct sk_buff *skb, struct net_device *dev)
|
||||
{
|
||||
static const u8 SNAP_RFC1024[6] = { 0xaa, 0xaa, 0x03, 0x00, 0x00, 0x00 };
|
||||
struct atmel_private *priv = netdev_priv(dev);
|
||||
struct ieee80211_hdr header;
|
||||
unsigned long flags;
|
||||
@ -853,7 +852,7 @@ static netdev_tx_t start_tx(struct sk_buff *skb, struct net_device *dev)
|
||||
}
|
||||
|
||||
if (priv->use_wpa)
|
||||
memcpy(&header.addr4, SNAP_RFC1024, ETH_ALEN);
|
||||
memcpy(&header.addr4, rfc1042_header, ETH_ALEN);
|
||||
|
||||
header.frame_control = cpu_to_le16(frame_ctl);
|
||||
/* Copy the wireless header into the card */
|
||||
|
@ -5569,7 +5569,7 @@ static struct b43_wl *b43_wireless_init(struct b43_bus_dev *dev)
|
||||
/* fill hw info */
|
||||
ieee80211_hw_set(hw, RX_INCLUDES_FCS);
|
||||
ieee80211_hw_set(hw, SIGNAL_DBM);
|
||||
|
||||
ieee80211_hw_set(hw, MFP_CAPABLE);
|
||||
hw->wiphy->interface_modes =
|
||||
BIT(NL80211_IFTYPE_AP) |
|
||||
BIT(NL80211_IFTYPE_MESH_POINT) |
|
||||
|
@ -3801,6 +3801,7 @@ static int b43legacy_wireless_init(struct ssb_device *dev)
|
||||
/* fill hw info */
|
||||
ieee80211_hw_set(hw, RX_INCLUDES_FCS);
|
||||
ieee80211_hw_set(hw, SIGNAL_DBM);
|
||||
ieee80211_hw_set(hw, MFP_CAPABLE); /* Allow WPA3 in software */
|
||||
|
||||
hw->wiphy->interface_modes =
|
||||
BIT(NL80211_IFTYPE_AP) |
|
||||
|
@ -43,7 +43,8 @@
|
||||
|
||||
#define SDIO_FUNC1_BLOCKSIZE 64
|
||||
#define SDIO_FUNC2_BLOCKSIZE 512
|
||||
#define SDIO_4359_FUNC2_BLOCKSIZE 256
|
||||
#define SDIO_4373_FUNC2_BLOCKSIZE 256
|
||||
#define SDIO_435X_FUNC2_BLOCKSIZE 256
|
||||
/* Maximum milliseconds to wait for F2 to come up */
|
||||
#define SDIO_WAIT_F2RDY 3000
|
||||
|
||||
@ -910,13 +911,28 @@ static int brcmf_sdiod_probe(struct brcmf_sdio_dev *sdiodev)
|
||||
sdio_release_host(sdiodev->func1);
|
||||
goto out;
|
||||
}
|
||||
if (sdiodev->func2->device == SDIO_DEVICE_ID_BROADCOM_4359)
|
||||
f2_blksz = SDIO_4359_FUNC2_BLOCKSIZE;
|
||||
switch (sdiodev->func2->device) {
|
||||
case SDIO_DEVICE_ID_CYPRESS_4373:
|
||||
f2_blksz = SDIO_4373_FUNC2_BLOCKSIZE;
|
||||
break;
|
||||
case SDIO_DEVICE_ID_BROADCOM_4359:
|
||||
/* fallthrough */
|
||||
case SDIO_DEVICE_ID_BROADCOM_4354:
|
||||
/* fallthrough */
|
||||
case SDIO_DEVICE_ID_BROADCOM_4356:
|
||||
f2_blksz = SDIO_435X_FUNC2_BLOCKSIZE;
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
ret = sdio_set_block_size(sdiodev->func2, f2_blksz);
|
||||
if (ret) {
|
||||
brcmf_err("Failed to set F2 blocksize\n");
|
||||
sdio_release_host(sdiodev->func1);
|
||||
goto out;
|
||||
} else {
|
||||
brcmf_dbg(SDIO, "set F2 blocksize to %d\n", f2_blksz);
|
||||
}
|
||||
|
||||
/* increase F2 timeout */
|
||||
|
@ -41,9 +41,22 @@
|
||||
/* watermark expressed in number of words */
|
||||
#define DEFAULT_F2_WATERMARK 0x8
|
||||
#define CY_4373_F2_WATERMARK 0x40
|
||||
#define CY_4373_F1_MESBUSYCTRL (CY_4373_F2_WATERMARK | SBSDIO_MESBUSYCTRL_ENAB)
|
||||
#define CY_43012_F2_WATERMARK 0x60
|
||||
#define CY_4359_F2_WATERMARK 0x40
|
||||
#define CY_4359_F1_MESBUSYCTRL (CY_4359_F2_WATERMARK | SBSDIO_MESBUSYCTRL_ENAB)
|
||||
#define CY_43012_MES_WATERMARK 0x50
|
||||
#define CY_43012_MESBUSYCTRL (CY_43012_MES_WATERMARK | \
|
||||
SBSDIO_MESBUSYCTRL_ENAB)
|
||||
#define CY_4339_F2_WATERMARK 48
|
||||
#define CY_4339_MES_WATERMARK 80
|
||||
#define CY_4339_MESBUSYCTRL (CY_4339_MES_WATERMARK | \
|
||||
SBSDIO_MESBUSYCTRL_ENAB)
|
||||
#define CY_43455_F2_WATERMARK 0x60
|
||||
#define CY_43455_MES_WATERMARK 0x50
|
||||
#define CY_43455_MESBUSYCTRL (CY_43455_MES_WATERMARK | \
|
||||
SBSDIO_MESBUSYCTRL_ENAB)
|
||||
#define CY_435X_F2_WATERMARK 0x40
|
||||
#define CY_435X_F1_MESBUSYCTRL (CY_435X_F2_WATERMARK | \
|
||||
SBSDIO_MESBUSYCTRL_ENAB)
|
||||
|
||||
#ifdef DEBUG
|
||||
|
||||
@ -4195,8 +4208,7 @@ static void brcmf_sdio_firmware_callback(struct device *dev, int err,
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_DEVICE_CTL, devctl,
|
||||
&err);
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_FUNC1_MESBUSYCTRL,
|
||||
CY_4373_F2_WATERMARK |
|
||||
SBSDIO_MESBUSYCTRL_ENAB, &err);
|
||||
CY_4373_F1_MESBUSYCTRL, &err);
|
||||
break;
|
||||
case SDIO_DEVICE_ID_CYPRESS_43012:
|
||||
brcmf_dbg(INFO, "set F2 watermark to 0x%x*4 bytes\n",
|
||||
@ -4208,19 +4220,51 @@ static void brcmf_sdio_firmware_callback(struct device *dev, int err,
|
||||
devctl |= SBSDIO_DEVCTL_F2WM_ENAB;
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_DEVICE_CTL, devctl,
|
||||
&err);
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_FUNC1_MESBUSYCTRL,
|
||||
CY_43012_MESBUSYCTRL, &err);
|
||||
break;
|
||||
case SDIO_DEVICE_ID_BROADCOM_4359:
|
||||
brcmf_dbg(INFO, "set F2 watermark to 0x%x*4 bytes\n",
|
||||
CY_4359_F2_WATERMARK);
|
||||
case SDIO_DEVICE_ID_BROADCOM_4339:
|
||||
brcmf_dbg(INFO, "set F2 watermark to 0x%x*4 bytes for 4339\n",
|
||||
CY_4339_F2_WATERMARK);
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_WATERMARK,
|
||||
CY_4359_F2_WATERMARK, &err);
|
||||
CY_4339_F2_WATERMARK, &err);
|
||||
devctl = brcmf_sdiod_readb(sdiod, SBSDIO_DEVICE_CTL,
|
||||
&err);
|
||||
devctl |= SBSDIO_DEVCTL_F2WM_ENAB;
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_DEVICE_CTL, devctl,
|
||||
&err);
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_FUNC1_MESBUSYCTRL,
|
||||
CY_4359_F1_MESBUSYCTRL, &err);
|
||||
CY_4339_MESBUSYCTRL, &err);
|
||||
break;
|
||||
case SDIO_DEVICE_ID_BROADCOM_43455:
|
||||
brcmf_dbg(INFO, "set F2 watermark to 0x%x*4 bytes for 43455\n",
|
||||
CY_43455_F2_WATERMARK);
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_WATERMARK,
|
||||
CY_43455_F2_WATERMARK, &err);
|
||||
devctl = brcmf_sdiod_readb(sdiod, SBSDIO_DEVICE_CTL,
|
||||
&err);
|
||||
devctl |= SBSDIO_DEVCTL_F2WM_ENAB;
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_DEVICE_CTL, devctl,
|
||||
&err);
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_FUNC1_MESBUSYCTRL,
|
||||
CY_43455_MESBUSYCTRL, &err);
|
||||
break;
|
||||
case SDIO_DEVICE_ID_BROADCOM_4359:
|
||||
/* fallthrough */
|
||||
case SDIO_DEVICE_ID_BROADCOM_4354:
|
||||
/* fallthrough */
|
||||
case SDIO_DEVICE_ID_BROADCOM_4356:
|
||||
brcmf_dbg(INFO, "set F2 watermark to 0x%x*4 bytes\n",
|
||||
CY_435X_F2_WATERMARK);
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_WATERMARK,
|
||||
CY_435X_F2_WATERMARK, &err);
|
||||
devctl = brcmf_sdiod_readb(sdiod, SBSDIO_DEVICE_CTL,
|
||||
&err);
|
||||
devctl |= SBSDIO_DEVCTL_F2WM_ENAB;
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_DEVICE_CTL, devctl,
|
||||
&err);
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_FUNC1_MESBUSYCTRL,
|
||||
CY_435X_F1_MESBUSYCTRL, &err);
|
||||
break;
|
||||
default:
|
||||
brcmf_sdiod_writeb(sdiod, SBSDIO_WATERMARK,
|
||||
|
@ -1925,6 +1925,10 @@ static netdev_tx_t mpi_start_xmit(struct sk_buff *skb,
|
||||
airo_print_err(dev->name, "%s: skb == NULL!",__func__);
|
||||
return NETDEV_TX_OK;
|
||||
}
|
||||
if (skb_padto(skb, ETH_ZLEN)) {
|
||||
dev->stats.tx_dropped++;
|
||||
return NETDEV_TX_OK;
|
||||
}
|
||||
npacks = skb_queue_len (&ai->txq);
|
||||
|
||||
if (npacks >= MAXTXQ - 1) {
|
||||
@ -2127,6 +2131,10 @@ static netdev_tx_t airo_start_xmit(struct sk_buff *skb,
|
||||
airo_print_err(dev->name, "%s: skb == NULL!", __func__);
|
||||
return NETDEV_TX_OK;
|
||||
}
|
||||
if (skb_padto(skb, ETH_ZLEN)) {
|
||||
dev->stats.tx_dropped++;
|
||||
return NETDEV_TX_OK;
|
||||
}
|
||||
|
||||
/* Find a vacant FID */
|
||||
for( i = 0; i < MAX_FIDS / 2 && (fids[i] & 0xffff0000); i++ );
|
||||
@ -2201,6 +2209,10 @@ static netdev_tx_t airo_start_xmit11(struct sk_buff *skb,
|
||||
airo_print_err(dev->name, "%s: skb == NULL!", __func__);
|
||||
return NETDEV_TX_OK;
|
||||
}
|
||||
if (skb_padto(skb, ETH_ZLEN)) {
|
||||
dev->stats.tx_dropped++;
|
||||
return NETDEV_TX_OK;
|
||||
}
|
||||
|
||||
/* Find a vacant FID */
|
||||
for( i = MAX_FIDS / 2; i < MAX_FIDS && (fids[i] & 0xffff0000); i++ );
|
||||
|
@ -57,7 +57,7 @@
|
||||
#include "iwl-prph.h"
|
||||
|
||||
/* Highest firmware API version supported */
|
||||
#define IWL_22000_UCODE_API_MAX 55
|
||||
#define IWL_22000_UCODE_API_MAX 56
|
||||
|
||||
/* Lowest firmware API version supported */
|
||||
#define IWL_22000_UCODE_API_MIN 39
|
||||
|
@ -58,44 +58,121 @@
|
||||
*
|
||||
*****************************************************************************/
|
||||
|
||||
#include <linux/uuid.h>
|
||||
#include "iwl-drv.h"
|
||||
#include "iwl-debug.h"
|
||||
#include "acpi.h"
|
||||
#include "fw/runtime.h"
|
||||
|
||||
void *iwl_acpi_get_object(struct device *dev, acpi_string method)
|
||||
static const guid_t intel_wifi_guid = GUID_INIT(0xF21202BF, 0x8F78, 0x4DC6,
|
||||
0xA5, 0xB3, 0x1F, 0x73,
|
||||
0x8E, 0x28, 0x5A, 0xDE);
|
||||
|
||||
static int iwl_acpi_get_handle(struct device *dev, acpi_string method,
|
||||
acpi_handle *ret_handle)
|
||||
{
|
||||
acpi_handle root_handle;
|
||||
acpi_handle handle;
|
||||
struct acpi_buffer buf = {ACPI_ALLOCATE_BUFFER, NULL};
|
||||
acpi_status status;
|
||||
|
||||
root_handle = ACPI_HANDLE(dev);
|
||||
if (!root_handle) {
|
||||
IWL_DEBUG_DEV_RADIO(dev,
|
||||
"Could not retrieve root port ACPI handle\n");
|
||||
return ERR_PTR(-ENOENT);
|
||||
"ACPI: Could not retrieve root port handle\n");
|
||||
return -ENOENT;
|
||||
}
|
||||
|
||||
/* Get the method's handle */
|
||||
status = acpi_get_handle(root_handle, method, &handle);
|
||||
status = acpi_get_handle(root_handle, method, ret_handle);
|
||||
if (ACPI_FAILURE(status)) {
|
||||
IWL_DEBUG_DEV_RADIO(dev, "%s method not found\n", method);
|
||||
return ERR_PTR(-ENOENT);
|
||||
IWL_DEBUG_DEV_RADIO(dev,
|
||||
"ACPI: %s method not found\n", method);
|
||||
return -ENOENT;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
void *iwl_acpi_get_object(struct device *dev, acpi_string method)
|
||||
{
|
||||
struct acpi_buffer buf = {ACPI_ALLOCATE_BUFFER, NULL};
|
||||
acpi_handle handle;
|
||||
acpi_status status;
|
||||
int ret;
|
||||
|
||||
ret = iwl_acpi_get_handle(dev, method, &handle);
|
||||
if (ret)
|
||||
return ERR_PTR(-ENOENT);
|
||||
|
||||
/* Call the method with no arguments */
|
||||
status = acpi_evaluate_object(handle, NULL, NULL, &buf);
|
||||
if (ACPI_FAILURE(status)) {
|
||||
IWL_DEBUG_DEV_RADIO(dev, "%s invocation failed (0x%x)\n",
|
||||
IWL_DEBUG_DEV_RADIO(dev,
|
||||
"ACPI: %s method invocation failed (status: 0x%x)\n",
|
||||
method, status);
|
||||
return ERR_PTR(-ENOENT);
|
||||
}
|
||||
|
||||
return buf.pointer;
|
||||
}
|
||||
IWL_EXPORT_SYMBOL(iwl_acpi_get_object);
|
||||
|
||||
/**
|
||||
* Generic function for evaluating a method defined in the device specific
|
||||
* method (DSM) interface. The returned acpi object must be freed by calling
|
||||
* function.
|
||||
*/
|
||||
void *iwl_acpi_get_dsm_object(struct device *dev, int rev, int func,
|
||||
union acpi_object *args)
|
||||
{
|
||||
union acpi_object *obj;
|
||||
|
||||
obj = acpi_evaluate_dsm(ACPI_HANDLE(dev), &intel_wifi_guid, rev, func,
|
||||
args);
|
||||
if (!obj) {
|
||||
IWL_DEBUG_DEV_RADIO(dev,
|
||||
"ACPI: DSM method invocation failed (rev: %d, func:%d)\n",
|
||||
rev, func);
|
||||
return ERR_PTR(-ENOENT);
|
||||
}
|
||||
return obj;
|
||||
}
|
||||
|
||||
/**
|
||||
* Evaluate a DSM with no arguments and a single u8 return value (inside a
|
||||
* buffer object), verify and return that value.
|
||||
*/
|
||||
int iwl_acpi_get_dsm_u8(struct device *dev, int rev, int func)
|
||||
{
|
||||
union acpi_object *obj;
|
||||
int ret;
|
||||
|
||||
obj = iwl_acpi_get_dsm_object(dev, rev, func, NULL);
|
||||
if (IS_ERR(obj))
|
||||
return -ENOENT;
|
||||
|
||||
if (obj->type != ACPI_TYPE_BUFFER) {
|
||||
IWL_DEBUG_DEV_RADIO(dev,
|
||||
"ACPI: DSM method did not return a valid object, type=%d\n",
|
||||
obj->type);
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
if (obj->buffer.length != sizeof(u8)) {
|
||||
IWL_DEBUG_DEV_RADIO(dev,
|
||||
"ACPI: DSM method returned invalid buffer, length=%d\n",
|
||||
obj->buffer.length);
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
ret = obj->buffer.pointer[0];
|
||||
IWL_DEBUG_DEV_RADIO(dev,
|
||||
"ACPI: DSM method evaluated: func=%d, ret=%d\n",
|
||||
func, ret);
|
||||
out:
|
||||
ACPI_FREE(obj);
|
||||
return ret;
|
||||
}
|
||||
IWL_EXPORT_SYMBOL(iwl_acpi_get_dsm_u8);
|
||||
|
||||
union acpi_object *iwl_acpi_get_wifi_pkg(struct device *dev,
|
||||
union acpi_object *data,
|
||||
int data_size, int *tbl_rev)
|
||||
|
@ -127,12 +127,23 @@ struct iwl_geo_profile {
|
||||
u8 values[ACPI_GEO_TABLE_SIZE];
|
||||
};
|
||||
|
||||
enum iwl_dsm_funcs_rev_0 {
|
||||
DSM_FUNC_QUERY = 0,
|
||||
DSM_FUNC_DISABLE_SRD = 1,
|
||||
DSM_FUNC_ENABLE_INDONESIA_5G2 = 2,
|
||||
};
|
||||
|
||||
#ifdef CONFIG_ACPI
|
||||
|
||||
struct iwl_fw_runtime;
|
||||
|
||||
void *iwl_acpi_get_object(struct device *dev, acpi_string method);
|
||||
|
||||
void *iwl_acpi_get_dsm_object(struct device *dev, int rev, int func,
|
||||
union acpi_object *args);
|
||||
|
||||
int iwl_acpi_get_dsm_u8(struct device *dev, int rev, int func);
|
||||
|
||||
union acpi_object *iwl_acpi_get_wifi_pkg(struct device *dev,
|
||||
union acpi_object *data,
|
||||
int data_size, int *tbl_rev);
|
||||
@ -192,6 +203,17 @@ static inline void *iwl_acpi_get_object(struct device *dev, acpi_string method)
|
||||
return ERR_PTR(-ENOENT);
|
||||
}
|
||||
|
||||
static inline void *iwl_acpi_get_dsm_object(struct device *dev, int rev,
|
||||
int func, union acpi_object *args)
|
||||
{
|
||||
return ERR_PTR(-ENOENT);
|
||||
}
|
||||
|
||||
static inline int iwl_acpi_get_dsm_u8(struct device *dev, int rev, int func)
|
||||
{
|
||||
return -ENOENT;
|
||||
}
|
||||
|
||||
static inline union acpi_object *iwl_acpi_get_wifi_pkg(struct device *dev,
|
||||
union acpi_object *data,
|
||||
int data_size,
|
||||
|
@ -550,13 +550,11 @@ struct iwl_tof_range_req_ap_entry_v4 {
|
||||
/**
|
||||
* enum iwl_location_cipher - location cipher selection
|
||||
* @IWL_LOCATION_CIPHER_CCMP_128: CCMP 128
|
||||
* @IWL_LOCATION_CIPHER_CCMP_256: CCMP 256
|
||||
* @IWL_LOCATION_CIPHER_GCMP_128: GCMP 128
|
||||
* @IWL_LOCATION_CIPHER_GCMP_256: GCMP 256
|
||||
*/
|
||||
enum iwl_location_cipher {
|
||||
IWL_LOCATION_CIPHER_CCMP_128,
|
||||
IWL_LOCATION_CIPHER_CCMP_256,
|
||||
IWL_LOCATION_CIPHER_GCMP_128,
|
||||
IWL_LOCATION_CIPHER_GCMP_256,
|
||||
};
|
||||
@ -577,7 +575,8 @@ enum iwl_location_cipher {
|
||||
* @samples_per_burst: the number of FTMs pairs in single Burst (1-31);
|
||||
* @num_of_bursts: Recommended value to be sent to the AP. 2s Exponent of
|
||||
* the number of measurement iterations (min 2^0 = 1, max 2^14)
|
||||
* @reserved: For alignment and future use
|
||||
* @sta_id: the station id of the AP. Only relevant when associated to the AP,
|
||||
* otherwise should be set to &IWL_MVM_INVALID_STA.
|
||||
* @cipher: pairwise cipher suite for secured measurement.
|
||||
* &enum iwl_location_cipher.
|
||||
* @hltk: HLTK to be used for secured 11az measurement
|
||||
@ -586,7 +585,8 @@ enum iwl_location_cipher {
|
||||
* If &IWL_INITIATOR_AP_FLAGS_USE_CALIB is set, the fw will use the
|
||||
* calibration value that corresponds to the rx bandwidth of the FTM
|
||||
* frame.
|
||||
* @reserved2: For alignment and future use.
|
||||
* @beacon_interval: beacon interval of the AP in TUs. Only required if
|
||||
* &IWL_INITIATOR_AP_FLAGS_TB is set.
|
||||
*/
|
||||
struct iwl_tof_range_req_ap_entry {
|
||||
__le32 initiator_ap_flags;
|
||||
@ -598,13 +598,13 @@ struct iwl_tof_range_req_ap_entry {
|
||||
__le16 burst_period;
|
||||
u8 samples_per_burst;
|
||||
u8 num_of_bursts;
|
||||
u8 reserved;
|
||||
u8 sta_id;
|
||||
u8 cipher;
|
||||
u8 hltk[HLTK_11AZ_LEN];
|
||||
u8 tk[TK_11AZ_LEN];
|
||||
__le16 calib[IWL_TOF_BW_NUM];
|
||||
__le16 reserved2;
|
||||
} __packed; /* LOCATION_RANGE_REQ_AP_ENTRY_CMD_API_S_VER_5 */
|
||||
__le16 beacon_interval;
|
||||
} __packed; /* LOCATION_RANGE_REQ_AP_ENTRY_CMD_API_S_VER_6 */
|
||||
|
||||
/**
|
||||
* enum iwl_tof_response_mode
|
||||
|
@ -8,7 +8,7 @@
|
||||
* Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
|
||||
* Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
|
||||
* Copyright(c) 2016 - 2017 Intel Deutschland GmbH
|
||||
* Copyright(C) 2018 - 2019 Intel Corporation
|
||||
* Copyright(C) 2018 - 2020 Intel Corporation
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of version 2 of the GNU General Public License as
|
||||
@ -31,7 +31,7 @@
|
||||
* Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
|
||||
* Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
|
||||
* Copyright(c) 2016 - 2017 Intel Deutschland GmbH
|
||||
* Copyright(C) 2018 - 2019 Intel Corporation
|
||||
* Copyright(C) 2018 - 2020 Intel Corporation
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
@ -74,6 +74,11 @@ enum iwl_regulatory_and_nvm_subcmd_ids {
|
||||
*/
|
||||
NVM_ACCESS_COMPLETE = 0x0,
|
||||
|
||||
/**
|
||||
* @LARI_CONFIG_CHANGE: &struct iwl_lari_config_change_cmd
|
||||
*/
|
||||
LARI_CONFIG_CHANGE = 0x1,
|
||||
|
||||
/**
|
||||
* @NVM_GET_INFO:
|
||||
* Command is &struct iwl_nvm_get_info,
|
||||
@ -446,4 +451,29 @@ struct iwl_tas_config_cmd {
|
||||
__le32 black_list_size;
|
||||
__le32 black_list_array[IWL_TAS_BLACK_LIST_MAX];
|
||||
} __packed; /* TAS_CONFIG_CMD_API_S_VER_2 */
|
||||
|
||||
/**
|
||||
* enum iwl_lari_configs - bit masks for the various LARI config operations
|
||||
* @LARI_CONFIG_DISABLE_11AC_UKRAINE_MSK: disable 11ac in ukraine
|
||||
* @LARI_CONFIG_CHANGE_ETSI_TO_PASSIVE_MSK: ETSI 5.8GHz SRD passive scan
|
||||
* @LARI_CONFIG_CHANGE_ETSI_TO_DISABLED_MSK: ETSI 5.8GHz SRD disabled
|
||||
* @LARI_CONFIG_ENABLE_5G2_IN_INDONESIA_MSK: enable 5.15/5.35GHz bands in
|
||||
* Indonesia
|
||||
*/
|
||||
enum iwl_lari_config_masks {
|
||||
LARI_CONFIG_DISABLE_11AC_UKRAINE_MSK = BIT(0),
|
||||
LARI_CONFIG_CHANGE_ETSI_TO_PASSIVE_MSK = BIT(1),
|
||||
LARI_CONFIG_CHANGE_ETSI_TO_DISABLED_MSK = BIT(2),
|
||||
LARI_CONFIG_ENABLE_5G2_IN_INDONESIA_MSK = BIT(3),
|
||||
};
|
||||
|
||||
/**
|
||||
* struct iwl_lari_config_change_cmd - change LARI configuration
|
||||
* @config_bitmap: bit map of the config commands. each bit will trigger a
|
||||
* different predefined FW config operation
|
||||
*/
|
||||
struct iwl_lari_config_change_cmd {
|
||||
__le32 config_bitmap;
|
||||
} __packed; /* LARI_CHANGE_CONF_CMD_S_VER_1 */
|
||||
|
||||
#endif /* __iwl_fw_api_nvm_reg_h__ */
|
||||
|
@ -5,7 +5,7 @@
|
||||
*
|
||||
* GPL LICENSE SUMMARY
|
||||
*
|
||||
* Copyright(c) 2018 Intel Corporation
|
||||
* Copyright(c) 2018, 2020 Intel Corporation
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of version 2 of the GNU General Public License as
|
||||
@ -18,7 +18,7 @@
|
||||
*
|
||||
* BSD LICENSE
|
||||
*
|
||||
* Copyright(c) 2018 Intel Corporation
|
||||
* Copyright(c) 2018, 2020 Intel Corporation
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
@ -93,6 +93,11 @@ enum iwl_prph_scratch_mtr_format {
|
||||
* @IWL_PRPH_SCRATCH_MTR_FORMAT: a mask for the size of the tfd.
|
||||
* There are 4 optional values: 0: 16 bit, 1: 32 bit, 2: 64 bit,
|
||||
* 3: 256 bit.
|
||||
* @IWL_PRPH_SCRATCH_RB_SIZE_EXT_MASK: RB size full information, ignored
|
||||
* by older firmware versions, so set IWL_PRPH_SCRATCH_RB_SIZE_4K
|
||||
* appropriately; use the below values for this.
|
||||
* @IWL_PRPH_SCRATCH_RB_SIZE_EXT_8K: 8kB RB size
|
||||
* @IWL_PRPH_SCRATCH_RB_SIZE_EXT_12K: 12kB RB size
|
||||
*/
|
||||
enum iwl_prph_scratch_flags {
|
||||
IWL_PRPH_SCRATCH_EARLY_DEBUG_EN = BIT(4),
|
||||
@ -103,6 +108,9 @@ enum iwl_prph_scratch_flags {
|
||||
IWL_PRPH_SCRATCH_RB_SIZE_4K = BIT(16),
|
||||
IWL_PRPH_SCRATCH_MTR_MODE = BIT(17),
|
||||
IWL_PRPH_SCRATCH_MTR_FORMAT = BIT(18) | BIT(19),
|
||||
IWL_PRPH_SCRATCH_RB_SIZE_EXT_MASK = 0xf << 20,
|
||||
IWL_PRPH_SCRATCH_RB_SIZE_EXT_8K = 8 << 20,
|
||||
IWL_PRPH_SCRATCH_RB_SIZE_EXT_12K = 9 << 20,
|
||||
};
|
||||
|
||||
/*
|
||||
|
@ -240,6 +240,7 @@ enum iwl_nvm_channel_flags {
|
||||
* @REG_CAPA_40MHZ_FORBIDDEN: 11n channel with a width of 40Mhz is forbidden
|
||||
* for this regulatory domain (valid only in 5Ghz).
|
||||
* @REG_CAPA_DC_HIGH_ENABLED: DC HIGH allowed.
|
||||
* @REG_CAPA_11AX_DISABLED: 11ax is forbidden for this regulatory domain.
|
||||
*/
|
||||
enum iwl_reg_capa_flags {
|
||||
REG_CAPA_BF_CCD_LOW_BAND = BIT(0),
|
||||
@ -250,6 +251,7 @@ enum iwl_reg_capa_flags {
|
||||
REG_CAPA_MCS_9_ALLOWED = BIT(5),
|
||||
REG_CAPA_40MHZ_FORBIDDEN = BIT(7),
|
||||
REG_CAPA_DC_HIGH_ENABLED = BIT(9),
|
||||
REG_CAPA_11AX_DISABLED = BIT(10),
|
||||
};
|
||||
|
||||
static inline void iwl_nvm_print_channel_flags(struct device *dev, u32 level,
|
||||
@ -1115,6 +1117,9 @@ static u32 iwl_nvm_get_regdom_bw_flags(const u16 *nvm_chan,
|
||||
flags |= NL80211_RRF_NO_160MHZ;
|
||||
}
|
||||
|
||||
if (cap_flags & REG_CAPA_11AX_DISABLED)
|
||||
flags |= NL80211_RRF_NO_HE;
|
||||
|
||||
return flags;
|
||||
}
|
||||
|
||||
|
@ -795,6 +795,132 @@ struct iwl_trans_debug {
|
||||
u32 domains_bitmap;
|
||||
};
|
||||
|
||||
struct iwl_dma_ptr {
|
||||
dma_addr_t dma;
|
||||
void *addr;
|
||||
size_t size;
|
||||
};
|
||||
|
||||
struct iwl_cmd_meta {
|
||||
/* only for SYNC commands, iff the reply skb is wanted */
|
||||
struct iwl_host_cmd *source;
|
||||
u32 flags;
|
||||
u32 tbs;
|
||||
};
|
||||
|
||||
/*
|
||||
* The FH will write back to the first TB only, so we need to copy some data
|
||||
* into the buffer regardless of whether it should be mapped or not.
|
||||
* This indicates how big the first TB must be to include the scratch buffer
|
||||
* and the assigned PN.
|
||||
* Since PN location is 8 bytes at offset 12, it's 20 now.
|
||||
* If we make it bigger then allocations will be bigger and copy slower, so
|
||||
* that's probably not useful.
|
||||
*/
|
||||
#define IWL_FIRST_TB_SIZE 20
|
||||
#define IWL_FIRST_TB_SIZE_ALIGN ALIGN(IWL_FIRST_TB_SIZE, 64)
|
||||
|
||||
struct iwl_pcie_txq_entry {
|
||||
void *cmd;
|
||||
struct sk_buff *skb;
|
||||
/* buffer to free after command completes */
|
||||
const void *free_buf;
|
||||
struct iwl_cmd_meta meta;
|
||||
};
|
||||
|
||||
struct iwl_pcie_first_tb_buf {
|
||||
u8 buf[IWL_FIRST_TB_SIZE_ALIGN];
|
||||
};
|
||||
|
||||
/**
|
||||
* struct iwl_txq - Tx Queue for DMA
|
||||
* @q: generic Rx/Tx queue descriptor
|
||||
* @tfds: transmit frame descriptors (DMA memory)
|
||||
* @first_tb_bufs: start of command headers, including scratch buffers, for
|
||||
* the writeback -- this is DMA memory and an array holding one buffer
|
||||
* for each command on the queue
|
||||
* @first_tb_dma: DMA address for the first_tb_bufs start
|
||||
* @entries: transmit entries (driver state)
|
||||
* @lock: queue lock
|
||||
* @stuck_timer: timer that fires if queue gets stuck
|
||||
* @trans: pointer back to transport (for timer)
|
||||
* @need_update: indicates need to update read/write index
|
||||
* @ampdu: true if this queue is an ampdu queue for an specific RA/TID
|
||||
* @wd_timeout: queue watchdog timeout (jiffies) - per queue
|
||||
* @frozen: tx stuck queue timer is frozen
|
||||
* @frozen_expiry_remainder: remember how long until the timer fires
|
||||
* @bc_tbl: byte count table of the queue (relevant only for gen2 transport)
|
||||
* @write_ptr: 1-st empty entry (index) host_w
|
||||
* @read_ptr: last used entry (index) host_r
|
||||
* @dma_addr: physical addr for BD's
|
||||
* @n_window: safe queue window
|
||||
* @id: queue id
|
||||
* @low_mark: low watermark, resume queue if free space more than this
|
||||
* @high_mark: high watermark, stop queue if free space less than this
|
||||
*
|
||||
* A Tx queue consists of circular buffer of BDs (a.k.a. TFDs, transmit frame
|
||||
* descriptors) and required locking structures.
|
||||
*
|
||||
* Note the difference between TFD_QUEUE_SIZE_MAX and n_window: the hardware
|
||||
* always assumes 256 descriptors, so TFD_QUEUE_SIZE_MAX is always 256 (unless
|
||||
* there might be HW changes in the future). For the normal TX
|
||||
* queues, n_window, which is the size of the software queue data
|
||||
* is also 256; however, for the command queue, n_window is only
|
||||
* 32 since we don't need so many commands pending. Since the HW
|
||||
* still uses 256 BDs for DMA though, TFD_QUEUE_SIZE_MAX stays 256.
|
||||
* This means that we end up with the following:
|
||||
* HW entries: | 0 | ... | N * 32 | ... | N * 32 + 31 | ... | 255 |
|
||||
* SW entries: | 0 | ... | 31 |
|
||||
* where N is a number between 0 and 7. This means that the SW
|
||||
* data is a window overlayed over the HW queue.
|
||||
*/
|
||||
struct iwl_txq {
|
||||
void *tfds;
|
||||
struct iwl_pcie_first_tb_buf *first_tb_bufs;
|
||||
dma_addr_t first_tb_dma;
|
||||
struct iwl_pcie_txq_entry *entries;
|
||||
/* lock for syncing changes on the queue */
|
||||
spinlock_t lock;
|
||||
unsigned long frozen_expiry_remainder;
|
||||
struct timer_list stuck_timer;
|
||||
struct iwl_trans *trans;
|
||||
bool need_update;
|
||||
bool frozen;
|
||||
bool ampdu;
|
||||
int block;
|
||||
unsigned long wd_timeout;
|
||||
struct sk_buff_head overflow_q;
|
||||
struct iwl_dma_ptr bc_tbl;
|
||||
|
||||
int write_ptr;
|
||||
int read_ptr;
|
||||
dma_addr_t dma_addr;
|
||||
int n_window;
|
||||
u32 id;
|
||||
int low_mark;
|
||||
int high_mark;
|
||||
|
||||
bool overflow_tx;
|
||||
};
|
||||
|
||||
/**
|
||||
* struct iwl_trans_txqs - transport tx queues data
|
||||
*
|
||||
* @queue_used - bit mask of used queues
|
||||
* @queue_stopped - bit mask of stopped queues
|
||||
*/
|
||||
struct iwl_trans_txqs {
|
||||
unsigned long queue_used[BITS_TO_LONGS(IWL_MAX_TVQM_QUEUES)];
|
||||
unsigned long queue_stopped[BITS_TO_LONGS(IWL_MAX_TVQM_QUEUES)];
|
||||
struct iwl_txq *txq[IWL_MAX_TVQM_QUEUES];
|
||||
struct {
|
||||
u8 fifo;
|
||||
u8 q_id;
|
||||
unsigned int wdg_timeout;
|
||||
} cmd;
|
||||
|
||||
};
|
||||
|
||||
/**
|
||||
* struct iwl_trans - transport common data
|
||||
*
|
||||
@ -828,6 +954,7 @@ struct iwl_trans_debug {
|
||||
* @system_pm_mode: the system-wide power management mode in use.
|
||||
* This mode is set dynamically, depending on the WoWLAN values
|
||||
* configured from the userspace at runtime.
|
||||
* @iwl_trans_txqs: transport tx queues data.
|
||||
*/
|
||||
struct iwl_trans {
|
||||
const struct iwl_trans_ops *ops;
|
||||
@ -875,6 +1002,7 @@ struct iwl_trans {
|
||||
enum iwl_plat_pm_mode system_pm_mode;
|
||||
|
||||
const char *name;
|
||||
struct iwl_trans_txqs txqs;
|
||||
|
||||
/* pointer to trans specific struct */
|
||||
/*Ensure that this pointer will always be aligned to sizeof pointer */
|
||||
|
@ -391,9 +391,9 @@ iwl_mvm_ftm_put_target_v3(struct iwl_mvm *mvm,
|
||||
}
|
||||
|
||||
static int
|
||||
iwl_mvm_ftm_put_target(struct iwl_mvm *mvm,
|
||||
struct cfg80211_pmsr_request_peer *peer,
|
||||
struct iwl_tof_range_req_ap_entry_v4 *target)
|
||||
iwl_mvm_ftm_put_target_v4(struct iwl_mvm *mvm,
|
||||
struct cfg80211_pmsr_request_peer *peer,
|
||||
struct iwl_tof_range_req_ap_entry_v4 *target)
|
||||
{
|
||||
int ret;
|
||||
|
||||
@ -408,6 +408,38 @@ iwl_mvm_ftm_put_target(struct iwl_mvm *mvm,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
iwl_mvm_ftm_put_target(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
struct cfg80211_pmsr_request_peer *peer,
|
||||
struct iwl_tof_range_req_ap_entry *target)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = iwl_mvm_ftm_target_chandef_v2(mvm, peer, &target->channel_num,
|
||||
&target->format_bw,
|
||||
&target->ctrl_ch_position);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
iwl_mvm_ftm_put_target_common(mvm, peer, (void *)target);
|
||||
|
||||
if (vif->bss_conf.assoc &&
|
||||
!memcmp(peer->addr, vif->bss_conf.bssid, ETH_ALEN)) {
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
|
||||
target->sta_id = mvmvif->ap_sta_id;
|
||||
} else {
|
||||
target->sta_id = IWL_MVM_INVALID_STA;
|
||||
}
|
||||
|
||||
/*
|
||||
* TODO: Beacon interval is currently unknown, so use the common value
|
||||
* of 100 TUs.
|
||||
*/
|
||||
target->beacon_interval = cpu_to_le16(100);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int iwl_mvm_ftm_send_cmd(struct iwl_mvm *mvm, struct iwl_host_cmd *hcmd)
|
||||
{
|
||||
u32 status;
|
||||
@ -496,7 +528,7 @@ static int iwl_mvm_ftm_start_v8(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
for (i = 0; i < cmd.num_of_ap; i++) {
|
||||
struct cfg80211_pmsr_request_peer *peer = &req->peers[i];
|
||||
|
||||
err = iwl_mvm_ftm_put_target(mvm, peer, &cmd.ap[i]);
|
||||
err = iwl_mvm_ftm_put_target_v4(mvm, peer, &cmd.ap[i]);
|
||||
if (err)
|
||||
return err;
|
||||
}
|
||||
@ -521,8 +553,9 @@ static int iwl_mvm_ftm_start_v9(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
|
||||
for (i = 0; i < cmd.num_of_ap; i++) {
|
||||
struct cfg80211_pmsr_request_peer *peer = &req->peers[i];
|
||||
struct iwl_tof_range_req_ap_entry *target = &cmd.ap[i];
|
||||
|
||||
err = iwl_mvm_ftm_put_target(mvm, peer, (void *)&cmd.ap[i]);
|
||||
err = iwl_mvm_ftm_put_target(mvm, vif, peer, target);
|
||||
if (err)
|
||||
return err;
|
||||
}
|
||||
@ -548,6 +581,7 @@ int iwl_mvm_ftm_start(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
|
||||
switch (cmd_ver) {
|
||||
case 9:
|
||||
case 10:
|
||||
err = iwl_mvm_ftm_start_v9(mvm, vif, req);
|
||||
break;
|
||||
case 8:
|
||||
|
@ -988,6 +988,44 @@ static void iwl_mvm_tas_init(struct iwl_mvm *mvm)
|
||||
if (ret < 0)
|
||||
IWL_DEBUG_RADIO(mvm, "failed to send TAS_CONFIG (%d)\n", ret);
|
||||
}
|
||||
|
||||
static bool iwl_mvm_eval_dsm_indonesia_5g2(struct iwl_mvm *mvm)
|
||||
{
|
||||
int ret = iwl_acpi_get_dsm_u8((&mvm->fwrt)->dev, 0,
|
||||
DSM_FUNC_ENABLE_INDONESIA_5G2);
|
||||
|
||||
IWL_DEBUG_RADIO(mvm,
|
||||
"Evaluated DSM function ENABLE_INDONESIA_5G2, ret=%d\n",
|
||||
ret);
|
||||
|
||||
return ret == 1;
|
||||
}
|
||||
|
||||
static void iwl_mvm_lari_cfg(struct iwl_mvm *mvm)
|
||||
{
|
||||
int ret;
|
||||
struct iwl_lari_config_change_cmd cmd = {};
|
||||
|
||||
if (iwl_mvm_eval_dsm_indonesia_5g2(mvm))
|
||||
cmd.config_bitmap |=
|
||||
cpu_to_le32(LARI_CONFIG_ENABLE_5G2_IN_INDONESIA_MSK);
|
||||
|
||||
/* apply more config masks here */
|
||||
|
||||
if (cmd.config_bitmap) {
|
||||
IWL_DEBUG_RADIO(mvm,
|
||||
"sending LARI_CONFIG_CHANGE, config_bitmap=0x%x\n",
|
||||
le32_to_cpu(cmd.config_bitmap));
|
||||
ret = iwl_mvm_send_cmd_pdu(mvm,
|
||||
WIDE_ID(REGULATORY_AND_NVM_GROUP,
|
||||
LARI_CONFIG_CHANGE),
|
||||
0, sizeof(cmd), &cmd);
|
||||
if (ret < 0)
|
||||
IWL_DEBUG_RADIO(mvm,
|
||||
"Failed to send LARI_CONFIG_CHANGE (%d)\n",
|
||||
ret);
|
||||
}
|
||||
}
|
||||
#else /* CONFIG_ACPI */
|
||||
|
||||
inline int iwl_mvm_sar_select_profile(struct iwl_mvm *mvm,
|
||||
@ -1019,6 +1057,10 @@ static int iwl_mvm_ppag_init(struct iwl_mvm *mvm)
|
||||
static void iwl_mvm_tas_init(struct iwl_mvm *mvm)
|
||||
{
|
||||
}
|
||||
|
||||
static void iwl_mvm_lari_cfg(struct iwl_mvm *mvm)
|
||||
{
|
||||
}
|
||||
#endif /* CONFIG_ACPI */
|
||||
|
||||
void iwl_mvm_send_recovery_cmd(struct iwl_mvm *mvm, u32 flags)
|
||||
@ -1293,6 +1335,7 @@ int iwl_mvm_up(struct iwl_mvm *mvm)
|
||||
if (ret)
|
||||
goto error;
|
||||
|
||||
iwl_mvm_lari_cfg(mvm);
|
||||
/*
|
||||
* RTNL is not taken during Ct-kill, but we don't need to scan/Tx
|
||||
* anyway, so don't init MCC.
|
||||
|
@ -1208,14 +1208,13 @@ void __iwl_mvm_mac_stop(struct iwl_mvm *mvm)
|
||||
*/
|
||||
flush_work(&mvm->roc_done_wk);
|
||||
|
||||
iwl_mvm_rm_aux_sta(mvm);
|
||||
|
||||
iwl_mvm_stop_device(mvm);
|
||||
|
||||
iwl_mvm_async_handlers_purge(mvm);
|
||||
/* async_handlers_list is empty and will stay empty: HW is stopped */
|
||||
|
||||
/* the fw is stopped, the aux sta is dead: clean up driver state */
|
||||
iwl_mvm_del_aux_sta(mvm);
|
||||
|
||||
/*
|
||||
* Clear IN_HW_RESTART and HW_RESTART_REQUESTED flag when stopping the
|
||||
* hw (as restart_complete() won't be called in this case) and mac80211
|
||||
|
@ -2093,18 +2093,26 @@ int iwl_mvm_rm_snif_sta(struct iwl_mvm *mvm, struct ieee80211_vif *vif)
|
||||
return ret;
|
||||
}
|
||||
|
||||
int iwl_mvm_rm_aux_sta(struct iwl_mvm *mvm)
|
||||
{
|
||||
int ret;
|
||||
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
|
||||
iwl_mvm_disable_txq(mvm, NULL, mvm->aux_queue, IWL_MAX_TID_COUNT, 0);
|
||||
ret = iwl_mvm_rm_sta_common(mvm, mvm->aux_sta.sta_id);
|
||||
if (ret)
|
||||
IWL_WARN(mvm, "Failed sending remove station\n");
|
||||
iwl_mvm_dealloc_int_sta(mvm, &mvm->aux_sta);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
void iwl_mvm_dealloc_snif_sta(struct iwl_mvm *mvm)
|
||||
{
|
||||
iwl_mvm_dealloc_int_sta(mvm, &mvm->snif_sta);
|
||||
}
|
||||
|
||||
void iwl_mvm_del_aux_sta(struct iwl_mvm *mvm)
|
||||
{
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
|
||||
iwl_mvm_dealloc_int_sta(mvm, &mvm->aux_sta);
|
||||
}
|
||||
|
||||
/*
|
||||
* Send the add station command for the vif's broadcast station.
|
||||
* Assumes that the station was already allocated.
|
||||
|
@ -8,7 +8,7 @@
|
||||
* Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
|
||||
* Copyright(c) 2013 - 2014 Intel Mobile Communications GmbH
|
||||
* Copyright(c) 2015 - 2016 Intel Deutschland GmbH
|
||||
* Copyright(c) 2018 - 2019 Intel Corporation
|
||||
* Copyright(c) 2018 - 2020 Intel Corporation
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of version 2 of the GNU General Public License as
|
||||
@ -31,7 +31,7 @@
|
||||
* Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
|
||||
* Copyright(c) 2013 - 2014 Intel Mobile Communications GmbH
|
||||
* Copyright(c) 2015 - 2016 Intel Deutschland GmbH
|
||||
* Copyright(c) 2018 - 2019 Intel Corporation
|
||||
* Copyright(c) 2018 - 2020 Intel Corporation
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
@ -541,7 +541,7 @@ int iwl_mvm_sta_tx_agg(struct iwl_mvm *mvm, struct ieee80211_sta *sta,
|
||||
int tid, u8 queue, bool start);
|
||||
|
||||
int iwl_mvm_add_aux_sta(struct iwl_mvm *mvm);
|
||||
void iwl_mvm_del_aux_sta(struct iwl_mvm *mvm);
|
||||
int iwl_mvm_rm_aux_sta(struct iwl_mvm *mvm);
|
||||
|
||||
int iwl_mvm_alloc_bcast_sta(struct iwl_mvm *mvm, struct ieee80211_vif *vif);
|
||||
int iwl_mvm_send_add_bcast_sta(struct iwl_mvm *mvm, struct ieee80211_vif *vif);
|
||||
|
@ -138,9 +138,17 @@ int iwl_pcie_ctxt_info_gen3_init(struct iwl_trans *trans,
|
||||
case IWL_AMSDU_2K:
|
||||
break;
|
||||
case IWL_AMSDU_4K:
|
||||
control_flags |= IWL_PRPH_SCRATCH_RB_SIZE_4K;
|
||||
break;
|
||||
case IWL_AMSDU_8K:
|
||||
control_flags |= IWL_PRPH_SCRATCH_RB_SIZE_4K;
|
||||
/* if firmware supports the ext size, tell it */
|
||||
control_flags |= IWL_PRPH_SCRATCH_RB_SIZE_EXT_8K;
|
||||
break;
|
||||
case IWL_AMSDU_12K:
|
||||
control_flags |= IWL_PRPH_SCRATCH_RB_SIZE_4K;
|
||||
/* if firmware supports the ext size, tell it */
|
||||
control_flags |= IWL_PRPH_SCRATCH_RB_SIZE_EXT_12K;
|
||||
break;
|
||||
}
|
||||
|
||||
@ -213,7 +221,7 @@ int iwl_pcie_ctxt_info_gen3_init(struct iwl_trans *trans,
|
||||
ctxt_info_gen3->tr_idx_arr_size =
|
||||
cpu_to_le16(IWL_NUM_OF_TRANSFER_RINGS);
|
||||
ctxt_info_gen3->mtr_base_addr =
|
||||
cpu_to_le64(trans_pcie->txq[trans_pcie->cmd_queue]->dma_addr);
|
||||
cpu_to_le64(trans->txqs.txq[trans->txqs.cmd.q_id]->dma_addr);
|
||||
ctxt_info_gen3->mcr_base_addr =
|
||||
cpu_to_le64(trans_pcie->rxq->used_bd_dma);
|
||||
ctxt_info_gen3->mtr_size =
|
||||
|
@ -6,7 +6,7 @@
|
||||
* GPL LICENSE SUMMARY
|
||||
*
|
||||
* Copyright(c) 2017 Intel Deutschland GmbH
|
||||
* Copyright(c) 2018 - 2019 Intel Corporation
|
||||
* Copyright(c) 2018 - 2020 Intel Corporation
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of version 2 of the GNU General Public License as
|
||||
@ -20,7 +20,7 @@
|
||||
* BSD LICENSE
|
||||
*
|
||||
* Copyright(c) 2017 Intel Deutschland GmbH
|
||||
* Copyright(c) 2018 - 2019 Intel Corporation
|
||||
* Copyright(c) 2018 - 2020 Intel Corporation
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
@ -263,7 +263,7 @@ int iwl_pcie_ctxt_info_init(struct iwl_trans *trans,
|
||||
|
||||
/* initialize TX command queue */
|
||||
ctxt_info->hcmd_cfg.cmd_queue_addr =
|
||||
cpu_to_le64(trans_pcie->txq[trans_pcie->cmd_queue]->dma_addr);
|
||||
cpu_to_le64(trans->txqs.txq[trans->txqs.cmd.q_id]->dma_addr);
|
||||
ctxt_info->hcmd_cfg.cmd_queue_size =
|
||||
TFD_QUEUE_CB_SIZE(IWL_CMD_QUEUE_SIZE);
|
||||
|
||||
|
@ -246,12 +246,6 @@ struct iwl_rb_allocator {
|
||||
struct work_struct rx_alloc;
|
||||
};
|
||||
|
||||
struct iwl_dma_ptr {
|
||||
dma_addr_t dma;
|
||||
void *addr;
|
||||
size_t size;
|
||||
};
|
||||
|
||||
/**
|
||||
* iwl_queue_inc_wrap - increment queue index, wrap back to beginning
|
||||
* @index -- current index
|
||||
@ -290,107 +284,6 @@ static inline int iwl_queue_dec_wrap(struct iwl_trans *trans, int index)
|
||||
(trans->trans_cfg->base_params->max_tfd_queue_size - 1);
|
||||
}
|
||||
|
||||
struct iwl_cmd_meta {
|
||||
/* only for SYNC commands, iff the reply skb is wanted */
|
||||
struct iwl_host_cmd *source;
|
||||
u32 flags;
|
||||
u32 tbs;
|
||||
};
|
||||
|
||||
/*
|
||||
* The FH will write back to the first TB only, so we need to copy some data
|
||||
* into the buffer regardless of whether it should be mapped or not.
|
||||
* This indicates how big the first TB must be to include the scratch buffer
|
||||
* and the assigned PN.
|
||||
* Since PN location is 8 bytes at offset 12, it's 20 now.
|
||||
* If we make it bigger then allocations will be bigger and copy slower, so
|
||||
* that's probably not useful.
|
||||
*/
|
||||
#define IWL_FIRST_TB_SIZE 20
|
||||
#define IWL_FIRST_TB_SIZE_ALIGN ALIGN(IWL_FIRST_TB_SIZE, 64)
|
||||
|
||||
struct iwl_pcie_txq_entry {
|
||||
void *cmd;
|
||||
struct sk_buff *skb;
|
||||
/* buffer to free after command completes */
|
||||
const void *free_buf;
|
||||
struct iwl_cmd_meta meta;
|
||||
};
|
||||
|
||||
struct iwl_pcie_first_tb_buf {
|
||||
u8 buf[IWL_FIRST_TB_SIZE_ALIGN];
|
||||
};
|
||||
|
||||
/**
|
||||
* struct iwl_txq - Tx Queue for DMA
|
||||
* @q: generic Rx/Tx queue descriptor
|
||||
* @tfds: transmit frame descriptors (DMA memory)
|
||||
* @first_tb_bufs: start of command headers, including scratch buffers, for
|
||||
* the writeback -- this is DMA memory and an array holding one buffer
|
||||
* for each command on the queue
|
||||
* @first_tb_dma: DMA address for the first_tb_bufs start
|
||||
* @entries: transmit entries (driver state)
|
||||
* @lock: queue lock
|
||||
* @stuck_timer: timer that fires if queue gets stuck
|
||||
* @trans_pcie: pointer back to transport (for timer)
|
||||
* @need_update: indicates need to update read/write index
|
||||
* @ampdu: true if this queue is an ampdu queue for an specific RA/TID
|
||||
* @wd_timeout: queue watchdog timeout (jiffies) - per queue
|
||||
* @frozen: tx stuck queue timer is frozen
|
||||
* @frozen_expiry_remainder: remember how long until the timer fires
|
||||
* @bc_tbl: byte count table of the queue (relevant only for gen2 transport)
|
||||
* @write_ptr: 1-st empty entry (index) host_w
|
||||
* @read_ptr: last used entry (index) host_r
|
||||
* @dma_addr: physical addr for BD's
|
||||
* @n_window: safe queue window
|
||||
* @id: queue id
|
||||
* @low_mark: low watermark, resume queue if free space more than this
|
||||
* @high_mark: high watermark, stop queue if free space less than this
|
||||
*
|
||||
* A Tx queue consists of circular buffer of BDs (a.k.a. TFDs, transmit frame
|
||||
* descriptors) and required locking structures.
|
||||
*
|
||||
* Note the difference between TFD_QUEUE_SIZE_MAX and n_window: the hardware
|
||||
* always assumes 256 descriptors, so TFD_QUEUE_SIZE_MAX is always 256 (unless
|
||||
* there might be HW changes in the future). For the normal TX
|
||||
* queues, n_window, which is the size of the software queue data
|
||||
* is also 256; however, for the command queue, n_window is only
|
||||
* 32 since we don't need so many commands pending. Since the HW
|
||||
* still uses 256 BDs for DMA though, TFD_QUEUE_SIZE_MAX stays 256.
|
||||
* This means that we end up with the following:
|
||||
* HW entries: | 0 | ... | N * 32 | ... | N * 32 + 31 | ... | 255 |
|
||||
* SW entries: | 0 | ... | 31 |
|
||||
* where N is a number between 0 and 7. This means that the SW
|
||||
* data is a window overlayed over the HW queue.
|
||||
*/
|
||||
struct iwl_txq {
|
||||
void *tfds;
|
||||
struct iwl_pcie_first_tb_buf *first_tb_bufs;
|
||||
dma_addr_t first_tb_dma;
|
||||
struct iwl_pcie_txq_entry *entries;
|
||||
spinlock_t lock;
|
||||
unsigned long frozen_expiry_remainder;
|
||||
struct timer_list stuck_timer;
|
||||
struct iwl_trans_pcie *trans_pcie;
|
||||
bool need_update;
|
||||
bool frozen;
|
||||
bool ampdu;
|
||||
int block;
|
||||
unsigned long wd_timeout;
|
||||
struct sk_buff_head overflow_q;
|
||||
struct iwl_dma_ptr bc_tbl;
|
||||
|
||||
int write_ptr;
|
||||
int read_ptr;
|
||||
dma_addr_t dma_addr;
|
||||
int n_window;
|
||||
u32 id;
|
||||
int low_mark;
|
||||
int high_mark;
|
||||
|
||||
bool overflow_tx;
|
||||
};
|
||||
|
||||
static inline dma_addr_t
|
||||
iwl_pcie_get_first_tb_dma(struct iwl_txq *txq, int idx)
|
||||
{
|
||||
@ -561,9 +454,6 @@ struct iwl_trans_pcie {
|
||||
struct dma_pool *bc_pool;
|
||||
|
||||
struct iwl_txq *txq_memory;
|
||||
struct iwl_txq *txq[IWL_MAX_TVQM_QUEUES];
|
||||
unsigned long queue_used[BITS_TO_LONGS(IWL_MAX_TVQM_QUEUES)];
|
||||
unsigned long queue_stopped[BITS_TO_LONGS(IWL_MAX_TVQM_QUEUES)];
|
||||
|
||||
/* PCI bus related data */
|
||||
struct pci_dev *pci_dev;
|
||||
@ -577,10 +467,7 @@ struct iwl_trans_pcie {
|
||||
|
||||
u8 page_offs, dev_cmd_offs;
|
||||
|
||||
u8 cmd_queue;
|
||||
u8 def_rx_queue;
|
||||
u8 cmd_fifo;
|
||||
unsigned int cmd_q_wdg_timeout;
|
||||
u8 n_no_reclaim_cmds;
|
||||
u8 no_reclaim_cmds[MAX_NO_RECLAIM_CMDS];
|
||||
u8 max_tbs;
|
||||
@ -983,9 +870,7 @@ void iwl_pcie_handle_rfkill_irq(struct iwl_trans *trans);
|
||||
static inline void iwl_wake_queue(struct iwl_trans *trans,
|
||||
struct iwl_txq *txq)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
|
||||
if (test_and_clear_bit(txq->id, trans_pcie->queue_stopped)) {
|
||||
if (test_and_clear_bit(txq->id, trans->txqs.queue_stopped)) {
|
||||
IWL_DEBUG_TX_QUEUES(trans, "Wake hwq %d\n", txq->id);
|
||||
iwl_op_mode_queue_not_full(trans->op_mode, txq->id);
|
||||
}
|
||||
@ -994,9 +879,7 @@ static inline void iwl_wake_queue(struct iwl_trans *trans,
|
||||
static inline void iwl_stop_queue(struct iwl_trans *trans,
|
||||
struct iwl_txq *txq)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
|
||||
if (!test_and_set_bit(txq->id, trans_pcie->queue_stopped)) {
|
||||
if (!test_and_set_bit(txq->id, trans->txqs.queue_stopped)) {
|
||||
iwl_op_mode_queue_full(trans->op_mode, txq->id);
|
||||
IWL_DEBUG_TX_QUEUES(trans, "Stop hwq %d\n", txq->id);
|
||||
} else
|
||||
|
@ -1284,7 +1284,7 @@ static void iwl_pcie_rx_handle_rb(struct iwl_trans *trans,
|
||||
int i)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[trans_pcie->cmd_queue];
|
||||
struct iwl_txq *txq = trans->txqs.txq[trans->txqs.cmd.q_id];
|
||||
bool page_stolen = false;
|
||||
int max_len = trans_pcie->rx_buf_bytes;
|
||||
u32 offset = 0;
|
||||
@ -1671,9 +1671,9 @@ static void iwl_pcie_irq_handle_error(struct iwl_trans *trans)
|
||||
}
|
||||
|
||||
for (i = 0; i < trans->trans_cfg->base_params->num_of_queues; i++) {
|
||||
if (!trans_pcie->txq[i])
|
||||
if (!trans->txqs.txq[i])
|
||||
continue;
|
||||
del_timer(&trans_pcie->txq[i]->stuck_timer);
|
||||
del_timer(&trans->txqs.txq[i]->stuck_timer);
|
||||
}
|
||||
|
||||
/* The STATUS_FW_ERROR bit is set in this function. This must happen
|
||||
|
@ -6,7 +6,7 @@
|
||||
* GPL LICENSE SUMMARY
|
||||
*
|
||||
* Copyright(c) 2017 Intel Deutschland GmbH
|
||||
* Copyright(c) 2018 - 2019 Intel Corporation
|
||||
* Copyright(c) 2018 - 2020 Intel Corporation
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of version 2 of the GNU General Public License as
|
||||
@ -20,7 +20,7 @@
|
||||
* BSD LICENSE
|
||||
*
|
||||
* Copyright(c) 2017 Intel Deutschland GmbH
|
||||
* Copyright(c) 2018 - 2019 Intel Corporation
|
||||
* Copyright(c) 2018 - 2020 Intel Corporation
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
@ -245,7 +245,7 @@ static int iwl_pcie_gen2_nic_init(struct iwl_trans *trans)
|
||||
return -ENOMEM;
|
||||
|
||||
/* Allocate or reset and init all Tx and Command queues */
|
||||
if (iwl_pcie_gen2_tx_init(trans, trans_pcie->cmd_queue, queue_size))
|
||||
if (iwl_pcie_gen2_tx_init(trans, trans->txqs.cmd.q_id, queue_size))
|
||||
return -ENOMEM;
|
||||
|
||||
/* enable shadow regs in HW */
|
||||
@ -262,8 +262,9 @@ void iwl_trans_pcie_gen2_fw_alive(struct iwl_trans *trans, u32 scd_addr)
|
||||
iwl_pcie_reset_ict(trans);
|
||||
|
||||
/* make sure all queue are not stopped/used */
|
||||
memset(trans_pcie->queue_stopped, 0, sizeof(trans_pcie->queue_stopped));
|
||||
memset(trans_pcie->queue_used, 0, sizeof(trans_pcie->queue_used));
|
||||
memset(trans->txqs.queue_stopped, 0,
|
||||
sizeof(trans->txqs.queue_stopped));
|
||||
memset(trans->txqs.queue_used, 0, sizeof(trans->txqs.queue_used));
|
||||
|
||||
/* now that we got alive we can free the fw image & the context info.
|
||||
* paging memory cannot be freed included since FW will still use it
|
||||
|
@ -5,10 +5,9 @@
|
||||
*
|
||||
* GPL LICENSE SUMMARY
|
||||
*
|
||||
* Copyright(c) 2007 - 2015 Intel Corporation. All rights reserved.
|
||||
* Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
|
||||
* Copyright(c) 2016 - 2017 Intel Deutschland GmbH
|
||||
* Copyright(c) 2018 - 2019 Intel Corporation
|
||||
* Copyright(c) 2007 - 2015, 2018 - 2020 Intel Corporation
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of version 2 of the GNU General Public License as
|
||||
@ -28,10 +27,9 @@
|
||||
*
|
||||
* BSD LICENSE
|
||||
*
|
||||
* Copyright(c) 2005 - 2015 Intel Corporation. All rights reserved.
|
||||
* Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
|
||||
* Copyright(c) 2016 - 2017 Intel Deutschland GmbH
|
||||
* Copyright(c) 2018 - 2019 Intel Corporation
|
||||
* Copyright(c) 2007 - 2015, 2018 - 2020 Intel Corporation
|
||||
* All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
@ -1495,14 +1493,10 @@ static int iwl_trans_pcie_d3_suspend(struct iwl_trans *trans, bool test,
|
||||
int ret;
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
|
||||
/*
|
||||
* Family IWL_DEVICE_FAMILY_AX210 and above persist mode is set by FW.
|
||||
*/
|
||||
if (!reset && trans->trans_cfg->device_family < IWL_DEVICE_FAMILY_AX210) {
|
||||
if (!reset)
|
||||
/* Enable persistence mode to avoid reset */
|
||||
iwl_set_bit(trans, CSR_HW_IF_CONFIG_REG,
|
||||
CSR_HW_IF_CONFIG_REG_PERSIST_MODE);
|
||||
}
|
||||
|
||||
if (trans->trans_cfg->device_family >= IWL_DEVICE_FAMILY_AX210) {
|
||||
iwl_write_umac_prph(trans, UREG_DOORBELL_TO_ISR6,
|
||||
@ -1910,9 +1904,9 @@ static void iwl_trans_pcie_configure(struct iwl_trans *trans,
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
|
||||
trans_pcie->cmd_queue = trans_cfg->cmd_queue;
|
||||
trans_pcie->cmd_fifo = trans_cfg->cmd_fifo;
|
||||
trans_pcie->cmd_q_wdg_timeout = trans_cfg->cmd_q_wdg_timeout;
|
||||
trans->txqs.cmd.q_id = trans_cfg->cmd_queue;
|
||||
trans->txqs.cmd.fifo = trans_cfg->cmd_fifo;
|
||||
trans->txqs.cmd.wdg_timeout = trans_cfg->cmd_q_wdg_timeout;
|
||||
if (WARN_ON(trans_cfg->n_no_reclaim_cmds > MAX_NO_RECLAIM_CMDS))
|
||||
trans_pcie->n_no_reclaim_cmds = 0;
|
||||
else
|
||||
@ -2205,11 +2199,10 @@ static void iwl_trans_pcie_freeze_txq_timer(struct iwl_trans *trans,
|
||||
unsigned long txqs,
|
||||
bool freeze)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
int queue;
|
||||
|
||||
for_each_set_bit(queue, &txqs, BITS_PER_LONG) {
|
||||
struct iwl_txq *txq = trans_pcie->txq[queue];
|
||||
struct iwl_txq *txq = trans->txqs.txq[queue];
|
||||
unsigned long now;
|
||||
|
||||
spin_lock_bh(&txq->lock);
|
||||
@ -2257,13 +2250,12 @@ static void iwl_trans_pcie_freeze_txq_timer(struct iwl_trans *trans,
|
||||
|
||||
static void iwl_trans_pcie_block_txq_ptrs(struct iwl_trans *trans, bool block)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
int i;
|
||||
|
||||
for (i = 0; i < trans->trans_cfg->base_params->num_of_queues; i++) {
|
||||
struct iwl_txq *txq = trans_pcie->txq[i];
|
||||
struct iwl_txq *txq = trans->txqs.txq[i];
|
||||
|
||||
if (i == trans_pcie->cmd_queue)
|
||||
if (i == trans->txqs.cmd.q_id)
|
||||
continue;
|
||||
|
||||
spin_lock_bh(&txq->lock);
|
||||
@ -2332,7 +2324,6 @@ static int iwl_trans_pcie_rxq_dma_data(struct iwl_trans *trans, int queue,
|
||||
|
||||
static int iwl_trans_pcie_wait_txq_empty(struct iwl_trans *trans, int txq_idx)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq;
|
||||
unsigned long now = jiffies;
|
||||
bool overflow_tx;
|
||||
@ -2342,11 +2333,11 @@ static int iwl_trans_pcie_wait_txq_empty(struct iwl_trans *trans, int txq_idx)
|
||||
if (test_bit(STATUS_TRANS_DEAD, &trans->status))
|
||||
return -ENODEV;
|
||||
|
||||
if (!test_bit(txq_idx, trans_pcie->queue_used))
|
||||
if (!test_bit(txq_idx, trans->txqs.queue_used))
|
||||
return -EINVAL;
|
||||
|
||||
IWL_DEBUG_TX_QUEUES(trans, "Emptying queue %d...\n", txq_idx);
|
||||
txq = trans_pcie->txq[txq_idx];
|
||||
txq = trans->txqs.txq[txq_idx];
|
||||
|
||||
spin_lock_bh(&txq->lock);
|
||||
overflow_tx = txq->overflow_tx ||
|
||||
@ -2394,7 +2385,6 @@ static int iwl_trans_pcie_wait_txq_empty(struct iwl_trans *trans, int txq_idx)
|
||||
|
||||
static int iwl_trans_pcie_wait_txqs_empty(struct iwl_trans *trans, u32 txq_bm)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
int cnt;
|
||||
int ret = 0;
|
||||
|
||||
@ -2403,9 +2393,9 @@ static int iwl_trans_pcie_wait_txqs_empty(struct iwl_trans *trans, u32 txq_bm)
|
||||
cnt < trans->trans_cfg->base_params->num_of_queues;
|
||||
cnt++) {
|
||||
|
||||
if (cnt == trans_pcie->cmd_queue)
|
||||
if (cnt == trans->txqs.cmd.q_id)
|
||||
continue;
|
||||
if (!test_bit(cnt, trans_pcie->queue_used))
|
||||
if (!test_bit(cnt, trans->txqs.queue_used))
|
||||
continue;
|
||||
if (!(BIT(cnt) & txq_bm))
|
||||
continue;
|
||||
@ -2579,13 +2569,12 @@ static int iwl_dbgfs_tx_queue_seq_show(struct seq_file *seq, void *v)
|
||||
struct iwl_dbgfs_tx_queue_priv *priv = seq->private;
|
||||
struct iwl_dbgfs_tx_queue_state *state = v;
|
||||
struct iwl_trans *trans = priv->trans;
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[state->pos];
|
||||
struct iwl_txq *txq = trans->txqs.txq[state->pos];
|
||||
|
||||
seq_printf(seq, "hwq %.3u: used=%d stopped=%d ",
|
||||
(unsigned int)state->pos,
|
||||
!!test_bit(state->pos, trans_pcie->queue_used),
|
||||
!!test_bit(state->pos, trans_pcie->queue_stopped));
|
||||
!!test_bit(state->pos, trans->txqs.queue_used),
|
||||
!!test_bit(state->pos, trans->txqs.queue_stopped));
|
||||
if (txq)
|
||||
seq_printf(seq,
|
||||
"read=%u write=%u need_update=%d frozen=%d n_window=%d ampdu=%d",
|
||||
@ -2595,7 +2584,7 @@ static int iwl_dbgfs_tx_queue_seq_show(struct seq_file *seq, void *v)
|
||||
else
|
||||
seq_puts(seq, "(unallocated)");
|
||||
|
||||
if (state->pos == trans_pcie->cmd_queue)
|
||||
if (state->pos == trans->txqs.cmd.q_id)
|
||||
seq_puts(seq, " (HCMD)");
|
||||
seq_puts(seq, "\n");
|
||||
|
||||
@ -3271,7 +3260,7 @@ static struct iwl_trans_dump_data
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_fw_error_dump_data *data;
|
||||
struct iwl_txq *cmdq = trans_pcie->txq[trans_pcie->cmd_queue];
|
||||
struct iwl_txq *cmdq = trans->txqs.txq[trans->txqs.cmd.q_id];
|
||||
struct iwl_fw_error_dump_txcmd *txcmd;
|
||||
struct iwl_trans_dump_data *dump_data;
|
||||
u32 len, num_rbs = 0, monitor_len = 0;
|
||||
|
@ -64,7 +64,6 @@
|
||||
*/
|
||||
void iwl_pcie_gen2_tx_stop(struct iwl_trans *trans)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
int txq_id;
|
||||
|
||||
/*
|
||||
@ -72,12 +71,13 @@ void iwl_pcie_gen2_tx_stop(struct iwl_trans *trans)
|
||||
* queues. This happens when we have an rfkill interrupt.
|
||||
* Since we stop Tx altogether - mark the queues as stopped.
|
||||
*/
|
||||
memset(trans_pcie->queue_stopped, 0, sizeof(trans_pcie->queue_stopped));
|
||||
memset(trans_pcie->queue_used, 0, sizeof(trans_pcie->queue_used));
|
||||
memset(trans->txqs.queue_stopped, 0,
|
||||
sizeof(trans->txqs.queue_stopped));
|
||||
memset(trans->txqs.queue_used, 0, sizeof(trans->txqs.queue_used));
|
||||
|
||||
/* Unmap DMA from host system and free skb's */
|
||||
for (txq_id = 0; txq_id < ARRAY_SIZE(trans_pcie->txq); txq_id++) {
|
||||
if (!trans_pcie->txq[txq_id])
|
||||
for (txq_id = 0; txq_id < ARRAY_SIZE(trans->txqs.txq); txq_id++) {
|
||||
if (!trans->txqs.txq[txq_id])
|
||||
continue;
|
||||
iwl_pcie_gen2_txq_unmap(trans, txq_id);
|
||||
}
|
||||
@ -716,7 +716,7 @@ int iwl_trans_pcie_gen2_tx(struct iwl_trans *trans, struct sk_buff *skb,
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_cmd_meta *out_meta;
|
||||
struct iwl_txq *txq = trans_pcie->txq[txq_id];
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
u16 cmd_len;
|
||||
int idx;
|
||||
void *tfd;
|
||||
@ -725,7 +725,7 @@ int iwl_trans_pcie_gen2_tx(struct iwl_trans *trans, struct sk_buff *skb,
|
||||
"queue %d out of range", txq_id))
|
||||
return -EINVAL;
|
||||
|
||||
if (WARN_ONCE(!test_bit(txq_id, trans_pcie->queue_used),
|
||||
if (WARN_ONCE(!test_bit(txq_id, trans->txqs.queue_used),
|
||||
"TX on unused queue %d\n", txq_id))
|
||||
return -EINVAL;
|
||||
|
||||
@ -819,7 +819,7 @@ static int iwl_pcie_gen2_enqueue_hcmd(struct iwl_trans *trans,
|
||||
struct iwl_host_cmd *cmd)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[trans_pcie->cmd_queue];
|
||||
struct iwl_txq *txq = trans->txqs.txq[trans->txqs.cmd.q_id];
|
||||
struct iwl_device_cmd *out_cmd;
|
||||
struct iwl_cmd_meta *out_meta;
|
||||
unsigned long flags;
|
||||
@ -931,7 +931,7 @@ static int iwl_pcie_gen2_enqueue_hcmd(struct iwl_trans *trans,
|
||||
cpu_to_le16(cmd_size - sizeof(struct iwl_cmd_header_wide));
|
||||
out_cmd->hdr_wide.reserved = 0;
|
||||
out_cmd->hdr_wide.sequence =
|
||||
cpu_to_le16(QUEUE_TO_SEQ(trans_pcie->cmd_queue) |
|
||||
cpu_to_le16(QUEUE_TO_SEQ(trans->txqs.cmd.q_id) |
|
||||
INDEX_TO_SEQ(txq->write_ptr));
|
||||
|
||||
cmd_pos = sizeof(struct iwl_cmd_header_wide);
|
||||
@ -979,7 +979,7 @@ static int iwl_pcie_gen2_enqueue_hcmd(struct iwl_trans *trans,
|
||||
"Sending command %s (%.2x.%.2x), seq: 0x%04X, %d bytes at %d[%d]:%d\n",
|
||||
iwl_get_cmd_string(trans, cmd->id), group_id,
|
||||
out_cmd->hdr.cmd, le16_to_cpu(out_cmd->hdr.sequence),
|
||||
cmd_size, txq->write_ptr, idx, trans_pcie->cmd_queue);
|
||||
cmd_size, txq->write_ptr, idx, trans->txqs.cmd.q_id);
|
||||
|
||||
/* start the TFD with the minimum copy bytes */
|
||||
tb0_size = min_t(int, copy_size, IWL_FIRST_TB_SIZE);
|
||||
@ -1056,7 +1056,7 @@ static int iwl_pcie_gen2_send_hcmd_sync(struct iwl_trans *trans,
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
const char *cmd_str = iwl_get_cmd_string(trans, cmd->id);
|
||||
struct iwl_txq *txq = trans_pcie->txq[trans_pcie->cmd_queue];
|
||||
struct iwl_txq *txq = trans->txqs.txq[trans->txqs.cmd.q_id];
|
||||
int cmd_idx;
|
||||
int ret;
|
||||
|
||||
@ -1175,14 +1175,14 @@ int iwl_trans_pcie_gen2_send_hcmd(struct iwl_trans *trans,
|
||||
void iwl_pcie_gen2_txq_unmap(struct iwl_trans *trans, int txq_id)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[txq_id];
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
|
||||
spin_lock_bh(&txq->lock);
|
||||
while (txq->write_ptr != txq->read_ptr) {
|
||||
IWL_DEBUG_TX_REPLY(trans, "Q %d Free %d\n",
|
||||
txq_id, txq->read_ptr);
|
||||
|
||||
if (txq_id != trans_pcie->cmd_queue) {
|
||||
if (txq_id != trans->txqs.cmd.q_id) {
|
||||
int idx = iwl_pcie_get_cmd_index(txq, txq->read_ptr);
|
||||
struct sk_buff *skb = txq->entries[idx].skb;
|
||||
|
||||
@ -1240,7 +1240,6 @@ void iwl_pcie_gen2_txq_free_memory(struct iwl_trans *trans,
|
||||
*/
|
||||
static void iwl_pcie_gen2_txq_free(struct iwl_trans *trans, int txq_id)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq;
|
||||
int i;
|
||||
|
||||
@ -1248,7 +1247,7 @@ static void iwl_pcie_gen2_txq_free(struct iwl_trans *trans, int txq_id)
|
||||
"queue %d out of range", txq_id))
|
||||
return;
|
||||
|
||||
txq = trans_pcie->txq[txq_id];
|
||||
txq = trans->txqs.txq[txq_id];
|
||||
|
||||
if (WARN_ON(!txq))
|
||||
return;
|
||||
@ -1256,7 +1255,7 @@ static void iwl_pcie_gen2_txq_free(struct iwl_trans *trans, int txq_id)
|
||||
iwl_pcie_gen2_txq_unmap(trans, txq_id);
|
||||
|
||||
/* De-alloc array of command/tx buffers */
|
||||
if (txq_id == trans_pcie->cmd_queue)
|
||||
if (txq_id == trans->txqs.cmd.q_id)
|
||||
for (i = 0; i < txq->n_window; i++) {
|
||||
kzfree(txq->entries[i].cmd);
|
||||
kzfree(txq->entries[i].free_buf);
|
||||
@ -1265,9 +1264,9 @@ static void iwl_pcie_gen2_txq_free(struct iwl_trans *trans, int txq_id)
|
||||
|
||||
iwl_pcie_gen2_txq_free_memory(trans, txq);
|
||||
|
||||
trans_pcie->txq[txq_id] = NULL;
|
||||
trans->txqs.txq[txq_id] = NULL;
|
||||
|
||||
clear_bit(txq_id, trans_pcie->queue_used);
|
||||
clear_bit(txq_id, trans->txqs.queue_used);
|
||||
}
|
||||
|
||||
int iwl_trans_pcie_dyn_txq_alloc_dma(struct iwl_trans *trans,
|
||||
@ -1327,7 +1326,6 @@ int iwl_trans_pcie_txq_alloc_response(struct iwl_trans *trans,
|
||||
struct iwl_txq *txq,
|
||||
struct iwl_host_cmd *hcmd)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_tx_queue_cfg_rsp *rsp;
|
||||
int ret, qid;
|
||||
u32 wr_ptr;
|
||||
@ -1342,20 +1340,20 @@ int iwl_trans_pcie_txq_alloc_response(struct iwl_trans *trans,
|
||||
qid = le16_to_cpu(rsp->queue_number);
|
||||
wr_ptr = le16_to_cpu(rsp->write_pointer);
|
||||
|
||||
if (qid >= ARRAY_SIZE(trans_pcie->txq)) {
|
||||
if (qid >= ARRAY_SIZE(trans->txqs.txq)) {
|
||||
WARN_ONCE(1, "queue index %d unsupported", qid);
|
||||
ret = -EIO;
|
||||
goto error_free_resp;
|
||||
}
|
||||
|
||||
if (test_and_set_bit(qid, trans_pcie->queue_used)) {
|
||||
if (test_and_set_bit(qid, trans->txqs.queue_used)) {
|
||||
WARN_ONCE(1, "queue %d already used", qid);
|
||||
ret = -EIO;
|
||||
goto error_free_resp;
|
||||
}
|
||||
|
||||
txq->id = qid;
|
||||
trans_pcie->txq[qid] = txq;
|
||||
trans->txqs.txq[qid] = txq;
|
||||
wr_ptr &= (trans->trans_cfg->base_params->max_tfd_queue_size - 1);
|
||||
|
||||
/* Place first TFD at index corresponding to start sequence number */
|
||||
@ -1413,8 +1411,6 @@ int iwl_trans_pcie_dyn_txq_alloc(struct iwl_trans *trans,
|
||||
|
||||
void iwl_trans_pcie_dyn_txq_free(struct iwl_trans *trans, int queue)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
|
||||
if (WARN(queue >= IWL_MAX_TVQM_QUEUES,
|
||||
"queue %d out of range", queue))
|
||||
return;
|
||||
@ -1425,7 +1421,7 @@ void iwl_trans_pcie_dyn_txq_free(struct iwl_trans *trans, int queue)
|
||||
* allow the op_mode to call txq_disable after it already called
|
||||
* stop_device.
|
||||
*/
|
||||
if (!test_and_clear_bit(queue, trans_pcie->queue_used)) {
|
||||
if (!test_and_clear_bit(queue, trans->txqs.queue_used)) {
|
||||
WARN_ONCE(test_bit(STATUS_DEVICE_ENABLED, &trans->status),
|
||||
"queue %d not used", queue);
|
||||
return;
|
||||
@ -1433,22 +1429,21 @@ void iwl_trans_pcie_dyn_txq_free(struct iwl_trans *trans, int queue)
|
||||
|
||||
iwl_pcie_gen2_txq_unmap(trans, queue);
|
||||
|
||||
iwl_pcie_gen2_txq_free_memory(trans, trans_pcie->txq[queue]);
|
||||
trans_pcie->txq[queue] = NULL;
|
||||
iwl_pcie_gen2_txq_free_memory(trans, trans->txqs.txq[queue]);
|
||||
trans->txqs.txq[queue] = NULL;
|
||||
|
||||
IWL_DEBUG_TX_QUEUES(trans, "Deactivate queue %d\n", queue);
|
||||
}
|
||||
|
||||
void iwl_pcie_gen2_tx_free(struct iwl_trans *trans)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
int i;
|
||||
|
||||
memset(trans_pcie->queue_used, 0, sizeof(trans_pcie->queue_used));
|
||||
memset(trans->txqs.queue_used, 0, sizeof(trans->txqs.queue_used));
|
||||
|
||||
/* Free all TX queues */
|
||||
for (i = 0; i < ARRAY_SIZE(trans_pcie->txq); i++) {
|
||||
if (!trans_pcie->txq[i])
|
||||
for (i = 0; i < ARRAY_SIZE(trans->txqs.txq); i++) {
|
||||
if (!trans->txqs.txq[i])
|
||||
continue;
|
||||
|
||||
iwl_pcie_gen2_txq_free(trans, i);
|
||||
@ -1457,35 +1452,34 @@ void iwl_pcie_gen2_tx_free(struct iwl_trans *trans)
|
||||
|
||||
int iwl_pcie_gen2_tx_init(struct iwl_trans *trans, int txq_id, int queue_size)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *queue;
|
||||
int ret;
|
||||
|
||||
/* alloc and init the tx queue */
|
||||
if (!trans_pcie->txq[txq_id]) {
|
||||
if (!trans->txqs.txq[txq_id]) {
|
||||
queue = kzalloc(sizeof(*queue), GFP_KERNEL);
|
||||
if (!queue) {
|
||||
IWL_ERR(trans, "Not enough memory for tx queue\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
trans_pcie->txq[txq_id] = queue;
|
||||
trans->txqs.txq[txq_id] = queue;
|
||||
ret = iwl_pcie_txq_alloc(trans, queue, queue_size, true);
|
||||
if (ret) {
|
||||
IWL_ERR(trans, "Tx %d queue init failed\n", txq_id);
|
||||
goto error;
|
||||
}
|
||||
} else {
|
||||
queue = trans_pcie->txq[txq_id];
|
||||
queue = trans->txqs.txq[txq_id];
|
||||
}
|
||||
|
||||
ret = iwl_pcie_txq_init(trans, queue, queue_size,
|
||||
(txq_id == trans_pcie->cmd_queue));
|
||||
(txq_id == trans->txqs.cmd.q_id));
|
||||
if (ret) {
|
||||
IWL_ERR(trans, "Tx %d queue alloc failed\n", txq_id);
|
||||
goto error;
|
||||
}
|
||||
trans_pcie->txq[txq_id]->id = txq_id;
|
||||
set_bit(txq_id, trans_pcie->queue_used);
|
||||
trans->txqs.txq[txq_id]->id = txq_id;
|
||||
set_bit(txq_id, trans->txqs.queue_used);
|
||||
|
||||
return 0;
|
||||
|
||||
|
@ -183,8 +183,7 @@ void iwl_pcie_free_dma_ptr(struct iwl_trans *trans, struct iwl_dma_ptr *ptr)
|
||||
static void iwl_pcie_txq_stuck_timer(struct timer_list *t)
|
||||
{
|
||||
struct iwl_txq *txq = from_timer(txq, t, stuck_timer);
|
||||
struct iwl_trans_pcie *trans_pcie = txq->trans_pcie;
|
||||
struct iwl_trans *trans = iwl_trans_pcie_get_trans(trans_pcie);
|
||||
struct iwl_trans *trans = txq->trans;
|
||||
|
||||
spin_lock(&txq->lock);
|
||||
/* check if triggered erroneously */
|
||||
@ -262,7 +261,7 @@ static void iwl_pcie_txq_inval_byte_cnt_tbl(struct iwl_trans *trans,
|
||||
|
||||
WARN_ON(read_ptr >= TFD_QUEUE_SIZE_MAX);
|
||||
|
||||
if (txq_id != trans_pcie->cmd_queue)
|
||||
if (txq_id != trans->txqs.cmd.q_id)
|
||||
sta_id = tx_cmd->sta_id;
|
||||
|
||||
bc_ent = cpu_to_le16(1 | (sta_id << 12));
|
||||
@ -280,7 +279,6 @@ static void iwl_pcie_txq_inval_byte_cnt_tbl(struct iwl_trans *trans,
|
||||
static void iwl_pcie_txq_inc_wr_ptr(struct iwl_trans *trans,
|
||||
struct iwl_txq *txq)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
u32 reg = 0;
|
||||
int txq_id = txq->id;
|
||||
|
||||
@ -293,7 +291,7 @@ static void iwl_pcie_txq_inc_wr_ptr(struct iwl_trans *trans,
|
||||
* 3. there is a chance that the NIC is asleep
|
||||
*/
|
||||
if (!trans->trans_cfg->base_params->shadow_reg_enable &&
|
||||
txq_id != trans_pcie->cmd_queue &&
|
||||
txq_id != trans->txqs.cmd.q_id &&
|
||||
test_bit(STATUS_TPOWER_PMI, &trans->status)) {
|
||||
/*
|
||||
* wake up nic if it's powered down ...
|
||||
@ -324,13 +322,12 @@ static void iwl_pcie_txq_inc_wr_ptr(struct iwl_trans *trans,
|
||||
|
||||
void iwl_pcie_txq_check_wrptrs(struct iwl_trans *trans)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
int i;
|
||||
|
||||
for (i = 0; i < trans->trans_cfg->base_params->num_of_queues; i++) {
|
||||
struct iwl_txq *txq = trans_pcie->txq[i];
|
||||
struct iwl_txq *txq = trans->txqs.txq[i];
|
||||
|
||||
if (!test_bit(i, trans_pcie->queue_used))
|
||||
if (!test_bit(i, trans->txqs.queue_used))
|
||||
continue;
|
||||
|
||||
spin_lock_bh(&txq->lock);
|
||||
@ -535,7 +532,7 @@ int iwl_pcie_txq_alloc(struct iwl_trans *trans, struct iwl_txq *txq,
|
||||
tfd_sz = trans_pcie->tfd_size * slots_num;
|
||||
|
||||
timer_setup(&txq->stuck_timer, iwl_pcie_txq_stuck_timer, 0);
|
||||
txq->trans_pcie = trans_pcie;
|
||||
txq->trans = trans;
|
||||
|
||||
txq->n_window = slots_num;
|
||||
|
||||
@ -661,14 +658,14 @@ static void iwl_pcie_clear_cmd_in_flight(struct iwl_trans *trans)
|
||||
static void iwl_pcie_txq_unmap(struct iwl_trans *trans, int txq_id)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[txq_id];
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
|
||||
spin_lock_bh(&txq->lock);
|
||||
while (txq->write_ptr != txq->read_ptr) {
|
||||
IWL_DEBUG_TX_REPLY(trans, "Q %d Free %d\n",
|
||||
txq_id, txq->read_ptr);
|
||||
|
||||
if (txq_id != trans_pcie->cmd_queue) {
|
||||
if (txq_id != trans->txqs.cmd.q_id) {
|
||||
struct sk_buff *skb = txq->entries[txq->read_ptr].skb;
|
||||
|
||||
if (WARN_ON_ONCE(!skb))
|
||||
@ -683,7 +680,7 @@ static void iwl_pcie_txq_unmap(struct iwl_trans *trans, int txq_id)
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&trans_pcie->reg_lock, flags);
|
||||
if (txq_id == trans_pcie->cmd_queue)
|
||||
if (txq_id == trans->txqs.cmd.q_id)
|
||||
iwl_pcie_clear_cmd_in_flight(trans);
|
||||
spin_unlock_irqrestore(&trans_pcie->reg_lock, flags);
|
||||
}
|
||||
@ -712,7 +709,7 @@ static void iwl_pcie_txq_unmap(struct iwl_trans *trans, int txq_id)
|
||||
static void iwl_pcie_txq_free(struct iwl_trans *trans, int txq_id)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[txq_id];
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
struct device *dev = trans->dev;
|
||||
int i;
|
||||
|
||||
@ -722,7 +719,7 @@ static void iwl_pcie_txq_free(struct iwl_trans *trans, int txq_id)
|
||||
iwl_pcie_txq_unmap(trans, txq_id);
|
||||
|
||||
/* De-alloc array of command/tx buffers */
|
||||
if (txq_id == trans_pcie->cmd_queue)
|
||||
if (txq_id == trans->txqs.cmd.q_id)
|
||||
for (i = 0; i < txq->n_window; i++) {
|
||||
kzfree(txq->entries[i].cmd);
|
||||
kzfree(txq->entries[i].free_buf);
|
||||
@ -761,8 +758,9 @@ void iwl_pcie_tx_start(struct iwl_trans *trans, u32 scd_base_addr)
|
||||
SCD_CONTEXT_MEM_LOWER_BOUND) / sizeof(u32);
|
||||
|
||||
/* make sure all queue are not stopped/used */
|
||||
memset(trans_pcie->queue_stopped, 0, sizeof(trans_pcie->queue_stopped));
|
||||
memset(trans_pcie->queue_used, 0, sizeof(trans_pcie->queue_used));
|
||||
memset(trans->txqs.queue_stopped, 0,
|
||||
sizeof(trans->txqs.queue_stopped));
|
||||
memset(trans->txqs.queue_used, 0, sizeof(trans->txqs.queue_used));
|
||||
|
||||
trans_pcie->scd_base_addr =
|
||||
iwl_read_prph(trans, SCD_SRAM_BASE_ADDR);
|
||||
@ -784,9 +782,9 @@ void iwl_pcie_tx_start(struct iwl_trans *trans, u32 scd_base_addr)
|
||||
if (trans->trans_cfg->base_params->scd_chain_ext_wa)
|
||||
iwl_write_prph(trans, SCD_CHAINEXT_EN, 0);
|
||||
|
||||
iwl_trans_ac_txq_enable(trans, trans_pcie->cmd_queue,
|
||||
trans_pcie->cmd_fifo,
|
||||
trans_pcie->cmd_q_wdg_timeout);
|
||||
iwl_trans_ac_txq_enable(trans, trans->txqs.cmd.q_id,
|
||||
trans->txqs.cmd.fifo,
|
||||
trans->txqs.cmd.wdg_timeout);
|
||||
|
||||
/* Activate all Tx DMA/FIFO channels */
|
||||
iwl_scd_activate_fifos(trans);
|
||||
@ -822,7 +820,7 @@ void iwl_trans_pcie_tx_reset(struct iwl_trans *trans)
|
||||
|
||||
for (txq_id = 0; txq_id < trans->trans_cfg->base_params->num_of_queues;
|
||||
txq_id++) {
|
||||
struct iwl_txq *txq = trans_pcie->txq[txq_id];
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
if (trans->trans_cfg->use_tfh)
|
||||
iwl_write_direct64(trans,
|
||||
FH_MEM_CBBC_QUEUE(trans, txq_id),
|
||||
@ -898,8 +896,9 @@ int iwl_pcie_tx_stop(struct iwl_trans *trans)
|
||||
* queues. This happens when we have an rfkill interrupt.
|
||||
* Since we stop Tx altogether - mark the queues as stopped.
|
||||
*/
|
||||
memset(trans_pcie->queue_stopped, 0, sizeof(trans_pcie->queue_stopped));
|
||||
memset(trans_pcie->queue_used, 0, sizeof(trans_pcie->queue_used));
|
||||
memset(trans->txqs.queue_stopped, 0,
|
||||
sizeof(trans->txqs.queue_stopped));
|
||||
memset(trans->txqs.queue_used, 0, sizeof(trans->txqs.queue_used));
|
||||
|
||||
/* This can happen: start_hw, stop_device */
|
||||
if (!trans_pcie->txq_memory)
|
||||
@ -923,7 +922,7 @@ void iwl_pcie_tx_free(struct iwl_trans *trans)
|
||||
int txq_id;
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
|
||||
memset(trans_pcie->queue_used, 0, sizeof(trans_pcie->queue_used));
|
||||
memset(trans->txqs.queue_used, 0, sizeof(trans->txqs.queue_used));
|
||||
|
||||
/* Tx queues */
|
||||
if (trans_pcie->txq_memory) {
|
||||
@ -931,7 +930,7 @@ void iwl_pcie_tx_free(struct iwl_trans *trans)
|
||||
txq_id < trans->trans_cfg->base_params->num_of_queues;
|
||||
txq_id++) {
|
||||
iwl_pcie_txq_free(trans, txq_id);
|
||||
trans_pcie->txq[txq_id] = NULL;
|
||||
trans->txqs.txq[txq_id] = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
@ -992,7 +991,7 @@ static int iwl_pcie_tx_alloc(struct iwl_trans *trans)
|
||||
/* Alloc and init all Tx queues, including the command queue (#4/#9) */
|
||||
for (txq_id = 0; txq_id < trans->trans_cfg->base_params->num_of_queues;
|
||||
txq_id++) {
|
||||
bool cmd_queue = (txq_id == trans_pcie->cmd_queue);
|
||||
bool cmd_queue = (txq_id == trans->txqs.cmd.q_id);
|
||||
|
||||
if (cmd_queue)
|
||||
slots_num = max_t(u32, IWL_CMD_QUEUE_SIZE,
|
||||
@ -1000,14 +999,14 @@ static int iwl_pcie_tx_alloc(struct iwl_trans *trans)
|
||||
else
|
||||
slots_num = max_t(u32, IWL_DEFAULT_QUEUE_SIZE,
|
||||
trans->cfg->min_256_ba_txq_size);
|
||||
trans_pcie->txq[txq_id] = &trans_pcie->txq_memory[txq_id];
|
||||
ret = iwl_pcie_txq_alloc(trans, trans_pcie->txq[txq_id],
|
||||
trans->txqs.txq[txq_id] = &trans_pcie->txq_memory[txq_id];
|
||||
ret = iwl_pcie_txq_alloc(trans, trans->txqs.txq[txq_id],
|
||||
slots_num, cmd_queue);
|
||||
if (ret) {
|
||||
IWL_ERR(trans, "Tx %d queue alloc failed\n", txq_id);
|
||||
goto error;
|
||||
}
|
||||
trans_pcie->txq[txq_id]->id = txq_id;
|
||||
trans->txqs.txq[txq_id]->id = txq_id;
|
||||
}
|
||||
|
||||
return 0;
|
||||
@ -1046,7 +1045,7 @@ int iwl_pcie_tx_init(struct iwl_trans *trans)
|
||||
/* Alloc and init all Tx queues, including the command queue (#4/#9) */
|
||||
for (txq_id = 0; txq_id < trans->trans_cfg->base_params->num_of_queues;
|
||||
txq_id++) {
|
||||
bool cmd_queue = (txq_id == trans_pcie->cmd_queue);
|
||||
bool cmd_queue = (txq_id == trans->txqs.cmd.q_id);
|
||||
|
||||
if (cmd_queue)
|
||||
slots_num = max_t(u32, IWL_CMD_QUEUE_SIZE,
|
||||
@ -1054,7 +1053,7 @@ int iwl_pcie_tx_init(struct iwl_trans *trans)
|
||||
else
|
||||
slots_num = max_t(u32, IWL_DEFAULT_QUEUE_SIZE,
|
||||
trans->cfg->min_256_ba_txq_size);
|
||||
ret = iwl_pcie_txq_init(trans, trans_pcie->txq[txq_id],
|
||||
ret = iwl_pcie_txq_init(trans, trans->txqs.txq[txq_id],
|
||||
slots_num, cmd_queue);
|
||||
if (ret) {
|
||||
IWL_ERR(trans, "Tx %d queue init failed\n", txq_id);
|
||||
@ -1068,7 +1067,7 @@ int iwl_pcie_tx_init(struct iwl_trans *trans)
|
||||
* Circular buffer (TFD queue in DRAM) physical base address
|
||||
*/
|
||||
iwl_write_direct32(trans, FH_MEM_CBBC_QUEUE(trans, txq_id),
|
||||
trans_pcie->txq[txq_id]->dma_addr >> 8);
|
||||
trans->txqs.txq[txq_id]->dma_addr >> 8);
|
||||
}
|
||||
|
||||
iwl_set_bits_prph(trans, SCD_GP_CTRL, SCD_GP_CTRL_AUTO_ACTIVE_MODE);
|
||||
@ -1113,18 +1112,18 @@ void iwl_trans_pcie_reclaim(struct iwl_trans *trans, int txq_id, int ssn,
|
||||
struct sk_buff_head *skbs)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[txq_id];
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
int tfd_num = iwl_pcie_get_cmd_index(txq, ssn);
|
||||
int read_ptr = iwl_pcie_get_cmd_index(txq, txq->read_ptr);
|
||||
int last_to_free;
|
||||
|
||||
/* This function is not meant to release cmd queue*/
|
||||
if (WARN_ON(txq_id == trans_pcie->cmd_queue))
|
||||
if (WARN_ON(txq_id == trans->txqs.cmd.q_id))
|
||||
return;
|
||||
|
||||
spin_lock_bh(&txq->lock);
|
||||
|
||||
if (!test_bit(txq_id, trans_pcie->queue_used)) {
|
||||
if (!test_bit(txq_id, trans->txqs.queue_used)) {
|
||||
IWL_DEBUG_TX_QUEUES(trans, "Q %d inactive - ignoring idx %d\n",
|
||||
txq_id, ssn);
|
||||
goto out;
|
||||
@ -1176,7 +1175,7 @@ void iwl_trans_pcie_reclaim(struct iwl_trans *trans, int txq_id, int ssn,
|
||||
iwl_pcie_txq_progress(txq);
|
||||
|
||||
if (iwl_queue_space(trans, txq) > txq->low_mark &&
|
||||
test_bit(txq_id, trans_pcie->queue_stopped)) {
|
||||
test_bit(txq_id, trans->txqs.queue_stopped)) {
|
||||
struct sk_buff_head overflow_skbs;
|
||||
|
||||
__skb_queue_head_init(&overflow_skbs);
|
||||
@ -1229,8 +1228,7 @@ void iwl_trans_pcie_reclaim(struct iwl_trans *trans, int txq_id, int ssn,
|
||||
/* Set wr_ptr of specific device and txq */
|
||||
void iwl_trans_pcie_set_q_ptrs(struct iwl_trans *trans, int txq_id, int ptr)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[txq_id];
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
|
||||
spin_lock_bh(&txq->lock);
|
||||
|
||||
@ -1290,7 +1288,7 @@ static int iwl_pcie_set_cmd_in_flight(struct iwl_trans *trans,
|
||||
static void iwl_pcie_cmdq_reclaim(struct iwl_trans *trans, int txq_id, int idx)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[txq_id];
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
unsigned long flags;
|
||||
int nfreed = 0;
|
||||
u16 r;
|
||||
@ -1302,7 +1300,7 @@ static void iwl_pcie_cmdq_reclaim(struct iwl_trans *trans, int txq_id, int idx)
|
||||
|
||||
if (idx >= trans->trans_cfg->base_params->max_tfd_queue_size ||
|
||||
(!iwl_queue_used(txq, idx))) {
|
||||
WARN_ONCE(test_bit(txq_id, trans_pcie->queue_used),
|
||||
WARN_ONCE(test_bit(txq_id, trans->txqs.queue_used),
|
||||
"%s: Read index for DMA queue txq id (%d), index %d is out of range [0-%d] %d %d.\n",
|
||||
__func__, txq_id, idx,
|
||||
trans->trans_cfg->base_params->max_tfd_queue_size,
|
||||
@ -1364,11 +1362,11 @@ bool iwl_trans_pcie_txq_enable(struct iwl_trans *trans, int txq_id, u16 ssn,
|
||||
unsigned int wdg_timeout)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[txq_id];
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
int fifo = -1;
|
||||
bool scd_bug = false;
|
||||
|
||||
if (test_and_set_bit(txq_id, trans_pcie->queue_used))
|
||||
if (test_and_set_bit(txq_id, trans->txqs.queue_used))
|
||||
WARN_ONCE(1, "queue %d already used - expect issues", txq_id);
|
||||
|
||||
txq->wd_timeout = msecs_to_jiffies(wdg_timeout);
|
||||
@ -1377,7 +1375,7 @@ bool iwl_trans_pcie_txq_enable(struct iwl_trans *trans, int txq_id, u16 ssn,
|
||||
fifo = cfg->fifo;
|
||||
|
||||
/* Disable the scheduler prior configuring the cmd queue */
|
||||
if (txq_id == trans_pcie->cmd_queue &&
|
||||
if (txq_id == trans->txqs.cmd.q_id &&
|
||||
trans_pcie->scd_set_active)
|
||||
iwl_scd_enable_set_active(trans, 0);
|
||||
|
||||
@ -1385,7 +1383,7 @@ bool iwl_trans_pcie_txq_enable(struct iwl_trans *trans, int txq_id, u16 ssn,
|
||||
iwl_scd_txq_set_inactive(trans, txq_id);
|
||||
|
||||
/* Set this queue as a chain-building queue unless it is CMD */
|
||||
if (txq_id != trans_pcie->cmd_queue)
|
||||
if (txq_id != trans->txqs.cmd.q_id)
|
||||
iwl_scd_txq_set_chain(trans, txq_id);
|
||||
|
||||
if (cfg->aggregate) {
|
||||
@ -1455,7 +1453,7 @@ bool iwl_trans_pcie_txq_enable(struct iwl_trans *trans, int txq_id, u16 ssn,
|
||||
SCD_QUEUE_STTS_REG_MSK);
|
||||
|
||||
/* enable the scheduler for this queue (only) */
|
||||
if (txq_id == trans_pcie->cmd_queue &&
|
||||
if (txq_id == trans->txqs.cmd.q_id &&
|
||||
trans_pcie->scd_set_active)
|
||||
iwl_scd_enable_set_active(trans, BIT(txq_id));
|
||||
|
||||
@ -1474,8 +1472,7 @@ bool iwl_trans_pcie_txq_enable(struct iwl_trans *trans, int txq_id, u16 ssn,
|
||||
void iwl_trans_pcie_txq_set_shared_mode(struct iwl_trans *trans, u32 txq_id,
|
||||
bool shared_mode)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[txq_id];
|
||||
struct iwl_txq *txq = trans->txqs.txq[txq_id];
|
||||
|
||||
txq->ampdu = !shared_mode;
|
||||
}
|
||||
@ -1488,8 +1485,8 @@ void iwl_trans_pcie_txq_disable(struct iwl_trans *trans, int txq_id,
|
||||
SCD_TX_STTS_QUEUE_OFFSET(txq_id);
|
||||
static const u32 zero_val[4] = {};
|
||||
|
||||
trans_pcie->txq[txq_id]->frozen_expiry_remainder = 0;
|
||||
trans_pcie->txq[txq_id]->frozen = false;
|
||||
trans->txqs.txq[txq_id]->frozen_expiry_remainder = 0;
|
||||
trans->txqs.txq[txq_id]->frozen = false;
|
||||
|
||||
/*
|
||||
* Upon HW Rfkill - we stop the device, and then stop the queues
|
||||
@ -1497,7 +1494,7 @@ void iwl_trans_pcie_txq_disable(struct iwl_trans *trans, int txq_id,
|
||||
* allow the op_mode to call txq_disable after it already called
|
||||
* stop_device.
|
||||
*/
|
||||
if (!test_and_clear_bit(txq_id, trans_pcie->queue_used)) {
|
||||
if (!test_and_clear_bit(txq_id, trans->txqs.queue_used)) {
|
||||
WARN_ONCE(test_bit(STATUS_DEVICE_ENABLED, &trans->status),
|
||||
"queue %d not used", txq_id);
|
||||
return;
|
||||
@ -1511,7 +1508,7 @@ void iwl_trans_pcie_txq_disable(struct iwl_trans *trans, int txq_id,
|
||||
}
|
||||
|
||||
iwl_pcie_txq_unmap(trans, txq_id);
|
||||
trans_pcie->txq[txq_id]->ampdu = false;
|
||||
trans->txqs.txq[txq_id]->ampdu = false;
|
||||
|
||||
IWL_DEBUG_TX_QUEUES(trans, "Deactivate queue %d\n", txq_id);
|
||||
}
|
||||
@ -1531,7 +1528,7 @@ static int iwl_pcie_enqueue_hcmd(struct iwl_trans *trans,
|
||||
struct iwl_host_cmd *cmd)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[trans_pcie->cmd_queue];
|
||||
struct iwl_txq *txq = trans->txqs.txq[trans->txqs.cmd.q_id];
|
||||
struct iwl_device_cmd *out_cmd;
|
||||
struct iwl_cmd_meta *out_meta;
|
||||
unsigned long flags;
|
||||
@ -1657,7 +1654,7 @@ static int iwl_pcie_enqueue_hcmd(struct iwl_trans *trans,
|
||||
sizeof(struct iwl_cmd_header_wide));
|
||||
out_cmd->hdr_wide.reserved = 0;
|
||||
out_cmd->hdr_wide.sequence =
|
||||
cpu_to_le16(QUEUE_TO_SEQ(trans_pcie->cmd_queue) |
|
||||
cpu_to_le16(QUEUE_TO_SEQ(trans->txqs.cmd.q_id) |
|
||||
INDEX_TO_SEQ(txq->write_ptr));
|
||||
|
||||
cmd_pos = sizeof(struct iwl_cmd_header_wide);
|
||||
@ -1665,7 +1662,7 @@ static int iwl_pcie_enqueue_hcmd(struct iwl_trans *trans,
|
||||
} else {
|
||||
out_cmd->hdr.cmd = iwl_cmd_opcode(cmd->id);
|
||||
out_cmd->hdr.sequence =
|
||||
cpu_to_le16(QUEUE_TO_SEQ(trans_pcie->cmd_queue) |
|
||||
cpu_to_le16(QUEUE_TO_SEQ(trans->txqs.cmd.q_id) |
|
||||
INDEX_TO_SEQ(txq->write_ptr));
|
||||
out_cmd->hdr.group_id = 0;
|
||||
|
||||
@ -1716,7 +1713,7 @@ static int iwl_pcie_enqueue_hcmd(struct iwl_trans *trans,
|
||||
iwl_get_cmd_string(trans, cmd->id),
|
||||
group_id, out_cmd->hdr.cmd,
|
||||
le16_to_cpu(out_cmd->hdr.sequence),
|
||||
cmd_size, txq->write_ptr, idx, trans_pcie->cmd_queue);
|
||||
cmd_size, txq->write_ptr, idx, trans->txqs.cmd.q_id);
|
||||
|
||||
/* start the TFD with the minimum copy bytes */
|
||||
tb0_size = min_t(int, copy_size, IWL_FIRST_TB_SIZE);
|
||||
@ -1816,14 +1813,14 @@ void iwl_pcie_hcmd_complete(struct iwl_trans *trans,
|
||||
struct iwl_device_cmd *cmd;
|
||||
struct iwl_cmd_meta *meta;
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[trans_pcie->cmd_queue];
|
||||
struct iwl_txq *txq = trans->txqs.txq[trans->txqs.cmd.q_id];
|
||||
|
||||
/* If a Tx command is being handled and it isn't in the actual
|
||||
* command queue then there a command routing bug has been introduced
|
||||
* in the queue management code. */
|
||||
if (WARN(txq_id != trans_pcie->cmd_queue,
|
||||
if (WARN(txq_id != trans->txqs.cmd.q_id,
|
||||
"wrong command queue %d (should be %d), sequence 0x%X readp=%d writep=%d\n",
|
||||
txq_id, trans_pcie->cmd_queue, sequence, txq->read_ptr,
|
||||
txq_id, trans->txqs.cmd.q_id, sequence, txq->read_ptr,
|
||||
txq->write_ptr)) {
|
||||
iwl_print_hex_error(trans, pkt, 32);
|
||||
return;
|
||||
@ -1895,7 +1892,7 @@ static int iwl_pcie_send_hcmd_sync(struct iwl_trans *trans,
|
||||
struct iwl_host_cmd *cmd)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_txq *txq = trans_pcie->txq[trans_pcie->cmd_queue];
|
||||
struct iwl_txq *txq = trans->txqs.txq[trans->txqs.cmd.q_id];
|
||||
int cmd_idx;
|
||||
int ret;
|
||||
|
||||
@ -2129,7 +2126,8 @@ static int iwl_fill_data_tbs_amsdu(struct iwl_trans *trans, struct sk_buff *skb,
|
||||
u16 tb1_len)
|
||||
{
|
||||
struct iwl_tx_cmd *tx_cmd = (void *)dev_cmd->payload;
|
||||
struct iwl_trans_pcie *trans_pcie = txq->trans_pcie;
|
||||
struct iwl_trans_pcie *trans_pcie =
|
||||
IWL_TRANS_GET_PCIE_TRANS(txq->trans);
|
||||
struct ieee80211_hdr *hdr = (void *)skb->data;
|
||||
unsigned int snap_ip_tcp_hdrlen, ip_hdrlen, total_len, hdr_room;
|
||||
unsigned int mss = skb_shinfo(skb)->gso_size;
|
||||
@ -2332,9 +2330,9 @@ int iwl_trans_pcie_tx(struct iwl_trans *trans, struct sk_buff *skb,
|
||||
u16 wifi_seq;
|
||||
bool amsdu;
|
||||
|
||||
txq = trans_pcie->txq[txq_id];
|
||||
txq = trans->txqs.txq[txq_id];
|
||||
|
||||
if (WARN_ONCE(!test_bit(txq_id, trans_pcie->queue_used),
|
||||
if (WARN_ONCE(!test_bit(txq_id, trans->txqs.queue_used),
|
||||
"TX on unused queue %d\n", txq_id))
|
||||
return -EINVAL;
|
||||
|
||||
|
@ -62,9 +62,6 @@ int lbs_process_rxed_packet(struct lbs_private *priv, struct sk_buff *skb)
|
||||
struct rxpd *p_rx_pd;
|
||||
int hdrchop;
|
||||
struct ethhdr *p_ethhdr;
|
||||
static const u8 rfc1042_eth_hdr[] = {
|
||||
0xaa, 0xaa, 0x03, 0x00, 0x00, 0x00
|
||||
};
|
||||
|
||||
BUG_ON(!skb);
|
||||
|
||||
@ -102,7 +99,7 @@ int lbs_process_rxed_packet(struct lbs_private *priv, struct sk_buff *skb)
|
||||
sizeof(p_rx_pkt->eth803_hdr.src_addr));
|
||||
|
||||
if (memcmp(&p_rx_pkt->rfc1042_hdr,
|
||||
rfc1042_eth_hdr, sizeof(rfc1042_eth_hdr)) == 0) {
|
||||
rfc1042_header, sizeof(rfc1042_header)) == 0) {
|
||||
/*
|
||||
* Replace the 803 header and rfc1042 header (llc/snap) with an
|
||||
* EthernetII header, keep the src/dst and snap_type (ethertype)
|
||||
|
@ -4335,6 +4335,11 @@ int mwifiex_register_cfg80211(struct mwifiex_adapter *adapter)
|
||||
wiphy->iface_combinations = &mwifiex_iface_comb_ap_sta;
|
||||
wiphy->n_iface_combinations = 1;
|
||||
|
||||
if (adapter->max_sta_conn > adapter->max_p2p_conn)
|
||||
wiphy->max_ap_assoc_sta = adapter->max_sta_conn;
|
||||
else
|
||||
wiphy->max_ap_assoc_sta = adapter->max_p2p_conn;
|
||||
|
||||
/* Initialize cipher suits */
|
||||
wiphy->cipher_suites = mwifiex_cipher_suites;
|
||||
wiphy->n_cipher_suites = ARRAY_SIZE(mwifiex_cipher_suites);
|
||||
|
@ -1495,6 +1495,7 @@ int mwifiex_ret_get_hw_spec(struct mwifiex_private *priv,
|
||||
struct mwifiex_adapter *adapter = priv->adapter;
|
||||
struct mwifiex_ie_types_header *tlv;
|
||||
struct hw_spec_api_rev *api_rev;
|
||||
struct hw_spec_max_conn *max_conn;
|
||||
u16 resp_size, api_id;
|
||||
int i, left_len, parsed_len = 0;
|
||||
|
||||
@ -1581,8 +1582,21 @@ int mwifiex_ret_get_hw_spec(struct mwifiex_private *priv,
|
||||
adapter->fw_api_ver =
|
||||
api_rev->major_ver;
|
||||
mwifiex_dbg(adapter, INFO,
|
||||
"Firmware api version %d\n",
|
||||
adapter->fw_api_ver);
|
||||
"Firmware api version %d.%d\n",
|
||||
adapter->fw_api_ver,
|
||||
api_rev->minor_ver);
|
||||
break;
|
||||
case UAP_FW_API_VER_ID:
|
||||
mwifiex_dbg(adapter, INFO,
|
||||
"uAP api version %d.%d\n",
|
||||
api_rev->major_ver,
|
||||
api_rev->minor_ver);
|
||||
break;
|
||||
case CHANRPT_API_VER_ID:
|
||||
mwifiex_dbg(adapter, INFO,
|
||||
"channel report api version %d.%d\n",
|
||||
api_rev->major_ver,
|
||||
api_rev->minor_ver);
|
||||
break;
|
||||
default:
|
||||
mwifiex_dbg(adapter, FATAL,
|
||||
@ -1591,6 +1605,17 @@ int mwifiex_ret_get_hw_spec(struct mwifiex_private *priv,
|
||||
break;
|
||||
}
|
||||
break;
|
||||
case TLV_TYPE_MAX_CONN:
|
||||
max_conn = (struct hw_spec_max_conn *)tlv;
|
||||
adapter->max_p2p_conn = max_conn->max_p2p_conn;
|
||||
adapter->max_sta_conn = max_conn->max_sta_conn;
|
||||
mwifiex_dbg(adapter, INFO,
|
||||
"max p2p connections: %u\n",
|
||||
adapter->max_p2p_conn);
|
||||
mwifiex_dbg(adapter, INFO,
|
||||
"max sta connections: %u\n",
|
||||
adapter->max_sta_conn);
|
||||
break;
|
||||
default:
|
||||
mwifiex_dbg(adapter, FATAL,
|
||||
"Unknown GET_HW_SPEC TLV type: %#x\n",
|
||||
|
@ -220,6 +220,7 @@ enum MWIFIEX_802_11_PRIVACY_FILTER {
|
||||
#define TLV_TYPE_BSS_MODE (PROPRIETARY_TLV_BASE_ID + 206)
|
||||
#define TLV_TYPE_RANDOM_MAC (PROPRIETARY_TLV_BASE_ID + 236)
|
||||
#define TLV_TYPE_CHAN_ATTR_CFG (PROPRIETARY_TLV_BASE_ID + 237)
|
||||
#define TLV_TYPE_MAX_CONN (PROPRIETARY_TLV_BASE_ID + 279)
|
||||
|
||||
#define MWIFIEX_TX_DATA_BUF_SIZE_2K 2048
|
||||
|
||||
@ -1052,6 +1053,8 @@ struct host_cmd_ds_802_11_ps_mode_enh {
|
||||
enum API_VER_ID {
|
||||
KEY_API_VER_ID = 1,
|
||||
FW_API_VER_ID = 2,
|
||||
UAP_FW_API_VER_ID = 3,
|
||||
CHANRPT_API_VER_ID = 4,
|
||||
};
|
||||
|
||||
struct hw_spec_api_rev {
|
||||
@ -2386,4 +2389,11 @@ struct mwifiex_opt_sleep_confirm {
|
||||
__le16 action;
|
||||
__le16 resp_ctrl;
|
||||
} __packed;
|
||||
|
||||
struct hw_spec_max_conn {
|
||||
struct mwifiex_ie_types_header header;
|
||||
u8 max_p2p_conn;
|
||||
u8 max_sta_conn;
|
||||
} __packed;
|
||||
|
||||
#endif /* !_MWIFIEX_FW_H_ */
|
||||
|
@ -1022,6 +1022,7 @@ struct mwifiex_adapter {
|
||||
bool ext_scan;
|
||||
u8 fw_api_ver;
|
||||
u8 key_api_major_ver, key_api_minor_ver;
|
||||
u8 max_p2p_conn, max_sta_conn;
|
||||
struct memory_type_mapping *mem_type_mapping_tbl;
|
||||
u8 num_mem_types;
|
||||
bool scan_chan_gap_enabled;
|
||||
|
@ -51,7 +51,7 @@ static int mt76_rx_queues_read(struct seq_file *s, void *data)
|
||||
struct mt76_dev *dev = dev_get_drvdata(s->private);
|
||||
int i, queued;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->q_rx); i++) {
|
||||
mt76_for_each_q_rx(dev, i) {
|
||||
struct mt76_queue *q = &dev->q_rx[i];
|
||||
|
||||
if (!q->ndesc)
|
||||
|
@ -576,7 +576,7 @@ mt76_dma_init(struct mt76_dev *dev)
|
||||
|
||||
init_dummy_netdev(&dev->napi_dev);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->q_rx); i++) {
|
||||
mt76_for_each_q_rx(dev, i) {
|
||||
netif_napi_add(&dev->napi_dev, &dev->napi[i], mt76_dma_rx_poll,
|
||||
64);
|
||||
mt76_dma_rx_fill(dev, &dev->q_rx[i]);
|
||||
@ -610,7 +610,7 @@ void mt76_dma_cleanup(struct mt76_dev *dev)
|
||||
for (i = 0; i < ARRAY_SIZE(dev->q_tx); i++)
|
||||
mt76_dma_tx_cleanup(dev, i, true);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->q_rx); i++) {
|
||||
mt76_for_each_q_rx(dev, i) {
|
||||
netif_napi_del(&dev->napi[i]);
|
||||
mt76_dma_rx_cleanup(dev, &dev->q_rx[i]);
|
||||
}
|
||||
|
@ -288,8 +288,8 @@ enum {
|
||||
MT76_REMOVED,
|
||||
MT76_READING_STATS,
|
||||
MT76_STATE_POWER_OFF,
|
||||
MT76_STATE_PS,
|
||||
MT76_STATE_SUSPEND,
|
||||
MT76_STATE_ROC,
|
||||
};
|
||||
|
||||
struct mt76_hw_cap {
|
||||
@ -537,8 +537,8 @@ struct mt76_dev {
|
||||
wait_queue_head_t tx_wait;
|
||||
struct sk_buff_head status_list;
|
||||
|
||||
unsigned long wcid_mask[MT76_N_WCIDS / BITS_PER_LONG];
|
||||
unsigned long wcid_phy_mask[MT76_N_WCIDS / BITS_PER_LONG];
|
||||
u32 wcid_mask[DIV_ROUND_UP(MT76_N_WCIDS, 32)];
|
||||
u32 wcid_phy_mask[DIV_ROUND_UP(MT76_N_WCIDS, 32)];
|
||||
|
||||
struct mt76_wcid global_wcid;
|
||||
struct mt76_wcid __rcu *wcid[MT76_N_WCIDS];
|
||||
@ -671,6 +671,10 @@ static inline u16 mt76_rev(struct mt76_dev *dev)
|
||||
#define mt76_queue_tx_cleanup(dev, ...) (dev)->mt76.queue_ops->tx_cleanup(&((dev)->mt76), __VA_ARGS__)
|
||||
#define mt76_queue_kick(dev, ...) (dev)->mt76.queue_ops->kick(&((dev)->mt76), __VA_ARGS__)
|
||||
|
||||
#define mt76_for_each_q_rx(dev, i) \
|
||||
for (i = 0; i < ARRAY_SIZE((dev)->q_rx) && \
|
||||
(dev)->q_rx[i].ndesc; i++)
|
||||
|
||||
struct mt76_dev *mt76_alloc_device(struct device *pdev, unsigned int size,
|
||||
const struct ieee80211_ops *ops,
|
||||
const struct mt76_driver_ops *drv_ops);
|
||||
|
@ -473,7 +473,7 @@ mt7603_rx_get_wcid(struct mt7603_dev *dev, u8 idx, bool unicast)
|
||||
struct mt7603_sta *sta;
|
||||
struct mt76_wcid *wcid;
|
||||
|
||||
if (idx >= ARRAY_SIZE(dev->mt76.wcid))
|
||||
if (idx >= MT7603_WTBL_SIZE)
|
||||
return NULL;
|
||||
|
||||
wcid = rcu_dereference(dev->mt76.wcid[idx]);
|
||||
@ -1238,7 +1238,7 @@ void mt7603_mac_add_txs(struct mt7603_dev *dev, void *data)
|
||||
if (pid == MT_PACKET_ID_NO_ACK)
|
||||
return;
|
||||
|
||||
if (wcidx >= ARRAY_SIZE(dev->mt76.wcid))
|
||||
if (wcidx >= MT7603_WTBL_SIZE)
|
||||
return;
|
||||
|
||||
rcu_read_lock();
|
||||
@ -1438,8 +1438,9 @@ static void mt7603_mac_watchdog_reset(struct mt7603_dev *dev)
|
||||
for (i = 0; i < __MT_TXQ_MAX; i++)
|
||||
mt76_queue_tx_cleanup(dev, i, true);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->mt76.q_rx); i++)
|
||||
mt76_for_each_q_rx(&dev->mt76, i) {
|
||||
mt76_queue_rx_reset(dev, i);
|
||||
}
|
||||
|
||||
mt7603_dma_sched_reset(dev);
|
||||
|
||||
|
@ -111,6 +111,12 @@ mt7615_eeprom_parse_hw_band_cap(struct mt7615_dev *dev)
|
||||
return;
|
||||
}
|
||||
|
||||
if (is_mt7611(&dev->mt76)) {
|
||||
/* 5GHz only */
|
||||
dev->mt76.cap.has_5ghz = true;
|
||||
return;
|
||||
}
|
||||
|
||||
val = FIELD_GET(MT_EE_NIC_WIFI_CONF_BAND_SEL,
|
||||
eeprom[MT_EE_WIFI_CONF]);
|
||||
switch (val) {
|
||||
@ -310,6 +316,7 @@ static void mt7615_cal_free_data(struct mt7615_dev *dev)
|
||||
mt7622_apply_cal_free_data(dev);
|
||||
break;
|
||||
case 0x7615:
|
||||
case 0x7611:
|
||||
mt7615_apply_cal_free_data(dev);
|
||||
break;
|
||||
}
|
||||
|
@ -139,8 +139,10 @@ void mt7615_check_offload_capability(struct mt7615_dev *dev)
|
||||
ieee80211_hw_set(hw, SUPPORTS_PS);
|
||||
ieee80211_hw_set(hw, SUPPORTS_DYNAMIC_PS);
|
||||
|
||||
wiphy->max_remain_on_channel_duration = 5000;
|
||||
wiphy->features |= NL80211_FEATURE_SCHED_SCAN_RANDOM_MAC_ADDR |
|
||||
NL80211_FEATURE_SCAN_RANDOM_MAC_ADDR |
|
||||
WIPHY_FLAG_HAS_REMAIN_ON_CHANNEL |
|
||||
NL80211_FEATURE_P2P_GO_CTWIN |
|
||||
NL80211_FEATURE_P2P_GO_OPPPS;
|
||||
} else {
|
||||
@ -149,6 +151,8 @@ void mt7615_check_offload_capability(struct mt7615_dev *dev)
|
||||
dev->ops->sched_scan_start = NULL;
|
||||
dev->ops->sched_scan_stop = NULL;
|
||||
dev->ops->set_rekey_data = NULL;
|
||||
dev->ops->remain_on_channel = NULL;
|
||||
dev->ops->cancel_remain_on_channel = NULL;
|
||||
|
||||
wiphy->max_sched_scan_plan_interval = 0;
|
||||
wiphy->max_sched_scan_ie_len = 0;
|
||||
@ -368,12 +372,6 @@ int mt7615_register_ext_phy(struct mt7615_dev *dev)
|
||||
if (phy)
|
||||
return 0;
|
||||
|
||||
INIT_DELAYED_WORK(&phy->mac_work, mt7615_mac_work);
|
||||
INIT_DELAYED_WORK(&phy->scan_work, mt7615_scan_work);
|
||||
skb_queue_head_init(&phy->scan_event_list);
|
||||
|
||||
INIT_WORK(&phy->ps_work, mt7615_ps_work);
|
||||
|
||||
mt7615_cap_dbdc_enable(dev);
|
||||
mphy = mt76_alloc_phy(&dev->mt76, sizeof(*phy), &mt7615_ops);
|
||||
if (!mphy)
|
||||
@ -386,6 +384,14 @@ int mt7615_register_ext_phy(struct mt7615_dev *dev)
|
||||
mphy->antenna_mask = BIT(hweight8(phy->chainmask)) - 1;
|
||||
mt7615_init_wiphy(mphy->hw);
|
||||
|
||||
INIT_DELAYED_WORK(&phy->mac_work, mt7615_mac_work);
|
||||
INIT_DELAYED_WORK(&phy->scan_work, mt7615_scan_work);
|
||||
skb_queue_head_init(&phy->scan_event_list);
|
||||
|
||||
INIT_WORK(&phy->roc_work, mt7615_roc_work);
|
||||
timer_setup(&phy->roc_timer, mt7615_roc_timer, 0);
|
||||
init_waitqueue_head(&phy->roc_wait);
|
||||
|
||||
mt7615_mac_set_scs(phy, true);
|
||||
|
||||
/*
|
||||
@ -437,9 +443,11 @@ void mt7615_init_device(struct mt7615_dev *dev)
|
||||
INIT_LIST_HEAD(&dev->sta_poll_list);
|
||||
spin_lock_init(&dev->sta_poll_lock);
|
||||
init_waitqueue_head(&dev->reset_wait);
|
||||
init_waitqueue_head(&dev->phy.roc_wait);
|
||||
|
||||
INIT_WORK(&dev->reset_work, mt7615_mac_reset_work);
|
||||
INIT_WORK(&dev->phy.ps_work, mt7615_ps_work);
|
||||
INIT_WORK(&dev->phy.roc_work, mt7615_roc_work);
|
||||
timer_setup(&dev->phy.roc_timer, mt7615_roc_timer, 0);
|
||||
|
||||
mt7615_init_wiphy(hw);
|
||||
dev->mphy.sband_2g.sband.ht_cap.cap |= IEEE80211_HT_CAP_LDPC_CODING;
|
||||
|
@ -61,7 +61,7 @@ static struct mt76_wcid *mt7615_rx_get_wcid(struct mt7615_dev *dev,
|
||||
struct mt7615_sta *sta;
|
||||
struct mt76_wcid *wcid;
|
||||
|
||||
if (idx >= ARRAY_SIZE(dev->mt76.wcid))
|
||||
if (idx >= MT7615_WTBL_SIZE)
|
||||
return NULL;
|
||||
|
||||
wcid = rcu_dereference(dev->mt76.wcid[idx]);
|
||||
@ -175,7 +175,8 @@ mt7615_get_status_freq_info(struct mt7615_dev *dev, struct mt76_phy *mphy,
|
||||
struct mt76_rx_status *status, u8 chfreq)
|
||||
{
|
||||
if (!test_bit(MT76_HW_SCANNING, &mphy->state) &&
|
||||
!test_bit(MT76_HW_SCHED_SCANNING, &mphy->state)) {
|
||||
!test_bit(MT76_HW_SCHED_SCANNING, &mphy->state) &&
|
||||
!test_bit(MT76_STATE_ROC, &mphy->state)) {
|
||||
status->freq = mphy->chandef.chan->center_freq;
|
||||
status->band = mphy->chandef.chan->band;
|
||||
return;
|
||||
@ -1302,7 +1303,7 @@ static void mt7615_mac_add_txs(struct mt7615_dev *dev, void *data)
|
||||
if (pid == MT_PACKET_ID_NO_ACK)
|
||||
return;
|
||||
|
||||
if (wcidx >= ARRAY_SIZE(dev->mt76.wcid))
|
||||
if (wcidx >= MT7615_WTBL_SIZE)
|
||||
return;
|
||||
|
||||
rcu_read_lock();
|
||||
@ -1819,8 +1820,9 @@ void mt7615_dma_reset(struct mt7615_dev *dev)
|
||||
for (i = 0; i < __MT_TXQ_MAX; i++)
|
||||
mt76_queue_tx_cleanup(dev, i, true);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->mt76.q_rx); i++)
|
||||
mt76_for_each_q_rx(&dev->mt76, i) {
|
||||
mt76_queue_rx_reset(dev, i);
|
||||
}
|
||||
|
||||
mt76_set(dev, MT_WPDMA_GLO_CFG,
|
||||
MT_WPDMA_GLO_CFG_RX_DMA_EN | MT_WPDMA_GLO_CFG_TX_DMA_EN |
|
||||
@ -1849,8 +1851,13 @@ void mt7615_mac_reset_work(struct work_struct *work)
|
||||
set_bit(MT76_MCU_RESET, &dev->mphy.state);
|
||||
wake_up(&dev->mt76.mcu.wait);
|
||||
cancel_delayed_work_sync(&dev->phy.mac_work);
|
||||
if (phy2)
|
||||
del_timer_sync(&dev->phy.roc_timer);
|
||||
cancel_work_sync(&dev->phy.roc_work);
|
||||
if (phy2) {
|
||||
cancel_delayed_work_sync(&phy2->mac_work);
|
||||
del_timer_sync(&phy2->roc_timer);
|
||||
cancel_work_sync(&phy2->roc_work);
|
||||
}
|
||||
|
||||
/* lock/unlock all queues to ensure that no tx is pending */
|
||||
mt76_txq_schedule_all(&dev->mphy);
|
||||
|
@ -71,7 +71,8 @@ static void mt7615_stop(struct ieee80211_hw *hw)
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
|
||||
cancel_delayed_work_sync(&phy->mac_work);
|
||||
cancel_work_sync(&phy->ps_work);
|
||||
del_timer_sync(&phy->roc_timer);
|
||||
cancel_work_sync(&phy->roc_work);
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
|
||||
@ -289,12 +290,11 @@ mt7615_queue_key_update(struct mt7615_dev *dev, enum set_key_cmd cmd,
|
||||
wd->type = MT7615_WTBL_KEY_DESC;
|
||||
wd->sta = msta;
|
||||
|
||||
wd->key.key = kzalloc(key->keylen, GFP_KERNEL);
|
||||
wd->key.key = kmemdup(key->key, key->keylen, GFP_KERNEL);
|
||||
if (!wd->key.key) {
|
||||
kfree(wd);
|
||||
return -ENOMEM;
|
||||
}
|
||||
memcpy(wd->key.key, key->key, key->keylen);
|
||||
wd->key.cipher = key->cipher;
|
||||
wd->key.keyidx = key->keyidx;
|
||||
wd->key.keylen = key->keylen;
|
||||
@ -360,20 +360,6 @@ static int mt7615_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
|
||||
return mt7615_mac_wtbl_set_key(dev, wcid, key, cmd);
|
||||
}
|
||||
|
||||
void mt7615_ps_work(struct work_struct *work)
|
||||
{
|
||||
struct mt7615_phy *phy;
|
||||
|
||||
phy = (struct mt7615_phy *)container_of(work, struct mt7615_phy,
|
||||
ps_work);
|
||||
|
||||
mutex_lock(&phy->dev->mt76.mutex);
|
||||
ieee80211_iterate_active_interfaces(phy->mt76->hw,
|
||||
IEEE80211_IFACE_ITER_RESUME_ALL,
|
||||
m7615_mcu_set_ps_iter, phy);
|
||||
mutex_unlock(&phy->dev->mt76.mutex);
|
||||
}
|
||||
|
||||
static int mt7615_config(struct ieee80211_hw *hw, u32 changed)
|
||||
{
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
@ -399,14 +385,6 @@ static int mt7615_config(struct ieee80211_hw *hw, u32 changed)
|
||||
mt76_wr(dev, MT_WF_RFCR(band), phy->rxfilter);
|
||||
}
|
||||
|
||||
if (changed & IEEE80211_CONF_CHANGE_PS) {
|
||||
if (hw->conf.flags & IEEE80211_CONF_PS)
|
||||
set_bit(MT76_STATE_PS, &phy->mt76->state);
|
||||
else
|
||||
clear_bit(MT76_STATE_PS, &phy->mt76->state);
|
||||
ieee80211_queue_work(hw, &phy->ps_work);
|
||||
}
|
||||
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
return ret;
|
||||
@ -509,6 +487,9 @@ static void mt7615_bss_info_changed(struct ieee80211_hw *hw,
|
||||
BSS_CHANGED_BEACON_ENABLED))
|
||||
mt7615_mcu_add_beacon(dev, hw, vif, info->enable_beacon);
|
||||
|
||||
if (changed & BSS_CHANGED_PS)
|
||||
mt7615_mcu_set_vif_ps(dev, vif);
|
||||
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
}
|
||||
|
||||
@ -791,6 +772,37 @@ mt7615_set_antenna(struct ieee80211_hw *hw, u32 tx_ant, u32 rx_ant)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void mt7615_roc_iter(void *priv, u8 *mac,
|
||||
struct ieee80211_vif *vif)
|
||||
{
|
||||
struct mt7615_phy *phy = priv;
|
||||
|
||||
mt7615_mcu_set_roc(phy, vif, NULL, 0);
|
||||
}
|
||||
|
||||
void mt7615_roc_work(struct work_struct *work)
|
||||
{
|
||||
struct mt7615_phy *phy;
|
||||
|
||||
phy = (struct mt7615_phy *)container_of(work, struct mt7615_phy,
|
||||
roc_work);
|
||||
|
||||
if (!test_and_clear_bit(MT76_STATE_ROC, &phy->mt76->state))
|
||||
return;
|
||||
|
||||
ieee80211_iterate_active_interfaces(phy->mt76->hw,
|
||||
IEEE80211_IFACE_ITER_RESUME_ALL,
|
||||
mt7615_roc_iter, phy);
|
||||
ieee80211_remain_on_channel_expired(phy->mt76->hw);
|
||||
}
|
||||
|
||||
void mt7615_roc_timer(struct timer_list *timer)
|
||||
{
|
||||
struct mt7615_phy *phy = from_timer(phy, timer, roc_timer);
|
||||
|
||||
ieee80211_queue_work(phy->mt76->hw, &phy->roc_work);
|
||||
}
|
||||
|
||||
void mt7615_scan_work(struct work_struct *work)
|
||||
{
|
||||
struct mt7615_phy *phy;
|
||||
@ -864,6 +876,50 @@ mt7615_stop_sched_scan(struct ieee80211_hw *hw, struct ieee80211_vif *vif)
|
||||
return mt7615_mcu_sched_scan_enable(mphy->priv, vif, false);
|
||||
}
|
||||
|
||||
static int mt7615_remain_on_channel(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_channel *chan,
|
||||
int duration,
|
||||
enum ieee80211_roc_type type)
|
||||
{
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
int err;
|
||||
|
||||
if (test_and_set_bit(MT76_STATE_ROC, &phy->mt76->state))
|
||||
return 0;
|
||||
|
||||
err = mt7615_mcu_set_roc(phy, vif, chan, duration);
|
||||
if (err < 0) {
|
||||
clear_bit(MT76_STATE_ROC, &phy->mt76->state);
|
||||
return err;
|
||||
}
|
||||
|
||||
if (!wait_event_timeout(phy->roc_wait, phy->roc_grant, HZ)) {
|
||||
mt7615_mcu_set_roc(phy, vif, NULL, 0);
|
||||
clear_bit(MT76_STATE_ROC, &phy->mt76->state);
|
||||
|
||||
return -ETIMEDOUT;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int mt7615_cancel_remain_on_channel(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif)
|
||||
{
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
|
||||
if (!test_and_clear_bit(MT76_STATE_ROC, &phy->mt76->state))
|
||||
return 0;
|
||||
|
||||
del_timer_sync(&phy->roc_timer);
|
||||
cancel_work_sync(&phy->roc_work);
|
||||
|
||||
mt7615_mcu_set_roc(phy, vif, NULL, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_PM
|
||||
static int mt7615_suspend(struct ieee80211_hw *hw,
|
||||
struct cfg80211_wowlan *wowlan)
|
||||
@ -978,6 +1034,8 @@ const struct ieee80211_ops mt7615_ops = {
|
||||
.cancel_hw_scan = mt7615_cancel_hw_scan,
|
||||
.sched_scan_start = mt7615_start_sched_scan,
|
||||
.sched_scan_stop = mt7615_stop_sched_scan,
|
||||
.remain_on_channel = mt7615_remain_on_channel,
|
||||
.cancel_remain_on_channel = mt7615_cancel_remain_on_channel,
|
||||
#ifdef CONFIG_PM
|
||||
.suspend = mt7615_suspend,
|
||||
.resume = mt7615_resume,
|
||||
|
@ -359,6 +359,33 @@ mt7615_mcu_scan_event(struct mt7615_dev *dev, struct sk_buff *skb)
|
||||
MT7615_HW_SCAN_TIMEOUT);
|
||||
}
|
||||
|
||||
static void
|
||||
mt7615_mcu_roc_event(struct mt7615_dev *dev, struct sk_buff *skb)
|
||||
{
|
||||
struct mt7615_roc_tlv *event;
|
||||
struct mt7615_phy *phy;
|
||||
struct mt76_phy *mphy;
|
||||
int duration;
|
||||
|
||||
skb_pull(skb, sizeof(struct mt7615_mcu_rxd));
|
||||
event = (struct mt7615_roc_tlv *)skb->data;
|
||||
|
||||
if (event->dbdc_band && dev->mt76.phy2)
|
||||
mphy = dev->mt76.phy2;
|
||||
else
|
||||
mphy = &dev->mt76.phy;
|
||||
|
||||
ieee80211_ready_on_channel(mphy->hw);
|
||||
|
||||
phy = (struct mt7615_phy *)mphy->priv;
|
||||
phy->roc_grant = true;
|
||||
wake_up(&phy->roc_wait);
|
||||
|
||||
duration = le32_to_cpu(event->max_interval);
|
||||
mod_timer(&phy->roc_timer,
|
||||
round_jiffies_up(jiffies + msecs_to_jiffies(duration)));
|
||||
}
|
||||
|
||||
static void
|
||||
mt7615_mcu_beacon_loss_iter(void *priv, u8 *mac, struct ieee80211_vif *vif)
|
||||
{
|
||||
@ -426,6 +453,9 @@ mt7615_mcu_rx_unsolicited_event(struct mt7615_dev *dev, struct sk_buff *skb)
|
||||
case MCU_EVENT_BSS_BEACON_LOSS:
|
||||
mt7615_mcu_beacon_loss_event(dev, skb);
|
||||
break;
|
||||
case MCU_EVENT_ROC:
|
||||
mt7615_mcu_roc_event(dev, skb);
|
||||
break;
|
||||
case MCU_EVENT_SCHED_SCAN_DONE:
|
||||
case MCU_EVENT_SCAN_DONE:
|
||||
mt7615_mcu_scan_event(dev, skb);
|
||||
@ -451,6 +481,7 @@ void mt7615_mcu_rx_event(struct mt7615_dev *dev, struct sk_buff *skb)
|
||||
rxd->eid == MCU_EVENT_SCHED_SCAN_DONE ||
|
||||
rxd->eid == MCU_EVENT_BSS_ABSENCE ||
|
||||
rxd->eid == MCU_EVENT_SCAN_DONE ||
|
||||
rxd->eid == MCU_EVENT_ROC ||
|
||||
!rxd->seq)
|
||||
mt7615_mcu_rx_unsolicited_event(dev, skb);
|
||||
else
|
||||
@ -2741,11 +2772,9 @@ int mt7615_mcu_set_sku_en(struct mt7615_phy *phy, bool enable)
|
||||
sizeof(req), true);
|
||||
}
|
||||
|
||||
void m7615_mcu_set_ps_iter(void *priv, u8 *mac, struct ieee80211_vif *vif)
|
||||
int mt7615_mcu_set_vif_ps(struct mt7615_dev *dev, struct ieee80211_vif *vif)
|
||||
{
|
||||
struct mt7615_vif *mvif = (struct mt7615_vif *)vif->drv_priv;
|
||||
struct mt7615_phy *phy = priv;
|
||||
struct mt76_phy *mphy = phy->mt76;
|
||||
struct {
|
||||
u8 bss_idx;
|
||||
u8 ps_state; /* 0: device awake
|
||||
@ -2754,12 +2783,14 @@ void m7615_mcu_set_ps_iter(void *priv, u8 *mac, struct ieee80211_vif *vif)
|
||||
*/
|
||||
} req = {
|
||||
.bss_idx = mvif->idx,
|
||||
.ps_state = test_bit(MT76_STATE_PS, &mphy->state) ? 2 : 0,
|
||||
.ps_state = vif->bss_conf.ps ? 2 : 0,
|
||||
};
|
||||
|
||||
if (vif->type == NL80211_IFTYPE_STATION)
|
||||
__mt76_mcu_send_msg(&phy->dev->mt76, MCU_CMD_SET_PS_PROFILE,
|
||||
&req, sizeof(req), false);
|
||||
if (vif->type != NL80211_IFTYPE_STATION)
|
||||
return -ENOTSUPP;
|
||||
|
||||
return __mt76_mcu_send_msg(&dev->mt76, MCU_CMD_SET_PS_PROFILE,
|
||||
&req, sizeof(req), false);
|
||||
}
|
||||
|
||||
int mt7615_mcu_set_channel_domain(struct mt7615_phy *phy)
|
||||
@ -2867,6 +2898,7 @@ int mt7615_mcu_hw_scan(struct mt7615_phy *phy, struct ieee80211_vif *vif,
|
||||
n_ssids++;
|
||||
}
|
||||
req->ssid_type = n_ssids ? BIT(2) : BIT(0);
|
||||
req->ssid_type_ext = n_ssids ? BIT(0) : 0;
|
||||
req->ssids_num = n_ssids;
|
||||
|
||||
/* increase channel time for passive scan */
|
||||
@ -3601,6 +3633,26 @@ int mt7615_mcu_update_gtk_rekey(struct ieee80211_hw *hw,
|
||||
}
|
||||
#endif /* CONFIG_PM */
|
||||
|
||||
int mt7615_mcu_set_roc(struct mt7615_phy *phy, struct ieee80211_vif *vif,
|
||||
struct ieee80211_channel *chan, int duration)
|
||||
{
|
||||
struct mt7615_vif *mvif = (struct mt7615_vif *)vif->drv_priv;
|
||||
struct mt7615_dev *dev = phy->dev;
|
||||
struct mt7615_roc_tlv req = {
|
||||
.bss_idx = mvif->idx,
|
||||
.active = !chan,
|
||||
.max_interval = cpu_to_le32(duration),
|
||||
.primary_chan = chan ? chan->hw_value : 0,
|
||||
.band = chan ? chan->band : 0,
|
||||
.req_type = 2,
|
||||
};
|
||||
|
||||
phy->roc_grant = false;
|
||||
|
||||
return __mt76_mcu_send_msg(&dev->mt76, MCU_CMD_SET_ROC, &req,
|
||||
sizeof(req), false);
|
||||
}
|
||||
|
||||
int mt7615_mcu_set_p2p_oppps(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif)
|
||||
{
|
||||
|
@ -82,6 +82,7 @@ enum {
|
||||
MCU_EVENT_ACCESS_REG = 0x02,
|
||||
MCU_EVENT_MT_PATCH_SEM = 0x04,
|
||||
MCU_EVENT_SCAN_DONE = 0x0d,
|
||||
MCU_EVENT_ROC = 0x10,
|
||||
MCU_EVENT_BSS_ABSENCE = 0x11,
|
||||
MCU_EVENT_BSS_BEACON_LOSS = 0x13,
|
||||
MCU_EVENT_CH_PRIVILEGE = 0x18,
|
||||
@ -326,7 +327,8 @@ struct mt7615_hw_scan_req {
|
||||
*/
|
||||
u8 ssid_type; /* BIT(0) wildcard SSID
|
||||
* BIT(1) P2P wildcard SSID
|
||||
* BIT(2) specified SSID
|
||||
* BIT(2) specified SSID + wildcard SSID
|
||||
* BIT(2) + ssid_type_ext BIT(0) specified SSID only
|
||||
*/
|
||||
u8 ssids_num;
|
||||
u8 probe_req_num; /* Number of probe request for each SSID */
|
||||
@ -361,7 +363,8 @@ struct mt7615_hw_scan_req {
|
||||
struct mt7615_mcu_scan_ssid ext_ssids[6];
|
||||
u8 bssid[ETH_ALEN];
|
||||
u8 random_mac[ETH_ALEN]; /* valid when BIT(1) in scan_func is set. */
|
||||
u8 pad[64];
|
||||
u8 pad[63];
|
||||
u8 ssid_type_ext;
|
||||
} __packed;
|
||||
|
||||
#define SCAN_DONE_EVENT_MAX_CHANNEL_NUM 64
|
||||
@ -525,6 +528,23 @@ struct mt7615_gtk_rekey_tlv {
|
||||
u8 reserverd[3];
|
||||
} __packed;
|
||||
|
||||
struct mt7615_roc_tlv {
|
||||
u8 bss_idx;
|
||||
u8 token;
|
||||
u8 active;
|
||||
u8 primary_chan;
|
||||
u8 sco;
|
||||
u8 band;
|
||||
u8 width; /* To support 80/160MHz bandwidth */
|
||||
u8 freq_seg1; /* To support 80/160MHz bandwidth */
|
||||
u8 freq_seg2; /* To support 80/160MHz bandwidth */
|
||||
u8 req_type;
|
||||
u8 dbdc_band;
|
||||
u8 rsv0;
|
||||
__le32 max_interval; /* ms */
|
||||
u8 rsv1[8];
|
||||
} __packed;
|
||||
|
||||
/* offload mcu commands */
|
||||
enum {
|
||||
MCU_CMD_START_HW_SCAN = MCU_CE_PREFIX | 0x03,
|
||||
@ -533,6 +553,7 @@ enum {
|
||||
MCU_CMD_SET_BSS_CONNECTED = MCU_CE_PREFIX | 0x16,
|
||||
MCU_CMD_SET_BSS_ABORT = MCU_CE_PREFIX | 0x17,
|
||||
MCU_CMD_CANCEL_HW_SCAN = MCU_CE_PREFIX | 0x1b,
|
||||
MCU_CMD_SET_ROC = MCU_CE_PREFIX | 0x1c,
|
||||
MCU_CMD_SET_P2P_OPPPS = MCU_CE_PREFIX | 0x33,
|
||||
MCU_CMD_SCHED_SCAN_ENABLE = MCU_CE_PREFIX | 0x61,
|
||||
MCU_CMD_SCHED_SCAN_REQ = MCU_CE_PREFIX | 0x62,
|
||||
|
@ -201,7 +201,10 @@ struct mt7615_phy {
|
||||
struct sk_buff_head scan_event_list;
|
||||
struct delayed_work scan_work;
|
||||
|
||||
struct work_struct ps_work;
|
||||
struct work_struct roc_work;
|
||||
struct timer_list roc_timer;
|
||||
wait_queue_head_t roc_wait;
|
||||
bool roc_grant;
|
||||
};
|
||||
|
||||
#define mt7615_mcu_add_tx_ba(dev, ...) (dev)->mcu_ops->add_tx_ba((dev), __VA_ARGS__)
|
||||
@ -411,7 +414,7 @@ static inline bool is_mt7622(struct mt76_dev *dev)
|
||||
|
||||
static inline bool is_mt7615(struct mt76_dev *dev)
|
||||
{
|
||||
return mt76_chip(dev) == 0x7615;
|
||||
return mt76_chip(dev) == 0x7615 || mt76_chip(dev) == 0x7611;
|
||||
}
|
||||
|
||||
static inline bool is_mt7663(struct mt76_dev *dev)
|
||||
@ -419,6 +422,11 @@ static inline bool is_mt7663(struct mt76_dev *dev)
|
||||
return mt76_chip(dev) == 0x7663;
|
||||
}
|
||||
|
||||
static inline bool is_mt7611(struct mt76_dev *dev)
|
||||
{
|
||||
return mt76_chip(dev) == 0x7611;
|
||||
}
|
||||
|
||||
static inline void mt7615_irq_enable(struct mt7615_dev *dev, u32 mask)
|
||||
{
|
||||
mt76_set_irq_mask(&dev->mt76, 0, 0, mask);
|
||||
@ -441,7 +449,8 @@ static inline u16 mt7615_wtbl_size(struct mt7615_dev *dev)
|
||||
|
||||
void mt7615_dma_reset(struct mt7615_dev *dev);
|
||||
void mt7615_scan_work(struct work_struct *work);
|
||||
void mt7615_ps_work(struct work_struct *work);
|
||||
void mt7615_roc_work(struct work_struct *work);
|
||||
void mt7615_roc_timer(struct timer_list *timer);
|
||||
void mt7615_init_txpower(struct mt7615_dev *dev,
|
||||
struct ieee80211_supported_band *sband);
|
||||
void mt7615_phy_init(struct mt7615_dev *dev);
|
||||
@ -527,11 +536,13 @@ int mt7615_mcu_set_radar_th(struct mt7615_dev *dev, int index,
|
||||
int mt7615_mcu_set_sku_en(struct mt7615_phy *phy, bool enable);
|
||||
int mt7615_mcu_apply_rx_dcoc(struct mt7615_phy *phy);
|
||||
int mt7615_mcu_apply_tx_dpd(struct mt7615_phy *phy);
|
||||
void m7615_mcu_set_ps_iter(void *priv, u8 *mac, struct ieee80211_vif *vif);
|
||||
int mt7615_mcu_set_vif_ps(struct mt7615_dev *dev, struct ieee80211_vif *vif);
|
||||
int mt7615_dfs_init_radar_detector(struct mt7615_phy *phy);
|
||||
|
||||
int mt7615_mcu_set_p2p_oppps(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif);
|
||||
int mt7615_mcu_set_roc(struct mt7615_phy *phy, struct ieee80211_vif *vif,
|
||||
struct ieee80211_channel *chan, int duration);
|
||||
int mt7615_firmware_own(struct mt7615_dev *dev);
|
||||
int mt7615_driver_own(struct mt7615_dev *dev);
|
||||
|
||||
|
@ -14,6 +14,7 @@
|
||||
static const struct pci_device_id mt7615_pci_device_table[] = {
|
||||
{ PCI_DEVICE(0x14c3, 0x7615) },
|
||||
{ PCI_DEVICE(0x14c3, 0x7663) },
|
||||
{ PCI_DEVICE(0x14c3, 0x7611) },
|
||||
{ },
|
||||
};
|
||||
|
||||
@ -85,8 +86,9 @@ static int mt7615_pci_suspend(struct pci_dev *pdev, pm_message_t state)
|
||||
napi_disable(&mdev->tx_napi);
|
||||
tasklet_kill(&mdev->tx_tasklet);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(mdev->q_rx); i++)
|
||||
mt76_for_each_q_rx(mdev, i) {
|
||||
napi_disable(&mdev->napi[i]);
|
||||
}
|
||||
tasklet_kill(&dev->irq_tasklet);
|
||||
|
||||
mt7615_dma_reset(dev);
|
||||
@ -119,8 +121,9 @@ static int mt7615_pci_suspend(struct pci_dev *pdev, pm_message_t state)
|
||||
return 0;
|
||||
|
||||
restore:
|
||||
for (i = 0; i < ARRAY_SIZE(mdev->q_rx); i++)
|
||||
mt76_for_each_q_rx(mdev, i) {
|
||||
napi_enable(&mdev->napi[i]);
|
||||
}
|
||||
napi_enable(&mdev->tx_napi);
|
||||
if (hif_suspend)
|
||||
mt7615_mcu_set_hif_suspend(dev, false);
|
||||
@ -155,7 +158,7 @@ static int mt7615_pci_resume(struct pci_dev *pdev)
|
||||
if (pdma_reset)
|
||||
dev_err(mdev->dev, "PDMA engine must be reinitialized\n");
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(mdev->q_rx); i++) {
|
||||
mt76_for_each_q_rx(mdev, i) {
|
||||
napi_enable(&mdev->napi[i]);
|
||||
napi_schedule(&mdev->napi[i]);
|
||||
}
|
||||
|
@ -51,7 +51,8 @@ static void mt7663u_stop(struct ieee80211_hw *hw)
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
cancel_work_sync(&phy->ps_work);
|
||||
del_timer_sync(&phy->roc_timer);
|
||||
cancel_work_sync(&phy->roc_work);
|
||||
cancel_delayed_work_sync(&phy->scan_work);
|
||||
cancel_delayed_work_sync(&phy->mac_work);
|
||||
mt76u_stop_tx(&dev->mt76);
|
||||
|
@ -15,6 +15,7 @@
|
||||
#include "mt76x02_dfs.h"
|
||||
#include "mt76x02_dma.h"
|
||||
|
||||
#define MT76x02_N_WCIDS 128
|
||||
#define MT_CALIBRATE_INTERVAL HZ
|
||||
#define MT_MAC_WORK_INTERVAL (HZ / 10)
|
||||
|
||||
@ -246,7 +247,7 @@ mt76x02_rx_get_sta(struct mt76_dev *dev, u8 idx)
|
||||
{
|
||||
struct mt76_wcid *wcid;
|
||||
|
||||
if (idx >= ARRAY_SIZE(dev->wcid))
|
||||
if (idx >= MT76x02_N_WCIDS)
|
||||
return NULL;
|
||||
|
||||
wcid = rcu_dereference(dev->wcid[idx]);
|
||||
|
@ -561,7 +561,7 @@ void mt76x02_send_tx_status(struct mt76x02_dev *dev,
|
||||
|
||||
rcu_read_lock();
|
||||
|
||||
if (stat->wcid < ARRAY_SIZE(dev->mt76.wcid))
|
||||
if (stat->wcid < MT76x02_N_WCIDS)
|
||||
wcid = rcu_dereference(dev->mt76.wcid[stat->wcid]);
|
||||
|
||||
if (wcid && wcid->sta) {
|
||||
|
@ -20,7 +20,7 @@ int mt76x02_mcu_msg_send(struct mt76_dev *mdev, int cmd, const void *data,
|
||||
int ret;
|
||||
u8 seq;
|
||||
|
||||
if (mt76_is_mmio(&dev->mt76) && dev->mcu_timeout)
|
||||
if (dev->mcu_timeout)
|
||||
return -EIO;
|
||||
|
||||
skb = mt76_mcu_msg_alloc(mdev, data, len);
|
||||
|
@ -415,7 +415,7 @@ static void mt76x02_reset_state(struct mt76x02_dev *dev)
|
||||
ieee80211_iter_keys_rcu(dev->mt76.hw, NULL, mt76x02_key_sync, NULL);
|
||||
rcu_read_unlock();
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->mt76.wcid); i++) {
|
||||
for (i = 0; i < MT76x02_N_WCIDS; i++) {
|
||||
struct ieee80211_sta *sta;
|
||||
struct ieee80211_vif *vif;
|
||||
struct mt76x02_sta *msta;
|
||||
@ -489,8 +489,9 @@ static void mt76x02_watchdog_reset(struct mt76x02_dev *dev)
|
||||
for (i = 0; i < __MT_TXQ_MAX; i++)
|
||||
mt76_queue_tx_cleanup(dev, i, true);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->mt76.q_rx); i++)
|
||||
mt76_for_each_q_rx(&dev->mt76, i) {
|
||||
mt76_queue_rx_reset(dev, i);
|
||||
}
|
||||
|
||||
mt76x02_mac_start(dev);
|
||||
|
||||
|
@ -249,7 +249,7 @@ int mt76x02_sta_add(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
|
||||
memset(msta, 0, sizeof(*msta));
|
||||
|
||||
idx = mt76_wcid_alloc(dev->mt76.wcid_mask, ARRAY_SIZE(dev->mt76.wcid));
|
||||
idx = mt76_wcid_alloc(dev->mt76.wcid_mask, MT76x02_N_WCIDS);
|
||||
if (idx < 0)
|
||||
return -ENOSPC;
|
||||
|
||||
|
@ -173,14 +173,14 @@ mt7915_txbf_stat_read_phy(struct mt7915_phy *phy, struct seq_file *s)
|
||||
|
||||
/* Tx Beamformee Rx NDPA & Tx feedback report */
|
||||
cnt = mt76_rr(dev, MT_ETBF_TX_NDP_BFRP(ext_phy));
|
||||
seq_printf(s, "Tx Beamformee sucessful feedback frames: %ld\n",
|
||||
seq_printf(s, "Tx Beamformee successful feedback frames: %ld\n",
|
||||
FIELD_GET(MT_ETBF_TX_FB_CPL, cnt));
|
||||
seq_printf(s, "Tx Beamformee feedback triggerd counts: %ld\n",
|
||||
seq_printf(s, "Tx Beamformee feedback triggered counts: %ld\n",
|
||||
FIELD_GET(MT_ETBF_TX_FB_TRI, cnt));
|
||||
|
||||
/* Tx SU counters */
|
||||
cnt = mt76_rr(dev, MT_MIB_DR11(ext_phy));
|
||||
seq_printf(s, "Tx single-user sucessful MPDU counts: %d\n", cnt);
|
||||
seq_printf(s, "Tx single-user successful MPDU counts: %d\n", cnt);
|
||||
|
||||
seq_puts(s, "\n");
|
||||
}
|
||||
|
@ -592,7 +592,6 @@ int mt7915_register_ext_phy(struct mt7915_dev *dev)
|
||||
if (phy)
|
||||
return 0;
|
||||
|
||||
INIT_DELAYED_WORK(&phy->mac_work, mt7915_mac_work);
|
||||
mt7915_cap_dbdc_enable(dev);
|
||||
mphy = mt76_alloc_phy(&dev->mt76, sizeof(*phy), &mt7915_ops);
|
||||
if (!mphy)
|
||||
@ -605,6 +604,8 @@ int mt7915_register_ext_phy(struct mt7915_dev *dev)
|
||||
mphy->antenna_mask = BIT(hweight8(phy->chainmask)) - 1;
|
||||
mt7915_init_wiphy(mphy->hw);
|
||||
|
||||
INIT_DELAYED_WORK(&phy->mac_work, mt7915_mac_work);
|
||||
|
||||
/*
|
||||
* Make the secondary PHY MAC address local without overlapping with
|
||||
* the usual MAC address allocation scheme on multiple virtual interfaces
|
||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user