]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
wifi: rt2x00: add nvmem eeprom support
authorRosen Penev <rosenp@gmail.com>
Mon, 27 Oct 2025 18:06:38 +0000 (11:06 -0700)
committerJohannes Berg <johannes.berg@intel.com>
Thu, 30 Oct 2025 07:39:18 +0000 (08:39 +0100)
Some embedded platforms have eeproms located in flash. Add nvmem support
to handle this. Support is added for PCI and SOC backends.

Signed-off-by: Rosen Penev <rosenp@gmail.com>
Link: https://patch.msgid.link/20251027180639.3797-1-rosenp@gmail.com
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
drivers/net/wireless/ralink/rt2x00/rt2800lib.c
drivers/net/wireless/ralink/rt2x00/rt2800lib.h
drivers/net/wireless/ralink/rt2x00/rt2800pci.c
drivers/net/wireless/ralink/rt2x00/rt2800soc.c

index f07152fa37255e38d531c26f25bd689082b6e73d..65d0f805459cefd92de93c2996a67ac05a72c909 100644 (file)
@@ -24,6 +24,7 @@
 #include <linux/crc-ccitt.h>
 #include <linux/kernel.h>
 #include <linux/module.h>
+#include <linux/nvmem-consumer.h>
 #include <linux/slab.h>
 
 #include "rt2x00.h"
@@ -10962,6 +10963,36 @@ int rt2800_read_eeprom_efuse(struct rt2x00_dev *rt2x00dev)
 }
 EXPORT_SYMBOL_GPL(rt2800_read_eeprom_efuse);
 
+int rt2800_read_eeprom_nvmem(struct rt2x00_dev *rt2x00dev)
+{
+       struct device_node *np = rt2x00dev->dev->of_node;
+       unsigned int len = rt2x00dev->ops->eeprom_size;
+       struct nvmem_cell *cell;
+       const void *data;
+       size_t retlen;
+
+       cell = of_nvmem_cell_get(np, "eeprom");
+       if (IS_ERR(cell))
+               return PTR_ERR(cell);
+
+       data = nvmem_cell_read(cell, &retlen);
+       nvmem_cell_put(cell);
+
+       if (IS_ERR(data))
+               return PTR_ERR(data);
+
+       if (retlen != len) {
+               dev_err(rt2x00dev->dev, "invalid eeprom size, required: 0x%04x\n", len);
+               kfree(data);
+               return -EINVAL;
+       }
+
+       memcpy(rt2x00dev->eeprom, data, len);
+       kfree(data);
+       return 0;
+}
+EXPORT_SYMBOL_GPL(rt2800_read_eeprom_nvmem);
+
 static u8 rt2800_get_txmixer_gain_24g(struct rt2x00_dev *rt2x00dev)
 {
        u16 word;
index 620a3d9872cec51594c6bfce2f7eefa965a4972a..a3c3a751f57e3b982b5a2f334b1b6e6342b85e96 100644 (file)
@@ -248,6 +248,8 @@ void rt2800_disable_radio(struct rt2x00_dev *rt2x00dev);
 int rt2800_efuse_detect(struct rt2x00_dev *rt2x00dev);
 int rt2800_read_eeprom_efuse(struct rt2x00_dev *rt2x00dev);
 
+int rt2800_read_eeprom_nvmem(struct rt2x00_dev *rt2x00dev);
+
 int rt2800_probe_hw(struct rt2x00_dev *rt2x00dev);
 
 void rt2800_get_key_seq(struct ieee80211_hw *hw,
index 14c45aba836f2d5a4644021e814455b255791743..4fa14bb573add555a2880084237bb730987ef436 100644 (file)
@@ -278,6 +278,9 @@ static int rt2800pci_read_eeprom(struct rt2x00_dev *rt2x00dev)
 {
        int retval;
 
+       if (!rt2800_read_eeprom_nvmem(rt2x00dev))
+               return 0;
+
        if (rt2800pci_efuse_detect(rt2x00dev))
                retval = rt2800pci_read_eeprom_efuse(rt2x00dev);
        else
index 8f510a84e7f16fa60a8a64aca4dcd4b75d6d17c9..5c29201b34c8bca5cb3e47c48f7ed5898e7993e0 100644 (file)
@@ -92,8 +92,12 @@ static int rt2800soc_set_device_state(struct rt2x00_dev *rt2x00dev,
 
 static int rt2800soc_read_eeprom(struct rt2x00_dev *rt2x00dev)
 {
-       void __iomem *base_addr = ioremap(0x1F040000, EEPROM_SIZE);
+       void __iomem *base_addr;
 
+       if (!rt2800_read_eeprom_nvmem(rt2x00dev))
+               return 0;
+
+       base_addr = ioremap(0x1F040000, EEPROM_SIZE);
        if (!base_addr)
                return -ENOMEM;