On Thu, Aug 15, 2019 at 12:20:54PM +0200, Felix Fietkau wrote: > On 2019-08-15 12:09, Stanislaw Gruszka wrote: > >> Hi Stanislaw, > >> > >> Can you please try if disabling/enabling the tx tasklet during hw key > >> configuration fixes the issue? > >> Doing something like: > >> > >> tasklet_disable(tx_tasklet) > >> mt76x02_set_key() > >> tasklet_enable(tx_tasklet) > > > > It does not help with the problem. > > > >> Moreover, have you double checked if there is any performance impact > >> of not using hw encryption? > > > > I didn't observe any, but realized on this machine I have > > aesni_intel encryption accelerator. After rebuild kernel without > > CONFIG_CRYPTO_AES_NI_INTEL, 'perf top' showed extra 20% of cpu usage > > in aes_encrypt() when sending data with HW encryption disabled. > > > >> If so, I guess it is better to just redefine mt76_wake_tx_queue for > >> mt76x0e and run mt76_txq_schedule for 7630e: > >> > >> void mt76x0e_wake_tx_queue(struct ieee80211_hw *hw, struct ieee80211_txq *txq) > >> { > >> if (is_mt7630(dev)) { > >> mt76_txq_schedule(dev, txq->ac); > >> } else { > >> tasklet_schedule(&dev->tx_tasklet); > >> } > >> } > > > > Not sure about reduction of lock contention for which the tx_tasklet > > was introduced here, but looks ok for me as fix. > I think if we work around the bug like this, it can easily come back to > bite us again later. I'm not into workarounds any kind, but this is really strange issue, maybe FW bug that triggers just by slightly different driver behaviour. > I don't see any logical explanation as to how this > makes a difference with hardware encryption. > Also, I think it would be helpful to figure out what key operation (if > any) triggers this, adding or removing keys. Seems not to be related with set_key operation at all. We set 2 HW keys at the beginning and hang happen after some tx/rx traffic without any re-keyring. I'm not sure why disabling HW encryption helps. Maybe it is due to ordering or timing. With SW encryption we spend more time in mac80211 before pass skb's to the driver. Or maybe we just mix some HW keys and SW (group) keys in way that FW does not like. > Maybe it could also help if we change the order in which the WCID table > entries are updated, i.e. changing MT_WCID_ATTR first when removing keys. > > Maybe temporarily clearing MT_MAC_SYS_CTRL_ENABLE_TX before the key > update and setting it again afterwards could also help. I tested below patch and it did not help. Stanislaw diff --git a/drivers/net/wireless/mediatek/mt76/mt76x02_mac.c b/drivers/net/wireless/mediatek/mt76/mt76x02_mac.c index 82bafb5ac326..846652f2b3f1 100644 --- a/drivers/net/wireless/mediatek/mt76/mt76x02_mac.c +++ b/drivers/net/wireless/mediatek/mt76/mt76x02_mac.c @@ -105,11 +105,14 @@ int mt76x02_mac_wcid_set_key(struct mt76x02_dev *dev, u8 idx, if (cipher == MT_CIPHER_NONE && key) return -EOPNOTSUPP; + if (!key) + mt76_rmw_field(dev, MT_WCID_ATTR(idx), MT_WCID_ATTR_PKEY_MODE, MT_CIPHER_NONE); + mt76_wr_copy(dev, MT_WCID_KEY(idx), key_data, sizeof(key_data)); - mt76_rmw_field(dev, MT_WCID_ATTR(idx), MT_WCID_ATTR_PKEY_MODE, cipher); memset(iv_data, 0, sizeof(iv_data)); if (key) { + mt76_rmw_field(dev, MT_WCID_ATTR(idx), MT_WCID_ATTR_PKEY_MODE, cipher); mt76_rmw_field(dev, MT_WCID_ATTR(idx), MT_WCID_ATTR_PAIRWISE, !!(key->flags & IEEE80211_KEY_FLAG_PAIRWISE)); diff --git a/drivers/net/wireless/mediatek/mt76/mt76x02_util.c b/drivers/net/wireless/mediatek/mt76/mt76x02_util.c index fa45ed280ab1..6f624e3329f9 100644 --- a/drivers/net/wireless/mediatek/mt76/mt76x02_util.c +++ b/drivers/net/wireless/mediatek/mt76/mt76x02_util.c @@ -391,7 +391,7 @@ int mt76x02_ampdu_action(struct ieee80211_hw *hw, struct ieee80211_vif *vif, } EXPORT_SYMBOL_GPL(mt76x02_ampdu_action); -int mt76x02_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd, +int __mt76x02_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd, struct ieee80211_vif *vif, struct ieee80211_sta *sta, struct ieee80211_key_conf *key) { @@ -466,6 +466,35 @@ int mt76x02_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd, return mt76x02_mac_wcid_set_key(dev, msta->wcid.idx, key); } + +int mt76x02_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 mt76x02_dev *dev = hw->priv; + int ret; + int i = 200; + + tasklet_disable(&dev->mt76.tx_tasklet); + + /* Page count on TxQ */ + while (i-- && ((mt76_rr(dev, 0x0438) & 0xffffffff) || + (mt76_rr(dev, 0x0a30) & 0x000000ff) || + (mt76_rr(dev, 0x0a34) & 0x00ff00ff))) + msleep(10); + + if (!mt76_poll(dev, MT_MAC_STATUS, MT_MAC_STATUS_TX, 0, 1000)) + dev_warn(dev->mt76.dev, "Warning: SET KEY: MAC TX did not stop!\n"); + + mt76_clear(dev, MT_MAC_SYS_CTRL, MT_MAC_SYS_CTRL_ENABLE_TX); + + ret = __mt76x02_set_key(hw, cmd, vif, sta, key); + + mt76_set(dev, MT_MAC_SYS_CTRL, MT_MAC_SYS_CTRL_ENABLE_TX); + tasklet_enable(&dev->mt76.tx_tasklet); + + return ret; +} EXPORT_SYMBOL_GPL(mt76x02_set_key); int mt76x02_conf_tx(struct ieee80211_hw *hw, struct ieee80211_vif *vif,