1 // SPDX-License-Identifier: GPL-2.0+
3 * Copyright (C) 2019 DENX Software Engineering
4 * Lukasz Majewski, DENX Software Engineering, lukma@denx.de
11 #include <dm/pinctrl.h>
13 #include "pinctrl-mxs.h"
15 DECLARE_GLOBAL_DATA_PTR;
17 struct mxs_pinctrl_priv {
19 const struct mxs_regs *regs;
22 static unsigned long mxs_dt_node_to_map(struct udevice *conf)
24 unsigned long config = 0;
28 ret = dev_read_u32(conf, "fsl,drive-strength", &val);
30 config = val | MA_PRESENT;
32 ret = dev_read_u32(conf, "fsl,voltage", &val);
34 config |= val << VOL_SHIFT | VOL_PRESENT;
36 ret = dev_read_u32(conf, "fsl,pull-up", &val);
38 config |= val << PULL_SHIFT | PULL_PRESENT;
43 static int mxs_pinctrl_set_mux(struct udevice *dev, u32 val, int bank, int pin)
45 struct mxs_pinctrl_priv *iomux = dev_get_priv(dev);
46 int muxsel = MUXID_TO_MUXSEL(val), shift;
49 reg = iomux->base + iomux->regs->muxsel;
50 reg += bank * 0x20 + pin / 16 * 0x10;
53 mxs_pinctrl_rmwl(muxsel, 0x3, shift, reg);
54 debug(" mux %d,", muxsel);
59 static int mxs_pinctrl_set_state(struct udevice *dev, struct udevice *conf)
61 struct mxs_pinctrl_priv *iomux = dev_get_priv(dev);
62 u32 *pin_data, val, ma, vol, pull;
63 int npins, size, i, ret;
66 debug("\n%s: set state: %s\n", __func__, conf->name);
68 size = dev_read_size(conf, "fsl,pinmux-ids");
72 if (!size || size % sizeof(int)) {
73 dev_err(dev, "Invalid fsl,pinmux-ids property in %s\n",
78 npins = size / sizeof(int);
80 pin_data = devm_kzalloc(dev, size, 0);
84 ret = dev_read_u32_array(conf, "fsl,pinmux-ids", pin_data, npins);
86 dev_err(dev, "Error reading pin data.\n");
87 devm_kfree(dev, pin_data);
91 config = mxs_dt_node_to_map(conf);
93 ma = CONFIG_TO_MA(config);
94 vol = CONFIG_TO_VOL(config);
95 pull = CONFIG_TO_PULL(config);
97 for (i = 0; i < npins; i++) {
98 int pinid, bank, pin, shift;
103 pinid = MUXID_TO_PINID(val);
104 bank = PINID_TO_BANK(pinid);
105 pin = PINID_TO_PIN(pinid);
107 debug("(val: 0x%x) pin %d,", val, pinid);
109 mxs_pinctrl_set_mux(dev, val, bank, pin);
111 debug(" ma: %d, vol: %d, pull: %d\n", ma, vol, pull);
114 reg = iomux->base + iomux->regs->drive;
115 reg += bank * 0x40 + pin / 8 * 0x10;
118 if (config & MA_PRESENT) {
120 mxs_pinctrl_rmwl(ma, 0x3, shift, reg);
124 if (config & VOL_PRESENT) {
125 shift = pin % 8 * 4 + 2;
127 writel(1 << shift, reg + SET);
129 writel(1 << shift, reg + CLR);
133 if (config & PULL_PRESENT) {
134 reg = iomux->base + iomux->regs->pull;
138 writel(1 << shift, reg + SET);
140 writel(1 << shift, reg + CLR);
144 devm_kfree(dev, pin_data);
148 static struct pinctrl_ops mxs_pinctrl_ops = {
149 .set_state = mxs_pinctrl_set_state,
152 static int mxs_pinctrl_probe(struct udevice *dev)
154 struct mxs_pinctrl_priv *iomux = dev_get_priv(dev);
156 iomux->base = dev_read_addr_ptr(dev);
157 iomux->regs = (struct mxs_regs *)dev_get_driver_data(dev);
162 static const struct mxs_regs imx23_regs = {
168 static const struct mxs_regs imx28_regs = {
174 static const struct udevice_id mxs_pinctrl_match[] = {
175 { .compatible = "fsl,imx23-pinctrl", .data = (ulong)&imx23_regs },
176 { .compatible = "fsl,imx28-pinctrl", .data = (ulong)&imx28_regs },
180 U_BOOT_DRIVER(mxs_pinctrl) = {
181 .name = "mxs-pinctrl",
182 .id = UCLASS_PINCTRL,
183 .of_match = of_match_ptr(mxs_pinctrl_match),
184 .probe = mxs_pinctrl_probe,
185 #if !CONFIG_IS_ENABLED(OF_PLATDATA)
186 .bind = dm_scan_fdt_dev,
188 .priv_auto_alloc_size = sizeof(struct mxs_pinctrl_priv),
189 .ops = &mxs_pinctrl_ops,