Path: blob/main/sys/contrib/dev/mediatek/mt76/mt7615/usb_sdio.c
48526 views
// SPDX-License-Identifier: ISC1/* Copyright (C) 2020 MediaTek Inc.2*3* Author: Lorenzo Bianconi <[email protected]>4* Sean Wang <[email protected]>5*/67#include <linux/kernel.h>8#include <linux/module.h>9#include <linux/usb.h>1011#include "mt7615.h"12#include "mac.h"13#include "mcu.h"14#include "regs.h"1516const u32 mt7663_usb_sdio_reg_map[] = {17[MT_TOP_CFG_BASE] = 0x80020000,18[MT_HW_BASE] = 0x80000000,19[MT_DMA_SHDL_BASE] = 0x5000a000,20[MT_HIF_BASE] = 0x50000000,21[MT_CSR_BASE] = 0x40000000,22[MT_EFUSE_ADDR_BASE] = 0x78011000,23[MT_TOP_MISC_BASE] = 0x81020000,24[MT_PLE_BASE] = 0x82060000,25[MT_PSE_BASE] = 0x82068000,26[MT_PP_BASE] = 0x8206c000,27[MT_WTBL_BASE_ADDR] = 0x820e0000,28[MT_CFG_BASE] = 0x820f0000,29[MT_AGG_BASE] = 0x820f2000,30[MT_ARB_BASE] = 0x820f3000,31[MT_TMAC_BASE] = 0x820f4000,32[MT_RMAC_BASE] = 0x820f5000,33[MT_DMA_BASE] = 0x820f7000,34[MT_PF_BASE] = 0x820f8000,35[MT_WTBL_BASE_ON] = 0x820f9000,36[MT_WTBL_BASE_OFF] = 0x820f9800,37[MT_LPON_BASE] = 0x820fb000,38[MT_MIB_BASE] = 0x820fd000,39};40EXPORT_SYMBOL_GPL(mt7663_usb_sdio_reg_map);4142static void43mt7663_usb_sdio_write_txwi(struct mt7615_dev *dev, struct mt76_wcid *wcid,44enum mt76_txq_id qid, struct ieee80211_sta *sta,45struct ieee80211_key_conf *key, int pid,46struct sk_buff *skb)47{48__le32 *txwi = (__le32 *)(skb->data - MT_USB_TXD_SIZE);4950memset(txwi, 0, MT_USB_TXD_SIZE);51mt7615_mac_write_txwi(dev, txwi, skb, wcid, sta, pid, key, qid, false);52skb_push(skb, MT_USB_TXD_SIZE);53}5455static int mt7663_usb_sdio_set_rates(struct mt7615_dev *dev,56struct mt7615_wtbl_rate_desc *wrd)57{58struct mt7615_rate_desc *rate = &wrd->rate;59struct mt7615_sta *sta = wrd->sta;60u32 w5, w27, addr, val;61u16 idx;6263lockdep_assert_held(&dev->mt76.mutex);6465if (!sta)66return -EINVAL;6768if (!mt76_poll(dev, MT_WTBL_UPDATE, MT_WTBL_UPDATE_BUSY, 0, 5000))69return -ETIMEDOUT;7071addr = mt7615_mac_wtbl_addr(dev, sta->wcid.idx);7273w27 = mt76_rr(dev, addr + 27 * 4);74w27 &= ~MT_WTBL_W27_CC_BW_SEL;75w27 |= FIELD_PREP(MT_WTBL_W27_CC_BW_SEL, rate->bw);7677w5 = mt76_rr(dev, addr + 5 * 4);78w5 &= ~(MT_WTBL_W5_BW_CAP | MT_WTBL_W5_CHANGE_BW_RATE |79MT_WTBL_W5_MPDU_OK_COUNT |80MT_WTBL_W5_MPDU_FAIL_COUNT |81MT_WTBL_W5_RATE_IDX);82w5 |= FIELD_PREP(MT_WTBL_W5_BW_CAP, rate->bw) |83FIELD_PREP(MT_WTBL_W5_CHANGE_BW_RATE,84rate->bw_idx ? rate->bw_idx - 1 : 7);8586mt76_wr(dev, MT_WTBL_RIUCR0, w5);8788mt76_wr(dev, MT_WTBL_RIUCR1,89FIELD_PREP(MT_WTBL_RIUCR1_RATE0, rate->probe_val) |90FIELD_PREP(MT_WTBL_RIUCR1_RATE1, rate->val[0]) |91FIELD_PREP(MT_WTBL_RIUCR1_RATE2_LO, rate->val[1]));9293mt76_wr(dev, MT_WTBL_RIUCR2,94FIELD_PREP(MT_WTBL_RIUCR2_RATE2_HI, rate->val[1] >> 8) |95FIELD_PREP(MT_WTBL_RIUCR2_RATE3, rate->val[1]) |96FIELD_PREP(MT_WTBL_RIUCR2_RATE4, rate->val[2]) |97FIELD_PREP(MT_WTBL_RIUCR2_RATE5_LO, rate->val[2]));9899mt76_wr(dev, MT_WTBL_RIUCR3,100FIELD_PREP(MT_WTBL_RIUCR3_RATE5_HI, rate->val[2] >> 4) |101FIELD_PREP(MT_WTBL_RIUCR3_RATE6, rate->val[3]) |102FIELD_PREP(MT_WTBL_RIUCR3_RATE7, rate->val[3]));103104mt76_wr(dev, MT_WTBL_UPDATE,105FIELD_PREP(MT_WTBL_UPDATE_WLAN_IDX, sta->wcid.idx) |106MT_WTBL_UPDATE_RATE_UPDATE |107MT_WTBL_UPDATE_TX_COUNT_CLEAR);108109mt76_wr(dev, addr + 27 * 4, w27);110111sta->rate_probe = sta->rateset[rate->rateset].probe_rate.idx != -1;112113idx = sta->vif->mt76.omac_idx;114idx = idx > HW_BSSID_MAX ? HW_BSSID_0 : idx;115addr = idx > 1 ? MT_LPON_TCR2(idx): MT_LPON_TCR0(idx);116117mt76_rmw(dev, addr, MT_LPON_TCR_MODE, MT_LPON_TCR_READ); /* TSF read */118val = mt76_rr(dev, MT_LPON_UTTR0);119sta->rate_set_tsf = (val & ~BIT(0)) | rate->rateset;120121if (!(sta->wcid.tx_info & MT_WCID_TX_INFO_SET))122mt76_poll(dev, MT_WTBL_UPDATE, MT_WTBL_UPDATE_BUSY, 0, 5000);123124sta->rate_count = 2 * MT7615_RATE_RETRY * sta->n_rates;125sta->wcid.tx_info |= MT_WCID_TX_INFO_SET;126127return 0;128}129130static void mt7663_usb_sdio_rate_work(struct work_struct *work)131{132struct mt7615_wtbl_rate_desc *wrd, *wrd_next;133struct list_head wrd_list;134struct mt7615_dev *dev;135136dev = (struct mt7615_dev *)container_of(work, struct mt7615_dev,137rate_work);138139INIT_LIST_HEAD(&wrd_list);140spin_lock_bh(&dev->mt76.lock);141list_splice_init(&dev->wrd_head, &wrd_list);142spin_unlock_bh(&dev->mt76.lock);143144list_for_each_entry_safe(wrd, wrd_next, &wrd_list, node) {145list_del(&wrd->node);146147mt7615_mutex_acquire(dev);148mt7663_usb_sdio_set_rates(dev, wrd);149mt7615_mutex_release(dev);150151kfree(wrd);152}153}154155bool mt7663_usb_sdio_tx_status_data(struct mt76_dev *mdev, u8 *update)156{157struct mt7615_dev *dev = container_of(mdev, struct mt7615_dev, mt76);158159mt7615_mutex_acquire(dev);160mt7615_mac_sta_poll(dev);161mt7615_mutex_release(dev);162163return false;164}165EXPORT_SYMBOL_GPL(mt7663_usb_sdio_tx_status_data);166167void mt7663_usb_sdio_tx_complete_skb(struct mt76_dev *mdev,168struct mt76_queue_entry *e)169{170unsigned int headroom = MT_USB_TXD_SIZE;171172if (mt76_is_usb(mdev))173headroom += MT_USB_HDR_SIZE;174skb_pull(e->skb, headroom);175176mt76_tx_complete_skb(mdev, e->wcid, e->skb);177}178EXPORT_SYMBOL_GPL(mt7663_usb_sdio_tx_complete_skb);179180int mt7663_usb_sdio_tx_prepare_skb(struct mt76_dev *mdev, void *txwi_ptr,181enum mt76_txq_id qid, struct mt76_wcid *wcid,182struct ieee80211_sta *sta,183struct mt76_tx_info *tx_info)184{185struct mt7615_dev *dev = container_of(mdev, struct mt7615_dev, mt76);186struct sk_buff *skb = tx_info->skb;187struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);188struct ieee80211_key_conf *key = info->control.hw_key;189struct mt7615_sta *msta;190int pad, err, pktid;191192msta = wcid ? container_of(wcid, struct mt7615_sta, wcid) : NULL;193if (!wcid)194wcid = &dev->mt76.global_wcid;195196if ((info->flags & IEEE80211_TX_CTL_RATE_CTRL_PROBE) &&197msta && !msta->rate_probe) {198/* request to configure sampling rate */199spin_lock_bh(&dev->mt76.lock);200mt7615_mac_set_rates(&dev->phy, msta, &info->control.rates[0],201msta->rates);202spin_unlock_bh(&dev->mt76.lock);203}204205pktid = mt76_tx_status_skb_add(&dev->mt76, wcid, skb);206mt7663_usb_sdio_write_txwi(dev, wcid, qid, sta, key, pktid, skb);207if (mt76_is_usb(mdev)) {208u32 len = skb->len;209210put_unaligned_le32(len, skb_push(skb, sizeof(len)));211pad = round_up(skb->len, 4) + 4 - skb->len;212} else {213pad = round_up(skb->len, 4) - skb->len;214}215216err = mt76_skb_adjust_pad(skb, pad);217if (err)218/* Release pktid in case of error. */219idr_remove(&wcid->pktid, pktid);220221return err;222}223EXPORT_SYMBOL_GPL(mt7663_usb_sdio_tx_prepare_skb);224225static int mt7663u_dma_sched_init(struct mt7615_dev *dev)226{227int i;228229mt76_rmw(dev, MT_DMA_SHDL(MT_DMASHDL_PKT_MAX_SIZE),230MT_DMASHDL_PKT_MAX_SIZE_PLE | MT_DMASHDL_PKT_MAX_SIZE_PSE,231FIELD_PREP(MT_DMASHDL_PKT_MAX_SIZE_PLE, 1) |232FIELD_PREP(MT_DMASHDL_PKT_MAX_SIZE_PSE, 8));233234/* disable refill group 5 - group 15 and raise group 2235* and 3 as high priority.236*/237mt76_wr(dev, MT_DMA_SHDL(MT_DMASHDL_REFILL), 0xffe00006);238mt76_clear(dev, MT_DMA_SHDL(MT_DMASHDL_PAGE), BIT(16));239240for (i = 0; i < 5; i++)241mt76_wr(dev, MT_DMA_SHDL(MT_DMASHDL_GROUP_QUOTA(i)),242FIELD_PREP(MT_DMASHDL_GROUP_QUOTA_MIN, 0x3) |243FIELD_PREP(MT_DMASHDL_GROUP_QUOTA_MAX, 0x1ff));244245mt76_wr(dev, MT_DMA_SHDL(MT_DMASHDL_Q_MAP(0)), 0x42104210);246mt76_wr(dev, MT_DMA_SHDL(MT_DMASHDL_Q_MAP(1)), 0x42104210);247248mt76_wr(dev, MT_DMA_SHDL(MT_DMASHDL_Q_MAP(2)), 0x4444);249250/* group pririority from high to low:251* 15 (cmd groups) > 4 > 3 > 2 > 1 > 0.252*/253mt76_wr(dev, MT_DMA_SHDL(MT_DMASHDL_SCHED_SET0), 0x6501234f);254mt76_wr(dev, MT_DMA_SHDL(MT_DMASHDL_SCHED_SET1), 0xedcba987);255mt76_wr(dev, MT_DMA_SHDL(MT_DMASHDL_OPTIONAL), 0x7004801c);256257mt76_wr(dev, MT_UDMA_WLCFG_1,258FIELD_PREP(MT_WL_TX_TMOUT_LMT, 80000) |259FIELD_PREP(MT_WL_RX_AGG_PKT_LMT, 1));260261/* setup UDMA Rx Flush */262mt76_clear(dev, MT_UDMA_WLCFG_0, MT_WL_RX_FLUSH);263/* hif reset */264mt76_set(dev, MT_HIF_RST, MT_HIF_LOGIC_RST_N);265266mt76_set(dev, MT_UDMA_WLCFG_0,267MT_WL_RX_AGG_EN | MT_WL_RX_EN | MT_WL_TX_EN |268MT_WL_RX_MPSZ_PAD0 | MT_TICK_1US_EN |269MT_WL_TX_TMOUT_FUNC_EN);270mt76_rmw(dev, MT_UDMA_WLCFG_0, MT_WL_RX_AGG_LMT | MT_WL_RX_AGG_TO,271FIELD_PREP(MT_WL_RX_AGG_LMT, 32) |272FIELD_PREP(MT_WL_RX_AGG_TO, 100));273274return 0;275}276277static int mt7663_usb_sdio_init_hardware(struct mt7615_dev *dev)278{279int ret, idx;280281ret = mt7615_eeprom_init(dev, MT_EFUSE_BASE);282if (ret < 0)283return ret;284285if (mt76_is_usb(&dev->mt76)) {286ret = mt7663u_dma_sched_init(dev);287if (ret)288return ret;289}290291set_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);292293/* Beacon and mgmt frames should occupy wcid 0 */294idx = mt76_wcid_alloc(dev->mt76.wcid_mask, MT7615_WTBL_STA - 1);295if (idx)296return -ENOSPC;297298dev->mt76.global_wcid.idx = idx;299dev->mt76.global_wcid.hw_key_idx = -1;300rcu_assign_pointer(dev->mt76.wcid[idx], &dev->mt76.global_wcid);301302return 0;303}304305int mt7663_usb_sdio_register_device(struct mt7615_dev *dev)306{307struct ieee80211_hw *hw = mt76_hw(dev);308int err;309310INIT_WORK(&dev->rate_work, mt7663_usb_sdio_rate_work);311INIT_LIST_HEAD(&dev->wrd_head);312mt7615_init_device(dev);313314err = mt7663_usb_sdio_init_hardware(dev);315if (err)316return err;317318hw->extra_tx_headroom += MT_USB_TXD_SIZE;319if (mt76_is_usb(&dev->mt76)) {320hw->extra_tx_headroom += MT_USB_HDR_SIZE;321/* check hw sg support in order to enable AMSDU */322if (dev->mt76.usb.sg_en)323hw->max_tx_fragments = MT_HW_TXP_MAX_BUF_NUM;324else325hw->max_tx_fragments = 1;326}327328err = mt76_register_device(&dev->mt76, true, mt76_rates,329ARRAY_SIZE(mt76_rates));330if (err < 0)331return err;332333if (!dev->mt76.usb.sg_en) {334struct ieee80211_sta_vht_cap *vht_cap;335336/* decrease max A-MSDU size if SG is not supported */337vht_cap = &dev->mphy.sband_5g.sband.vht_cap;338vht_cap->cap &= ~IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_11454;339}340341ieee80211_queue_work(hw, &dev->mcu_work);342mt7615_init_txpower(dev, &dev->mphy.sband_2g.sband);343mt7615_init_txpower(dev, &dev->mphy.sband_5g.sband);344345#if defined(CONFIG_MT7615_DEBUGFS)346return mt7615_init_debugfs(dev);347#else348return 0;349#endif350}351EXPORT_SYMBOL_GPL(mt7663_usb_sdio_register_device);352353MODULE_AUTHOR("Lorenzo Bianconi <[email protected]>");354MODULE_AUTHOR("Sean Wang <[email protected]>");355MODULE_DESCRIPTION("MediaTek MT7663 SDIO/USB helpers");356MODULE_LICENSE("Dual BSD/GPL");357358359