blob: 27f274ff022416a6d66e895b92a0239fedefc5fe [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Wenyou Yang9319a752017-03-23 12:44:37 +08002/*
3 * Atmel PIO pinctrl driver
4 *
5 * Copyright (C) 2016 Atmel Corporation
6 * Wenyou.Yang <wenyou.yang@atmel.com>
Wenyou Yang9319a752017-03-23 12:44:37 +08007 */
8
9#include <common.h>
Simon Glass9d922452017-05-17 17:18:03 -060010#include <dm.h>
Wenyou Yang9319a752017-03-23 12:44:37 +080011#include <dm/pinctrl.h>
Wenyou Yange61ed482017-09-14 11:07:42 +080012#include <asm/hardware.h>
Wenyou Yang9319a752017-03-23 12:44:37 +080013#include <linux/io.h>
14#include <linux/err.h>
15#include <mach/at91_pio.h>
16
17DECLARE_GLOBAL_DATA_PTR;
18
19#define MAX_GPIO_BANKS 5
20#define MAX_NB_GPIO_PER_BANK 32
21
22#define MAX_PINMUX_ENTRIES 200
23
24struct at91_pinctrl_priv {
25 struct at91_port *reg_base[MAX_GPIO_BANKS];
26 u32 nbanks;
27};
28
29#define PULL_UP BIT(0)
30#define MULTI_DRIVE BIT(1)
31#define DEGLITCH BIT(2)
32#define PULL_DOWN BIT(3)
33#define DIS_SCHMIT BIT(4)
34#define DRIVE_STRENGTH_SHIFT 5
35#define DRIVE_STRENGTH_MASK 0x3
36#define DRIVE_STRENGTH (DRIVE_STRENGTH_MASK << DRIVE_STRENGTH_SHIFT)
37#define OUTPUT BIT(7)
38#define OUTPUT_VAL_SHIFT 8
39#define OUTPUT_VAL (0x1 << OUTPUT_VAL_SHIFT)
Claudiu Beznea068d4c0a2019-03-25 10:34:00 +000040#define SLEWRATE_SHIFT 9
41#define SLEWRATE_MASK 0x1
42#define SLEWRATE (SLEWRATE_MASK << SLEWRATE_SHIFT)
Wenyou Yang9319a752017-03-23 12:44:37 +080043#define DEBOUNCE BIT(16)
44#define DEBOUNCE_VAL_SHIFT 17
45#define DEBOUNCE_VAL (0x3fff << DEBOUNCE_VAL_SHIFT)
46
47/**
48 * These defines will translated the dt binding settings to our internal
49 * settings. They are not necessarily the same value as the register setting.
50 * The actual drive strength current of low, medium and high must be looked up
51 * from the corresponding device datasheet. This value is different for pins
52 * that are even in the same banks. It is also dependent on VCC.
53 * DRIVE_STRENGTH_DEFAULT is just a placeholder to avoid changing the drive
54 * strength when there is no dt config for it.
55 */
Claudiu Beznea04d4ec92019-03-25 10:33:56 +000056enum drive_strength_bit {
57 DRIVE_STRENGTH_BIT_DEF,
58 DRIVE_STRENGTH_BIT_LOW,
59 DRIVE_STRENGTH_BIT_MED,
60 DRIVE_STRENGTH_BIT_HI,
61};
62
63#define DRIVE_STRENGTH_BIT_MSK(name) (DRIVE_STRENGTH_BIT_##name << \
64 DRIVE_STRENGTH_SHIFT)
Wenyou Yang9319a752017-03-23 12:44:37 +080065
Claudiu Beznea068d4c0a2019-03-25 10:34:00 +000066enum slewrate_bit {
67 SLEWRATE_BIT_DIS,
68 SLEWRATE_BIT_ENA,
69};
70
71#define SLEWRATE_BIT_MSK(name) (SLEWRATE_BIT_##name << SLEWRATE_SHIFT)
72
Wenyou Yang9319a752017-03-23 12:44:37 +080073enum at91_mux {
74 AT91_MUX_GPIO = 0,
75 AT91_MUX_PERIPH_A = 1,
76 AT91_MUX_PERIPH_B = 2,
77 AT91_MUX_PERIPH_C = 3,
78 AT91_MUX_PERIPH_D = 4,
79};
80
81/**
82 * struct at91_pinctrl_mux_ops - describes an AT91 mux ops group
83 * on new IP with support for periph C and D the way to mux in
84 * periph A and B has changed
85 * So provide the right callbacks
86 * if not present means the IP does not support it
87 * @mux_A_periph: assign the corresponding pin to the peripheral A function.
88 * @mux_B_periph: assign the corresponding pin to the peripheral B function.
89 * @mux_C_periph: assign the corresponding pin to the peripheral C function.
90 * @mux_D_periph: assign the corresponding pin to the peripheral D function.
91 * @set_deglitch: enable/disable the deglitch feature.
92 * @set_debounce: enable/disable the debounce feature.
93 * @set_pulldown: enable/disable the pulldown feature.
94 * @disable_schmitt_trig: disable schmitt trigger
95 */
96struct at91_pinctrl_mux_ops {
97 void (*mux_A_periph)(struct at91_port *pio, u32 mask);
98 void (*mux_B_periph)(struct at91_port *pio, u32 mask);
99 void (*mux_C_periph)(struct at91_port *pio, u32 mask);
100 void (*mux_D_periph)(struct at91_port *pio, u32 mask);
101 void (*set_deglitch)(struct at91_port *pio, u32 mask, bool is_on);
102 void (*set_debounce)(struct at91_port *pio, u32 mask, bool is_on,
103 u32 div);
104 void (*set_pulldown)(struct at91_port *pio, u32 mask, bool is_on);
105 void (*disable_schmitt_trig)(struct at91_port *pio, u32 mask);
106 void (*set_drivestrength)(struct at91_port *pio, u32 pin,
107 u32 strength);
Claudiu Beznea068d4c0a2019-03-25 10:34:00 +0000108 void (*set_slewrate)(struct at91_port *pio, u32 pin, u32 slewrate);
Wenyou Yang9319a752017-03-23 12:44:37 +0800109};
110
111static u32 two_bit_pin_value_shift_amount(u32 pin)
112{
113 /* return the shift value for a pin for "two bit" per pin registers,
114 * i.e. drive strength */
115 return 2 * ((pin >= MAX_NB_GPIO_PER_BANK/2)
116 ? pin - MAX_NB_GPIO_PER_BANK/2 : pin);
117}
118
119static void at91_mux_disable_interrupt(struct at91_port *pio, u32 mask)
120{
121 writel(mask, &pio->idr);
122}
123
124static void at91_mux_set_pullup(struct at91_port *pio, u32 mask, bool on)
125{
126 if (on)
127 writel(mask, &pio->mux.pio3.ppddr);
128
129 writel(mask, (on ? &pio->puer : &pio->pudr));
130}
131
132static void at91_mux_set_output(struct at91_port *pio, unsigned mask,
133 bool is_on, bool val)
134{
135 writel(mask, (val ? &pio->sodr : &pio->codr));
136 writel(mask, (is_on ? &pio->oer : &pio->odr));
137}
138
139static void at91_mux_set_multidrive(struct at91_port *pio, u32 mask, bool on)
140{
141 writel(mask, (on ? &pio->mder : &pio->mddr));
142}
143
144static void at91_mux_set_A_periph(struct at91_port *pio, u32 mask)
145{
146 writel(mask, &pio->mux.pio2.asr);
147}
148
149static void at91_mux_set_B_periph(struct at91_port *pio, u32 mask)
150{
151 writel(mask, &pio->mux.pio2.bsr);
152}
153
154static void at91_mux_pio3_set_A_periph(struct at91_port *pio, u32 mask)
155{
156 writel(readl(&pio->mux.pio3.abcdsr1) & ~mask, &pio->mux.pio3.abcdsr1);
157 writel(readl(&pio->mux.pio3.abcdsr2) & ~mask, &pio->mux.pio3.abcdsr2);
158}
159
160static void at91_mux_pio3_set_B_periph(struct at91_port *pio, u32 mask)
161{
162 writel(readl(&pio->mux.pio3.abcdsr1) | mask, &pio->mux.pio3.abcdsr1);
163 writel(readl(&pio->mux.pio3.abcdsr2) & ~mask, &pio->mux.pio3.abcdsr2);
164}
165
166static void at91_mux_pio3_set_C_periph(struct at91_port *pio, u32 mask)
167{
168 writel(readl(&pio->mux.pio3.abcdsr1) & ~mask, &pio->mux.pio3.abcdsr1);
169 writel(readl(&pio->mux.pio3.abcdsr2) | mask, &pio->mux.pio3.abcdsr2);
170}
171
172static void at91_mux_pio3_set_D_periph(struct at91_port *pio, u32 mask)
173{
174 writel(readl(&pio->mux.pio3.abcdsr1) | mask, &pio->mux.pio3.abcdsr1);
175 writel(readl(&pio->mux.pio3.abcdsr2) | mask, &pio->mux.pio3.abcdsr2);
176}
177
178static void at91_mux_set_deglitch(struct at91_port *pio, u32 mask, bool is_on)
179{
180 writel(mask, (is_on ? &pio->ifer : &pio->ifdr));
181}
182
183static void at91_mux_pio3_set_deglitch(struct at91_port *pio,
184 u32 mask, bool is_on)
185{
186 if (is_on)
187 writel(mask, &pio->mux.pio3.ifscdr);
188 at91_mux_set_deglitch(pio, mask, is_on);
189}
190
191static void at91_mux_pio3_set_debounce(struct at91_port *pio, u32 mask,
192 bool is_on, u32 div)
193{
194 if (is_on) {
195 writel(mask, &pio->mux.pio3.ifscer);
196 writel(div & PIO_SCDR_DIV, &pio->mux.pio3.scdr);
197 writel(mask, &pio->ifer);
198 } else {
199 writel(mask, &pio->mux.pio3.ifscdr);
200 }
201}
202
203static void at91_mux_pio3_set_pulldown(struct at91_port *pio,
204 u32 mask, bool is_on)
205{
206 if (is_on)
207 writel(mask, &pio->pudr);
208
209 writel(mask, (is_on ? &pio->mux.pio3.ppder : &pio->mux.pio3.ppddr));
210}
211
212static void at91_mux_pio3_disable_schmitt_trig(struct at91_port *pio,
213 u32 mask)
214{
215 writel(readl(&pio->schmitt) | mask, &pio->schmitt);
216}
217
218static void set_drive_strength(void *reg, u32 pin, u32 strength)
219{
220 u32 shift = two_bit_pin_value_shift_amount(pin);
221
222 clrsetbits_le32(reg, DRIVE_STRENGTH_MASK << shift, strength << shift);
223}
224
225static void at91_mux_sama5d3_set_drivestrength(struct at91_port *pio,
226 u32 pin, u32 setting)
227{
228 void *reg;
229
230 reg = &pio->driver12;
231 if (pin >= MAX_NB_GPIO_PER_BANK / 2)
232 reg = &pio->driver2;
233
234 /* do nothing if setting is zero */
235 if (!setting)
236 return;
237
238 /* strength is 1 to 1 with setting for SAMA5 */
239 set_drive_strength(reg, pin, setting);
240}
241
242static void at91_mux_sam9x5_set_drivestrength(struct at91_port *pio,
243 u32 pin, u32 setting)
244{
245 void *reg;
246
247 reg = &pio->driver1;
248 if (pin >= MAX_NB_GPIO_PER_BANK / 2)
249 reg = &pio->driver12;
250
251 /* do nothing if setting is zero */
252 if (!setting)
253 return;
254
255 /* strength is inverse on SAM9x5s with our defines
256 * 0 = hi, 1 = med, 2 = low, 3 = rsvd */
Claudiu Beznea04d4ec92019-03-25 10:33:56 +0000257 setting = DRIVE_STRENGTH_BIT_MSK(HI) - setting;
Wenyou Yang9319a752017-03-23 12:44:37 +0800258
259 set_drive_strength(reg, pin, setting);
260}
261
Claudiu Beznea1a6a8282019-03-25 10:33:57 +0000262static void at91_mux_sam9x60_set_drivestrength(struct at91_port *pio, u32 pin,
263 u32 setting)
264{
265 void *reg = &pio->driver12;
266 u32 tmp;
267
268 if (setting <= DRIVE_STRENGTH_BIT_DEF ||
269 setting == DRIVE_STRENGTH_BIT_MED ||
270 setting > DRIVE_STRENGTH_BIT_HI)
271 return;
272
273 tmp = readl(reg);
274
275 /* Strength is 0: low, 1: hi */
276 if (setting == DRIVE_STRENGTH_BIT_LOW)
277 tmp &= ~BIT(pin);
278 else
279 tmp |= BIT(pin);
280
281 writel(tmp, reg);
282}
283
Claudiu Beznea068d4c0a2019-03-25 10:34:00 +0000284static void at91_mux_sam9x60_set_slewrate(struct at91_port *pio, u32 pin,
285 u32 setting)
286{
287 void *reg = &pio->reserved12[3];
288 u32 tmp;
289
290 if (setting < SLEWRATE_BIT_DIS || setting > SLEWRATE_BIT_ENA)
291 return;
292
293 tmp = readl(reg);
294
295 if (setting == SLEWRATE_BIT_DIS)
296 tmp &= ~BIT(pin);
297 else
298 tmp |= BIT(pin);
299
300 writel(tmp, reg);
301}
302
Wenyou Yang9319a752017-03-23 12:44:37 +0800303static struct at91_pinctrl_mux_ops at91rm9200_ops = {
304 .mux_A_periph = at91_mux_set_A_periph,
305 .mux_B_periph = at91_mux_set_B_periph,
306 .set_deglitch = at91_mux_set_deglitch,
307};
308
309static struct at91_pinctrl_mux_ops at91sam9x5_ops = {
310 .mux_A_periph = at91_mux_pio3_set_A_periph,
311 .mux_B_periph = at91_mux_pio3_set_B_periph,
312 .mux_C_periph = at91_mux_pio3_set_C_periph,
313 .mux_D_periph = at91_mux_pio3_set_D_periph,
314 .set_deglitch = at91_mux_pio3_set_deglitch,
315 .set_debounce = at91_mux_pio3_set_debounce,
316 .set_pulldown = at91_mux_pio3_set_pulldown,
317 .disable_schmitt_trig = at91_mux_pio3_disable_schmitt_trig,
318 .set_drivestrength = at91_mux_sam9x5_set_drivestrength,
319};
320
321static struct at91_pinctrl_mux_ops sama5d3_ops = {
322 .mux_A_periph = at91_mux_pio3_set_A_periph,
323 .mux_B_periph = at91_mux_pio3_set_B_periph,
324 .mux_C_periph = at91_mux_pio3_set_C_periph,
325 .mux_D_periph = at91_mux_pio3_set_D_periph,
326 .set_deglitch = at91_mux_pio3_set_deglitch,
327 .set_debounce = at91_mux_pio3_set_debounce,
328 .set_pulldown = at91_mux_pio3_set_pulldown,
329 .disable_schmitt_trig = at91_mux_pio3_disable_schmitt_trig,
330 .set_drivestrength = at91_mux_sama5d3_set_drivestrength,
331};
332
Claudiu Beznea1a6a8282019-03-25 10:33:57 +0000333static struct at91_pinctrl_mux_ops sam9x60_ops = {
334 .mux_A_periph = at91_mux_pio3_set_A_periph,
335 .mux_B_periph = at91_mux_pio3_set_B_periph,
336 .mux_C_periph = at91_mux_pio3_set_C_periph,
337 .mux_D_periph = at91_mux_pio3_set_D_periph,
338 .set_deglitch = at91_mux_pio3_set_deglitch,
339 .set_debounce = at91_mux_pio3_set_debounce,
340 .set_pulldown = at91_mux_pio3_set_pulldown,
341 .disable_schmitt_trig = at91_mux_pio3_disable_schmitt_trig,
342 .set_drivestrength = at91_mux_sam9x60_set_drivestrength,
Claudiu Beznea068d4c0a2019-03-25 10:34:00 +0000343 .set_slewrate = at91_mux_sam9x60_set_slewrate,
Claudiu Beznea1a6a8282019-03-25 10:33:57 +0000344};
345
Wenyou Yang9319a752017-03-23 12:44:37 +0800346static void at91_mux_gpio_disable(struct at91_port *pio, u32 mask)
347{
348 writel(mask, &pio->pdr);
349}
350
351static void at91_mux_gpio_enable(struct at91_port *pio, u32 mask, bool input)
352{
353 writel(mask, &pio->per);
354 writel(mask, (input ? &pio->odr : &pio->oer));
355}
356
357static int at91_pmx_set(struct at91_pinctrl_mux_ops *ops,
358 struct at91_port *pio, u32 mask, enum at91_mux mux)
359{
360 at91_mux_disable_interrupt(pio, mask);
361 switch (mux) {
362 case AT91_MUX_GPIO:
363 at91_mux_gpio_enable(pio, mask, 1);
364 break;
365 case AT91_MUX_PERIPH_A:
366 ops->mux_A_periph(pio, mask);
367 break;
368 case AT91_MUX_PERIPH_B:
369 ops->mux_B_periph(pio, mask);
370 break;
371 case AT91_MUX_PERIPH_C:
372 if (!ops->mux_C_periph)
373 return -EINVAL;
374 ops->mux_C_periph(pio, mask);
375 break;
376 case AT91_MUX_PERIPH_D:
377 if (!ops->mux_D_periph)
378 return -EINVAL;
379 ops->mux_D_periph(pio, mask);
380 break;
381 }
382 if (mux)
383 at91_mux_gpio_disable(pio, mask);
384
385 return 0;
386}
387
388static int at91_pinconf_set(struct at91_pinctrl_mux_ops *ops,
389 struct at91_port *pio, u32 pin, u32 config)
390{
391 u32 mask = BIT(pin);
392
393 if ((config & PULL_UP) && (config & PULL_DOWN))
394 return -EINVAL;
395
396 at91_mux_set_output(pio, mask, config & OUTPUT,
397 (config & OUTPUT_VAL) >> OUTPUT_VAL_SHIFT);
398 at91_mux_set_pullup(pio, mask, config & PULL_UP);
399 at91_mux_set_multidrive(pio, mask, config & MULTI_DRIVE);
400 if (ops->set_deglitch)
401 ops->set_deglitch(pio, mask, config & DEGLITCH);
402 if (ops->set_debounce)
403 ops->set_debounce(pio, mask, config & DEBOUNCE,
404 (config & DEBOUNCE_VAL) >> DEBOUNCE_VAL_SHIFT);
405 if (ops->set_pulldown)
406 ops->set_pulldown(pio, mask, config & PULL_DOWN);
407 if (ops->disable_schmitt_trig && config & DIS_SCHMIT)
408 ops->disable_schmitt_trig(pio, mask);
409 if (ops->set_drivestrength)
410 ops->set_drivestrength(pio, pin,
411 (config & DRIVE_STRENGTH) >> DRIVE_STRENGTH_SHIFT);
Claudiu Beznea068d4c0a2019-03-25 10:34:00 +0000412 if (ops->set_slewrate)
413 ops->set_slewrate(pio, pin,
414 (config & SLEWRATE) >> SLEWRATE_SHIFT);
Wenyou Yang9319a752017-03-23 12:44:37 +0800415
416 return 0;
417}
418
419static int at91_pin_check_config(struct udevice *dev, u32 bank, u32 pin)
420{
421 struct at91_pinctrl_priv *priv = dev_get_priv(dev);
422
423 if (bank >= priv->nbanks) {
424 debug("pin conf bank %d >= nbanks %d\n", bank, priv->nbanks);
425 return -EINVAL;
426 }
427
428 if (pin >= MAX_NB_GPIO_PER_BANK) {
429 debug("pin conf pin %d >= %d\n", pin, MAX_NB_GPIO_PER_BANK);
430 return -EINVAL;
431 }
432
433 return 0;
434}
435
436static int at91_pinctrl_set_state(struct udevice *dev, struct udevice *config)
437{
438 struct at91_pinctrl_priv *priv = dev_get_priv(dev);
439 const void *blob = gd->fdt_blob;
Simon Glassda409cc2017-05-17 17:18:09 -0600440 int node = dev_of_offset(config);
Wenyou Yang9319a752017-03-23 12:44:37 +0800441 u32 cells[MAX_PINMUX_ENTRIES];
442 const u32 *list = cells;
443 u32 bank, pin;
444 u32 conf, mask, count, i;
445 int size;
446 int ret;
447 enum at91_mux mux;
448 struct at91_port *pio;
449 struct at91_pinctrl_mux_ops *ops =
450 (struct at91_pinctrl_mux_ops *)dev_get_driver_data(dev);
451
452 /*
453 * the binding format is atmel,pins = <bank pin mux CONFIG ...>,
454 * do sanity check and calculate pins number
455 */
456 size = fdtdec_get_int_array_count(blob, node, "atmel,pins",
457 cells, ARRAY_SIZE(cells));
458
459 /* we do not check return since it's safe node passed down */
460 count = size >> 2;
461 if (!count)
462 return -EINVAL;
463
464 for (i = 0; i < count; i++) {
465 bank = *list++;
466 pin = *list++;
467 mux = *list++;
468 conf = *list++;
469
470 ret = at91_pin_check_config(dev, bank, pin);
471 if (ret)
472 return ret;
473
474 pio = priv->reg_base[bank];
475 mask = BIT(pin);
476
477 ret = at91_pmx_set(ops, pio, mask, mux);
478 if (ret)
479 return ret;
480
481 ret = at91_pinconf_set(ops, pio, pin, conf);
482 if (ret)
483 return ret;
484 }
485
486 return 0;
487}
488
489const struct pinctrl_ops at91_pinctrl_ops = {
490 .set_state = at91_pinctrl_set_state,
491};
492
493static int at91_pinctrl_probe(struct udevice *dev)
494{
495 struct at91_pinctrl_priv *priv = dev_get_priv(dev);
496 fdt_addr_t addr_base;
497 int index;
498
499 for (index = 0; index < MAX_GPIO_BANKS; index++) {
Simon Glassa821c4a2017-05-17 17:18:05 -0600500 addr_base = devfdt_get_addr_index(dev, index);
Wenyou Yang9319a752017-03-23 12:44:37 +0800501 if (addr_base == FDT_ADDR_T_NONE)
502 break;
503
504 priv->reg_base[index] = (struct at91_port *)addr_base;
505 }
506
507 priv->nbanks = index;
508
509 return 0;
510}
511
512static const struct udevice_id at91_pinctrl_match[] = {
513 { .compatible = "atmel,sama5d3-pinctrl", .data = (ulong)&sama5d3_ops },
514 { .compatible = "atmel,at91sam9x5-pinctrl", .data = (ulong)&at91sam9x5_ops },
515 { .compatible = "atmel,at91rm9200-pinctrl", .data = (ulong)&at91rm9200_ops },
Claudiu Bezneabe6e2402019-03-25 10:33:59 +0000516 { .compatible = "microchip,sam9x60-pinctrl", .data = (ulong)&sam9x60_ops },
Wenyou Yang9319a752017-03-23 12:44:37 +0800517 {}
518};
519
520U_BOOT_DRIVER(at91_pinctrl) = {
521 .name = "pinctrl_at91",
522 .id = UCLASS_PINCTRL,
523 .of_match = at91_pinctrl_match,
524 .probe = at91_pinctrl_probe,
525 .priv_auto_alloc_size = sizeof(struct at91_pinctrl_priv),
526 .ops = &at91_pinctrl_ops,
527};