]> git.proxmox.com Git - mirror_ubuntu-jammy-kernel.git/commitdiff
mt76: mt7615: fix key set/delete issues
authorFelix Fietkau <nbd@nbd.name>
Mon, 8 Feb 2021 16:07:27 +0000 (17:07 +0100)
committerFelix Fietkau <nbd@nbd.name>
Sun, 11 Apr 2021 16:50:38 +0000 (18:50 +0200)
There were multiple issues in the current key set/remove code:
- deleting a key with the previous key index deletes the current key
- BIP key would only be uploaded correctly initially and corrupted on rekey

Rework the code to better keep track of multiple keys and check for the
key index before deleting the current key

Signed-off-by: Felix Fietkau <nbd@nbd.name>
drivers/net/wireless/mediatek/mt76/mt76.h
drivers/net/wireless/mediatek/mt76/mt7615/mac.c
drivers/net/wireless/mediatek/mt76/mt7615/main.c

index 695b58d7eadc64a7a266190a5d606950d723f4f7..5561f81c64f7b35ea499ec9e21228b7ab1cc6bed 100644 (file)
@@ -223,6 +223,7 @@ struct mt76_wcid {
 
        u16 idx;
        u8 hw_key_idx;
+       u8 hw_key_idx2;
 
        u8 sta:1;
        u8 ext_phy:1;
index 6a835568f4a3460df6b1b6493ab1fc5bf3d8d201..c97ec70772f396efa9196395fba01dfec89113c0 100644 (file)
@@ -1033,7 +1033,7 @@ EXPORT_SYMBOL_GPL(mt7615_mac_set_rates);
 static int
 mt7615_mac_wtbl_update_key(struct mt7615_dev *dev, struct mt76_wcid *wcid,
                           struct ieee80211_key_conf *key,
-                          enum mt7615_cipher_type cipher,
+                          enum mt7615_cipher_type cipher, u16 cipher_mask,
                           enum set_key_cmd cmd)
 {
        u32 addr = mt7615_mac_wtbl_addr(dev, wcid->idx) + 30 * 4;
@@ -1050,22 +1050,22 @@ mt7615_mac_wtbl_update_key(struct mt7615_dev *dev, struct mt76_wcid *wcid,
                        memcpy(data + 16, key->key + 24, 8);
                        memcpy(data + 24, key->key + 16, 8);
                } else {
-                       if (cipher != MT_CIPHER_BIP_CMAC_128 && wcid->cipher)
-                               memmove(data + 16, data, 16);
-                       if (cipher != MT_CIPHER_BIP_CMAC_128 || !wcid->cipher)
+                       if (cipher_mask == BIT(cipher))
                                memcpy(data, key->key, key->keylen);
-                       else if (cipher == MT_CIPHER_BIP_CMAC_128)
+                       else if (cipher != MT_CIPHER_BIP_CMAC_128)
+                               memcpy(data, key->key, 16);
+                       if (cipher == MT_CIPHER_BIP_CMAC_128)
                                memcpy(data + 16, key->key, 16);
                }
        } else {
-               if (wcid->cipher & ~BIT(cipher)) {
-                       if (cipher != MT_CIPHER_BIP_CMAC_128)
-                               memmove(data, data + 16, 16);
+               if (cipher == MT_CIPHER_BIP_CMAC_128)
                        memset(data + 16, 0, 16);
-               } else {
+               else if (cipher_mask)
+                       memset(data, 0, 16);
+               if (!cipher_mask)
                        memset(data, 0, sizeof(data));
-               }
        }
+
        mt76_wr_copy(dev, addr, data, sizeof(data));
 
        return 0;
@@ -1073,7 +1073,7 @@ mt7615_mac_wtbl_update_key(struct mt7615_dev *dev, struct mt76_wcid *wcid,
 
 static int
 mt7615_mac_wtbl_update_pk(struct mt7615_dev *dev, struct mt76_wcid *wcid,
-                         enum mt7615_cipher_type cipher,
+                         enum mt7615_cipher_type cipher, u16 cipher_mask,
                          int keyidx, enum set_key_cmd cmd)
 {
        u32 addr = mt7615_mac_wtbl_addr(dev, wcid->idx), w0, w1;
@@ -1083,20 +1083,23 @@ mt7615_mac_wtbl_update_pk(struct mt7615_dev *dev, struct mt76_wcid *wcid,
 
        w0 = mt76_rr(dev, addr);
        w1 = mt76_rr(dev, addr + 4);
-       if (cmd == SET_KEY) {
-               w0 |= MT_WTBL_W0_RX_KEY_VALID |
-                     FIELD_PREP(MT_WTBL_W0_RX_IK_VALID,
-                                cipher == MT_CIPHER_BIP_CMAC_128);
-               if (cipher != MT_CIPHER_BIP_CMAC_128 ||
-                   !wcid->cipher)
-                       w0 |= FIELD_PREP(MT_WTBL_W0_KEY_IDX, keyidx);
-       }  else {
-               if (!(wcid->cipher & ~BIT(cipher)))
-                       w0 &= ~(MT_WTBL_W0_RX_KEY_VALID |
-                               MT_WTBL_W0_KEY_IDX);
-               if (cipher == MT_CIPHER_BIP_CMAC_128)
-                       w0 &= ~MT_WTBL_W0_RX_IK_VALID;
+
+       if (cipher_mask)
+               w0 |= MT_WTBL_W0_RX_KEY_VALID;
+       else
+               w0 &= ~(MT_WTBL_W0_RX_KEY_VALID | MT_WTBL_W0_KEY_IDX);
+       if (cipher_mask & BIT(MT_CIPHER_BIP_CMAC_128))
+               w0 |= MT_WTBL_W0_RX_IK_VALID;
+       else
+               w0 &= ~MT_WTBL_W0_RX_IK_VALID;
+
+       if (cmd == SET_KEY &&
+           (cipher != MT_CIPHER_BIP_CMAC_128 ||
+            cipher_mask == BIT(cipher))) {
+               w0 &= ~MT_WTBL_W0_KEY_IDX;
+               w0 |= FIELD_PREP(MT_WTBL_W0_KEY_IDX, keyidx);
        }
+
        mt76_wr(dev, MT_WTBL_RICR0, w0);
        mt76_wr(dev, MT_WTBL_RICR1, w1);
 
@@ -1109,24 +1112,25 @@ mt7615_mac_wtbl_update_pk(struct mt7615_dev *dev, struct mt76_wcid *wcid,
 
 static void
 mt7615_mac_wtbl_update_cipher(struct mt7615_dev *dev, struct mt76_wcid *wcid,
-                             enum mt7615_cipher_type cipher,
+                             enum mt7615_cipher_type cipher, u16 cipher_mask,
                              enum set_key_cmd cmd)
 {
        u32 addr = mt7615_mac_wtbl_addr(dev, wcid->idx);
 
-       if (cmd == SET_KEY) {
-               if (cipher != MT_CIPHER_BIP_CMAC_128 || !wcid->cipher)
-                       mt76_rmw(dev, addr + 2 * 4, MT_WTBL_W2_KEY_TYPE,
-                                FIELD_PREP(MT_WTBL_W2_KEY_TYPE, cipher));
-       } else {
-               if (cipher != MT_CIPHER_BIP_CMAC_128 &&
-                   wcid->cipher & BIT(MT_CIPHER_BIP_CMAC_128))
-                       mt76_rmw(dev, addr + 2 * 4, MT_WTBL_W2_KEY_TYPE,
-                                FIELD_PREP(MT_WTBL_W2_KEY_TYPE,
-                                           MT_CIPHER_BIP_CMAC_128));
-               else if (!(wcid->cipher & ~BIT(cipher)))
-                       mt76_clear(dev, addr + 2 * 4, MT_WTBL_W2_KEY_TYPE);
+       if (!cipher_mask) {
+               mt76_clear(dev, addr + 2 * 4, MT_WTBL_W2_KEY_TYPE);
+               return;
        }
+
+       if (cmd != SET_KEY)
+               return;
+
+       if (cipher == MT_CIPHER_BIP_CMAC_128 &&
+           cipher_mask & ~BIT(MT_CIPHER_BIP_CMAC_128))
+               return;
+
+       mt76_rmw(dev, addr + 2 * 4, MT_WTBL_W2_KEY_TYPE,
+                FIELD_PREP(MT_WTBL_W2_KEY_TYPE, cipher));
 }
 
 int __mt7615_mac_wtbl_set_key(struct mt7615_dev *dev,
@@ -1135,25 +1139,30 @@ int __mt7615_mac_wtbl_set_key(struct mt7615_dev *dev,
                              enum set_key_cmd cmd)
 {
        enum mt7615_cipher_type cipher;
+       u16 cipher_mask = wcid->cipher;
        int err;
 
        cipher = mt7615_mac_get_cipher(key->cipher);
        if (cipher == MT_CIPHER_NONE)
                return -EOPNOTSUPP;
 
-       mt7615_mac_wtbl_update_cipher(dev, wcid, cipher, cmd);
-       err = mt7615_mac_wtbl_update_key(dev, wcid, key, cipher, cmd);
+       if (cmd == SET_KEY)
+               cipher_mask |= BIT(cipher);
+       else
+               cipher_mask &= ~BIT(cipher);
+
+       mt7615_mac_wtbl_update_cipher(dev, wcid, cipher, cipher_mask, cmd);
+       err = mt7615_mac_wtbl_update_key(dev, wcid, key, cipher, cipher_mask,
+                                        cmd);
        if (err < 0)
                return err;
 
-       err = mt7615_mac_wtbl_update_pk(dev, wcid, cipher, key->keyidx, cmd);
+       err = mt7615_mac_wtbl_update_pk(dev, wcid, cipher, cipher_mask,
+                                       key->keyidx, cmd);
        if (err < 0)
                return err;
 
-       if (cmd == SET_KEY)
-               wcid->cipher |= BIT(cipher);
-       else
-               wcid->cipher &= ~BIT(cipher);
+       wcid->cipher = cipher_mask;
 
        return 0;
 }
index 25faf486d2795605edf6e53f18298959ca69f5ad..1aa6928f88fc25596a5a30db5d0fd28f6eca7d83 100644 (file)
@@ -337,7 +337,8 @@ static int mt7615_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
        struct mt7615_sta *msta = sta ? (struct mt7615_sta *)sta->drv_priv :
                                  &mvif->sta;
        struct mt76_wcid *wcid = &msta->wcid;
-       int idx = key->keyidx, err;
+       int idx = key->keyidx, err = 0;
+       u8 *wcid_keyidx = &wcid->hw_key_idx;
 
        /* The hardware does not support per-STA RX GTK, fallback
         * to software mode for these.
@@ -352,6 +353,7 @@ static int mt7615_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
        /* fall back to sw encryption for unsupported ciphers */
        switch (key->cipher) {
        case WLAN_CIPHER_SUITE_AES_CMAC:
+               wcid_keyidx = &wcid->hw_key_idx2;
                key->flags |= IEEE80211_KEY_FLAG_GENERATE_MMIE;
                break;
        case WLAN_CIPHER_SUITE_TKIP:
@@ -369,12 +371,13 @@ static int mt7615_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
 
        mt7615_mutex_acquire(dev);
 
-       if (cmd == SET_KEY) {
-               key->hw_key_idx = wcid->idx;
-               wcid->hw_key_idx = idx;
-       } else if (idx == wcid->hw_key_idx) {
-               wcid->hw_key_idx = -1;
-       }
+       if (cmd == SET_KEY)
+               *wcid_keyidx = idx;
+       else if (idx == *wcid_keyidx)
+               *wcid_keyidx = -1;
+       else
+               goto out;
+
        mt76_wcid_key_setup(&dev->mt76, wcid,
                            cmd == SET_KEY ? key : NULL);
 
@@ -383,6 +386,7 @@ static int mt7615_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd,
        else
                err = __mt7615_mac_wtbl_set_key(dev, wcid, key, cmd);
 
+out:
        mt7615_mutex_release(dev);
 
        return err;