Commit 0c1ce988 authored by Lorenzo Bianconi's avatar Lorenzo Bianconi Committed by Felix Fietkau

mt76: mt7921: add wifi reset support

Introduce wifi chip reset support for mt7921 device to recover mcu
hangs.
Co-developed-by: default avatarSean Wang <sean.wang@mediatek.com>
Signed-off-by: default avatarSean Wang <sean.wang@mediatek.com>
Signed-off-by: default avatarLorenzo Bianconi <lorenzo@kernel.org>
Signed-off-by: default avatarFelix Fietkau <nbd@nbd.name>
parent c001df97
...@@ -143,7 +143,7 @@ mt7921_mac_init_band(struct mt7921_dev *dev, u8 band) ...@@ -143,7 +143,7 @@ mt7921_mac_init_band(struct mt7921_dev *dev, u8 band)
mt76_clear(dev, MT_DMA_DCR0(band), MT_DMA_DCR0_RXD_G5_EN); mt76_clear(dev, MT_DMA_DCR0(band), MT_DMA_DCR0_RXD_G5_EN);
} }
static void mt7921_mac_init(struct mt7921_dev *dev) void mt7921_mac_init(struct mt7921_dev *dev)
{ {
int i; int i;
...@@ -233,7 +233,6 @@ int mt7921_register_device(struct mt7921_dev *dev) ...@@ -233,7 +233,6 @@ int mt7921_register_device(struct mt7921_dev *dev)
INIT_LIST_HEAD(&dev->sta_poll_list); INIT_LIST_HEAD(&dev->sta_poll_list);
spin_lock_init(&dev->sta_poll_lock); spin_lock_init(&dev->sta_poll_lock);
init_waitqueue_head(&dev->reset_wait);
INIT_WORK(&dev->reset_work, mt7921_mac_reset_work); INIT_WORK(&dev->reset_work, mt7921_mac_reset_work);
ret = mt7921_init_hardware(dev); ret = mt7921_init_hardware(dev);
......
...@@ -1184,43 +1184,77 @@ void mt7921_update_channel(struct mt76_dev *mdev) ...@@ -1184,43 +1184,77 @@ void mt7921_update_channel(struct mt76_dev *mdev)
mt76_connac_power_save_sched(&dev->mphy, &dev->pm); mt76_connac_power_save_sched(&dev->mphy, &dev->pm);
} }
static bool static int
mt7921_wait_reset_state(struct mt7921_dev *dev, u32 state) mt7921_wfsys_reset(struct mt7921_dev *dev)
{ {
bool ret; mt76_set(dev, 0x70002600, BIT(0));
msleep(200);
mt76_clear(dev, 0x70002600, BIT(0));
ret = wait_event_timeout(dev->reset_wait, return __mt76_poll_msec(&dev->mt76, MT_WFSYS_SW_RST_B,
(READ_ONCE(dev->reset_state) & state), WFSYS_SW_INIT_DONE, WFSYS_SW_INIT_DONE, 500);
MT7921_RESET_TIMEOUT);
WARN(!ret, "Timeout waiting for MCU reset state %x\n", state);
return ret;
} }
static void static void
mt7921_dma_reset(struct mt7921_phy *phy) mt7921_dma_reset(struct mt7921_dev *dev)
{ {
struct mt7921_dev *dev = phy->dev;
int i; int i;
/* reset */
mt76_clear(dev, MT_WFDMA0_RST,
MT_WFDMA0_RST_DMASHDL_ALL_RST | MT_WFDMA0_RST_LOGIC_RST);
mt76_set(dev, MT_WFDMA0_RST,
MT_WFDMA0_RST_DMASHDL_ALL_RST | MT_WFDMA0_RST_LOGIC_RST);
/* disable WFDMA0 */
mt76_clear(dev, MT_WFDMA0_GLO_CFG, mt76_clear(dev, MT_WFDMA0_GLO_CFG,
MT_WFDMA0_GLO_CFG_TX_DMA_EN | MT_WFDMA0_GLO_CFG_RX_DMA_EN); MT_WFDMA0_GLO_CFG_TX_DMA_EN | MT_WFDMA0_GLO_CFG_RX_DMA_EN |
MT_WFDMA0_GLO_CFG_CSR_DISP_BASE_PTR_CHAIN_EN |
MT_WFDMA0_GLO_CFG_OMIT_TX_INFO |
MT_WFDMA0_GLO_CFG_OMIT_RX_INFO |
MT_WFDMA0_GLO_CFG_OMIT_RX_INFO_PFET2);
usleep_range(1000, 2000); mt76_poll(dev, MT_WFDMA0_GLO_CFG,
MT_WFDMA0_GLO_CFG_TX_DMA_BUSY |
MT_WFDMA0_GLO_CFG_RX_DMA_BUSY, 0, 1000);
mt76_queue_tx_cleanup(dev, dev->mt76.q_mcu[MT_MCUQ_WA], true); /* reset hw queues */
for (i = 0; i < __MT_TXQ_MAX; i++) for (i = 0; i < __MT_TXQ_MAX; i++)
mt76_queue_tx_cleanup(dev, phy->mt76->q_tx[i], true); mt76_queue_reset(dev, dev->mphy.q_tx[i]);
mt76_for_each_q_rx(&dev->mt76, i) { for (i = 0; i < __MT_MCUQ_MAX; i++)
mt76_queue_rx_reset(dev, i); mt76_queue_reset(dev, dev->mt76.q_mcu[i]);
}
mt76_for_each_q_rx(&dev->mt76, i)
mt76_queue_reset(dev, &dev->mt76.q_rx[i]);
/* re-init prefetch settings after reset */ /* configure perfetch settings */
mt7921_dma_prefetch(dev); mt7921_dma_prefetch(dev);
/* reset dma idx */
mt76_wr(dev, MT_WFDMA0_RST_DTX_PTR, ~0);
/* configure delay interrupt */
mt76_wr(dev, MT_WFDMA0_PRI_DLY_INT_CFG0, 0);
mt76_set(dev, MT_WFDMA0_GLO_CFG,
MT_WFDMA0_GLO_CFG_TX_WB_DDONE |
MT_WFDMA0_GLO_CFG_FIFO_LITTLE_ENDIAN |
MT_WFDMA0_GLO_CFG_CLK_GAT_DIS |
MT_WFDMA0_GLO_CFG_OMIT_TX_INFO |
MT_WFDMA0_GLO_CFG_CSR_DISP_BASE_PTR_CHAIN_EN |
MT_WFDMA0_GLO_CFG_OMIT_RX_INFO_PFET2);
mt76_set(dev, MT_WFDMA0_GLO_CFG, mt76_set(dev, MT_WFDMA0_GLO_CFG,
MT_WFDMA0_GLO_CFG_TX_DMA_EN | MT_WFDMA0_GLO_CFG_RX_DMA_EN); MT_WFDMA0_GLO_CFG_TX_DMA_EN | MT_WFDMA0_GLO_CFG_RX_DMA_EN);
mt76_set(dev, 0x54000120, BIT(1));
/* enable interrupts for TX/RX rings */
mt7921_irq_enable(dev,
MT_INT_RX_DONE_ALL | MT_INT_TX_DONE_ALL |
MT_INT_MCU_CMD);
} }
void mt7921_tx_token_put(struct mt7921_dev *dev) void mt7921_tx_token_put(struct mt7921_dev *dev)
...@@ -1244,71 +1278,125 @@ void mt7921_tx_token_put(struct mt7921_dev *dev) ...@@ -1244,71 +1278,125 @@ void mt7921_tx_token_put(struct mt7921_dev *dev)
idr_destroy(&dev->token); idr_destroy(&dev->token);
} }
/* system error recovery */ static void
void mt7921_mac_reset_work(struct work_struct *work) mt7921_vif_connect_iter(void *priv, u8 *mac,
struct ieee80211_vif *vif)
{ {
struct mt7921_dev *dev; struct mt7921_vif *mvif = (struct mt7921_vif *)vif->drv_priv;
struct mt7921_dev *dev = mvif->phy->dev;
dev = container_of(work, struct mt7921_dev, reset_work); ieee80211_disconnect(vif, true);
if (!(READ_ONCE(dev->reset_state) & MT_MCU_CMD_STOP_DMA)) mt76_connac_mcu_uni_add_dev(&dev->mphy, vif, &mvif->sta.wcid, true);
return; mt7921_mcu_set_tx(dev, vif);
}
static int
mt7921_mac_reset(struct mt7921_dev *dev)
{
int i, err;
mt76_connac_free_pending_tx_skbs(&dev->pm, NULL);
ieee80211_stop_queues(mt76_hw(dev)); mt76_wr(dev, MT_WFDMA0_HOST_INT_ENA, 0);
mt76_wr(dev, MT_PCIE_MAC_INT_ENABLE, 0x0);
set_bit(MT76_RESET, &dev->mphy.state);
set_bit(MT76_MCU_RESET, &dev->mphy.state); set_bit(MT76_MCU_RESET, &dev->mphy.state);
wake_up(&dev->mt76.mcu.wait); wake_up(&dev->mt76.mcu.wait);
cancel_delayed_work_sync(&dev->mphy.mac_work); skb_queue_purge(&dev->mt76.mcu.res_q);
/* lock/unlock all queues to ensure that no tx is pending */
mt76_txq_schedule_all(&dev->mphy); mt76_txq_schedule_all(&dev->mphy);
mt76_worker_disable(&dev->mt76.tx_worker); mt76_worker_disable(&dev->mt76.tx_worker);
napi_disable(&dev->mt76.napi[0]); napi_disable(&dev->mt76.napi[MT_RXQ_MAIN]);
napi_disable(&dev->mt76.napi[1]); napi_disable(&dev->mt76.napi[MT_RXQ_MCU]);
napi_disable(&dev->mt76.napi[2]); napi_disable(&dev->mt76.napi[MT_RXQ_MCU_WA]);
napi_disable(&dev->mt76.tx_napi); napi_disable(&dev->mt76.tx_napi);
mt7921_mutex_acquire(dev);
mt76_wr(dev, MT_MCU_INT_EVENT, MT_MCU_INT_EVENT_DMA_STOPPED);
mt7921_tx_token_put(dev); mt7921_tx_token_put(dev);
idr_init(&dev->token); idr_init(&dev->token);
if (mt7921_wait_reset_state(dev, MT_MCU_CMD_RESET_DONE)) { /* clean up hw queues */
mt7921_dma_reset(&dev->phy); for (i = 0; i < ARRAY_SIZE(dev->mt76.phy.q_tx); i++)
mt76_queue_tx_cleanup(dev, dev->mphy.q_tx[i], true);
mt76_wr(dev, MT_MCU_INT_EVENT, MT_MCU_INT_EVENT_DMA_INIT); for (i = 0; i < ARRAY_SIZE(dev->mt76.q_mcu); i++)
mt7921_wait_reset_state(dev, MT_MCU_CMD_RECOVERY_DONE); mt76_queue_tx_cleanup(dev, dev->mt76.q_mcu[i], true);
}
clear_bit(MT76_MCU_RESET, &dev->mphy.state); mt76_for_each_q_rx(&dev->mt76, i)
clear_bit(MT76_RESET, &dev->mphy.state); mt76_queue_rx_cleanup(dev, &dev->mt76.q_rx[i]);
mt7921_wfsys_reset(dev);
mt7921_dma_reset(dev);
mt76_for_each_q_rx(&dev->mt76, i) {
mt76_queue_rx_reset(dev, i);
napi_enable(&dev->mt76.napi[i]);
napi_schedule(&dev->mt76.napi[i]);
}
mt76_worker_enable(&dev->mt76.tx_worker);
napi_enable(&dev->mt76.tx_napi); napi_enable(&dev->mt76.tx_napi);
napi_schedule(&dev->mt76.tx_napi); napi_schedule(&dev->mt76.tx_napi);
mt76_worker_enable(&dev->mt76.tx_worker);
napi_enable(&dev->mt76.napi[0]); clear_bit(MT76_MCU_RESET, &dev->mphy.state);
napi_schedule(&dev->mt76.napi[0]);
napi_enable(&dev->mt76.napi[1]); mt76_wr(dev, MT_WFDMA0_HOST_INT_ENA, 0);
napi_schedule(&dev->mt76.napi[1]); mt76_wr(dev, MT_PCIE_MAC_INT_ENABLE, 0xff);
mt7921_irq_enable(dev,
MT_INT_RX_DONE_ALL | MT_INT_TX_DONE_ALL |
MT_INT_MCU_CMD);
napi_enable(&dev->mt76.napi[2]); err = mt7921_run_firmware(dev);
napi_schedule(&dev->mt76.napi[2]); if (err)
return err;
ieee80211_wake_queues(mt76_hw(dev)); err = mt7921_mcu_set_eeprom(dev);
if (err)
return err;
mt76_wr(dev, MT_MCU_INT_EVENT, MT_MCU_INT_EVENT_RESET_DONE); mt7921_mac_init(dev);
mt7921_wait_reset_state(dev, MT_MCU_CMD_NORMAL_STATE); return __mt7921_start(&dev->phy);
}
mt7921_mutex_release(dev); /* system error recovery */
void mt7921_mac_reset_work(struct work_struct *work)
{
struct ieee80211_hw *hw;
struct mt7921_dev *dev;
int i;
ieee80211_queue_delayed_work(mt76_hw(dev), &dev->mphy.mac_work, dev = container_of(work, struct mt7921_dev, reset_work);
MT7921_WATCHDOG_TIME); hw = mt76_hw(dev);
dev_err(dev->mt76.dev, "chip reset\n");
ieee80211_stop_queues(hw);
cancel_delayed_work_sync(&dev->mphy.mac_work);
cancel_delayed_work_sync(&dev->pm.ps_work);
cancel_work_sync(&dev->pm.wake_work);
mutex_lock(&dev->mt76.mutex);
for (i = 0; i < 10; i++) {
if (!mt7921_mac_reset(dev))
break;
}
mutex_unlock(&dev->mt76.mutex);
if (i == 10)
dev_err(dev->mt76.dev, "chip reset failed\n");
ieee80211_wake_queues(hw);
ieee80211_iterate_active_interfaces(hw,
IEEE80211_IFACE_ITER_RESUME_ALL,
mt7921_vif_connect_iter, 0);
}
void mt7921_reset(struct mt76_dev *mdev)
{
struct mt7921_dev *dev = container_of(mdev, struct mt7921_dev, mt76);
queue_work(dev->mt76.wq, &dev->reset_work);
} }
static void static void
...@@ -1505,4 +1593,5 @@ void mt7921_coredump_work(struct work_struct *work) ...@@ -1505,4 +1593,5 @@ void mt7921_coredump_work(struct work_struct *work)
} }
dev_coredumpv(dev->mt76.dev, dump, MT76_CONNAC_COREDUMP_SZ, dev_coredumpv(dev->mt76.dev, dump, MT76_CONNAC_COREDUMP_SZ,
GFP_KERNEL); GFP_KERNEL);
mt7921_reset(&dev->mt76);
} }
...@@ -952,6 +952,7 @@ int mt7921_mcu_init(struct mt7921_dev *dev) ...@@ -952,6 +952,7 @@ int mt7921_mcu_init(struct mt7921_dev *dev)
.mcu_skb_send_msg = mt7921_mcu_send_message, .mcu_skb_send_msg = mt7921_mcu_send_message,
.mcu_parse_response = mt7921_mcu_parse_response, .mcu_parse_response = mt7921_mcu_parse_response,
.mcu_restart = mt7921_mcu_restart, .mcu_restart = mt7921_mcu_restart,
.mcu_reset = mt7921_reset,
}; };
dev->mt76.mcu_ops = &mt7921_mcu_ops; dev->mt76.mcu_ops = &mt7921_mcu_ops;
...@@ -1269,6 +1270,7 @@ int mt7921_mcu_drv_pmctrl(struct mt7921_dev *dev) ...@@ -1269,6 +1270,7 @@ int mt7921_mcu_drv_pmctrl(struct mt7921_dev *dev)
if (i == MT7921_DRV_OWN_RETRY_COUNT) { if (i == MT7921_DRV_OWN_RETRY_COUNT) {
dev_err(dev->mt76.dev, "driver own failed\n"); dev_err(dev->mt76.dev, "driver own failed\n");
mt7921_reset(&dev->mt76);
return -EIO; return -EIO;
} }
...@@ -1295,6 +1297,7 @@ int mt7921_mcu_fw_pmctrl(struct mt7921_dev *dev) ...@@ -1295,6 +1297,7 @@ int mt7921_mcu_fw_pmctrl(struct mt7921_dev *dev)
if (i == MT7921_DRV_OWN_RETRY_COUNT) { if (i == MT7921_DRV_OWN_RETRY_COUNT) {
dev_err(dev->mt76.dev, "firmware own failed\n"); dev_err(dev->mt76.dev, "firmware own failed\n");
mt7921_reset(&dev->mt76);
return -EIO; return -EIO;
} }
......
...@@ -151,8 +151,6 @@ struct mt7921_dev { ...@@ -151,8 +151,6 @@ struct mt7921_dev {
struct work_struct init_work; struct work_struct init_work;
struct work_struct reset_work; struct work_struct reset_work;
wait_queue_head_t reset_wait;
u32 reset_state;
struct list_head sta_poll_list; struct list_head sta_poll_list;
spinlock_t sta_poll_lock; spinlock_t sta_poll_lock;
...@@ -283,6 +281,7 @@ mt7921_l1_rmw(struct mt7921_dev *dev, u32 addr, u32 mask, u32 val) ...@@ -283,6 +281,7 @@ mt7921_l1_rmw(struct mt7921_dev *dev, u32 addr, u32 mask, u32 val)
#define mt7921_l1_set(dev, addr, val) mt7921_l1_rmw(dev, addr, 0, val) #define mt7921_l1_set(dev, addr, val) mt7921_l1_rmw(dev, addr, 0, val)
#define mt7921_l1_clear(dev, addr, val) mt7921_l1_rmw(dev, addr, val, 0) #define mt7921_l1_clear(dev, addr, val) mt7921_l1_rmw(dev, addr, val, 0)
void mt7921_mac_init(struct mt7921_dev *dev);
bool mt7921_mac_wtbl_update(struct mt7921_dev *dev, int idx, u32 mask); bool mt7921_mac_wtbl_update(struct mt7921_dev *dev, int idx, u32 mask);
void mt7921_mac_reset_counters(struct mt7921_phy *phy); void mt7921_mac_reset_counters(struct mt7921_phy *phy);
void mt7921_mac_write_txwi(struct mt7921_dev *dev, __le32 *txwi, void mt7921_mac_write_txwi(struct mt7921_dev *dev, __le32 *txwi,
...@@ -298,6 +297,7 @@ void mt7921_mac_sta_remove(struct mt76_dev *mdev, struct ieee80211_vif *vif, ...@@ -298,6 +297,7 @@ void mt7921_mac_sta_remove(struct mt76_dev *mdev, struct ieee80211_vif *vif,
struct ieee80211_sta *sta); struct ieee80211_sta *sta);
void mt7921_mac_work(struct work_struct *work); void mt7921_mac_work(struct work_struct *work);
void mt7921_mac_reset_work(struct work_struct *work); void mt7921_mac_reset_work(struct work_struct *work);
void mt7921_reset(struct mt76_dev *mdev);
int mt7921_tx_prepare_skb(struct mt76_dev *mdev, void *txwi_ptr, int mt7921_tx_prepare_skb(struct mt76_dev *mdev, void *txwi_ptr,
enum mt76_txq_id qid, struct mt76_wcid *wcid, enum mt76_txq_id qid, struct mt76_wcid *wcid,
struct ieee80211_sta *sta, struct ieee80211_sta *sta,
......
...@@ -418,6 +418,10 @@ ...@@ -418,6 +418,10 @@
#define PCIE_LPCR_HOST_CLR_OWN BIT(1) #define PCIE_LPCR_HOST_CLR_OWN BIT(1)
#define PCIE_LPCR_HOST_SET_OWN BIT(0) #define PCIE_LPCR_HOST_SET_OWN BIT(0)
#define MT_WFSYS_SW_RST_B 0x18000140
#define WFSYS_SW_RST_B BIT(0)
#define WFSYS_SW_INIT_DONE BIT(4)
#define MT_CONN_ON_MISC 0x7c0600f0 #define MT_CONN_ON_MISC 0x7c0600f0
#define MT_TOP_MISC2_FW_N9_RDY GENMASK(1, 0) #define MT_TOP_MISC2_FW_N9_RDY GENMASK(1, 0)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment