2 * Copyright (C) 2015 Masahiro Yamada <yamada.masahiro@socionext.com>
4 * SPDX-License-Identifier: GPL-2.0+
10 #include <linux/err.h>
11 #include <linux/sizes.h>
12 #include <dm/device.h>
13 #include <dm/pinctrl.h>
15 #include "pinctrl-uniphier.h"
17 static int uniphier_pinctrl_get_groups_count(struct udevice *dev)
19 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
21 return priv->socdata->groups_count;
24 static const char *uniphier_pinctrl_get_group_name(struct udevice *dev,
27 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
29 return priv->socdata->groups[selector].name;
32 static int uniphier_pinmux_get_functions_count(struct udevice *dev)
34 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
36 return priv->socdata->functions_count;
39 static const char *uniphier_pinmux_get_function_name(struct udevice *dev,
42 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
44 return priv->socdata->functions[selector];
47 static void uniphier_pinconf_input_enable(struct udevice *dev, unsigned pin)
49 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
50 int pins_count = priv->socdata->pins_count;
51 const struct uniphier_pinctrl_pin *pins = priv->socdata->pins;
54 for (i = 0; i < pins_count; i++) {
55 if (pins[i].number == pin) {
59 iectrl = uniphier_pin_get_iectrl(pins[i].data);
60 tmp = readl(priv->base + UNIPHIER_PINCTRL_IECTRL);
62 writel(tmp, priv->base + UNIPHIER_PINCTRL_IECTRL);
67 static void uniphier_pinmux_set_one(struct udevice *dev, unsigned pin,
70 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
71 unsigned mux_bits = priv->socdata->mux_bits;
72 unsigned reg_stride = priv->socdata->reg_stride;
73 unsigned reg, reg_end, shift, mask;
76 /* some pins need input-enabling */
77 uniphier_pinconf_input_enable(dev, pin);
79 reg = UNIPHIER_PINCTRL_PINMUX_BASE + pin * mux_bits / 32 * reg_stride;
80 reg_end = reg + reg_stride;
81 shift = pin * mux_bits % 32;
82 mask = (1U << mux_bits) - 1;
85 * If reg_stride is greater than 4, the MSB of each pinsel shall be
86 * stored in the offset+4.
88 for (; reg < reg_end; reg += 4) {
89 tmp = readl(priv->base + reg);
90 tmp &= ~(mask << shift);
91 tmp |= (mask & muxval) << shift;
92 writel(tmp, priv->base + reg);
97 if (priv->socdata->load_pinctrl)
98 writel(1, priv->base + UNIPHIER_PINCTRL_LOAD_PINMUX);
101 static int uniphier_pinmux_group_set(struct udevice *dev,
102 unsigned group_selector,
103 unsigned func_selector)
105 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
106 const struct uniphier_pinctrl_group *grp =
107 &priv->socdata->groups[group_selector];
110 for (i = 0; i < grp->num_pins; i++)
111 uniphier_pinmux_set_one(dev, grp->pins[i], grp->muxvals[i]);
116 const struct pinctrl_ops uniphier_pinctrl_ops = {
117 .get_groups_count = uniphier_pinctrl_get_groups_count,
118 .get_group_name = uniphier_pinctrl_get_group_name,
119 .get_functions_count = uniphier_pinmux_get_functions_count,
120 .get_function_name = uniphier_pinmux_get_function_name,
121 .pinmux_group_set = uniphier_pinmux_group_set,
122 .set_state = pinctrl_generic_set_state,
125 int uniphier_pinctrl_probe(struct udevice *dev,
126 struct uniphier_pinctrl_socdata *socdata)
128 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
131 addr = dev_get_addr(dev);
132 if (addr == FDT_ADDR_T_NONE)
135 priv->base = map_sysmem(addr, SZ_4K);
139 priv->socdata = socdata;
144 int uniphier_pinctrl_remove(struct udevice *dev)
146 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
148 unmap_sysmem(priv->base);