mt76 patches for 5.7
* dual-band concurrent support for MT7615 * fixes for rx path race conditions * EEPROM fixes * MAC address handling fixes * coverage class support for MT7615 * beacon fixes for USB devices * MT7615 LED support * minor cleanups/fixes for all drivers * set_antenna support for MT7615 * tracing improvements * preparation for supporting new USB devices * tx power fixes -----BEGIN PGP SIGNATURE----- Comment: GPGTools - http://gpgtools.org iEYEABECAAYFAl5G3tEACgkQ130UHQKnbvXVtgCdEBlcmjViCqkCPXStdSxUy9o9 JCsAn2TnPduCoTCB1jo1foJbNXn5XzuG =Z/l3 -----END PGP SIGNATURE----- Merge tag 'mt76-for-kvalo-2020-02-14' of https://github.com/nbd168/wireless mt76 patches for 5.7 * dual-band concurrent support for MT7615 * fixes for rx path race conditions * EEPROM fixes * MAC address handling fixes * coverage class support for MT7615 * beacon fixes for USB devices * MT7615 LED support * minor cleanups/fixes for all drivers * set_antenna support for MT7615 * tracing improvements * preparation for supporting new USB devices * tx power fixes
This commit is contained in:
commit
6065bb8a9c
@ -4,17 +4,27 @@ This node provides properties for configuring the MediaTek mt76xx wireless
|
||||
device. The node is expected to be specified as a child node of the PCI
|
||||
controller to which the wireless chip is connected.
|
||||
|
||||
Alternatively, it can specify the wireless part of the MT7628/MT7688 SoC.
|
||||
For SoC, use the compatible string "mediatek,mt7628-wmac" and the following
|
||||
properties:
|
||||
Alternatively, it can specify the wireless part of the MT7628/MT7688 or
|
||||
MT7622 SoC. For SoC, use the following compatible strings:
|
||||
|
||||
compatible:
|
||||
- "mediatek,mt7628-wmac" for MT7628/MT7688
|
||||
- "mediatek,mt7622-wmac" for MT7622
|
||||
|
||||
properties:
|
||||
- reg: Address and length of the register set for the device.
|
||||
- interrupts: Main device interrupt
|
||||
|
||||
MT7622 specific properties:
|
||||
- power-domains: phandle to the power domain that the WMAC is part of
|
||||
- mediatek,infracfg: phandle to the infrastructure bus fabric syscon node
|
||||
|
||||
Optional properties:
|
||||
|
||||
- ieee80211-freq-limit: See ieee80211.txt
|
||||
- mediatek,mtd-eeprom: Specify a MTD partition + offset containing EEPROM data
|
||||
- big-endian: if the radio eeprom partition is written in big-endian, specify
|
||||
this property
|
||||
|
||||
The MAC address can as well be set with corresponding optional properties
|
||||
defined in net/ethernet.txt.
|
||||
@ -31,6 +41,7 @@ Optional nodes:
|
||||
reg = <0x0000 0 0 0 0>;
|
||||
ieee80211-freq-limit = <5000000 6000000>;
|
||||
mediatek,mtd-eeprom = <&factory 0x8000>;
|
||||
big-endian;
|
||||
|
||||
led {
|
||||
led-sources = <2>;
|
||||
@ -50,3 +61,15 @@ wmac: wmac@10300000 {
|
||||
|
||||
mediatek,mtd-eeprom = <&factory 0x0000>;
|
||||
};
|
||||
|
||||
MT7622 example:
|
||||
|
||||
wmac: wmac@18000000 {
|
||||
compatible = "mediatek,mt7622-wmac";
|
||||
reg = <0 0x18000000 0 0x100000>;
|
||||
interrupts = <GIC_SPI 211 IRQ_TYPE_LEVEL_LOW>;
|
||||
|
||||
mediatek,infracfg = <&infracfg>;
|
||||
|
||||
power-domains = <&scpsys MT7622_POWER_DOMAIN_WB>;
|
||||
};
|
||||
|
@ -6,7 +6,7 @@ obj-$(CONFIG_MT76x02_USB) += mt76x02-usb.o
|
||||
|
||||
mt76-y := \
|
||||
mmio.o util.o trace.o dma.o mac80211.o debugfs.o eeprom.o \
|
||||
tx.o agg-rx.o mcu.o airtime.o
|
||||
tx.o agg-rx.o mcu.o
|
||||
|
||||
mt76-$(CONFIG_PCI) += pci.o
|
||||
|
||||
|
@ -4,7 +4,13 @@
|
||||
*/
|
||||
#include "mt76.h"
|
||||
|
||||
#define REORDER_TIMEOUT (HZ / 10)
|
||||
static unsigned long mt76_aggr_tid_to_timeo(u8 tidno)
|
||||
{
|
||||
/* Currently voice traffic (AC_VO) always runs without aggregation,
|
||||
* no special handling is needed. AC_BE/AC_BK use tids 0-3. Just check
|
||||
* for non AC_BK/AC_BE and set smaller timeout for it. */
|
||||
return HZ / (tidno >= 4 ? 25 : 10);
|
||||
}
|
||||
|
||||
static void
|
||||
mt76_aggr_release(struct mt76_rx_tid *tid, struct sk_buff_head *frames, int idx)
|
||||
@ -71,7 +77,8 @@ mt76_rx_aggr_check_release(struct mt76_rx_tid *tid, struct sk_buff_head *frames)
|
||||
nframes--;
|
||||
status = (struct mt76_rx_status *)skb->cb;
|
||||
if (!time_after(jiffies,
|
||||
status->reorder_time + REORDER_TIMEOUT))
|
||||
status->reorder_time +
|
||||
mt76_aggr_tid_to_timeo(tid->num)))
|
||||
continue;
|
||||
|
||||
mt76_rx_aggr_release_frames(tid, frames, status->seqno);
|
||||
@ -101,7 +108,7 @@ mt76_rx_aggr_reorder_work(struct work_struct *work)
|
||||
|
||||
if (nframes)
|
||||
ieee80211_queue_delayed_work(tid->dev->hw, &tid->reorder_work,
|
||||
REORDER_TIMEOUT);
|
||||
mt76_aggr_tid_to_timeo(tid->num));
|
||||
mt76_rx_complete(dev, &frames, NULL);
|
||||
|
||||
rcu_read_unlock();
|
||||
@ -225,7 +232,7 @@ void mt76_rx_aggr_reorder(struct sk_buff *skb, struct sk_buff_head *frames)
|
||||
mt76_rx_aggr_release_head(tid, frames);
|
||||
|
||||
ieee80211_queue_delayed_work(tid->dev->hw, &tid->reorder_work,
|
||||
REORDER_TIMEOUT);
|
||||
mt76_aggr_tid_to_timeo(tid->num));
|
||||
|
||||
out:
|
||||
spin_unlock_bh(&tid->lock);
|
||||
@ -245,6 +252,7 @@ int mt76_rx_aggr_start(struct mt76_dev *dev, struct mt76_wcid *wcid, u8 tidno,
|
||||
tid->dev = dev;
|
||||
tid->head = ssn;
|
||||
tid->size = size;
|
||||
tid->num = tidno;
|
||||
INIT_DELAYED_WORK(&tid->reorder_work, mt76_rx_aggr_reorder_work);
|
||||
spin_lock_init(&tid->lock);
|
||||
|
||||
@ -268,6 +276,7 @@ static void mt76_rx_aggr_shutdown(struct mt76_dev *dev, struct mt76_rx_tid *tid)
|
||||
if (!skb)
|
||||
continue;
|
||||
|
||||
tid->reorder_buf[i] = NULL;
|
||||
tid->nframes--;
|
||||
dev_kfree_skb(skb);
|
||||
}
|
||||
|
@ -1,326 +0,0 @@
|
||||
// SPDX-License-Identifier: ISC
|
||||
/*
|
||||
* Copyright (C) 2019 Felix Fietkau <nbd@nbd.name>
|
||||
*/
|
||||
|
||||
#include "mt76.h"
|
||||
|
||||
#define AVG_PKT_SIZE 1024
|
||||
|
||||
/* Number of bits for an average sized packet */
|
||||
#define MCS_NBITS (AVG_PKT_SIZE << 3)
|
||||
|
||||
/* Number of symbols for a packet with (bps) bits per symbol */
|
||||
#define MCS_NSYMS(bps) DIV_ROUND_UP(MCS_NBITS, (bps))
|
||||
|
||||
/* Transmission time (1024 usec) for a packet containing (syms) * symbols */
|
||||
#define MCS_SYMBOL_TIME(sgi, syms) \
|
||||
(sgi ? \
|
||||
((syms) * 18 * 1024 + 4 * 1024) / 5 : /* syms * 3.6 us */ \
|
||||
((syms) * 1024) << 2 /* syms * 4 us */ \
|
||||
)
|
||||
|
||||
/* Transmit duration for the raw data part of an average sized packet */
|
||||
#define MCS_DURATION(streams, sgi, bps) \
|
||||
MCS_SYMBOL_TIME(sgi, MCS_NSYMS((streams) * (bps)))
|
||||
|
||||
#define BW_20 0
|
||||
#define BW_40 1
|
||||
#define BW_80 2
|
||||
|
||||
/*
|
||||
* Define group sort order: HT40 -> SGI -> #streams
|
||||
*/
|
||||
#define MT_MAX_STREAMS 4
|
||||
#define MT_HT_STREAM_GROUPS 4 /* BW(=2) * SGI(=2) */
|
||||
#define MT_VHT_STREAM_GROUPS 6 /* BW(=3) * SGI(=2) */
|
||||
|
||||
#define MT_HT_GROUPS_NB (MT_MAX_STREAMS * \
|
||||
MT_HT_STREAM_GROUPS)
|
||||
#define MT_VHT_GROUPS_NB (MT_MAX_STREAMS * \
|
||||
MT_VHT_STREAM_GROUPS)
|
||||
#define MT_GROUPS_NB (MT_HT_GROUPS_NB + \
|
||||
MT_VHT_GROUPS_NB)
|
||||
|
||||
#define MT_HT_GROUP_0 0
|
||||
#define MT_VHT_GROUP_0 (MT_HT_GROUP_0 + MT_HT_GROUPS_NB)
|
||||
|
||||
#define MCS_GROUP_RATES 10
|
||||
|
||||
#define HT_GROUP_IDX(_streams, _sgi, _ht40) \
|
||||
MT_HT_GROUP_0 + \
|
||||
MT_MAX_STREAMS * 2 * _ht40 + \
|
||||
MT_MAX_STREAMS * _sgi + \
|
||||
_streams - 1
|
||||
|
||||
#define _MAX(a, b) (((a)>(b))?(a):(b))
|
||||
|
||||
#define GROUP_SHIFT(duration) \
|
||||
_MAX(0, 16 - __builtin_clz(duration))
|
||||
|
||||
/* MCS rate information for an MCS group */
|
||||
#define __MCS_GROUP(_streams, _sgi, _ht40, _s) \
|
||||
[HT_GROUP_IDX(_streams, _sgi, _ht40)] = { \
|
||||
.shift = _s, \
|
||||
.duration = { \
|
||||
MCS_DURATION(_streams, _sgi, _ht40 ? 54 : 26) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, _ht40 ? 108 : 52) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, _ht40 ? 162 : 78) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, _ht40 ? 216 : 104) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, _ht40 ? 324 : 156) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, _ht40 ? 432 : 208) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, _ht40 ? 486 : 234) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, _ht40 ? 540 : 260) >> _s \
|
||||
} \
|
||||
}
|
||||
|
||||
#define MCS_GROUP_SHIFT(_streams, _sgi, _ht40) \
|
||||
GROUP_SHIFT(MCS_DURATION(_streams, _sgi, _ht40 ? 54 : 26))
|
||||
|
||||
#define MCS_GROUP(_streams, _sgi, _ht40) \
|
||||
__MCS_GROUP(_streams, _sgi, _ht40, \
|
||||
MCS_GROUP_SHIFT(_streams, _sgi, _ht40))
|
||||
|
||||
#define VHT_GROUP_IDX(_streams, _sgi, _bw) \
|
||||
(MT_VHT_GROUP_0 + \
|
||||
MT_MAX_STREAMS * 2 * (_bw) + \
|
||||
MT_MAX_STREAMS * (_sgi) + \
|
||||
(_streams) - 1)
|
||||
|
||||
#define BW2VBPS(_bw, r3, r2, r1) \
|
||||
(_bw == BW_80 ? r3 : _bw == BW_40 ? r2 : r1)
|
||||
|
||||
#define __VHT_GROUP(_streams, _sgi, _bw, _s) \
|
||||
[VHT_GROUP_IDX(_streams, _sgi, _bw)] = { \
|
||||
.shift = _s, \
|
||||
.duration = { \
|
||||
MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 117, 54, 26)) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 234, 108, 52)) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 351, 162, 78)) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 468, 216, 104)) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 702, 324, 156)) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 936, 432, 208)) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 1053, 486, 234)) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 1170, 540, 260)) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 1404, 648, 312)) >> _s, \
|
||||
MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 1560, 720, 346)) >> _s \
|
||||
} \
|
||||
}
|
||||
|
||||
#define VHT_GROUP_SHIFT(_streams, _sgi, _bw) \
|
||||
GROUP_SHIFT(MCS_DURATION(_streams, _sgi, \
|
||||
BW2VBPS(_bw, 117, 54, 26)))
|
||||
|
||||
#define VHT_GROUP(_streams, _sgi, _bw) \
|
||||
__VHT_GROUP(_streams, _sgi, _bw, \
|
||||
VHT_GROUP_SHIFT(_streams, _sgi, _bw))
|
||||
|
||||
struct mcs_group {
|
||||
u8 shift;
|
||||
u16 duration[MCS_GROUP_RATES];
|
||||
};
|
||||
|
||||
static const struct mcs_group airtime_mcs_groups[] = {
|
||||
MCS_GROUP(1, 0, BW_20),
|
||||
MCS_GROUP(2, 0, BW_20),
|
||||
MCS_GROUP(3, 0, BW_20),
|
||||
MCS_GROUP(4, 0, BW_20),
|
||||
|
||||
MCS_GROUP(1, 1, BW_20),
|
||||
MCS_GROUP(2, 1, BW_20),
|
||||
MCS_GROUP(3, 1, BW_20),
|
||||
MCS_GROUP(4, 1, BW_20),
|
||||
|
||||
MCS_GROUP(1, 0, BW_40),
|
||||
MCS_GROUP(2, 0, BW_40),
|
||||
MCS_GROUP(3, 0, BW_40),
|
||||
MCS_GROUP(4, 0, BW_40),
|
||||
|
||||
MCS_GROUP(1, 1, BW_40),
|
||||
MCS_GROUP(2, 1, BW_40),
|
||||
MCS_GROUP(3, 1, BW_40),
|
||||
MCS_GROUP(4, 1, BW_40),
|
||||
|
||||
VHT_GROUP(1, 0, BW_20),
|
||||
VHT_GROUP(2, 0, BW_20),
|
||||
VHT_GROUP(3, 0, BW_20),
|
||||
VHT_GROUP(4, 0, BW_20),
|
||||
|
||||
VHT_GROUP(1, 1, BW_20),
|
||||
VHT_GROUP(2, 1, BW_20),
|
||||
VHT_GROUP(3, 1, BW_20),
|
||||
VHT_GROUP(4, 1, BW_20),
|
||||
|
||||
VHT_GROUP(1, 0, BW_40),
|
||||
VHT_GROUP(2, 0, BW_40),
|
||||
VHT_GROUP(3, 0, BW_40),
|
||||
VHT_GROUP(4, 0, BW_40),
|
||||
|
||||
VHT_GROUP(1, 1, BW_40),
|
||||
VHT_GROUP(2, 1, BW_40),
|
||||
VHT_GROUP(3, 1, BW_40),
|
||||
VHT_GROUP(4, 1, BW_40),
|
||||
|
||||
VHT_GROUP(1, 0, BW_80),
|
||||
VHT_GROUP(2, 0, BW_80),
|
||||
VHT_GROUP(3, 0, BW_80),
|
||||
VHT_GROUP(4, 0, BW_80),
|
||||
|
||||
VHT_GROUP(1, 1, BW_80),
|
||||
VHT_GROUP(2, 1, BW_80),
|
||||
VHT_GROUP(3, 1, BW_80),
|
||||
VHT_GROUP(4, 1, BW_80),
|
||||
};
|
||||
|
||||
static u32
|
||||
mt76_calc_legacy_rate_duration(const struct ieee80211_rate *rate, bool short_pre,
|
||||
int len)
|
||||
{
|
||||
u32 duration;
|
||||
|
||||
switch (rate->hw_value >> 8) {
|
||||
case MT_PHY_TYPE_CCK:
|
||||
duration = 144 + 48; /* preamble + PLCP */
|
||||
if (short_pre)
|
||||
duration >>= 1;
|
||||
|
||||
duration += 10; /* SIFS */
|
||||
break;
|
||||
case MT_PHY_TYPE_OFDM:
|
||||
duration = 20 + 16; /* premable + SIFS */
|
||||
break;
|
||||
default:
|
||||
WARN_ON_ONCE(1);
|
||||
return 0;
|
||||
}
|
||||
|
||||
len <<= 3;
|
||||
duration += (len * 10) / rate->bitrate;
|
||||
|
||||
return duration;
|
||||
}
|
||||
|
||||
u32 mt76_calc_rx_airtime(struct mt76_dev *dev, struct mt76_rx_status *status,
|
||||
int len)
|
||||
{
|
||||
struct ieee80211_supported_band *sband;
|
||||
const struct ieee80211_rate *rate;
|
||||
bool sgi = status->enc_flags & RX_ENC_FLAG_SHORT_GI;
|
||||
bool sp = status->enc_flags & RX_ENC_FLAG_SHORTPRE;
|
||||
int bw, streams;
|
||||
u32 duration;
|
||||
int group, idx;
|
||||
|
||||
switch (status->bw) {
|
||||
case RATE_INFO_BW_20:
|
||||
bw = BW_20;
|
||||
break;
|
||||
case RATE_INFO_BW_40:
|
||||
bw = BW_40;
|
||||
break;
|
||||
case RATE_INFO_BW_80:
|
||||
bw = BW_80;
|
||||
break;
|
||||
default:
|
||||
WARN_ON_ONCE(1);
|
||||
return 0;
|
||||
}
|
||||
|
||||
switch (status->encoding) {
|
||||
case RX_ENC_LEGACY:
|
||||
if (WARN_ON_ONCE(status->band > NL80211_BAND_5GHZ))
|
||||
return 0;
|
||||
|
||||
sband = dev->hw->wiphy->bands[status->band];
|
||||
if (!sband || status->rate_idx >= sband->n_bitrates)
|
||||
return 0;
|
||||
|
||||
rate = &sband->bitrates[status->rate_idx];
|
||||
|
||||
return mt76_calc_legacy_rate_duration(rate, sp, len);
|
||||
case RX_ENC_VHT:
|
||||
streams = status->nss;
|
||||
idx = status->rate_idx;
|
||||
group = VHT_GROUP_IDX(streams, sgi, bw);
|
||||
break;
|
||||
case RX_ENC_HT:
|
||||
streams = ((status->rate_idx >> 3) & 3) + 1;
|
||||
idx = status->rate_idx & 7;
|
||||
group = HT_GROUP_IDX(streams, sgi, bw);
|
||||
break;
|
||||
default:
|
||||
WARN_ON_ONCE(1);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (WARN_ON_ONCE(streams > 4))
|
||||
return 0;
|
||||
|
||||
duration = airtime_mcs_groups[group].duration[idx];
|
||||
duration <<= airtime_mcs_groups[group].shift;
|
||||
duration *= len;
|
||||
duration /= AVG_PKT_SIZE;
|
||||
duration /= 1024;
|
||||
|
||||
duration += 36 + (streams << 2);
|
||||
|
||||
return duration;
|
||||
}
|
||||
|
||||
u32 mt76_calc_tx_airtime(struct mt76_dev *dev, struct ieee80211_tx_info *info,
|
||||
int len)
|
||||
{
|
||||
struct mt76_rx_status stat = {
|
||||
.band = info->band,
|
||||
};
|
||||
u32 duration = 0;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(info->status.rates); i++) {
|
||||
struct ieee80211_tx_rate *rate = &info->status.rates[i];
|
||||
u32 cur_duration;
|
||||
|
||||
if (rate->idx < 0 || !rate->count)
|
||||
break;
|
||||
|
||||
if (rate->flags & IEEE80211_TX_RC_80_MHZ_WIDTH)
|
||||
stat.bw = RATE_INFO_BW_80;
|
||||
else if (rate->flags & IEEE80211_TX_RC_40_MHZ_WIDTH)
|
||||
stat.bw = RATE_INFO_BW_40;
|
||||
else
|
||||
stat.bw = RATE_INFO_BW_20;
|
||||
|
||||
stat.enc_flags = 0;
|
||||
if (rate->flags & IEEE80211_TX_RC_USE_SHORT_PREAMBLE)
|
||||
stat.enc_flags |= RX_ENC_FLAG_SHORTPRE;
|
||||
if (rate->flags & IEEE80211_TX_RC_SHORT_GI)
|
||||
stat.enc_flags |= RX_ENC_FLAG_SHORT_GI;
|
||||
|
||||
stat.rate_idx = rate->idx;
|
||||
if (rate->flags & IEEE80211_TX_RC_VHT_MCS) {
|
||||
stat.encoding = RX_ENC_VHT;
|
||||
stat.rate_idx = ieee80211_rate_get_vht_mcs(rate);
|
||||
stat.nss = ieee80211_rate_get_vht_nss(rate);
|
||||
} else if (rate->flags & IEEE80211_TX_RC_MCS) {
|
||||
stat.encoding = RX_ENC_HT;
|
||||
} else {
|
||||
stat.encoding = RX_ENC_LEGACY;
|
||||
}
|
||||
|
||||
cur_duration = mt76_calc_rx_airtime(dev, &stat, len);
|
||||
duration += cur_duration * rate->count;
|
||||
}
|
||||
|
||||
return duration;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_calc_tx_airtime);
|
@ -132,6 +132,11 @@ mt76_dma_sync_idx(struct mt76_dev *dev, struct mt76_queue *q)
|
||||
writel(q->ndesc, &q->regs->ring_size);
|
||||
q->head = readl(&q->regs->dma_idx);
|
||||
q->tail = q->head;
|
||||
}
|
||||
|
||||
static void
|
||||
mt76_dma_kick_queue(struct mt76_dev *dev, struct mt76_queue *q)
|
||||
{
|
||||
writel(q->head, &q->regs->cpu_idx);
|
||||
}
|
||||
|
||||
@ -141,7 +146,7 @@ mt76_dma_tx_cleanup(struct mt76_dev *dev, enum mt76_txq_id qid, bool flush)
|
||||
struct mt76_sw_queue *sq = &dev->q_tx[qid];
|
||||
struct mt76_queue *q = sq->q;
|
||||
struct mt76_queue_entry entry;
|
||||
unsigned int n_swq_queued[4] = {};
|
||||
unsigned int n_swq_queued[8] = {};
|
||||
unsigned int n_queued = 0;
|
||||
bool wake = false;
|
||||
int i, last;
|
||||
@ -178,15 +183,25 @@ mt76_dma_tx_cleanup(struct mt76_dev *dev, enum mt76_txq_id qid, bool flush)
|
||||
spin_lock_bh(&q->lock);
|
||||
|
||||
q->queued -= n_queued;
|
||||
for (i = 0; i < ARRAY_SIZE(n_swq_queued); i++) {
|
||||
for (i = 0; i < 4; i++) {
|
||||
if (!n_swq_queued[i])
|
||||
continue;
|
||||
|
||||
dev->q_tx[i].swq_queued -= n_swq_queued[i];
|
||||
}
|
||||
|
||||
if (flush)
|
||||
/* ext PHY */
|
||||
for (i = 0; i < 4; i++) {
|
||||
if (!n_swq_queued[i])
|
||||
continue;
|
||||
|
||||
dev->q_tx[__MT_TXQ_MAX + i].swq_queued -= n_swq_queued[4 + i];
|
||||
}
|
||||
|
||||
if (flush) {
|
||||
mt76_dma_sync_idx(dev, q);
|
||||
mt76_dma_kick_queue(dev, q);
|
||||
}
|
||||
|
||||
wake = wake && q->stopped &&
|
||||
qid < IEEE80211_NUM_ACS && q->queued < q->ndesc - 8;
|
||||
@ -238,7 +253,9 @@ mt76_dma_dequeue(struct mt76_dev *dev, struct mt76_queue *q, bool flush,
|
||||
if (!q->queued)
|
||||
return NULL;
|
||||
|
||||
if (!flush && !(q->desc[idx].ctrl & cpu_to_le32(MT_DMA_CTL_DMA_DONE)))
|
||||
if (flush)
|
||||
q->desc[idx].ctrl |= cpu_to_le32(MT_DMA_CTL_DMA_DONE);
|
||||
else if (!(q->desc[idx].ctrl & cpu_to_le32(MT_DMA_CTL_DMA_DONE)))
|
||||
return NULL;
|
||||
|
||||
q->tail = (q->tail + 1) % q->ndesc;
|
||||
@ -247,12 +264,6 @@ mt76_dma_dequeue(struct mt76_dev *dev, struct mt76_queue *q, bool flush,
|
||||
return mt76_dma_get_buf(dev, q, idx, len, info, more);
|
||||
}
|
||||
|
||||
static void
|
||||
mt76_dma_kick_queue(struct mt76_dev *dev, struct mt76_queue *q)
|
||||
{
|
||||
writel(q->head, &q->regs->cpu_idx);
|
||||
}
|
||||
|
||||
static int
|
||||
mt76_dma_tx_queue_skb_raw(struct mt76_dev *dev, enum mt76_txq_id qid,
|
||||
struct sk_buff *skb, u32 tx_info)
|
||||
@ -261,10 +272,13 @@ mt76_dma_tx_queue_skb_raw(struct mt76_dev *dev, enum mt76_txq_id qid,
|
||||
struct mt76_queue_buf buf;
|
||||
dma_addr_t addr;
|
||||
|
||||
if (q->queued + 1 >= q->ndesc - 1)
|
||||
goto error;
|
||||
|
||||
addr = dma_map_single(dev->dev, skb->data, skb->len,
|
||||
DMA_TO_DEVICE);
|
||||
if (unlikely(dma_mapping_error(dev->dev, addr)))
|
||||
return -ENOMEM;
|
||||
goto error;
|
||||
|
||||
buf.addr = addr;
|
||||
buf.len = skb->len;
|
||||
@ -275,6 +289,10 @@ mt76_dma_tx_queue_skb_raw(struct mt76_dev *dev, enum mt76_txq_id qid,
|
||||
spin_unlock_bh(&q->lock);
|
||||
|
||||
return 0;
|
||||
|
||||
error:
|
||||
dev_kfree_skb(skb);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
static int
|
||||
@ -286,6 +304,7 @@ mt76_dma_tx_queue_skb(struct mt76_dev *dev, enum mt76_txq_id qid,
|
||||
struct mt76_tx_info tx_info = {
|
||||
.skb = skb,
|
||||
};
|
||||
struct ieee80211_hw *hw;
|
||||
int len, n = 0, ret = -ENOMEM;
|
||||
struct mt76_queue_entry e;
|
||||
struct mt76_txwi_cache *t;
|
||||
@ -295,7 +314,8 @@ mt76_dma_tx_queue_skb(struct mt76_dev *dev, enum mt76_txq_id qid,
|
||||
|
||||
t = mt76_get_txwi(dev);
|
||||
if (!t) {
|
||||
ieee80211_free_txskb(dev->hw, skb);
|
||||
hw = mt76_tx_status_get_hw(dev, skb);
|
||||
ieee80211_free_txskb(hw, skb);
|
||||
return -ENOMEM;
|
||||
}
|
||||
txwi = mt76_get_txwi_ptr(dev, t);
|
||||
@ -427,7 +447,7 @@ mt76_dma_rx_reset(struct mt76_dev *dev, enum mt76_rxq_id qid)
|
||||
int i;
|
||||
|
||||
for (i = 0; i < q->ndesc; i++)
|
||||
q->desc[i].ctrl &= ~cpu_to_le32(MT_DMA_CTL_DMA_DONE);
|
||||
q->desc[i].ctrl = cpu_to_le32(MT_DMA_CTL_DMA_DONE);
|
||||
|
||||
mt76_dma_rx_cleanup(dev, q);
|
||||
mt76_dma_sync_idx(dev, q);
|
||||
@ -528,6 +548,7 @@ mt76_dma_rx_poll(struct napi_struct *napi, int budget)
|
||||
dev = container_of(napi->dev, struct mt76_dev, napi_dev);
|
||||
qid = napi - dev->napi;
|
||||
|
||||
local_bh_disable();
|
||||
rcu_read_lock();
|
||||
|
||||
do {
|
||||
@ -537,6 +558,7 @@ mt76_dma_rx_poll(struct napi_struct *napi, int budget)
|
||||
} while (cur && done < budget);
|
||||
|
||||
rcu_read_unlock();
|
||||
local_bh_enable();
|
||||
|
||||
if (done < budget && napi_complete(napi))
|
||||
dev->drv->rx_poll_complete(dev, qid);
|
||||
@ -555,7 +577,6 @@ mt76_dma_init(struct mt76_dev *dev)
|
||||
netif_napi_add(&dev->napi_dev, &dev->napi[i], mt76_dma_rx_poll,
|
||||
64);
|
||||
mt76_dma_rx_fill(dev, &dev->q_rx[i]);
|
||||
skb_queue_head_init(&dev->rx_skb[i]);
|
||||
napi_enable(&dev->napi[i]);
|
||||
}
|
||||
|
||||
|
@ -64,6 +64,16 @@ mt76_get_of_eeprom(struct mt76_dev *dev, int len)
|
||||
goto out_put_node;
|
||||
}
|
||||
|
||||
if (of_property_read_bool(dev->dev->of_node, "big-endian")) {
|
||||
u8 *data = (u8 *)dev->eeprom.data;
|
||||
int i;
|
||||
|
||||
/* convert eeprom data in Little Endian */
|
||||
for (i = 0; i < round_down(len, 2); i += 2)
|
||||
put_unaligned_le16(get_unaligned_be16(&data[i]),
|
||||
&data[i]);
|
||||
}
|
||||
|
||||
out_put_node:
|
||||
of_node_put(np);
|
||||
return ret;
|
||||
@ -77,13 +87,11 @@ mt76_eeprom_override(struct mt76_dev *dev)
|
||||
{
|
||||
#ifdef CONFIG_OF
|
||||
struct device_node *np = dev->dev->of_node;
|
||||
const u8 *mac;
|
||||
const u8 *mac = NULL;
|
||||
|
||||
if (!np)
|
||||
return;
|
||||
|
||||
mac = of_get_mac_address(np);
|
||||
if (!IS_ERR(mac))
|
||||
if (np)
|
||||
mac = of_get_mac_address(np);
|
||||
if (!IS_ERR_OR_NULL(mac))
|
||||
ether_addr_copy(dev->macaddr, mac);
|
||||
#endif
|
||||
|
||||
|
@ -121,7 +121,7 @@ static void mt76_init_stream_cap(struct mt76_dev *dev,
|
||||
bool vht)
|
||||
{
|
||||
struct ieee80211_sta_ht_cap *ht_cap = &sband->ht_cap;
|
||||
int i, nstream = hweight8(dev->antenna_mask);
|
||||
int i, nstream = hweight8(dev->phy.antenna_mask);
|
||||
struct ieee80211_sta_vht_cap *vht_cap;
|
||||
u16 mcs_map = 0;
|
||||
|
||||
@ -156,9 +156,9 @@ static void mt76_init_stream_cap(struct mt76_dev *dev,
|
||||
void mt76_set_stream_caps(struct mt76_dev *dev, bool vht)
|
||||
{
|
||||
if (dev->cap.has_2ghz)
|
||||
mt76_init_stream_cap(dev, &dev->sband_2g.sband, false);
|
||||
mt76_init_stream_cap(dev, &dev->phy.sband_2g.sband, false);
|
||||
if (dev->cap.has_5ghz)
|
||||
mt76_init_stream_cap(dev, &dev->sband_5g.sband, vht);
|
||||
mt76_init_stream_cap(dev, &dev->phy.sband_5g.sband, vht);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_set_stream_caps);
|
||||
|
||||
@ -187,8 +187,6 @@ mt76_init_sband(struct mt76_dev *dev, struct mt76_sband *msband,
|
||||
sband->n_channels = n_chan;
|
||||
sband->bitrates = rates;
|
||||
sband->n_bitrates = n_rates;
|
||||
dev->chandef.chan = &sband->channels[0];
|
||||
dev->chan_state = &msband->chan[0];
|
||||
|
||||
ht_cap = &sband->ht_cap;
|
||||
ht_cap->ht_supported = true;
|
||||
@ -223,9 +221,9 @@ static int
|
||||
mt76_init_sband_2g(struct mt76_dev *dev, struct ieee80211_rate *rates,
|
||||
int n_rates)
|
||||
{
|
||||
dev->hw->wiphy->bands[NL80211_BAND_2GHZ] = &dev->sband_2g.sband;
|
||||
dev->hw->wiphy->bands[NL80211_BAND_2GHZ] = &dev->phy.sband_2g.sband;
|
||||
|
||||
return mt76_init_sband(dev, &dev->sband_2g,
|
||||
return mt76_init_sband(dev, &dev->phy.sband_2g,
|
||||
mt76_channels_2ghz,
|
||||
ARRAY_SIZE(mt76_channels_2ghz),
|
||||
rates, n_rates, false);
|
||||
@ -235,18 +233,19 @@ static int
|
||||
mt76_init_sband_5g(struct mt76_dev *dev, struct ieee80211_rate *rates,
|
||||
int n_rates, bool vht)
|
||||
{
|
||||
dev->hw->wiphy->bands[NL80211_BAND_5GHZ] = &dev->sband_5g.sband;
|
||||
dev->hw->wiphy->bands[NL80211_BAND_5GHZ] = &dev->phy.sband_5g.sband;
|
||||
|
||||
return mt76_init_sband(dev, &dev->sband_5g,
|
||||
return mt76_init_sband(dev, &dev->phy.sband_5g,
|
||||
mt76_channels_5ghz,
|
||||
ARRAY_SIZE(mt76_channels_5ghz),
|
||||
rates, n_rates, vht);
|
||||
}
|
||||
|
||||
static void
|
||||
mt76_check_sband(struct mt76_dev *dev, int band)
|
||||
mt76_check_sband(struct mt76_phy *phy, struct mt76_sband *msband,
|
||||
enum nl80211_band band)
|
||||
{
|
||||
struct ieee80211_supported_band *sband = dev->hw->wiphy->bands[band];
|
||||
struct ieee80211_supported_band *sband = &msband->sband;
|
||||
bool found = false;
|
||||
int i;
|
||||
|
||||
@ -261,71 +260,41 @@ mt76_check_sband(struct mt76_dev *dev, int band)
|
||||
break;
|
||||
}
|
||||
|
||||
if (found)
|
||||
if (found) {
|
||||
phy->chandef.chan = &sband->channels[0];
|
||||
phy->chan_state = &msband->chan[0];
|
||||
return;
|
||||
}
|
||||
|
||||
sband->n_channels = 0;
|
||||
dev->hw->wiphy->bands[band] = NULL;
|
||||
phy->hw->wiphy->bands[band] = NULL;
|
||||
}
|
||||
|
||||
struct mt76_dev *
|
||||
mt76_alloc_device(struct device *pdev, unsigned int size,
|
||||
const struct ieee80211_ops *ops,
|
||||
const struct mt76_driver_ops *drv_ops)
|
||||
static void
|
||||
mt76_phy_init(struct mt76_dev *dev, struct ieee80211_hw *hw)
|
||||
{
|
||||
struct ieee80211_hw *hw;
|
||||
struct mt76_dev *dev;
|
||||
|
||||
hw = ieee80211_alloc_hw(size, ops);
|
||||
if (!hw)
|
||||
return NULL;
|
||||
|
||||
dev = hw->priv;
|
||||
dev->hw = hw;
|
||||
dev->dev = pdev;
|
||||
dev->drv = drv_ops;
|
||||
|
||||
spin_lock_init(&dev->rx_lock);
|
||||
spin_lock_init(&dev->lock);
|
||||
spin_lock_init(&dev->cc_lock);
|
||||
mutex_init(&dev->mutex);
|
||||
init_waitqueue_head(&dev->tx_wait);
|
||||
skb_queue_head_init(&dev->status_list);
|
||||
|
||||
tasklet_init(&dev->tx_tasklet, mt76_tx_tasklet, (unsigned long)dev);
|
||||
|
||||
return dev;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_alloc_device);
|
||||
|
||||
int mt76_register_device(struct mt76_dev *dev, bool vht,
|
||||
struct ieee80211_rate *rates, int n_rates)
|
||||
{
|
||||
struct ieee80211_hw *hw = dev->hw;
|
||||
struct wiphy *wiphy = hw->wiphy;
|
||||
int ret;
|
||||
|
||||
dev_set_drvdata(dev->dev, dev);
|
||||
|
||||
INIT_LIST_HEAD(&dev->txwi_cache);
|
||||
|
||||
SET_IEEE80211_DEV(hw, dev->dev);
|
||||
SET_IEEE80211_PERM_ADDR(hw, dev->macaddr);
|
||||
|
||||
wiphy->features |= NL80211_FEATURE_ACTIVE_MONITOR;
|
||||
wiphy->flags |= WIPHY_FLAG_HAS_CHANNEL_SWITCH;
|
||||
|
||||
wiphy_ext_feature_set(wiphy, NL80211_EXT_FEATURE_CQM_RSSI_LIST);
|
||||
wiphy_ext_feature_set(wiphy, NL80211_EXT_FEATURE_AIRTIME_FAIRNESS);
|
||||
wiphy_ext_feature_set(wiphy, NL80211_EXT_FEATURE_AQL);
|
||||
|
||||
wiphy->available_antennas_tx = dev->antenna_mask;
|
||||
wiphy->available_antennas_rx = dev->antenna_mask;
|
||||
wiphy->available_antennas_tx = dev->phy.antenna_mask;
|
||||
wiphy->available_antennas_rx = dev->phy.antenna_mask;
|
||||
|
||||
hw->txq_data_size = sizeof(struct mt76_txq);
|
||||
hw->max_tx_fragments = 16;
|
||||
|
||||
if (!hw->max_tx_fragments)
|
||||
hw->max_tx_fragments = 16;
|
||||
|
||||
ieee80211_hw_set(hw, SIGNAL_DBM);
|
||||
ieee80211_hw_set(hw, PS_NULLFUNC_STACK);
|
||||
ieee80211_hw_set(hw, HOST_BROADCAST_PS_BUFFERING);
|
||||
ieee80211_hw_set(hw, AMPDU_AGGREGATION);
|
||||
ieee80211_hw_set(hw, SUPPORTS_RC_TABLE);
|
||||
ieee80211_hw_set(hw, SUPPORT_FAST_XMIT);
|
||||
@ -337,7 +306,6 @@ int mt76_register_device(struct mt76_dev *dev, bool vht,
|
||||
ieee80211_hw_set(hw, AP_LINK_PS);
|
||||
ieee80211_hw_set(hw, REPORTS_TX_ACK_STATUS);
|
||||
ieee80211_hw_set(hw, NEEDS_UNIQUE_STA_ADDR);
|
||||
ieee80211_hw_set(hw, SUPPORTS_REORDERING_BUFFER);
|
||||
|
||||
wiphy->flags |= WIPHY_FLAG_IBSS_RSN;
|
||||
wiphy->interface_modes =
|
||||
@ -347,6 +315,135 @@ int mt76_register_device(struct mt76_dev *dev, bool vht,
|
||||
BIT(NL80211_IFTYPE_MESH_POINT) |
|
||||
#endif
|
||||
BIT(NL80211_IFTYPE_ADHOC);
|
||||
}
|
||||
|
||||
struct mt76_phy *
|
||||
mt76_alloc_phy(struct mt76_dev *dev, unsigned int size,
|
||||
const struct ieee80211_ops *ops)
|
||||
{
|
||||
struct ieee80211_hw *hw;
|
||||
struct mt76_phy *phy;
|
||||
unsigned int phy_size, chan_size;
|
||||
unsigned int size_2g, size_5g;
|
||||
void *priv;
|
||||
|
||||
phy_size = ALIGN(sizeof(*phy), 8);
|
||||
chan_size = sizeof(dev->phy.sband_2g.chan[0]);
|
||||
size_2g = ALIGN(ARRAY_SIZE(mt76_channels_2ghz) * chan_size, 8);
|
||||
size_5g = ALIGN(ARRAY_SIZE(mt76_channels_5ghz) * chan_size, 8);
|
||||
|
||||
size += phy_size + size_2g + size_5g;
|
||||
hw = ieee80211_alloc_hw(size, ops);
|
||||
if (!hw)
|
||||
return NULL;
|
||||
|
||||
phy = hw->priv;
|
||||
phy->dev = dev;
|
||||
phy->hw = hw;
|
||||
|
||||
mt76_phy_init(dev, hw);
|
||||
|
||||
priv = hw->priv + phy_size;
|
||||
|
||||
phy->sband_2g = dev->phy.sband_2g;
|
||||
phy->sband_2g.chan = priv;
|
||||
priv += size_2g;
|
||||
|
||||
phy->sband_5g = dev->phy.sband_5g;
|
||||
phy->sband_5g.chan = priv;
|
||||
priv += size_5g;
|
||||
|
||||
phy->priv = priv;
|
||||
|
||||
hw->wiphy->bands[NL80211_BAND_2GHZ] = &phy->sband_2g.sband;
|
||||
hw->wiphy->bands[NL80211_BAND_5GHZ] = &phy->sband_5g.sband;
|
||||
|
||||
mt76_check_sband(phy, &phy->sband_2g, NL80211_BAND_2GHZ);
|
||||
mt76_check_sband(phy, &phy->sband_5g, NL80211_BAND_5GHZ);
|
||||
|
||||
return phy;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_alloc_phy);
|
||||
|
||||
int
|
||||
mt76_register_phy(struct mt76_phy *phy)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = ieee80211_register_hw(phy->hw);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
phy->dev->phy2 = phy;
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_register_phy);
|
||||
|
||||
void
|
||||
mt76_unregister_phy(struct mt76_phy *phy)
|
||||
{
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
|
||||
dev->phy2 = NULL;
|
||||
mt76_tx_status_check(dev, NULL, true);
|
||||
ieee80211_unregister_hw(phy->hw);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_unregister_phy);
|
||||
|
||||
struct mt76_dev *
|
||||
mt76_alloc_device(struct device *pdev, unsigned int size,
|
||||
const struct ieee80211_ops *ops,
|
||||
const struct mt76_driver_ops *drv_ops)
|
||||
{
|
||||
struct ieee80211_hw *hw;
|
||||
struct mt76_phy *phy;
|
||||
struct mt76_dev *dev;
|
||||
int i;
|
||||
|
||||
hw = ieee80211_alloc_hw(size, ops);
|
||||
if (!hw)
|
||||
return NULL;
|
||||
|
||||
dev = hw->priv;
|
||||
dev->hw = hw;
|
||||
dev->dev = pdev;
|
||||
dev->drv = drv_ops;
|
||||
|
||||
phy = &dev->phy;
|
||||
phy->dev = dev;
|
||||
phy->hw = hw;
|
||||
|
||||
spin_lock_init(&dev->rx_lock);
|
||||
spin_lock_init(&dev->lock);
|
||||
spin_lock_init(&dev->cc_lock);
|
||||
mutex_init(&dev->mutex);
|
||||
init_waitqueue_head(&dev->tx_wait);
|
||||
skb_queue_head_init(&dev->status_list);
|
||||
|
||||
skb_queue_head_init(&dev->mcu.res_q);
|
||||
init_waitqueue_head(&dev->mcu.wait);
|
||||
mutex_init(&dev->mcu.mutex);
|
||||
|
||||
INIT_LIST_HEAD(&dev->txwi_cache);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->q_rx); i++)
|
||||
skb_queue_head_init(&dev->rx_skb[i]);
|
||||
|
||||
tasklet_init(&dev->tx_tasklet, mt76_tx_tasklet, (unsigned long)dev);
|
||||
|
||||
return dev;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_alloc_device);
|
||||
|
||||
int mt76_register_device(struct mt76_dev *dev, bool vht,
|
||||
struct ieee80211_rate *rates, int n_rates)
|
||||
{
|
||||
struct ieee80211_hw *hw = dev->hw;
|
||||
struct mt76_phy *phy = &dev->phy;
|
||||
int ret;
|
||||
|
||||
dev_set_drvdata(dev->dev, dev);
|
||||
mt76_phy_init(dev, hw);
|
||||
|
||||
if (dev->cap.has_2ghz) {
|
||||
ret = mt76_init_sband_2g(dev, rates, n_rates);
|
||||
@ -360,9 +457,9 @@ int mt76_register_device(struct mt76_dev *dev, bool vht,
|
||||
return ret;
|
||||
}
|
||||
|
||||
wiphy_read_of_freq_limits(dev->hw->wiphy);
|
||||
mt76_check_sband(dev, NL80211_BAND_2GHZ);
|
||||
mt76_check_sband(dev, NL80211_BAND_5GHZ);
|
||||
wiphy_read_of_freq_limits(hw->wiphy);
|
||||
mt76_check_sband(&dev->phy, &phy->sband_2g, NL80211_BAND_2GHZ);
|
||||
mt76_check_sband(&dev->phy, &phy->sband_5g, NL80211_BAND_5GHZ);
|
||||
|
||||
if (IS_ENABLED(CONFIG_MT76_LEDS)) {
|
||||
ret = mt76_led_init(dev);
|
||||
@ -394,7 +491,10 @@ EXPORT_SYMBOL_GPL(mt76_free_device);
|
||||
|
||||
void mt76_rx(struct mt76_dev *dev, enum mt76_rxq_id q, struct sk_buff *skb)
|
||||
{
|
||||
if (!test_bit(MT76_STATE_RUNNING, &dev->state)) {
|
||||
struct mt76_rx_status *status = (struct mt76_rx_status *)skb->cb;
|
||||
struct mt76_phy *phy = mt76_dev_phy(dev, status->ext_phy);
|
||||
|
||||
if (!test_bit(MT76_STATE_RUNNING, &phy->state)) {
|
||||
dev_kfree_skb(skb);
|
||||
return;
|
||||
}
|
||||
@ -403,13 +503,16 @@ void mt76_rx(struct mt76_dev *dev, enum mt76_rxq_id q, struct sk_buff *skb)
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_rx);
|
||||
|
||||
bool mt76_has_tx_pending(struct mt76_dev *dev)
|
||||
bool mt76_has_tx_pending(struct mt76_phy *phy)
|
||||
{
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
struct mt76_queue *q;
|
||||
int i;
|
||||
int i, offset;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->q_tx); i++) {
|
||||
q = dev->q_tx[i].q;
|
||||
offset = __MT_TXQ_MAX * (phy != &dev->phy);
|
||||
|
||||
for (i = 0; i < __MT_TXQ_MAX; i++) {
|
||||
q = dev->q_tx[offset + i].q;
|
||||
if (q && q->queued)
|
||||
return true;
|
||||
}
|
||||
@ -419,37 +522,45 @@ bool mt76_has_tx_pending(struct mt76_dev *dev)
|
||||
EXPORT_SYMBOL_GPL(mt76_has_tx_pending);
|
||||
|
||||
static struct mt76_channel_state *
|
||||
mt76_channel_state(struct mt76_dev *dev, struct ieee80211_channel *c)
|
||||
mt76_channel_state(struct mt76_phy *phy, struct ieee80211_channel *c)
|
||||
{
|
||||
struct mt76_sband *msband;
|
||||
int idx;
|
||||
|
||||
if (c->band == NL80211_BAND_2GHZ)
|
||||
msband = &dev->sband_2g;
|
||||
msband = &phy->sband_2g;
|
||||
else
|
||||
msband = &dev->sband_5g;
|
||||
msband = &phy->sband_5g;
|
||||
|
||||
idx = c - &msband->sband.channels[0];
|
||||
return &msband->chan[idx];
|
||||
}
|
||||
|
||||
static void
|
||||
mt76_update_survey_active_time(struct mt76_phy *phy, ktime_t time)
|
||||
{
|
||||
struct mt76_channel_state *state = phy->chan_state;
|
||||
|
||||
state->cc_active += ktime_to_us(ktime_sub(time,
|
||||
phy->survey_time));
|
||||
phy->survey_time = time;
|
||||
}
|
||||
|
||||
void mt76_update_survey(struct mt76_dev *dev)
|
||||
{
|
||||
struct mt76_channel_state *state = dev->chan_state;
|
||||
ktime_t cur_time;
|
||||
|
||||
if (!test_bit(MT76_STATE_RUNNING, &dev->state))
|
||||
return;
|
||||
|
||||
if (dev->drv->update_survey)
|
||||
dev->drv->update_survey(dev);
|
||||
|
||||
cur_time = ktime_get_boottime();
|
||||
state->cc_active += ktime_to_us(ktime_sub(cur_time,
|
||||
dev->survey_time));
|
||||
dev->survey_time = cur_time;
|
||||
mt76_update_survey_active_time(&dev->phy, cur_time);
|
||||
if (dev->phy2)
|
||||
mt76_update_survey_active_time(dev->phy2, cur_time);
|
||||
|
||||
if (dev->drv->drv_flags & MT_DRV_SW_RX_AIRTIME) {
|
||||
struct mt76_channel_state *state = dev->phy.chan_state;
|
||||
|
||||
spin_lock_bh(&dev->cc_lock);
|
||||
state->cc_bss_rx += dev->cur_cc_bss_rx;
|
||||
dev->cur_cc_bss_rx = 0;
|
||||
@ -458,31 +569,33 @@ void mt76_update_survey(struct mt76_dev *dev)
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_update_survey);
|
||||
|
||||
void mt76_set_channel(struct mt76_dev *dev)
|
||||
void mt76_set_channel(struct mt76_phy *phy)
|
||||
{
|
||||
struct ieee80211_hw *hw = dev->hw;
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
struct ieee80211_hw *hw = phy->hw;
|
||||
struct cfg80211_chan_def *chandef = &hw->conf.chandef;
|
||||
bool offchannel = hw->conf.flags & IEEE80211_CONF_OFFCHANNEL;
|
||||
int timeout = HZ / 5;
|
||||
|
||||
wait_event_timeout(dev->tx_wait, !mt76_has_tx_pending(dev), timeout);
|
||||
wait_event_timeout(dev->tx_wait, !mt76_has_tx_pending(phy), timeout);
|
||||
mt76_update_survey(dev);
|
||||
|
||||
dev->chandef = *chandef;
|
||||
dev->chan_state = mt76_channel_state(dev, chandef->chan);
|
||||
phy->chandef = *chandef;
|
||||
phy->chan_state = mt76_channel_state(phy, chandef->chan);
|
||||
|
||||
if (!offchannel)
|
||||
dev->main_chan = chandef->chan;
|
||||
phy->main_chan = chandef->chan;
|
||||
|
||||
if (chandef->chan != dev->main_chan)
|
||||
memset(dev->chan_state, 0, sizeof(*dev->chan_state));
|
||||
if (chandef->chan != phy->main_chan)
|
||||
memset(phy->chan_state, 0, sizeof(*phy->chan_state));
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_set_channel);
|
||||
|
||||
int mt76_get_survey(struct ieee80211_hw *hw, int idx,
|
||||
struct survey_info *survey)
|
||||
{
|
||||
struct mt76_dev *dev = hw->priv;
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
struct mt76_sband *sband;
|
||||
struct ieee80211_channel *chan;
|
||||
struct mt76_channel_state *state;
|
||||
@ -492,10 +605,10 @@ int mt76_get_survey(struct ieee80211_hw *hw, int idx,
|
||||
if (idx == 0 && dev->drv->update_survey)
|
||||
mt76_update_survey(dev);
|
||||
|
||||
sband = &dev->sband_2g;
|
||||
sband = &phy->sband_2g;
|
||||
if (idx >= sband->sband.n_channels) {
|
||||
idx -= sband->sband.n_channels;
|
||||
sband = &dev->sband_5g;
|
||||
sband = &phy->sband_5g;
|
||||
}
|
||||
|
||||
if (idx >= sband->sband.n_channels) {
|
||||
@ -504,13 +617,16 @@ int mt76_get_survey(struct ieee80211_hw *hw, int idx,
|
||||
}
|
||||
|
||||
chan = &sband->sband.channels[idx];
|
||||
state = mt76_channel_state(dev, chan);
|
||||
state = mt76_channel_state(phy, chan);
|
||||
|
||||
memset(survey, 0, sizeof(*survey));
|
||||
survey->channel = chan;
|
||||
survey->filled = SURVEY_INFO_TIME | SURVEY_INFO_TIME_BUSY;
|
||||
survey->filled |= dev->drv->survey_flags;
|
||||
if (chan == dev->main_chan) {
|
||||
if (state->noise)
|
||||
survey->filled |= SURVEY_INFO_NOISE_DBM;
|
||||
|
||||
if (chan == phy->main_chan) {
|
||||
survey->filled |= SURVEY_INFO_IN_USE;
|
||||
|
||||
if (dev->drv->drv_flags & MT_DRV_SW_RX_AIRTIME)
|
||||
@ -520,6 +636,7 @@ int mt76_get_survey(struct ieee80211_hw *hw, int idx,
|
||||
survey->time_busy = div_u64(state->cc_busy, 1000);
|
||||
survey->time_rx = div_u64(state->cc_rx, 1000);
|
||||
survey->time = div_u64(state->cc_active, 1000);
|
||||
survey->noise = state->noise;
|
||||
|
||||
spin_lock_bh(&dev->cc_lock);
|
||||
survey->time_bss_rx = div_u64(state->cc_bss_rx, 1000);
|
||||
@ -555,8 +672,12 @@ void mt76_wcid_key_setup(struct mt76_dev *dev, struct mt76_wcid *wcid,
|
||||
}
|
||||
EXPORT_SYMBOL(mt76_wcid_key_setup);
|
||||
|
||||
static struct ieee80211_sta *mt76_rx_convert(struct sk_buff *skb)
|
||||
static void
|
||||
mt76_rx_convert(struct mt76_dev *dev, struct sk_buff *skb,
|
||||
struct ieee80211_hw **hw,
|
||||
struct ieee80211_sta **sta)
|
||||
{
|
||||
|
||||
struct ieee80211_rx_status *status = IEEE80211_SKB_RXCB(skb);
|
||||
struct mt76_rx_status mstat;
|
||||
|
||||
@ -581,7 +702,8 @@ static struct ieee80211_sta *mt76_rx_convert(struct sk_buff *skb)
|
||||
memcpy(status->chain_signal, mstat.chain_signal,
|
||||
sizeof(mstat.chain_signal));
|
||||
|
||||
return wcid_to_sta(mstat.wcid);
|
||||
*sta = wcid_to_sta(mstat.wcid);
|
||||
*hw = mt76_phy_hw(dev, mstat.ext_phy);
|
||||
}
|
||||
|
||||
static int
|
||||
@ -628,10 +750,18 @@ mt76_airtime_report(struct mt76_dev *dev, struct mt76_rx_status *status,
|
||||
int len)
|
||||
{
|
||||
struct mt76_wcid *wcid = status->wcid;
|
||||
struct ieee80211_rx_status info = {
|
||||
.enc_flags = status->enc_flags,
|
||||
.rate_idx = status->rate_idx,
|
||||
.encoding = status->encoding,
|
||||
.band = status->band,
|
||||
.nss = status->nss,
|
||||
.bw = status->bw,
|
||||
};
|
||||
struct ieee80211_sta *sta;
|
||||
u32 airtime;
|
||||
|
||||
airtime = mt76_calc_rx_airtime(dev, status, len);
|
||||
airtime = ieee80211_calc_rx_airtime(dev->hw, &info, len);
|
||||
spin_lock(&dev->cc_lock);
|
||||
dev->cur_cc_bss_rx += airtime;
|
||||
spin_unlock(&dev->cc_lock);
|
||||
@ -707,12 +837,14 @@ mt76_check_sta(struct mt76_dev *dev, struct sk_buff *skb)
|
||||
struct mt76_rx_status *status = (struct mt76_rx_status *)skb->cb;
|
||||
struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data;
|
||||
struct ieee80211_sta *sta;
|
||||
struct ieee80211_hw *hw;
|
||||
struct mt76_wcid *wcid = status->wcid;
|
||||
bool ps;
|
||||
int i;
|
||||
|
||||
hw = mt76_phy_hw(dev, status->ext_phy);
|
||||
if (ieee80211_is_pspoll(hdr->frame_control) && !wcid) {
|
||||
sta = ieee80211_find_sta_by_ifaddr(dev->hw, hdr->addr2, NULL);
|
||||
sta = ieee80211_find_sta_by_ifaddr(hw, hdr->addr2, NULL);
|
||||
if (sta)
|
||||
wcid = status->wcid = (struct mt76_wcid *)sta->drv_priv;
|
||||
}
|
||||
@ -770,7 +902,7 @@ mt76_check_sta(struct mt76_dev *dev, struct sk_buff *skb)
|
||||
|
||||
mtxq = (struct mt76_txq *)sta->txq[i]->drv_priv;
|
||||
if (!skb_queue_empty(&mtxq->retry_q))
|
||||
ieee80211_schedule_txq(dev->hw, sta->txq[i]);
|
||||
ieee80211_schedule_txq(hw, sta->txq[i]);
|
||||
}
|
||||
}
|
||||
|
||||
@ -778,6 +910,7 @@ void mt76_rx_complete(struct mt76_dev *dev, struct sk_buff_head *frames,
|
||||
struct napi_struct *napi)
|
||||
{
|
||||
struct ieee80211_sta *sta;
|
||||
struct ieee80211_hw *hw;
|
||||
struct sk_buff *skb;
|
||||
|
||||
spin_lock(&dev->rx_lock);
|
||||
@ -787,8 +920,8 @@ void mt76_rx_complete(struct mt76_dev *dev, struct sk_buff_head *frames,
|
||||
continue;
|
||||
}
|
||||
|
||||
sta = mt76_rx_convert(skb);
|
||||
ieee80211_rx_napi(dev->hw, sta, skb, napi);
|
||||
mt76_rx_convert(dev, skb, &hw, &sta);
|
||||
ieee80211_rx_napi(hw, sta, skb, napi);
|
||||
}
|
||||
spin_unlock(&dev->rx_lock);
|
||||
}
|
||||
@ -812,7 +945,7 @@ EXPORT_SYMBOL_GPL(mt76_rx_poll_complete);
|
||||
|
||||
static int
|
||||
mt76_sta_add(struct mt76_dev *dev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta)
|
||||
struct ieee80211_sta *sta, bool ext_phy)
|
||||
{
|
||||
struct mt76_wcid *wcid = (struct mt76_wcid *)sta->drv_priv;
|
||||
int ret;
|
||||
@ -837,6 +970,9 @@ mt76_sta_add(struct mt76_dev *dev, struct ieee80211_vif *vif,
|
||||
}
|
||||
|
||||
ewma_signal_init(&wcid->rssi);
|
||||
if (ext_phy)
|
||||
mt76_wcid_mask_set(dev->wcid_phy_mask, wcid->idx);
|
||||
wcid->ext_phy = ext_phy;
|
||||
rcu_assign_pointer(dev->wcid[wcid->idx], wcid);
|
||||
|
||||
out:
|
||||
@ -851,9 +987,6 @@ void __mt76_sta_remove(struct mt76_dev *dev, struct ieee80211_vif *vif,
|
||||
struct mt76_wcid *wcid = (struct mt76_wcid *)sta->drv_priv;
|
||||
int i, idx = wcid->idx;
|
||||
|
||||
rcu_assign_pointer(dev->wcid[idx], NULL);
|
||||
synchronize_rcu();
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(wcid->aggr); i++)
|
||||
mt76_rx_aggr_stop(dev, wcid, i);
|
||||
|
||||
@ -863,7 +996,8 @@ void __mt76_sta_remove(struct mt76_dev *dev, struct ieee80211_vif *vif,
|
||||
mt76_tx_status_check(dev, wcid, true);
|
||||
for (i = 0; i < ARRAY_SIZE(sta->txq); i++)
|
||||
mt76_txq_remove(dev, sta->txq[i]);
|
||||
mt76_wcid_free(dev->wcid_mask, idx);
|
||||
mt76_wcid_mask_clear(dev->wcid_mask, idx);
|
||||
mt76_wcid_mask_clear(dev->wcid_phy_mask, idx);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(__mt76_sta_remove);
|
||||
|
||||
@ -881,11 +1015,13 @@ int mt76_sta_state(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
enum ieee80211_sta_state old_state,
|
||||
enum ieee80211_sta_state new_state)
|
||||
{
|
||||
struct mt76_dev *dev = hw->priv;
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
bool ext_phy = phy != &dev->phy;
|
||||
|
||||
if (old_state == IEEE80211_STA_NOTEXIST &&
|
||||
new_state == IEEE80211_STA_NONE)
|
||||
return mt76_sta_add(dev, vif, sta);
|
||||
return mt76_sta_add(dev, vif, sta, ext_phy);
|
||||
|
||||
if (old_state == IEEE80211_STA_AUTH &&
|
||||
new_state == IEEE80211_STA_ASSOC &&
|
||||
@ -900,30 +1036,27 @@ int mt76_sta_state(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_sta_state);
|
||||
|
||||
void mt76_sta_pre_rcu_remove(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta)
|
||||
{
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
struct mt76_wcid *wcid = (struct mt76_wcid *)sta->drv_priv;
|
||||
|
||||
mutex_lock(&dev->mutex);
|
||||
rcu_assign_pointer(dev->wcid[wcid->idx], NULL);
|
||||
mutex_unlock(&dev->mutex);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_sta_pre_rcu_remove);
|
||||
|
||||
int mt76_get_txpower(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
int *dbm)
|
||||
{
|
||||
struct mt76_dev *dev = hw->priv;
|
||||
int n_chains = hweight8(dev->antenna_mask);
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
int n_chains = hweight8(phy->antenna_mask);
|
||||
int delta = mt76_tx_power_nss_delta(n_chains);
|
||||
|
||||
*dbm = DIV_ROUND_UP(dev->txpower_cur, 2);
|
||||
|
||||
/* convert from per-chain power to combined
|
||||
* output power
|
||||
*/
|
||||
switch (n_chains) {
|
||||
case 4:
|
||||
*dbm += 6;
|
||||
break;
|
||||
case 3:
|
||||
*dbm += 4;
|
||||
break;
|
||||
case 2:
|
||||
*dbm += 3;
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
*dbm = DIV_ROUND_UP(phy->txpower_cur + delta, 2);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -1005,11 +1138,11 @@ int mt76_get_rate(struct mt76_dev *dev,
|
||||
int i, offset = 0, len = sband->n_bitrates;
|
||||
|
||||
if (cck) {
|
||||
if (sband == &dev->sband_5g.sband)
|
||||
if (sband == &dev->phy.sband_5g.sband)
|
||||
return 0;
|
||||
|
||||
idx &= ~BIT(2); /* short preamble */
|
||||
} else if (sband == &dev->sband_2g.sband) {
|
||||
} else if (sband == &dev->phy.sband_2g.sband) {
|
||||
offset = 4;
|
||||
}
|
||||
|
||||
@ -1025,27 +1158,28 @@ EXPORT_SYMBOL_GPL(mt76_get_rate);
|
||||
void mt76_sw_scan(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
const u8 *mac)
|
||||
{
|
||||
struct mt76_dev *dev = hw->priv;
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
|
||||
set_bit(MT76_SCANNING, &dev->state);
|
||||
set_bit(MT76_SCANNING, &phy->state);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_sw_scan);
|
||||
|
||||
void mt76_sw_scan_complete(struct ieee80211_hw *hw, struct ieee80211_vif *vif)
|
||||
{
|
||||
struct mt76_dev *dev = hw->priv;
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
|
||||
clear_bit(MT76_SCANNING, &dev->state);
|
||||
clear_bit(MT76_SCANNING, &phy->state);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_sw_scan_complete);
|
||||
|
||||
int mt76_get_antenna(struct ieee80211_hw *hw, u32 *tx_ant, u32 *rx_ant)
|
||||
{
|
||||
struct mt76_dev *dev = hw->priv;
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
|
||||
mutex_lock(&dev->mutex);
|
||||
*tx_ant = dev->antenna_mask;
|
||||
*rx_ant = dev->antenna_mask;
|
||||
*tx_ant = phy->antenna_mask;
|
||||
*rx_ant = phy->antenna_mask;
|
||||
mutex_unlock(&dev->mutex);
|
||||
|
||||
return 0;
|
||||
|
@ -24,7 +24,6 @@ mt76_mcu_msg_alloc(const void *data, int head_len,
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_mcu_msg_alloc);
|
||||
|
||||
/* mmio */
|
||||
struct sk_buff *mt76_mcu_get_response(struct mt76_dev *dev,
|
||||
unsigned long expires)
|
||||
{
|
||||
@ -34,16 +33,17 @@ struct sk_buff *mt76_mcu_get_response(struct mt76_dev *dev,
|
||||
return NULL;
|
||||
|
||||
timeout = expires - jiffies;
|
||||
wait_event_timeout(dev->mmio.mcu.wait,
|
||||
!skb_queue_empty(&dev->mmio.mcu.res_q),
|
||||
wait_event_timeout(dev->mcu.wait,
|
||||
(!skb_queue_empty(&dev->mcu.res_q) ||
|
||||
test_bit(MT76_MCU_RESET, &dev->phy.state)),
|
||||
timeout);
|
||||
return skb_dequeue(&dev->mmio.mcu.res_q);
|
||||
return skb_dequeue(&dev->mcu.res_q);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_mcu_get_response);
|
||||
|
||||
void mt76_mcu_rx_event(struct mt76_dev *dev, struct sk_buff *skb)
|
||||
{
|
||||
skb_queue_tail(&dev->mmio.mcu.res_q, skb);
|
||||
wake_up(&dev->mmio.mcu.wait);
|
||||
skb_queue_tail(&dev->mcu.res_q, skb);
|
||||
wake_up(&dev->mcu.wait);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_mcu_rx_event);
|
||||
|
@ -94,9 +94,6 @@ void mt76_mmio_init(struct mt76_dev *dev, void __iomem *regs)
|
||||
dev->bus = &mt76_mmio_ops;
|
||||
dev->mmio.regs = regs;
|
||||
|
||||
skb_queue_head_init(&dev->mmio.mcu.res_q);
|
||||
init_waitqueue_head(&dev->mmio.mcu.wait);
|
||||
spin_lock_init(&dev->mmio.irq_lock);
|
||||
mutex_init(&dev->mmio.mcu.mutex);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_mmio_init);
|
||||
|
@ -22,6 +22,7 @@
|
||||
#define MT_SKB_HEAD_LEN 128
|
||||
|
||||
struct mt76_dev;
|
||||
struct mt76_phy;
|
||||
struct mt76_wcid;
|
||||
|
||||
struct mt76_reg_pair {
|
||||
@ -177,6 +178,9 @@ enum mt76_wcid_flags {
|
||||
|
||||
#define MT76_N_WCIDS 128
|
||||
|
||||
/* stored in ieee80211_tx_info::hw_queue */
|
||||
#define MT_TX_HW_QUEUE_EXT_PHY BIT(3)
|
||||
|
||||
DECLARE_EWMA(signal, 10, 8);
|
||||
|
||||
#define MT_WCID_TX_INFO_RATE GENMASK(15, 0)
|
||||
@ -196,6 +200,7 @@ struct mt76_wcid {
|
||||
u8 hw_key_idx;
|
||||
|
||||
u8 sta:1;
|
||||
u8 ext_phy:1;
|
||||
|
||||
u8 rx_check_pn;
|
||||
u8 rx_key_pn[IEEE80211_NUM_TIDS][6];
|
||||
@ -237,6 +242,8 @@ struct mt76_rx_tid {
|
||||
u8 size;
|
||||
u8 nframes;
|
||||
|
||||
u8 num;
|
||||
|
||||
u8 started:1, stopped:1, timer_pending:1;
|
||||
|
||||
struct sk_buff *reorder_buf[];
|
||||
@ -267,6 +274,7 @@ enum {
|
||||
MT76_STATE_MCU_RUNNING,
|
||||
MT76_SCANNING,
|
||||
MT76_RESET,
|
||||
MT76_MCU_RESET,
|
||||
MT76_REMOVED,
|
||||
MT76_READING_STATS,
|
||||
};
|
||||
@ -279,6 +287,7 @@ struct mt76_hw_cap {
|
||||
#define MT_DRV_TXWI_NO_FREE BIT(0)
|
||||
#define MT_DRV_TX_ALIGNED4_SKBS BIT(1)
|
||||
#define MT_DRV_SW_RX_AIRTIME BIT(2)
|
||||
#define MT_DRV_RX_DMA_HDR BIT(3)
|
||||
|
||||
struct mt76_driver_ops {
|
||||
u32 drv_flags;
|
||||
@ -321,6 +330,8 @@ struct mt76_channel_state {
|
||||
u64 cc_rx;
|
||||
u64 cc_bss_rx;
|
||||
u64 cc_tx;
|
||||
|
||||
s8 noise;
|
||||
};
|
||||
|
||||
struct mt76_sband {
|
||||
@ -350,12 +361,15 @@ struct mt76_rate_power {
|
||||
enum mt_vendor_req {
|
||||
MT_VEND_DEV_MODE = 0x1,
|
||||
MT_VEND_WRITE = 0x2,
|
||||
MT_VEND_POWER_ON = 0x4,
|
||||
MT_VEND_MULTI_WRITE = 0x6,
|
||||
MT_VEND_MULTI_READ = 0x7,
|
||||
MT_VEND_READ_EEPROM = 0x9,
|
||||
MT_VEND_WRITE_FCE = 0x42,
|
||||
MT_VEND_WRITE_CFG = 0x46,
|
||||
MT_VEND_READ_CFG = 0x47,
|
||||
MT_VEND_READ_EXT = 0x63,
|
||||
MT_VEND_WRITE_EXT = 0x66,
|
||||
};
|
||||
|
||||
enum mt76u_in_ep {
|
||||
@ -374,20 +388,27 @@ enum mt76u_out_ep {
|
||||
__MT_EP_OUT_MAX,
|
||||
};
|
||||
|
||||
struct mt76_mcu {
|
||||
struct mutex mutex;
|
||||
u32 msg_seq;
|
||||
|
||||
struct sk_buff_head res_q;
|
||||
wait_queue_head_t wait;
|
||||
};
|
||||
|
||||
#define MT_TX_SG_MAX_SIZE 8
|
||||
#define MT_RX_SG_MAX_SIZE 1
|
||||
#define MT_RX_SG_MAX_SIZE 4
|
||||
#define MT_NUM_TX_ENTRIES 256
|
||||
#define MT_NUM_RX_ENTRIES 128
|
||||
#define MCU_RESP_URB_SIZE 1024
|
||||
struct mt76_usb {
|
||||
struct mutex usb_ctrl_mtx;
|
||||
union {
|
||||
u8 data[32];
|
||||
__le32 reg_val;
|
||||
};
|
||||
__le32 reg_val;
|
||||
u8 *data;
|
||||
u16 data_len;
|
||||
|
||||
struct tasklet_struct rx_tasklet;
|
||||
struct workqueue_struct *stat_wq;
|
||||
struct workqueue_struct *wq;
|
||||
struct work_struct stat_work;
|
||||
|
||||
u8 out_ep[__MT_EP_OUT_MAX];
|
||||
@ -395,10 +416,7 @@ struct mt76_usb {
|
||||
bool sg_en;
|
||||
|
||||
struct mt76u_mcu {
|
||||
struct mutex mutex;
|
||||
u8 *data;
|
||||
u32 msg_seq;
|
||||
|
||||
/* multiple reads */
|
||||
struct mt76_reg_pair *rp;
|
||||
int rp_len;
|
||||
@ -408,14 +426,6 @@ struct mt76_usb {
|
||||
};
|
||||
|
||||
struct mt76_mmio {
|
||||
struct mt76e_mcu {
|
||||
struct mutex mutex;
|
||||
|
||||
wait_queue_head_t wait;
|
||||
struct sk_buff_head res_q;
|
||||
|
||||
u32 msg_seq;
|
||||
} mcu;
|
||||
void __iomem *regs;
|
||||
spinlock_t irq_lock;
|
||||
u32 irqmask;
|
||||
@ -433,6 +443,7 @@ struct mt76_rx_status {
|
||||
|
||||
u8 iv[6];
|
||||
|
||||
u8 ext_phy:1;
|
||||
u8 aggr:1;
|
||||
u8 tid;
|
||||
u16 seqno;
|
||||
@ -449,12 +460,33 @@ struct mt76_rx_status {
|
||||
s8 chain_signal[IEEE80211_MAX_CHAINS];
|
||||
};
|
||||
|
||||
struct mt76_dev {
|
||||
struct mt76_phy {
|
||||
struct ieee80211_hw *hw;
|
||||
struct mt76_dev *dev;
|
||||
void *priv;
|
||||
|
||||
unsigned long state;
|
||||
|
||||
struct cfg80211_chan_def chandef;
|
||||
struct ieee80211_channel *main_chan;
|
||||
|
||||
struct mt76_channel_state *chan_state;
|
||||
ktime_t survey_time;
|
||||
|
||||
struct mt76_sband sband_2g;
|
||||
struct mt76_sband sband_5g;
|
||||
|
||||
int txpower_cur;
|
||||
u8 antenna_mask;
|
||||
};
|
||||
|
||||
struct mt76_dev {
|
||||
struct mt76_phy phy; /* must be first */
|
||||
|
||||
struct mt76_phy *phy2;
|
||||
|
||||
struct ieee80211_hw *hw;
|
||||
|
||||
spinlock_t lock;
|
||||
spinlock_t cc_lock;
|
||||
|
||||
@ -471,14 +503,15 @@ struct mt76_dev {
|
||||
const struct mt76_mcu_ops *mcu_ops;
|
||||
struct device *dev;
|
||||
|
||||
struct mt76_mcu mcu;
|
||||
|
||||
struct net_device napi_dev;
|
||||
spinlock_t rx_lock;
|
||||
struct napi_struct napi[__MT_RXQ_MAX];
|
||||
struct sk_buff_head rx_skb[__MT_RXQ_MAX];
|
||||
u32 ampdu_ref;
|
||||
|
||||
struct list_head txwi_cache;
|
||||
struct mt76_sw_queue q_tx[__MT_TXQ_MAX];
|
||||
struct mt76_sw_queue q_tx[2 * __MT_TXQ_MAX];
|
||||
struct mt76_queue q_rx[__MT_RXQ_MAX];
|
||||
const struct mt76_queue_ops *queue_ops;
|
||||
int tx_dma_idx[4];
|
||||
@ -491,32 +524,25 @@ struct mt76_dev {
|
||||
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];
|
||||
|
||||
struct mt76_wcid global_wcid;
|
||||
struct mt76_wcid __rcu *wcid[MT76_N_WCIDS];
|
||||
|
||||
u8 macaddr[ETH_ALEN];
|
||||
u32 rev;
|
||||
unsigned long state;
|
||||
|
||||
u32 aggr_stats[32];
|
||||
|
||||
u8 antenna_mask;
|
||||
u16 chainmask;
|
||||
|
||||
struct tasklet_struct pre_tbtt_tasklet;
|
||||
int beacon_int;
|
||||
u8 beacon_mask;
|
||||
|
||||
struct mt76_sband sband_2g;
|
||||
struct mt76_sband sband_5g;
|
||||
struct debugfs_blob_wrapper eeprom;
|
||||
struct debugfs_blob_wrapper otp;
|
||||
struct mt76_hw_cap cap;
|
||||
|
||||
struct mt76_rate_power rate_power;
|
||||
int txpower_conf;
|
||||
int txpower_cur;
|
||||
|
||||
enum nl80211_dfs_regions region;
|
||||
|
||||
@ -529,8 +555,6 @@ struct mt76_dev {
|
||||
|
||||
u8 csa_complete;
|
||||
|
||||
ktime_t survey_time;
|
||||
|
||||
u32 rxfilter;
|
||||
|
||||
union {
|
||||
@ -581,7 +605,17 @@ enum mt76_phy_type {
|
||||
#define __mt76_rmw_field(_dev, _reg, _field, _val) \
|
||||
__mt76_rmw(_dev, _reg, _field, FIELD_PREP(_field, _val))
|
||||
|
||||
#define mt76_hw(dev) (dev)->mt76.hw
|
||||
#define mt76_hw(dev) (dev)->mphy.hw
|
||||
|
||||
static inline struct ieee80211_hw *
|
||||
mt76_wcid_hw(struct mt76_dev *dev, u8 wcid)
|
||||
{
|
||||
if (wcid <= MT76_N_WCIDS &&
|
||||
mt76_wcid_mask_test(dev->wcid_phy_mask, wcid))
|
||||
return dev->phy2->hw;
|
||||
|
||||
return dev->phy.hw;
|
||||
}
|
||||
|
||||
bool __mt76_poll(struct mt76_dev *dev, u32 offset, u32 mask, u32 val,
|
||||
int timeout);
|
||||
@ -624,6 +658,11 @@ int mt76_register_device(struct mt76_dev *dev, bool vht,
|
||||
struct ieee80211_rate *rates, int n_rates);
|
||||
void mt76_unregister_device(struct mt76_dev *dev);
|
||||
void mt76_free_device(struct mt76_dev *dev);
|
||||
void mt76_unregister_phy(struct mt76_phy *phy);
|
||||
|
||||
struct mt76_phy *mt76_alloc_phy(struct mt76_dev *dev, unsigned int size,
|
||||
const struct ieee80211_ops *ops);
|
||||
int mt76_register_phy(struct mt76_phy *phy);
|
||||
|
||||
struct dentry *mt76_register_debugfs(struct mt76_dev *dev);
|
||||
int mt76_queues_read(struct seq_file *s, void *data);
|
||||
@ -633,6 +672,20 @@ void mt76_seq_puts_array(struct seq_file *file, const char *str,
|
||||
int mt76_eeprom_init(struct mt76_dev *dev, int len);
|
||||
void mt76_eeprom_override(struct mt76_dev *dev);
|
||||
|
||||
static inline struct mt76_phy *
|
||||
mt76_dev_phy(struct mt76_dev *dev, bool phy_ext)
|
||||
{
|
||||
if (phy_ext && dev->phy2)
|
||||
return dev->phy2;
|
||||
return &dev->phy;
|
||||
}
|
||||
|
||||
static inline struct ieee80211_hw *
|
||||
mt76_phy_hw(struct mt76_dev *dev, bool phy_ext)
|
||||
{
|
||||
return mt76_dev_phy(dev, phy_ext)->hw;
|
||||
}
|
||||
|
||||
static inline u8 *
|
||||
mt76_get_txwi_ptr(struct mt76_dev *dev, struct mt76_txwi_cache *t)
|
||||
{
|
||||
@ -701,24 +754,31 @@ static inline bool mt76_is_skb_pktid(u8 pktid)
|
||||
return pktid >= MT_PACKET_ID_FIRST;
|
||||
}
|
||||
|
||||
static inline u8 mt76_tx_power_nss_delta(u8 nss)
|
||||
{
|
||||
static const u8 nss_delta[4] = { 0, 6, 9, 12 };
|
||||
|
||||
return nss_delta[nss - 1];
|
||||
}
|
||||
|
||||
void mt76_rx(struct mt76_dev *dev, enum mt76_rxq_id q, struct sk_buff *skb);
|
||||
void mt76_tx(struct mt76_dev *dev, struct ieee80211_sta *sta,
|
||||
void mt76_tx(struct mt76_phy *dev, struct ieee80211_sta *sta,
|
||||
struct mt76_wcid *wcid, struct sk_buff *skb);
|
||||
void mt76_txq_init(struct mt76_dev *dev, struct ieee80211_txq *txq);
|
||||
void mt76_txq_remove(struct mt76_dev *dev, struct ieee80211_txq *txq);
|
||||
void mt76_wake_tx_queue(struct ieee80211_hw *hw, struct ieee80211_txq *txq);
|
||||
void mt76_stop_tx_queues(struct mt76_dev *dev, struct ieee80211_sta *sta,
|
||||
bool send_bar);
|
||||
void mt76_txq_schedule(struct mt76_dev *dev, enum mt76_txq_id qid);
|
||||
void mt76_txq_schedule_all(struct mt76_dev *dev);
|
||||
void mt76_txq_schedule(struct mt76_phy *phy, enum mt76_txq_id qid);
|
||||
void mt76_txq_schedule_all(struct mt76_phy *phy);
|
||||
void mt76_tx_tasklet(unsigned long data);
|
||||
void mt76_release_buffered_frames(struct ieee80211_hw *hw,
|
||||
struct ieee80211_sta *sta,
|
||||
u16 tids, int nframes,
|
||||
enum ieee80211_frame_release_type reason,
|
||||
bool more_data);
|
||||
bool mt76_has_tx_pending(struct mt76_dev *dev);
|
||||
void mt76_set_channel(struct mt76_dev *dev);
|
||||
bool mt76_has_tx_pending(struct mt76_phy *phy);
|
||||
void mt76_set_channel(struct mt76_phy *phy);
|
||||
void mt76_update_survey(struct mt76_dev *dev);
|
||||
int mt76_get_survey(struct ieee80211_hw *hw, int idx,
|
||||
struct survey_info *survey);
|
||||
@ -752,8 +812,10 @@ int mt76_sta_state(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
enum ieee80211_sta_state new_state);
|
||||
void __mt76_sta_remove(struct mt76_dev *dev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta);
|
||||
void mt76_sta_pre_rcu_remove(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta);
|
||||
|
||||
int mt76_get_min_avg_rssi(struct mt76_dev *dev);
|
||||
int mt76_get_min_avg_rssi(struct mt76_dev *dev, bool ext_phy);
|
||||
|
||||
int mt76_get_txpower(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
int *dbm);
|
||||
@ -771,10 +833,22 @@ void mt76_sw_scan(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
const u8 *mac);
|
||||
void mt76_sw_scan_complete(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif);
|
||||
u32 mt76_calc_tx_airtime(struct mt76_dev *dev, struct ieee80211_tx_info *info,
|
||||
int len);
|
||||
|
||||
/* internal */
|
||||
static inline struct ieee80211_hw *
|
||||
mt76_tx_status_get_hw(struct mt76_dev *dev, struct sk_buff *skb)
|
||||
{
|
||||
struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
|
||||
struct ieee80211_hw *hw = dev->phy.hw;
|
||||
|
||||
if ((info->hw_queue & MT_TX_HW_QUEUE_EXT_PHY) && dev->phy2)
|
||||
hw = dev->phy2->hw;
|
||||
|
||||
info->hw_queue &= ~MT_TX_HW_QUEUE_EXT_PHY;
|
||||
|
||||
return hw;
|
||||
}
|
||||
|
||||
void mt76_tx_free(struct mt76_dev *dev);
|
||||
struct mt76_txwi_cache *mt76_get_txwi(struct mt76_dev *dev);
|
||||
void mt76_put_txwi(struct mt76_dev *dev, struct mt76_txwi_cache *t);
|
||||
@ -783,8 +857,6 @@ void mt76_rx_complete(struct mt76_dev *dev, struct sk_buff_head *frames,
|
||||
void mt76_rx_poll_complete(struct mt76_dev *dev, enum mt76_rxq_id q,
|
||||
struct napi_struct *napi);
|
||||
void mt76_rx_aggr_reorder(struct sk_buff *skb, struct sk_buff_head *frames);
|
||||
u32 mt76_calc_rx_airtime(struct mt76_dev *dev, struct mt76_rx_status *status,
|
||||
int len);
|
||||
|
||||
/* usb */
|
||||
static inline bool mt76u_urb_error(struct urb *urb)
|
||||
@ -804,7 +876,7 @@ static inline u8 q2ep(u8 qid)
|
||||
|
||||
static inline int
|
||||
mt76u_bulk_msg(struct mt76_dev *dev, void *data, int len, int *actual_len,
|
||||
int timeout)
|
||||
int timeout, int ep)
|
||||
{
|
||||
struct usb_interface *uintf = to_usb_interface(dev->dev);
|
||||
struct usb_device *udev = interface_to_usbdev(uintf);
|
||||
@ -812,20 +884,23 @@ mt76u_bulk_msg(struct mt76_dev *dev, void *data, int len, int *actual_len,
|
||||
unsigned int pipe;
|
||||
|
||||
if (actual_len)
|
||||
pipe = usb_rcvbulkpipe(udev, usb->in_ep[MT_EP_IN_CMD_RESP]);
|
||||
pipe = usb_rcvbulkpipe(udev, usb->in_ep[ep]);
|
||||
else
|
||||
pipe = usb_sndbulkpipe(udev, usb->out_ep[MT_EP_OUT_INBAND_CMD]);
|
||||
pipe = usb_sndbulkpipe(udev, usb->out_ep[ep]);
|
||||
|
||||
return usb_bulk_msg(udev, pipe, data, len, actual_len, timeout);
|
||||
}
|
||||
|
||||
int mt76u_skb_dma_info(struct sk_buff *skb, u32 info);
|
||||
int mt76u_vendor_request(struct mt76_dev *dev, u8 req,
|
||||
u8 req_type, u16 val, u16 offset,
|
||||
void *buf, size_t len);
|
||||
void mt76u_single_wr(struct mt76_dev *dev, const u8 req,
|
||||
const u16 offset, const u32 val);
|
||||
int mt76u_init(struct mt76_dev *dev, struct usb_interface *intf);
|
||||
void mt76u_deinit(struct mt76_dev *dev);
|
||||
int mt76u_init(struct mt76_dev *dev, struct usb_interface *intf,
|
||||
bool ext);
|
||||
int mt76u_alloc_mcu_queue(struct mt76_dev *dev);
|
||||
int mt76u_alloc_queues(struct mt76_dev *dev);
|
||||
void mt76u_stop_tx(struct mt76_dev *dev);
|
||||
void mt76u_stop_rx(struct mt76_dev *dev);
|
||||
|
@ -1,6 +1,7 @@
|
||||
// SPDX-License-Identifier: ISC
|
||||
|
||||
#include "mt7603.h"
|
||||
#include "../trace.h"
|
||||
|
||||
void mt7603_rx_poll_complete(struct mt76_dev *mdev, enum mt76_rxq_id q)
|
||||
{
|
||||
@ -17,9 +18,11 @@ irqreturn_t mt7603_irq_handler(int irq, void *dev_instance)
|
||||
intr = mt76_rr(dev, MT_INT_SOURCE_CSR);
|
||||
mt76_wr(dev, MT_INT_SOURCE_CSR, intr);
|
||||
|
||||
if (!test_bit(MT76_STATE_INITIALIZED, &dev->mt76.state))
|
||||
if (!test_bit(MT76_STATE_INITIALIZED, &dev->mphy.state))
|
||||
return IRQ_NONE;
|
||||
|
||||
trace_dev_irq(&dev->mt76, intr, dev->mt76.mmio.irqmask);
|
||||
|
||||
intr &= dev->mt76.mmio.irqmask;
|
||||
|
||||
if (intr & MT_INT_MAC_IRQ3) {
|
||||
|
@ -30,6 +30,16 @@ mt7603_init_tx_queue(struct mt7603_dev *dev, struct mt76_sw_queue *q,
|
||||
static void
|
||||
mt7603_rx_loopback_skb(struct mt7603_dev *dev, struct sk_buff *skb)
|
||||
{
|
||||
static const u8 tid_to_ac[8] = {
|
||||
IEEE80211_AC_BE,
|
||||
IEEE80211_AC_BK,
|
||||
IEEE80211_AC_BK,
|
||||
IEEE80211_AC_BE,
|
||||
IEEE80211_AC_VI,
|
||||
IEEE80211_AC_VI,
|
||||
IEEE80211_AC_VO,
|
||||
IEEE80211_AC_VO
|
||||
};
|
||||
__le32 *txd = (__le32 *)skb->data;
|
||||
struct ieee80211_hdr *hdr;
|
||||
struct ieee80211_sta *sta;
|
||||
@ -38,7 +48,7 @@ mt7603_rx_loopback_skb(struct mt7603_dev *dev, struct sk_buff *skb)
|
||||
void *priv;
|
||||
int idx;
|
||||
u32 val;
|
||||
u8 tid;
|
||||
u8 tid = 0;
|
||||
|
||||
if (skb->len < MT_TXD_SIZE + sizeof(struct ieee80211_hdr))
|
||||
goto free;
|
||||
@ -56,15 +66,16 @@ mt7603_rx_loopback_skb(struct mt7603_dev *dev, struct sk_buff *skb)
|
||||
|
||||
priv = msta = container_of(wcid, struct mt7603_sta, wcid);
|
||||
val = le32_to_cpu(txd[0]);
|
||||
skb_set_queue_mapping(skb, FIELD_GET(MT_TXD0_Q_IDX, val));
|
||||
|
||||
val &= ~(MT_TXD0_P_IDX | MT_TXD0_Q_IDX);
|
||||
val |= FIELD_PREP(MT_TXD0_Q_IDX, MT_TX_HW_QUEUE_MGMT);
|
||||
txd[0] = cpu_to_le32(val);
|
||||
|
||||
sta = container_of(priv, struct ieee80211_sta, drv_priv);
|
||||
hdr = (struct ieee80211_hdr *)&skb->data[MT_TXD_SIZE];
|
||||
tid = *ieee80211_get_qos_ctl(hdr) & IEEE80211_QOS_CTL_TID_MASK;
|
||||
if (ieee80211_is_data_qos(hdr->frame_control))
|
||||
tid = *ieee80211_get_qos_ctl(hdr) &
|
||||
IEEE80211_QOS_CTL_TAG1D_MASK;
|
||||
skb_set_queue_mapping(skb, tid_to_ac[tid]);
|
||||
ieee80211_sta_set_buffered(sta, tid, true);
|
||||
|
||||
spin_lock_bh(&dev->ps_lock);
|
||||
@ -210,7 +221,7 @@ int mt7603_dma_init(struct mt7603_dev *dev)
|
||||
return ret;
|
||||
|
||||
ret = mt7603_init_rx_queue(dev, &dev->mt76.q_rx[MT_RXQ_MCU], 1,
|
||||
MT_MCU_RING_SIZE, MT_RX_BUF_SIZE);
|
||||
MT7603_MCU_RX_RING_SIZE, MT_RX_BUF_SIZE);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
|
@ -113,7 +113,7 @@ mt7603_dma_sched_init(struct mt7603_dev *dev)
|
||||
static void
|
||||
mt7603_phy_init(struct mt7603_dev *dev)
|
||||
{
|
||||
int rx_chains = dev->mt76.antenna_mask;
|
||||
int rx_chains = dev->mphy.antenna_mask;
|
||||
int tx_chains = hweight8(rx_chains) - 1;
|
||||
|
||||
mt76_rmw(dev, MT_WF_RMAC_RMCR,
|
||||
@ -284,7 +284,7 @@ mt7603_init_hardware(struct mt7603_dev *dev)
|
||||
mt76_wr(dev, MT_WPDMA_GLO_CFG, 0x52000850);
|
||||
mt7603_mac_dma_start(dev);
|
||||
dev->rxfilter = mt76_rr(dev, MT_WF_RFCR);
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);
|
||||
|
||||
for (i = 0; i < MT7603_WTBL_SIZE; i++) {
|
||||
mt76_wr(dev, MT_PSE_RTA, MT_PSE_RTA_BUSY | MT_PSE_RTA_WRITE |
|
||||
@ -363,9 +363,9 @@ static void mt7603_led_set_config(struct mt76_dev *mt76, u8 delay_on,
|
||||
mt76);
|
||||
u32 val, addr;
|
||||
|
||||
val = MT_LED_STATUS_DURATION(0xffff) |
|
||||
MT_LED_STATUS_OFF(delay_off) |
|
||||
MT_LED_STATUS_ON(delay_on);
|
||||
val = FIELD_PREP(MT_LED_STATUS_DURATION, 0xffff) |
|
||||
FIELD_PREP(MT_LED_STATUS_OFF, delay_off) |
|
||||
FIELD_PREP(MT_LED_STATUS_ON, delay_on);
|
||||
|
||||
addr = mt7603_reg_map(dev, MT_LED_STATUS_0(mt76->led_pin));
|
||||
mt76_wr(dev, addr, val);
|
||||
@ -493,12 +493,12 @@ mt7603_init_txpower(struct mt7603_dev *dev,
|
||||
target_power += max_offset;
|
||||
|
||||
dev->tx_power_limit = target_power;
|
||||
dev->mt76.txpower_cur = target_power;
|
||||
dev->mphy.txpower_cur = target_power;
|
||||
|
||||
target_power = DIV_ROUND_UP(target_power, 2);
|
||||
|
||||
/* add 3 dBm for 2SS devices (combined output) */
|
||||
if (dev->mt76.antenna_mask & BIT(1))
|
||||
if (dev->mphy.antenna_mask & BIT(1))
|
||||
target_power += 3;
|
||||
|
||||
for (i = 0; i < sband->n_channels; i++) {
|
||||
@ -535,9 +535,9 @@ int mt7603_register_device(struct mt7603_dev *dev)
|
||||
(unsigned long)dev);
|
||||
|
||||
/* Check for 7688, which only has 1SS */
|
||||
dev->mt76.antenna_mask = 3;
|
||||
dev->mphy.antenna_mask = 3;
|
||||
if (mt76_rr(dev, MT_EFUSE_BASE + 0x64) & BIT(4))
|
||||
dev->mt76.antenna_mask = 1;
|
||||
dev->mphy.antenna_mask = 1;
|
||||
|
||||
dev->slottime = 9;
|
||||
|
||||
@ -557,6 +557,7 @@ int mt7603_register_device(struct mt7603_dev *dev)
|
||||
wiphy->n_iface_combinations = ARRAY_SIZE(if_comb);
|
||||
|
||||
ieee80211_hw_set(hw, TX_STATUS_NO_AMPDU_LEN);
|
||||
ieee80211_hw_set(hw, HOST_BROADCAST_PS_BUFFERING);
|
||||
|
||||
/* init led callbacks */
|
||||
if (IS_ENABLED(CONFIG_MT76_LEDS)) {
|
||||
@ -564,7 +565,6 @@ int mt7603_register_device(struct mt7603_dev *dev)
|
||||
dev->mt76.led_cdev.blink_set = mt7603_led_set_blink;
|
||||
}
|
||||
|
||||
wiphy->flags |= WIPHY_FLAG_HAS_CHANNEL_SWITCH;
|
||||
wiphy->reg_notifier = mt7603_regd_notifier;
|
||||
|
||||
ret = mt76_register_device(&dev->mt76, true, mt7603_rates,
|
||||
@ -573,7 +573,7 @@ int mt7603_register_device(struct mt7603_dev *dev)
|
||||
return ret;
|
||||
|
||||
mt7603_init_debugfs(dev);
|
||||
mt7603_init_txpower(dev, &dev->mt76.sband_2g.sband);
|
||||
mt7603_init_txpower(dev, &dev->mphy.sband_2g.sband);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -4,6 +4,7 @@
|
||||
#include <linux/timekeeping.h>
|
||||
#include "mt7603.h"
|
||||
#include "mac.h"
|
||||
#include "../trace.h"
|
||||
|
||||
#define MT_PSE_PAGE_SIZE 128
|
||||
|
||||
@ -53,7 +54,7 @@ void mt7603_mac_set_timing(struct mt7603_dev *dev)
|
||||
int sifs;
|
||||
u32 val;
|
||||
|
||||
if (dev->mt76.chandef.chan->band == NL80211_BAND_5GHZ)
|
||||
if (dev->mphy.chandef.chan->band == NL80211_BAND_5GHZ)
|
||||
sifs = 16;
|
||||
else
|
||||
sifs = 10;
|
||||
@ -456,7 +457,7 @@ void mt7603_mac_sta_poll(struct mt7603_dev *dev)
|
||||
return;
|
||||
|
||||
spin_lock_bh(&dev->mt76.cc_lock);
|
||||
dev->mt76.chan_state->cc_tx += total_airtime;
|
||||
dev->mphy.chan_state->cc_tx += total_airtime;
|
||||
spin_unlock_bh(&dev->mt76.cc_lock);
|
||||
}
|
||||
|
||||
@ -502,7 +503,7 @@ mt7603_mac_fill_rx(struct mt7603_dev *dev, struct sk_buff *skb)
|
||||
memset(status, 0, sizeof(*status));
|
||||
|
||||
i = FIELD_GET(MT_RXD1_NORMAL_CH_FREQ, rxd1);
|
||||
sband = (i & 1) ? &dev->mt76.sband_5g.sband : &dev->mt76.sband_2g.sband;
|
||||
sband = (i & 1) ? &dev->mphy.sband_5g.sband : &dev->mphy.sband_2g.sband;
|
||||
i >>= 1;
|
||||
|
||||
idx = FIELD_GET(MT_RXD2_NORMAL_WLAN_IDX, rxd2);
|
||||
@ -531,12 +532,12 @@ mt7603_mac_fill_rx(struct mt7603_dev *dev, struct sk_buff *skb)
|
||||
|
||||
/* all subframes of an A-MPDU have the same timestamp */
|
||||
if (dev->rx_ampdu_ts != rxd[12]) {
|
||||
if (!++dev->mt76.ampdu_ref)
|
||||
dev->mt76.ampdu_ref++;
|
||||
if (!++dev->ampdu_ref)
|
||||
dev->ampdu_ref++;
|
||||
}
|
||||
dev->rx_ampdu_ts = rxd[12];
|
||||
|
||||
status->ampdu_ref = dev->mt76.ampdu_ref;
|
||||
status->ampdu_ref = dev->ampdu_ref;
|
||||
}
|
||||
|
||||
remove_pad = rxd1 & MT_RXD1_NORMAL_HDR_OFFSET;
|
||||
@ -609,7 +610,7 @@ mt7603_mac_fill_rx(struct mt7603_dev *dev, struct sk_buff *skb)
|
||||
|
||||
status->rate_idx = i;
|
||||
|
||||
status->chains = dev->mt76.antenna_mask;
|
||||
status->chains = dev->mphy.antenna_mask;
|
||||
status->chain_signal[0] = FIELD_GET(MT_RXV4_IB_RSSI0, rxdg3) +
|
||||
dev->rssi_offset[0];
|
||||
status->chain_signal[1] = FIELD_GET(MT_RXV4_IB_RSSI1, rxdg3) +
|
||||
@ -668,7 +669,7 @@ mt7603_mac_tx_rate_val(struct mt7603_dev *dev,
|
||||
*bw = 1;
|
||||
} else {
|
||||
const struct ieee80211_rate *r;
|
||||
int band = dev->mt76.chandef.chan->band;
|
||||
int band = dev->mphy.chandef.chan->band;
|
||||
u16 val;
|
||||
|
||||
nss = 1;
|
||||
@ -1156,10 +1157,10 @@ out:
|
||||
cck = true;
|
||||
/* fall through */
|
||||
case MT_PHY_TYPE_OFDM:
|
||||
if (dev->mt76.chandef.chan->band == NL80211_BAND_5GHZ)
|
||||
sband = &dev->mt76.sband_5g.sband;
|
||||
if (dev->mphy.chandef.chan->band == NL80211_BAND_5GHZ)
|
||||
sband = &dev->mphy.sband_5g.sband;
|
||||
else
|
||||
sband = &dev->mt76.sband_2g.sband;
|
||||
sband = &dev->mphy.sband_2g.sband;
|
||||
final_rate &= GENMASK(5, 0);
|
||||
final_rate = mt76_get_rate(&dev->mt76, sband, final_rate,
|
||||
cck);
|
||||
@ -1193,6 +1194,8 @@ mt7603_mac_add_txs_skb(struct mt7603_dev *dev, struct mt7603_sta *sta, int pid,
|
||||
if (pid < MT_PACKET_ID_FIRST)
|
||||
return false;
|
||||
|
||||
trace_mac_txdone(mdev, sta->wcid.idx, pid);
|
||||
|
||||
mt76_tx_status_lock(mdev, &list);
|
||||
skb = mt76_tx_status_skb_get(mdev, &sta->wcid, pid, &list);
|
||||
if (skb) {
|
||||
@ -1389,10 +1392,10 @@ static void mt7603_mac_watchdog_reset(struct mt7603_dev *dev)
|
||||
int i;
|
||||
|
||||
ieee80211_stop_queues(dev->mt76.hw);
|
||||
set_bit(MT76_RESET, &dev->mt76.state);
|
||||
set_bit(MT76_RESET, &dev->mphy.state);
|
||||
|
||||
/* lock/unlock all queues to ensure that no tx is pending */
|
||||
mt76_txq_schedule_all(&dev->mt76);
|
||||
mt76_txq_schedule_all(&dev->mphy);
|
||||
|
||||
tasklet_disable(&dev->mt76.tx_tasklet);
|
||||
tasklet_disable(&dev->mt76.pre_tbtt_tasklet);
|
||||
@ -1426,7 +1429,7 @@ static void mt7603_mac_watchdog_reset(struct mt7603_dev *dev)
|
||||
|
||||
mt7603_pse_client_reset(dev);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->mt76.q_tx); i++)
|
||||
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++)
|
||||
@ -1439,7 +1442,7 @@ static void mt7603_mac_watchdog_reset(struct mt7603_dev *dev)
|
||||
mt7603_irq_enable(dev, mask);
|
||||
|
||||
skip_dma_reset:
|
||||
clear_bit(MT76_RESET, &dev->mt76.state);
|
||||
clear_bit(MT76_RESET, &dev->mphy.state);
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
tasklet_enable(&dev->mt76.tx_tasklet);
|
||||
@ -1456,7 +1459,7 @@ skip_dma_reset:
|
||||
napi_schedule(&dev->mt76.napi[1]);
|
||||
|
||||
ieee80211_wake_queues(dev->mt76.hw);
|
||||
mt76_txq_schedule_all(&dev->mt76);
|
||||
mt76_txq_schedule_all(&dev->mphy);
|
||||
}
|
||||
|
||||
static u32 mt7603_dma_debug(struct mt7603_dev *dev, u8 index)
|
||||
@ -1574,7 +1577,7 @@ void mt7603_update_channel(struct mt76_dev *mdev)
|
||||
struct mt7603_dev *dev = container_of(mdev, struct mt7603_dev, mt76);
|
||||
struct mt76_channel_state *state;
|
||||
|
||||
state = mdev->chan_state;
|
||||
state = mdev->phy.chan_state;
|
||||
state->cc_busy += mt76_rr(dev, MT_MIB_STAT_CCA);
|
||||
}
|
||||
|
||||
@ -1737,7 +1740,7 @@ mt7603_false_cca_check(struct mt7603_dev *dev)
|
||||
|
||||
mt7603_cca_stats_reset(dev);
|
||||
|
||||
min_signal = mt76_get_min_avg_rssi(&dev->mt76);
|
||||
min_signal = mt76_get_min_avg_rssi(&dev->mt76, false);
|
||||
if (!min_signal) {
|
||||
dev->sensitivity = 0;
|
||||
dev->last_cca_adj = jiffies;
|
||||
|
@ -15,8 +15,8 @@ mt7603_start(struct ieee80211_hw *hw)
|
||||
|
||||
mt7603_mac_reset_counters(dev);
|
||||
mt7603_mac_start(dev);
|
||||
dev->mt76.survey_time = ktime_get_boottime();
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
dev->mphy.survey_time = ktime_get_boottime();
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
mt7603_mac_work(&dev->mt76.mac_work.work);
|
||||
|
||||
return 0;
|
||||
@ -27,7 +27,7 @@ mt7603_stop(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct mt7603_dev *dev = hw->priv;
|
||||
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
cancel_delayed_work_sync(&dev->mt76.mac_work);
|
||||
mt7603_mac_stop(dev);
|
||||
}
|
||||
@ -143,16 +143,16 @@ mt7603_set_channel(struct mt7603_dev *dev, struct cfg80211_chan_def *def)
|
||||
tasklet_disable(&dev->mt76.pre_tbtt_tasklet);
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
set_bit(MT76_RESET, &dev->mt76.state);
|
||||
set_bit(MT76_RESET, &dev->mphy.state);
|
||||
|
||||
mt7603_beacon_set_timer(dev, -1, 0);
|
||||
mt76_set_channel(&dev->mt76);
|
||||
mt76_set_channel(&dev->mphy);
|
||||
mt7603_mac_stop(dev);
|
||||
|
||||
if (def->width == NL80211_CHAN_WIDTH_40)
|
||||
bw = MT_BW_40;
|
||||
|
||||
dev->mt76.chandef = *def;
|
||||
dev->mphy.chandef = *def;
|
||||
mt76_rmw_field(dev, MT_AGG_BWCR, MT_AGG_BWCR_BW, bw);
|
||||
ret = mt7603_mcu_set_channel(dev);
|
||||
if (ret) {
|
||||
@ -176,9 +176,9 @@ mt7603_set_channel(struct mt7603_dev *dev, struct cfg80211_chan_def *def)
|
||||
mt7603_mac_set_timing(dev);
|
||||
mt7603_mac_start(dev);
|
||||
|
||||
clear_bit(MT76_RESET, &dev->mt76.state);
|
||||
clear_bit(MT76_RESET, &dev->mphy.state);
|
||||
|
||||
mt76_txq_schedule_all(&dev->mt76);
|
||||
mt76_txq_schedule_all(&dev->mphy);
|
||||
|
||||
ieee80211_queue_delayed_work(mt76_hw(dev), &dev->mt76.mac_work,
|
||||
msecs_to_jiffies(MT7603_WATCHDOG_TIME));
|
||||
@ -187,10 +187,10 @@ mt7603_set_channel(struct mt7603_dev *dev, struct cfg80211_chan_def *def)
|
||||
mt76_clear(dev, MT_MIB_CTL, MT_MIB_CTL_READ_CLR_DIS);
|
||||
mt76_set(dev, MT_MIB_CTL,
|
||||
MT_MIB_CTL_CCA_NAV_TX | MT_MIB_CTL_PSCCA_TIME);
|
||||
mt76_rr(dev, MT_MIB_STAT_PSCCA);
|
||||
mt76_rr(dev, MT_MIB_STAT_CCA);
|
||||
mt7603_cca_stats_reset(dev);
|
||||
|
||||
dev->mt76.survey_time = ktime_get_boottime();
|
||||
dev->mphy.survey_time = ktime_get_boottime();
|
||||
|
||||
mt7603_init_edcca(dev);
|
||||
|
||||
@ -642,7 +642,7 @@ mt7603_set_coverage_class(struct ieee80211_hw *hw, s16 coverage_class)
|
||||
{
|
||||
struct mt7603_dev *dev = hw->priv;
|
||||
|
||||
dev->coverage_class = coverage_class;
|
||||
dev->coverage_class = max_t(s16, coverage_class, 0);
|
||||
mt7603_mac_set_timing(dev);
|
||||
}
|
||||
|
||||
@ -667,7 +667,7 @@ static void mt7603_tx(struct ieee80211_hw *hw,
|
||||
wcid = &mvif->sta.wcid;
|
||||
}
|
||||
|
||||
mt76_tx(&dev->mt76, control->sta, wcid, skb);
|
||||
mt76_tx(&dev->mphy, control->sta, wcid, skb);
|
||||
}
|
||||
|
||||
const struct ieee80211_ops mt7603_ops = {
|
||||
@ -680,6 +680,7 @@ const struct ieee80211_ops mt7603_ops = {
|
||||
.configure_filter = mt7603_configure_filter,
|
||||
.bss_info_changed = mt7603_bss_info_changed,
|
||||
.sta_state = mt76_sta_state,
|
||||
.sta_pre_rcu_remove = mt76_sta_pre_rcu_remove,
|
||||
.set_key = mt7603_set_key,
|
||||
.conf_tx = mt7603_conf_tx,
|
||||
.sw_scan_start = mt76_sw_scan,
|
||||
|
@ -22,9 +22,9 @@ __mt7603_mcu_msg_send(struct mt7603_dev *dev, struct sk_buff *skb,
|
||||
struct mt7603_mcu_txd *txd;
|
||||
u8 seq;
|
||||
|
||||
seq = ++mdev->mmio.mcu.msg_seq & 0xf;
|
||||
seq = ++mdev->mcu.msg_seq & 0xf;
|
||||
if (!seq)
|
||||
seq = ++mdev->mmio.mcu.msg_seq & 0xf;
|
||||
seq = ++mdev->mcu.msg_seq & 0xf;
|
||||
|
||||
txd = (struct mt7603_mcu_txd *)skb_push(skb, hdrlen);
|
||||
memset(txd, 0, hdrlen);
|
||||
@ -67,7 +67,7 @@ mt7603_mcu_msg_send(struct mt76_dev *mdev, int cmd, const void *data,
|
||||
if (!skb)
|
||||
return -ENOMEM;
|
||||
|
||||
mutex_lock(&mdev->mmio.mcu.mutex);
|
||||
mutex_lock(&mdev->mcu.mutex);
|
||||
|
||||
ret = __mt7603_mcu_msg_send(dev, skb, cmd, &seq);
|
||||
if (ret)
|
||||
@ -97,7 +97,7 @@ mt7603_mcu_msg_send(struct mt76_dev *mdev, int cmd, const void *data,
|
||||
}
|
||||
|
||||
out:
|
||||
mutex_unlock(&mdev->mmio.mcu.mutex);
|
||||
mutex_unlock(&mdev->mcu.mutex);
|
||||
|
||||
return ret;
|
||||
}
|
||||
@ -277,7 +277,7 @@ int mt7603_mcu_init(struct mt7603_dev *dev)
|
||||
void mt7603_mcu_exit(struct mt7603_dev *dev)
|
||||
{
|
||||
__mt76_mcu_restart(&dev->mt76);
|
||||
skb_queue_purge(&dev->mt76.mmio.mcu.res_q);
|
||||
skb_queue_purge(&dev->mt76.mcu.res_q);
|
||||
}
|
||||
|
||||
int mt7603_mcu_set_eeprom(struct mt7603_dev *dev)
|
||||
@ -397,7 +397,7 @@ static int mt7603_mcu_set_tx_power(struct mt7603_dev *dev)
|
||||
u8 temp_comp_power[17];
|
||||
u8 reserved;
|
||||
} req = {
|
||||
.center_channel = dev->mt76.chandef.chan->hw_value,
|
||||
.center_channel = dev->mphy.chandef.chan->hw_value,
|
||||
#define EEP_VAL(n) ((u8 *)dev->mt76.eeprom.data)[n]
|
||||
.tssi = EEP_VAL(MT_EE_NIC_CONF_1 + 1),
|
||||
.temp_comp = EEP_VAL(MT_EE_NIC_CONF_1),
|
||||
@ -430,9 +430,9 @@ static int mt7603_mcu_set_tx_power(struct mt7603_dev *dev)
|
||||
|
||||
int mt7603_mcu_set_channel(struct mt7603_dev *dev)
|
||||
{
|
||||
struct cfg80211_chan_def *chandef = &dev->mt76.chandef;
|
||||
struct cfg80211_chan_def *chandef = &dev->mphy.chandef;
|
||||
struct ieee80211_hw *hw = mt76_hw(dev);
|
||||
int n_chains = hweight8(dev->mt76.antenna_mask);
|
||||
int n_chains = hweight8(dev->mphy.antenna_mask);
|
||||
struct {
|
||||
u8 control_chan;
|
||||
u8 center_chan;
|
||||
@ -452,7 +452,7 @@ int mt7603_mcu_set_channel(struct mt7603_dev *dev)
|
||||
s8 tx_power;
|
||||
int i, ret;
|
||||
|
||||
if (dev->mt76.chandef.width == NL80211_CHAN_WIDTH_40) {
|
||||
if (dev->mphy.chandef.width == NL80211_CHAN_WIDTH_40) {
|
||||
req.bw = MT_BW_40;
|
||||
if (chandef->center_freq1 > chandef->chan->center_freq)
|
||||
req.center_chan += 2;
|
||||
@ -461,11 +461,11 @@ int mt7603_mcu_set_channel(struct mt7603_dev *dev)
|
||||
}
|
||||
|
||||
tx_power = hw->conf.power_level * 2;
|
||||
if (dev->mt76.antenna_mask == 3)
|
||||
if (dev->mphy.antenna_mask == 3)
|
||||
tx_power -= 6;
|
||||
tx_power = min(tx_power, dev->tx_power_limit);
|
||||
|
||||
dev->mt76.txpower_cur = tx_power;
|
||||
dev->mphy.txpower_cur = tx_power;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(req.txpower); i++)
|
||||
req.txpower[i] = tx_power;
|
||||
|
@ -15,6 +15,7 @@
|
||||
|
||||
#define MT7603_RATE_RETRY 2
|
||||
|
||||
#define MT7603_MCU_RX_RING_SIZE 64
|
||||
#define MT7603_RX_RING_SIZE 128
|
||||
|
||||
#define MT7603_FIRMWARE_E1 "mt7603_e1.bin"
|
||||
@ -98,7 +99,10 @@ enum mt7603_reset_cause {
|
||||
};
|
||||
|
||||
struct mt7603_dev {
|
||||
struct mt76_dev mt76; /* must be first */
|
||||
union { /* must be first */
|
||||
struct mt76_dev mt76;
|
||||
struct mt76_phy mphy;
|
||||
};
|
||||
|
||||
const struct mt76_bus_ops *bus_ops;
|
||||
|
||||
@ -115,6 +119,7 @@ struct mt7603_dev {
|
||||
u32 false_cca_ofdm, false_cca_cck;
|
||||
unsigned long last_cca_adj;
|
||||
|
||||
u32 ampdu_ref;
|
||||
__le32 rx_ampdu_ts;
|
||||
u8 rssi_offset[3];
|
||||
|
||||
|
@ -585,18 +585,9 @@ enum {
|
||||
|
||||
#define MT_LED_STATUS_0(_n) MT_LED_PHYS(0x10 + ((_n) * 8))
|
||||
#define MT_LED_STATUS_1(_n) MT_LED_PHYS(0x14 + ((_n) * 8))
|
||||
#define MT_LED_STATUS_OFF_MASK GENMASK(31, 24)
|
||||
#define MT_LED_STATUS_OFF(_v) (((_v) << \
|
||||
__ffs(MT_LED_STATUS_OFF_MASK)) & \
|
||||
MT_LED_STATUS_OFF_MASK)
|
||||
#define MT_LED_STATUS_ON_MASK GENMASK(23, 16)
|
||||
#define MT_LED_STATUS_ON(_v) (((_v) << \
|
||||
__ffs(MT_LED_STATUS_ON_MASK)) & \
|
||||
MT_LED_STATUS_ON_MASK)
|
||||
#define MT_LED_STATUS_DURATION_MASK GENMASK(15, 0)
|
||||
#define MT_LED_STATUS_DURATION(_v) (((_v) << \
|
||||
__ffs(MT_LED_STATUS_DURATION_MASK)) &\
|
||||
MT_LED_STATUS_DURATION_MASK)
|
||||
#define MT_LED_STATUS_OFF GENMASK(31, 24)
|
||||
#define MT_LED_STATUS_ON GENMASK(23, 16)
|
||||
#define MT_LED_STATUS_DURATION GENMASK(15, 0)
|
||||
|
||||
#define MT_CLIENT_BASE_PHYS_ADDR 0x800c0000
|
||||
|
||||
|
@ -11,3 +11,14 @@ config MT7615E
|
||||
MU-MIMO up to 4 users/group and 160MHz channels.
|
||||
|
||||
To compile this driver as a module, choose M here.
|
||||
|
||||
config MT7622_WMAC
|
||||
bool "MT7622 (SoC) WMAC support"
|
||||
depends on MT7615E
|
||||
depends on ARCH_MEDIATEK || COMPILE_TEST
|
||||
select REGMAP
|
||||
default y
|
||||
help
|
||||
This adds support for the built-in WMAC on MT7622 SoC devices
|
||||
which has the same feature set as a MT7615, but limited to
|
||||
2.4 GHz only.
|
||||
|
@ -2,5 +2,8 @@
|
||||
|
||||
obj-$(CONFIG_MT7615E) += mt7615e.o
|
||||
|
||||
mt7615e-y := pci.o init.o dma.o eeprom.o main.o mcu.o mac.o \
|
||||
debugfs.o
|
||||
CFLAGS_trace.o := -I$(src)
|
||||
|
||||
mt7615e-y := pci.o init.o dma.o eeprom.o main.o mcu.o mac.o mmio.o \
|
||||
debugfs.o trace.o
|
||||
mt7615e-$(CONFIG_MT7622_WMAC) += soc.o
|
||||
|
@ -7,6 +7,9 @@ mt7615_radar_pattern_set(void *data, u64 val)
|
||||
{
|
||||
struct mt7615_dev *dev = data;
|
||||
|
||||
if (!mt7615_wait_for_mcu_init(dev))
|
||||
return 0;
|
||||
|
||||
return mt7615_mcu_rdd_send_pattern(dev);
|
||||
}
|
||||
|
||||
@ -18,6 +21,9 @@ mt7615_scs_set(void *data, u64 val)
|
||||
{
|
||||
struct mt7615_dev *dev = data;
|
||||
|
||||
if (!mt7615_wait_for_mcu_init(dev))
|
||||
return 0;
|
||||
|
||||
mt7615_mac_set_scs(dev, val);
|
||||
|
||||
return 0;
|
||||
@ -36,6 +42,84 @@ mt7615_scs_get(void *data, u64 *val)
|
||||
DEFINE_DEBUGFS_ATTRIBUTE(fops_scs, mt7615_scs_get,
|
||||
mt7615_scs_set, "%lld\n");
|
||||
|
||||
static int
|
||||
mt7615_dbdc_set(void *data, u64 val)
|
||||
{
|
||||
struct mt7615_dev *dev = data;
|
||||
|
||||
if (!mt7615_wait_for_mcu_init(dev))
|
||||
return 0;
|
||||
|
||||
if (val)
|
||||
mt7615_register_ext_phy(dev);
|
||||
else
|
||||
mt7615_unregister_ext_phy(dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
mt7615_dbdc_get(void *data, u64 *val)
|
||||
{
|
||||
struct mt7615_dev *dev = data;
|
||||
|
||||
*val = !!mt7615_ext_phy(dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
DEFINE_DEBUGFS_ATTRIBUTE(fops_dbdc, mt7615_dbdc_get,
|
||||
mt7615_dbdc_set, "%lld\n");
|
||||
|
||||
static int
|
||||
mt7615_fw_debug_set(void *data, u64 val)
|
||||
{
|
||||
struct mt7615_dev *dev = data;
|
||||
|
||||
if (!mt7615_wait_for_mcu_init(dev))
|
||||
return 0;
|
||||
|
||||
dev->fw_debug = val;
|
||||
mt7615_mcu_fw_log_2_host(dev, dev->fw_debug ? 2 : 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
mt7615_fw_debug_get(void *data, u64 *val)
|
||||
{
|
||||
struct mt7615_dev *dev = data;
|
||||
|
||||
*val = dev->fw_debug;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
DEFINE_DEBUGFS_ATTRIBUTE(fops_fw_debug, mt7615_fw_debug_get,
|
||||
mt7615_fw_debug_set, "%lld\n");
|
||||
|
||||
static int
|
||||
mt7615_reset_test_set(void *data, u64 val)
|
||||
{
|
||||
struct mt7615_dev *dev = data;
|
||||
struct sk_buff *skb;
|
||||
|
||||
if (!mt7615_wait_for_mcu_init(dev))
|
||||
return 0;
|
||||
|
||||
skb = alloc_skb(1, GFP_KERNEL);
|
||||
if (!skb)
|
||||
return -ENOMEM;
|
||||
|
||||
skb_put(skb, 1);
|
||||
mt76_tx_queue_skb_raw(dev, 0, skb, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
DEFINE_DEBUGFS_ATTRIBUTE(fops_reset_test, NULL,
|
||||
mt7615_reset_test_set, "%lld\n");
|
||||
|
||||
static int
|
||||
mt7615_ampdu_stat_read(struct seq_file *file, void *data)
|
||||
{
|
||||
@ -74,15 +158,28 @@ static const struct file_operations fops_ampdu_stat = {
|
||||
.release = single_release,
|
||||
};
|
||||
|
||||
static void
|
||||
mt7615_radio_read_phy(struct mt7615_phy *phy, struct seq_file *s)
|
||||
{
|
||||
struct mt7615_dev *dev = dev_get_drvdata(s->private);
|
||||
bool ext_phy = phy != &dev->phy;
|
||||
|
||||
if (!phy)
|
||||
return;
|
||||
|
||||
seq_printf(s, "Radio %d sensitivity: ofdm=%d cck=%d\n", ext_phy,
|
||||
phy->ofdm_sensitivity, phy->cck_sensitivity);
|
||||
seq_printf(s, "Radio %d false CCA: ofdm=%d cck=%d\n", ext_phy,
|
||||
phy->false_cca_ofdm, phy->false_cca_cck);
|
||||
}
|
||||
|
||||
static int
|
||||
mt7615_radio_read(struct seq_file *s, void *data)
|
||||
{
|
||||
struct mt7615_dev *dev = dev_get_drvdata(s->private);
|
||||
|
||||
seq_printf(s, "Sensitivity: ofdm=%d cck=%d\n",
|
||||
dev->ofdm_sensitivity, dev->cck_sensitivity);
|
||||
seq_printf(s, "False CCA: ofdm=%d cck=%d\n",
|
||||
dev->false_cca_ofdm, dev->false_cca_cck);
|
||||
mt7615_radio_read_phy(&dev->phy, s);
|
||||
mt7615_radio_read_phy(mt7615_ext_phy(dev), s);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -92,6 +189,9 @@ static int mt7615_read_temperature(struct seq_file *s, void *data)
|
||||
struct mt7615_dev *dev = dev_get_drvdata(s->private);
|
||||
int temp;
|
||||
|
||||
if (!mt7615_wait_for_mcu_init(dev))
|
||||
return 0;
|
||||
|
||||
/* cpu */
|
||||
temp = mt7615_mcu_get_temperature(dev, 0);
|
||||
seq_printf(s, "Temperature: %d\n", temp);
|
||||
@ -164,12 +264,18 @@ int mt7615_init_debugfs(struct mt7615_dev *dev)
|
||||
if (!dir)
|
||||
return -ENOMEM;
|
||||
|
||||
debugfs_create_devm_seqfile(dev->mt76.dev, "queues", dir,
|
||||
mt7615_queues_read);
|
||||
if (is_mt7615(&dev->mt76))
|
||||
debugfs_create_devm_seqfile(dev->mt76.dev, "queues", dir,
|
||||
mt7615_queues_read);
|
||||
else
|
||||
debugfs_create_devm_seqfile(dev->mt76.dev, "queues", dir,
|
||||
mt76_queues_read);
|
||||
debugfs_create_devm_seqfile(dev->mt76.dev, "acq", dir,
|
||||
mt7615_queues_acq);
|
||||
debugfs_create_file("ampdu_stat", 0400, dir, dev, &fops_ampdu_stat);
|
||||
debugfs_create_file("scs", 0600, dir, dev, &fops_scs);
|
||||
debugfs_create_file("dbdc", 0600, dir, dev, &fops_dbdc);
|
||||
debugfs_create_file("fw_debug", 0600, dir, dev, &fops_fw_debug);
|
||||
debugfs_create_devm_seqfile(dev->mt76.dev, "radio", dir,
|
||||
mt7615_radio_read);
|
||||
debugfs_create_u32("dfs_hw_pattern", 0400, dir, &dev->hw_pattern);
|
||||
@ -184,6 +290,8 @@ int mt7615_init_debugfs(struct mt7615_dev *dev)
|
||||
&dev->radar_pattern.power);
|
||||
debugfs_create_file("radar_trigger", 0200, dir, dev,
|
||||
&fops_radar_pattern);
|
||||
debugfs_create_file("reset_test", 0200, dir, dev,
|
||||
&fops_reset_test);
|
||||
debugfs_create_devm_seqfile(dev->mt76.dev, "temperature", dir,
|
||||
mt7615_read_temperature);
|
||||
|
||||
|
@ -12,31 +12,7 @@
|
||||
#include "mac.h"
|
||||
|
||||
static int
|
||||
mt7615_init_tx_queues(struct mt7615_dev *dev, int n_desc)
|
||||
{
|
||||
struct mt76_sw_queue *q;
|
||||
struct mt76_queue *hwq;
|
||||
int err, i;
|
||||
|
||||
hwq = devm_kzalloc(dev->mt76.dev, sizeof(*hwq), GFP_KERNEL);
|
||||
if (!hwq)
|
||||
return -ENOMEM;
|
||||
|
||||
err = mt76_queue_alloc(dev, hwq, 0, n_desc, 0, MT_TX_RING_BASE);
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
for (i = 0; i < MT_TXQ_MCU; i++) {
|
||||
q = &dev->mt76.q_tx[i];
|
||||
INIT_LIST_HEAD(&q->swq);
|
||||
q->q = hwq;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
mt7615_init_mcu_queue(struct mt7615_dev *dev, struct mt76_sw_queue *q,
|
||||
mt7615_init_tx_queue(struct mt7615_dev *dev, struct mt76_sw_queue *q,
|
||||
int idx, int n_desc)
|
||||
{
|
||||
struct mt76_queue *hwq;
|
||||
@ -56,6 +32,68 @@ mt7615_init_mcu_queue(struct mt7615_dev *dev, struct mt76_sw_queue *q,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
mt7622_init_tx_queues_multi(struct mt7615_dev *dev)
|
||||
{
|
||||
static const u8 wmm_queue_map[] = {
|
||||
MT7622_TXQ_AC0,
|
||||
MT7622_TXQ_AC1,
|
||||
MT7622_TXQ_AC2,
|
||||
MT7622_TXQ_AC3,
|
||||
};
|
||||
int ret;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(wmm_queue_map); i++) {
|
||||
ret = mt7615_init_tx_queue(dev, &dev->mt76.q_tx[i],
|
||||
wmm_queue_map[i],
|
||||
MT7615_TX_RING_SIZE / 2);
|
||||
if (ret)
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = mt7615_init_tx_queue(dev, &dev->mt76.q_tx[MT_TXQ_PSD],
|
||||
MT7622_TXQ_MGMT, MT7615_TX_MGMT_RING_SIZE);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = mt7615_init_tx_queue(dev, &dev->mt76.q_tx[MT_TXQ_MCU],
|
||||
MT7622_TXQ_MCU, MT7615_TX_MCU_RING_SIZE);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int
|
||||
mt7615_init_tx_queues(struct mt7615_dev *dev)
|
||||
{
|
||||
struct mt76_sw_queue *q;
|
||||
int ret, i;
|
||||
|
||||
ret = mt7615_init_tx_queue(dev, &dev->mt76.q_tx[MT_TXQ_FWDL],
|
||||
MT7615_TXQ_FWDL,
|
||||
MT7615_TX_FWDL_RING_SIZE);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (!is_mt7615(&dev->mt76))
|
||||
return mt7622_init_tx_queues_multi(dev);
|
||||
|
||||
ret = mt7615_init_tx_queue(dev, &dev->mt76.q_tx[0], 0,
|
||||
MT7615_TX_RING_SIZE);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
for (i = 1; i < MT_TXQ_MCU; i++) {
|
||||
q = &dev->mt76.q_tx[i];
|
||||
INIT_LIST_HEAD(&q->swq);
|
||||
q->q = dev->mt76.q_tx[0].q;
|
||||
}
|
||||
|
||||
ret = mt7615_init_tx_queue(dev, &dev->mt76.q_tx[MT_TXQ_MCU],
|
||||
MT7615_TXQ_MCU,
|
||||
MT7615_TX_MCU_RING_SIZE);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void mt7615_queue_rx_skb(struct mt76_dev *mdev, enum mt76_rxq_id q,
|
||||
struct sk_buff *skb)
|
||||
{
|
||||
@ -90,25 +128,32 @@ void mt7615_queue_rx_skb(struct mt76_dev *mdev, enum mt76_rxq_id q,
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
mt7615_tx_cleanup(struct mt7615_dev *dev)
|
||||
{
|
||||
int i;
|
||||
|
||||
mt76_queue_tx_cleanup(dev, MT_TXQ_MCU, false);
|
||||
if (is_mt7615(&dev->mt76)) {
|
||||
mt76_queue_tx_cleanup(dev, MT_TXQ_BE, false);
|
||||
} else {
|
||||
for (i = 0; i < IEEE80211_NUM_ACS; i++)
|
||||
mt76_queue_tx_cleanup(dev, i, false);
|
||||
}
|
||||
}
|
||||
|
||||
static int mt7615_poll_tx(struct napi_struct *napi, int budget)
|
||||
{
|
||||
static const u8 queue_map[] = {
|
||||
MT_TXQ_MCU,
|
||||
MT_TXQ_BE
|
||||
};
|
||||
struct mt7615_dev *dev;
|
||||
int i;
|
||||
|
||||
dev = container_of(napi, struct mt7615_dev, mt76.tx_napi);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(queue_map); i++)
|
||||
mt76_queue_tx_cleanup(dev, queue_map[i], false);
|
||||
mt7615_tx_cleanup(dev);
|
||||
|
||||
if (napi_complete_done(napi, 0))
|
||||
mt7615_irq_enable(dev, MT_INT_TX_DONE_ALL);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(queue_map); i++)
|
||||
mt76_queue_tx_cleanup(dev, queue_map[i], false);
|
||||
mt7615_tx_cleanup(dev);
|
||||
|
||||
mt7615_mac_sta_poll(dev);
|
||||
|
||||
@ -117,8 +162,33 @@ static int mt7615_poll_tx(struct napi_struct *napi, int budget)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void mt7622_dma_sched_init(struct mt7615_dev *dev)
|
||||
{
|
||||
u32 reg = mt7615_reg_map(dev, MT_DMASHDL_BASE);
|
||||
int i;
|
||||
|
||||
mt76_rmw(dev, reg + MT_DMASHDL_PKT_MAX_SIZE,
|
||||
MT_DMASHDL_PKT_MAX_SIZE_PLE | MT_DMASHDL_PKT_MAX_SIZE_PSE,
|
||||
FIELD_PREP(MT_DMASHDL_PKT_MAX_SIZE_PLE, 1) |
|
||||
FIELD_PREP(MT_DMASHDL_PKT_MAX_SIZE_PSE, 8));
|
||||
|
||||
for (i = 0; i <= 5; i++)
|
||||
mt76_wr(dev, reg + MT_DMASHDL_GROUP_QUOTA(i),
|
||||
FIELD_PREP(MT_DMASHDL_GROUP_QUOTA_MIN, 0x10) |
|
||||
FIELD_PREP(MT_DMASHDL_GROUP_QUOTA_MAX, 0x800));
|
||||
|
||||
mt76_wr(dev, reg + MT_DMASHDL_Q_MAP(0), 0x42104210);
|
||||
mt76_wr(dev, reg + MT_DMASHDL_Q_MAP(1), 0x42104210);
|
||||
mt76_wr(dev, reg + MT_DMASHDL_Q_MAP(2), 0x5);
|
||||
mt76_wr(dev, reg + MT_DMASHDL_Q_MAP(3), 0);
|
||||
|
||||
mt76_wr(dev, reg + MT_DMASHDL_SCHED_SET0, 0x6012345f);
|
||||
mt76_wr(dev, reg + MT_DMASHDL_SCHED_SET1, 0xedcba987);
|
||||
}
|
||||
|
||||
int mt7615_dma_init(struct mt7615_dev *dev)
|
||||
{
|
||||
int rx_ring_size = MT7615_RX_RING_SIZE;
|
||||
int ret;
|
||||
|
||||
mt76_dma_attach(&dev->mt76);
|
||||
@ -126,9 +196,12 @@ int mt7615_dma_init(struct mt7615_dev *dev)
|
||||
mt76_wr(dev, MT_WPDMA_GLO_CFG,
|
||||
MT_WPDMA_GLO_CFG_TX_WRITEBACK_DONE |
|
||||
MT_WPDMA_GLO_CFG_FIFO_LITTLE_ENDIAN |
|
||||
MT_WPDMA_GLO_CFG_FIRST_TOKEN_ONLY |
|
||||
MT_WPDMA_GLO_CFG_OMIT_TX_INFO);
|
||||
|
||||
if (!is_mt7622(&dev->mt76))
|
||||
mt76_set(dev, MT_WPDMA_GLO_CFG,
|
||||
MT_WPDMA_GLO_CFG_FIRST_TOKEN_ONLY);
|
||||
|
||||
mt76_rmw_field(dev, MT_WPDMA_GLO_CFG,
|
||||
MT_WPDMA_GLO_CFG_TX_BT_SIZE_BIT0, 0x1);
|
||||
|
||||
@ -141,28 +214,19 @@ int mt7615_dma_init(struct mt7615_dev *dev)
|
||||
mt76_rmw_field(dev, MT_WPDMA_GLO_CFG,
|
||||
MT_WPDMA_GLO_CFG_MULTI_DMA_EN, 0x3);
|
||||
|
||||
mt76_wr(dev, MT_WPDMA_GLO_CFG1, 0x1);
|
||||
mt76_wr(dev, MT_WPDMA_TX_PRE_CFG, 0xf0000);
|
||||
mt76_wr(dev, MT_WPDMA_RX_PRE_CFG, 0xf7f0000);
|
||||
mt76_wr(dev, MT_WPDMA_ABT_CFG, 0x4000026);
|
||||
mt76_wr(dev, MT_WPDMA_ABT_CFG1, 0x18811881);
|
||||
mt76_set(dev, 0x7158, BIT(16));
|
||||
mt76_clear(dev, 0x7000, BIT(23));
|
||||
if (is_mt7615(&dev->mt76)) {
|
||||
mt76_wr(dev, MT_WPDMA_GLO_CFG1, 0x1);
|
||||
mt76_wr(dev, MT_WPDMA_TX_PRE_CFG, 0xf0000);
|
||||
mt76_wr(dev, MT_WPDMA_RX_PRE_CFG, 0xf7f0000);
|
||||
mt76_wr(dev, MT_WPDMA_ABT_CFG, 0x4000026);
|
||||
mt76_wr(dev, MT_WPDMA_ABT_CFG1, 0x18811881);
|
||||
mt76_set(dev, 0x7158, BIT(16));
|
||||
mt76_clear(dev, 0x7000, BIT(23));
|
||||
}
|
||||
|
||||
mt76_wr(dev, MT_WPDMA_RST_IDX, ~0);
|
||||
|
||||
ret = mt7615_init_tx_queues(dev, MT7615_TX_RING_SIZE);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = mt7615_init_mcu_queue(dev, &dev->mt76.q_tx[MT_TXQ_MCU],
|
||||
MT7615_TXQ_MCU,
|
||||
MT7615_TX_MCU_RING_SIZE);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = mt7615_init_mcu_queue(dev, &dev->mt76.q_tx[MT_TXQ_FWDL],
|
||||
MT7615_TXQ_FWDL,
|
||||
MT7615_TX_FWDL_RING_SIZE);
|
||||
ret = mt7615_init_tx_queues(dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
@ -173,9 +237,11 @@ int mt7615_dma_init(struct mt7615_dev *dev)
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (!is_mt7615(&dev->mt76))
|
||||
rx_ring_size /= 2;
|
||||
|
||||
ret = mt76_queue_alloc(dev, &dev->mt76.q_rx[MT_RXQ_MAIN], 0,
|
||||
MT7615_RX_RING_SIZE, MT_RX_BUF_SIZE,
|
||||
MT_RX_RING_BASE);
|
||||
rx_ring_size, MT_RX_BUF_SIZE, MT_RX_RING_BASE);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
@ -199,7 +265,11 @@ int mt7615_dma_init(struct mt7615_dev *dev)
|
||||
MT_WPDMA_GLO_CFG_RX_DMA_EN);
|
||||
|
||||
/* enable interrupts for TX/RX rings */
|
||||
mt7615_irq_enable(dev, MT_INT_RX_DONE_ALL | MT_INT_TX_DONE_ALL);
|
||||
mt7615_irq_enable(dev, MT_INT_RX_DONE_ALL | MT_INT_TX_DONE_ALL |
|
||||
MT_INT_MCU_CMD);
|
||||
|
||||
if (is_mt7622(&dev->mt76))
|
||||
mt7622_dma_sched_init(dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -84,6 +84,7 @@ static int mt7615_check_eeprom(struct mt76_dev *dev)
|
||||
|
||||
switch (val) {
|
||||
case 0x7615:
|
||||
case 0x7622:
|
||||
return 0;
|
||||
default:
|
||||
return -EINVAL;
|
||||
@ -93,7 +94,7 @@ static int mt7615_check_eeprom(struct mt76_dev *dev)
|
||||
static void mt7615_eeprom_parse_hw_cap(struct mt7615_dev *dev)
|
||||
{
|
||||
u8 *eeprom = dev->mt76.eeprom.data;
|
||||
u8 tx_mask, rx_mask, max_nss;
|
||||
u8 tx_mask, max_nss;
|
||||
u32 val;
|
||||
|
||||
val = FIELD_GET(MT_EE_NIC_WIFI_CONF_BAND_SEL,
|
||||
@ -111,22 +112,21 @@ static void mt7615_eeprom_parse_hw_cap(struct mt7615_dev *dev)
|
||||
break;
|
||||
}
|
||||
|
||||
if (is_mt7622(&dev->mt76))
|
||||
dev->mt76.cap.has_5ghz = false;
|
||||
|
||||
/* read tx-rx mask from eeprom */
|
||||
val = mt76_rr(dev, MT_TOP_STRAP_STA);
|
||||
max_nss = val & MT_TOP_3NSS ? 3 : 4;
|
||||
|
||||
rx_mask = FIELD_GET(MT_EE_NIC_CONF_RX_MASK,
|
||||
eeprom[MT_EE_NIC_CONF_0]);
|
||||
if (!rx_mask || rx_mask > max_nss)
|
||||
rx_mask = max_nss;
|
||||
|
||||
tx_mask = FIELD_GET(MT_EE_NIC_CONF_TX_MASK,
|
||||
eeprom[MT_EE_NIC_CONF_0]);
|
||||
if (!tx_mask || tx_mask > max_nss)
|
||||
tx_mask = max_nss;
|
||||
|
||||
dev->mt76.chainmask = tx_mask << 8 | rx_mask;
|
||||
dev->mt76.antenna_mask = BIT(tx_mask) - 1;
|
||||
dev->chainmask = BIT(tx_mask) - 1;
|
||||
dev->mphy.antenna_mask = dev->chainmask;
|
||||
dev->phy.chainmask = dev->chainmask;
|
||||
}
|
||||
|
||||
int mt7615_eeprom_get_power_index(struct mt7615_dev *dev,
|
||||
@ -209,6 +209,26 @@ static void mt7615_apply_cal_free_data(struct mt7615_dev *dev)
|
||||
eeprom[ical_nocheck[i]] = otp[ical_nocheck[i]];
|
||||
}
|
||||
|
||||
static void mt7622_apply_cal_free_data(struct mt7615_dev *dev)
|
||||
{
|
||||
static const u16 ical[] = {
|
||||
0x53, 0x54, 0x55, 0x56, 0xf4, 0xf7, 0x144, 0x156, 0x15b
|
||||
};
|
||||
u8 *eeprom = dev->mt76.eeprom.data;
|
||||
u8 *otp = dev->mt76.otp.data;
|
||||
int i;
|
||||
|
||||
if (!otp)
|
||||
return;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(ical); i++) {
|
||||
if (!otp[ical[i]])
|
||||
continue;
|
||||
|
||||
eeprom[ical[i]] = otp[ical[i]];
|
||||
}
|
||||
}
|
||||
|
||||
int mt7615_eeprom_init(struct mt7615_dev *dev)
|
||||
{
|
||||
int ret;
|
||||
@ -221,6 +241,8 @@ int mt7615_eeprom_init(struct mt7615_dev *dev)
|
||||
if (ret && dev->mt76.otp.data)
|
||||
memcpy(dev->mt76.eeprom.data, dev->mt76.otp.data,
|
||||
MT7615_EEPROM_SIZE);
|
||||
else if (is_mt7622(&dev->mt76))
|
||||
mt7622_apply_cal_free_data(dev);
|
||||
else
|
||||
mt7615_apply_cal_free_data(dev);
|
||||
|
||||
|
@ -21,7 +21,8 @@ enum mt7615_eeprom_field {
|
||||
MT_EE_TX2_5G_G0_TARGET_POWER = 0x142,
|
||||
MT_EE_TX3_5G_G0_TARGET_POWER = 0x16a,
|
||||
|
||||
__MT_EE_MAX = 0x3bf
|
||||
MT7615_EE_MAX = 0x3bf,
|
||||
MT7622_EE_MAX = 0x3db,
|
||||
};
|
||||
|
||||
#define MT_EE_NIC_CONF_TX_MASK GENMASK(7, 4)
|
||||
|
@ -13,9 +13,9 @@
|
||||
|
||||
static void mt7615_phy_init(struct mt7615_dev *dev)
|
||||
{
|
||||
/* disable band 0 rf low power beacon mode */
|
||||
mt76_rmw(dev, MT_WF_PHY_WF2_RFCTRL0, MT_WF_PHY_WF2_RFCTRL0_LPBCN_EN,
|
||||
MT_WF_PHY_WF2_RFCTRL0_LPBCN_EN);
|
||||
/* disable rf low power beacon mode */
|
||||
mt76_set(dev, MT_WF_PHY_WF2_RFCTRL0(0), MT_WF_PHY_WF2_RFCTRL0_LPBCN_EN);
|
||||
mt76_set(dev, MT_WF_PHY_WF2_RFCTRL0(1), MT_WF_PHY_WF2_RFCTRL0_LPBCN_EN);
|
||||
}
|
||||
|
||||
static void mt7615_mac_init(struct mt7615_dev *dev)
|
||||
@ -28,17 +28,17 @@ static void mt7615_mac_init(struct mt7615_dev *dev)
|
||||
MT_CFG_CCR_MAC_D0_1X_GC_EN | MT_CFG_CCR_MAC_D0_2X_GC_EN |
|
||||
MT_CFG_CCR_MAC_D1_1X_GC_EN | MT_CFG_CCR_MAC_D1_2X_GC_EN);
|
||||
|
||||
val = mt76_rmw(dev, MT_TMAC_TRCR0,
|
||||
val = mt76_rmw(dev, MT_TMAC_TRCR(0),
|
||||
MT_TMAC_TRCR_CCA_SEL | MT_TMAC_TRCR_SEC_CCA_SEL,
|
||||
FIELD_PREP(MT_TMAC_TRCR_CCA_SEL, 2) |
|
||||
FIELD_PREP(MT_TMAC_TRCR_SEC_CCA_SEL, 0));
|
||||
mt76_wr(dev, MT_TMAC_TRCR1, val);
|
||||
mt76_wr(dev, MT_TMAC_TRCR(1), val);
|
||||
|
||||
val = MT_AGG_ACR_PKT_TIME_EN | MT_AGG_ACR_NO_BA_AR_RULE |
|
||||
FIELD_PREP(MT_AGG_ACR_CFEND_RATE, 0x49) | /* 24M */
|
||||
FIELD_PREP(MT_AGG_ACR_BAR_RATE, 0x4b); /* 6M */
|
||||
mt76_wr(dev, MT_AGG_ACR0, val);
|
||||
mt76_wr(dev, MT_AGG_ACR1, val);
|
||||
FIELD_PREP(MT_AGG_ACR_CFEND_RATE, MT7615_CFEND_RATE_DEFAULT) |
|
||||
FIELD_PREP(MT_AGG_ACR_BAR_RATE, MT7615_BAR_RATE_DEFAULT);
|
||||
mt76_wr(dev, MT_AGG_ACR(0), val);
|
||||
mt76_wr(dev, MT_AGG_ACR(1), val);
|
||||
|
||||
mt76_rmw_field(dev, MT_TMAC_CTCR0,
|
||||
MT_TMAC_CTCR0_INS_DDLMT_REFTIME, 0x3f);
|
||||
@ -50,36 +50,36 @@ static void mt7615_mac_init(struct mt7615_dev *dev)
|
||||
MT_TMAC_CTCR0_INS_DDLMT_VHT_SMPDU_EN |
|
||||
MT_TMAC_CTCR0_INS_DDLMT_EN);
|
||||
|
||||
mt7615_mcu_set_rts_thresh(dev, 0x92b);
|
||||
mt7615_mcu_set_rts_thresh(&dev->phy, 0x92b);
|
||||
mt7615_mac_set_scs(dev, true);
|
||||
|
||||
mt76_rmw(dev, MT_AGG_SCR, MT_AGG_SCR_NLNAV_MID_PTEC_DIS,
|
||||
MT_AGG_SCR_NLNAV_MID_PTEC_DIS);
|
||||
|
||||
mt7615_mcu_init_mac(dev);
|
||||
|
||||
mt76_wr(dev, MT_DMA_DCR0, MT_DMA_DCR0_RX_VEC_DROP |
|
||||
FIELD_PREP(MT_DMA_DCR0_MAX_RX_LEN, 3072));
|
||||
|
||||
mt76_wr(dev, MT_AGG_ARUCR,
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(0), 7) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(1), 2) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(2), 2) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(3), 2) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(4), 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(5), 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(6), 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(7), 1));
|
||||
val = FIELD_PREP(MT_AGG_ARxCR_LIMIT(0), 7) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(1), 2) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(2), 2) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(3), 2) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(4), 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(5), 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(6), 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(7), 1);
|
||||
mt76_wr(dev, MT_AGG_ARUCR(0), val);
|
||||
mt76_wr(dev, MT_AGG_ARUCR(1), val);
|
||||
|
||||
mt76_wr(dev, MT_AGG_ARDCR,
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(0), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(1), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(2), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(3), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(4), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(5), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(6), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(7), MT7615_RATE_RETRY - 1));
|
||||
val = FIELD_PREP(MT_AGG_ARxCR_LIMIT(0), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(1), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(2), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(3), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(4), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(5), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(6), MT7615_RATE_RETRY - 1) |
|
||||
FIELD_PREP(MT_AGG_ARxCR_LIMIT(7), MT7615_RATE_RETRY - 1);
|
||||
mt76_wr(dev, MT_AGG_ARDCR(0), val);
|
||||
mt76_wr(dev, MT_AGG_ARDCR(1), val);
|
||||
|
||||
mt76_wr(dev, MT_AGG_ARCR,
|
||||
(FIELD_PREP(MT_AGG_ARCR_RTS_RATE_THR, 2) |
|
||||
@ -95,8 +95,8 @@ static void mt7615_mac_init(struct mt7615_dev *dev)
|
||||
MT_DMA_RCFR0_RX_DROPPED_MCAST;
|
||||
set = FIELD_PREP(MT_DMA_RCFR0_RX_DROPPED_UCAST, 2) |
|
||||
FIELD_PREP(MT_DMA_RCFR0_RX_DROPPED_MCAST, 2);
|
||||
mt76_rmw(dev, MT_DMA_BN0RCFR0, mask, set);
|
||||
mt76_rmw(dev, MT_DMA_BN1RCFR0, mask, set);
|
||||
mt76_rmw(dev, MT_DMA_RCFR0(0), mask, set);
|
||||
mt76_rmw(dev, MT_DMA_RCFR0(1), mask, set);
|
||||
|
||||
for (i = 0; i < MT7615_WTBL_SIZE; i++)
|
||||
mt7615_mac_wtbl_update(dev, i,
|
||||
@ -106,12 +106,33 @@ static void mt7615_mac_init(struct mt7615_dev *dev)
|
||||
mt76_set(dev, MT_WF_RMAC_MIB_AIRTIME0, MT_WF_RMAC_MIB_RXTIME_EN);
|
||||
}
|
||||
|
||||
bool mt7615_wait_for_mcu_init(struct mt7615_dev *dev)
|
||||
{
|
||||
flush_work(&dev->mcu_work);
|
||||
|
||||
return test_bit(MT76_STATE_MCU_RUNNING, &dev->mphy.state);
|
||||
}
|
||||
|
||||
static void mt7615_init_work(struct work_struct *work)
|
||||
{
|
||||
struct mt7615_dev *dev = container_of(work, struct mt7615_dev, mcu_work);
|
||||
|
||||
if (mt7615_mcu_init(dev))
|
||||
return;
|
||||
|
||||
mt7615_mcu_set_eeprom(dev);
|
||||
mt7615_mac_init(dev);
|
||||
mt7615_phy_init(dev);
|
||||
mt7615_mcu_del_wtbl_all(dev);
|
||||
}
|
||||
|
||||
static int mt7615_init_hardware(struct mt7615_dev *dev)
|
||||
{
|
||||
int ret, idx;
|
||||
|
||||
mt76_wr(dev, MT_INT_SOURCE_CSR, ~0);
|
||||
|
||||
INIT_WORK(&dev->mcu_work, mt7615_init_work);
|
||||
spin_lock_init(&dev->token_lock);
|
||||
idr_init(&dev->token);
|
||||
|
||||
@ -123,17 +144,7 @@ static int mt7615_init_hardware(struct mt7615_dev *dev)
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mt76.state);
|
||||
|
||||
ret = mt7615_mcu_init(dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
mt7615_mcu_set_eeprom(dev);
|
||||
mt7615_mac_init(dev);
|
||||
mt7615_phy_init(dev);
|
||||
mt7615_mcu_ctrl_pm_state(dev, 0);
|
||||
mt7615_mcu_del_wtbl_all(dev);
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);
|
||||
|
||||
/* Beacon and mgmt frames should occupy wcid 0 */
|
||||
idx = mt76_wcid_alloc(dev->mt76.wcid_mask, MT7615_WTBL_STA - 1);
|
||||
@ -199,13 +210,66 @@ static const struct ieee80211_iface_combination if_comb[] = {
|
||||
}
|
||||
};
|
||||
|
||||
static void
|
||||
mt7615_led_set_config(struct led_classdev *led_cdev,
|
||||
u8 delay_on, u8 delay_off)
|
||||
{
|
||||
struct mt7615_dev *dev;
|
||||
struct mt76_dev *mt76;
|
||||
u32 val, addr;
|
||||
|
||||
mt76 = container_of(led_cdev, struct mt76_dev, led_cdev);
|
||||
dev = container_of(mt76, struct mt7615_dev, mt76);
|
||||
val = FIELD_PREP(MT_LED_STATUS_DURATION, 0xffff) |
|
||||
FIELD_PREP(MT_LED_STATUS_OFF, delay_off) |
|
||||
FIELD_PREP(MT_LED_STATUS_ON, delay_on);
|
||||
|
||||
addr = mt7615_reg_map(dev, MT_LED_STATUS_0(mt76->led_pin));
|
||||
mt76_wr(dev, addr, val);
|
||||
addr = mt7615_reg_map(dev, MT_LED_STATUS_1(mt76->led_pin));
|
||||
mt76_wr(dev, addr, val);
|
||||
|
||||
val = MT_LED_CTRL_REPLAY(mt76->led_pin) |
|
||||
MT_LED_CTRL_KICK(mt76->led_pin);
|
||||
if (mt76->led_al)
|
||||
val |= MT_LED_CTRL_POLARITY(mt76->led_pin);
|
||||
addr = mt7615_reg_map(dev, MT_LED_CTRL);
|
||||
mt76_wr(dev, addr, val);
|
||||
}
|
||||
|
||||
static int
|
||||
mt7615_led_set_blink(struct led_classdev *led_cdev,
|
||||
unsigned long *delay_on,
|
||||
unsigned long *delay_off)
|
||||
{
|
||||
u8 delta_on, delta_off;
|
||||
|
||||
delta_off = max_t(u8, *delay_off / 10, 1);
|
||||
delta_on = max_t(u8, *delay_on / 10, 1);
|
||||
|
||||
mt7615_led_set_config(led_cdev, delta_on, delta_off);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
mt7615_led_set_brightness(struct led_classdev *led_cdev,
|
||||
enum led_brightness brightness)
|
||||
{
|
||||
if (!brightness)
|
||||
mt7615_led_set_config(led_cdev, 0, 0xff);
|
||||
else
|
||||
mt7615_led_set_config(led_cdev, 0xff, 0);
|
||||
}
|
||||
|
||||
static void
|
||||
mt7615_init_txpower(struct mt7615_dev *dev,
|
||||
struct ieee80211_supported_band *sband)
|
||||
{
|
||||
int i, n_chains = hweight8(dev->mt76.antenna_mask), target_chains;
|
||||
int i, n_chains = hweight8(dev->mphy.antenna_mask), target_chains;
|
||||
u8 *eep = (u8 *)dev->mt76.eeprom.data;
|
||||
enum nl80211_band band = sband->band;
|
||||
int delta = mt76_tx_power_nss_delta(n_chains);
|
||||
|
||||
target_chains = mt7615_ext_pa_enabled(dev, band) ? 1 : n_chains;
|
||||
for (i = 0; i < sband->n_channels; i++) {
|
||||
@ -220,21 +284,7 @@ mt7615_init_txpower(struct mt7615_dev *dev,
|
||||
target_power = max(target_power, eep[index]);
|
||||
}
|
||||
|
||||
target_power = DIV_ROUND_UP(target_power, 2);
|
||||
switch (n_chains) {
|
||||
case 4:
|
||||
target_power += 6;
|
||||
break;
|
||||
case 3:
|
||||
target_power += 4;
|
||||
break;
|
||||
case 2:
|
||||
target_power += 3;
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
target_power = DIV_ROUND_UP(target_power + delta, 2);
|
||||
chan->max_power = min_t(int, chan->max_reg_power,
|
||||
target_power);
|
||||
chan->orig_mpwr = target_power;
|
||||
@ -246,74 +296,181 @@ mt7615_regd_notifier(struct wiphy *wiphy,
|
||||
struct regulatory_request *request)
|
||||
{
|
||||
struct ieee80211_hw *hw = wiphy_to_ieee80211_hw(wiphy);
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct cfg80211_chan_def *chandef = &dev->mt76.chandef;
|
||||
|
||||
if (request->dfs_region == dev->mt76.region)
|
||||
return;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt76_phy *mphy = hw->priv;
|
||||
struct mt7615_phy *phy = mphy->priv;
|
||||
struct cfg80211_chan_def *chandef = &mphy->chandef;
|
||||
|
||||
dev->mt76.region = request->dfs_region;
|
||||
|
||||
if (!(chandef->chan->flags & IEEE80211_CHAN_RADAR))
|
||||
return;
|
||||
|
||||
mt7615_dfs_stop_radar_detector(dev);
|
||||
if (request->dfs_region == NL80211_DFS_UNSET)
|
||||
mt7615_mcu_rdd_cmd(dev, RDD_CAC_END, MT_HW_RDD0,
|
||||
MT_RX_SEL0, 0);
|
||||
else
|
||||
mt7615_dfs_start_radar_detector(dev);
|
||||
mt7615_dfs_init_radar_detector(phy);
|
||||
}
|
||||
|
||||
int mt7615_register_device(struct mt7615_dev *dev)
|
||||
static void
|
||||
mt7615_init_wiphy(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct ieee80211_hw *hw = mt76_hw(dev);
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
struct wiphy *wiphy = hw->wiphy;
|
||||
int ret;
|
||||
|
||||
INIT_DELAYED_WORK(&dev->mt76.mac_work, mt7615_mac_work);
|
||||
INIT_LIST_HEAD(&dev->sta_poll_list);
|
||||
spin_lock_init(&dev->sta_poll_lock);
|
||||
|
||||
ret = mt7615_init_hardware(dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
hw->queues = 4;
|
||||
hw->max_rates = 3;
|
||||
hw->max_report_rates = 7;
|
||||
hw->max_rate_tries = 11;
|
||||
|
||||
phy->slottime = 9;
|
||||
|
||||
hw->sta_data_size = sizeof(struct mt7615_sta);
|
||||
hw->vif_data_size = sizeof(struct mt7615_vif);
|
||||
|
||||
wiphy->iface_combinations = if_comb;
|
||||
wiphy->n_iface_combinations = ARRAY_SIZE(if_comb);
|
||||
wiphy->reg_notifier = mt7615_regd_notifier;
|
||||
wiphy->flags |= WIPHY_FLAG_HAS_CHANNEL_SWITCH;
|
||||
|
||||
wiphy_ext_feature_set(wiphy, NL80211_EXT_FEATURE_VHT_IBSS);
|
||||
|
||||
ieee80211_hw_set(hw, TX_STATUS_NO_AMPDU_LEN);
|
||||
|
||||
dev->mt76.sband_2g.sband.ht_cap.cap |= IEEE80211_HT_CAP_LDPC_CODING;
|
||||
dev->mt76.sband_5g.sband.ht_cap.cap |= IEEE80211_HT_CAP_LDPC_CODING;
|
||||
dev->mt76.sband_5g.sband.vht_cap.cap |=
|
||||
if (is_mt7615(&phy->dev->mt76))
|
||||
hw->max_tx_fragments = MT_TXP_MAX_BUF_NUM;
|
||||
else
|
||||
hw->max_tx_fragments = MT_HW_TXP_MAX_BUF_NUM;
|
||||
}
|
||||
|
||||
static void
|
||||
mt7615_cap_dbdc_enable(struct mt7615_dev *dev)
|
||||
{
|
||||
dev->mphy.sband_5g.sband.vht_cap.cap &=
|
||||
~(IEEE80211_VHT_CAP_SHORT_GI_160 |
|
||||
IEEE80211_VHT_CAP_SUPP_CHAN_WIDTH_160_80PLUS80MHZ);
|
||||
if (dev->chainmask == 0xf)
|
||||
dev->mphy.antenna_mask = dev->chainmask >> 2;
|
||||
else
|
||||
dev->mphy.antenna_mask = dev->chainmask >> 1;
|
||||
dev->phy.chainmask = dev->mphy.antenna_mask;
|
||||
mt76_set_stream_caps(&dev->mt76, true);
|
||||
}
|
||||
|
||||
static void
|
||||
mt7615_cap_dbdc_disable(struct mt7615_dev *dev)
|
||||
{
|
||||
dev->mphy.sband_5g.sband.vht_cap.cap |=
|
||||
IEEE80211_VHT_CAP_SHORT_GI_160 |
|
||||
IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_11454 |
|
||||
IEEE80211_VHT_CAP_MAX_A_MPDU_LENGTH_EXPONENT_MASK |
|
||||
IEEE80211_VHT_CAP_SUPP_CHAN_WIDTH_160_80PLUS80MHZ;
|
||||
dev->dfs_state = -1;
|
||||
dev->mphy.antenna_mask = dev->chainmask;
|
||||
dev->phy.chainmask = dev->chainmask;
|
||||
mt76_set_stream_caps(&dev->mt76, true);
|
||||
}
|
||||
|
||||
int mt7615_register_ext_phy(struct mt7615_dev *dev)
|
||||
{
|
||||
struct mt7615_phy *phy = mt7615_ext_phy(dev);
|
||||
struct mt76_phy *mphy;
|
||||
int ret;
|
||||
|
||||
if (!is_mt7615(&dev->mt76))
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
if (test_bit(MT76_STATE_RUNNING, &dev->mphy.state))
|
||||
return -EINVAL;
|
||||
|
||||
if (phy)
|
||||
return 0;
|
||||
|
||||
mt7615_cap_dbdc_enable(dev);
|
||||
mphy = mt76_alloc_phy(&dev->mt76, sizeof(*phy), &mt7615_ops);
|
||||
if (!mphy)
|
||||
return -ENOMEM;
|
||||
|
||||
phy = mphy->priv;
|
||||
phy->dev = dev;
|
||||
phy->mt76 = mphy;
|
||||
phy->chainmask = dev->chainmask & ~dev->phy.chainmask;
|
||||
mphy->antenna_mask = BIT(hweight8(phy->chainmask)) - 1;
|
||||
mt7615_init_wiphy(mphy->hw);
|
||||
|
||||
/*
|
||||
* Make the secondary PHY MAC address local without overlapping with
|
||||
* the usual MAC address allocation scheme on multiple virtual interfaces
|
||||
*/
|
||||
mphy->hw->wiphy->perm_addr[0] |= 2;
|
||||
mphy->hw->wiphy->perm_addr[0] ^= BIT(7);
|
||||
|
||||
/* second phy can only handle 5 GHz */
|
||||
mphy->sband_2g.sband.n_channels = 0;
|
||||
mphy->hw->wiphy->bands[NL80211_BAND_2GHZ] = NULL;
|
||||
|
||||
/* The second interface does not get any packets unless it has a vif */
|
||||
ieee80211_hw_set(mphy->hw, WANT_MONITOR_VIF);
|
||||
|
||||
ret = mt76_register_phy(mphy);
|
||||
if (ret)
|
||||
ieee80211_free_hw(mphy->hw);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
void mt7615_unregister_ext_phy(struct mt7615_dev *dev)
|
||||
{
|
||||
struct mt7615_phy *phy = mt7615_ext_phy(dev);
|
||||
struct mt76_phy *mphy = dev->mt76.phy2;
|
||||
|
||||
if (!phy)
|
||||
return;
|
||||
|
||||
mt7615_cap_dbdc_disable(dev);
|
||||
mt76_unregister_phy(mphy);
|
||||
ieee80211_free_hw(mphy->hw);
|
||||
}
|
||||
|
||||
|
||||
int mt7615_register_device(struct mt7615_dev *dev)
|
||||
{
|
||||
struct ieee80211_hw *hw = mt76_hw(dev);
|
||||
int ret;
|
||||
|
||||
dev->phy.dev = dev;
|
||||
dev->phy.mt76 = &dev->mt76.phy;
|
||||
dev->mt76.phy.priv = &dev->phy;
|
||||
INIT_DELAYED_WORK(&dev->mt76.mac_work, mt7615_mac_work);
|
||||
INIT_LIST_HEAD(&dev->sta_poll_list);
|
||||
spin_lock_init(&dev->sta_poll_lock);
|
||||
init_waitqueue_head(&dev->reset_wait);
|
||||
INIT_WORK(&dev->reset_work, mt7615_mac_reset_work);
|
||||
|
||||
ret = mt7622_wmac_init(dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = mt7615_init_hardware(dev);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
mt7615_init_wiphy(hw);
|
||||
dev->mphy.sband_2g.sband.ht_cap.cap |= IEEE80211_HT_CAP_LDPC_CODING;
|
||||
dev->mphy.sband_5g.sband.ht_cap.cap |= IEEE80211_HT_CAP_LDPC_CODING;
|
||||
dev->mphy.sband_5g.sband.vht_cap.cap |=
|
||||
IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_11454 |
|
||||
IEEE80211_VHT_CAP_MAX_A_MPDU_LENGTH_EXPONENT_MASK;
|
||||
mt7615_cap_dbdc_disable(dev);
|
||||
dev->phy.dfs_state = -1;
|
||||
|
||||
/* init led callbacks */
|
||||
if (IS_ENABLED(CONFIG_MT76_LEDS)) {
|
||||
dev->mt76.led_cdev.brightness_set = mt7615_led_set_brightness;
|
||||
dev->mt76.led_cdev.blink_set = mt7615_led_set_blink;
|
||||
}
|
||||
|
||||
ret = mt76_register_device(&dev->mt76, true, mt7615_rates,
|
||||
ARRAY_SIZE(mt7615_rates));
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
mt7615_init_txpower(dev, &dev->mt76.sband_2g.sband);
|
||||
mt7615_init_txpower(dev, &dev->mt76.sband_5g.sband);
|
||||
|
||||
hw->max_tx_fragments = MT_TXP_MAX_BUF_NUM;
|
||||
ieee80211_queue_work(mt76_hw(dev), &dev->mcu_work);
|
||||
mt7615_init_txpower(dev, &dev->mphy.sband_2g.sband);
|
||||
mt7615_init_txpower(dev, &dev->mphy.sband_5g.sband);
|
||||
|
||||
return mt7615_init_debugfs(dev);
|
||||
}
|
||||
@ -321,10 +478,15 @@ int mt7615_register_device(struct mt7615_dev *dev)
|
||||
void mt7615_unregister_device(struct mt7615_dev *dev)
|
||||
{
|
||||
struct mt76_txwi_cache *txwi;
|
||||
bool mcu_running;
|
||||
int id;
|
||||
|
||||
mcu_running = mt7615_wait_for_mcu_init(dev);
|
||||
|
||||
mt7615_unregister_ext_phy(dev);
|
||||
mt76_unregister_device(&dev->mt76);
|
||||
mt7615_mcu_exit(dev);
|
||||
if (mcu_running)
|
||||
mt7615_mcu_exit(dev);
|
||||
mt7615_dma_cleanup(dev);
|
||||
|
||||
spin_lock_bh(&dev->token_lock);
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -103,6 +103,11 @@ enum rx_pkt_type {
|
||||
#define MT_RXV4_RCPI1 GENMASK(15, 8)
|
||||
#define MT_RXV4_RCPI0 GENMASK(7, 0)
|
||||
|
||||
#define MT_RXV6_NF3 GENMASK(31, 24)
|
||||
#define MT_RXV6_NF2 GENMASK(23, 16)
|
||||
#define MT_RXV6_NF1 GENMASK(15, 8)
|
||||
#define MT_RXV6_NF0 GENMASK(7, 0)
|
||||
|
||||
enum tx_header_format {
|
||||
MT_HDR_FORMAT_802_3,
|
||||
MT_HDR_FORMAT_CMD,
|
||||
@ -126,6 +131,10 @@ enum tx_pkt_queue_idx {
|
||||
MT_LMAC_BMC0,
|
||||
MT_LMAC_BCN0,
|
||||
MT_LMAC_PSMP0,
|
||||
MT_LMAC_ALTX1,
|
||||
MT_LMAC_BMC1,
|
||||
MT_LMAC_BCN1,
|
||||
MT_LMAC_PSMP1,
|
||||
};
|
||||
|
||||
enum tx_port_idx {
|
||||
@ -229,8 +238,27 @@ enum tx_phy_bandwidth {
|
||||
#define MT_TX_RATE_IDX GENMASK(5, 0)
|
||||
|
||||
#define MT_TXP_MAX_BUF_NUM 6
|
||||
#define MT_HW_TXP_MAX_MSDU_NUM 4
|
||||
#define MT_HW_TXP_MAX_BUF_NUM 4
|
||||
|
||||
struct mt7615_txp {
|
||||
#define MT_MSDU_ID_VALID BIT(15)
|
||||
|
||||
#define MT_TXD_LEN_MSDU_LAST BIT(14)
|
||||
#define MT_TXD_LEN_AMSDU_LAST BIT(15)
|
||||
|
||||
struct mt7615_txp_ptr {
|
||||
__le32 buf0;
|
||||
__le16 len0;
|
||||
__le16 len1;
|
||||
__le32 buf1;
|
||||
} __packed __aligned(4);
|
||||
|
||||
struct mt7615_hw_txp {
|
||||
__le16 msdu_id[MT_HW_TXP_MAX_MSDU_NUM];
|
||||
struct mt7615_txp_ptr ptr[MT_HW_TXP_MAX_BUF_NUM / 2];
|
||||
} __packed __aligned(4);
|
||||
|
||||
struct mt7615_fw_txp {
|
||||
__le16 flags;
|
||||
__le16 token;
|
||||
u8 bss_idx;
|
||||
@ -239,7 +267,14 @@ struct mt7615_txp {
|
||||
u8 nbuf;
|
||||
__le32 buf[MT_TXP_MAX_BUF_NUM];
|
||||
__le16 len[MT_TXP_MAX_BUF_NUM];
|
||||
} __packed;
|
||||
} __packed __aligned(4);
|
||||
|
||||
struct mt7615_txp_common {
|
||||
union {
|
||||
struct mt7615_fw_txp fw;
|
||||
struct mt7615_hw_txp hw;
|
||||
};
|
||||
};
|
||||
|
||||
struct mt7615_tx_free {
|
||||
__le16 rx_byte_cnt;
|
||||
@ -247,7 +282,7 @@ struct mt7615_tx_free {
|
||||
u8 txd_cnt;
|
||||
u8 rsv[3];
|
||||
__le16 token[];
|
||||
} __packed;
|
||||
} __packed __aligned(4);
|
||||
|
||||
#define MT_TX_FREE_MSDU_ID_CNT GENMASK(6, 0)
|
||||
|
||||
@ -302,6 +337,38 @@ struct mt7615_tx_free {
|
||||
#define MT_TXS6_F1_RCPI_1 GENMASK(15, 8)
|
||||
#define MT_TXS6_F1_RCPI_0 GENMASK(7, 0)
|
||||
|
||||
struct mt7615_dfs_pulse {
|
||||
u32 max_width; /* us */
|
||||
int max_pwr; /* dbm */
|
||||
int min_pwr; /* dbm */
|
||||
u32 min_stgr_pri; /* us */
|
||||
u32 max_stgr_pri; /* us */
|
||||
u32 min_cr_pri; /* us */
|
||||
u32 max_cr_pri; /* us */
|
||||
};
|
||||
|
||||
struct mt7615_dfs_pattern {
|
||||
u8 enb;
|
||||
u8 stgr;
|
||||
u8 min_crpn;
|
||||
u8 max_crpn;
|
||||
u8 min_crpr;
|
||||
u8 min_pw;
|
||||
u8 max_pw;
|
||||
u32 min_pri;
|
||||
u32 max_pri;
|
||||
u8 min_crbn;
|
||||
u8 max_crbn;
|
||||
u8 min_stgpn;
|
||||
u8 max_stgpn;
|
||||
u8 min_stgpr;
|
||||
};
|
||||
|
||||
struct mt7615_dfs_radar_spec {
|
||||
struct mt7615_dfs_pulse pulse_th;
|
||||
struct mt7615_dfs_pattern radar_pattern[16];
|
||||
};
|
||||
|
||||
enum mt7615_cipher_type {
|
||||
MT_CIPHER_NONE,
|
||||
MT_CIPHER_WEP40,
|
||||
@ -317,7 +384,7 @@ enum mt7615_cipher_type {
|
||||
MT_CIPHER_GCMP_256,
|
||||
};
|
||||
|
||||
static inline struct mt7615_txp *
|
||||
static inline struct mt7615_txp_common *
|
||||
mt7615_txwi_to_txp(struct mt76_dev *dev, struct mt76_txwi_cache *t)
|
||||
{
|
||||
u8 *txwi;
|
||||
@ -327,7 +394,7 @@ mt7615_txwi_to_txp(struct mt76_dev *dev, struct mt76_txwi_cache *t)
|
||||
|
||||
txwi = mt76_get_txwi_ptr(dev, t);
|
||||
|
||||
return (struct mt7615_txp *)(txwi + MT_TXD_SIZE);
|
||||
return (struct mt7615_txp_common *)(txwi + MT_TXD_SIZE);
|
||||
}
|
||||
|
||||
#endif
|
||||
|
@ -11,27 +11,85 @@
|
||||
#include <linux/pci.h>
|
||||
#include <linux/module.h>
|
||||
#include "mt7615.h"
|
||||
#include "mcu.h"
|
||||
|
||||
static bool mt7615_dev_running(struct mt7615_dev *dev)
|
||||
{
|
||||
struct mt7615_phy *phy;
|
||||
|
||||
if (test_bit(MT76_STATE_RUNNING, &dev->mphy.state))
|
||||
return true;
|
||||
|
||||
phy = mt7615_ext_phy(dev);
|
||||
|
||||
return phy && test_bit(MT76_STATE_RUNNING, &phy->mt76->state);
|
||||
}
|
||||
|
||||
static int mt7615_start(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
bool running;
|
||||
|
||||
if (!mt7615_wait_for_mcu_init(dev))
|
||||
return -EIO;
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
|
||||
running = mt7615_dev_running(dev);
|
||||
|
||||
if (!running) {
|
||||
mt7615_mcu_ctrl_pm_state(dev, 0, 0);
|
||||
mt7615_mcu_set_mac_enable(dev, 0, true);
|
||||
mt7615_mac_enable_nf(dev, 0);
|
||||
}
|
||||
|
||||
if (phy != &dev->phy) {
|
||||
mt7615_mcu_ctrl_pm_state(dev, 1, 0);
|
||||
mt7615_mcu_set_mac_enable(dev, 1, true);
|
||||
mt7615_mac_enable_nf(dev, 1);
|
||||
}
|
||||
|
||||
mt7615_mcu_set_chan_info(phy, MCU_EXT_CMD_SET_RX_PATH);
|
||||
|
||||
set_bit(MT76_STATE_RUNNING, &phy->mt76->state);
|
||||
|
||||
if (running)
|
||||
goto out;
|
||||
|
||||
mt7615_mac_reset_counters(dev);
|
||||
|
||||
dev->mt76.survey_time = ktime_get_boottime();
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
ieee80211_queue_delayed_work(mt76_hw(dev), &dev->mt76.mac_work,
|
||||
MT7615_WATCHDOG_TIME);
|
||||
|
||||
out:
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void mt7615_stop(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
cancel_delayed_work_sync(&dev->mt76.mac_work);
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
|
||||
clear_bit(MT76_STATE_RUNNING, &phy->mt76->state);
|
||||
|
||||
if (phy != &dev->phy) {
|
||||
mt7615_mcu_ctrl_pm_state(dev, 1, 1);
|
||||
mt7615_mcu_set_mac_enable(dev, 1, false);
|
||||
}
|
||||
|
||||
if (!mt7615_dev_running(dev)) {
|
||||
cancel_delayed_work_sync(&dev->mt76.mac_work);
|
||||
|
||||
mt7615_mcu_ctrl_pm_state(dev, 0, 1);
|
||||
mt7615_mcu_set_mac_enable(dev, 0, false);
|
||||
}
|
||||
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
}
|
||||
|
||||
static int get_omac_idx(enum nl80211_iftype type, u32 mask)
|
||||
@ -39,6 +97,7 @@ static int get_omac_idx(enum nl80211_iftype type, u32 mask)
|
||||
int i;
|
||||
|
||||
switch (type) {
|
||||
case NL80211_IFTYPE_MONITOR:
|
||||
case NL80211_IFTYPE_AP:
|
||||
case NL80211_IFTYPE_MESH_POINT:
|
||||
case NL80211_IFTYPE_ADHOC:
|
||||
@ -70,8 +129,10 @@ static int mt7615_add_interface(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif)
|
||||
{
|
||||
struct mt7615_vif *mvif = (struct mt7615_vif *)vif->drv_priv;
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
struct mt76_txq *mtxq;
|
||||
bool ext_phy = phy != &dev->phy;
|
||||
int idx, ret = 0;
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
@ -89,9 +150,12 @@ static int mt7615_add_interface(struct ieee80211_hw *hw,
|
||||
}
|
||||
mvif->omac_idx = idx;
|
||||
|
||||
/* TODO: DBDC support. Use band 0 for now */
|
||||
mvif->band_idx = 0;
|
||||
mvif->wmm_idx = mvif->idx % MT7615_MAX_WMM_SETS;
|
||||
mvif->band_idx = ext_phy;
|
||||
if (mt7615_ext_phy(dev))
|
||||
mvif->wmm_idx = ext_phy * (MT7615_MAX_WMM_SETS / 2) +
|
||||
mvif->idx % (MT7615_MAX_WMM_SETS / 2);
|
||||
else
|
||||
mvif->wmm_idx = mvif->idx % MT7615_MAX_WMM_SETS;
|
||||
|
||||
ret = mt7615_mcu_set_dev_info(dev, vif, 1);
|
||||
if (ret)
|
||||
@ -99,18 +163,25 @@ static int mt7615_add_interface(struct ieee80211_hw *hw,
|
||||
|
||||
dev->vif_mask |= BIT(mvif->idx);
|
||||
dev->omac_mask |= BIT(mvif->omac_idx);
|
||||
phy->omac_mask |= BIT(mvif->omac_idx);
|
||||
|
||||
mt7615_mcu_set_dbdc(dev);
|
||||
|
||||
idx = MT7615_WTBL_RESERVED - mvif->idx;
|
||||
|
||||
INIT_LIST_HEAD(&mvif->sta.poll_list);
|
||||
mvif->sta.wcid.idx = idx;
|
||||
mvif->sta.wcid.ext_phy = mvif->band_idx;
|
||||
mvif->sta.wcid.hw_key_idx = -1;
|
||||
mt7615_mac_wtbl_update(dev, idx,
|
||||
MT_WTBL_UPDATE_ADM_COUNT_CLEAR);
|
||||
|
||||
rcu_assign_pointer(dev->mt76.wcid[idx], &mvif->sta.wcid);
|
||||
mtxq = (struct mt76_txq *)vif->txq->drv_priv;
|
||||
mtxq->wcid = &mvif->sta.wcid;
|
||||
mt76_txq_init(&dev->mt76, vif->txq);
|
||||
if (vif->txq) {
|
||||
mtxq = (struct mt76_txq *)vif->txq->drv_priv;
|
||||
mtxq->wcid = &mvif->sta.wcid;
|
||||
mt76_txq_init(&dev->mt76, vif->txq);
|
||||
}
|
||||
|
||||
out:
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
@ -123,7 +194,8 @@ static void mt7615_remove_interface(struct ieee80211_hw *hw,
|
||||
{
|
||||
struct mt7615_vif *mvif = (struct mt7615_vif *)vif->drv_priv;
|
||||
struct mt7615_sta *msta = &mvif->sta;
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
int idx = msta->wcid.idx;
|
||||
|
||||
/* TODO: disable beacon for the bss */
|
||||
@ -131,11 +203,13 @@ static void mt7615_remove_interface(struct ieee80211_hw *hw,
|
||||
mt7615_mcu_set_dev_info(dev, vif, 0);
|
||||
|
||||
rcu_assign_pointer(dev->mt76.wcid[idx], NULL);
|
||||
mt76_txq_remove(&dev->mt76, vif->txq);
|
||||
if (vif->txq)
|
||||
mt76_txq_remove(&dev->mt76, vif->txq);
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
dev->vif_mask &= ~BIT(mvif->idx);
|
||||
dev->omac_mask &= ~BIT(mvif->omac_idx);
|
||||
phy->omac_mask &= ~BIT(mvif->omac_idx);
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
spin_lock_bh(&dev->sta_poll_lock);
|
||||
@ -144,34 +218,38 @@ static void mt7615_remove_interface(struct ieee80211_hw *hw,
|
||||
spin_unlock_bh(&dev->sta_poll_lock);
|
||||
}
|
||||
|
||||
static int mt7615_set_channel(struct mt7615_dev *dev)
|
||||
static int mt7615_set_channel(struct mt7615_phy *phy)
|
||||
{
|
||||
struct mt7615_dev *dev = phy->dev;
|
||||
bool ext_phy = phy != &dev->phy;
|
||||
int ret;
|
||||
|
||||
cancel_delayed_work_sync(&dev->mt76.mac_work);
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
set_bit(MT76_RESET, &dev->mt76.state);
|
||||
set_bit(MT76_RESET, &phy->mt76->state);
|
||||
|
||||
mt7615_dfs_check_channel(dev);
|
||||
phy->dfs_state = -1;
|
||||
mt76_set_channel(phy->mt76);
|
||||
|
||||
mt76_set_channel(&dev->mt76);
|
||||
|
||||
ret = mt7615_mcu_set_channel(dev);
|
||||
ret = mt7615_mcu_set_chan_info(phy, MCU_EXT_CMD_CHANNEL_SWITCH);
|
||||
if (ret)
|
||||
goto out;
|
||||
|
||||
ret = mt7615_dfs_init_radar_detector(dev);
|
||||
mt7615_mac_cca_stats_reset(dev);
|
||||
dev->mt76.survey_time = ktime_get_boottime();
|
||||
mt7615_mac_set_timing(phy);
|
||||
ret = mt7615_dfs_init_radar_detector(phy);
|
||||
mt7615_mac_cca_stats_reset(phy);
|
||||
mt7615_mcu_set_sku_en(phy, true);
|
||||
|
||||
mt7615_mac_reset_counters(dev);
|
||||
phy->noise = 0;
|
||||
phy->chfreq = mt76_rr(dev, MT_CHFREQ(ext_phy));
|
||||
|
||||
out:
|
||||
clear_bit(MT76_RESET, &dev->mt76.state);
|
||||
clear_bit(MT76_RESET, &phy->mt76->state);
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
mt76_txq_schedule_all(&dev->mt76);
|
||||
mt76_txq_schedule_all(phy->mt76);
|
||||
ieee80211_queue_delayed_work(mt76_hw(dev), &dev->mt76.mac_work,
|
||||
MT7615_WATCHDOG_TIME);
|
||||
return ret;
|
||||
@ -181,7 +259,7 @@ static int mt7615_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
|
||||
struct ieee80211_vif *vif, struct ieee80211_sta *sta,
|
||||
struct ieee80211_key_conf *key)
|
||||
{
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt7615_vif *mvif = (struct mt7615_vif *)vif->drv_priv;
|
||||
struct mt7615_sta *msta = sta ? (struct mt7615_sta *)sta->drv_priv :
|
||||
&mvif->sta;
|
||||
@ -230,27 +308,27 @@ static int mt7615_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
|
||||
|
||||
static int mt7615_config(struct ieee80211_hw *hw, u32 changed)
|
||||
{
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
bool band = phy != &dev->phy;
|
||||
int ret = 0;
|
||||
|
||||
if (changed & IEEE80211_CONF_CHANGE_CHANNEL) {
|
||||
if (changed & (IEEE80211_CONF_CHANGE_CHANNEL |
|
||||
IEEE80211_CONF_CHANGE_POWER)) {
|
||||
ieee80211_stop_queues(hw);
|
||||
ret = mt7615_set_channel(dev);
|
||||
ret = mt7615_set_channel(phy);
|
||||
ieee80211_wake_queues(hw);
|
||||
}
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
|
||||
if (changed & IEEE80211_CONF_CHANGE_POWER)
|
||||
ret = mt7615_mcu_set_tx_power(dev);
|
||||
|
||||
if (changed & IEEE80211_CONF_CHANGE_MONITOR) {
|
||||
if (!(hw->conf.flags & IEEE80211_CONF_MONITOR))
|
||||
dev->mt76.rxfilter |= MT_WF_RFCR_DROP_OTHER_UC;
|
||||
phy->rxfilter |= MT_WF_RFCR_DROP_OTHER_UC;
|
||||
else
|
||||
dev->mt76.rxfilter &= ~MT_WF_RFCR_DROP_OTHER_UC;
|
||||
phy->rxfilter &= ~MT_WF_RFCR_DROP_OTHER_UC;
|
||||
|
||||
mt76_wr(dev, MT_WF_RFCR, dev->mt76.rxfilter);
|
||||
mt76_wr(dev, MT_WF_RFCR(band), phy->rxfilter);
|
||||
}
|
||||
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
@ -263,7 +341,7 @@ mt7615_conf_tx(struct ieee80211_hw *hw, struct ieee80211_vif *vif, u16 queue,
|
||||
const struct ieee80211_tx_queue_params *params)
|
||||
{
|
||||
struct mt7615_vif *mvif = (struct mt7615_vif *)vif->drv_priv;
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
|
||||
queue += mvif->wmm_idx * MT7615_MAX_WMM_SETS;
|
||||
|
||||
@ -275,7 +353,10 @@ static void mt7615_configure_filter(struct ieee80211_hw *hw,
|
||||
unsigned int *total_flags,
|
||||
u64 multicast)
|
||||
{
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
bool band = phy != &dev->phy;
|
||||
|
||||
u32 ctl_flags = MT_WF_RFCR1_DROP_ACK |
|
||||
MT_WF_RFCR1_DROP_BF_POLL |
|
||||
MT_WF_RFCR1_DROP_BA |
|
||||
@ -285,21 +366,21 @@ static void mt7615_configure_filter(struct ieee80211_hw *hw,
|
||||
|
||||
#define MT76_FILTER(_flag, _hw) do { \
|
||||
flags |= *total_flags & FIF_##_flag; \
|
||||
dev->mt76.rxfilter &= ~(_hw); \
|
||||
dev->mt76.rxfilter |= !(flags & FIF_##_flag) * (_hw); \
|
||||
phy->rxfilter &= ~(_hw); \
|
||||
phy->rxfilter |= !(flags & FIF_##_flag) * (_hw); \
|
||||
} while (0)
|
||||
|
||||
dev->mt76.rxfilter &= ~(MT_WF_RFCR_DROP_OTHER_BSS |
|
||||
MT_WF_RFCR_DROP_OTHER_BEACON |
|
||||
MT_WF_RFCR_DROP_FRAME_REPORT |
|
||||
MT_WF_RFCR_DROP_PROBEREQ |
|
||||
MT_WF_RFCR_DROP_MCAST_FILTERED |
|
||||
MT_WF_RFCR_DROP_MCAST |
|
||||
MT_WF_RFCR_DROP_BCAST |
|
||||
MT_WF_RFCR_DROP_DUPLICATE |
|
||||
MT_WF_RFCR_DROP_A2_BSSID |
|
||||
MT_WF_RFCR_DROP_UNWANTED_CTL |
|
||||
MT_WF_RFCR_DROP_STBC_MULTI);
|
||||
phy->rxfilter &= ~(MT_WF_RFCR_DROP_OTHER_BSS |
|
||||
MT_WF_RFCR_DROP_OTHER_BEACON |
|
||||
MT_WF_RFCR_DROP_FRAME_REPORT |
|
||||
MT_WF_RFCR_DROP_PROBEREQ |
|
||||
MT_WF_RFCR_DROP_MCAST_FILTERED |
|
||||
MT_WF_RFCR_DROP_MCAST |
|
||||
MT_WF_RFCR_DROP_BCAST |
|
||||
MT_WF_RFCR_DROP_DUPLICATE |
|
||||
MT_WF_RFCR_DROP_A2_BSSID |
|
||||
MT_WF_RFCR_DROP_UNWANTED_CTL |
|
||||
MT_WF_RFCR_DROP_STBC_MULTI);
|
||||
|
||||
MT76_FILTER(OTHER_BSS, MT_WF_RFCR_DROP_OTHER_TIM |
|
||||
MT_WF_RFCR_DROP_A3_MAC |
|
||||
@ -313,12 +394,12 @@ static void mt7615_configure_filter(struct ieee80211_hw *hw,
|
||||
MT_WF_RFCR_DROP_NDPA);
|
||||
|
||||
*total_flags = flags;
|
||||
mt76_wr(dev, MT_WF_RFCR, dev->mt76.rxfilter);
|
||||
mt76_wr(dev, MT_WF_RFCR(band), phy->rxfilter);
|
||||
|
||||
if (*total_flags & FIF_CONTROL)
|
||||
mt76_clear(dev, MT_WF_RFCR1, ctl_flags);
|
||||
mt76_clear(dev, MT_WF_RFCR1(band), ctl_flags);
|
||||
else
|
||||
mt76_set(dev, MT_WF_RFCR1, ctl_flags);
|
||||
mt76_set(dev, MT_WF_RFCR1(band), ctl_flags);
|
||||
}
|
||||
|
||||
static void mt7615_bss_info_changed(struct ieee80211_hw *hw,
|
||||
@ -326,24 +407,32 @@ static void mt7615_bss_info_changed(struct ieee80211_hw *hw,
|
||||
struct ieee80211_bss_conf *info,
|
||||
u32 changed)
|
||||
{
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
|
||||
if (changed & BSS_CHANGED_ASSOC)
|
||||
mt7615_mcu_set_bss_info(dev, vif, info->assoc);
|
||||
|
||||
/* TODO: update beacon content
|
||||
* BSS_CHANGED_BEACON
|
||||
*/
|
||||
if (changed & BSS_CHANGED_ERP_SLOT) {
|
||||
int slottime = info->use_short_slot ? 9 : 20;
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
|
||||
if (slottime != phy->slottime) {
|
||||
phy->slottime = slottime;
|
||||
mt7615_mac_set_timing(phy);
|
||||
}
|
||||
}
|
||||
|
||||
if (changed & BSS_CHANGED_BEACON_ENABLED) {
|
||||
mt7615_mcu_set_bss_info(dev, vif, info->enable_beacon);
|
||||
mt7615_mcu_wtbl_bmc(dev, vif, info->enable_beacon);
|
||||
mt7615_mcu_set_sta_rec_bmc(dev, vif, info->enable_beacon);
|
||||
mt7615_mcu_set_bcn(dev, vif, info->enable_beacon);
|
||||
mt7615_mcu_set_bmc(dev, vif, info->enable_beacon);
|
||||
}
|
||||
|
||||
if (changed & (BSS_CHANGED_BEACON |
|
||||
BSS_CHANGED_BEACON_ENABLED))
|
||||
mt7615_mcu_set_bcn(hw, vif, info->enable_beacon);
|
||||
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
}
|
||||
|
||||
@ -352,15 +441,15 @@ mt7615_channel_switch_beacon(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif,
|
||||
struct cfg80211_chan_def *chandef)
|
||||
{
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
mt7615_mcu_set_bcn(dev, vif, true);
|
||||
mt7615_mcu_set_bcn(hw, vif, true);
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
}
|
||||
|
||||
int mt7615_sta_add(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta)
|
||||
int mt7615_mac_sta_add(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta)
|
||||
{
|
||||
struct mt7615_dev *dev = container_of(mdev, struct mt7615_dev, mt76);
|
||||
struct mt7615_sta *msta = (struct mt7615_sta *)sta->drv_priv;
|
||||
@ -375,33 +464,23 @@ int mt7615_sta_add(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
msta->vif = mvif;
|
||||
msta->wcid.sta = 1;
|
||||
msta->wcid.idx = idx;
|
||||
msta->wcid.ext_phy = mvif->band_idx;
|
||||
|
||||
mt7615_mac_wtbl_update(dev, idx,
|
||||
MT_WTBL_UPDATE_ADM_COUNT_CLEAR);
|
||||
|
||||
mt7615_mcu_add_wtbl(dev, vif, sta);
|
||||
mt7615_mcu_set_sta_rec(dev, vif, sta, 1);
|
||||
mt7615_mcu_set_sta(dev, vif, sta, 1);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void mt7615_sta_assoc(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta)
|
||||
{
|
||||
struct mt7615_dev *dev = container_of(mdev, struct mt7615_dev, mt76);
|
||||
|
||||
if (sta->ht_cap.ht_supported)
|
||||
mt7615_mcu_set_ht_cap(dev, vif, sta);
|
||||
}
|
||||
|
||||
void mt7615_sta_remove(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta)
|
||||
void mt7615_mac_sta_remove(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta)
|
||||
{
|
||||
struct mt7615_dev *dev = container_of(mdev, struct mt7615_dev, mt76);
|
||||
struct mt7615_sta *msta = (struct mt7615_sta *)sta->drv_priv;
|
||||
|
||||
mt7615_mcu_set_sta_rec(dev, vif, sta, 0);
|
||||
mt7615_mcu_del_wtbl(dev, sta);
|
||||
|
||||
mt7615_mcu_set_sta(dev, vif, sta, 0);
|
||||
mt7615_mac_wtbl_update(dev, msta->wcid.idx,
|
||||
MT_WTBL_UPDATE_ADM_COUNT_CLEAR);
|
||||
|
||||
@ -415,7 +494,8 @@ static void mt7615_sta_rate_tbl_update(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta)
|
||||
{
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
struct mt7615_sta *msta = (struct mt7615_sta *)sta->drv_priv;
|
||||
struct ieee80211_sta_rates *sta_rates = rcu_dereference(sta->rates);
|
||||
int i;
|
||||
@ -430,7 +510,7 @@ static void mt7615_sta_rate_tbl_update(struct ieee80211_hw *hw,
|
||||
break;
|
||||
}
|
||||
msta->n_rates = i;
|
||||
mt7615_mac_set_rates(dev, msta, NULL, msta->rates);
|
||||
mt7615_mac_set_rates(phy, msta, NULL, msta->rates);
|
||||
msta->rate_probe = false;
|
||||
spin_unlock_bh(&dev->mt76.lock);
|
||||
}
|
||||
@ -439,7 +519,8 @@ static void mt7615_tx(struct ieee80211_hw *hw,
|
||||
struct ieee80211_tx_control *control,
|
||||
struct sk_buff *skb)
|
||||
{
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt76_phy *mphy = hw->priv;
|
||||
struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
|
||||
struct ieee80211_vif *vif = info->control.vif;
|
||||
struct mt76_wcid *wcid = &dev->mt76.global_wcid;
|
||||
@ -458,15 +539,16 @@ static void mt7615_tx(struct ieee80211_hw *hw,
|
||||
wcid = &mvif->sta.wcid;
|
||||
}
|
||||
|
||||
mt76_tx(&dev->mt76, control->sta, wcid, skb);
|
||||
mt76_tx(mphy, control->sta, wcid, skb);
|
||||
}
|
||||
|
||||
static int mt7615_set_rts_threshold(struct ieee80211_hw *hw, u32 val)
|
||||
{
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
mt7615_mcu_set_rts_thresh(dev, val);
|
||||
mt7615_mcu_set_rts_thresh(phy, val);
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
return 0;
|
||||
@ -477,7 +559,7 @@ mt7615_ampdu_action(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
struct ieee80211_ampdu_params *params)
|
||||
{
|
||||
enum ieee80211_ampdu_mlme_action action = params->action;
|
||||
struct mt7615_dev *dev = hw->priv;
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct ieee80211_sta *sta = params->sta;
|
||||
struct ieee80211_txq *txq = sta->txq[params->tid];
|
||||
struct mt7615_sta *msta = (struct mt7615_sta *)sta->drv_priv;
|
||||
@ -527,6 +609,92 @@ mt7615_ampdu_action(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int
|
||||
mt7615_sta_add(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta)
|
||||
{
|
||||
return mt76_sta_state(hw, vif, sta, IEEE80211_STA_NOTEXIST,
|
||||
IEEE80211_STA_NONE);
|
||||
}
|
||||
|
||||
static int
|
||||
mt7615_sta_remove(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta)
|
||||
{
|
||||
return mt76_sta_state(hw, vif, sta, IEEE80211_STA_NONE,
|
||||
IEEE80211_STA_NOTEXIST);
|
||||
}
|
||||
|
||||
static int
|
||||
mt7615_get_stats(struct ieee80211_hw *hw,
|
||||
struct ieee80211_low_level_stats *stats)
|
||||
{
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
struct mib_stats *mib = &phy->mib;
|
||||
|
||||
stats->dot11RTSSuccessCount = mib->rts_cnt;
|
||||
stats->dot11RTSFailureCount = mib->rts_retries_cnt;
|
||||
stats->dot11FCSErrorCount = mib->fcs_err_cnt;
|
||||
stats->dot11ACKFailureCount = mib->ack_fail_cnt;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static u64
|
||||
mt7615_get_tsf(struct ieee80211_hw *hw, struct ieee80211_vif *vif)
|
||||
{
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
union {
|
||||
u64 t64;
|
||||
u32 t32[2];
|
||||
} tsf;
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
|
||||
mt76_set(dev, MT_LPON_T0CR, MT_LPON_T0CR_MODE); /* TSF read */
|
||||
tsf.t32[0] = mt76_rr(dev, MT_LPON_UTTR0);
|
||||
tsf.t32[1] = mt76_rr(dev, MT_LPON_UTTR1);
|
||||
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
return tsf.t64;
|
||||
}
|
||||
|
||||
static void
|
||||
mt7615_set_coverage_class(struct ieee80211_hw *hw, s16 coverage_class)
|
||||
{
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
|
||||
phy->coverage_class = max_t(s16, coverage_class, 0);
|
||||
mt7615_mac_set_timing(phy);
|
||||
}
|
||||
|
||||
static int
|
||||
mt7615_set_antenna(struct ieee80211_hw *hw, u32 tx_ant, u32 rx_ant)
|
||||
{
|
||||
struct mt7615_dev *dev = mt7615_hw_dev(hw);
|
||||
struct mt7615_phy *phy = mt7615_hw_phy(hw);
|
||||
int max_nss = hweight8(hw->wiphy->available_antennas_tx);
|
||||
bool ext_phy = phy != &dev->phy;
|
||||
|
||||
if (!tx_ant || tx_ant != rx_ant || ffs(tx_ant) > max_nss)
|
||||
return -EINVAL;
|
||||
|
||||
if ((BIT(hweight8(tx_ant)) - 1) != tx_ant)
|
||||
tx_ant = BIT(ffs(tx_ant) - 1) - 1;
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
|
||||
phy->mt76->antenna_mask = tx_ant;
|
||||
phy->chainmask = ext_phy ? tx_ant << 2 : tx_ant;
|
||||
|
||||
mt76_set_stream_caps(&dev->mt76, true);
|
||||
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
const struct ieee80211_ops mt7615_ops = {
|
||||
.tx = mt7615_tx,
|
||||
.start = mt7615_start,
|
||||
@ -537,7 +705,9 @@ const struct ieee80211_ops mt7615_ops = {
|
||||
.conf_tx = mt7615_conf_tx,
|
||||
.configure_filter = mt7615_configure_filter,
|
||||
.bss_info_changed = mt7615_bss_info_changed,
|
||||
.sta_state = mt76_sta_state,
|
||||
.sta_add = mt7615_sta_add,
|
||||
.sta_remove = mt7615_sta_remove,
|
||||
.sta_pre_rcu_remove = mt76_sta_pre_rcu_remove,
|
||||
.set_key = mt7615_set_key,
|
||||
.ampdu_action = mt7615_ampdu_action,
|
||||
.set_rts_threshold = mt7615_set_rts_threshold,
|
||||
@ -548,6 +718,38 @@ const struct ieee80211_ops mt7615_ops = {
|
||||
.release_buffered_frames = mt76_release_buffered_frames,
|
||||
.get_txpower = mt76_get_txpower,
|
||||
.channel_switch_beacon = mt7615_channel_switch_beacon,
|
||||
.get_stats = mt7615_get_stats,
|
||||
.get_tsf = mt7615_get_tsf,
|
||||
.get_survey = mt76_get_survey,
|
||||
.get_antenna = mt76_get_antenna,
|
||||
.set_antenna = mt7615_set_antenna,
|
||||
.set_coverage_class = mt7615_set_coverage_class,
|
||||
};
|
||||
|
||||
static int __init mt7615_init(void)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = pci_register_driver(&mt7615_pci_driver);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (IS_ENABLED(CONFIG_MT7622_WMAC)) {
|
||||
ret = platform_driver_register(&mt7622_wmac_driver);
|
||||
if (ret)
|
||||
pci_unregister_driver(&mt7615_pci_driver);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void __exit mt7615_exit(void)
|
||||
{
|
||||
if (IS_ENABLED(CONFIG_MT7622_WMAC))
|
||||
platform_driver_unregister(&mt7622_wmac_driver);
|
||||
pci_unregister_driver(&mt7615_pci_driver);
|
||||
}
|
||||
|
||||
module_init(mt7615_init);
|
||||
module_exit(mt7615_exit);
|
||||
MODULE_LICENSE("Dual BSD/GPL");
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -45,6 +45,62 @@ enum {
|
||||
MCU_EXT_EVENT_CSA_NOTIFY = 0x4f,
|
||||
};
|
||||
|
||||
enum {
|
||||
MT_SKU_CCK_1_2 = 0,
|
||||
MT_SKU_CCK_55_11,
|
||||
MT_SKU_OFDM_6_9,
|
||||
MT_SKU_OFDM_12_18,
|
||||
MT_SKU_OFDM_24_36,
|
||||
MT_SKU_OFDM_48,
|
||||
MT_SKU_OFDM_54,
|
||||
MT_SKU_HT20_0_8,
|
||||
MT_SKU_HT20_32,
|
||||
MT_SKU_HT20_1_2_9_10,
|
||||
MT_SKU_HT20_3_4_11_12,
|
||||
MT_SKU_HT20_5_13,
|
||||
MT_SKU_HT20_6_14,
|
||||
MT_SKU_HT20_7_15,
|
||||
MT_SKU_HT40_0_8,
|
||||
MT_SKU_HT40_32,
|
||||
MT_SKU_HT40_1_2_9_10,
|
||||
MT_SKU_HT40_3_4_11_12,
|
||||
MT_SKU_HT40_5_13,
|
||||
MT_SKU_HT40_6_14,
|
||||
MT_SKU_HT40_7_15,
|
||||
MT_SKU_VHT20_0,
|
||||
MT_SKU_VHT20_1_2,
|
||||
MT_SKU_VHT20_3_4,
|
||||
MT_SKU_VHT20_5_6,
|
||||
MT_SKU_VHT20_7,
|
||||
MT_SKU_VHT20_8,
|
||||
MT_SKU_VHT20_9,
|
||||
MT_SKU_VHT40_0,
|
||||
MT_SKU_VHT40_1_2,
|
||||
MT_SKU_VHT40_3_4,
|
||||
MT_SKU_VHT40_5_6,
|
||||
MT_SKU_VHT40_7,
|
||||
MT_SKU_VHT40_8,
|
||||
MT_SKU_VHT40_9,
|
||||
MT_SKU_VHT80_0,
|
||||
MT_SKU_VHT80_1_2,
|
||||
MT_SKU_VHT80_3_4,
|
||||
MT_SKU_VHT80_5_6,
|
||||
MT_SKU_VHT80_7,
|
||||
MT_SKU_VHT80_8,
|
||||
MT_SKU_VHT80_9,
|
||||
MT_SKU_VHT160_0,
|
||||
MT_SKU_VHT160_1_2,
|
||||
MT_SKU_VHT160_3_4,
|
||||
MT_SKU_VHT160_5_6,
|
||||
MT_SKU_VHT160_7,
|
||||
MT_SKU_VHT160_8,
|
||||
MT_SKU_VHT160_9,
|
||||
MT_SKU_1SS_DELTA,
|
||||
MT_SKU_2SS_DELTA,
|
||||
MT_SKU_3SS_DELTA,
|
||||
MT_SKU_4SS_DELTA,
|
||||
};
|
||||
|
||||
struct mt7615_mcu_rxd {
|
||||
__le32 rxd[4];
|
||||
|
||||
@ -60,6 +116,52 @@ struct mt7615_mcu_rxd {
|
||||
u8 s2d_index;
|
||||
};
|
||||
|
||||
struct mt7615_mcu_rdd_report {
|
||||
struct mt7615_mcu_rxd rxd;
|
||||
|
||||
u8 idx;
|
||||
u8 long_detected;
|
||||
u8 constant_prf_detected;
|
||||
u8 staggered_prf_detected;
|
||||
u8 radar_type_idx;
|
||||
u8 periodic_pulse_num;
|
||||
u8 long_pulse_num;
|
||||
u8 hw_pulse_num;
|
||||
|
||||
u8 out_lpn;
|
||||
u8 out_spn;
|
||||
u8 out_crpn;
|
||||
u8 out_crpw;
|
||||
u8 out_crbn;
|
||||
u8 out_stgpn;
|
||||
u8 out_stgpw;
|
||||
|
||||
u8 _rsv[2];
|
||||
|
||||
__le32 out_pri_const;
|
||||
__le32 out_pri_stg[3];
|
||||
|
||||
struct {
|
||||
__le32 start;
|
||||
__le16 pulse_width;
|
||||
__le16 pulse_power;
|
||||
} long_pulse[32];
|
||||
|
||||
struct {
|
||||
__le32 start;
|
||||
__le16 pulse_width;
|
||||
__le16 pulse_power;
|
||||
} periodic_pulse[32];
|
||||
|
||||
struct {
|
||||
__le32 start;
|
||||
__le16 pulse_width;
|
||||
__le16 pulse_power;
|
||||
u8 sc_pass;
|
||||
u8 sw_reset;
|
||||
} hw_pulse[32];
|
||||
};
|
||||
|
||||
#define MCU_PQ_ID(p, q) (((p) << 15) | ((q) << 10))
|
||||
#define MCU_PKT_ID 0xa0
|
||||
|
||||
@ -93,6 +195,7 @@ enum {
|
||||
MCU_EXT_CMD_PM_STATE_CTRL = 0x07,
|
||||
MCU_EXT_CMD_CHANNEL_SWITCH = 0x08,
|
||||
MCU_EXT_CMD_SET_TX_POWER_CTRL = 0x11,
|
||||
MCU_EXT_CMD_FW_LOG_2_HOST = 0x13,
|
||||
MCU_EXT_CMD_EFUSE_BUFFER_MODE = 0x21,
|
||||
MCU_EXT_CMD_STA_REC_UPDATE = 0x25,
|
||||
MCU_EXT_CMD_BSS_INFO_UPDATE = 0x26,
|
||||
@ -102,9 +205,12 @@ enum {
|
||||
MCU_EXT_CMD_WTBL_UPDATE = 0x32,
|
||||
MCU_EXT_CMD_SET_RDD_CTRL = 0x3a,
|
||||
MCU_EXT_CMD_PROTECT_CTRL = 0x3e,
|
||||
MCU_EXT_CMD_DBDC_CTRL = 0x45,
|
||||
MCU_EXT_CMD_MAC_INIT_CTRL = 0x46,
|
||||
MCU_EXT_CMD_BCN_OFFLOAD = 0x49,
|
||||
MCU_EXT_CMD_SET_RX_PATH = 0x4e,
|
||||
MCU_EXT_CMD_TX_POWER_FEATURE_CTRL = 0x58,
|
||||
MCU_EXT_CMD_SET_RDD_TH = 0x7c,
|
||||
MCU_EXT_CMD_SET_RDD_PATTERN = 0x7d,
|
||||
};
|
||||
|
||||
@ -156,6 +262,18 @@ enum {
|
||||
DEV_INFO_MAX_NUM
|
||||
};
|
||||
|
||||
enum {
|
||||
DBDC_TYPE_WMM,
|
||||
DBDC_TYPE_MGMT,
|
||||
DBDC_TYPE_BSS,
|
||||
DBDC_TYPE_MBSS,
|
||||
DBDC_TYPE_REPEATER,
|
||||
DBDC_TYPE_MU,
|
||||
DBDC_TYPE_BF,
|
||||
DBDC_TYPE_PTA,
|
||||
__DBDC_TYPE_MAX,
|
||||
};
|
||||
|
||||
struct bss_info_omac {
|
||||
__le16 tag;
|
||||
__le16 len;
|
||||
@ -447,9 +565,10 @@ struct sta_rec_ba {
|
||||
__le16 winsize;
|
||||
} __packed;
|
||||
|
||||
#define MT7615_STA_REC_UPDATE_MAX_SIZE (sizeof(struct sta_rec_basic) + \
|
||||
sizeof(struct sta_rec_ht) + \
|
||||
sizeof(struct sta_rec_vht))
|
||||
struct sta_rec_wtbl {
|
||||
__le16 tag;
|
||||
__le16 len;
|
||||
} __packed;
|
||||
|
||||
enum {
|
||||
STA_REC_BASIC,
|
||||
@ -464,6 +583,7 @@ enum {
|
||||
STA_REC_HT,
|
||||
STA_REC_VHT,
|
||||
STA_REC_APPS,
|
||||
STA_REC_WTBL = 13,
|
||||
STA_REC_MAX_NUM
|
||||
};
|
||||
|
||||
|
115
drivers/net/wireless/mediatek/mt76/mt7615/mmio.c
Normal file
115
drivers/net/wireless/mediatek/mt76/mt7615/mmio.c
Normal file
@ -0,0 +1,115 @@
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/module.h>
|
||||
|
||||
#include "mt7615.h"
|
||||
#include "mac.h"
|
||||
#include "../trace.h"
|
||||
|
||||
u32 mt7615_reg_map(struct mt7615_dev *dev, u32 addr)
|
||||
{
|
||||
u32 base = addr & MT_MCU_PCIE_REMAP_2_BASE;
|
||||
u32 offset = addr & MT_MCU_PCIE_REMAP_2_OFFSET;
|
||||
|
||||
mt76_wr(dev, MT_MCU_PCIE_REMAP_2, base);
|
||||
|
||||
return MT_PCIE_REMAP_BASE_2 + offset;
|
||||
}
|
||||
|
||||
static void
|
||||
mt7615_rx_poll_complete(struct mt76_dev *mdev, enum mt76_rxq_id q)
|
||||
{
|
||||
struct mt7615_dev *dev = container_of(mdev, struct mt7615_dev, mt76);
|
||||
|
||||
mt7615_irq_enable(dev, MT_INT_RX_DONE(q));
|
||||
}
|
||||
|
||||
static irqreturn_t mt7615_irq_handler(int irq, void *dev_instance)
|
||||
{
|
||||
struct mt7615_dev *dev = dev_instance;
|
||||
u32 intr;
|
||||
|
||||
intr = mt76_rr(dev, MT_INT_SOURCE_CSR);
|
||||
mt76_wr(dev, MT_INT_SOURCE_CSR, intr);
|
||||
|
||||
if (!test_bit(MT76_STATE_INITIALIZED, &dev->mphy.state))
|
||||
return IRQ_NONE;
|
||||
|
||||
trace_dev_irq(&dev->mt76, intr, dev->mt76.mmio.irqmask);
|
||||
|
||||
intr &= dev->mt76.mmio.irqmask;
|
||||
|
||||
if (intr & MT_INT_TX_DONE_ALL) {
|
||||
mt7615_irq_disable(dev, MT_INT_TX_DONE_ALL);
|
||||
napi_schedule(&dev->mt76.tx_napi);
|
||||
}
|
||||
|
||||
if (intr & MT_INT_RX_DONE(0)) {
|
||||
mt7615_irq_disable(dev, MT_INT_RX_DONE(0));
|
||||
napi_schedule(&dev->mt76.napi[0]);
|
||||
}
|
||||
|
||||
if (intr & MT_INT_RX_DONE(1)) {
|
||||
mt7615_irq_disable(dev, MT_INT_RX_DONE(1));
|
||||
napi_schedule(&dev->mt76.napi[1]);
|
||||
}
|
||||
|
||||
if (intr & MT_INT_MCU_CMD) {
|
||||
u32 val = mt76_rr(dev, MT_MCU_CMD);
|
||||
|
||||
if (val & MT_MCU_CMD_ERROR_MASK) {
|
||||
dev->reset_state = val;
|
||||
ieee80211_queue_work(mt76_hw(dev), &dev->reset_work);
|
||||
wake_up(&dev->reset_wait);
|
||||
}
|
||||
}
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
int mt7615_mmio_probe(struct device *pdev, void __iomem *mem_base, int irq)
|
||||
{
|
||||
static const struct mt76_driver_ops drv_ops = {
|
||||
/* txwi_size = txd size + txp size */
|
||||
.txwi_size = MT_TXD_SIZE + sizeof(struct mt7615_txp_common),
|
||||
.drv_flags = MT_DRV_TXWI_NO_FREE,
|
||||
.survey_flags = SURVEY_INFO_TIME_TX |
|
||||
SURVEY_INFO_TIME_RX |
|
||||
SURVEY_INFO_TIME_BSS_RX,
|
||||
.tx_prepare_skb = mt7615_tx_prepare_skb,
|
||||
.tx_complete_skb = mt7615_tx_complete_skb,
|
||||
.rx_skb = mt7615_queue_rx_skb,
|
||||
.rx_poll_complete = mt7615_rx_poll_complete,
|
||||
.sta_ps = mt7615_sta_ps,
|
||||
.sta_add = mt7615_mac_sta_add,
|
||||
.sta_remove = mt7615_mac_sta_remove,
|
||||
.update_survey = mt7615_update_channel,
|
||||
};
|
||||
struct mt7615_dev *dev;
|
||||
struct mt76_dev *mdev;
|
||||
int ret;
|
||||
|
||||
mdev = mt76_alloc_device(pdev, sizeof(*dev), &mt7615_ops, &drv_ops);
|
||||
if (!mdev)
|
||||
return -ENOMEM;
|
||||
|
||||
dev = container_of(mdev, struct mt7615_dev, mt76);
|
||||
mt76_mmio_init(&dev->mt76, mem_base);
|
||||
|
||||
mdev->rev = (mt76_rr(dev, MT_HW_CHIPID) << 16) |
|
||||
(mt76_rr(dev, MT_HW_REV) & 0xff);
|
||||
dev_dbg(mdev->dev, "ASIC revision: %04x\n", mdev->rev);
|
||||
|
||||
ret = devm_request_irq(mdev->dev, irq, mt7615_irq_handler,
|
||||
IRQF_SHARED, KBUILD_MODNAME, dev);
|
||||
if (ret)
|
||||
goto error;
|
||||
|
||||
ret = mt7615_register_device(dev);
|
||||
if (ret)
|
||||
goto error;
|
||||
|
||||
return 0;
|
||||
error:
|
||||
ieee80211_free_hw(mt76_hw(dev));
|
||||
return ret;
|
||||
}
|
@ -6,6 +6,7 @@
|
||||
|
||||
#include <linux/interrupt.h>
|
||||
#include <linux/ktime.h>
|
||||
#include <linux/regmap.h>
|
||||
#include "../mt76.h"
|
||||
#include "regs.h"
|
||||
|
||||
@ -17,9 +18,11 @@
|
||||
MT7615_MAX_INTERFACES)
|
||||
|
||||
#define MT7615_WATCHDOG_TIME (HZ / 10)
|
||||
#define MT7615_RESET_TIMEOUT (30 * HZ)
|
||||
#define MT7615_RATE_RETRY 2
|
||||
|
||||
#define MT7615_TX_RING_SIZE 1024
|
||||
#define MT7615_TX_MGMT_RING_SIZE 128
|
||||
#define MT7615_TX_MCU_RING_SIZE 128
|
||||
#define MT7615_TX_FWDL_RING_SIZE 128
|
||||
|
||||
@ -30,14 +33,30 @@
|
||||
#define MT7615_FIRMWARE_N9 "mediatek/mt7615_n9.bin"
|
||||
#define MT7615_ROM_PATCH "mediatek/mt7615_rom_patch.bin"
|
||||
|
||||
#define MT7622_FIRMWARE_N9 "mediatek/mt7622_n9.bin"
|
||||
#define MT7622_ROM_PATCH "mediatek/mt7622_rom_patch.bin"
|
||||
|
||||
#define MT7615_FIRMWARE_V1 1
|
||||
#define MT7615_FIRMWARE_V2 2
|
||||
|
||||
#define MT7615_EEPROM_SIZE 1024
|
||||
#define MT7615_TOKEN_SIZE 4096
|
||||
|
||||
#define MT_FRAC_SCALE 12
|
||||
#define MT_FRAC(val, div) (((val) << MT_FRAC_SCALE) / (div))
|
||||
|
||||
#define MT_CHFREQ_VALID BIT(7)
|
||||
#define MT_CHFREQ_DBDC_IDX BIT(6)
|
||||
#define MT_CHFREQ_SEQ GENMASK(5, 0)
|
||||
|
||||
#define MT7615_BAR_RATE_DEFAULT 0x4b /* OFDM 6M */
|
||||
#define MT7615_CFEND_RATE_DEFAULT 0x49 /* OFDM 24M */
|
||||
#define MT7615_CFEND_RATE_11B 0x03 /* 11B LP, 11M */
|
||||
|
||||
struct mt7615_vif;
|
||||
struct mt7615_sta;
|
||||
struct mt7615_dfs_pulse;
|
||||
struct mt7615_dfs_pattern;
|
||||
|
||||
enum mt7615_hw_txq_id {
|
||||
MT7615_TXQ_MAIN,
|
||||
@ -46,6 +65,16 @@ enum mt7615_hw_txq_id {
|
||||
MT7615_TXQ_FWDL,
|
||||
};
|
||||
|
||||
enum mt7622_hw_txq_id {
|
||||
MT7622_TXQ_AC0,
|
||||
MT7622_TXQ_AC1,
|
||||
MT7622_TXQ_AC2,
|
||||
MT7622_TXQ_FWDL = MT7615_TXQ_FWDL,
|
||||
MT7622_TXQ_AC3,
|
||||
MT7622_TXQ_MGMT,
|
||||
MT7622_TXQ_MCU = 15,
|
||||
};
|
||||
|
||||
struct mt7615_rate_set {
|
||||
struct ieee80211_tx_rate probe_rate;
|
||||
struct ieee80211_tx_rate rates[4];
|
||||
@ -79,12 +108,61 @@ struct mt7615_vif {
|
||||
struct mt7615_sta sta;
|
||||
};
|
||||
|
||||
struct mib_stats {
|
||||
u32 ack_fail_cnt;
|
||||
u32 fcs_err_cnt;
|
||||
u32 rts_cnt;
|
||||
u32 rts_retries_cnt;
|
||||
};
|
||||
|
||||
struct mt7615_phy {
|
||||
struct mt76_phy *mt76;
|
||||
struct mt7615_dev *dev;
|
||||
|
||||
u32 rxfilter;
|
||||
u32 omac_mask;
|
||||
|
||||
u16 noise;
|
||||
|
||||
unsigned long last_cca_adj;
|
||||
int false_cca_ofdm, false_cca_cck;
|
||||
s8 ofdm_sensitivity;
|
||||
s8 cck_sensitivity;
|
||||
|
||||
u16 chainmask;
|
||||
|
||||
s16 coverage_class;
|
||||
u8 slottime;
|
||||
|
||||
u8 chfreq;
|
||||
u8 rdd_state;
|
||||
int dfs_state;
|
||||
|
||||
__le32 rx_ampdu_ts;
|
||||
u32 ampdu_ref;
|
||||
|
||||
struct mib_stats mib;
|
||||
};
|
||||
|
||||
struct mt7615_dev {
|
||||
struct mt76_dev mt76; /* must be first */
|
||||
union { /* must be first */
|
||||
struct mt76_dev mt76;
|
||||
struct mt76_phy mphy;
|
||||
};
|
||||
|
||||
struct mt7615_phy phy;
|
||||
u32 vif_mask;
|
||||
u32 omac_mask;
|
||||
|
||||
__le32 rx_ampdu_ts;
|
||||
u16 chainmask;
|
||||
|
||||
struct regmap *infracfg;
|
||||
|
||||
struct work_struct mcu_work;
|
||||
|
||||
struct work_struct reset_work;
|
||||
wait_queue_head_t reset_wait;
|
||||
u32 reset_state;
|
||||
|
||||
struct list_head sta_poll_list;
|
||||
spinlock_t sta_poll_lock;
|
||||
@ -96,17 +174,15 @@ struct mt7615_dev {
|
||||
s16 power;
|
||||
} radar_pattern;
|
||||
u32 hw_pattern;
|
||||
int dfs_state;
|
||||
|
||||
int false_cca_ofdm, false_cca_cck;
|
||||
unsigned long last_cca_adj;
|
||||
u8 mac_work_count;
|
||||
s8 ofdm_sensitivity;
|
||||
s8 cck_sensitivity;
|
||||
bool scs_en;
|
||||
bool fw_debug;
|
||||
|
||||
spinlock_t token_lock;
|
||||
struct idr token;
|
||||
|
||||
u8 fw_ver;
|
||||
};
|
||||
|
||||
enum {
|
||||
@ -134,11 +210,6 @@ enum {
|
||||
EXT_BSSID_END
|
||||
};
|
||||
|
||||
enum {
|
||||
MT_HW_RDD0,
|
||||
MT_HW_RDD1,
|
||||
};
|
||||
|
||||
enum {
|
||||
MT_RX_SEL0,
|
||||
MT_RX_SEL1,
|
||||
@ -158,13 +229,53 @@ enum mt7615_rdd_cmd {
|
||||
RDD_RESUME_BF,
|
||||
};
|
||||
|
||||
static inline struct mt7615_phy *
|
||||
mt7615_hw_phy(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
|
||||
return phy->priv;
|
||||
}
|
||||
|
||||
static inline struct mt7615_dev *
|
||||
mt7615_hw_dev(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
|
||||
return container_of(phy->dev, struct mt7615_dev, mt76);
|
||||
}
|
||||
|
||||
static inline struct mt7615_phy *
|
||||
mt7615_ext_phy(struct mt7615_dev *dev)
|
||||
{
|
||||
struct mt76_phy *phy = dev->mt76.phy2;
|
||||
|
||||
if (!phy)
|
||||
return NULL;
|
||||
|
||||
return phy->priv;
|
||||
}
|
||||
|
||||
extern const struct ieee80211_ops mt7615_ops;
|
||||
extern struct pci_driver mt7615_pci_driver;
|
||||
extern struct platform_driver mt7622_wmac_driver;
|
||||
|
||||
#ifdef CONFIG_MT7622_WMAC
|
||||
int mt7622_wmac_init(struct mt7615_dev *dev);
|
||||
#else
|
||||
static inline int mt7622_wmac_init(struct mt7615_dev *dev)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
int mt7615_mmio_probe(struct device *pdev, void __iomem *mem_base, int irq);
|
||||
u32 mt7615_reg_map(struct mt7615_dev *dev, u32 addr);
|
||||
|
||||
int mt7615_register_device(struct mt7615_dev *dev);
|
||||
void mt7615_unregister_device(struct mt7615_dev *dev);
|
||||
int mt7615_register_ext_phy(struct mt7615_dev *dev);
|
||||
void mt7615_unregister_ext_phy(struct mt7615_dev *dev);
|
||||
int mt7615_eeprom_init(struct mt7615_dev *dev);
|
||||
int mt7615_eeprom_get_power_index(struct mt7615_dev *dev,
|
||||
struct ieee80211_channel *chan,
|
||||
@ -172,26 +283,22 @@ int mt7615_eeprom_get_power_index(struct mt7615_dev *dev,
|
||||
int mt7615_dma_init(struct mt7615_dev *dev);
|
||||
void mt7615_dma_cleanup(struct mt7615_dev *dev);
|
||||
int mt7615_mcu_init(struct mt7615_dev *dev);
|
||||
bool mt7615_wait_for_mcu_init(struct mt7615_dev *dev);
|
||||
int mt7615_mcu_set_dev_info(struct mt7615_dev *dev,
|
||||
struct ieee80211_vif *vif, bool enable);
|
||||
int mt7615_mcu_set_bss_info(struct mt7615_dev *dev, struct ieee80211_vif *vif,
|
||||
int en);
|
||||
void mt7615_mac_set_rates(struct mt7615_dev *dev, struct mt7615_sta *sta,
|
||||
void mt7615_mac_set_rates(struct mt7615_phy *phy, struct mt7615_sta *sta,
|
||||
struct ieee80211_tx_rate *probe_rate,
|
||||
struct ieee80211_tx_rate *rates);
|
||||
int mt7615_mcu_wtbl_bmc(struct mt7615_dev *dev, struct ieee80211_vif *vif,
|
||||
bool enable);
|
||||
int mt7615_mcu_add_wtbl(struct mt7615_dev *dev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta);
|
||||
int mt7615_mcu_del_wtbl(struct mt7615_dev *dev, struct ieee80211_sta *sta);
|
||||
int mt7615_mcu_del_wtbl_all(struct mt7615_dev *dev);
|
||||
int mt7615_mcu_set_sta_rec_bmc(struct mt7615_dev *dev,
|
||||
struct ieee80211_vif *vif, bool en);
|
||||
int mt7615_mcu_set_sta_rec(struct mt7615_dev *dev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta, bool en);
|
||||
int mt7615_mcu_set_bcn(struct mt7615_dev *dev, struct ieee80211_vif *vif,
|
||||
int mt7615_mcu_set_bmc(struct mt7615_dev *dev, struct ieee80211_vif *vif,
|
||||
bool en);
|
||||
int mt7615_mcu_set_sta(struct mt7615_dev *dev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta, bool en);
|
||||
int mt7615_mcu_set_bcn(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
int en);
|
||||
int mt7615_mcu_set_channel(struct mt7615_dev *dev);
|
||||
int mt7615_mcu_set_chan_info(struct mt7615_phy *phy, int cmd);
|
||||
int mt7615_mcu_set_wmm(struct mt7615_dev *dev, u8 queue,
|
||||
const struct ieee80211_tx_queue_params *params);
|
||||
int mt7615_mcu_set_tx_ba(struct mt7615_dev *dev,
|
||||
@ -200,30 +307,24 @@ int mt7615_mcu_set_tx_ba(struct mt7615_dev *dev,
|
||||
int mt7615_mcu_set_rx_ba(struct mt7615_dev *dev,
|
||||
struct ieee80211_ampdu_params *params,
|
||||
bool add);
|
||||
int mt7615_mcu_set_ht_cap(struct mt7615_dev *dev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta);
|
||||
void mt7615_mcu_rx_event(struct mt7615_dev *dev, struct sk_buff *skb);
|
||||
int mt7615_mcu_rdd_cmd(struct mt7615_dev *dev,
|
||||
enum mt7615_rdd_cmd cmd, u8 index,
|
||||
u8 rx_sel, u8 val);
|
||||
int mt7615_dfs_start_radar_detector(struct mt7615_dev *dev);
|
||||
int mt7615_dfs_stop_radar_detector(struct mt7615_dev *dev);
|
||||
int mt7615_mcu_rdd_send_pattern(struct mt7615_dev *dev);
|
||||
int mt7615_mcu_fw_log_2_host(struct mt7615_dev *dev, u8 ctrl);
|
||||
|
||||
static inline bool is_mt7622(struct mt76_dev *dev)
|
||||
{
|
||||
if (!IS_ENABLED(CONFIG_MT7622_WMAC))
|
||||
return false;
|
||||
|
||||
return mt76_chip(dev) == 0x7622;
|
||||
}
|
||||
|
||||
static inline void mt7615_dfs_check_channel(struct mt7615_dev *dev)
|
||||
static inline bool is_mt7615(struct mt76_dev *dev)
|
||||
{
|
||||
enum nl80211_chan_width width = dev->mt76.chandef.width;
|
||||
u32 freq = dev->mt76.chandef.chan->center_freq;
|
||||
struct ieee80211_hw *hw = mt76_hw(dev);
|
||||
|
||||
if (hw->conf.chandef.chan->center_freq != freq ||
|
||||
hw->conf.chandef.width != width)
|
||||
dev->dfs_state = -1;
|
||||
return mt76_chip(dev) == 0x7615;
|
||||
}
|
||||
|
||||
static inline void mt7615_irq_enable(struct mt7615_dev *dev, u32 mask)
|
||||
@ -239,26 +340,29 @@ static inline void mt7615_irq_disable(struct mt7615_dev *dev, u32 mask)
|
||||
void mt7615_update_channel(struct mt76_dev *mdev);
|
||||
bool mt7615_mac_wtbl_update(struct mt7615_dev *dev, int idx, u32 mask);
|
||||
void mt7615_mac_reset_counters(struct mt7615_dev *dev);
|
||||
void mt7615_mac_cca_stats_reset(struct mt7615_dev *dev);
|
||||
void mt7615_mac_cca_stats_reset(struct mt7615_phy *phy);
|
||||
void mt7615_mac_set_scs(struct mt7615_dev *dev, bool enable);
|
||||
void mt7615_mac_enable_nf(struct mt7615_dev *dev, bool ext_phy);
|
||||
void mt7615_mac_sta_poll(struct mt7615_dev *dev);
|
||||
int mt7615_mac_write_txwi(struct mt7615_dev *dev, __le32 *txwi,
|
||||
struct sk_buff *skb, struct mt76_wcid *wcid,
|
||||
struct ieee80211_sta *sta, int pid,
|
||||
struct ieee80211_key_conf *key);
|
||||
void mt7615_mac_set_timing(struct mt7615_phy *phy);
|
||||
int mt7615_mac_fill_rx(struct mt7615_dev *dev, struct sk_buff *skb);
|
||||
void mt7615_mac_add_txs(struct mt7615_dev *dev, void *data);
|
||||
void mt7615_mac_tx_free(struct mt7615_dev *dev, struct sk_buff *skb);
|
||||
int mt7615_mac_wtbl_set_key(struct mt7615_dev *dev, struct mt76_wcid *wcid,
|
||||
struct ieee80211_key_conf *key,
|
||||
enum set_key_cmd cmd);
|
||||
void mt7615_mac_reset_work(struct work_struct *work);
|
||||
|
||||
int mt7615_mcu_set_dbdc(struct mt7615_dev *dev);
|
||||
int mt7615_mcu_set_eeprom(struct mt7615_dev *dev);
|
||||
int mt7615_mcu_init_mac(struct mt7615_dev *dev);
|
||||
int mt7615_mcu_set_rts_thresh(struct mt7615_dev *dev, u32 val);
|
||||
int mt7615_mcu_ctrl_pm_state(struct mt7615_dev *dev, int enter);
|
||||
int mt7615_mcu_set_mac_enable(struct mt7615_dev *dev, int band, bool enable);
|
||||
int mt7615_mcu_set_rts_thresh(struct mt7615_phy *phy, u32 val);
|
||||
int mt7615_mcu_ctrl_pm_state(struct mt7615_dev *dev, int band, int enter);
|
||||
int mt7615_mcu_get_temperature(struct mt7615_dev *dev, int index);
|
||||
int mt7615_mcu_set_tx_power(struct mt7615_dev *dev);
|
||||
void mt7615_mcu_exit(struct mt7615_dev *dev);
|
||||
|
||||
int mt7615_tx_prepare_skb(struct mt76_dev *mdev, void *txwi_ptr,
|
||||
@ -272,17 +376,20 @@ void mt7615_tx_complete_skb(struct mt76_dev *mdev, enum mt76_txq_id qid,
|
||||
void mt7615_queue_rx_skb(struct mt76_dev *mdev, enum mt76_rxq_id q,
|
||||
struct sk_buff *skb);
|
||||
void mt7615_sta_ps(struct mt76_dev *mdev, struct ieee80211_sta *sta, bool ps);
|
||||
int mt7615_sta_add(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta);
|
||||
void mt7615_sta_assoc(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta);
|
||||
void mt7615_sta_remove(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
int mt7615_mac_sta_add(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta);
|
||||
void mt7615_mac_sta_remove(struct mt76_dev *mdev, struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta);
|
||||
void mt7615_mac_work(struct work_struct *work);
|
||||
void mt7615_txp_skb_unmap(struct mt76_dev *dev,
|
||||
struct mt76_txwi_cache *txwi);
|
||||
int mt76_dfs_start_rdd(struct mt7615_dev *dev, bool force);
|
||||
int mt7615_dfs_init_radar_detector(struct mt7615_dev *dev);
|
||||
int mt7615_mcu_set_fcc5_lpn(struct mt7615_dev *dev, int val);
|
||||
int mt7615_mcu_set_pulse_th(struct mt7615_dev *dev,
|
||||
const struct mt7615_dfs_pulse *pulse);
|
||||
int mt7615_mcu_set_radar_th(struct mt7615_dev *dev, int index,
|
||||
const struct mt7615_dfs_pattern *pattern);
|
||||
int mt7615_mcu_set_sku_en(struct mt7615_phy *phy, bool enable);
|
||||
int mt7615_dfs_init_radar_detector(struct mt7615_phy *phy);
|
||||
|
||||
int mt7615_init_debugfs(struct mt7615_dev *dev);
|
||||
|
||||
|
56
drivers/net/wireless/mediatek/mt76/mt7615/mt7615_trace.h
Normal file
56
drivers/net/wireless/mediatek/mt76/mt7615/mt7615_trace.h
Normal file
@ -0,0 +1,56 @@
|
||||
/* SPDX-License-Identifier: ISC */
|
||||
/*
|
||||
* Copyright (C) 2019 Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
*/
|
||||
|
||||
#if !defined(__MT7615_TRACE_H) || defined(TRACE_HEADER_MULTI_READ)
|
||||
#define __MT7615_TRACE_H
|
||||
|
||||
#include <linux/tracepoint.h>
|
||||
#include "mt7615.h"
|
||||
|
||||
#undef TRACE_SYSTEM
|
||||
#define TRACE_SYSTEM mt7615
|
||||
|
||||
#define MAXNAME 32
|
||||
#define DEV_ENTRY __array(char, wiphy_name, 32)
|
||||
#define DEV_ASSIGN strlcpy(__entry->wiphy_name, \
|
||||
wiphy_name(mt76_hw(dev)->wiphy), MAXNAME)
|
||||
#define DEV_PR_FMT "%s"
|
||||
#define DEV_PR_ARG __entry->wiphy_name
|
||||
|
||||
#define TOKEN_ENTRY __field(u16, token)
|
||||
#define TOKEN_ASSIGN __entry->token = token
|
||||
#define TOKEN_PR_FMT " %d"
|
||||
#define TOKEN_PR_ARG __entry->token
|
||||
|
||||
DECLARE_EVENT_CLASS(dev_token,
|
||||
TP_PROTO(struct mt7615_dev *dev, u16 token),
|
||||
TP_ARGS(dev, token),
|
||||
TP_STRUCT__entry(
|
||||
DEV_ENTRY
|
||||
TOKEN_ENTRY
|
||||
),
|
||||
TP_fast_assign(
|
||||
DEV_ASSIGN;
|
||||
TOKEN_ASSIGN;
|
||||
),
|
||||
TP_printk(
|
||||
DEV_PR_FMT TOKEN_PR_FMT,
|
||||
DEV_PR_ARG, TOKEN_PR_ARG
|
||||
)
|
||||
);
|
||||
|
||||
DEFINE_EVENT(dev_token, mac_tx_free,
|
||||
TP_PROTO(struct mt7615_dev *dev, u16 token),
|
||||
TP_ARGS(dev, token)
|
||||
);
|
||||
|
||||
#endif
|
||||
|
||||
#undef TRACE_INCLUDE_PATH
|
||||
#define TRACE_INCLUDE_PATH .
|
||||
#undef TRACE_INCLUDE_FILE
|
||||
#define TRACE_INCLUDE_FILE mt7615_trace
|
||||
|
||||
#include <trace/define_trace.h>
|
@ -10,84 +10,15 @@
|
||||
#include <linux/pci.h>
|
||||
|
||||
#include "mt7615.h"
|
||||
#include "mac.h"
|
||||
|
||||
static const struct pci_device_id mt7615_pci_device_table[] = {
|
||||
{ PCI_DEVICE(0x14c3, 0x7615) },
|
||||
{ },
|
||||
};
|
||||
|
||||
u32 mt7615_reg_map(struct mt7615_dev *dev, u32 addr)
|
||||
{
|
||||
u32 base = addr & MT_MCU_PCIE_REMAP_2_BASE;
|
||||
u32 offset = addr & MT_MCU_PCIE_REMAP_2_OFFSET;
|
||||
|
||||
mt76_wr(dev, MT_MCU_PCIE_REMAP_2, base);
|
||||
|
||||
return MT_PCIE_REMAP_BASE_2 + offset;
|
||||
}
|
||||
|
||||
static void
|
||||
mt7615_rx_poll_complete(struct mt76_dev *mdev, enum mt76_rxq_id q)
|
||||
{
|
||||
struct mt7615_dev *dev = container_of(mdev, struct mt7615_dev, mt76);
|
||||
|
||||
mt7615_irq_enable(dev, MT_INT_RX_DONE(q));
|
||||
}
|
||||
|
||||
static irqreturn_t mt7615_irq_handler(int irq, void *dev_instance)
|
||||
{
|
||||
struct mt7615_dev *dev = dev_instance;
|
||||
u32 intr;
|
||||
|
||||
intr = mt76_rr(dev, MT_INT_SOURCE_CSR);
|
||||
mt76_wr(dev, MT_INT_SOURCE_CSR, intr);
|
||||
|
||||
if (!test_bit(MT76_STATE_INITIALIZED, &dev->mt76.state))
|
||||
return IRQ_NONE;
|
||||
|
||||
intr &= dev->mt76.mmio.irqmask;
|
||||
|
||||
if (intr & MT_INT_TX_DONE_ALL) {
|
||||
mt7615_irq_disable(dev, MT_INT_TX_DONE_ALL);
|
||||
napi_schedule(&dev->mt76.tx_napi);
|
||||
}
|
||||
|
||||
if (intr & MT_INT_RX_DONE(0)) {
|
||||
mt7615_irq_disable(dev, MT_INT_RX_DONE(0));
|
||||
napi_schedule(&dev->mt76.napi[0]);
|
||||
}
|
||||
|
||||
if (intr & MT_INT_RX_DONE(1)) {
|
||||
mt7615_irq_disable(dev, MT_INT_RX_DONE(1));
|
||||
napi_schedule(&dev->mt76.napi[1]);
|
||||
}
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
static int mt7615_pci_probe(struct pci_dev *pdev,
|
||||
const struct pci_device_id *id)
|
||||
{
|
||||
static const struct mt76_driver_ops drv_ops = {
|
||||
/* txwi_size = txd size + txp size */
|
||||
.txwi_size = MT_TXD_SIZE + sizeof(struct mt7615_txp),
|
||||
.drv_flags = MT_DRV_TXWI_NO_FREE,
|
||||
.survey_flags = SURVEY_INFO_TIME_TX |
|
||||
SURVEY_INFO_TIME_RX |
|
||||
SURVEY_INFO_TIME_BSS_RX,
|
||||
.tx_prepare_skb = mt7615_tx_prepare_skb,
|
||||
.tx_complete_skb = mt7615_tx_complete_skb,
|
||||
.rx_skb = mt7615_queue_rx_skb,
|
||||
.rx_poll_complete = mt7615_rx_poll_complete,
|
||||
.sta_ps = mt7615_sta_ps,
|
||||
.sta_add = mt7615_sta_add,
|
||||
.sta_assoc = mt7615_sta_assoc,
|
||||
.sta_remove = mt7615_sta_remove,
|
||||
.update_survey = mt7615_update_channel,
|
||||
};
|
||||
struct mt7615_dev *dev;
|
||||
struct mt76_dev *mdev;
|
||||
int ret;
|
||||
|
||||
ret = pcim_enable_device(pdev);
|
||||
@ -104,31 +35,7 @@ static int mt7615_pci_probe(struct pci_dev *pdev,
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
mdev = mt76_alloc_device(&pdev->dev, sizeof(*dev), &mt7615_ops,
|
||||
&drv_ops);
|
||||
if (!mdev)
|
||||
return -ENOMEM;
|
||||
|
||||
dev = container_of(mdev, struct mt7615_dev, mt76);
|
||||
mt76_mmio_init(&dev->mt76, pcim_iomap_table(pdev)[0]);
|
||||
|
||||
mdev->rev = (mt76_rr(dev, MT_HW_CHIPID) << 16) |
|
||||
(mt76_rr(dev, MT_HW_REV) & 0xff);
|
||||
dev_dbg(mdev->dev, "ASIC revision: %04x\n", mdev->rev);
|
||||
|
||||
ret = devm_request_irq(mdev->dev, pdev->irq, mt7615_irq_handler,
|
||||
IRQF_SHARED, KBUILD_MODNAME, dev);
|
||||
if (ret)
|
||||
goto error;
|
||||
|
||||
ret = mt7615_register_device(dev);
|
||||
if (ret)
|
||||
goto error;
|
||||
|
||||
return 0;
|
||||
error:
|
||||
ieee80211_free_hw(mt76_hw(dev));
|
||||
return ret;
|
||||
return mt7615_mmio_probe(&pdev->dev, pcim_iomap_table(pdev)[0], pdev->irq);
|
||||
}
|
||||
|
||||
static void mt7615_pci_remove(struct pci_dev *pdev)
|
||||
@ -146,10 +53,7 @@ struct pci_driver mt7615_pci_driver = {
|
||||
.remove = mt7615_pci_remove,
|
||||
};
|
||||
|
||||
module_pci_driver(mt7615_pci_driver);
|
||||
|
||||
MODULE_DEVICE_TABLE(pci, mt7615_pci_device_table);
|
||||
MODULE_FIRMWARE(MT7615_FIRMWARE_CR4);
|
||||
MODULE_FIRMWARE(MT7615_FIRMWARE_N9);
|
||||
MODULE_FIRMWARE(MT7615_ROM_PATCH);
|
||||
MODULE_LICENSE("Dual BSD/GPL");
|
||||
|
@ -8,6 +8,10 @@
|
||||
#define MT_HW_CHIPID 0x1008
|
||||
#define MT_TOP_STRAP_STA 0x1010
|
||||
#define MT_TOP_3NSS BIT(24)
|
||||
|
||||
#define MT_TOP_OFF_RSV 0x1128
|
||||
#define MT_TOP_OFF_RSV_FW_STATE GENMASK(18, 16)
|
||||
|
||||
#define MT_TOP_MISC2 0x1134
|
||||
#define MT_TOP_MISC2_FW_STATE GENMASK(2, 0)
|
||||
|
||||
@ -31,14 +35,21 @@
|
||||
#define MT_CFG_LPCR_HOST_FW_OWN BIT(0)
|
||||
#define MT_CFG_LPCR_HOST_DRV_OWN BIT(1)
|
||||
|
||||
#define MT_MCU_INT_EVENT MT_HIF(0x1f8)
|
||||
#define MT_MCU_INT_EVENT_PDMA_STOPPED BIT(0)
|
||||
#define MT_MCU_INT_EVENT_PDMA_INIT BIT(1)
|
||||
#define MT_MCU_INT_EVENT_SER_TRIGGER BIT(2)
|
||||
#define MT_MCU_INT_EVENT_RESET_DONE BIT(3)
|
||||
|
||||
#define MT_INT_SOURCE_CSR MT_HIF(0x200)
|
||||
#define MT_INT_MASK_CSR MT_HIF(0x204)
|
||||
#define MT_DELAY_INT_CFG MT_HIF(0x210)
|
||||
|
||||
#define MT_INT_RX_DONE(_n) BIT(_n)
|
||||
#define MT_INT_RX_DONE_ALL GENMASK(1, 0)
|
||||
#define MT_INT_TX_DONE_ALL GENMASK(7, 4)
|
||||
#define MT_INT_TX_DONE_ALL GENMASK(19, 4)
|
||||
#define MT_INT_TX_DONE(_n) BIT((_n) + 4)
|
||||
#define MT_INT_MCU_CMD BIT(30)
|
||||
|
||||
#define MT_WPDMA_GLO_CFG MT_HIF(0x208)
|
||||
#define MT_WPDMA_GLO_CFG_TX_DMA_EN BIT(0)
|
||||
@ -49,6 +60,7 @@
|
||||
#define MT_WPDMA_GLO_CFG_TX_WRITEBACK_DONE BIT(6)
|
||||
#define MT_WPDMA_GLO_CFG_BIG_ENDIAN BIT(7)
|
||||
#define MT_WPDMA_GLO_CFG_TX_BT_SIZE_BIT0 BIT(9)
|
||||
#define MT_WPDMA_GLO_CFG_BYPASS_TX_SCH BIT(9) /* MT7622 */
|
||||
#define MT_WPDMA_GLO_CFG_MULTI_DMA_EN GENMASK(11, 10)
|
||||
#define MT_WPDMA_GLO_CFG_FIFO_LITTLE_ENDIAN BIT(12)
|
||||
#define MT_WPDMA_GLO_CFG_TX_BT_SIZE_BIT21 GENMASK(23, 22)
|
||||
@ -58,6 +70,22 @@
|
||||
|
||||
#define MT_WPDMA_RST_IDX MT_HIF(0x20c)
|
||||
|
||||
#define MT_WPDMA_MEM_RNG_ERR MT_HIF(0x224)
|
||||
|
||||
#define MT_MCU_CMD MT_HIF(0x234)
|
||||
#define MT_MCU_CMD_CLEAR_FW_OWN BIT(0)
|
||||
#define MT_MCU_CMD_STOP_PDMA_FW_RELOAD BIT(1)
|
||||
#define MT_MCU_CMD_STOP_PDMA BIT(2)
|
||||
#define MT_MCU_CMD_RESET_DONE BIT(3)
|
||||
#define MT_MCU_CMD_RECOVERY_DONE BIT(4)
|
||||
#define MT_MCU_CMD_NORMAL_STATE BIT(5)
|
||||
#define MT_MCU_CMD_LMAC_ERROR BIT(24)
|
||||
#define MT_MCU_CMD_PSE_ERROR BIT(25)
|
||||
#define MT_MCU_CMD_PLE_ERROR BIT(26)
|
||||
#define MT_MCU_CMD_PDMA_ERROR BIT(27)
|
||||
#define MT_MCU_CMD_PCIE_ERROR BIT(28)
|
||||
#define MT_MCU_CMD_ERROR_MASK (GENMASK(5, 1) | GENMASK(28, 24))
|
||||
|
||||
#define MT_TX_RING_BASE MT_HIF(0x300)
|
||||
#define MT_RX_RING_BASE MT_HIF(0x400)
|
||||
|
||||
@ -81,36 +109,35 @@
|
||||
#define MT_WF_PHY_BASE 0x10000
|
||||
#define MT_WF_PHY(ofs) (MT_WF_PHY_BASE + (ofs))
|
||||
|
||||
#define MT_WF_PHY_WF2_RFCTRL0 MT_WF_PHY(0x1900)
|
||||
#define MT_WF_PHY_WF2_RFCTRL0(n) MT_WF_PHY(0x1900 + (n) * 0x400)
|
||||
#define MT_WF_PHY_WF2_RFCTRL0_LPBCN_EN BIT(9)
|
||||
|
||||
#define MT_WF_PHY_R0_B0_PHYMUX_5 MT_WF_PHY(0x0614)
|
||||
#define MT_WF_PHY_R0_PHYMUX_5(_phy) MT_WF_PHY(0x0614 + ((_phy) << 9))
|
||||
|
||||
#define MT_WF_PHY_R0_B0_PHYCTRL_STS0 MT_WF_PHY(0x020c)
|
||||
#define MT_WF_PHY_R0_PHYCTRL_STS0(_phy) MT_WF_PHY(0x020c + ((_phy) << 9))
|
||||
#define MT_WF_PHYCTRL_STAT_PD_OFDM GENMASK(31, 16)
|
||||
#define MT_WF_PHYCTRL_STAT_PD_CCK GENMASK(15, 0)
|
||||
|
||||
#define MT_WF_PHY_R0_B0_PHYCTRL_STS5 MT_WF_PHY(0x0220)
|
||||
#define MT_WF_PHY_R0_PHYCTRL_STS5(_phy) MT_WF_PHY(0x0220 + ((_phy) << 9))
|
||||
#define MT_WF_PHYCTRL_STAT_MDRDY_OFDM GENMASK(31, 16)
|
||||
#define MT_WF_PHYCTRL_STAT_MDRDY_CCK GENMASK(15, 0)
|
||||
|
||||
#define MT_WF_PHY_B0_MIN_PRI_PWR MT_WF_PHY(0x229c)
|
||||
#define MT_WF_PHY_B0_PD_OFDM_MASK GENMASK(28, 20)
|
||||
#define MT_WF_PHY_B0_PD_OFDM(v) ((v) << 20)
|
||||
#define MT_WF_PHY_B0_PD_BLK BIT(19)
|
||||
#define MT_WF_PHY_MIN_PRI_PWR(_phy) MT_WF_PHY((_phy) ? 0x084 : 0x229c)
|
||||
#define MT_WF_PHY_PD_OFDM_MASK(_phy) ((_phy) ? GENMASK(24, 16) : \
|
||||
GENMASK(28, 20))
|
||||
#define MT_WF_PHY_PD_OFDM(_phy, v) ((v) << ((_phy) ? 16 : 20))
|
||||
#define MT_WF_PHY_PD_BLK(_phy) ((_phy) ? BIT(25) : BIT(19))
|
||||
|
||||
#define MT_WF_PHY_B1_MIN_PRI_PWR MT_WF_PHY(0x084)
|
||||
#define MT_WF_PHY_B1_PD_OFDM_MASK GENMASK(24, 16)
|
||||
#define MT_WF_PHY_B1_PD_OFDM(v) ((v) << 16)
|
||||
#define MT_WF_PHY_B1_PD_BLK BIT(25)
|
||||
#define MT_WF_PHY_RXTD_BASE MT_WF_PHY(0x2200)
|
||||
#define MT_WF_PHY_RXTD(_n) (MT_WF_PHY_RXTD_BASE + ((_n) << 2))
|
||||
|
||||
#define MT_WF_PHY_B0_RXTD_CCK_PD MT_WF_PHY(0x2310)
|
||||
#define MT_WF_PHY_B0_PD_CCK_MASK GENMASK(8, 1)
|
||||
#define MT_WF_PHY_B0_PD_CCK(v) ((v) << 1)
|
||||
#define MT_WF_PHY_RXTD_CCK_PD(_phy) MT_WF_PHY((_phy) ? 0x2314 : 0x2310)
|
||||
#define MT_WF_PHY_PD_CCK_MASK(_phy) (_phy) ? GENMASK(31, 24) : \
|
||||
GENMASK(8, 1)
|
||||
#define MT_WF_PHY_PD_CCK(_phy, v) ((v) << ((_phy) ? 24 : 1))
|
||||
|
||||
#define MT_WF_PHY_B1_RXTD_CCK_PD MT_WF_PHY(0x2314)
|
||||
#define MT_WF_PHY_B1_PD_CCK_MASK GENMASK(31, 24)
|
||||
#define MT_WF_PHY_B1_PD_CCK(v) ((v) << 24)
|
||||
#define MT_WF_PHY_RXTD2_BASE MT_WF_PHY(0x2a00)
|
||||
#define MT_WF_PHY_RXTD2(_n) (MT_WF_PHY_RXTD2_BASE + ((_n) << 2))
|
||||
|
||||
#define MT_WF_CFG_BASE 0x20200
|
||||
#define MT_WF_CFG(ofs) (MT_WF_CFG_BASE + (ofs))
|
||||
@ -131,8 +158,8 @@
|
||||
#define MT_AGG_ARCR_RATE_DOWN_RATIO_EN BIT(19)
|
||||
#define MT_AGG_ARCR_RATE_UP_EXTRA_TH GENMASK(22, 20)
|
||||
|
||||
#define MT_AGG_ARUCR MT_WF_AGG(0x018)
|
||||
#define MT_AGG_ARDCR MT_WF_AGG(0x01c)
|
||||
#define MT_AGG_ARUCR(_band) MT_WF_AGG(0x018 + (_band) * 0x100)
|
||||
#define MT_AGG_ARDCR(_band) MT_WF_AGG(0x01c + (_band) * 0x100)
|
||||
#define MT_AGG_ARxCR_LIMIT_SHIFT(_n) (4 * (_n))
|
||||
#define MT_AGG_ARxCR_LIMIT(_n) GENMASK(2 + \
|
||||
MT_AGG_ARxCR_LIMIT_SHIFT(_n), \
|
||||
@ -142,8 +169,7 @@
|
||||
#define MT_AGG_ASRCR1 MT_WF_AGG(0x064)
|
||||
#define MT_AGG_ASRCR_RANGE(val, n) (((val) >> ((n) << 3)) & GENMASK(5, 0))
|
||||
|
||||
#define MT_AGG_ACR0 MT_WF_AGG(0x070)
|
||||
#define MT_AGG_ACR1 MT_WF_AGG(0x170)
|
||||
#define MT_AGG_ACR(_band) MT_WF_AGG(0x070 + (_band) * 0x100)
|
||||
#define MT_AGG_ACR_NO_BA_RULE BIT(0)
|
||||
#define MT_AGG_ACR_NO_BA_AR_RULE BIT(1)
|
||||
#define MT_AGG_ACR_PKT_TIME_EN BIT(2)
|
||||
@ -153,14 +179,33 @@
|
||||
#define MT_AGG_SCR MT_WF_AGG(0x0fc)
|
||||
#define MT_AGG_SCR_NLNAV_MID_PTEC_DIS BIT(3)
|
||||
|
||||
#define MT_WF_ARB_BASE 0x20c00
|
||||
#define MT_WF_ARB(ofs) (MT_WF_ARB_BASE + (ofs))
|
||||
|
||||
#define MT_ARB_SCR MT_WF_ARB(0x080)
|
||||
#define MT_ARB_SCR_TX0_DISABLE BIT(8)
|
||||
#define MT_ARB_SCR_RX0_DISABLE BIT(9)
|
||||
#define MT_ARB_SCR_TX1_DISABLE BIT(10)
|
||||
#define MT_ARB_SCR_RX1_DISABLE BIT(11)
|
||||
|
||||
#define MT_WF_TMAC_BASE 0x21000
|
||||
#define MT_WF_TMAC(ofs) (MT_WF_TMAC_BASE + (ofs))
|
||||
|
||||
#define MT_TMAC_TRCR0 MT_WF_TMAC(0x09c)
|
||||
#define MT_TMAC_TRCR1 MT_WF_TMAC(0x070)
|
||||
#define MT_TMAC_CDTR MT_WF_TMAC(0x090)
|
||||
#define MT_TMAC_ODTR MT_WF_TMAC(0x094)
|
||||
#define MT_TIMEOUT_VAL_PLCP GENMASK(15, 0)
|
||||
#define MT_TIMEOUT_VAL_CCA GENMASK(31, 16)
|
||||
|
||||
#define MT_TMAC_TRCR(_band) MT_WF_TMAC((_band) ? 0x070 : 0x09c)
|
||||
#define MT_TMAC_TRCR_CCA_SEL GENMASK(31, 30)
|
||||
#define MT_TMAC_TRCR_SEC_CCA_SEL GENMASK(29, 28)
|
||||
|
||||
#define MT_TMAC_ICR(_band) MT_WF_TMAC((_band) ? 0x074 : 0x0a4)
|
||||
#define MT_IFS_EIFS GENMASK(8, 0)
|
||||
#define MT_IFS_RIFS GENMASK(14, 10)
|
||||
#define MT_IFS_SIFS GENMASK(22, 16)
|
||||
#define MT_IFS_SLOT GENMASK(30, 24)
|
||||
|
||||
#define MT_TMAC_CTCR0 MT_WF_TMAC(0x0f4)
|
||||
#define MT_TMAC_CTCR0_INS_DDLMT_REFTIME GENMASK(5, 0)
|
||||
#define MT_TMAC_CTCR0_INS_DDLMT_DENSITY GENMASK(15, 12)
|
||||
@ -170,7 +215,7 @@
|
||||
#define MT_WF_RMAC_BASE 0x21200
|
||||
#define MT_WF_RMAC(ofs) (MT_WF_RMAC_BASE + (ofs))
|
||||
|
||||
#define MT_WF_RFCR MT_WF_RMAC(0x000)
|
||||
#define MT_WF_RFCR(_band) MT_WF_RMAC((_band) ? 0x100 : 0x000)
|
||||
#define MT_WF_RFCR_DROP_STBC_MULTI BIT(0)
|
||||
#define MT_WF_RFCR_DROP_FCSFAIL BIT(1)
|
||||
#define MT_WF_RFCR_DROP_VERSION BIT(3)
|
||||
@ -193,13 +238,15 @@
|
||||
#define MT_WF_RFCR_DROP_NDPA BIT(20)
|
||||
#define MT_WF_RFCR_DROP_UNWANTED_CTL BIT(21)
|
||||
|
||||
#define MT_WF_RFCR1 MT_WF_RMAC(0x004)
|
||||
#define MT_WF_RFCR1(_band) MT_WF_RMAC((_band) ? 0x104 : 0x004)
|
||||
#define MT_WF_RFCR1_DROP_ACK BIT(4)
|
||||
#define MT_WF_RFCR1_DROP_BF_POLL BIT(5)
|
||||
#define MT_WF_RFCR1_DROP_BA BIT(6)
|
||||
#define MT_WF_RFCR1_DROP_CFEND BIT(7)
|
||||
#define MT_WF_RFCR1_DROP_CFACK BIT(8)
|
||||
|
||||
#define MT_CHFREQ(_band) MT_WF_RMAC((_band) ? 0x130 : 0x030)
|
||||
|
||||
#define MT_WF_RMAC_MIB_TIME0 MT_WF_RMAC(0x03c4)
|
||||
#define MT_WF_RMAC_MIB_RXTIME_CLR BIT(31)
|
||||
#define MT_WF_RMAC_MIB_RXTIME_EN BIT(30)
|
||||
@ -207,6 +254,7 @@
|
||||
#define MT_WF_RMAC_MIB_AIRTIME0 MT_WF_RMAC(0x0380)
|
||||
|
||||
#define MT_WF_RMAC_MIB_TIME5 MT_WF_RMAC(0x03d8)
|
||||
#define MT_WF_RMAC_MIB_TIME6 MT_WF_RMAC(0x03dc)
|
||||
#define MT_MIB_OBSSTIME_MASK GENMASK(23, 0)
|
||||
|
||||
#define MT_WF_DMA_BASE 0x21800
|
||||
@ -216,8 +264,7 @@
|
||||
#define MT_DMA_DCR0_MAX_RX_LEN GENMASK(15, 2)
|
||||
#define MT_DMA_DCR0_RX_VEC_DROP BIT(17)
|
||||
|
||||
#define MT_DMA_BN0RCFR0 MT_WF_DMA(0x070)
|
||||
#define MT_DMA_BN1RCFR0 MT_WF_DMA(0x0b0)
|
||||
#define MT_DMA_RCFR0(_band) MT_WF_DMA(0x070 + (_band) * 0x40)
|
||||
#define MT_DMA_RCFR0_MCU_RX_MGMT BIT(2)
|
||||
#define MT_DMA_RCFR0_MCU_RX_CTL_NON_BAR BIT(3)
|
||||
#define MT_DMA_RCFR0_MCU_RX_CTL_BAR BIT(4)
|
||||
@ -294,9 +341,9 @@
|
||||
#define MT_WF_MIB(ofs) (MT_WF_MIB_BASE + (ofs))
|
||||
|
||||
#define MT_MIB_M0_MISC_CR MT_WF_MIB(0x00c)
|
||||
#define MT_MIB_MB_SDR0(n) MT_WF_MIB(0x100 + ((n) << 4))
|
||||
#define MT_MIB_RTS_RETRIES_COUNT_MASK GENMASK(31, 16)
|
||||
#define MT_MIB_RTS_COUNT_MASK GENMASK(15, 0)
|
||||
|
||||
#define MT_MIB_SDR3(n) MT_WF_MIB(0x014 + ((n) << 9))
|
||||
#define MT_MIB_SDR3_FCS_ERR_MASK GENMASK(15, 0)
|
||||
|
||||
#define MT_MIB_SDR9(n) MT_WF_MIB(0x02c + ((n) << 9))
|
||||
#define MT_MIB_SDR9_BUSY_MASK GENMASK(23, 0)
|
||||
@ -309,8 +356,56 @@
|
||||
#define MT_MIB_SDR37(n) MT_WF_MIB(0x09c + ((n) << 9))
|
||||
#define MT_MIB_SDR37_RXTIME_MASK GENMASK(23, 0)
|
||||
|
||||
#define MT_MIB_MB_SDR0(_band, n) MT_WF_MIB(0x100 + ((_band) << 9) + \
|
||||
((n) << 4))
|
||||
#define MT_MIB_RTS_RETRIES_COUNT_MASK GENMASK(31, 16)
|
||||
#define MT_MIB_RTS_COUNT_MASK GENMASK(15, 0)
|
||||
|
||||
#define MT_MIB_MB_SDR1(_band, n) MT_WF_MIB(0x104 + ((_band) << 9) + \
|
||||
((n) << 4))
|
||||
#define MT_MIB_ACK_FAIL_COUNT_MASK GENMASK(31, 16)
|
||||
|
||||
#define MT_TX_AGG_CNT(n) MT_WF_MIB(0xa8 + ((n) << 2))
|
||||
|
||||
#define MT_DMASHDL_BASE 0x5000a000
|
||||
#define MT_DMASHDL_OPTIONAL 0x008
|
||||
#define MT_DMASHDL_PAGE 0x00c
|
||||
|
||||
#define MT_DMASHDL_REFILL 0x010
|
||||
|
||||
#define MT_DMASHDL_PKT_MAX_SIZE 0x01c
|
||||
#define MT_DMASHDL_PKT_MAX_SIZE_PLE GENMASK(11, 0)
|
||||
#define MT_DMASHDL_PKT_MAX_SIZE_PSE GENMASK(27, 16)
|
||||
|
||||
#define MT_DMASHDL_GROUP_QUOTA(_n) (0x020 + ((_n) << 2))
|
||||
#define MT_DMASHDL_GROUP_QUOTA_MIN GENMASK(11, 0)
|
||||
#define MT_DMASHDL_GROUP_QUOTA_MAX GENMASK(27, 16)
|
||||
|
||||
#define MT_DMASHDL_SCHED_SET0 0x0b0
|
||||
#define MT_DMASHDL_SCHED_SET1 0x0b4
|
||||
|
||||
#define MT_DMASHDL_Q_MAP(_n) (0x0d0 + ((_n) << 2))
|
||||
#define MT_DMASHDL_Q_MAP_MASK GENMASK(3, 0)
|
||||
#define MT_DMASHDL_Q_MAP_SHIFT(_n) (4 * ((_n) % 8))
|
||||
|
||||
#define MT_LED_BASE_PHYS 0x80024000
|
||||
#define MT_LED_PHYS(_n) (MT_LED_BASE_PHYS + (_n))
|
||||
|
||||
#define MT_LED_CTRL MT_LED_PHYS(0x00)
|
||||
|
||||
#define MT_LED_CTRL_REPLAY(_n) BIT(0 + (8 * (_n)))
|
||||
#define MT_LED_CTRL_POLARITY(_n) BIT(1 + (8 * (_n)))
|
||||
#define MT_LED_CTRL_TX_BLINK_MODE(_n) BIT(2 + (8 * (_n)))
|
||||
#define MT_LED_CTRL_TX_MANUAL_BLINK(_n) BIT(3 + (8 * (_n)))
|
||||
#define MT_LED_CTRL_TX_OVER_BLINK(_n) BIT(5 + (8 * (_n)))
|
||||
#define MT_LED_CTRL_KICK(_n) BIT(7 + (8 * (_n)))
|
||||
|
||||
#define MT_LED_STATUS_0(_n) MT_LED_PHYS(0x10 + ((_n) * 8))
|
||||
#define MT_LED_STATUS_1(_n) MT_LED_PHYS(0x14 + ((_n) * 8))
|
||||
#define MT_LED_STATUS_OFF GENMASK(31, 24)
|
||||
#define MT_LED_STATUS_ON GENMASK(23, 16)
|
||||
#define MT_LED_STATUS_DURATION GENMASK(15, 0)
|
||||
|
||||
#define MT_EFUSE_BASE 0x81070000
|
||||
#define MT_EFUSE_BASE_CTRL 0x000
|
||||
#define MT_EFUSE_BASE_CTRL_EMPTY BIT(30)
|
||||
@ -328,4 +423,8 @@
|
||||
#define MT_EFUSE_WDATA(_i) (0x010 + ((_i) * 4))
|
||||
#define MT_EFUSE_RDATA(_i) (0x030 + ((_i) * 4))
|
||||
|
||||
/* INFRACFG host register range on MT7622 */
|
||||
#define MT_INFRACFG_MISC 0x700
|
||||
#define MT_INFRACFG_MISC_AP2CONN_WAKE BIT(1)
|
||||
|
||||
#endif
|
||||
|
77
drivers/net/wireless/mediatek/mt76/mt7615/soc.c
Normal file
77
drivers/net/wireless/mediatek/mt76/mt7615/soc.c
Normal file
@ -0,0 +1,77 @@
|
||||
// SPDX-License-Identifier: ISC
|
||||
/* Copyright (C) 2019 MediaTek Inc.
|
||||
*
|
||||
* Author: Ryder Lee <ryder.lee@mediatek.com>
|
||||
* Felix Fietkau <nbd@nbd.name>
|
||||
*/
|
||||
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/regmap.h>
|
||||
#include <linux/mfd/syscon.h>
|
||||
#include <linux/of.h>
|
||||
#include "mt7615.h"
|
||||
|
||||
int mt7622_wmac_init(struct mt7615_dev *dev)
|
||||
{
|
||||
struct device_node *np = dev->mt76.dev->of_node;
|
||||
|
||||
if (!is_mt7622(&dev->mt76))
|
||||
return 0;
|
||||
|
||||
dev->infracfg = syscon_regmap_lookup_by_phandle(np, "mediatek,infracfg");
|
||||
if (IS_ERR(dev->infracfg)) {
|
||||
dev_err(dev->mt76.dev, "Cannot find infracfg controller\n");
|
||||
return PTR_ERR(dev->infracfg);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int mt7622_wmac_probe(struct platform_device *pdev)
|
||||
{
|
||||
struct resource *res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||
void __iomem *mem_base;
|
||||
int irq;
|
||||
|
||||
irq = platform_get_irq(pdev, 0);
|
||||
if (irq < 0) {
|
||||
dev_err(&pdev->dev, "Failed to get device IRQ\n");
|
||||
return irq;
|
||||
}
|
||||
|
||||
mem_base = devm_ioremap_resource(&pdev->dev, res);
|
||||
if (IS_ERR(mem_base)) {
|
||||
dev_err(&pdev->dev, "Failed to get memory resource\n");
|
||||
return PTR_ERR(mem_base);
|
||||
}
|
||||
|
||||
return mt7615_mmio_probe(&pdev->dev, mem_base, irq);
|
||||
}
|
||||
|
||||
static int mt7622_wmac_remove(struct platform_device *pdev)
|
||||
{
|
||||
struct mt7615_dev *dev = platform_get_drvdata(pdev);
|
||||
|
||||
mt7615_unregister_device(dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct of_device_id mt7622_wmac_of_match[] = {
|
||||
{ .compatible = "mediatek,mt7622-wmac" },
|
||||
{},
|
||||
};
|
||||
|
||||
struct platform_driver mt7622_wmac_driver = {
|
||||
.driver = {
|
||||
.name = "mt7622-wmac",
|
||||
.of_match_table = mt7622_wmac_of_match,
|
||||
},
|
||||
.probe = mt7622_wmac_probe,
|
||||
.remove = mt7622_wmac_remove,
|
||||
};
|
||||
|
||||
MODULE_FIRMWARE(MT7622_FIRMWARE_N9);
|
||||
MODULE_FIRMWARE(MT7622_ROM_PATCH);
|
12
drivers/net/wireless/mediatek/mt76/mt7615/trace.c
Normal file
12
drivers/net/wireless/mediatek/mt76/mt7615/trace.c
Normal file
@ -0,0 +1,12 @@
|
||||
// SPDX-License-Identifier: ISC
|
||||
/*
|
||||
* Copyright (C) 2019 Lorenzo Bianconi <lorenzo@kernel.org>
|
||||
*/
|
||||
|
||||
#include <linux/module.h>
|
||||
|
||||
#ifndef __CHECKER__
|
||||
#define CREATE_TRACE_POINTS
|
||||
#include "mt7615_trace.h"
|
||||
|
||||
#endif
|
@ -109,7 +109,7 @@ static void mt76x0_set_freq_offset(struct mt76x02_dev *dev)
|
||||
|
||||
void mt76x0_read_rx_gain(struct mt76x02_dev *dev)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
struct mt76x02_rx_freq_cal *caldata = &dev->cal.rx;
|
||||
s8 val, lna_5g[3], lna_2g;
|
||||
u16 rssi_offset;
|
||||
@ -129,7 +129,7 @@ void mt76x0_read_rx_gain(struct mt76x02_dev *dev)
|
||||
|
||||
static s8 mt76x0_get_delta(struct mt76x02_dev *dev)
|
||||
{
|
||||
struct cfg80211_chan_def *chandef = &dev->mt76.chandef;
|
||||
struct cfg80211_chan_def *chandef = &dev->mphy.chandef;
|
||||
u8 val;
|
||||
|
||||
if (chandef->width == NL80211_CHAN_WIDTH_80) {
|
||||
|
@ -264,12 +264,12 @@ int mt76x0_register_device(struct mt76x02_dev *dev)
|
||||
|
||||
if (dev->mt76.cap.has_5ghz) {
|
||||
/* overwrite unsupported features */
|
||||
mt76x0_vht_cap_mask(&dev->mt76.sband_5g.sband);
|
||||
mt76x0_init_txpower(dev, &dev->mt76.sband_5g.sband);
|
||||
mt76x0_vht_cap_mask(&dev->mphy.sband_5g.sband);
|
||||
mt76x0_init_txpower(dev, &dev->mphy.sband_5g.sband);
|
||||
}
|
||||
|
||||
if (dev->mt76.cap.has_2ghz)
|
||||
mt76x0_init_txpower(dev, &dev->mt76.sband_2g.sband);
|
||||
mt76x0_init_txpower(dev, &dev->mphy.sband_2g.sband);
|
||||
|
||||
mt76x02_init_debugfs(dev);
|
||||
|
||||
|
@ -16,7 +16,7 @@ mt76x0_set_channel(struct mt76x02_dev *dev, struct cfg80211_chan_def *chandef)
|
||||
if (mt76_is_mmio(&dev->mt76))
|
||||
tasklet_disable(&dev->dfs_pd.dfs_tasklet);
|
||||
|
||||
mt76_set_channel(&dev->mt76);
|
||||
mt76_set_channel(&dev->mphy);
|
||||
mt76x0_phy_set_channel(dev, chandef);
|
||||
|
||||
mt76x02_mac_cc_reset(dev);
|
||||
@ -28,7 +28,7 @@ mt76x0_set_channel(struct mt76x02_dev *dev, struct cfg80211_chan_def *chandef)
|
||||
}
|
||||
mt76x02_pre_tbtt_enable(dev, true);
|
||||
|
||||
mt76_txq_schedule_all(&dev->mt76);
|
||||
mt76_txq_schedule_all(&dev->mphy);
|
||||
}
|
||||
|
||||
int mt76x0_config(struct ieee80211_hw *hw, u32 changed)
|
||||
@ -44,9 +44,9 @@ int mt76x0_config(struct ieee80211_hw *hw, u32 changed)
|
||||
}
|
||||
|
||||
if (changed & IEEE80211_CONF_CHANGE_POWER) {
|
||||
dev->mt76.txpower_conf = hw->conf.power_level * 2;
|
||||
dev->txpower_conf = hw->conf.power_level * 2;
|
||||
|
||||
if (test_bit(MT76_STATE_RUNNING, &dev->mt76.state))
|
||||
if (test_bit(MT76_STATE_RUNNING, &dev->mphy.state))
|
||||
mt76x0_phy_set_txpower(dev);
|
||||
}
|
||||
|
||||
|
@ -20,7 +20,7 @@ static int mt76x0e_start(struct ieee80211_hw *hw)
|
||||
MT_MAC_WORK_INTERVAL);
|
||||
ieee80211_queue_delayed_work(dev->mt76.hw, &dev->cal_work,
|
||||
MT_CALIBRATE_INTERVAL);
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -47,7 +47,7 @@ static void mt76x0e_stop(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct mt76x02_dev *dev = hw->priv;
|
||||
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
mt76x0e_stop_hw(dev);
|
||||
}
|
||||
|
||||
@ -67,6 +67,7 @@ static const struct ieee80211_ops mt76x0e_ops = {
|
||||
.configure_filter = mt76x02_configure_filter,
|
||||
.bss_info_changed = mt76x02_bss_info_changed,
|
||||
.sta_state = mt76_sta_state,
|
||||
.sta_pre_rcu_remove = mt76_sta_pre_rcu_remove,
|
||||
.set_key = mt76x02_set_key,
|
||||
.conf_tx = mt76x02_conf_tx,
|
||||
.sw_scan_start = mt76_sw_scan,
|
||||
@ -124,7 +125,7 @@ static int mt76x0e_register_device(struct mt76x02_dev *dev)
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -195,7 +196,7 @@ error:
|
||||
|
||||
static void mt76x0e_cleanup(struct mt76x02_dev *dev)
|
||||
{
|
||||
clear_bit(MT76_STATE_INITIALIZED, &dev->mt76.state);
|
||||
clear_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);
|
||||
tasklet_disable(&dev->mt76.pre_tbtt_tasklet);
|
||||
mt76x0_chip_onoff(dev, false, false);
|
||||
mt76x0e_stop_hw(dev);
|
||||
|
@ -126,7 +126,7 @@ int mt76x0e_mcu_init(struct mt76x02_dev *dev)
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
set_bit(MT76_STATE_MCU_RUNNING, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_MCU_RUNNING, &dev->mphy.state);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -23,7 +23,7 @@ mt76x0_rf_csr_wr(struct mt76x02_dev *dev, u32 offset, u8 value)
|
||||
int ret = 0;
|
||||
u8 bank, reg;
|
||||
|
||||
if (test_bit(MT76_REMOVED, &dev->mt76.state))
|
||||
if (test_bit(MT76_REMOVED, &dev->mphy.state))
|
||||
return -ENODEV;
|
||||
|
||||
bank = MT_RF_BANK(offset);
|
||||
@ -62,7 +62,7 @@ static int mt76x0_rf_csr_rr(struct mt76x02_dev *dev, u32 offset)
|
||||
u32 val;
|
||||
u8 bank, reg;
|
||||
|
||||
if (test_bit(MT76_REMOVED, &dev->mt76.state))
|
||||
if (test_bit(MT76_REMOVED, &dev->mphy.state))
|
||||
return -ENODEV;
|
||||
|
||||
bank = MT_RF_BANK(offset);
|
||||
@ -109,7 +109,7 @@ mt76x0_rf_wr(struct mt76x02_dev *dev, u32 offset, u8 val)
|
||||
};
|
||||
|
||||
WARN_ON_ONCE(!test_bit(MT76_STATE_MCU_RUNNING,
|
||||
&dev->mt76.state));
|
||||
&dev->mphy.state));
|
||||
return mt76_wr_rp(dev, MT_MCU_MEMMAP_RF, &pair, 1);
|
||||
} else {
|
||||
return mt76x0_rf_csr_wr(dev, offset, val);
|
||||
@ -127,7 +127,7 @@ static int mt76x0_rf_rr(struct mt76x02_dev *dev, u32 offset)
|
||||
};
|
||||
|
||||
WARN_ON_ONCE(!test_bit(MT76_STATE_MCU_RUNNING,
|
||||
&dev->mt76.state));
|
||||
&dev->mphy.state));
|
||||
ret = mt76_rd_rp(dev, MT_MCU_MEMMAP_RF, &pair, 1);
|
||||
val = pair.value;
|
||||
} else {
|
||||
@ -502,7 +502,7 @@ mt76x0_phy_bbp_set_bw(struct mt76x02_dev *dev, enum nl80211_chan_width width)
|
||||
|
||||
static void mt76x0_phy_tssi_dc_calibrate(struct mt76x02_dev *dev)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
u32 val;
|
||||
|
||||
if (chan->band == NL80211_BAND_5GHZ)
|
||||
@ -543,7 +543,7 @@ static int
|
||||
mt76x0_phy_tssi_adc_calibrate(struct mt76x02_dev *dev, s16 *ltssi,
|
||||
u8 *info)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
u32 val;
|
||||
|
||||
val = (chan->band == NL80211_BAND_5GHZ) ? 0x80055 : 0x80050;
|
||||
@ -696,7 +696,7 @@ mt76x0_phy_get_delta_power(struct mt76x02_dev *dev, u8 tx_mode,
|
||||
s8 target_power, s8 target_pa_power,
|
||||
s16 ltssi)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
int tssi_target = target_power << 12, tssi_slope;
|
||||
int tssi_offset, tssi_db, ret;
|
||||
u32 data;
|
||||
@ -844,12 +844,12 @@ void mt76x0_phy_set_txpower(struct mt76x02_dev *dev)
|
||||
struct mt76_rate_power *t = &dev->mt76.rate_power;
|
||||
s8 info;
|
||||
|
||||
mt76x0_get_tx_power_per_rate(dev, dev->mt76.chandef.chan, t);
|
||||
mt76x0_get_power_info(dev, dev->mt76.chandef.chan, &info);
|
||||
mt76x0_get_tx_power_per_rate(dev, dev->mphy.chandef.chan, t);
|
||||
mt76x0_get_power_info(dev, dev->mphy.chandef.chan, &info);
|
||||
|
||||
mt76x02_add_rate_power_offset(t, info);
|
||||
mt76x02_limit_rate_power(t, dev->mt76.txpower_conf);
|
||||
dev->mt76.txpower_cur = mt76x02_get_max_rate_power(t);
|
||||
mt76x02_limit_rate_power(t, dev->txpower_conf);
|
||||
dev->mphy.txpower_cur = mt76x02_get_max_rate_power(t);
|
||||
mt76x02_add_rate_power_offset(t, -info);
|
||||
|
||||
dev->target_power = info;
|
||||
@ -858,7 +858,7 @@ void mt76x0_phy_set_txpower(struct mt76x02_dev *dev)
|
||||
|
||||
void mt76x0_phy_calibrate(struct mt76x02_dev *dev, bool power_on)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
int is_5ghz = (chan->band == NL80211_BAND_5GHZ) ? 1 : 0;
|
||||
u32 val, tx_alc, reg_val;
|
||||
|
||||
@ -933,7 +933,7 @@ void mt76x0_phy_set_channel(struct mt76x02_dev *dev,
|
||||
FIELD_PREP(MT_EXT_CCA_CFG_CCA3, 0) |
|
||||
FIELD_PREP(MT_EXT_CCA_CFG_CCA_MASK, BIT(3)),
|
||||
};
|
||||
bool scan = test_bit(MT76_SCANNING, &dev->mt76.state);
|
||||
bool scan = test_bit(MT76_SCANNING, &dev->mphy.state);
|
||||
int ch_group_index, freq, freq1;
|
||||
u8 channel;
|
||||
u32 val;
|
||||
@ -1037,7 +1037,7 @@ static void mt76x0_phy_temp_sensor(struct mt76x02_dev *dev)
|
||||
|
||||
if (abs(val - dev->cal.temp_vco) > 20) {
|
||||
mt76x02_mcu_calibrate(dev, MCU_CAL_VCO,
|
||||
dev->mt76.chandef.chan->hw_value);
|
||||
dev->mphy.chandef.chan->hw_value);
|
||||
dev->cal.temp_vco = val;
|
||||
}
|
||||
if (abs(val - dev->cal.temp) > 30) {
|
||||
@ -1057,7 +1057,7 @@ static void mt76x0_phy_set_gain_val(struct mt76x02_dev *dev)
|
||||
|
||||
mt76_rmw_field(dev, MT_BBP(AGC, 8), MT_BBP_AGC_GAIN, gain);
|
||||
|
||||
if ((dev->mt76.chandef.chan->flags & IEEE80211_CHAN_RADAR) &&
|
||||
if ((dev->mphy.chandef.chan->flags & IEEE80211_CHAN_RADAR) &&
|
||||
!is_mt7630(dev))
|
||||
mt76x02_phy_dfs_adjust_agc(dev);
|
||||
}
|
||||
@ -1069,7 +1069,7 @@ mt76x0_phy_update_channel_gain(struct mt76x02_dev *dev)
|
||||
u8 gain_delta;
|
||||
int low_gain;
|
||||
|
||||
dev->cal.avg_rssi_all = mt76_get_min_avg_rssi(&dev->mt76);
|
||||
dev->cal.avg_rssi_all = mt76_get_min_avg_rssi(&dev->mt76, false);
|
||||
if (!dev->cal.avg_rssi_all)
|
||||
dev->cal.avg_rssi_all = -75;
|
||||
|
||||
|
@ -71,7 +71,7 @@ static void mt76x0_init_usb_dma(struct mt76x02_dev *dev)
|
||||
|
||||
static void mt76x0u_cleanup(struct mt76x02_dev *dev)
|
||||
{
|
||||
clear_bit(MT76_STATE_INITIALIZED, &dev->mt76.state);
|
||||
clear_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);
|
||||
mt76x0_chip_onoff(dev, false, false);
|
||||
mt76u_queues_deinit(&dev->mt76);
|
||||
}
|
||||
@ -80,13 +80,13 @@ static void mt76x0u_stop(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct mt76x02_dev *dev = hw->priv;
|
||||
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
cancel_delayed_work_sync(&dev->cal_work);
|
||||
cancel_delayed_work_sync(&dev->mt76.mac_work);
|
||||
mt76u_stop_tx(&dev->mt76);
|
||||
mt76x02u_exit_beacon_config(dev);
|
||||
|
||||
if (test_bit(MT76_REMOVED, &dev->mt76.state))
|
||||
if (test_bit(MT76_REMOVED, &dev->mphy.state))
|
||||
return;
|
||||
|
||||
if (!mt76_poll(dev, MT_USB_DMA_CFG, MT_USB_DMA_CFG_TX_BUSY, 0, 1000))
|
||||
@ -112,7 +112,7 @@ static int mt76x0u_start(struct ieee80211_hw *hw)
|
||||
MT_MAC_WORK_INTERVAL);
|
||||
ieee80211_queue_delayed_work(dev->mt76.hw, &dev->cal_work,
|
||||
MT_CALIBRATE_INTERVAL);
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -126,6 +126,7 @@ static const struct ieee80211_ops mt76x0u_ops = {
|
||||
.configure_filter = mt76x02_configure_filter,
|
||||
.bss_info_changed = mt76x02_bss_info_changed,
|
||||
.sta_state = mt76_sta_state,
|
||||
.sta_pre_rcu_remove = mt76_sta_pre_rcu_remove,
|
||||
.set_key = mt76x02_set_key,
|
||||
.conf_tx = mt76x02_conf_tx,
|
||||
.sw_scan_start = mt76_sw_scan,
|
||||
@ -172,8 +173,14 @@ static int mt76x0u_init_hardware(struct mt76x02_dev *dev, bool reset)
|
||||
static int mt76x0u_register_device(struct mt76x02_dev *dev)
|
||||
{
|
||||
struct ieee80211_hw *hw = dev->mt76.hw;
|
||||
struct mt76_usb *usb = &dev->mt76.usb;
|
||||
int err;
|
||||
|
||||
usb->mcu.data = devm_kmalloc(dev->mt76.dev, MCU_RESP_URB_SIZE,
|
||||
GFP_KERNEL);
|
||||
if (!usb->mcu.data)
|
||||
return -ENOMEM;
|
||||
|
||||
err = mt76u_alloc_queues(&dev->mt76);
|
||||
if (err < 0)
|
||||
goto out_err;
|
||||
@ -182,17 +189,13 @@ static int mt76x0u_register_device(struct mt76x02_dev *dev)
|
||||
if (err < 0)
|
||||
goto out_err;
|
||||
|
||||
/* check hw sg support in order to enable AMSDU */
|
||||
hw->max_tx_fragments = dev->mt76.usb.sg_en ? MT_TX_SG_MAX_SIZE : 1;
|
||||
err = mt76x0_register_device(dev);
|
||||
if (err < 0)
|
||||
goto out_err;
|
||||
|
||||
/* check hw sg support in order to enable AMSDU */
|
||||
if (dev->mt76.usb.sg_en)
|
||||
hw->max_tx_fragments = MT_TX_SG_MAX_SIZE;
|
||||
else
|
||||
hw->max_tx_fragments = 1;
|
||||
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);
|
||||
|
||||
return 0;
|
||||
|
||||
@ -240,7 +243,7 @@ static int mt76x0u_probe(struct usb_interface *usb_intf,
|
||||
usb_set_intfdata(usb_intf, dev);
|
||||
|
||||
mt76x02u_init_mcu(mdev);
|
||||
ret = mt76u_init(mdev, usb_intf);
|
||||
ret = mt76u_init(mdev, usb_intf, false);
|
||||
if (ret)
|
||||
goto err;
|
||||
|
||||
@ -283,7 +286,7 @@ err:
|
||||
static void mt76x0_disconnect(struct usb_interface *usb_intf)
|
||||
{
|
||||
struct mt76x02_dev *dev = usb_get_intfdata(usb_intf);
|
||||
bool initialized = test_bit(MT76_STATE_INITIALIZED, &dev->mt76.state);
|
||||
bool initialized = test_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);
|
||||
|
||||
if (!initialized)
|
||||
return;
|
||||
@ -304,7 +307,7 @@ static int __maybe_unused mt76x0_suspend(struct usb_interface *usb_intf,
|
||||
struct mt76x02_dev *dev = usb_get_intfdata(usb_intf);
|
||||
|
||||
mt76u_stop_rx(&dev->mt76);
|
||||
clear_bit(MT76_STATE_MCU_RUNNING, &dev->mt76.state);
|
||||
clear_bit(MT76_STATE_MCU_RUNNING, &dev->mphy.state);
|
||||
mt76x0_chip_onoff(dev, false, false);
|
||||
|
||||
return 0;
|
||||
|
@ -168,7 +168,7 @@ int mt76x0u_mcu_init(struct mt76x02_dev *dev)
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
set_bit(MT76_STATE_MCU_RUNNING, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_MCU_RUNNING, &dev->mphy.state);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -70,18 +70,23 @@ struct mt76x02_beacon_ops {
|
||||
(dev)->beacon_ops->pre_tbtt_enable(dev, enable)
|
||||
|
||||
struct mt76x02_dev {
|
||||
struct mt76_dev mt76; /* must be first */
|
||||
union { /* must be first */
|
||||
struct mt76_dev mt76;
|
||||
struct mt76_phy mphy;
|
||||
};
|
||||
|
||||
struct mac_address macaddr_list[8];
|
||||
|
||||
struct mutex phy_mutex;
|
||||
|
||||
u16 vif_mask;
|
||||
u16 chainmask;
|
||||
|
||||
u8 txdone_seq;
|
||||
DECLARE_KFIFO_PTR(txstatus_fifo, struct mt76x02_tx_status);
|
||||
spinlock_t txstatus_fifo_lock;
|
||||
u32 tx_airtime;
|
||||
u32 ampdu_ref;
|
||||
|
||||
struct sk_buff *rx_head;
|
||||
|
||||
@ -93,8 +98,7 @@ struct mt76x02_dev {
|
||||
|
||||
const struct mt76x02_beacon_ops *beacon_ops;
|
||||
|
||||
struct sk_buff *beacons[8];
|
||||
u8 beacon_data_mask;
|
||||
u8 beacon_data_count;
|
||||
|
||||
u8 tbtt_count;
|
||||
|
||||
@ -104,13 +108,14 @@ struct mt76x02_dev {
|
||||
|
||||
struct mt76x02_calibration cal;
|
||||
|
||||
int txpower_conf;
|
||||
s8 target_power;
|
||||
s8 target_power_delta[2];
|
||||
bool enable_tpc;
|
||||
|
||||
bool no_2ghz;
|
||||
|
||||
u8 coverage_class;
|
||||
s16 coverage_class;
|
||||
u8 slottime;
|
||||
|
||||
struct mt76x02_dfs_pattern_detector dfs_pd;
|
||||
|
@ -26,78 +26,40 @@ static int
|
||||
mt76x02_write_beacon(struct mt76x02_dev *dev, int offset, struct sk_buff *skb)
|
||||
{
|
||||
int beacon_len = dev->beacon_ops->slot_size;
|
||||
struct mt76x02_txwi txwi;
|
||||
|
||||
if (WARN_ON_ONCE(beacon_len < skb->len + sizeof(struct mt76x02_txwi)))
|
||||
return -ENOSPC;
|
||||
|
||||
mt76x02_mac_write_txwi(dev, &txwi, skb, NULL, NULL, skb->len);
|
||||
/* USB devices already reserve enough skb headroom for txwi's. This
|
||||
* helps to save slow copies over USB.
|
||||
*/
|
||||
if (mt76_is_usb(&dev->mt76)) {
|
||||
struct mt76x02_txwi *txwi;
|
||||
|
||||
mt76_wr_copy(dev, offset, &txwi, sizeof(txwi));
|
||||
offset += sizeof(txwi);
|
||||
txwi = (struct mt76x02_txwi *)(skb->data - sizeof(*txwi));
|
||||
mt76x02_mac_write_txwi(dev, txwi, skb, NULL, NULL, skb->len);
|
||||
skb_push(skb, sizeof(*txwi));
|
||||
} else {
|
||||
struct mt76x02_txwi txwi;
|
||||
|
||||
mt76x02_mac_write_txwi(dev, &txwi, skb, NULL, NULL, skb->len);
|
||||
mt76_wr_copy(dev, offset, &txwi, sizeof(txwi));
|
||||
offset += sizeof(txwi);
|
||||
}
|
||||
|
||||
mt76_wr_copy(dev, offset, skb->data, skb->len);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
__mt76x02_mac_set_beacon(struct mt76x02_dev *dev, u8 bcn_idx,
|
||||
struct sk_buff *skb)
|
||||
void mt76x02_mac_set_beacon(struct mt76x02_dev *dev,
|
||||
struct sk_buff *skb)
|
||||
{
|
||||
int beacon_len = dev->beacon_ops->slot_size;
|
||||
int beacon_addr = MT_BEACON_BASE + (beacon_len * bcn_idx);
|
||||
int ret = 0;
|
||||
int i;
|
||||
int bcn_len = dev->beacon_ops->slot_size;
|
||||
int bcn_addr = MT_BEACON_BASE + (bcn_len * dev->beacon_data_count);
|
||||
|
||||
/* Prevent corrupt transmissions during update */
|
||||
mt76_set(dev, MT_BCN_BYPASS_MASK, BIT(bcn_idx));
|
||||
|
||||
if (skb) {
|
||||
ret = mt76x02_write_beacon(dev, beacon_addr, skb);
|
||||
if (!ret)
|
||||
dev->beacon_data_mask |= BIT(bcn_idx);
|
||||
} else {
|
||||
dev->beacon_data_mask &= ~BIT(bcn_idx);
|
||||
for (i = 0; i < beacon_len; i += 4)
|
||||
mt76_wr(dev, beacon_addr + i, 0);
|
||||
}
|
||||
|
||||
mt76_wr(dev, MT_BCN_BYPASS_MASK, 0xff00 | ~dev->beacon_data_mask);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
int mt76x02_mac_set_beacon(struct mt76x02_dev *dev, u8 vif_idx,
|
||||
struct sk_buff *skb)
|
||||
{
|
||||
bool force_update = false;
|
||||
int bcn_idx = 0;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->beacons); i++) {
|
||||
if (vif_idx == i) {
|
||||
force_update = !!dev->beacons[i] ^ !!skb;
|
||||
dev_kfree_skb(dev->beacons[i]);
|
||||
dev->beacons[i] = skb;
|
||||
__mt76x02_mac_set_beacon(dev, bcn_idx, skb);
|
||||
} else if (force_update && dev->beacons[i]) {
|
||||
__mt76x02_mac_set_beacon(dev, bcn_idx,
|
||||
dev->beacons[i]);
|
||||
}
|
||||
|
||||
bcn_idx += !!dev->beacons[i];
|
||||
}
|
||||
|
||||
for (i = bcn_idx; i < ARRAY_SIZE(dev->beacons); i++) {
|
||||
if (!(dev->beacon_data_mask & BIT(i)))
|
||||
break;
|
||||
|
||||
__mt76x02_mac_set_beacon(dev, i, NULL);
|
||||
}
|
||||
|
||||
mt76_rmw_field(dev, MT_MAC_BSSID_DW1, MT_MAC_BSSID_DW1_MBEACON_N,
|
||||
bcn_idx - 1);
|
||||
return 0;
|
||||
if (!mt76x02_write_beacon(dev, bcn_addr, skb))
|
||||
dev->beacon_data_count++;
|
||||
dev_kfree_skb(skb);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76x02_mac_set_beacon);
|
||||
|
||||
@ -116,7 +78,6 @@ void mt76x02_mac_set_beacon_enable(struct mt76x02_dev *dev,
|
||||
dev->mt76.beacon_mask |= BIT(mvif->idx);
|
||||
} else {
|
||||
dev->mt76.beacon_mask &= ~BIT(mvif->idx);
|
||||
mt76x02_mac_set_beacon(dev, mvif->idx, NULL);
|
||||
}
|
||||
|
||||
if (!!old_mask == !!dev->mt76.beacon_mask)
|
||||
@ -182,7 +143,7 @@ mt76x02_update_beacon_iter(void *priv, u8 *mac, struct ieee80211_vif *vif)
|
||||
if (!skb)
|
||||
return;
|
||||
|
||||
mt76x02_mac_set_beacon(dev, mvif->idx, skb);
|
||||
mt76x02_mac_set_beacon(dev, skb);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76x02_update_beacon_iter);
|
||||
|
||||
@ -241,17 +202,11 @@ EXPORT_SYMBOL_GPL(mt76x02_enqueue_buffered_bc);
|
||||
|
||||
void mt76x02_init_beacon_config(struct mt76x02_dev *dev)
|
||||
{
|
||||
int i;
|
||||
|
||||
mt76_clear(dev, MT_BEACON_TIME_CFG, (MT_BEACON_TIME_CFG_TIMER_EN |
|
||||
MT_BEACON_TIME_CFG_TBTT_EN |
|
||||
MT_BEACON_TIME_CFG_BEACON_TX));
|
||||
mt76_set(dev, MT_BEACON_TIME_CFG, MT_BEACON_TIME_CFG_SYNC_MODE);
|
||||
mt76_wr(dev, MT_BCN_BYPASS_MASK, 0xffff);
|
||||
|
||||
for (i = 0; i < 8; i++)
|
||||
mt76x02_mac_set_beacon(dev, i, NULL);
|
||||
|
||||
mt76x02_set_beacon_offsets(dev);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76x02_init_beacon_config);
|
||||
|
@ -307,8 +307,8 @@ static bool mt76x02_dfs_check_hw_pulse(struct mt76x02_dev *dev,
|
||||
pulse->period <= 100100);
|
||||
break;
|
||||
case NL80211_DFS_JP:
|
||||
if (dev->mt76.chandef.chan->center_freq >= 5250 &&
|
||||
dev->mt76.chandef.chan->center_freq <= 5350) {
|
||||
if (dev->mphy.chandef.chan->center_freq >= 5250 &&
|
||||
dev->mphy.chandef.chan->center_freq <= 5350) {
|
||||
/* JPW53 */
|
||||
if (pulse->w1 <= 130)
|
||||
ret = (pulse->period >= 28360 &&
|
||||
@ -616,7 +616,7 @@ static void mt76x02_dfs_tasklet(unsigned long arg)
|
||||
u32 engine_mask;
|
||||
int i;
|
||||
|
||||
if (test_bit(MT76_SCANNING, &dev->mt76.state))
|
||||
if (test_bit(MT76_SCANNING, &dev->mphy.state))
|
||||
goto out;
|
||||
|
||||
if (time_is_before_jiffies(dfs_pd->last_sw_check +
|
||||
@ -702,7 +702,7 @@ static void mt76x02_dfs_set_bbp_params(struct mt76x02_dev *dev)
|
||||
u8 i, shift;
|
||||
u32 data;
|
||||
|
||||
switch (dev->mt76.chandef.width) {
|
||||
switch (dev->mphy.chandef.width) {
|
||||
case NL80211_CHAN_WIDTH_40:
|
||||
shift = MT_DFS_NUM_ENGINES;
|
||||
break;
|
||||
@ -722,8 +722,8 @@ static void mt76x02_dfs_set_bbp_params(struct mt76x02_dev *dev)
|
||||
radar_specs = &etsi_radar_specs[shift];
|
||||
break;
|
||||
case NL80211_DFS_JP:
|
||||
if (dev->mt76.chandef.chan->center_freq >= 5250 &&
|
||||
dev->mt76.chandef.chan->center_freq <= 5350)
|
||||
if (dev->mphy.chandef.chan->center_freq >= 5250 &&
|
||||
dev->mphy.chandef.chan->center_freq <= 5350)
|
||||
radar_specs = &jp_w53_radar_specs[shift];
|
||||
else
|
||||
radar_specs = &jp_w56_radar_specs[shift];
|
||||
@ -822,7 +822,7 @@ EXPORT_SYMBOL_GPL(mt76x02_phy_dfs_adjust_agc);
|
||||
|
||||
void mt76x02_dfs_init_params(struct mt76x02_dev *dev)
|
||||
{
|
||||
struct cfg80211_chan_def *chandef = &dev->mt76.chandef;
|
||||
struct cfg80211_chan_def *chandef = &dev->mphy.chandef;
|
||||
|
||||
if ((chandef->chan->flags & IEEE80211_CHAN_RADAR) &&
|
||||
dev->mt76.region != NL80211_DFS_UNSET) {
|
||||
|
@ -6,6 +6,7 @@
|
||||
|
||||
#include "mt76x02.h"
|
||||
#include "mt76x02_trace.h"
|
||||
#include "trace.h"
|
||||
|
||||
void mt76x02_mac_reset_counters(struct mt76x02_dev *dev)
|
||||
{
|
||||
@ -200,7 +201,7 @@ mt76x02_mac_tx_rate_val(struct mt76x02_dev *dev,
|
||||
bw = 1;
|
||||
} else {
|
||||
const struct ieee80211_rate *r;
|
||||
int band = dev->mt76.chandef.chan->band;
|
||||
int band = dev->mphy.chandef.chan->band;
|
||||
u16 val;
|
||||
|
||||
r = &dev->mt76.hw->wiphy->bands[band]->bitrates[rate->idx];
|
||||
@ -344,7 +345,7 @@ void mt76x02_mac_write_txwi(struct mt76x02_dev *dev, struct mt76x02_txwi *txwi,
|
||||
u16 txwi_flags = 0;
|
||||
u8 nss;
|
||||
s8 txpwr_adj, max_txpwr_adj;
|
||||
u8 ccmp_pn[8], nstreams = dev->mt76.chainmask & 0xf;
|
||||
u8 ccmp_pn[8], nstreams = dev->chainmask & 0xf;
|
||||
|
||||
memset(txwi, 0, sizeof(*txwi));
|
||||
|
||||
@ -386,7 +387,7 @@ void mt76x02_mac_write_txwi(struct mt76x02_dev *dev, struct mt76x02_txwi *txwi,
|
||||
max_txpwr_adj = mt76x02_tx_get_max_txpwr_adj(dev, rate);
|
||||
}
|
||||
|
||||
txpwr_adj = mt76x02_tx_get_txpwr_adj(dev, dev->mt76.txpower_conf,
|
||||
txpwr_adj = mt76x02_tx_get_txpwr_adj(dev, dev->txpower_conf,
|
||||
max_txpwr_adj);
|
||||
txwi->ctl2 = FIELD_PREP(MT_TX_PWR_ADJ, txpwr_adj);
|
||||
|
||||
@ -487,17 +488,17 @@ mt76x02_mac_fill_tx_status(struct mt76x02_dev *dev, struct mt76x02_sta *msta,
|
||||
first_rate |= st->pktid & MT_PKTID_RATE;
|
||||
|
||||
mt76x02_mac_process_tx_rate(&rate[0], first_rate,
|
||||
dev->mt76.chandef.chan->band);
|
||||
dev->mphy.chandef.chan->band);
|
||||
} else if (rate[0].idx < 0) {
|
||||
if (!msta)
|
||||
return;
|
||||
|
||||
mt76x02_mac_process_tx_rate(&rate[0], msta->wcid.tx_info,
|
||||
dev->mt76.chandef.chan->band);
|
||||
dev->mphy.chandef.chan->band);
|
||||
}
|
||||
|
||||
mt76x02_mac_process_tx_rate(&last_rate, st->rate,
|
||||
dev->mt76.chandef.chan->band);
|
||||
dev->mphy.chandef.chan->band);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(info->status.rates); i++) {
|
||||
retry--;
|
||||
@ -630,7 +631,7 @@ void mt76x02_send_tx_status(struct mt76x02_dev *dev,
|
||||
if (!len)
|
||||
goto out;
|
||||
|
||||
duration = mt76_calc_tx_airtime(&dev->mt76, &info, len);
|
||||
duration = ieee80211_calc_tx_airtime(mt76_hw(dev), &info, len);
|
||||
|
||||
spin_lock_bh(&dev->mt76.cc_lock);
|
||||
dev->tx_airtime += duration;
|
||||
@ -679,7 +680,7 @@ mt76x02_mac_process_rate(struct mt76x02_dev *dev,
|
||||
status->rate_idx = idx;
|
||||
break;
|
||||
case MT_PHY_TYPE_VHT: {
|
||||
u8 n_rxstream = dev->mt76.chainmask & 0xf;
|
||||
u8 n_rxstream = dev->chainmask & 0xf;
|
||||
|
||||
status->encoding = RX_ENC_VHT;
|
||||
status->rate_idx = FIELD_GET(MT_RATE_INDEX_VHT_IDX, idx);
|
||||
@ -741,6 +742,8 @@ void mt76x02_mac_setaddr(struct mt76x02_dev *dev, const u8 *addr)
|
||||
get_unaligned_le16(dev->mt76.macaddr + 4) |
|
||||
FIELD_PREP(MT_MAC_BSSID_DW1_MBSS_MODE, 3) | /* 8 APs + 8 STAs */
|
||||
MT_MAC_BSSID_DW1_MBSS_LOCAL_BIT);
|
||||
/* enable 7 additional beacon slots and control them with bypass mask */
|
||||
mt76_rmw_field(dev, MT_MAC_BSSID_DW1, MT_MAC_BSSID_DW1_MBEACON_N, 7);
|
||||
|
||||
for (i = 0; i < 16; i++)
|
||||
mt76x02_mac_set_bssid(dev, i, null_addr);
|
||||
@ -769,13 +772,13 @@ int mt76x02_mac_process_rx(struct mt76x02_dev *dev, struct sk_buff *skb,
|
||||
u16 rate = le16_to_cpu(rxwi->rate);
|
||||
u16 tid_sn = le16_to_cpu(rxwi->tid_sn);
|
||||
bool unicast = rxwi->rxinfo & cpu_to_le32(MT_RXINFO_UNICAST);
|
||||
int pad_len = 0, nstreams = dev->mt76.chainmask & 0xf;
|
||||
int pad_len = 0, nstreams = dev->chainmask & 0xf;
|
||||
s8 signal;
|
||||
u8 pn_len;
|
||||
u8 wcid;
|
||||
int len;
|
||||
|
||||
if (!test_bit(MT76_STATE_RUNNING, &dev->mt76.state))
|
||||
if (!test_bit(MT76_STATE_RUNNING, &dev->mphy.state))
|
||||
return -EINVAL;
|
||||
|
||||
if (rxinfo & MT_RXINFO_L2PAD)
|
||||
@ -824,7 +827,7 @@ int mt76x02_mac_process_rx(struct mt76x02_dev *dev, struct sk_buff *skb,
|
||||
|
||||
if (rxinfo & MT_RXINFO_AMPDU) {
|
||||
status->flag |= RX_FLAG_AMPDU_DETAILS;
|
||||
status->ampdu_ref = dev->mt76.ampdu_ref;
|
||||
status->ampdu_ref = dev->ampdu_ref;
|
||||
|
||||
/*
|
||||
* When receiving an A-MPDU subframe and RSSI info is not valid,
|
||||
@ -832,8 +835,8 @@ int mt76x02_mac_process_rx(struct mt76x02_dev *dev, struct sk_buff *skb,
|
||||
* are coming. The last one will have valid RSSI info
|
||||
*/
|
||||
if (rxinfo & MT_RXINFO_RSSI) {
|
||||
if (!++dev->mt76.ampdu_ref)
|
||||
dev->mt76.ampdu_ref++;
|
||||
if (!++dev->ampdu_ref)
|
||||
dev->ampdu_ref++;
|
||||
}
|
||||
}
|
||||
|
||||
@ -853,8 +856,8 @@ int mt76x02_mac_process_rx(struct mt76x02_dev *dev, struct sk_buff *skb,
|
||||
signal = max_t(s8, signal, status->chain_signal[1]);
|
||||
}
|
||||
status->signal = signal;
|
||||
status->freq = dev->mt76.chandef.chan->center_freq;
|
||||
status->band = dev->mt76.chandef.chan->band;
|
||||
status->freq = dev->mphy.chandef.chan->center_freq;
|
||||
status->band = dev->mphy.chandef.chan->band;
|
||||
|
||||
status->tid = FIELD_GET(MT_RXWI_TID, tid_sn);
|
||||
status->seqno = FIELD_GET(MT_RXWI_SN, tid_sn);
|
||||
@ -868,7 +871,7 @@ void mt76x02_mac_poll_tx_status(struct mt76x02_dev *dev, bool irq)
|
||||
u8 update = 1;
|
||||
bool ret;
|
||||
|
||||
if (!test_bit(MT76_STATE_RUNNING, &dev->mt76.state))
|
||||
if (!test_bit(MT76_STATE_RUNNING, &dev->mphy.state))
|
||||
return;
|
||||
|
||||
trace_mac_txstat_poll(dev);
|
||||
@ -908,7 +911,7 @@ void mt76x02_tx_complete_skb(struct mt76_dev *mdev, enum mt76_txq_id qid,
|
||||
|
||||
txwi_ptr = mt76_get_txwi_ptr(mdev, e->txwi);
|
||||
txwi = (struct mt76x02_txwi *)txwi_ptr;
|
||||
trace_mac_txdone_add(dev, txwi->wcid, txwi->pktid);
|
||||
trace_mac_txdone(mdev, txwi->wcid, txwi->pktid);
|
||||
|
||||
mt76_tx_complete_skb(mdev, e->skb);
|
||||
}
|
||||
@ -1018,7 +1021,7 @@ void mt76x02_update_channel(struct mt76_dev *mdev)
|
||||
struct mt76x02_dev *dev = container_of(mdev, struct mt76x02_dev, mt76);
|
||||
struct mt76_channel_state *state;
|
||||
|
||||
state = mdev->chan_state;
|
||||
state = mdev->phy.chan_state;
|
||||
state->cc_busy += mt76_rr(dev, MT_CH_BUSY);
|
||||
|
||||
spin_lock_bh(&dev->mt76.cc_lock);
|
||||
@ -1074,7 +1077,7 @@ void mt76x02_edcca_init(struct mt76x02_dev *dev)
|
||||
dev->ed_silent = 0;
|
||||
|
||||
if (dev->ed_monitor) {
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
u8 ed_th = chan->band == NL80211_BAND_5GHZ ? 0x0e : 0x20;
|
||||
|
||||
mt76_clear(dev, MT_TX_LINK_CFG, MT_TX_CFACK_EN);
|
||||
@ -1184,7 +1187,7 @@ void mt76x02_mac_work(struct work_struct *work)
|
||||
|
||||
void mt76x02_mac_cc_reset(struct mt76x02_dev *dev)
|
||||
{
|
||||
dev->mt76.survey_time = ktime_get_boottime();
|
||||
dev->mphy.survey_time = ktime_get_boottime();
|
||||
|
||||
mt76_wr(dev, MT_CH_TIME_CFG,
|
||||
MT_CH_TIME_CFG_TIMER_EN |
|
||||
|
@ -152,7 +152,7 @@ static inline bool mt76x02_wait_for_mac(struct mt76_dev *dev)
|
||||
int i;
|
||||
|
||||
for (i = 0; i < 500; i++) {
|
||||
if (test_bit(MT76_REMOVED, &dev->state))
|
||||
if (test_bit(MT76_REMOVED, &dev->phy.state))
|
||||
return false;
|
||||
|
||||
switch (dev->bus->rr(dev, MAC_CSR0)) {
|
||||
@ -201,8 +201,7 @@ void mt76x02_mac_work(struct work_struct *work);
|
||||
|
||||
void mt76x02_mac_cc_reset(struct mt76x02_dev *dev);
|
||||
void mt76x02_mac_set_bssid(struct mt76x02_dev *dev, u8 idx, const u8 *addr);
|
||||
int mt76x02_mac_set_beacon(struct mt76x02_dev *dev, u8 vif_idx,
|
||||
struct sk_buff *skb);
|
||||
void mt76x02_mac_set_beacon(struct mt76x02_dev *dev, struct sk_buff *skb);
|
||||
void mt76x02_mac_set_beacon_enable(struct mt76x02_dev *dev,
|
||||
struct ieee80211_vif *vif, bool enable);
|
||||
|
||||
|
@ -24,11 +24,11 @@ int mt76x02_mcu_msg_send(struct mt76_dev *mdev, int cmd, const void *data,
|
||||
if (!skb)
|
||||
return -ENOMEM;
|
||||
|
||||
mutex_lock(&mdev->mmio.mcu.mutex);
|
||||
mutex_lock(&mdev->mcu.mutex);
|
||||
|
||||
seq = ++mdev->mmio.mcu.msg_seq & 0xf;
|
||||
seq = ++mdev->mcu.msg_seq & 0xf;
|
||||
if (!seq)
|
||||
seq = ++mdev->mmio.mcu.msg_seq & 0xf;
|
||||
seq = ++mdev->mcu.msg_seq & 0xf;
|
||||
|
||||
tx_info = MT_MCU_MSG_TYPE_CMD |
|
||||
FIELD_PREP(MT_MCU_MSG_CMD_TYPE, cmd) |
|
||||
@ -65,7 +65,7 @@ int mt76x02_mcu_msg_send(struct mt76_dev *mdev, int cmd, const void *data,
|
||||
}
|
||||
|
||||
out:
|
||||
mutex_unlock(&mdev->mmio.mcu.mutex);
|
||||
mutex_unlock(&mdev->mcu.mutex);
|
||||
|
||||
return ret;
|
||||
}
|
||||
@ -141,7 +141,7 @@ int mt76x02_mcu_cleanup(struct mt76x02_dev *dev)
|
||||
mt76_wr(dev, MT_MCU_INT_LEVEL, 1);
|
||||
usleep_range(20000, 30000);
|
||||
|
||||
while ((skb = skb_dequeue(&dev->mt76.mmio.mcu.res_q)) != NULL)
|
||||
while ((skb = skb_dequeue(&dev->mt76.mcu.res_q)) != NULL)
|
||||
dev_kfree_skb(skb);
|
||||
|
||||
return 0;
|
||||
|
@ -9,7 +9,7 @@
|
||||
|
||||
#include "mt76x02.h"
|
||||
#include "mt76x02_mcu.h"
|
||||
#include "mt76x02_trace.h"
|
||||
#include "trace.h"
|
||||
|
||||
static void mt76x02_pre_tbtt_tasklet(unsigned long arg)
|
||||
{
|
||||
@ -24,10 +24,17 @@ static void mt76x02_pre_tbtt_tasklet(unsigned long arg)
|
||||
|
||||
mt76x02_resync_beacon_timer(dev);
|
||||
|
||||
/* Prevent corrupt transmissions during update */
|
||||
mt76_set(dev, MT_BCN_BYPASS_MASK, 0xffff);
|
||||
dev->beacon_data_count = 0;
|
||||
|
||||
ieee80211_iterate_active_interfaces_atomic(mt76_hw(dev),
|
||||
IEEE80211_IFACE_ITER_RESUME_ALL,
|
||||
mt76x02_update_beacon_iter, dev);
|
||||
|
||||
mt76_wr(dev, MT_BCN_BYPASS_MASK,
|
||||
0xff00 | ~(0xff00 >> dev->beacon_data_count));
|
||||
|
||||
mt76_csa_check(&dev->mt76);
|
||||
|
||||
if (dev->mt76.csa_complete)
|
||||
@ -151,7 +158,7 @@ static void mt76x02_tx_tasklet(unsigned long data)
|
||||
mt76x02_mac_poll_tx_status(dev, false);
|
||||
mt76x02_process_tx_status_fifo(dev);
|
||||
|
||||
mt76_txq_schedule_all(&dev->mt76);
|
||||
mt76_txq_schedule_all(&dev->mphy);
|
||||
}
|
||||
|
||||
static int mt76x02_poll_tx(struct napi_struct *napi, int budget)
|
||||
@ -261,10 +268,10 @@ irqreturn_t mt76x02_irq_handler(int irq, void *dev_instance)
|
||||
intr = mt76_rr(dev, MT_INT_SOURCE_CSR);
|
||||
mt76_wr(dev, MT_INT_SOURCE_CSR, intr);
|
||||
|
||||
if (!test_bit(MT76_STATE_INITIALIZED, &dev->mt76.state))
|
||||
if (!test_bit(MT76_STATE_INITIALIZED, &dev->mphy.state))
|
||||
return IRQ_NONE;
|
||||
|
||||
trace_dev_irq(dev, intr, dev->mt76.mmio.irqmask);
|
||||
trace_dev_irq(&dev->mt76, intr, dev->mt76.mmio.irqmask);
|
||||
|
||||
intr &= dev->mt76.mmio.irqmask;
|
||||
|
||||
@ -402,7 +409,7 @@ static void mt76x02_reset_state(struct mt76x02_dev *dev)
|
||||
|
||||
lockdep_assert_held(&dev->mt76.mutex);
|
||||
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
|
||||
rcu_read_lock();
|
||||
ieee80211_iter_keys_rcu(dev->mt76.hw, NULL, mt76x02_key_sync, NULL);
|
||||
@ -420,6 +427,8 @@ static void mt76x02_reset_state(struct mt76x02_dev *dev)
|
||||
if (!wcid)
|
||||
continue;
|
||||
|
||||
rcu_assign_pointer(dev->mt76.wcid[i], NULL);
|
||||
|
||||
priv = msta = container_of(wcid, struct mt76x02_sta, wcid);
|
||||
sta = container_of(priv, struct ieee80211_sta, drv_priv);
|
||||
|
||||
@ -441,7 +450,7 @@ static void mt76x02_watchdog_reset(struct mt76x02_dev *dev)
|
||||
int i;
|
||||
|
||||
ieee80211_stop_queues(dev->mt76.hw);
|
||||
set_bit(MT76_RESET, &dev->mt76.state);
|
||||
set_bit(MT76_RESET, &dev->mphy.state);
|
||||
|
||||
tasklet_disable(&dev->mt76.pre_tbtt_tasklet);
|
||||
tasklet_disable(&dev->mt76.tx_tasklet);
|
||||
@ -476,7 +485,7 @@ static void mt76x02_watchdog_reset(struct mt76x02_dev *dev)
|
||||
if (restart)
|
||||
mt76_mcu_restart(dev);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->mt76.q_tx); i++)
|
||||
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++)
|
||||
@ -496,7 +505,7 @@ static void mt76x02_watchdog_reset(struct mt76x02_dev *dev)
|
||||
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
clear_bit(MT76_RESET, &dev->mt76.state);
|
||||
clear_bit(MT76_RESET, &dev->mphy.state);
|
||||
|
||||
tasklet_enable(&dev->mt76.tx_tasklet);
|
||||
napi_enable(&dev->mt76.tx_napi);
|
||||
@ -514,7 +523,7 @@ static void mt76x02_watchdog_reset(struct mt76x02_dev *dev)
|
||||
ieee80211_restart_hw(dev->mt76.hw);
|
||||
} else {
|
||||
ieee80211_wake_queues(dev->mt76.hw);
|
||||
mt76_txq_schedule_all(&dev->mt76);
|
||||
mt76_txq_schedule_all(&dev->mphy);
|
||||
}
|
||||
}
|
||||
|
||||
@ -535,9 +544,9 @@ static void mt76x02_check_tx_hang(struct mt76x02_dev *dev)
|
||||
restart:
|
||||
mt76x02_watchdog_reset(dev);
|
||||
|
||||
mutex_lock(&dev->mt76.mmio.mcu.mutex);
|
||||
mutex_lock(&dev->mt76.mcu.mutex);
|
||||
dev->mcu_timeout = 0;
|
||||
mutex_unlock(&dev->mt76.mmio.mcu.mutex);
|
||||
mutex_unlock(&dev->mt76.mcu.mutex);
|
||||
|
||||
dev->tx_hang_reset++;
|
||||
dev->tx_hang_check = 0;
|
||||
|
@ -16,7 +16,7 @@ void mt76x02_phy_set_rxpath(struct mt76x02_dev *dev)
|
||||
val = mt76_rr(dev, MT_BBP(AGC, 0));
|
||||
val &= ~BIT(4);
|
||||
|
||||
switch (dev->mt76.chainmask & 0xf) {
|
||||
switch (dev->chainmask & 0xf) {
|
||||
case 2:
|
||||
val |= BIT(3);
|
||||
break;
|
||||
@ -35,7 +35,7 @@ void mt76x02_phy_set_txdac(struct mt76x02_dev *dev)
|
||||
{
|
||||
int txpath;
|
||||
|
||||
txpath = (dev->mt76.chainmask >> 8) & 0xf;
|
||||
txpath = (dev->chainmask >> 8) & 0xf;
|
||||
switch (txpath) {
|
||||
case 2:
|
||||
mt76_set(dev, MT_BBP(TXBE, 5), 0x3);
|
||||
|
@ -11,7 +11,7 @@
|
||||
static inline int
|
||||
mt76x02_get_rssi_gain_thresh(struct mt76x02_dev *dev)
|
||||
{
|
||||
switch (dev->mt76.chandef.width) {
|
||||
switch (dev->mphy.chandef.width) {
|
||||
case NL80211_CHAN_WIDTH_80:
|
||||
return -62;
|
||||
case NL80211_CHAN_WIDTH_40:
|
||||
@ -24,7 +24,7 @@ mt76x02_get_rssi_gain_thresh(struct mt76x02_dev *dev)
|
||||
static inline int
|
||||
mt76x02_get_low_rssi_gain_thresh(struct mt76x02_dev *dev)
|
||||
{
|
||||
switch (dev->mt76.chandef.width) {
|
||||
switch (dev->mphy.chandef.width) {
|
||||
case NL80211_CHAN_WIDTH_80:
|
||||
return -76;
|
||||
case NL80211_CHAN_WIDTH_40:
|
||||
|
@ -235,15 +235,9 @@
|
||||
#define MT_LED_S0(_n) (MT_LED_S0_BASE + 8 * (_n))
|
||||
#define MT_LED_S1_BASE 0x0780
|
||||
#define MT_LED_S1(_n) (MT_LED_S1_BASE + 8 * (_n))
|
||||
#define MT_LED_STATUS_OFF_MASK GENMASK(31, 24)
|
||||
#define MT_LED_STATUS_OFF(_v) (((_v) << __ffs(MT_LED_STATUS_OFF_MASK)) & \
|
||||
MT_LED_STATUS_OFF_MASK)
|
||||
#define MT_LED_STATUS_ON_MASK GENMASK(23, 16)
|
||||
#define MT_LED_STATUS_ON(_v) (((_v) << __ffs(MT_LED_STATUS_ON_MASK)) & \
|
||||
MT_LED_STATUS_ON_MASK)
|
||||
#define MT_LED_STATUS_DURATION_MASK GENMASK(15, 8)
|
||||
#define MT_LED_STATUS_DURATION(_v) (((_v) << __ffs(MT_LED_STATUS_DURATION_MASK)) & \
|
||||
MT_LED_STATUS_DURATION_MASK)
|
||||
#define MT_LED_STATUS_OFF GENMASK(31, 24)
|
||||
#define MT_LED_STATUS_ON GENMASK(23, 16)
|
||||
#define MT_LED_STATUS_DURATION GENMASK(15, 8)
|
||||
|
||||
#define MT_FCE_PSE_CTRL 0x0800
|
||||
#define MT_FCE_PARAMETERS 0x0804
|
||||
|
@ -20,7 +20,6 @@
|
||||
#define DEV_PR_ARG __entry->wiphy_name
|
||||
|
||||
#define TXID_ENTRY __field(u8, wcid) __field(u8, pktid)
|
||||
#define TXID_ASSIGN __entry->wcid = wcid; __entry->pktid = pktid
|
||||
#define TXID_PR_FMT " [%d:%d]"
|
||||
#define TXID_PR_ARG __entry->wcid, __entry->pktid
|
||||
|
||||
@ -36,28 +35,6 @@ DECLARE_EVENT_CLASS(dev_evt,
|
||||
TP_printk(DEV_PR_FMT, DEV_PR_ARG)
|
||||
);
|
||||
|
||||
DECLARE_EVENT_CLASS(dev_txid_evt,
|
||||
TP_PROTO(struct mt76x02_dev *dev, u8 wcid, u8 pktid),
|
||||
TP_ARGS(dev, wcid, pktid),
|
||||
TP_STRUCT__entry(
|
||||
DEV_ENTRY
|
||||
TXID_ENTRY
|
||||
),
|
||||
TP_fast_assign(
|
||||
DEV_ASSIGN;
|
||||
TXID_ASSIGN;
|
||||
),
|
||||
TP_printk(
|
||||
DEV_PR_FMT TXID_PR_FMT,
|
||||
DEV_PR_ARG, TXID_PR_ARG
|
||||
)
|
||||
);
|
||||
|
||||
DEFINE_EVENT(dev_txid_evt, mac_txdone_add,
|
||||
TP_PROTO(struct mt76x02_dev *dev, u8 wcid, u8 pktid),
|
||||
TP_ARGS(dev, wcid, pktid)
|
||||
);
|
||||
|
||||
DEFINE_EVENT(dev_evt, mac_txstat_poll,
|
||||
TP_PROTO(struct mt76x02_dev *dev),
|
||||
TP_ARGS(dev)
|
||||
@ -100,29 +77,6 @@ TRACE_EVENT(mac_txstat_fetch,
|
||||
)
|
||||
);
|
||||
|
||||
TRACE_EVENT(dev_irq,
|
||||
TP_PROTO(struct mt76x02_dev *dev, u32 val, u32 mask),
|
||||
|
||||
TP_ARGS(dev, val, mask),
|
||||
|
||||
TP_STRUCT__entry(
|
||||
DEV_ENTRY
|
||||
__field(u32, val)
|
||||
__field(u32, mask)
|
||||
),
|
||||
|
||||
TP_fast_assign(
|
||||
DEV_ASSIGN;
|
||||
__entry->val = val;
|
||||
__entry->mask = mask;
|
||||
),
|
||||
|
||||
TP_printk(
|
||||
DEV_PR_FMT " %08x & %08x",
|
||||
DEV_PR_ARG, __entry->val, __entry->mask
|
||||
)
|
||||
);
|
||||
|
||||
#endif
|
||||
|
||||
#undef TRACE_INCLUDE_PATH
|
||||
|
@ -28,7 +28,7 @@ void mt76x02_tx(struct ieee80211_hw *hw, struct ieee80211_tx_control *control,
|
||||
wcid = &mvif->group_wcid;
|
||||
}
|
||||
|
||||
mt76_tx(&dev->mt76, control->sta, wcid, skb);
|
||||
mt76_tx(&dev->mphy, control->sta, wcid, skb);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76x02_tx);
|
||||
|
||||
@ -39,7 +39,6 @@ void mt76x02_queue_rx_skb(struct mt76_dev *mdev, enum mt76_rxq_id q,
|
||||
void *rxwi = skb->data;
|
||||
|
||||
if (q == MT_RXQ_MCU) {
|
||||
/* this is used just by mmio code */
|
||||
mt76_mcu_rx_event(&dev->mt76, skb);
|
||||
return;
|
||||
}
|
||||
@ -74,7 +73,7 @@ s8 mt76x02_tx_get_max_txpwr_adj(struct mt76x02_dev *dev,
|
||||
} else if (rate->flags & IEEE80211_TX_RC_MCS) {
|
||||
max_txpwr = dev->mt76.rate_power.ht[rate->idx & 0xf];
|
||||
} else {
|
||||
enum nl80211_band band = dev->mt76.chandef.chan->band;
|
||||
enum nl80211_band band = dev->mphy.chandef.chan->band;
|
||||
|
||||
if (band == NL80211_BAND_2GHZ) {
|
||||
const struct ieee80211_rate *r;
|
||||
@ -96,7 +95,7 @@ s8 mt76x02_tx_get_max_txpwr_adj(struct mt76x02_dev *dev,
|
||||
|
||||
s8 mt76x02_tx_get_txpwr_adj(struct mt76x02_dev *dev, s8 txpwr, s8 max_txpwr_adj)
|
||||
{
|
||||
txpwr = min_t(s8, txpwr, dev->mt76.txpower_conf);
|
||||
txpwr = min_t(s8, txpwr, dev->txpower_conf);
|
||||
txpwr -= (dev->target_power + dev->target_power_delta[0]);
|
||||
txpwr = min_t(s8, txpwr, max_txpwr_adj);
|
||||
|
||||
|
@ -46,8 +46,7 @@ EXPORT_SYMBOL_GPL(mt76x02u_mac_start);
|
||||
|
||||
int mt76x02u_skb_dma_info(struct sk_buff *skb, int port, u32 flags)
|
||||
{
|
||||
struct sk_buff *iter, *last = skb;
|
||||
u32 info, pad;
|
||||
u32 info;
|
||||
|
||||
/* Buffer layout:
|
||||
* | 4B | xfer len | pad | 4B |
|
||||
@ -57,28 +56,8 @@ int mt76x02u_skb_dma_info(struct sk_buff *skb, int port, u32 flags)
|
||||
*/
|
||||
info = FIELD_PREP(MT_TXD_INFO_LEN, round_up(skb->len, 4)) |
|
||||
FIELD_PREP(MT_TXD_INFO_DPORT, port) | flags;
|
||||
put_unaligned_le32(info, skb_push(skb, sizeof(info)));
|
||||
|
||||
/* Add zero pad of 4 - 7 bytes */
|
||||
pad = round_up(skb->len, 4) + 4 - skb->len;
|
||||
|
||||
/* First packet of a A-MSDU burst keeps track of the whole burst
|
||||
* length, need to update length of it and the last packet.
|
||||
*/
|
||||
skb_walk_frags(skb, iter) {
|
||||
last = iter;
|
||||
if (!iter->next) {
|
||||
skb->data_len += pad;
|
||||
skb->len += pad;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (skb_pad(last, pad))
|
||||
return -ENOMEM;
|
||||
__skb_put(last, pad);
|
||||
|
||||
return 0;
|
||||
return mt76u_skb_dma_info(skb, info);
|
||||
}
|
||||
|
||||
int mt76x02u_tx_prepare_skb(struct mt76_dev *mdev, void *data,
|
||||
@ -198,7 +177,7 @@ static void mt76x02u_pre_tbtt_work(struct work_struct *work)
|
||||
container_of(work, struct mt76x02_dev, pre_tbtt_work);
|
||||
struct beacon_bc_data data = {};
|
||||
struct sk_buff *skb;
|
||||
int i, nbeacons;
|
||||
int nbeacons;
|
||||
|
||||
if (!dev->mt76.beacon_mask)
|
||||
return;
|
||||
@ -208,17 +187,30 @@ static void mt76x02u_pre_tbtt_work(struct work_struct *work)
|
||||
|
||||
mt76x02_resync_beacon_timer(dev);
|
||||
|
||||
/* Prevent corrupt transmissions during update */
|
||||
mt76_set(dev, MT_BCN_BYPASS_MASK, 0xffff);
|
||||
dev->beacon_data_count = 0;
|
||||
|
||||
ieee80211_iterate_active_interfaces(mt76_hw(dev),
|
||||
IEEE80211_IFACE_ITER_RESUME_ALL,
|
||||
mt76x02_update_beacon_iter, dev);
|
||||
|
||||
mt76_csa_check(&dev->mt76);
|
||||
|
||||
if (dev->mt76.csa_complete) {
|
||||
mt76_csa_finish(&dev->mt76);
|
||||
goto out;
|
||||
}
|
||||
|
||||
nbeacons = hweight8(dev->mt76.beacon_mask);
|
||||
mt76x02_enqueue_buffered_bc(dev, &data, N_BCN_SLOTS - nbeacons);
|
||||
|
||||
for (i = nbeacons; i < N_BCN_SLOTS; i++) {
|
||||
skb = __skb_dequeue(&data.q);
|
||||
mt76x02_mac_set_beacon(dev, i, skb);
|
||||
}
|
||||
while ((skb = __skb_dequeue(&data.q)) != NULL)
|
||||
mt76x02_mac_set_beacon(dev, skb);
|
||||
|
||||
out:
|
||||
mt76_wr(dev, MT_BCN_BYPASS_MASK,
|
||||
0xff00 | ~(0xff00 >> dev->beacon_data_count));
|
||||
|
||||
mt76x02u_restart_pre_tbtt_timer(dev);
|
||||
}
|
||||
@ -244,20 +236,11 @@ static void mt76x02u_pre_tbtt_enable(struct mt76x02_dev *dev, bool en)
|
||||
|
||||
static void mt76x02u_beacon_enable(struct mt76x02_dev *dev, bool en)
|
||||
{
|
||||
int i;
|
||||
|
||||
if (WARN_ON_ONCE(!dev->mt76.beacon_int))
|
||||
return;
|
||||
|
||||
if (en) {
|
||||
if (en)
|
||||
mt76x02u_start_pre_tbtt_timer(dev);
|
||||
} else {
|
||||
/* Timer is already stopped, only clean up
|
||||
* PS buffered frames if any.
|
||||
*/
|
||||
for (i = 0; i < N_BCN_SLOTS; i++)
|
||||
mt76x02_mac_set_beacon(dev, i, NULL);
|
||||
}
|
||||
}
|
||||
|
||||
void mt76x02u_init_beacon_config(struct mt76x02_dev *dev)
|
||||
@ -280,7 +263,7 @@ EXPORT_SYMBOL_GPL(mt76x02u_init_beacon_config);
|
||||
|
||||
void mt76x02u_exit_beacon_config(struct mt76x02_dev *dev)
|
||||
{
|
||||
if (!test_bit(MT76_REMOVED, &dev->mt76.state))
|
||||
if (!test_bit(MT76_REMOVED, &dev->mphy.state))
|
||||
mt76_clear(dev, MT_BEACON_TIME_CFG,
|
||||
MT_BEACON_TIME_CFG_TIMER_EN |
|
||||
MT_BEACON_TIME_CFG_SYNC_MODE |
|
||||
|
@ -55,7 +55,8 @@ static int mt76x02u_mcu_wait_resp(struct mt76_dev *dev, u8 seq)
|
||||
u32 rxfce;
|
||||
|
||||
for (i = 0; i < 5; i++) {
|
||||
ret = mt76u_bulk_msg(dev, data, MCU_RESP_URB_SIZE, &len, 300);
|
||||
ret = mt76u_bulk_msg(dev, data, MCU_RESP_URB_SIZE, &len,
|
||||
300, MT_EP_IN_CMD_RESP);
|
||||
if (ret == -ETIMEDOUT)
|
||||
continue;
|
||||
if (ret)
|
||||
@ -82,18 +83,17 @@ static int
|
||||
__mt76x02u_mcu_send_msg(struct mt76_dev *dev, struct sk_buff *skb,
|
||||
int cmd, bool wait_resp)
|
||||
{
|
||||
struct mt76_usb *usb = &dev->usb;
|
||||
int ret;
|
||||
u8 seq = 0;
|
||||
u32 info;
|
||||
int ret;
|
||||
|
||||
if (test_bit(MT76_REMOVED, &dev->state))
|
||||
if (test_bit(MT76_REMOVED, &dev->phy.state))
|
||||
return 0;
|
||||
|
||||
if (wait_resp) {
|
||||
seq = ++usb->mcu.msg_seq & 0xf;
|
||||
seq = ++dev->mcu.msg_seq & 0xf;
|
||||
if (!seq)
|
||||
seq = ++usb->mcu.msg_seq & 0xf;
|
||||
seq = ++dev->mcu.msg_seq & 0xf;
|
||||
}
|
||||
|
||||
info = FIELD_PREP(MT_MCU_MSG_CMD_SEQ, seq) |
|
||||
@ -103,7 +103,8 @@ __mt76x02u_mcu_send_msg(struct mt76_dev *dev, struct sk_buff *skb,
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = mt76u_bulk_msg(dev, skb->data, skb->len, NULL, 500);
|
||||
ret = mt76u_bulk_msg(dev, skb->data, skb->len, NULL, 500,
|
||||
MT_EP_OUT_INBAND_CMD);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
@ -119,7 +120,6 @@ static int
|
||||
mt76x02u_mcu_send_msg(struct mt76_dev *dev, int cmd, const void *data,
|
||||
int len, bool wait_resp)
|
||||
{
|
||||
struct mt76_usb *usb = &dev->usb;
|
||||
struct sk_buff *skb;
|
||||
int err;
|
||||
|
||||
@ -127,9 +127,9 @@ mt76x02u_mcu_send_msg(struct mt76_dev *dev, int cmd, const void *data,
|
||||
if (!skb)
|
||||
return -ENOMEM;
|
||||
|
||||
mutex_lock(&usb->mcu.mutex);
|
||||
mutex_lock(&dev->mcu.mutex);
|
||||
err = __mt76x02u_mcu_send_msg(dev, skb, cmd, wait_resp);
|
||||
mutex_unlock(&usb->mcu.mutex);
|
||||
mutex_unlock(&dev->mcu.mutex);
|
||||
|
||||
return err;
|
||||
}
|
||||
@ -143,9 +143,8 @@ static int
|
||||
mt76x02u_mcu_wr_rp(struct mt76_dev *dev, u32 base,
|
||||
const struct mt76_reg_pair *data, int n)
|
||||
{
|
||||
const int CMD_RANDOM_WRITE = 12;
|
||||
const int max_vals_per_cmd = MT_INBAND_PACKET_MAX_LEN / 8;
|
||||
struct mt76_usb *usb = &dev->usb;
|
||||
const int CMD_RANDOM_WRITE = 12;
|
||||
struct sk_buff *skb;
|
||||
int cnt, i, ret;
|
||||
|
||||
@ -164,9 +163,9 @@ mt76x02u_mcu_wr_rp(struct mt76_dev *dev, u32 base,
|
||||
skb_put_le32(skb, data[i].value);
|
||||
}
|
||||
|
||||
mutex_lock(&usb->mcu.mutex);
|
||||
mutex_lock(&dev->mcu.mutex);
|
||||
ret = __mt76x02u_mcu_send_msg(dev, skb, CMD_RANDOM_WRITE, cnt == n);
|
||||
mutex_unlock(&usb->mcu.mutex);
|
||||
mutex_unlock(&dev->mcu.mutex);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
@ -200,7 +199,7 @@ mt76x02u_mcu_rd_rp(struct mt76_dev *dev, u32 base,
|
||||
skb_put_le32(skb, data[i].value);
|
||||
}
|
||||
|
||||
mutex_lock(&usb->mcu.mutex);
|
||||
mutex_lock(&dev->mcu.mutex);
|
||||
|
||||
usb->mcu.rp = data;
|
||||
usb->mcu.rp_len = n;
|
||||
@ -211,7 +210,7 @@ mt76x02u_mcu_rd_rp(struct mt76_dev *dev, u32 base,
|
||||
|
||||
usb->mcu.rp = NULL;
|
||||
|
||||
mutex_unlock(&usb->mcu.mutex);
|
||||
mutex_unlock(&dev->mcu.mutex);
|
||||
|
||||
return ret;
|
||||
}
|
||||
@ -248,7 +247,8 @@ __mt76x02u_mcu_fw_send_data(struct mt76x02_dev *dev, u8 *data,
|
||||
|
||||
data_len = MT_CMD_HDR_LEN + len + sizeof(info);
|
||||
|
||||
err = mt76u_bulk_msg(&dev->mt76, data, data_len, NULL, 1000);
|
||||
err = mt76u_bulk_msg(&dev->mt76, data, data_len, NULL, 1000,
|
||||
MT_EP_OUT_INBAND_CMD);
|
||||
if (err) {
|
||||
dev_err(dev->mt76.dev, "firmware upload failed: %d\n", err);
|
||||
return err;
|
||||
|
@ -96,9 +96,9 @@ mt76x02_led_set_config(struct mt76_dev *mdev, u8 delay_on,
|
||||
mt76);
|
||||
u32 val;
|
||||
|
||||
val = MT_LED_STATUS_DURATION(0xff) |
|
||||
MT_LED_STATUS_OFF(delay_off) |
|
||||
MT_LED_STATUS_ON(delay_on);
|
||||
val = FIELD_PREP(MT_LED_STATUS_DURATION, 0xff) |
|
||||
FIELD_PREP(MT_LED_STATUS_OFF, delay_off) |
|
||||
FIELD_PREP(MT_LED_STATUS_ON, delay_on);
|
||||
|
||||
mt76_wr(dev, MT_LED_S0(mdev->led_pin), val);
|
||||
mt76_wr(dev, MT_LED_S1(mdev->led_pin), val);
|
||||
@ -166,7 +166,6 @@ void mt76x02_init_device(struct mt76x02_dev *dev)
|
||||
wiphy->reg_notifier = mt76x02_regd_notifier;
|
||||
wiphy->iface_combinations = mt76x02_if_comb;
|
||||
wiphy->n_iface_combinations = ARRAY_SIZE(mt76x02_if_comb);
|
||||
wiphy->flags |= WIPHY_FLAG_HAS_CHANNEL_SWITCH;
|
||||
|
||||
/* init led callbacks */
|
||||
if (IS_ENABLED(CONFIG_MT76_LEDS)) {
|
||||
@ -182,21 +181,22 @@ void mt76x02_init_device(struct mt76x02_dev *dev)
|
||||
hw->vif_data_size = sizeof(struct mt76x02_vif);
|
||||
|
||||
ieee80211_hw_set(hw, SUPPORTS_HT_CCK_RATES);
|
||||
ieee80211_hw_set(hw, HOST_BROADCAST_PS_BUFFERING);
|
||||
|
||||
dev->mt76.global_wcid.idx = 255;
|
||||
dev->mt76.global_wcid.hw_key_idx = -1;
|
||||
dev->slottime = 9;
|
||||
|
||||
if (is_mt76x2(dev)) {
|
||||
dev->mt76.sband_2g.sband.ht_cap.cap |=
|
||||
dev->mphy.sband_2g.sband.ht_cap.cap |=
|
||||
IEEE80211_HT_CAP_LDPC_CODING;
|
||||
dev->mt76.sband_5g.sband.ht_cap.cap |=
|
||||
dev->mphy.sband_5g.sband.ht_cap.cap |=
|
||||
IEEE80211_HT_CAP_LDPC_CODING;
|
||||
dev->mt76.chainmask = 0x202;
|
||||
dev->mt76.antenna_mask = 3;
|
||||
dev->chainmask = 0x202;
|
||||
dev->mphy.antenna_mask = 3;
|
||||
} else {
|
||||
dev->mt76.chainmask = 0x101;
|
||||
dev->mt76.antenna_mask = 1;
|
||||
dev->chainmask = 0x101;
|
||||
dev->mphy.antenna_mask = 1;
|
||||
}
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76x02_init_device);
|
||||
@ -325,7 +325,9 @@ mt76x02_add_interface(struct ieee80211_hw *hw, struct ieee80211_vif *vif)
|
||||
if (vif->type == NL80211_IFTYPE_STATION)
|
||||
idx += 8;
|
||||
|
||||
if (dev->vif_mask & BIT(idx))
|
||||
/* vif is already set or idx is 8 for AP/Mesh/... */
|
||||
if (dev->vif_mask & BIT(idx) ||
|
||||
(vif->type != NL80211_IFTYPE_STATION && idx > 7))
|
||||
return -EBUSY;
|
||||
|
||||
dev->vif_mask |= BIT(idx);
|
||||
@ -545,7 +547,7 @@ void mt76x02_set_coverage_class(struct ieee80211_hw *hw,
|
||||
struct mt76x02_dev *dev = hw->priv;
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
dev->coverage_class = coverage_class;
|
||||
dev->coverage_class = max_t(s16, coverage_class, 0);
|
||||
mt76x02_set_tx_ackto(dev);
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
}
|
||||
@ -602,7 +604,7 @@ void mt76x02_sw_scan_complete(struct ieee80211_hw *hw,
|
||||
{
|
||||
struct mt76x02_dev *dev = hw->priv;
|
||||
|
||||
clear_bit(MT76_SCANNING, &dev->mt76.state);
|
||||
clear_bit(MT76_SCANNING, &dev->mphy.state);
|
||||
if (dev->cal.gain_init_done) {
|
||||
/* Restore AGC gain and resume calibration after scanning. */
|
||||
dev->cal.low_gain = -1;
|
||||
|
@ -13,5 +13,3 @@ mt76x2e-y := \
|
||||
mt76x2u-y := \
|
||||
usb.o usb_init.o usb_main.o usb_mac.o usb_mcu.o \
|
||||
usb_phy.o
|
||||
|
||||
CFLAGS_pci_trace.o := -I$(src)
|
||||
|
@ -248,7 +248,7 @@ mt76x2_get_5g_rx_gain(struct mt76x02_dev *dev, u8 channel)
|
||||
|
||||
void mt76x2_read_rx_gain(struct mt76x02_dev *dev)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
int channel = chan->hw_value;
|
||||
s8 lna_5g[3], lna_2g;
|
||||
u8 lna;
|
||||
@ -455,7 +455,7 @@ EXPORT_SYMBOL_GPL(mt76x2_get_power_info);
|
||||
|
||||
int mt76x2_get_temp_comp(struct mt76x02_dev *dev, struct mt76x2_temp_comp *t)
|
||||
{
|
||||
enum nl80211_band band = dev->mt76.chandef.chan->band;
|
||||
enum nl80211_band band = dev->mphy.chandef.chan->band;
|
||||
u16 val, slope;
|
||||
u8 bounds;
|
||||
|
||||
|
@ -53,7 +53,7 @@ mt76x2_has_ext_lna(struct mt76x02_dev *dev)
|
||||
{
|
||||
u32 val = mt76x02_eeprom_get(dev, MT_EE_NIC_CONF_1);
|
||||
|
||||
if (dev->mt76.chandef.chan->band == NL80211_BAND_2GHZ)
|
||||
if (dev->mphy.chandef.chan->band == NL80211_BAND_2GHZ)
|
||||
return val & MT_EE_NIC_CONF_1_LNA_EXT_2G;
|
||||
else
|
||||
return val & MT_EE_NIC_CONF_1_LNA_EXT_5G;
|
||||
|
@ -82,7 +82,7 @@ void mt76_write_mac_initvals(struct mt76x02_dev *dev)
|
||||
{ MT_PBF_SYS_CTRL, 0x00080c00 },
|
||||
{ MT_PBF_CFG, 0x1efebcff },
|
||||
{ MT_FCE_PSE_CTRL, 0x00000001 },
|
||||
{ MT_MAC_SYS_CTRL, 0x0000000c },
|
||||
{ MT_MAC_SYS_CTRL, 0x00000000 },
|
||||
{ MT_MAX_LEN_CFG, 0x003e3f00 },
|
||||
{ MT_AMPDU_MAX_LEN_20M1S, 0xaaa99887 },
|
||||
{ MT_AMPDU_MAX_LEN_20M2S, 0x000000aa },
|
||||
|
@ -29,7 +29,7 @@ int mt76x2_mcu_set_channel(struct mt76x02_dev *dev, u8 channel, u8 bw,
|
||||
.idx = channel,
|
||||
.scan = scan,
|
||||
.bw = bw,
|
||||
.chainmask = cpu_to_le16(dev->mt76.chainmask),
|
||||
.chainmask = cpu_to_le16(dev->chainmask),
|
||||
};
|
||||
|
||||
/* first set the channel without the extension channel info */
|
||||
|
@ -30,7 +30,7 @@ static inline bool is_mt7612(struct mt76x02_dev *dev)
|
||||
|
||||
static inline bool mt76x2_channel_silent(struct mt76x02_dev *dev)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
|
||||
return ((chan->flags & IEEE80211_CHAN_RADAR) &&
|
||||
chan->dfs_state != NL80211_DFS_AVAILABLE);
|
||||
|
@ -239,7 +239,7 @@ static int mt76x2_init_hardware(struct mt76x02_dev *dev)
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);
|
||||
mt76x02_mac_start(dev);
|
||||
|
||||
ret = mt76x2_mcu_init(dev);
|
||||
@ -289,8 +289,8 @@ int mt76x2_register_device(struct mt76x02_dev *dev)
|
||||
goto fail;
|
||||
|
||||
mt76x02_init_debugfs(dev);
|
||||
mt76x2_init_txpower(dev, &dev->mt76.sband_2g.sband);
|
||||
mt76x2_init_txpower(dev, &dev->mt76.sband_5g.sband);
|
||||
mt76x2_init_txpower(dev, &dev->mphy.sband_2g.sband);
|
||||
mt76x2_init_txpower(dev, &dev->mphy.sband_5g.sband);
|
||||
|
||||
return 0;
|
||||
|
||||
|
@ -22,7 +22,7 @@ mt76x2_start(struct ieee80211_hw *hw)
|
||||
ieee80211_queue_delayed_work(mt76_hw(dev), &dev->wdt_work,
|
||||
MT_WATCHDOG_TIME);
|
||||
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -31,7 +31,7 @@ mt76x2_stop(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct mt76x02_dev *dev = hw->priv;
|
||||
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
mt76x2_stop_hardware(dev);
|
||||
}
|
||||
|
||||
@ -45,9 +45,9 @@ mt76x2_set_channel(struct mt76x02_dev *dev, struct cfg80211_chan_def *chandef)
|
||||
tasklet_disable(&dev->dfs_pd.dfs_tasklet);
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
set_bit(MT76_RESET, &dev->mt76.state);
|
||||
set_bit(MT76_RESET, &dev->mphy.state);
|
||||
|
||||
mt76_set_channel(&dev->mt76);
|
||||
mt76_set_channel(&dev->mphy);
|
||||
|
||||
mt76x2_mac_stop(dev, true);
|
||||
ret = mt76x2_phy_set_channel(dev, chandef);
|
||||
@ -57,13 +57,13 @@ mt76x2_set_channel(struct mt76x02_dev *dev, struct cfg80211_chan_def *chandef)
|
||||
|
||||
mt76x2_mac_resume(dev);
|
||||
|
||||
clear_bit(MT76_RESET, &dev->mt76.state);
|
||||
clear_bit(MT76_RESET, &dev->mphy.state);
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
tasklet_enable(&dev->dfs_pd.dfs_tasklet);
|
||||
tasklet_enable(&dev->mt76.pre_tbtt_tasklet);
|
||||
|
||||
mt76_txq_schedule_all(&dev->mt76);
|
||||
mt76_txq_schedule_all(&dev->mphy);
|
||||
|
||||
return ret;
|
||||
}
|
||||
@ -86,14 +86,14 @@ mt76x2_config(struct ieee80211_hw *hw, u32 changed)
|
||||
}
|
||||
|
||||
if (changed & IEEE80211_CONF_CHANGE_POWER) {
|
||||
dev->mt76.txpower_conf = hw->conf.power_level * 2;
|
||||
dev->txpower_conf = hw->conf.power_level * 2;
|
||||
|
||||
/* convert to per-chain power for 2x2 devices */
|
||||
dev->mt76.txpower_conf -= 6;
|
||||
dev->txpower_conf -= 6;
|
||||
|
||||
if (test_bit(MT76_STATE_RUNNING, &dev->mt76.state)) {
|
||||
if (test_bit(MT76_STATE_RUNNING, &dev->mphy.state)) {
|
||||
mt76x2_phy_set_txpower(dev);
|
||||
mt76x02_tx_set_txpwr_auto(dev, dev->mt76.txpower_conf);
|
||||
mt76x02_tx_set_txpwr_auto(dev, dev->txpower_conf);
|
||||
}
|
||||
}
|
||||
|
||||
@ -124,8 +124,8 @@ static int mt76x2_set_antenna(struct ieee80211_hw *hw, u32 tx_ant,
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
|
||||
dev->mt76.chainmask = (tx_ant == 3) ? 0x202 : 0x101;
|
||||
dev->mt76.antenna_mask = tx_ant;
|
||||
dev->chainmask = (tx_ant == 3) ? 0x202 : 0x101;
|
||||
dev->mphy.antenna_mask = tx_ant;
|
||||
|
||||
mt76_set_stream_caps(&dev->mt76, true);
|
||||
mt76x2_phy_set_antenna(dev);
|
||||
@ -145,6 +145,7 @@ const struct ieee80211_ops mt76x2_ops = {
|
||||
.configure_filter = mt76x02_configure_filter,
|
||||
.bss_info_changed = mt76x02_bss_info_changed,
|
||||
.sta_state = mt76_sta_state,
|
||||
.sta_pre_rcu_remove = mt76_sta_pre_rcu_remove,
|
||||
.set_key = mt76x02_set_key,
|
||||
.conf_tx = mt76x02_conf_tx,
|
||||
.sw_scan_start = mt76_sw_scan,
|
||||
|
@ -12,7 +12,7 @@
|
||||
static bool
|
||||
mt76x2_phy_tssi_init_cal(struct mt76x02_dev *dev)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
u32 flag = 0;
|
||||
|
||||
if (!mt76x2_tssi_enabled(dev))
|
||||
@ -35,7 +35,7 @@ mt76x2_phy_tssi_init_cal(struct mt76x02_dev *dev)
|
||||
static void
|
||||
mt76x2_phy_channel_calibrate(struct mt76x02_dev *dev, bool mac_stopped)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
bool is_5ghz = chan->band == NL80211_BAND_5GHZ;
|
||||
|
||||
if (dev->cal.channel_cal_done)
|
||||
@ -74,7 +74,7 @@ void mt76x2_phy_set_antenna(struct mt76x02_dev *dev)
|
||||
|
||||
val = mt76_rr(dev, MT_BBP(AGC, 0));
|
||||
val &= ~(BIT(4) | BIT(1));
|
||||
switch (dev->mt76.antenna_mask) {
|
||||
switch (dev->mphy.antenna_mask) {
|
||||
case 1:
|
||||
/* disable mac DAC control */
|
||||
mt76_clear(dev, MT_BBP(IBI, 9), BIT(11));
|
||||
@ -118,7 +118,7 @@ int mt76x2_phy_set_channel(struct mt76x02_dev *dev,
|
||||
struct cfg80211_chan_def *chandef)
|
||||
{
|
||||
struct ieee80211_channel *chan = chandef->chan;
|
||||
bool scan = test_bit(MT76_SCANNING, &dev->mt76.state);
|
||||
bool scan = test_bit(MT76_SCANNING, &dev->mphy.state);
|
||||
enum nl80211_band band = chan->band;
|
||||
u8 channel;
|
||||
|
||||
|
@ -136,8 +136,8 @@ mt76x2_get_min_rate_power(struct mt76_rate_power *r)
|
||||
|
||||
void mt76x2_phy_set_txpower(struct mt76x02_dev *dev)
|
||||
{
|
||||
enum nl80211_chan_width width = dev->mt76.chandef.width;
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
enum nl80211_chan_width width = dev->mphy.chandef.width;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
struct mt76x2_tx_power_info txp;
|
||||
int txp_0, txp_1, delta = 0;
|
||||
struct mt76_rate_power t = {};
|
||||
@ -152,8 +152,8 @@ void mt76x2_phy_set_txpower(struct mt76x02_dev *dev)
|
||||
|
||||
mt76x2_get_rate_power(dev, &t, chan);
|
||||
mt76x02_add_rate_power_offset(&t, txp.target_power + delta);
|
||||
mt76x02_limit_rate_power(&t, dev->mt76.txpower_conf);
|
||||
dev->mt76.txpower_cur = mt76x02_get_max_rate_power(&t);
|
||||
mt76x02_limit_rate_power(&t, dev->txpower_conf);
|
||||
dev->mphy.txpower_cur = mt76x02_get_max_rate_power(&t);
|
||||
|
||||
base_power = mt76x2_get_min_rate_power(&t);
|
||||
delta = base_power - txp.target_power;
|
||||
@ -202,7 +202,7 @@ EXPORT_SYMBOL_GPL(mt76x2_configure_tx_delay);
|
||||
|
||||
void mt76x2_phy_tssi_compensate(struct mt76x02_dev *dev)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
struct mt76x2_tx_power_info txp;
|
||||
struct mt76x2_tssi_comp t = {};
|
||||
|
||||
@ -252,12 +252,12 @@ mt76x2_phy_set_gain_val(struct mt76x02_dev *dev)
|
||||
|
||||
val = 0x1836 << 16;
|
||||
if (!mt76x2_has_ext_lna(dev) &&
|
||||
dev->mt76.chandef.width >= NL80211_CHAN_WIDTH_40)
|
||||
dev->mphy.chandef.width >= NL80211_CHAN_WIDTH_40)
|
||||
val = 0x1e42 << 16;
|
||||
|
||||
if (mt76x2_has_ext_lna(dev) &&
|
||||
dev->mt76.chandef.chan->band == NL80211_BAND_2GHZ &&
|
||||
dev->mt76.chandef.width < NL80211_CHAN_WIDTH_40)
|
||||
dev->mphy.chandef.chan->band == NL80211_BAND_2GHZ &&
|
||||
dev->mphy.chandef.width < NL80211_CHAN_WIDTH_40)
|
||||
val = 0x0f36 << 16;
|
||||
|
||||
val |= 0xf8;
|
||||
@ -267,7 +267,7 @@ mt76x2_phy_set_gain_val(struct mt76x02_dev *dev)
|
||||
mt76_wr(dev, MT_BBP(AGC, 9),
|
||||
val | FIELD_PREP(MT_BBP_AGC_GAIN, gain_val[1]));
|
||||
|
||||
if (dev->mt76.chandef.chan->flags & IEEE80211_CHAN_RADAR)
|
||||
if (dev->mphy.chandef.chan->flags & IEEE80211_CHAN_RADAR)
|
||||
mt76x02_phy_dfs_adjust_agc(dev);
|
||||
}
|
||||
|
||||
@ -280,7 +280,7 @@ void mt76x2_phy_update_channel_gain(struct mt76x02_dev *dev)
|
||||
int low_gain;
|
||||
u32 val;
|
||||
|
||||
dev->cal.avg_rssi_all = mt76_get_min_avg_rssi(&dev->mt76);
|
||||
dev->cal.avg_rssi_all = mt76_get_min_avg_rssi(&dev->mt76, false);
|
||||
if (!dev->cal.avg_rssi_all)
|
||||
dev->cal.avg_rssi_all = -75;
|
||||
|
||||
@ -297,7 +297,7 @@ void mt76x2_phy_update_channel_gain(struct mt76x02_dev *dev)
|
||||
return;
|
||||
}
|
||||
|
||||
if (dev->mt76.chandef.width == NL80211_CHAN_WIDTH_80) {
|
||||
if (dev->mphy.chandef.width == NL80211_CHAN_WIDTH_80) {
|
||||
mt76_wr(dev, MT_BBP(RXO, 14), 0x00560211);
|
||||
val = mt76_rr(dev, MT_BBP(AGC, 26)) & ~0xf;
|
||||
if (low_gain == 2)
|
||||
@ -315,11 +315,11 @@ void mt76x2_phy_update_channel_gain(struct mt76x02_dev *dev)
|
||||
low_gain_delta = 14;
|
||||
|
||||
agc_37 = 0x2121262c;
|
||||
if (dev->mt76.chandef.chan->band == NL80211_BAND_2GHZ)
|
||||
if (dev->mphy.chandef.chan->band == NL80211_BAND_2GHZ)
|
||||
agc_35 = 0x11111516;
|
||||
else if (low_gain == 2)
|
||||
agc_35 = agc_37 = 0x08080808;
|
||||
else if (dev->mt76.chandef.width == NL80211_CHAN_WIDTH_80)
|
||||
else if (dev->mphy.chandef.width == NL80211_CHAN_WIDTH_80)
|
||||
agc_35 = 0x10101014;
|
||||
else
|
||||
agc_35 = 0x11111116;
|
||||
|
@ -54,7 +54,7 @@ static int mt76x2u_probe(struct usb_interface *intf,
|
||||
usb_set_intfdata(intf, dev);
|
||||
|
||||
mt76x02u_init_mcu(mdev);
|
||||
err = mt76u_init(mdev, intf);
|
||||
err = mt76u_init(mdev, intf, false);
|
||||
if (err < 0)
|
||||
goto err;
|
||||
|
||||
@ -86,7 +86,7 @@ static void mt76x2u_disconnect(struct usb_interface *intf)
|
||||
struct mt76x02_dev *dev = usb_get_intfdata(intf);
|
||||
struct ieee80211_hw *hw = mt76_hw(dev);
|
||||
|
||||
set_bit(MT76_REMOVED, &dev->mt76.state);
|
||||
set_bit(MT76_REMOVED, &dev->mphy.state);
|
||||
ieee80211_unregister_hw(hw);
|
||||
mt76x2u_cleanup(dev);
|
||||
mt76u_deinit(&dev->mt76);
|
||||
|
@ -190,6 +190,7 @@ int mt76x2u_init_hardware(struct mt76x02_dev *dev)
|
||||
int mt76x2u_register_device(struct mt76x02_dev *dev)
|
||||
{
|
||||
struct ieee80211_hw *hw = mt76_hw(dev);
|
||||
struct mt76_usb *usb = &dev->mt76.usb;
|
||||
int err;
|
||||
|
||||
INIT_DELAYED_WORK(&dev->cal_work, mt76x2u_phy_calibrate);
|
||||
@ -199,6 +200,11 @@ int mt76x2u_register_device(struct mt76x02_dev *dev)
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
usb->mcu.data = devm_kmalloc(dev->mt76.dev, MCU_RESP_URB_SIZE,
|
||||
GFP_KERNEL);
|
||||
if (!usb->mcu.data)
|
||||
return -ENOMEM;
|
||||
|
||||
err = mt76u_alloc_queues(&dev->mt76);
|
||||
if (err < 0)
|
||||
goto fail;
|
||||
@ -207,22 +213,18 @@ int mt76x2u_register_device(struct mt76x02_dev *dev)
|
||||
if (err < 0)
|
||||
goto fail;
|
||||
|
||||
/* check hw sg support in order to enable AMSDU */
|
||||
hw->max_tx_fragments = dev->mt76.usb.sg_en ? MT_TX_SG_MAX_SIZE : 1;
|
||||
err = mt76_register_device(&dev->mt76, true, mt76x02_rates,
|
||||
ARRAY_SIZE(mt76x02_rates));
|
||||
if (err)
|
||||
goto fail;
|
||||
|
||||
/* check hw sg support in order to enable AMSDU */
|
||||
if (dev->mt76.usb.sg_en)
|
||||
hw->max_tx_fragments = MT_TX_SG_MAX_SIZE;
|
||||
else
|
||||
hw->max_tx_fragments = 1;
|
||||
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);
|
||||
|
||||
mt76x02_init_debugfs(dev);
|
||||
mt76x2_init_txpower(dev, &dev->mt76.sband_2g.sband);
|
||||
mt76x2_init_txpower(dev, &dev->mt76.sband_5g.sband);
|
||||
mt76x2_init_txpower(dev, &dev->mphy.sband_2g.sband);
|
||||
mt76x2_init_txpower(dev, &dev->mphy.sband_5g.sband);
|
||||
|
||||
return 0;
|
||||
|
||||
|
@ -98,7 +98,7 @@ int mt76x2u_mac_stop(struct mt76x02_dev *dev)
|
||||
bool stopped = false;
|
||||
u32 rts_cfg;
|
||||
|
||||
if (test_bit(MT76_REMOVED, &dev->mt76.state))
|
||||
if (test_bit(MT76_REMOVED, &dev->mphy.state))
|
||||
return -EIO;
|
||||
|
||||
rts_cfg = mt76_rr(dev, MT_TX_RTS_CFG);
|
||||
|
@ -17,7 +17,7 @@ static int mt76x2u_start(struct ieee80211_hw *hw)
|
||||
|
||||
ieee80211_queue_delayed_work(mt76_hw(dev), &dev->mt76.mac_work,
|
||||
MT_MAC_WORK_INTERVAL);
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
set_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -26,7 +26,7 @@ static void mt76x2u_stop(struct ieee80211_hw *hw)
|
||||
{
|
||||
struct mt76x02_dev *dev = hw->priv;
|
||||
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mt76.state);
|
||||
clear_bit(MT76_STATE_RUNNING, &dev->mphy.state);
|
||||
mt76u_stop_tx(&dev->mt76);
|
||||
mt76x2u_stop_hw(dev);
|
||||
}
|
||||
@ -41,9 +41,9 @@ mt76x2u_set_channel(struct mt76x02_dev *dev,
|
||||
mt76x02_pre_tbtt_enable(dev, false);
|
||||
|
||||
mutex_lock(&dev->mt76.mutex);
|
||||
set_bit(MT76_RESET, &dev->mt76.state);
|
||||
set_bit(MT76_RESET, &dev->mphy.state);
|
||||
|
||||
mt76_set_channel(&dev->mt76);
|
||||
mt76_set_channel(&dev->mphy);
|
||||
|
||||
mt76x2_mac_stop(dev, false);
|
||||
|
||||
@ -52,11 +52,11 @@ mt76x2u_set_channel(struct mt76x02_dev *dev,
|
||||
mt76x02_mac_cc_reset(dev);
|
||||
mt76x2_mac_resume(dev);
|
||||
|
||||
clear_bit(MT76_RESET, &dev->mt76.state);
|
||||
clear_bit(MT76_RESET, &dev->mphy.state);
|
||||
mutex_unlock(&dev->mt76.mutex);
|
||||
|
||||
mt76x02_pre_tbtt_enable(dev, true);
|
||||
mt76_txq_schedule_all(&dev->mt76);
|
||||
mt76_txq_schedule_all(&dev->mphy);
|
||||
|
||||
return err;
|
||||
}
|
||||
@ -78,12 +78,12 @@ mt76x2u_config(struct ieee80211_hw *hw, u32 changed)
|
||||
}
|
||||
|
||||
if (changed & IEEE80211_CONF_CHANGE_POWER) {
|
||||
dev->mt76.txpower_conf = hw->conf.power_level * 2;
|
||||
dev->txpower_conf = hw->conf.power_level * 2;
|
||||
|
||||
/* convert to per-chain power for 2x2 devices */
|
||||
dev->mt76.txpower_conf -= 6;
|
||||
dev->txpower_conf -= 6;
|
||||
|
||||
if (test_bit(MT76_STATE_RUNNING, &dev->mt76.state))
|
||||
if (test_bit(MT76_STATE_RUNNING, &dev->mphy.state))
|
||||
mt76x2_phy_set_txpower(dev);
|
||||
}
|
||||
|
||||
@ -105,6 +105,7 @@ const struct ieee80211_ops mt76x2u_ops = {
|
||||
.add_interface = mt76x02_add_interface,
|
||||
.remove_interface = mt76x02_remove_interface,
|
||||
.sta_state = mt76_sta_state,
|
||||
.sta_pre_rcu_remove = mt76_sta_pre_rcu_remove,
|
||||
.set_key = mt76x02_set_key,
|
||||
.ampdu_action = mt76x02_ampdu_action,
|
||||
.config = mt76x2u_config,
|
||||
|
@ -10,7 +10,7 @@
|
||||
static void
|
||||
mt76x2u_phy_channel_calibrate(struct mt76x02_dev *dev, bool mac_stopped)
|
||||
{
|
||||
struct ieee80211_channel *chan = dev->mt76.chandef.chan;
|
||||
struct ieee80211_channel *chan = dev->mphy.chandef.chan;
|
||||
bool is_5ghz = chan->band == NL80211_BAND_5GHZ;
|
||||
|
||||
if (dev->cal.channel_cal_done)
|
||||
@ -82,7 +82,7 @@ int mt76x2u_phy_set_channel(struct mt76x02_dev *dev,
|
||||
FIELD_PREP(MT_EXT_CCA_CFG_CCA3, 0) |
|
||||
FIELD_PREP(MT_EXT_CCA_CFG_CCA_MASK, BIT(3)),
|
||||
};
|
||||
bool scan = test_bit(MT76_SCANNING, &dev->mt76.state);
|
||||
bool scan = test_bit(MT76_SCANNING, &dev->mphy.state);
|
||||
struct ieee80211_channel *chan = chandef->chan;
|
||||
u8 channel = chan->hw_value, bw, bw_index;
|
||||
int ch_group_index, freq, freq1, ret;
|
||||
@ -185,7 +185,7 @@ int mt76x2u_phy_set_channel(struct mt76x02_dev *dev,
|
||||
struct ieee80211_channel *chan;
|
||||
u32 flag = 0;
|
||||
|
||||
chan = dev->mt76.chandef.chan;
|
||||
chan = dev->mphy.chandef.chan;
|
||||
if (chan->band == NL80211_BAND_5GHZ)
|
||||
flag |= BIT(0);
|
||||
if (mt76x02_ext_pa_enabled(dev, chan->band))
|
||||
|
@ -9,4 +9,7 @@
|
||||
#define CREATE_TRACE_POINTS
|
||||
#include "trace.h"
|
||||
|
||||
EXPORT_TRACEPOINT_SYMBOL_GPL(mac_txdone);
|
||||
EXPORT_TRACEPOINT_SYMBOL_GPL(dev_irq);
|
||||
|
||||
#endif
|
||||
|
@ -14,7 +14,7 @@
|
||||
|
||||
#define MAXNAME 32
|
||||
#define DEV_ENTRY __array(char, wiphy_name, 32)
|
||||
#define DEV_ASSIGN strlcpy(__entry->wiphy_name, \
|
||||
#define DEVICE_ASSIGN strlcpy(__entry->wiphy_name, \
|
||||
wiphy_name(dev->hw->wiphy), MAXNAME)
|
||||
#define DEV_PR_FMT "%s"
|
||||
#define DEV_PR_ARG __entry->wiphy_name
|
||||
@ -24,6 +24,11 @@
|
||||
#define REG_PR_FMT " %04x=%08x"
|
||||
#define REG_PR_ARG __entry->reg, __entry->val
|
||||
|
||||
#define TXID_ENTRY __field(u8, wcid) __field(u8, pktid)
|
||||
#define TXID_ASSIGN __entry->wcid = wcid; __entry->pktid = pktid
|
||||
#define TXID_PR_FMT " [%d:%d]"
|
||||
#define TXID_PR_ARG __entry->wcid, __entry->pktid
|
||||
|
||||
DECLARE_EVENT_CLASS(dev_reg_evt,
|
||||
TP_PROTO(struct mt76_dev *dev, u32 reg, u32 val),
|
||||
TP_ARGS(dev, reg, val),
|
||||
@ -32,7 +37,7 @@ DECLARE_EVENT_CLASS(dev_reg_evt,
|
||||
REG_ENTRY
|
||||
),
|
||||
TP_fast_assign(
|
||||
DEV_ASSIGN;
|
||||
DEVICE_ASSIGN;
|
||||
REG_ASSIGN;
|
||||
),
|
||||
TP_printk(
|
||||
@ -51,6 +56,51 @@ DEFINE_EVENT(dev_reg_evt, reg_wr,
|
||||
TP_ARGS(dev, reg, val)
|
||||
);
|
||||
|
||||
TRACE_EVENT(dev_irq,
|
||||
TP_PROTO(struct mt76_dev *dev, u32 val, u32 mask),
|
||||
|
||||
TP_ARGS(dev, val, mask),
|
||||
|
||||
TP_STRUCT__entry(
|
||||
DEV_ENTRY
|
||||
__field(u32, val)
|
||||
__field(u32, mask)
|
||||
),
|
||||
|
||||
TP_fast_assign(
|
||||
DEVICE_ASSIGN;
|
||||
__entry->val = val;
|
||||
__entry->mask = mask;
|
||||
),
|
||||
|
||||
TP_printk(
|
||||
DEV_PR_FMT " %08x & %08x",
|
||||
DEV_PR_ARG, __entry->val, __entry->mask
|
||||
)
|
||||
);
|
||||
|
||||
DECLARE_EVENT_CLASS(dev_txid_evt,
|
||||
TP_PROTO(struct mt76_dev *dev, u8 wcid, u8 pktid),
|
||||
TP_ARGS(dev, wcid, pktid),
|
||||
TP_STRUCT__entry(
|
||||
DEV_ENTRY
|
||||
TXID_ENTRY
|
||||
),
|
||||
TP_fast_assign(
|
||||
DEVICE_ASSIGN;
|
||||
TXID_ASSIGN;
|
||||
),
|
||||
TP_printk(
|
||||
DEV_PR_FMT TXID_PR_FMT,
|
||||
DEV_PR_ARG, TXID_PR_ARG
|
||||
)
|
||||
);
|
||||
|
||||
DEFINE_EVENT(dev_txid_evt, mac_txdone,
|
||||
TP_PROTO(struct mt76_dev *dev, u8 wcid, u8 pktid),
|
||||
TP_ARGS(dev, wcid, pktid)
|
||||
);
|
||||
|
||||
#endif
|
||||
|
||||
#undef TRACE_INCLUDE_PATH
|
||||
|
@ -109,13 +109,17 @@ void
|
||||
mt76_tx_status_unlock(struct mt76_dev *dev, struct sk_buff_head *list)
|
||||
__releases(&dev->status_list.unlock)
|
||||
{
|
||||
struct ieee80211_hw *hw;
|
||||
struct sk_buff *skb;
|
||||
|
||||
spin_unlock_bh(&dev->status_list.lock);
|
||||
__release(&dev->status_list.unlock);
|
||||
|
||||
while ((skb = __skb_dequeue(list)) != NULL)
|
||||
ieee80211_tx_status(dev->hw, skb);
|
||||
while ((skb = __skb_dequeue(list)) != NULL) {
|
||||
hw = mt76_tx_status_get_hw(dev, skb);
|
||||
ieee80211_tx_status(hw, skb);
|
||||
}
|
||||
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_tx_status_unlock);
|
||||
|
||||
@ -231,10 +235,12 @@ EXPORT_SYMBOL_GPL(mt76_tx_status_check);
|
||||
|
||||
void mt76_tx_complete_skb(struct mt76_dev *dev, struct sk_buff *skb)
|
||||
{
|
||||
struct ieee80211_hw *hw;
|
||||
struct sk_buff_head list;
|
||||
|
||||
if (!skb->prev) {
|
||||
ieee80211_free_txskb(dev->hw, skb);
|
||||
hw = mt76_tx_status_get_hw(dev, skb);
|
||||
ieee80211_free_txskb(hw, skb);
|
||||
return;
|
||||
}
|
||||
|
||||
@ -245,13 +251,15 @@ void mt76_tx_complete_skb(struct mt76_dev *dev, struct sk_buff *skb)
|
||||
EXPORT_SYMBOL_GPL(mt76_tx_complete_skb);
|
||||
|
||||
void
|
||||
mt76_tx(struct mt76_dev *dev, struct ieee80211_sta *sta,
|
||||
mt76_tx(struct mt76_phy *phy, struct ieee80211_sta *sta,
|
||||
struct mt76_wcid *wcid, struct sk_buff *skb)
|
||||
{
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
|
||||
struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data;
|
||||
struct mt76_queue *q;
|
||||
int qid = skb_get_queue_mapping(skb);
|
||||
bool ext_phy = phy != &dev->phy;
|
||||
|
||||
if (WARN_ON(qid >= MT_TXQ_PSD)) {
|
||||
qid = MT_TXQ_BE;
|
||||
@ -275,6 +283,9 @@ mt76_tx(struct mt76_dev *dev, struct ieee80211_sta *sta,
|
||||
mt76_check_agg_ssn(mtxq, skb);
|
||||
}
|
||||
|
||||
if (ext_phy)
|
||||
info->hw_queue |= MT_TX_HW_QUEUE_EXT_PHY;
|
||||
|
||||
q = dev->q_tx[qid].q;
|
||||
|
||||
spin_lock_bh(&q->lock);
|
||||
@ -282,7 +293,7 @@ mt76_tx(struct mt76_dev *dev, struct ieee80211_sta *sta,
|
||||
dev->queue_ops->kick(dev, q);
|
||||
|
||||
if (q->queued > q->ndesc - 8 && !q->stopped) {
|
||||
ieee80211_stop_queue(dev->hw, skb_get_queue_mapping(skb));
|
||||
ieee80211_stop_queue(phy->hw, skb_get_queue_mapping(skb));
|
||||
q->stopped = true;
|
||||
}
|
||||
|
||||
@ -291,9 +302,11 @@ mt76_tx(struct mt76_dev *dev, struct ieee80211_sta *sta,
|
||||
EXPORT_SYMBOL_GPL(mt76_tx);
|
||||
|
||||
static struct sk_buff *
|
||||
mt76_txq_dequeue(struct mt76_dev *dev, struct mt76_txq *mtxq, bool ps)
|
||||
mt76_txq_dequeue(struct mt76_phy *phy, struct mt76_txq *mtxq, bool ps)
|
||||
{
|
||||
struct ieee80211_txq *txq = mtxq_to_txq(mtxq);
|
||||
struct ieee80211_tx_info *info;
|
||||
bool ext_phy = phy != &phy->dev->phy;
|
||||
struct sk_buff *skb;
|
||||
|
||||
skb = skb_dequeue(&mtxq->retry_q);
|
||||
@ -306,10 +319,14 @@ mt76_txq_dequeue(struct mt76_dev *dev, struct mt76_txq *mtxq, bool ps)
|
||||
return skb;
|
||||
}
|
||||
|
||||
skb = ieee80211_tx_dequeue(dev->hw, txq);
|
||||
skb = ieee80211_tx_dequeue(phy->hw, txq);
|
||||
if (!skb)
|
||||
return NULL;
|
||||
|
||||
info = IEEE80211_SKB_CB(skb);
|
||||
if (ext_phy)
|
||||
info->hw_queue |= MT_TX_HW_QUEUE_EXT_PHY;
|
||||
|
||||
return skb;
|
||||
}
|
||||
|
||||
@ -335,7 +352,8 @@ mt76_release_buffered_frames(struct ieee80211_hw *hw, struct ieee80211_sta *sta,
|
||||
enum ieee80211_frame_release_type reason,
|
||||
bool more_data)
|
||||
{
|
||||
struct mt76_dev *dev = hw->priv;
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
struct sk_buff *last_skb = NULL;
|
||||
struct mt76_queue *hwq = dev->q_tx[MT_TXQ_PSD].q;
|
||||
int i;
|
||||
@ -350,7 +368,7 @@ mt76_release_buffered_frames(struct ieee80211_hw *hw, struct ieee80211_sta *sta,
|
||||
continue;
|
||||
|
||||
do {
|
||||
skb = mt76_txq_dequeue(dev, mtxq, true);
|
||||
skb = mt76_txq_dequeue(phy, mtxq, true);
|
||||
if (!skb)
|
||||
break;
|
||||
|
||||
@ -377,9 +395,10 @@ mt76_release_buffered_frames(struct ieee80211_hw *hw, struct ieee80211_sta *sta,
|
||||
EXPORT_SYMBOL_GPL(mt76_release_buffered_frames);
|
||||
|
||||
static int
|
||||
mt76_txq_send_burst(struct mt76_dev *dev, struct mt76_sw_queue *sq,
|
||||
mt76_txq_send_burst(struct mt76_phy *phy, struct mt76_sw_queue *sq,
|
||||
struct mt76_txq *mtxq)
|
||||
{
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
struct ieee80211_txq *txq = mtxq_to_txq(mtxq);
|
||||
enum mt76_txq_id qid = mt76_txq_get_qid(txq);
|
||||
struct mt76_wcid *wcid = mtxq->wcid;
|
||||
@ -395,7 +414,7 @@ mt76_txq_send_burst(struct mt76_dev *dev, struct mt76_sw_queue *sq,
|
||||
if (test_bit(MT_WCID_FLAG_PS, &wcid->flags))
|
||||
return 0;
|
||||
|
||||
skb = mt76_txq_dequeue(dev, mtxq, false);
|
||||
skb = mt76_txq_dequeue(phy, mtxq, false);
|
||||
if (!skb)
|
||||
return 0;
|
||||
|
||||
@ -423,10 +442,10 @@ mt76_txq_send_burst(struct mt76_dev *dev, struct mt76_sw_queue *sq,
|
||||
if (probe)
|
||||
break;
|
||||
|
||||
if (test_bit(MT76_RESET, &dev->state))
|
||||
if (test_bit(MT76_RESET, &phy->state))
|
||||
return -EBUSY;
|
||||
|
||||
skb = mt76_txq_dequeue(dev, mtxq, false);
|
||||
skb = mt76_txq_dequeue(phy, mtxq, false);
|
||||
if (!skb)
|
||||
break;
|
||||
|
||||
@ -464,8 +483,9 @@ mt76_txq_send_burst(struct mt76_dev *dev, struct mt76_sw_queue *sq,
|
||||
}
|
||||
|
||||
static int
|
||||
mt76_txq_schedule_list(struct mt76_dev *dev, enum mt76_txq_id qid)
|
||||
mt76_txq_schedule_list(struct mt76_phy *phy, enum mt76_txq_id qid)
|
||||
{
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
struct mt76_sw_queue *sq = &dev->q_tx[qid];
|
||||
struct mt76_queue *hwq = sq->q;
|
||||
struct ieee80211_txq *txq;
|
||||
@ -478,12 +498,12 @@ mt76_txq_schedule_list(struct mt76_dev *dev, enum mt76_txq_id qid)
|
||||
if (sq->swq_queued >= 4)
|
||||
break;
|
||||
|
||||
if (test_bit(MT76_RESET, &dev->state)) {
|
||||
if (test_bit(MT76_RESET, &phy->state)) {
|
||||
ret = -EBUSY;
|
||||
break;
|
||||
}
|
||||
|
||||
txq = ieee80211_next_txq(dev->hw, qid);
|
||||
txq = ieee80211_next_txq(phy->hw, qid);
|
||||
if (!txq)
|
||||
break;
|
||||
|
||||
@ -505,8 +525,8 @@ mt76_txq_schedule_list(struct mt76_dev *dev, enum mt76_txq_id qid)
|
||||
spin_lock_bh(&hwq->lock);
|
||||
}
|
||||
|
||||
ret += mt76_txq_send_burst(dev, sq, mtxq);
|
||||
ieee80211_return_txq(dev->hw, txq,
|
||||
ret += mt76_txq_send_burst(phy, sq, mtxq);
|
||||
ieee80211_return_txq(phy->hw, txq,
|
||||
!skb_queue_empty(&mtxq->retry_q));
|
||||
}
|
||||
spin_unlock_bh(&hwq->lock);
|
||||
@ -514,8 +534,9 @@ mt76_txq_schedule_list(struct mt76_dev *dev, enum mt76_txq_id qid)
|
||||
return ret;
|
||||
}
|
||||
|
||||
void mt76_txq_schedule(struct mt76_dev *dev, enum mt76_txq_id qid)
|
||||
void mt76_txq_schedule(struct mt76_phy *phy, enum mt76_txq_id qid)
|
||||
{
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
struct mt76_sw_queue *sq = &dev->q_tx[qid];
|
||||
int len;
|
||||
|
||||
@ -528,21 +549,21 @@ void mt76_txq_schedule(struct mt76_dev *dev, enum mt76_txq_id qid)
|
||||
rcu_read_lock();
|
||||
|
||||
do {
|
||||
ieee80211_txq_schedule_start(dev->hw, qid);
|
||||
len = mt76_txq_schedule_list(dev, qid);
|
||||
ieee80211_txq_schedule_end(dev->hw, qid);
|
||||
ieee80211_txq_schedule_start(phy->hw, qid);
|
||||
len = mt76_txq_schedule_list(phy, qid);
|
||||
ieee80211_txq_schedule_end(phy->hw, qid);
|
||||
} while (len > 0);
|
||||
|
||||
rcu_read_unlock();
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_txq_schedule);
|
||||
|
||||
void mt76_txq_schedule_all(struct mt76_dev *dev)
|
||||
void mt76_txq_schedule_all(struct mt76_phy *phy)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i <= MT_TXQ_BK; i++)
|
||||
mt76_txq_schedule(dev, i);
|
||||
mt76_txq_schedule(phy, i);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_txq_schedule_all);
|
||||
|
||||
@ -550,7 +571,9 @@ void mt76_tx_tasklet(unsigned long data)
|
||||
{
|
||||
struct mt76_dev *dev = (struct mt76_dev *)data;
|
||||
|
||||
mt76_txq_schedule_all(dev);
|
||||
mt76_txq_schedule_all(&dev->phy);
|
||||
if (dev->phy2)
|
||||
mt76_txq_schedule_all(dev->phy2);
|
||||
}
|
||||
|
||||
void mt76_stop_tx_queues(struct mt76_dev *dev, struct ieee80211_sta *sta,
|
||||
@ -578,9 +601,10 @@ EXPORT_SYMBOL_GPL(mt76_stop_tx_queues);
|
||||
|
||||
void mt76_wake_tx_queue(struct ieee80211_hw *hw, struct ieee80211_txq *txq)
|
||||
{
|
||||
struct mt76_dev *dev = hw->priv;
|
||||
struct mt76_phy *phy = hw->priv;
|
||||
struct mt76_dev *dev = phy->dev;
|
||||
|
||||
if (!test_bit(MT76_STATE_RUNNING, &dev->state))
|
||||
if (!test_bit(MT76_STATE_RUNNING, &phy->state))
|
||||
return;
|
||||
|
||||
tasklet_schedule(&dev->tx_tasklet);
|
||||
@ -589,6 +613,7 @@ EXPORT_SYMBOL_GPL(mt76_wake_tx_queue);
|
||||
|
||||
void mt76_txq_remove(struct mt76_dev *dev, struct ieee80211_txq *txq)
|
||||
{
|
||||
struct ieee80211_hw *hw;
|
||||
struct mt76_txq *mtxq;
|
||||
struct sk_buff *skb;
|
||||
|
||||
@ -597,8 +622,10 @@ void mt76_txq_remove(struct mt76_dev *dev, struct ieee80211_txq *txq)
|
||||
|
||||
mtxq = (struct mt76_txq *)txq->drv_priv;
|
||||
|
||||
while ((skb = skb_dequeue(&mtxq->retry_q)) != NULL)
|
||||
ieee80211_free_txskb(dev->hw, skb);
|
||||
while ((skb = skb_dequeue(&mtxq->retry_q)) != NULL) {
|
||||
hw = mt76_tx_status_get_hw(dev, skb);
|
||||
ieee80211_free_txskb(hw, skb);
|
||||
}
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_txq_remove);
|
||||
|
||||
|
@ -29,13 +29,13 @@ static int __mt76u_vendor_request(struct mt76_dev *dev, u8 req,
|
||||
pipe = (req_type & USB_DIR_IN) ? usb_rcvctrlpipe(udev, 0)
|
||||
: usb_sndctrlpipe(udev, 0);
|
||||
for (i = 0; i < MT_VEND_REQ_MAX_RETRY; i++) {
|
||||
if (test_bit(MT76_REMOVED, &dev->state))
|
||||
if (test_bit(MT76_REMOVED, &dev->phy.state))
|
||||
return -EIO;
|
||||
|
||||
ret = usb_control_msg(udev, pipe, req, req_type, val,
|
||||
offset, buf, len, MT_VEND_REQ_TOUT_MS);
|
||||
if (ret == -ENODEV)
|
||||
set_bit(MT76_REMOVED, &dev->state);
|
||||
set_bit(MT76_REMOVED, &dev->phy.state);
|
||||
if (ret >= 0 || ret == -ENODEV)
|
||||
return ret;
|
||||
usleep_range(5000, 10000);
|
||||
@ -62,12 +62,25 @@ int mt76u_vendor_request(struct mt76_dev *dev, u8 req,
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76u_vendor_request);
|
||||
|
||||
static u32 __mt76u_rr(struct mt76_dev *dev, u32 addr)
|
||||
static u32 ___mt76u_rr(struct mt76_dev *dev, u8 req, u32 addr)
|
||||
{
|
||||
struct mt76_usb *usb = &dev->usb;
|
||||
u32 data = ~0;
|
||||
u16 offset;
|
||||
int ret;
|
||||
|
||||
ret = __mt76u_vendor_request(dev, req,
|
||||
USB_DIR_IN | USB_TYPE_VENDOR,
|
||||
addr >> 16, addr, &usb->reg_val,
|
||||
sizeof(__le32));
|
||||
if (ret == sizeof(__le32))
|
||||
data = le32_to_cpu(usb->reg_val);
|
||||
trace_usb_reg_rr(dev, addr, data);
|
||||
|
||||
return data;
|
||||
}
|
||||
|
||||
static u32 __mt76u_rr(struct mt76_dev *dev, u32 addr)
|
||||
{
|
||||
u8 req;
|
||||
|
||||
switch (addr & MT_VEND_TYPE_MASK) {
|
||||
@ -81,16 +94,8 @@ static u32 __mt76u_rr(struct mt76_dev *dev, u32 addr)
|
||||
req = MT_VEND_MULTI_READ;
|
||||
break;
|
||||
}
|
||||
offset = addr & ~MT_VEND_TYPE_MASK;
|
||||
|
||||
ret = __mt76u_vendor_request(dev, req,
|
||||
USB_DIR_IN | USB_TYPE_VENDOR,
|
||||
0, offset, &usb->reg_val, sizeof(__le32));
|
||||
if (ret == sizeof(__le32))
|
||||
data = le32_to_cpu(usb->reg_val);
|
||||
trace_usb_reg_rr(dev, addr, data);
|
||||
|
||||
return data;
|
||||
return ___mt76u_rr(dev, req, addr & ~MT_VEND_TYPE_MASK);
|
||||
}
|
||||
|
||||
static u32 mt76u_rr(struct mt76_dev *dev, u32 addr)
|
||||
@ -104,10 +109,32 @@ static u32 mt76u_rr(struct mt76_dev *dev, u32 addr)
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void __mt76u_wr(struct mt76_dev *dev, u32 addr, u32 val)
|
||||
static u32 mt76u_rr_ext(struct mt76_dev *dev, u32 addr)
|
||||
{
|
||||
u32 ret;
|
||||
|
||||
mutex_lock(&dev->usb.usb_ctrl_mtx);
|
||||
ret = ___mt76u_rr(dev, MT_VEND_READ_EXT, addr);
|
||||
mutex_unlock(&dev->usb.usb_ctrl_mtx);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void ___mt76u_wr(struct mt76_dev *dev, u8 req,
|
||||
u32 addr, u32 val)
|
||||
{
|
||||
struct mt76_usb *usb = &dev->usb;
|
||||
u16 offset;
|
||||
|
||||
usb->reg_val = cpu_to_le32(val);
|
||||
__mt76u_vendor_request(dev, req,
|
||||
USB_DIR_OUT | USB_TYPE_VENDOR,
|
||||
addr >> 16, addr, &usb->reg_val,
|
||||
sizeof(__le32));
|
||||
trace_usb_reg_wr(dev, addr, val);
|
||||
}
|
||||
|
||||
static void __mt76u_wr(struct mt76_dev *dev, u32 addr, u32 val)
|
||||
{
|
||||
u8 req;
|
||||
|
||||
switch (addr & MT_VEND_TYPE_MASK) {
|
||||
@ -118,13 +145,7 @@ static void __mt76u_wr(struct mt76_dev *dev, u32 addr, u32 val)
|
||||
req = MT_VEND_MULTI_WRITE;
|
||||
break;
|
||||
}
|
||||
offset = addr & ~MT_VEND_TYPE_MASK;
|
||||
|
||||
usb->reg_val = cpu_to_le32(val);
|
||||
__mt76u_vendor_request(dev, req,
|
||||
USB_DIR_OUT | USB_TYPE_VENDOR, 0,
|
||||
offset, &usb->reg_val, sizeof(__le32));
|
||||
trace_usb_reg_wr(dev, addr, val);
|
||||
___mt76u_wr(dev, req, addr & ~MT_VEND_TYPE_MASK, val);
|
||||
}
|
||||
|
||||
static void mt76u_wr(struct mt76_dev *dev, u32 addr, u32 val)
|
||||
@ -134,6 +155,13 @@ static void mt76u_wr(struct mt76_dev *dev, u32 addr, u32 val)
|
||||
mutex_unlock(&dev->usb.usb_ctrl_mtx);
|
||||
}
|
||||
|
||||
static void mt76u_wr_ext(struct mt76_dev *dev, u32 addr, u32 val)
|
||||
{
|
||||
mutex_lock(&dev->usb.usb_ctrl_mtx);
|
||||
___mt76u_wr(dev, MT_VEND_WRITE_EXT, addr, val);
|
||||
mutex_unlock(&dev->usb.usb_ctrl_mtx);
|
||||
}
|
||||
|
||||
static u32 mt76u_rmw(struct mt76_dev *dev, u32 addr,
|
||||
u32 mask, u32 val)
|
||||
{
|
||||
@ -145,22 +173,94 @@ static u32 mt76u_rmw(struct mt76_dev *dev, u32 addr,
|
||||
return val;
|
||||
}
|
||||
|
||||
static u32 mt76u_rmw_ext(struct mt76_dev *dev, u32 addr,
|
||||
u32 mask, u32 val)
|
||||
{
|
||||
mutex_lock(&dev->usb.usb_ctrl_mtx);
|
||||
val |= ___mt76u_rr(dev, MT_VEND_READ_EXT, addr) & ~mask;
|
||||
___mt76u_wr(dev, MT_VEND_WRITE_EXT, addr, val);
|
||||
mutex_unlock(&dev->usb.usb_ctrl_mtx);
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
static void mt76u_copy(struct mt76_dev *dev, u32 offset,
|
||||
const void *data, int len)
|
||||
{
|
||||
struct mt76_usb *usb = &dev->usb;
|
||||
const u32 *val = data;
|
||||
int i, ret;
|
||||
const u8 *val = data;
|
||||
int ret;
|
||||
int current_batch_size;
|
||||
int i = 0;
|
||||
|
||||
/* Assure that always a multiple of 4 bytes are copied,
|
||||
* otherwise beacons can be corrupted.
|
||||
* See: "mt76: round up length on mt76_wr_copy"
|
||||
* Commit 850e8f6fbd5d0003b0
|
||||
*/
|
||||
len = round_up(len, 4);
|
||||
|
||||
mutex_lock(&usb->usb_ctrl_mtx);
|
||||
for (i = 0; i < DIV_ROUND_UP(len, 4); i++) {
|
||||
put_unaligned(val[i], (u32 *)usb->data);
|
||||
while (i < len) {
|
||||
current_batch_size = min_t(int, usb->data_len, len - i);
|
||||
memcpy(usb->data, val + i, current_batch_size);
|
||||
ret = __mt76u_vendor_request(dev, MT_VEND_MULTI_WRITE,
|
||||
USB_DIR_OUT | USB_TYPE_VENDOR,
|
||||
0, offset + i * 4, usb->data,
|
||||
sizeof(u32));
|
||||
0, offset + i, usb->data,
|
||||
current_batch_size);
|
||||
if (ret < 0)
|
||||
break;
|
||||
|
||||
i += current_batch_size;
|
||||
}
|
||||
mutex_unlock(&usb->usb_ctrl_mtx);
|
||||
}
|
||||
|
||||
static void mt76u_copy_ext(struct mt76_dev *dev, u32 offset,
|
||||
const void *data, int len)
|
||||
{
|
||||
struct mt76_usb *usb = &dev->usb;
|
||||
int ret, i = 0, batch_len;
|
||||
const u8 *val = data;
|
||||
|
||||
len = round_up(len, 4);
|
||||
mutex_lock(&usb->usb_ctrl_mtx);
|
||||
while (i < len) {
|
||||
batch_len = min_t(int, usb->data_len, len - i);
|
||||
memcpy(usb->data, val + i, batch_len);
|
||||
ret = __mt76u_vendor_request(dev, MT_VEND_WRITE_EXT,
|
||||
USB_DIR_OUT | USB_TYPE_VENDOR,
|
||||
(offset + i) >> 16, offset + i,
|
||||
usb->data, batch_len);
|
||||
if (ret < 0)
|
||||
break;
|
||||
|
||||
i += batch_len;
|
||||
}
|
||||
mutex_unlock(&usb->usb_ctrl_mtx);
|
||||
}
|
||||
|
||||
static void
|
||||
mt76u_read_copy_ext(struct mt76_dev *dev, u32 offset,
|
||||
void *data, int len)
|
||||
{
|
||||
struct mt76_usb *usb = &dev->usb;
|
||||
int i = 0, batch_len, ret;
|
||||
u8 *val = data;
|
||||
|
||||
len = round_up(len, 4);
|
||||
mutex_lock(&usb->usb_ctrl_mtx);
|
||||
while (i < len) {
|
||||
batch_len = min_t(int, usb->data_len, len - i);
|
||||
ret = __mt76u_vendor_request(dev, MT_VEND_READ_EXT,
|
||||
USB_DIR_IN | USB_TYPE_VENDOR,
|
||||
(offset + i) >> 16, offset + i,
|
||||
usb->data, batch_len);
|
||||
if (ret < 0)
|
||||
break;
|
||||
|
||||
memcpy(val + i, usb->data, batch_len);
|
||||
i += batch_len;
|
||||
}
|
||||
mutex_unlock(&usb->usb_ctrl_mtx);
|
||||
}
|
||||
@ -200,7 +300,7 @@ static int
|
||||
mt76u_wr_rp(struct mt76_dev *dev, u32 base,
|
||||
const struct mt76_reg_pair *data, int n)
|
||||
{
|
||||
if (test_bit(MT76_STATE_MCU_RUNNING, &dev->state))
|
||||
if (test_bit(MT76_STATE_MCU_RUNNING, &dev->phy.state))
|
||||
return dev->mcu_ops->mcu_wr_rp(dev, base, data, n);
|
||||
else
|
||||
return mt76u_req_wr_rp(dev, base, data, n);
|
||||
@ -227,7 +327,7 @@ static int
|
||||
mt76u_rd_rp(struct mt76_dev *dev, u32 base,
|
||||
struct mt76_reg_pair *data, int n)
|
||||
{
|
||||
if (test_bit(MT76_STATE_MCU_RUNNING, &dev->state))
|
||||
if (test_bit(MT76_STATE_MCU_RUNNING, &dev->phy.state))
|
||||
return dev->mcu_ops->mcu_rd_rp(dev, base, data, n);
|
||||
else
|
||||
return mt76u_req_rd_rp(dev, base, data, n);
|
||||
@ -306,11 +406,12 @@ mt76u_fill_rx_sg(struct mt76_dev *dev, struct mt76_queue *q, struct urb *urb,
|
||||
}
|
||||
|
||||
static int
|
||||
mt76u_refill_rx(struct mt76_dev *dev, struct urb *urb, int nsgs, gfp_t gfp)
|
||||
mt76u_refill_rx(struct mt76_dev *dev, struct mt76_queue *q,
|
||||
struct urb *urb, int nsgs, gfp_t gfp)
|
||||
{
|
||||
struct mt76_queue *q = &dev->q_rx[MT_RXQ_MAIN];
|
||||
enum mt76_rxq_id qid = q - &dev->q_rx[MT_RXQ_MAIN];
|
||||
|
||||
if (dev->usb.sg_en)
|
||||
if (qid == MT_RXQ_MAIN && dev->usb.sg_en)
|
||||
return mt76u_fill_rx_sg(dev, q, urb, nsgs, gfp);
|
||||
|
||||
urb->transfer_buffer_length = q->buf_size;
|
||||
@ -334,23 +435,25 @@ mt76u_urb_alloc(struct mt76_dev *dev, struct mt76_queue_entry *e,
|
||||
|
||||
usb_init_urb(e->urb);
|
||||
|
||||
if (dev->usb.sg_en)
|
||||
if (dev->usb.sg_en && sg_max_size > 0)
|
||||
e->urb->sg = (struct scatterlist *)(e->urb + 1);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
mt76u_rx_urb_alloc(struct mt76_dev *dev, struct mt76_queue_entry *e)
|
||||
mt76u_rx_urb_alloc(struct mt76_dev *dev, struct mt76_queue *q,
|
||||
struct mt76_queue_entry *e)
|
||||
{
|
||||
int err;
|
||||
enum mt76_rxq_id qid = q - &dev->q_rx[MT_RXQ_MAIN];
|
||||
int err, sg_size;
|
||||
|
||||
err = mt76u_urb_alloc(dev, e, MT_RX_SG_MAX_SIZE);
|
||||
sg_size = qid == MT_RXQ_MAIN ? MT_RX_SG_MAX_SIZE : 0;
|
||||
err = mt76u_urb_alloc(dev, e, sg_size);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
return mt76u_refill_rx(dev, e->urb, MT_RX_SG_MAX_SIZE,
|
||||
GFP_KERNEL);
|
||||
return mt76u_refill_rx(dev, q, e->urb, sg_size, GFP_KERNEL);
|
||||
}
|
||||
|
||||
static void mt76u_urb_free(struct urb *urb)
|
||||
@ -386,10 +489,9 @@ mt76u_fill_bulk_urb(struct mt76_dev *dev, int dir, int index,
|
||||
urb->context = context;
|
||||
}
|
||||
|
||||
static inline struct urb *
|
||||
mt76u_get_next_rx_entry(struct mt76_dev *dev)
|
||||
static struct urb *
|
||||
mt76u_get_next_rx_entry(struct mt76_queue *q)
|
||||
{
|
||||
struct mt76_queue *q = &dev->q_rx[MT_RXQ_MAIN];
|
||||
struct urb *urb = NULL;
|
||||
unsigned long flags;
|
||||
|
||||
@ -404,14 +506,17 @@ mt76u_get_next_rx_entry(struct mt76_dev *dev)
|
||||
return urb;
|
||||
}
|
||||
|
||||
static int mt76u_get_rx_entry_len(u8 *data, u32 data_len)
|
||||
static int
|
||||
mt76u_get_rx_entry_len(struct mt76_dev *dev, u8 *data,
|
||||
u32 data_len)
|
||||
{
|
||||
u16 dma_len, min_len;
|
||||
|
||||
dma_len = get_unaligned_le16(data);
|
||||
min_len = MT_DMA_HDR_LEN + MT_RX_RXWI_LEN +
|
||||
MT_FCE_INFO_LEN;
|
||||
if (dev->drv->drv_flags & MT_DRV_RX_DMA_HDR)
|
||||
return dma_len;
|
||||
|
||||
min_len = MT_DMA_HDR_LEN + MT_RX_RXWI_LEN + MT_FCE_INFO_LEN;
|
||||
if (data_len < min_len || !dma_len ||
|
||||
dma_len + MT_DMA_HDR_LEN > data_len ||
|
||||
(dma_len & 0x3))
|
||||
@ -420,11 +525,14 @@ static int mt76u_get_rx_entry_len(u8 *data, u32 data_len)
|
||||
}
|
||||
|
||||
static struct sk_buff *
|
||||
mt76u_build_rx_skb(void *data, int len, int buf_size)
|
||||
mt76u_build_rx_skb(struct mt76_dev *dev, void *data,
|
||||
int len, int buf_size)
|
||||
{
|
||||
int head_room, drv_flags = dev->drv->drv_flags;
|
||||
struct sk_buff *skb;
|
||||
|
||||
if (SKB_WITH_OVERHEAD(buf_size) < MT_DMA_HDR_LEN + len) {
|
||||
head_room = drv_flags & MT_DRV_RX_DMA_HDR ? 0 : MT_DMA_HDR_LEN;
|
||||
if (SKB_WITH_OVERHEAD(buf_size) < head_room + len) {
|
||||
struct page *page;
|
||||
|
||||
/* slow path, not enough space for data and
|
||||
@ -434,8 +542,8 @@ mt76u_build_rx_skb(void *data, int len, int buf_size)
|
||||
if (!skb)
|
||||
return NULL;
|
||||
|
||||
skb_put_data(skb, data + MT_DMA_HDR_LEN, MT_SKB_HEAD_LEN);
|
||||
data += (MT_DMA_HDR_LEN + MT_SKB_HEAD_LEN);
|
||||
skb_put_data(skb, data + head_room, MT_SKB_HEAD_LEN);
|
||||
data += head_room + MT_SKB_HEAD_LEN;
|
||||
page = virt_to_head_page(data);
|
||||
skb_add_rx_frag(skb, skb_shinfo(skb)->nr_frags,
|
||||
page, data - page_address(page),
|
||||
@ -449,30 +557,31 @@ mt76u_build_rx_skb(void *data, int len, int buf_size)
|
||||
if (!skb)
|
||||
return NULL;
|
||||
|
||||
skb_reserve(skb, MT_DMA_HDR_LEN);
|
||||
skb_reserve(skb, head_room);
|
||||
__skb_put(skb, len);
|
||||
|
||||
return skb;
|
||||
}
|
||||
|
||||
static int
|
||||
mt76u_process_rx_entry(struct mt76_dev *dev, struct urb *urb)
|
||||
mt76u_process_rx_entry(struct mt76_dev *dev, struct urb *urb,
|
||||
int buf_size)
|
||||
{
|
||||
struct mt76_queue *q = &dev->q_rx[MT_RXQ_MAIN];
|
||||
u8 *data = urb->num_sgs ? sg_virt(&urb->sg[0]) : urb->transfer_buffer;
|
||||
int data_len = urb->num_sgs ? urb->sg[0].length : urb->actual_length;
|
||||
int len, nsgs = 1;
|
||||
int len, nsgs = 1, head_room, drv_flags = dev->drv->drv_flags;
|
||||
struct sk_buff *skb;
|
||||
|
||||
if (!test_bit(MT76_STATE_INITIALIZED, &dev->state))
|
||||
if (!test_bit(MT76_STATE_INITIALIZED, &dev->phy.state))
|
||||
return 0;
|
||||
|
||||
len = mt76u_get_rx_entry_len(data, urb->actual_length);
|
||||
len = mt76u_get_rx_entry_len(dev, data, urb->actual_length);
|
||||
if (len < 0)
|
||||
return 0;
|
||||
|
||||
data_len = min_t(int, len, data_len - MT_DMA_HDR_LEN);
|
||||
skb = mt76u_build_rx_skb(data, data_len, q->buf_size);
|
||||
head_room = drv_flags & MT_DRV_RX_DMA_HDR ? 0 : MT_DMA_HDR_LEN;
|
||||
data_len = min_t(int, len, data_len - head_room);
|
||||
skb = mt76u_build_rx_skb(dev, data, data_len, buf_size);
|
||||
if (!skb)
|
||||
return 0;
|
||||
|
||||
@ -481,8 +590,8 @@ mt76u_process_rx_entry(struct mt76_dev *dev, struct urb *urb)
|
||||
data_len = min_t(int, len, urb->sg[nsgs].length);
|
||||
skb_add_rx_frag(skb, skb_shinfo(skb)->nr_frags,
|
||||
sg_page(&urb->sg[nsgs]),
|
||||
urb->sg[nsgs].offset,
|
||||
data_len, q->buf_size);
|
||||
urb->sg[nsgs].offset, data_len,
|
||||
buf_size);
|
||||
len -= data_len;
|
||||
nsgs++;
|
||||
}
|
||||
@ -493,8 +602,8 @@ mt76u_process_rx_entry(struct mt76_dev *dev, struct urb *urb)
|
||||
|
||||
static void mt76u_complete_rx(struct urb *urb)
|
||||
{
|
||||
struct mt76_dev *dev = urb->context;
|
||||
struct mt76_queue *q = &dev->q_rx[MT_RXQ_MAIN];
|
||||
struct mt76_dev *dev = dev_get_drvdata(&urb->dev->dev);
|
||||
struct mt76_queue *q = urb->context;
|
||||
unsigned long flags;
|
||||
|
||||
trace_rx_urb(dev, urb);
|
||||
@ -524,50 +633,62 @@ out:
|
||||
}
|
||||
|
||||
static int
|
||||
mt76u_submit_rx_buf(struct mt76_dev *dev, struct urb *urb)
|
||||
mt76u_submit_rx_buf(struct mt76_dev *dev, enum mt76_rxq_id qid,
|
||||
struct urb *urb)
|
||||
{
|
||||
mt76u_fill_bulk_urb(dev, USB_DIR_IN, MT_EP_IN_PKT_RX, urb,
|
||||
mt76u_complete_rx, dev);
|
||||
int ep = qid == MT_RXQ_MAIN ? MT_EP_IN_PKT_RX : MT_EP_IN_CMD_RESP;
|
||||
|
||||
mt76u_fill_bulk_urb(dev, USB_DIR_IN, ep, urb,
|
||||
mt76u_complete_rx, &dev->q_rx[qid]);
|
||||
trace_submit_urb(dev, urb);
|
||||
|
||||
return usb_submit_urb(urb, GFP_ATOMIC);
|
||||
}
|
||||
|
||||
static void mt76u_rx_tasklet(unsigned long data)
|
||||
static void
|
||||
mt76u_process_rx_queue(struct mt76_dev *dev, struct mt76_queue *q)
|
||||
{
|
||||
struct mt76_dev *dev = (struct mt76_dev *)data;
|
||||
int qid = q - &dev->q_rx[MT_RXQ_MAIN];
|
||||
struct urb *urb;
|
||||
int err, count;
|
||||
|
||||
rcu_read_lock();
|
||||
|
||||
while (true) {
|
||||
urb = mt76u_get_next_rx_entry(dev);
|
||||
urb = mt76u_get_next_rx_entry(q);
|
||||
if (!urb)
|
||||
break;
|
||||
|
||||
count = mt76u_process_rx_entry(dev, urb);
|
||||
count = mt76u_process_rx_entry(dev, urb, q->buf_size);
|
||||
if (count > 0) {
|
||||
err = mt76u_refill_rx(dev, urb, count, GFP_ATOMIC);
|
||||
err = mt76u_refill_rx(dev, q, urb, count, GFP_ATOMIC);
|
||||
if (err < 0)
|
||||
break;
|
||||
}
|
||||
mt76u_submit_rx_buf(dev, urb);
|
||||
mt76u_submit_rx_buf(dev, qid, urb);
|
||||
}
|
||||
mt76_rx_poll_complete(dev, MT_RXQ_MAIN, NULL);
|
||||
if (qid == MT_RXQ_MAIN)
|
||||
mt76_rx_poll_complete(dev, MT_RXQ_MAIN, NULL);
|
||||
}
|
||||
|
||||
static void mt76u_rx_tasklet(unsigned long data)
|
||||
{
|
||||
struct mt76_dev *dev = (struct mt76_dev *)data;
|
||||
struct mt76_queue *q = &dev->q_rx[MT_RXQ_MAIN];
|
||||
|
||||
rcu_read_lock();
|
||||
mt76u_process_rx_queue(dev, q);
|
||||
rcu_read_unlock();
|
||||
}
|
||||
|
||||
static int mt76u_submit_rx_buffers(struct mt76_dev *dev)
|
||||
static int
|
||||
mt76u_submit_rx_buffers(struct mt76_dev *dev, enum mt76_rxq_id qid)
|
||||
{
|
||||
struct mt76_queue *q = &dev->q_rx[MT_RXQ_MAIN];
|
||||
struct mt76_queue *q = &dev->q_rx[qid];
|
||||
unsigned long flags;
|
||||
int i, err = 0;
|
||||
|
||||
spin_lock_irqsave(&q->lock, flags);
|
||||
for (i = 0; i < q->ndesc; i++) {
|
||||
err = mt76u_submit_rx_buf(dev, q->entry[i].urb);
|
||||
err = mt76u_submit_rx_buf(dev, qid, q->entry[i].urb);
|
||||
if (err < 0)
|
||||
break;
|
||||
}
|
||||
@ -578,16 +699,12 @@ static int mt76u_submit_rx_buffers(struct mt76_dev *dev)
|
||||
return err;
|
||||
}
|
||||
|
||||
static int mt76u_alloc_rx(struct mt76_dev *dev)
|
||||
static int
|
||||
mt76u_alloc_rx_queue(struct mt76_dev *dev, enum mt76_rxq_id qid)
|
||||
{
|
||||
struct mt76_usb *usb = &dev->usb;
|
||||
struct mt76_queue *q = &dev->q_rx[MT_RXQ_MAIN];
|
||||
struct mt76_queue *q = &dev->q_rx[qid];
|
||||
int i, err;
|
||||
|
||||
usb->mcu.data = devm_kmalloc(dev->dev, MCU_RESP_URB_SIZE, GFP_KERNEL);
|
||||
if (!usb->mcu.data)
|
||||
return -ENOMEM;
|
||||
|
||||
spin_lock_init(&q->lock);
|
||||
q->entry = devm_kcalloc(dev->dev,
|
||||
MT_NUM_RX_ENTRIES, sizeof(*q->entry),
|
||||
@ -599,17 +716,23 @@ static int mt76u_alloc_rx(struct mt76_dev *dev)
|
||||
q->buf_size = PAGE_SIZE;
|
||||
|
||||
for (i = 0; i < q->ndesc; i++) {
|
||||
err = mt76u_rx_urb_alloc(dev, &q->entry[i]);
|
||||
err = mt76u_rx_urb_alloc(dev, q, &q->entry[i]);
|
||||
if (err < 0)
|
||||
return err;
|
||||
}
|
||||
|
||||
return mt76u_submit_rx_buffers(dev);
|
||||
return mt76u_submit_rx_buffers(dev, qid);
|
||||
}
|
||||
|
||||
static void mt76u_free_rx(struct mt76_dev *dev)
|
||||
int mt76u_alloc_mcu_queue(struct mt76_dev *dev)
|
||||
{
|
||||
return mt76u_alloc_rx_queue(dev, MT_RXQ_MCU);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76u_alloc_mcu_queue);
|
||||
|
||||
static void
|
||||
mt76u_free_rx_queue(struct mt76_dev *dev, struct mt76_queue *q)
|
||||
{
|
||||
struct mt76_queue *q = &dev->q_rx[MT_RXQ_MAIN];
|
||||
struct page *page;
|
||||
int i;
|
||||
|
||||
@ -624,13 +747,33 @@ static void mt76u_free_rx(struct mt76_dev *dev)
|
||||
memset(&q->rx_page, 0, sizeof(q->rx_page));
|
||||
}
|
||||
|
||||
void mt76u_stop_rx(struct mt76_dev *dev)
|
||||
static void mt76u_free_rx(struct mt76_dev *dev)
|
||||
{
|
||||
struct mt76_queue *q = &dev->q_rx[MT_RXQ_MAIN];
|
||||
struct mt76_queue *q;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < q->ndesc; i++)
|
||||
usb_poison_urb(q->entry[i].urb);
|
||||
for (i = 0; i < __MT_RXQ_MAX; i++) {
|
||||
q = &dev->q_rx[i];
|
||||
if (!q->ndesc)
|
||||
continue;
|
||||
|
||||
mt76u_free_rx_queue(dev, q);
|
||||
}
|
||||
}
|
||||
|
||||
void mt76u_stop_rx(struct mt76_dev *dev)
|
||||
{
|
||||
struct mt76_queue *q;
|
||||
int i, j;
|
||||
|
||||
for (i = 0; i < __MT_RXQ_MAX; i++) {
|
||||
q = &dev->q_rx[i];
|
||||
if (!q->ndesc)
|
||||
continue;
|
||||
|
||||
for (j = 0; j < q->ndesc; j++)
|
||||
usb_poison_urb(q->entry[j].urb);
|
||||
}
|
||||
|
||||
tasklet_kill(&dev->usb.rx_tasklet);
|
||||
}
|
||||
@ -638,13 +781,24 @@ EXPORT_SYMBOL_GPL(mt76u_stop_rx);
|
||||
|
||||
int mt76u_resume_rx(struct mt76_dev *dev)
|
||||
{
|
||||
struct mt76_queue *q = &dev->q_rx[MT_RXQ_MAIN];
|
||||
int i;
|
||||
struct mt76_queue *q;
|
||||
int i, j, err;
|
||||
|
||||
for (i = 0; i < q->ndesc; i++)
|
||||
usb_unpoison_urb(q->entry[i].urb);
|
||||
for (i = 0; i < __MT_RXQ_MAX; i++) {
|
||||
q = &dev->q_rx[i];
|
||||
|
||||
return mt76u_submit_rx_buffers(dev);
|
||||
if (!q->ndesc)
|
||||
continue;
|
||||
|
||||
for (j = 0; j < q->ndesc; j++)
|
||||
usb_unpoison_urb(q->entry[j].urb);
|
||||
|
||||
err = mt76u_submit_rx_buffers(dev, i);
|
||||
if (err < 0)
|
||||
return err;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76u_resume_rx);
|
||||
|
||||
@ -694,10 +848,11 @@ static void mt76u_tx_tasklet(unsigned long data)
|
||||
|
||||
spin_unlock_bh(&q->lock);
|
||||
|
||||
mt76_txq_schedule(dev, i);
|
||||
mt76_txq_schedule(&dev->phy, i);
|
||||
|
||||
if (!test_and_set_bit(MT76_READING_STATS, &dev->state))
|
||||
queue_work(dev->usb.stat_wq, &dev->usb.stat_work);
|
||||
if (dev->drv->tx_status_data &&
|
||||
!test_and_set_bit(MT76_READING_STATS, &dev->phy.state))
|
||||
queue_work(dev->usb.wq, &dev->usb.stat_work);
|
||||
if (wake)
|
||||
ieee80211_wake_queue(dev->hw, i);
|
||||
}
|
||||
@ -714,7 +869,7 @@ static void mt76u_tx_status_data(struct work_struct *work)
|
||||
dev = container_of(usb, struct mt76_dev, usb);
|
||||
|
||||
while (true) {
|
||||
if (test_bit(MT76_REMOVED, &dev->state))
|
||||
if (test_bit(MT76_REMOVED, &dev->phy.state))
|
||||
break;
|
||||
|
||||
if (!dev->drv->tx_status_data(dev, &update))
|
||||
@ -722,10 +877,10 @@ static void mt76u_tx_status_data(struct work_struct *work)
|
||||
count++;
|
||||
}
|
||||
|
||||
if (count && test_bit(MT76_STATE_RUNNING, &dev->state))
|
||||
queue_work(usb->stat_wq, &usb->stat_work);
|
||||
if (count && test_bit(MT76_STATE_RUNNING, &dev->phy.state))
|
||||
queue_work(usb->wq, &usb->stat_work);
|
||||
else
|
||||
clear_bit(MT76_READING_STATS, &dev->state);
|
||||
clear_bit(MT76_READING_STATS, &dev->phy.state);
|
||||
}
|
||||
|
||||
static void mt76u_complete_tx(struct urb *urb)
|
||||
@ -759,6 +914,35 @@ mt76u_tx_setup_buffers(struct mt76_dev *dev, struct sk_buff *skb,
|
||||
return urb->num_sgs;
|
||||
}
|
||||
|
||||
int mt76u_skb_dma_info(struct sk_buff *skb, u32 info)
|
||||
{
|
||||
struct sk_buff *iter, *last = skb;
|
||||
u32 pad;
|
||||
|
||||
put_unaligned_le32(info, skb_push(skb, sizeof(info)));
|
||||
/* Add zero pad of 4 - 7 bytes */
|
||||
pad = round_up(skb->len, 4) + 4 - skb->len;
|
||||
|
||||
/* First packet of a A-MSDU burst keeps track of the whole burst
|
||||
* length, need to update length of it and the last packet.
|
||||
*/
|
||||
skb_walk_frags(skb, iter) {
|
||||
last = iter;
|
||||
if (!iter->next) {
|
||||
skb->data_len += pad;
|
||||
skb->len += pad;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (skb_pad(last, pad))
|
||||
return -ENOMEM;
|
||||
__skb_put(last, pad);
|
||||
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76u_skb_dma_info);
|
||||
|
||||
static int
|
||||
mt76u_tx_queue_skb(struct mt76_dev *dev, enum mt76_txq_id qid,
|
||||
struct sk_buff *skb, struct mt76_wcid *wcid,
|
||||
@ -806,7 +990,7 @@ static void mt76u_tx_kick(struct mt76_dev *dev, struct mt76_queue *q)
|
||||
err = usb_submit_urb(urb, GFP_ATOMIC);
|
||||
if (err < 0) {
|
||||
if (err == -ENODEV)
|
||||
set_bit(MT76_REMOVED, &dev->state);
|
||||
set_bit(MT76_REMOVED, &dev->phy.state);
|
||||
else
|
||||
dev_err(dev->dev, "tx urb submit failed:%d\n",
|
||||
err);
|
||||
@ -816,6 +1000,14 @@ static void mt76u_tx_kick(struct mt76_dev *dev, struct mt76_queue *q)
|
||||
}
|
||||
}
|
||||
|
||||
static u8 mt76u_ac_to_hwq(struct mt76_dev *dev, u8 ac)
|
||||
{
|
||||
if (mt76_chip(dev) == 0x7663)
|
||||
return ac ^ 0x3;
|
||||
|
||||
return mt76_ac_to_hwq(ac);
|
||||
}
|
||||
|
||||
static int mt76u_alloc_tx(struct mt76_dev *dev)
|
||||
{
|
||||
struct mt76_queue *q;
|
||||
@ -834,7 +1026,7 @@ static int mt76u_alloc_tx(struct mt76_dev *dev)
|
||||
return -ENOMEM;
|
||||
|
||||
spin_lock_init(&q->lock);
|
||||
q->hw_idx = mt76_ac_to_hwq(i);
|
||||
q->hw_idx = mt76u_ac_to_hwq(dev, i);
|
||||
dev->q_tx[i].q = q;
|
||||
|
||||
q->entry = devm_kcalloc(dev->dev,
|
||||
@ -872,7 +1064,7 @@ void mt76u_stop_tx(struct mt76_dev *dev)
|
||||
struct mt76_queue *q;
|
||||
int i, j, ret;
|
||||
|
||||
ret = wait_event_timeout(dev->tx_wait, !mt76_has_tx_pending(dev),
|
||||
ret = wait_event_timeout(dev->tx_wait, !mt76_has_tx_pending(&dev->phy),
|
||||
HZ / 5);
|
||||
if (!ret) {
|
||||
dev_err(dev->dev, "timed out waiting for pending tx\n");
|
||||
@ -905,7 +1097,7 @@ void mt76u_stop_tx(struct mt76_dev *dev)
|
||||
}
|
||||
|
||||
cancel_work_sync(&dev->usb.stat_work);
|
||||
clear_bit(MT76_READING_STATS, &dev->state);
|
||||
clear_bit(MT76_READING_STATS, &dev->phy.state);
|
||||
|
||||
mt76_tx_status_check(dev, NULL, true);
|
||||
}
|
||||
@ -925,7 +1117,7 @@ int mt76u_alloc_queues(struct mt76_dev *dev)
|
||||
{
|
||||
int err;
|
||||
|
||||
err = mt76u_alloc_rx(dev);
|
||||
err = mt76u_alloc_rx_queue(dev, MT_RXQ_MAIN);
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
@ -938,14 +1130,20 @@ static const struct mt76_queue_ops usb_queue_ops = {
|
||||
.kick = mt76u_tx_kick,
|
||||
};
|
||||
|
||||
int mt76u_init(struct mt76_dev *dev,
|
||||
struct usb_interface *intf)
|
||||
void mt76u_deinit(struct mt76_dev *dev)
|
||||
{
|
||||
static const struct mt76_bus_ops mt76u_ops = {
|
||||
.rr = mt76u_rr,
|
||||
.wr = mt76u_wr,
|
||||
.rmw = mt76u_rmw,
|
||||
.write_copy = mt76u_copy,
|
||||
if (dev->usb.wq) {
|
||||
destroy_workqueue(dev->usb.wq);
|
||||
dev->usb.wq = NULL;
|
||||
}
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76u_deinit);
|
||||
|
||||
int mt76u_init(struct mt76_dev *dev,
|
||||
struct usb_interface *intf, bool ext)
|
||||
{
|
||||
static struct mt76_bus_ops mt76u_ops = {
|
||||
.read_copy = mt76u_read_copy_ext,
|
||||
.wr_rp = mt76u_wr_rp,
|
||||
.rd_rp = mt76u_rd_rp,
|
||||
.type = MT76_BUS_USB,
|
||||
@ -953,16 +1151,28 @@ int mt76u_init(struct mt76_dev *dev,
|
||||
struct usb_device *udev = interface_to_usbdev(intf);
|
||||
struct mt76_usb *usb = &dev->usb;
|
||||
|
||||
mt76u_ops.rr = ext ? mt76u_rr_ext : mt76u_rr;
|
||||
mt76u_ops.wr = ext ? mt76u_wr_ext : mt76u_wr;
|
||||
mt76u_ops.rmw = ext ? mt76u_rmw_ext : mt76u_rmw;
|
||||
mt76u_ops.write_copy = ext ? mt76u_copy_ext : mt76u_copy;
|
||||
|
||||
tasklet_init(&usb->rx_tasklet, mt76u_rx_tasklet, (unsigned long)dev);
|
||||
tasklet_init(&dev->tx_tasklet, mt76u_tx_tasklet, (unsigned long)dev);
|
||||
INIT_WORK(&usb->stat_work, mt76u_tx_status_data);
|
||||
skb_queue_head_init(&dev->rx_skb[MT_RXQ_MAIN]);
|
||||
|
||||
usb->stat_wq = alloc_workqueue("mt76u", WQ_UNBOUND, 0);
|
||||
if (!usb->stat_wq)
|
||||
usb->wq = alloc_workqueue("mt76u", WQ_UNBOUND, 0);
|
||||
if (!usb->wq)
|
||||
return -ENOMEM;
|
||||
|
||||
mutex_init(&usb->mcu.mutex);
|
||||
usb->data_len = usb_maxpacket(udev, usb_sndctrlpipe(udev, 0), 1);
|
||||
if (usb->data_len < 32)
|
||||
usb->data_len = 32;
|
||||
|
||||
usb->data = devm_kmalloc(dev->dev, usb->data_len, GFP_KERNEL);
|
||||
if (!usb->data) {
|
||||
mt76u_deinit(dev);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
mutex_init(&usb->usb_ctrl_mtx);
|
||||
dev->bus = &mt76u_ops;
|
||||
@ -976,14 +1186,5 @@ int mt76u_init(struct mt76_dev *dev,
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76u_init);
|
||||
|
||||
void mt76u_deinit(struct mt76_dev *dev)
|
||||
{
|
||||
if (dev->usb.stat_wq) {
|
||||
destroy_workqueue(dev->usb.stat_wq);
|
||||
dev->usb.stat_wq = NULL;
|
||||
}
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76u_deinit);
|
||||
|
||||
MODULE_AUTHOR("Lorenzo Bianconi <lorenzo.bianconi83@gmail.com>");
|
||||
MODULE_LICENSE("Dual BSD/GPL");
|
||||
|
@ -64,7 +64,7 @@ int mt76_wcid_alloc(unsigned long *mask, int size)
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(mt76_wcid_alloc);
|
||||
|
||||
int mt76_get_min_avg_rssi(struct mt76_dev *dev)
|
||||
int mt76_get_min_avg_rssi(struct mt76_dev *dev, bool ext_phy)
|
||||
{
|
||||
struct mt76_wcid *wcid;
|
||||
int i, j, min_rssi = 0;
|
||||
@ -75,14 +75,18 @@ int mt76_get_min_avg_rssi(struct mt76_dev *dev)
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(dev->wcid_mask); i++) {
|
||||
unsigned long mask = dev->wcid_mask[i];
|
||||
unsigned long phy_mask = dev->wcid_phy_mask[i];
|
||||
|
||||
if (!mask)
|
||||
continue;
|
||||
|
||||
for (j = i * BITS_PER_LONG; mask; j++, mask >>= 1) {
|
||||
for (j = i * BITS_PER_LONG; mask; j++, mask >>= 1, phy_mask >>= 1) {
|
||||
if (!(mask & 1))
|
||||
continue;
|
||||
|
||||
if (!!(phy_mask & 1) != ext_phy)
|
||||
continue;
|
||||
|
||||
wcid = rcu_dereference(dev->wcid[j]);
|
||||
if (!wcid)
|
||||
continue;
|
||||
|
@ -16,8 +16,20 @@
|
||||
|
||||
int mt76_wcid_alloc(unsigned long *mask, int size);
|
||||
|
||||
static inline bool
|
||||
mt76_wcid_mask_test(unsigned long *mask, int idx)
|
||||
{
|
||||
return mask[idx / BITS_PER_LONG] & BIT(idx % BITS_PER_LONG);
|
||||
}
|
||||
|
||||
static inline void
|
||||
mt76_wcid_free(unsigned long *mask, int idx)
|
||||
mt76_wcid_mask_set(unsigned long *mask, int idx)
|
||||
{
|
||||
mask[idx / BITS_PER_LONG] |= BIT(idx % BITS_PER_LONG);
|
||||
}
|
||||
|
||||
static inline void
|
||||
mt76_wcid_mask_clear(unsigned long *mask, int idx)
|
||||
{
|
||||
mask[idx / BITS_PER_LONG] &= ~BIT(idx % BITS_PER_LONG);
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user