[U-Boot] [PATCH 1/3] pinctrl: pinctrl-single: Handle different register width
Vladimir Olovyannikov
vladimir.olovyannikov at broadcom.com
Fri Nov 22 22:44:41 UTC 2019
From: Rayagonda Kokatanur <rayagonda.kokatanur at broadcom.com>
Add support to use different register read/write api's
based on register width.
Signed-off-by: Rayagonda Kokatanur <rayagonda.kokatanur at broadcom.com>
Signed-off-by: Vladimir Olovyannikov <vladimir.olovyannikov at broadcom.com>
---
drivers/pinctrl/pinctrl-single.c | 111 ++++++++++++++++++++++---------
1 file changed, 80 insertions(+), 31 deletions(-)
diff --git a/drivers/pinctrl/pinctrl-single.c b/drivers/pinctrl/pinctrl-single.c
index 1dfc97dcea..6c6a33e4c5 100644
--- a/drivers/pinctrl/pinctrl-single.c
+++ b/drivers/pinctrl/pinctrl-single.c
@@ -11,12 +11,23 @@
DECLARE_GLOBAL_DATA_PTR;
+/**
+ * struct single_pdata - pinctrl device instance
+ * @base first configuration register
+ * @offset index of last configuration register
+ * @mask configuration-value mask bits
+ * @width configuration register bit width
+ * @read register read function to use
+ * @write register write function to use
+ */
struct single_pdata {
- fdt_addr_t base; /* first configuration register */
- int offset; /* index of last configuration register */
- u32 mask; /* configuration-value mask bits */
- int width; /* configuration register bit width */
+ void __iomem *base;
+ int offset;
+ u32 mask;
+ int width;
bool bits_per_mux;
+ u32 (*read)(void __iomem *reg);
+ void (*write)(u32 val, void __iomem *reg);
};
struct single_fdt_pin_cfg {
@@ -30,6 +41,36 @@ struct single_fdt_bits_cfg {
fdt32_t mask; /* configuration register mask */
};
+static u32 __maybe_unused single_readb(void __iomem *reg)
+{
+ return readb(reg);
+}
+
+static u32 __maybe_unused single_readw(void __iomem *reg)
+{
+ return readw(reg);
+}
+
+static u32 __maybe_unused single_readl(void __iomem *reg)
+{
+ return readl(reg);
+}
+
+static void __maybe_unused single_writeb(u32 val, void __iomem *reg)
+{
+ writeb(val, reg);
+}
+
+static void __maybe_unused single_writew(u32 val, void __iomem *reg)
+{
+ writew(val, reg);
+}
+
+static void __maybe_unused single_writel(u32 val, void __iomem *reg)
+{
+ writel(val, reg);
+}
+
/**
* single_configure_pins() - Configure pins based on FDT data
*
@@ -55,24 +96,15 @@ static int single_configure_pins(struct udevice *dev,
for (n = 0; n < count; n++, pins++) {
reg = fdt32_to_cpu(pins->reg);
- if ((reg < 0) || (reg > pdata->offset)) {
+ if (reg > pdata->offset) {
dev_dbg(dev, " invalid register offset 0x%pa\n", ®);
continue;
}
- reg += pdata->base;
+ reg += (phys_addr_t)pdata->base;
val = fdt32_to_cpu(pins->val) & pdata->mask;
- switch (pdata->width) {
- case 16:
- writew((readw(reg) & ~pdata->mask) | val, reg);
- break;
- case 32:
- writel((readl(reg) & ~pdata->mask) | val, reg);
- break;
- default:
- dev_warn(dev, "unsupported register width %i\n",
- pdata->width);
- continue;
- }
+ val |= pdata->read((void __iomem *)reg) & ~pdata->mask;
+ pdata->write(val, (void __iomem *)reg);
+
dev_dbg(dev, " reg/val 0x%pa/0x%08x\n", ®, val);
}
return 0;
@@ -97,19 +129,9 @@ static int single_configure_bits(struct udevice *dev,
mask = fdt32_to_cpu(pins->mask);
val = fdt32_to_cpu(pins->val) & mask;
+ val |= pdata->read((void __iomem *)reg) & ~mask;
+ pdata->write(val, (void __iomem *)reg);
- switch (pdata->width) {
- case 16:
- writew((readw(reg) & ~mask) | val, reg);
- break;
- case 32:
- writel((readl(reg) & ~mask) | val, reg);
- break;
- default:
- dev_warn(dev, "unsupported register width %i\n",
- pdata->width);
- continue;
- }
dev_dbg(dev, " reg/val 0x%pa/0x%08x\n", ®, val);
}
return 0;
@@ -153,6 +175,32 @@ static int single_set_state(struct udevice *dev,
return len;
}
+static int single_probe(struct udevice *dev)
+{
+ struct single_pdata *pdata = dev->platdata;
+
+ switch (pdata->width) {
+ case 8:
+ pdata->read = single_readb;
+ pdata->write = single_writeb;
+ break;
+ case 16:
+ pdata->read = single_readw;
+ pdata->write = single_writew;
+ break;
+ case 32:
+ pdata->read = single_readl;
+ pdata->write = single_writel;
+ break;
+ default:
+ dev_warn(dev, "%s: unsupported register width %d\n",
+ __func__, pdata->width);
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
static int single_ofdata_to_platdata(struct udevice *dev)
{
fdt_addr_t addr;
@@ -174,7 +222,7 @@ static int single_ofdata_to_platdata(struct udevice *dev)
dev_dbg(dev, "no valid base register address\n");
return -EINVAL;
}
- pdata->base = addr;
+ pdata->base = (void __iomem *)addr;
pdata->mask = fdtdec_get_int(gd->fdt_blob, dev_of_offset(dev),
"pinctrl-single,function-mask",
@@ -201,4 +249,5 @@ U_BOOT_DRIVER(single_pinctrl) = {
.ops = &single_pinctrl_ops,
.platdata_auto_alloc_size = sizeof(struct single_pdata),
.ofdata_to_platdata = single_ofdata_to_platdata,
+ .probe = single_probe,
};
--
2.17.1
More information about the U-Boot
mailing list