]>
Commit | Line | Data |
---|---|---|
02239394 CL |
1 | // SPDX-License-Identifier: GPL-2.0+ |
2 | /* | |
3 | * (C) Copyright 2020 CS Group | |
4 | * Charles Frey <[email protected]> | |
5 | * | |
6 | * based on driver/gpio/mpc8xxx_gpio.c, which is | |
7 | * Copyright 2016 Mario Six, Guntermann & Drunck GmbH, [email protected] | |
8 | * | |
9 | * based on arch/powerpc/include/asm/mpc85xx_gpio.h, which is | |
10 | * Copyright 2010 eXMeritus, A Boeing Company | |
11 | */ | |
12 | ||
02239394 CL |
13 | #include <asm/io.h> |
14 | #include <dm.h> | |
15 | #include <mapmem.h> | |
16 | #include <asm/gpio.h> | |
17 | #include <malloc.h> | |
18 | ||
19 | enum { | |
20 | MPC8XX_CPM1_PORTA, | |
21 | MPC8XX_CPM1_PORTB, | |
22 | MPC8XX_CPM1_PORTC, | |
23 | MPC8XX_CPM1_PORTD, | |
24 | MPC8XX_CPM1_PORTE, | |
25 | }; | |
26 | ||
27 | /* | |
28 | * The MPC885 CPU CPM has 5 I/O ports, and each ports has different | |
29 | * register length : 16 bits for ports A,C,D and 32 bits for ports | |
30 | * B and E. | |
31 | * | |
32 | * This structure allows us to select the accessors according to the | |
33 | * port we are configuring. | |
34 | */ | |
35 | struct mpc8xx_gpio_data { | |
36 | /* The bank's register base in memory */ | |
37 | void __iomem *base; | |
38 | /* The address of the registers; used to identify the bank */ | |
39 | ulong addr; | |
40 | /* The GPIO count of the bank */ | |
41 | uint gpio_count; | |
42 | /* Type needed to use the correct accessors */ | |
43 | int type; | |
44 | }; | |
45 | ||
46 | /* Structure for ports A, C, D */ | |
47 | struct iop_16 { | |
48 | u16 pdir; | |
49 | u16 ppar; | |
50 | u16 podr; | |
51 | u16 pdat; | |
52 | }; | |
53 | ||
54 | /* Port B */ | |
55 | struct iop_32_b { | |
56 | u32 pdir; | |
57 | u32 ppar; | |
58 | u32 podr; | |
59 | u32 pdat; | |
60 | }; | |
61 | ||
62 | /* Port E */ | |
63 | struct iop_32_e { | |
64 | u32 pdir; | |
65 | u32 ppar; | |
66 | u32 psor; | |
67 | u32 podr; | |
68 | u32 pdat; | |
69 | }; | |
70 | ||
71 | union iop_32 { | |
72 | struct iop_32_b b; | |
73 | struct iop_32_e e; | |
74 | }; | |
75 | ||
76 | inline u32 gpio_mask(uint gpio, int type) | |
77 | { | |
78 | if (type == MPC8XX_CPM1_PORTB || type == MPC8XX_CPM1_PORTE) | |
79 | return 1U << (31 - (gpio)); | |
80 | else | |
81 | return 1U << (15 - (gpio)); | |
82 | } | |
83 | ||
84 | static inline u16 gpio16_get_val(void __iomem *base, u16 mask, int type) | |
85 | { | |
86 | struct iop_16 *regs = base; | |
87 | ||
88 | return in_be16(®s->pdat) & mask; | |
89 | } | |
90 | ||
91 | static inline u16 gpio16_get_dir(void __iomem *base, u16 mask, int type) | |
92 | { | |
93 | struct iop_16 *regs = base; | |
94 | ||
95 | return in_be16(®s->pdir) & mask; | |
96 | } | |
97 | ||
98 | static inline void gpio16_set_in(void __iomem *base, u16 gpios, int type) | |
99 | { | |
100 | struct iop_16 *regs = base; | |
101 | ||
102 | clrbits_be16(®s->pdat, gpios); | |
103 | /* GPDIR register 0 -> input */ | |
104 | clrbits_be16(®s->pdir, gpios); | |
105 | } | |
106 | ||
107 | static inline void gpio16_set_lo(void __iomem *base, u16 gpios, int type) | |
108 | { | |
109 | struct iop_16 *regs = base; | |
110 | ||
111 | clrbits_be16(®s->pdat, gpios); | |
112 | /* GPDIR register 1 -> output */ | |
113 | setbits_be16(®s->pdir, gpios); | |
114 | } | |
115 | ||
116 | static inline void gpio16_set_hi(void __iomem *base, u16 gpios, int type) | |
117 | { | |
118 | struct iop_16 *regs = base; | |
119 | ||
120 | setbits_be16(®s->pdat, gpios); | |
121 | /* GPDIR register 1 -> output */ | |
122 | setbits_be16(®s->pdir, gpios); | |
123 | } | |
124 | ||
125 | /* PORT B AND E */ | |
126 | static inline u32 gpio32_get_val(void __iomem *base, u32 mask, int type) | |
127 | { | |
128 | union iop_32 __iomem *regs = base; | |
129 | ||
130 | if (type == MPC8XX_CPM1_PORTB) | |
131 | return in_be32(®s->b.pdat) & mask; | |
132 | else | |
133 | return in_be32(®s->e.pdat) & mask; | |
134 | } | |
135 | ||
136 | static inline u32 gpio32_get_dir(void __iomem *base, u32 mask, int type) | |
137 | { | |
138 | union iop_32 __iomem *regs = base; | |
139 | ||
140 | if (type == MPC8XX_CPM1_PORTB) | |
141 | return in_be32(®s->b.pdir) & mask; | |
142 | else | |
143 | return in_be32(®s->e.pdir) & mask; | |
144 | } | |
145 | ||
146 | static inline void gpio32_set_in(void __iomem *base, u32 gpios, int type) | |
147 | { | |
148 | union iop_32 __iomem *regs = base; | |
149 | ||
150 | if (type == MPC8XX_CPM1_PORTB) { | |
151 | clrbits_be32(®s->b.pdat, gpios); | |
152 | /* GPDIR register 0 -> input */ | |
153 | clrbits_be32(®s->b.pdir, gpios); | |
154 | } else { /* Port E */ | |
155 | clrbits_be32(®s->e.pdat, gpios); | |
156 | /* GPDIR register 0 -> input */ | |
157 | clrbits_be32(®s->e.pdir, gpios); | |
158 | } | |
159 | } | |
160 | ||
161 | static inline void gpio32_set_lo(void __iomem *base, u32 gpios, int type) | |
162 | { | |
163 | union iop_32 __iomem *regs = base; | |
164 | ||
165 | if (type == MPC8XX_CPM1_PORTB) { | |
166 | clrbits_be32(®s->b.pdat, gpios); | |
167 | /* GPDIR register 1 -> output */ | |
168 | setbits_be32(®s->b.pdir, gpios); | |
169 | } else { | |
170 | clrbits_be32(®s->e.pdat, gpios); | |
171 | /* GPDIR register 1 -> output */ | |
172 | setbits_be32(®s->e.pdir, gpios); | |
173 | } | |
174 | } | |
175 | ||
176 | static inline void gpio32_set_hi(void __iomem *base, u32 gpios, int type) | |
177 | { | |
178 | union iop_32 __iomem *regs = base; | |
179 | ||
180 | if (type == MPC8XX_CPM1_PORTB) { | |
181 | setbits_be32(®s->b.pdat, gpios); | |
182 | /* GPDIR register 1 -> output */ | |
183 | setbits_be32(®s->b.pdir, gpios); | |
184 | } else { | |
185 | setbits_be32(®s->e.pdat, gpios); | |
186 | /* GPDIR register 1 -> output */ | |
187 | setbits_be32(®s->e.pdir, gpios); | |
188 | } | |
189 | } | |
190 | ||
191 | static int mpc8xx_gpio_direction_input(struct udevice *dev, uint gpio) | |
192 | { | |
193 | struct mpc8xx_gpio_data *data = dev_get_priv(dev); | |
194 | int type = data->type; | |
195 | ||
196 | if (type == MPC8XX_CPM1_PORTB || type == MPC8XX_CPM1_PORTE) | |
197 | gpio32_set_in(data->base, gpio_mask(gpio, type), type); | |
198 | else | |
199 | gpio16_set_in(data->base, gpio_mask(gpio, type), type); | |
200 | ||
201 | return 0; | |
202 | } | |
203 | ||
204 | static int mpc8xx_gpio_set_value(struct udevice *dev, uint gpio, int value) | |
205 | { | |
206 | struct mpc8xx_gpio_data *data = dev_get_priv(dev); | |
207 | int type = data->type; | |
208 | ||
209 | if (type == MPC8XX_CPM1_PORTB || type == MPC8XX_CPM1_PORTE) { | |
210 | if (value) | |
211 | gpio32_set_hi(data->base, gpio_mask(gpio, type), type); | |
212 | else | |
213 | gpio32_set_lo(data->base, gpio_mask(gpio, type), type); | |
214 | } else { | |
215 | if (value) | |
216 | gpio16_set_hi(data->base, gpio_mask(gpio, type), type); | |
217 | else | |
218 | gpio16_set_lo(data->base, gpio_mask(gpio, type), type); | |
219 | } | |
220 | ||
221 | return 0; | |
222 | } | |
223 | ||
224 | static int mpc8xx_gpio_direction_output(struct udevice *dev, uint gpio, | |
225 | int value) | |
226 | { | |
227 | return mpc8xx_gpio_set_value(dev, gpio, value); | |
228 | } | |
229 | ||
230 | static int mpc8xx_gpio_get_value(struct udevice *dev, uint gpio) | |
231 | { | |
232 | struct mpc8xx_gpio_data *data = dev_get_priv(dev); | |
233 | int type = data->type; | |
234 | ||
235 | /* Input -> read value from GPDAT register */ | |
236 | if (type == MPC8XX_CPM1_PORTB || type == MPC8XX_CPM1_PORTE) | |
237 | return gpio32_get_val(data->base, gpio_mask(gpio, type), type); | |
238 | else | |
239 | return gpio16_get_val(data->base, gpio_mask(gpio, type), type); | |
240 | } | |
241 | ||
242 | static int mpc8xx_gpio_get_function(struct udevice *dev, uint gpio) | |
243 | { | |
244 | struct mpc8xx_gpio_data *data = dev_get_priv(dev); | |
245 | int type = data->type; | |
246 | int dir; | |
247 | ||
248 | if (type == MPC8XX_CPM1_PORTB || type == MPC8XX_CPM1_PORTE) | |
249 | dir = gpio32_get_dir(data->base, gpio_mask(gpio, type), type); | |
250 | else | |
251 | dir = gpio16_get_dir(data->base, gpio_mask(gpio, type), type); | |
252 | return dir ? GPIOF_OUTPUT : GPIOF_INPUT; | |
253 | } | |
254 | ||
255 | static int mpc8xx_gpio_ofdata_to_platdata(struct udevice *dev) | |
256 | { | |
257 | struct mpc8xx_gpio_plat *plat = dev_get_plat(dev); | |
258 | fdt_addr_t addr; | |
259 | u32 reg[2]; | |
260 | ||
261 | dev_read_u32_array(dev, "reg", reg, 2); | |
262 | addr = dev_translate_address(dev, reg); | |
263 | ||
264 | plat->addr = addr; | |
265 | plat->size = reg[1]; | |
266 | plat->ngpios = dev_read_u32_default(dev, "ngpios", 32); | |
267 | ||
268 | return 0; | |
269 | } | |
270 | ||
271 | static int mpc8xx_gpio_platdata_to_priv(struct udevice *dev) | |
272 | { | |
273 | struct mpc8xx_gpio_data *priv = dev_get_priv(dev); | |
274 | struct mpc8xx_gpio_plat *plat = dev_get_plat(dev); | |
275 | unsigned long size = plat->size; | |
276 | int type; | |
277 | ||
278 | if (size == 0) | |
279 | size = 0x100; | |
280 | ||
281 | priv->addr = plat->addr; | |
282 | priv->base = map_sysmem(plat->addr, size); | |
283 | ||
284 | if (!priv->base) | |
285 | return -ENOMEM; | |
286 | ||
287 | priv->gpio_count = plat->ngpios; | |
288 | ||
289 | type = dev_get_driver_data(dev); | |
290 | ||
291 | if ((type == MPC8XX_CPM1_PORTA || type == MPC8XX_CPM1_PORTC || | |
292 | type == MPC8XX_CPM1_PORTD) && plat->ngpios == 32) | |
293 | priv->gpio_count = 16; | |
294 | ||
295 | priv->type = type; | |
296 | ||
297 | return 0; | |
298 | } | |
299 | ||
300 | static int mpc8xx_gpio_probe(struct udevice *dev) | |
301 | { | |
302 | struct gpio_dev_priv *uc_priv = dev_get_uclass_priv(dev); | |
303 | struct mpc8xx_gpio_data *data = dev_get_priv(dev); | |
304 | char name[32], *str; | |
305 | ||
306 | mpc8xx_gpio_platdata_to_priv(dev); | |
307 | ||
308 | snprintf(name, sizeof(name), "MPC@%lx_", data->addr); | |
309 | str = strdup(name); | |
310 | ||
311 | if (!str) | |
312 | return -ENOMEM; | |
313 | ||
314 | uc_priv->bank_name = str; | |
315 | uc_priv->gpio_count = data->gpio_count; | |
316 | ||
317 | return 0; | |
318 | } | |
319 | ||
320 | static const struct dm_gpio_ops gpio_mpc8xx_ops = { | |
321 | .direction_input = mpc8xx_gpio_direction_input, | |
322 | .direction_output = mpc8xx_gpio_direction_output, | |
323 | .get_value = mpc8xx_gpio_get_value, | |
324 | .set_value = mpc8xx_gpio_set_value, | |
325 | .get_function = mpc8xx_gpio_get_function, | |
326 | }; | |
327 | ||
328 | static const struct udevice_id mpc8xx_gpio_ids[] = { | |
329 | { .compatible = "fsl,cpm1-pario-bank-a", .data = MPC8XX_CPM1_PORTA }, | |
330 | { .compatible = "fsl,cpm1-pario-bank-b", .data = MPC8XX_CPM1_PORTB }, | |
331 | { .compatible = "fsl,cpm1-pario-bank-c", .data = MPC8XX_CPM1_PORTC }, | |
332 | { .compatible = "fsl,cpm1-pario-bank-d", .data = MPC8XX_CPM1_PORTD }, | |
333 | { .compatible = "fsl,cpm1-pario-bank-e", .data = MPC8XX_CPM1_PORTE }, | |
334 | { /* sentinel */ } | |
335 | }; | |
336 | ||
337 | U_BOOT_DRIVER(gpio_mpc8xx) = { | |
338 | .name = "gpio_mpc8xx", | |
339 | .id = UCLASS_GPIO, | |
340 | .ops = &gpio_mpc8xx_ops, | |
341 | .of_to_plat = mpc8xx_gpio_ofdata_to_platdata, | |
342 | .plat_auto = sizeof(struct mpc8xx_gpio_plat), | |
343 | .of_match = mpc8xx_gpio_ids, | |
344 | .probe = mpc8xx_gpio_probe, | |
345 | .priv_auto = sizeof(struct mpc8xx_gpio_data), | |
346 | }; |