2 * Copyright (C) 2015-2016 Socionext Inc.
3 * Author: Masahiro Yamada <yamada.masahiro@socionext.com>
5 * SPDX-License-Identifier: GPL-2.0+
10 #include <linux/sizes.h>
11 #include <dm/device.h>
12 #include <dm/pinctrl.h>
14 #include "pinctrl-uniphier.h"
16 #define UNIPHIER_PINCTRL_PINMUX_BASE 0x1000
17 #define UNIPHIER_PINCTRL_LOAD_PINMUX 0x1700
18 #define UNIPHIER_PINCTRL_IECTRL 0x1d00
20 static const char *uniphier_pinctrl_dummy_name = "_dummy";
22 static int uniphier_pinctrl_get_groups_count(struct udevice *dev)
24 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
26 return priv->socdata->groups_count;
29 static const char *uniphier_pinctrl_get_group_name(struct udevice *dev,
32 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
34 if (!priv->socdata->groups[selector].name)
35 return uniphier_pinctrl_dummy_name;
37 return priv->socdata->groups[selector].name;
40 static int uniphier_pinmux_get_functions_count(struct udevice *dev)
42 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
44 return priv->socdata->functions_count;
47 static const char *uniphier_pinmux_get_function_name(struct udevice *dev,
50 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
52 if (!priv->socdata->functions[selector])
53 return uniphier_pinctrl_dummy_name;
55 return priv->socdata->functions[selector];
58 static void uniphier_pinconf_input_enable_perpin(struct udevice *dev,
61 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
65 reg = UNIPHIER_PINCTRL_IECTRL + pin / 32 * 4;
68 tmp = readl(priv->base + reg);
70 writel(tmp, priv->base + reg);
73 static void uniphier_pinconf_input_enable_legacy(struct udevice *dev,
76 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
77 int pins_count = priv->socdata->pins_count;
78 const struct uniphier_pinctrl_pin *pins = priv->socdata->pins;
81 for (i = 0; i < pins_count; i++) {
82 if (pins[i].number == pin) {
86 iectrl = uniphier_pin_get_iectrl(pins[i].data);
87 tmp = readl(priv->base + UNIPHIER_PINCTRL_IECTRL);
89 writel(tmp, priv->base + UNIPHIER_PINCTRL_IECTRL);
94 static void uniphier_pinconf_input_enable(struct udevice *dev, unsigned pin)
96 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
98 if (priv->socdata->caps & UNIPHIER_PINCTRL_CAPS_PERPIN_IECTRL)
99 uniphier_pinconf_input_enable_perpin(dev, pin);
101 uniphier_pinconf_input_enable_legacy(dev, pin);
104 static void uniphier_pinmux_set_one(struct udevice *dev, unsigned pin,
107 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
108 unsigned mux_bits, reg_stride, reg, reg_end, shift, mask;
112 /* some pins need input-enabling */
113 uniphier_pinconf_input_enable(dev, pin);
116 return; /* dedicated pin; nothing to do for pin-mux */
118 if (priv->socdata->caps & UNIPHIER_PINCTRL_CAPS_DBGMUX_SEPARATE) {
121 * Normal 4 * n shift+3:shift
122 * Debug 4 * n shift+7:shift+4
130 * Normal 8 * n shift+3:shift
131 * Debug 8 * n + 4 shift+3:shift
135 load_pinctrl = false;
138 reg = UNIPHIER_PINCTRL_PINMUX_BASE + pin * mux_bits / 32 * reg_stride;
139 reg_end = reg + reg_stride;
140 shift = pin * mux_bits % 32;
141 mask = (1U << mux_bits) - 1;
144 * If reg_stride is greater than 4, the MSB of each pinsel shall be
145 * stored in the offset+4.
147 for (; reg < reg_end; reg += 4) {
148 tmp = readl(priv->base + reg);
149 tmp &= ~(mask << shift);
150 tmp |= (mask & muxval) << shift;
151 writel(tmp, priv->base + reg);
157 writel(1, priv->base + UNIPHIER_PINCTRL_LOAD_PINMUX);
160 static int uniphier_pinmux_group_set(struct udevice *dev,
161 unsigned group_selector,
162 unsigned func_selector)
164 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
165 const struct uniphier_pinctrl_group *grp =
166 &priv->socdata->groups[group_selector];
169 for (i = 0; i < grp->num_pins; i++)
170 uniphier_pinmux_set_one(dev, grp->pins[i], grp->muxvals[i]);
175 const struct pinctrl_ops uniphier_pinctrl_ops = {
176 .get_groups_count = uniphier_pinctrl_get_groups_count,
177 .get_group_name = uniphier_pinctrl_get_group_name,
178 .get_functions_count = uniphier_pinmux_get_functions_count,
179 .get_function_name = uniphier_pinmux_get_function_name,
180 .pinmux_group_set = uniphier_pinmux_group_set,
181 .set_state = pinctrl_generic_set_state,
184 int uniphier_pinctrl_probe(struct udevice *dev,
185 struct uniphier_pinctrl_socdata *socdata)
187 struct uniphier_pinctrl_priv *priv = dev_get_priv(dev);
190 addr = dev_get_addr(dev->parent);
191 if (addr == FDT_ADDR_T_NONE)
194 priv->base = devm_ioremap(dev, addr, SZ_4K);
198 priv->socdata = socdata;