blob: ee45aaf038afcfd73740d02a72cac3e33e7058ed [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Jens Scharsig8d065462010-02-03 22:46:16 +01002/*
Bo Shenf0adeaa2013-08-13 14:38:32 +08003 * Copyright (C) 2013 Bo Shen <voice.shen@atmel.com>
Jens Scharsig8d065462010-02-03 22:46:16 +01004 *
5 * Copyright (C) 2009 Jens Scharsig (js_at_ng@scharsoft.de)
6 *
7 * Copyright (C) 2005 HP Labs
Jens Scharsig8d065462010-02-03 22:46:16 +01008 */
9
10#include <config.h>
Wenyou Yange29b82b2017-03-23 12:46:21 +080011#include <clk.h>
Simon Glass4d717882014-10-29 13:08:57 -060012#include <dm.h>
Simon Glass9bc15642020-02-03 07:36:16 -070013#include <malloc.h>
Reinhard Meyerb06208c2010-11-07 13:26:14 +010014#include <asm/io.h>
Alexey Brodkin267d8e22014-02-26 17:47:58 +040015#include <linux/sizes.h>
Simon Glass4d717882014-10-29 13:08:57 -060016#include <asm/gpio.h>
Jens Scharsig8d065462010-02-03 22:46:16 +010017#include <asm/arch/hardware.h>
Jens Scharsig8d065462010-02-03 22:46:16 +010018#include <asm/arch/at91_pio.h>
Simon Glass4d717882014-10-29 13:08:57 -060019
20#define GPIO_PER_BANK 32
Jens Scharsig8d065462010-02-03 22:46:16 +010021
Bo Shen02d88142013-08-22 15:24:40 +080022static struct at91_port *at91_pio_get_port(unsigned port)
23{
24 switch (port) {
25 case AT91_PIO_PORTA:
26 return (struct at91_port *)ATMEL_BASE_PIOA;
27 case AT91_PIO_PORTB:
28 return (struct at91_port *)ATMEL_BASE_PIOB;
29 case AT91_PIO_PORTC:
30 return (struct at91_port *)ATMEL_BASE_PIOC;
31#if (ATMEL_PIO_PORTS > 3)
32 case AT91_PIO_PORTD:
33 return (struct at91_port *)ATMEL_BASE_PIOD;
34#if (ATMEL_PIO_PORTS > 4)
35 case AT91_PIO_PORTE:
36 return (struct at91_port *)ATMEL_BASE_PIOE;
37#endif
38#endif
39 default:
Wu, Josh383543a2014-05-07 16:50:45 +080040 printf("Error: at91_gpio: Fail to get PIO base!\n");
Bo Shen02d88142013-08-22 15:24:40 +080041 return NULL;
42 }
43}
44
Simon Glass4d717882014-10-29 13:08:57 -060045static void at91_set_port_pullup(struct at91_port *at91_port, unsigned offset,
46 int use_pullup)
47{
48 u32 mask;
49
50 mask = 1 << offset;
51 if (use_pullup)
52 writel(mask, &at91_port->puer);
53 else
54 writel(mask, &at91_port->pudr);
55 writel(mask, &at91_port->per);
56}
57
Jens Scharsig8d065462010-02-03 22:46:16 +010058int at91_set_pio_pullup(unsigned port, unsigned pin, int use_pullup)
59{
Bo Shen02d88142013-08-22 15:24:40 +080060 struct at91_port *at91_port = at91_pio_get_port(port);
Jens Scharsig8d065462010-02-03 22:46:16 +010061
Simon Glass4d717882014-10-29 13:08:57 -060062 if (at91_port && (pin < GPIO_PER_BANK))
63 at91_set_port_pullup(at91_port, pin, use_pullup);
Bo Shen02d88142013-08-22 15:24:40 +080064
Jens Scharsig8d065462010-02-03 22:46:16 +010065 return 0;
66}
67
68/*
69 * mux the pin to the "GPIO" peripheral role.
70 */
71int at91_set_pio_periph(unsigned port, unsigned pin, int use_pullup)
72{
Bo Shen02d88142013-08-22 15:24:40 +080073 struct at91_port *at91_port = at91_pio_get_port(port);
74 u32 mask;
Jens Scharsig8d065462010-02-03 22:46:16 +010075
Simon Glass4d717882014-10-29 13:08:57 -060076 if (at91_port && (pin < GPIO_PER_BANK)) {
Jens Scharsig8d065462010-02-03 22:46:16 +010077 mask = 1 << pin;
Bo Shen02d88142013-08-22 15:24:40 +080078 writel(mask, &at91_port->idr);
Jens Scharsig8d065462010-02-03 22:46:16 +010079 at91_set_pio_pullup(port, pin, use_pullup);
Bo Shen02d88142013-08-22 15:24:40 +080080 writel(mask, &at91_port->per);
Jens Scharsig8d065462010-02-03 22:46:16 +010081 }
Bo Shen02d88142013-08-22 15:24:40 +080082
Jens Scharsig8d065462010-02-03 22:46:16 +010083 return 0;
84}
85
86/*
87 * mux the pin to the "A" internal peripheral role.
88 */
89int at91_set_a_periph(unsigned port, unsigned pin, int use_pullup)
90{
Bo Shen02d88142013-08-22 15:24:40 +080091 struct at91_port *at91_port = at91_pio_get_port(port);
92 u32 mask;
Jens Scharsig8d065462010-02-03 22:46:16 +010093
Simon Glass4d717882014-10-29 13:08:57 -060094 if (at91_port && (pin < GPIO_PER_BANK)) {
Jens Scharsig8d065462010-02-03 22:46:16 +010095 mask = 1 << pin;
Bo Shen02d88142013-08-22 15:24:40 +080096 writel(mask, &at91_port->idr);
Jens Scharsig8d065462010-02-03 22:46:16 +010097 at91_set_pio_pullup(port, pin, use_pullup);
Wenyou Yang4a92a3e2017-03-23 12:44:36 +080098 writel(mask, &at91_port->mux.pio2.asr);
Bo Shen02d88142013-08-22 15:24:40 +080099 writel(mask, &at91_port->pdr);
Jens Scharsig8d065462010-02-03 22:46:16 +0100100 }
Bo Shen02d88142013-08-22 15:24:40 +0800101
Jens Scharsig8d065462010-02-03 22:46:16 +0100102 return 0;
103}
104
105/*
106 * mux the pin to the "B" internal peripheral role.
107 */
108int at91_set_b_periph(unsigned port, unsigned pin, int use_pullup)
109{
Bo Shen02d88142013-08-22 15:24:40 +0800110 struct at91_port *at91_port = at91_pio_get_port(port);
111 u32 mask;
Jens Scharsig8d065462010-02-03 22:46:16 +0100112
Simon Glass4d717882014-10-29 13:08:57 -0600113 if (at91_port && (pin < GPIO_PER_BANK)) {
Jens Scharsig8d065462010-02-03 22:46:16 +0100114 mask = 1 << pin;
Bo Shen02d88142013-08-22 15:24:40 +0800115 writel(mask, &at91_port->idr);
Jens Scharsig8d065462010-02-03 22:46:16 +0100116 at91_set_pio_pullup(port, pin, use_pullup);
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800117 writel(mask, &at91_port->mux.pio2.bsr);
Bo Shen02d88142013-08-22 15:24:40 +0800118 writel(mask, &at91_port->pdr);
Jens Scharsig8d065462010-02-03 22:46:16 +0100119 }
Bo Shen02d88142013-08-22 15:24:40 +0800120
Jens Scharsig8d065462010-02-03 22:46:16 +0100121 return 0;
122}
123
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800124/*
125 * mux the pin to the "A" internal peripheral role.
126 */
127int at91_pio3_set_a_periph(unsigned port, unsigned pin, int use_pullup)
128{
129 struct at91_port *at91_port = at91_pio_get_port(port);
130 u32 mask;
131
132 if (at91_port && (pin < GPIO_PER_BANK)) {
133 mask = 1 << pin;
134 writel(mask, &at91_port->idr);
135 at91_set_pio_pullup(port, pin, use_pullup);
136 writel(readl(&at91_port->mux.pio3.abcdsr1) & ~mask,
137 &at91_port->mux.pio3.abcdsr1);
138 writel(readl(&at91_port->mux.pio3.abcdsr2) & ~mask,
139 &at91_port->mux.pio3.abcdsr2);
140
141 writel(mask, &at91_port->pdr);
142 }
143
144 return 0;
145}
146
147/*
148 * mux the pin to the "B" internal peripheral role.
149 */
150int at91_pio3_set_b_periph(unsigned port, unsigned pin, int use_pullup)
151{
152 struct at91_port *at91_port = at91_pio_get_port(port);
153 u32 mask;
154
155 if (at91_port && (pin < GPIO_PER_BANK)) {
156 mask = 1 << pin;
157 writel(mask, &at91_port->idr);
158 at91_set_pio_pullup(port, pin, use_pullup);
159 writel(readl(&at91_port->mux.pio3.abcdsr1) | mask,
160 &at91_port->mux.pio3.abcdsr1);
161 writel(readl(&at91_port->mux.pio3.abcdsr2) & ~mask,
162 &at91_port->mux.pio3.abcdsr2);
163
164 writel(mask, &at91_port->pdr);
165 }
166
167 return 0;
168}
Bo Shen0ac13452012-05-20 15:50:00 +0000169/*
170 * mux the pin to the "C" internal peripheral role.
171 */
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800172int at91_pio3_set_c_periph(unsigned port, unsigned pin, int use_pullup)
Bo Shen0ac13452012-05-20 15:50:00 +0000173{
Bo Shen02d88142013-08-22 15:24:40 +0800174 struct at91_port *at91_port = at91_pio_get_port(port);
175 u32 mask;
Bo Shen0ac13452012-05-20 15:50:00 +0000176
Simon Glass4d717882014-10-29 13:08:57 -0600177 if (at91_port && (pin < GPIO_PER_BANK)) {
Bo Shen0ac13452012-05-20 15:50:00 +0000178 mask = 1 << pin;
Bo Shen02d88142013-08-22 15:24:40 +0800179 writel(mask, &at91_port->idr);
Bo Shen0ac13452012-05-20 15:50:00 +0000180 at91_set_pio_pullup(port, pin, use_pullup);
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800181 writel(readl(&at91_port->mux.pio3.abcdsr1) & ~mask,
182 &at91_port->mux.pio3.abcdsr1);
183 writel(readl(&at91_port->mux.pio3.abcdsr2) | mask,
184 &at91_port->mux.pio3.abcdsr2);
Bo Shen02d88142013-08-22 15:24:40 +0800185 writel(mask, &at91_port->pdr);
Bo Shen0ac13452012-05-20 15:50:00 +0000186 }
Bo Shen02d88142013-08-22 15:24:40 +0800187
Bo Shen0ac13452012-05-20 15:50:00 +0000188 return 0;
189}
190
191/*
192 * mux the pin to the "D" internal peripheral role.
193 */
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800194int at91_pio3_set_d_periph(unsigned port, unsigned pin, int use_pullup)
Bo Shen0ac13452012-05-20 15:50:00 +0000195{
Bo Shen02d88142013-08-22 15:24:40 +0800196 struct at91_port *at91_port = at91_pio_get_port(port);
197 u32 mask;
Bo Shen0ac13452012-05-20 15:50:00 +0000198
Simon Glass4d717882014-10-29 13:08:57 -0600199 if (at91_port && (pin < GPIO_PER_BANK)) {
Bo Shen0ac13452012-05-20 15:50:00 +0000200 mask = 1 << pin;
Bo Shen02d88142013-08-22 15:24:40 +0800201 writel(mask, &at91_port->idr);
Bo Shen0ac13452012-05-20 15:50:00 +0000202 at91_set_pio_pullup(port, pin, use_pullup);
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800203 writel(readl(&at91_port->mux.pio3.abcdsr1) | mask,
204 &at91_port->mux.pio3.abcdsr1);
205 writel(readl(&at91_port->mux.pio3.abcdsr2) | mask,
206 &at91_port->mux.pio3.abcdsr2);
Bo Shen02d88142013-08-22 15:24:40 +0800207 writel(mask, &at91_port->pdr);
Bo Shen0ac13452012-05-20 15:50:00 +0000208 }
Bo Shen02d88142013-08-22 15:24:40 +0800209
Bo Shen0ac13452012-05-20 15:50:00 +0000210 return 0;
211}
Bo Shen0ac13452012-05-20 15:50:00 +0000212
Simon Glassfa4689a2019-12-06 21:41:35 -0700213#if CONFIG_IS_ENABLED(DM_GPIO)
Simon Glass4d717882014-10-29 13:08:57 -0600214static bool at91_get_port_output(struct at91_port *at91_port, int offset)
215{
216 u32 mask, val;
217
218 mask = 1 << offset;
219 val = readl(&at91_port->osr);
220 return val & mask;
221}
Zixun LI6e399502024-11-13 11:10:22 +0100222
223static bool at91_is_port_gpio(struct at91_port *at91_port, int offset)
224{
225 u32 mask, val;
226
227 mask = 1 << offset;
228 val = readl(&at91_port->psr);
229 return !!(val & mask);
230}
Simon Glass4d717882014-10-29 13:08:57 -0600231#endif
232
233static void at91_set_port_input(struct at91_port *at91_port, int offset,
234 int use_pullup)
235{
236 u32 mask;
237
238 mask = 1 << offset;
239 writel(mask, &at91_port->idr);
240 at91_set_port_pullup(at91_port, offset, use_pullup);
241 writel(mask, &at91_port->odr);
242 writel(mask, &at91_port->per);
243}
244
Jens Scharsig8d065462010-02-03 22:46:16 +0100245/*
246 * mux the pin to the gpio controller (instead of "A" or "B" peripheral), and
247 * configure it for an input.
248 */
249int at91_set_pio_input(unsigned port, u32 pin, int use_pullup)
250{
Bo Shen02d88142013-08-22 15:24:40 +0800251 struct at91_port *at91_port = at91_pio_get_port(port);
Jens Scharsig8d065462010-02-03 22:46:16 +0100252
Simon Glass4d717882014-10-29 13:08:57 -0600253 if (at91_port && (pin < GPIO_PER_BANK))
254 at91_set_port_input(at91_port, pin, use_pullup);
Bo Shen02d88142013-08-22 15:24:40 +0800255
Jens Scharsig8d065462010-02-03 22:46:16 +0100256 return 0;
257}
258
Simon Glass4d717882014-10-29 13:08:57 -0600259static void at91_set_port_output(struct at91_port *at91_port, int offset,
260 int value)
261{
262 u32 mask;
263
264 mask = 1 << offset;
265 writel(mask, &at91_port->idr);
266 writel(mask, &at91_port->pudr);
267 if (value)
268 writel(mask, &at91_port->sodr);
269 else
270 writel(mask, &at91_port->codr);
271 writel(mask, &at91_port->oer);
272 writel(mask, &at91_port->per);
273}
274
Jens Scharsig8d065462010-02-03 22:46:16 +0100275/*
276 * mux the pin to the gpio controller (instead of "A" or "B" peripheral),
277 * and configure it for an output.
278 */
279int at91_set_pio_output(unsigned port, u32 pin, int value)
280{
Bo Shen02d88142013-08-22 15:24:40 +0800281 struct at91_port *at91_port = at91_pio_get_port(port);
Jens Scharsig8d065462010-02-03 22:46:16 +0100282
Simon Glass4d717882014-10-29 13:08:57 -0600283 if (at91_port && (pin < GPIO_PER_BANK))
284 at91_set_port_output(at91_port, pin, value);
Bo Shen02d88142013-08-22 15:24:40 +0800285
Jens Scharsig8d065462010-02-03 22:46:16 +0100286 return 0;
287}
288
289/*
290 * enable/disable the glitch filter. mostly used with IRQ handling.
291 */
292int at91_set_pio_deglitch(unsigned port, unsigned pin, int is_on)
293{
Bo Shen02d88142013-08-22 15:24:40 +0800294 struct at91_port *at91_port = at91_pio_get_port(port);
295 u32 mask;
Jens Scharsig8d065462010-02-03 22:46:16 +0100296
Simon Glass4d717882014-10-29 13:08:57 -0600297 if (at91_port && (pin < GPIO_PER_BANK)) {
Jens Scharsig8d065462010-02-03 22:46:16 +0100298 mask = 1 << pin;
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800299 if (is_on)
300 writel(mask, &at91_port->ifer);
301 else
302 writel(mask, &at91_port->ifdr);
303 }
304
305 return 0;
306}
307
308/*
309 * enable/disable the glitch filter. mostly used with IRQ handling.
310 */
311int at91_pio3_set_pio_deglitch(unsigned port, unsigned pin, int is_on)
312{
313 struct at91_port *at91_port = at91_pio_get_port(port);
314 u32 mask;
315
316 if (at91_port && (pin < GPIO_PER_BANK)) {
317 mask = 1 << pin;
Bo Shen0ac13452012-05-20 15:50:00 +0000318 if (is_on) {
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800319 writel(mask, &at91_port->mux.pio3.ifscdr);
Bo Shen02d88142013-08-22 15:24:40 +0800320 writel(mask, &at91_port->ifer);
Bo Shen0ac13452012-05-20 15:50:00 +0000321 } else {
Bo Shen02d88142013-08-22 15:24:40 +0800322 writel(mask, &at91_port->ifdr);
Bo Shen0ac13452012-05-20 15:50:00 +0000323 }
324 }
Bo Shen02d88142013-08-22 15:24:40 +0800325
Bo Shen0ac13452012-05-20 15:50:00 +0000326 return 0;
327}
328
Bo Shen0ac13452012-05-20 15:50:00 +0000329/*
330 * enable/disable the debounce filter.
331 */
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800332int at91_pio3_set_pio_debounce(unsigned port, unsigned pin, int is_on, int div)
Bo Shen0ac13452012-05-20 15:50:00 +0000333{
Bo Shen02d88142013-08-22 15:24:40 +0800334 struct at91_port *at91_port = at91_pio_get_port(port);
335 u32 mask;
Bo Shen0ac13452012-05-20 15:50:00 +0000336
Simon Glass4d717882014-10-29 13:08:57 -0600337 if (at91_port && (pin < GPIO_PER_BANK)) {
Bo Shen0ac13452012-05-20 15:50:00 +0000338 mask = 1 << pin;
339 if (is_on) {
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800340 writel(mask, &at91_port->mux.pio3.ifscer);
341 writel(div & PIO_SCDR_DIV, &at91_port->mux.pio3.scdr);
Bo Shen02d88142013-08-22 15:24:40 +0800342 writel(mask, &at91_port->ifer);
Bo Shen0ac13452012-05-20 15:50:00 +0000343 } else {
Bo Shen02d88142013-08-22 15:24:40 +0800344 writel(mask, &at91_port->ifdr);
Bo Shen0ac13452012-05-20 15:50:00 +0000345 }
346 }
Bo Shen02d88142013-08-22 15:24:40 +0800347
Bo Shen0ac13452012-05-20 15:50:00 +0000348 return 0;
349}
350
351/*
352 * enable/disable the pull-down.
353 * If pull-up already enabled while calling the function, we disable it.
354 */
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800355int at91_pio3_set_pio_pulldown(unsigned port, unsigned pin, int is_on)
Bo Shen0ac13452012-05-20 15:50:00 +0000356{
Bo Shen02d88142013-08-22 15:24:40 +0800357 struct at91_port *at91_port = at91_pio_get_port(port);
358 u32 mask;
Bo Shen0ac13452012-05-20 15:50:00 +0000359
Simon Glass4d717882014-10-29 13:08:57 -0600360 if (at91_port && (pin < GPIO_PER_BANK)) {
Bo Shen0ac13452012-05-20 15:50:00 +0000361 mask = 1 << pin;
Marek Vasutba91bd52016-05-04 23:05:23 +0200362 if (is_on) {
363 at91_set_pio_pullup(port, pin, 0);
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800364 writel(mask, &at91_port->mux.pio3.ppder);
Marek Vasutba91bd52016-05-04 23:05:23 +0200365 } else
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800366 writel(mask, &at91_port->mux.pio3.ppddr);
Bo Shen0ac13452012-05-20 15:50:00 +0000367 }
Bo Shen02d88142013-08-22 15:24:40 +0800368
Bo Shen0ac13452012-05-20 15:50:00 +0000369 return 0;
370}
371
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800372int at91_pio3_set_pio_pullup(unsigned port, unsigned pin, int use_pullup)
373{
374 struct at91_port *at91_port = at91_pio_get_port(port);
375
376 if (use_pullup)
377 at91_pio3_set_pio_pulldown(port, pin, 0);
378
379 if (at91_port && (pin < GPIO_PER_BANK))
380 at91_set_port_pullup(at91_port, pin, use_pullup);
381
382 return 0;
383}
384
Bo Shen0ac13452012-05-20 15:50:00 +0000385/*
386 * disable Schmitt trigger
387 */
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800388int at91_pio3_set_pio_disable_schmitt_trig(unsigned port, unsigned pin)
Bo Shen0ac13452012-05-20 15:50:00 +0000389{
Bo Shen02d88142013-08-22 15:24:40 +0800390 struct at91_port *at91_port = at91_pio_get_port(port);
391 u32 mask;
Bo Shen0ac13452012-05-20 15:50:00 +0000392
Simon Glass4d717882014-10-29 13:08:57 -0600393 if (at91_port && (pin < GPIO_PER_BANK)) {
Bo Shen0ac13452012-05-20 15:50:00 +0000394 mask = 1 << pin;
Bo Shen02d88142013-08-22 15:24:40 +0800395 writel(readl(&at91_port->schmitt) | mask,
396 &at91_port->schmitt);
Jens Scharsig8d065462010-02-03 22:46:16 +0100397 }
Bo Shen02d88142013-08-22 15:24:40 +0800398
Jens Scharsig8d065462010-02-03 22:46:16 +0100399 return 0;
400}
401
402/*
403 * enable/disable the multi-driver. This is only valid for output and
404 * allows the output pin to run as an open collector output.
405 */
406int at91_set_pio_multi_drive(unsigned port, unsigned pin, int is_on)
407{
Bo Shen02d88142013-08-22 15:24:40 +0800408 struct at91_port *at91_port = at91_pio_get_port(port);
409 u32 mask;
Jens Scharsig8d065462010-02-03 22:46:16 +0100410
Simon Glass4d717882014-10-29 13:08:57 -0600411 if (at91_port && (pin < GPIO_PER_BANK)) {
Jens Scharsig8d065462010-02-03 22:46:16 +0100412 mask = 1 << pin;
413 if (is_on)
Bo Shen02d88142013-08-22 15:24:40 +0800414 writel(mask, &at91_port->mder);
Jens Scharsig8d065462010-02-03 22:46:16 +0100415 else
Bo Shen02d88142013-08-22 15:24:40 +0800416 writel(mask, &at91_port->mddr);
Jens Scharsig8d065462010-02-03 22:46:16 +0100417 }
Bo Shen02d88142013-08-22 15:24:40 +0800418
Jens Scharsig8d065462010-02-03 22:46:16 +0100419 return 0;
420}
421
Simon Glass4d717882014-10-29 13:08:57 -0600422static void at91_set_port_value(struct at91_port *at91_port, int offset,
423 int value)
424{
425 u32 mask;
426
427 mask = 1 << offset;
428 if (value)
429 writel(mask, &at91_port->sodr);
430 else
431 writel(mask, &at91_port->codr);
432}
433
Jens Scharsig8d065462010-02-03 22:46:16 +0100434/*
435 * assuming the pin is muxed as a gpio output, set its value.
436 */
437int at91_set_pio_value(unsigned port, unsigned pin, int value)
438{
Bo Shen02d88142013-08-22 15:24:40 +0800439 struct at91_port *at91_port = at91_pio_get_port(port);
Jens Scharsig8d065462010-02-03 22:46:16 +0100440
Simon Glass4d717882014-10-29 13:08:57 -0600441 if (at91_port && (pin < GPIO_PER_BANK))
442 at91_set_port_value(at91_port, pin, value);
Bo Shen02d88142013-08-22 15:24:40 +0800443
Jens Scharsig8d065462010-02-03 22:46:16 +0100444 return 0;
445}
446
Simon Glass4d717882014-10-29 13:08:57 -0600447static int at91_get_port_value(struct at91_port *at91_port, int offset)
448{
449 u32 pdsr = 0, mask;
450
451 mask = 1 << offset;
452 pdsr = readl(&at91_port->pdsr) & mask;
453
454 return pdsr != 0;
455}
Jens Scharsig8d065462010-02-03 22:46:16 +0100456/*
457 * read the pin's value (works even if it's not muxed as a gpio).
458 */
459int at91_get_pio_value(unsigned port, unsigned pin)
460{
Bo Shen02d88142013-08-22 15:24:40 +0800461 struct at91_port *at91_port = at91_pio_get_port(port);
Jens Scharsig8d065462010-02-03 22:46:16 +0100462
Simon Glass4d717882014-10-29 13:08:57 -0600463 if (at91_port && (pin < GPIO_PER_BANK))
464 return at91_get_port_value(at91_port, pin);
Bo Shen02d88142013-08-22 15:24:40 +0800465
Simon Glass4d717882014-10-29 13:08:57 -0600466 return 0;
Jens Scharsig8d065462010-02-03 22:46:16 +0100467}
Bo Shenad1d2ac2013-08-13 14:38:31 +0800468
Simon Glassfa4689a2019-12-06 21:41:35 -0700469#if !CONFIG_IS_ENABLED(DM_GPIO)
Bo Shenad1d2ac2013-08-13 14:38:31 +0800470/* Common GPIO API */
471
Bo Shenad1d2ac2013-08-13 14:38:31 +0800472int gpio_request(unsigned gpio, const char *label)
473{
474 return 0;
475}
476
477int gpio_free(unsigned gpio)
478{
479 return 0;
480}
481
482int gpio_direction_input(unsigned gpio)
483{
484 at91_set_pio_input(at91_gpio_to_port(gpio),
485 at91_gpio_to_pin(gpio), 0);
486 return 0;
487}
488
489int gpio_direction_output(unsigned gpio, int value)
490{
491 at91_set_pio_output(at91_gpio_to_port(gpio),
492 at91_gpio_to_pin(gpio), value);
493 return 0;
494}
495
496int gpio_get_value(unsigned gpio)
497{
498 return at91_get_pio_value(at91_gpio_to_port(gpio),
499 at91_gpio_to_pin(gpio));
500}
501
502int gpio_set_value(unsigned gpio, int value)
503{
504 at91_set_pio_value(at91_gpio_to_port(gpio),
505 at91_gpio_to_pin(gpio), value);
506
507 return 0;
508}
Simon Glass4d717882014-10-29 13:08:57 -0600509#endif
510
Simon Glassfa4689a2019-12-06 21:41:35 -0700511#if CONFIG_IS_ENABLED(DM_GPIO)
Simon Glass4d717882014-10-29 13:08:57 -0600512
513struct at91_port_priv {
514 struct at91_port *regs;
515};
516
517/* set GPIO pin 'gpio' as an input */
518static int at91_gpio_direction_input(struct udevice *dev, unsigned offset)
519{
Axel Lin98b9b372015-01-31 14:47:34 +0800520 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glass4d717882014-10-29 13:08:57 -0600521
522 at91_set_port_input(port->regs, offset, 0);
523
524 return 0;
525}
526
527/* set GPIO pin 'gpio' as an output, with polarity 'value' */
528static int at91_gpio_direction_output(struct udevice *dev, unsigned offset,
529 int value)
530{
Axel Lin98b9b372015-01-31 14:47:34 +0800531 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glass4d717882014-10-29 13:08:57 -0600532
533 at91_set_port_output(port->regs, offset, value);
534
535 return 0;
536}
537
538/* read GPIO IN value of pin 'gpio' */
539static int at91_gpio_get_value(struct udevice *dev, unsigned offset)
540{
Axel Lin98b9b372015-01-31 14:47:34 +0800541 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glass4d717882014-10-29 13:08:57 -0600542
543 return at91_get_port_value(port->regs, offset);
544}
545
546/* write GPIO OUT value to pin 'gpio' */
547static int at91_gpio_set_value(struct udevice *dev, unsigned offset,
548 int value)
549{
Axel Lin98b9b372015-01-31 14:47:34 +0800550 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glass4d717882014-10-29 13:08:57 -0600551
552 at91_set_port_value(port->regs, offset, value);
553
554 return 0;
555}
556
557static int at91_gpio_get_function(struct udevice *dev, unsigned offset)
558{
Axel Lin98b9b372015-01-31 14:47:34 +0800559 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glass4d717882014-10-29 13:08:57 -0600560
Zixun LI6e399502024-11-13 11:10:22 +0100561 if (!at91_is_port_gpio(port->regs, offset))
562 return GPIOF_FUNC;
563
Simon Glass4d717882014-10-29 13:08:57 -0600564 if (at91_get_port_output(port->regs, offset))
565 return GPIOF_OUTPUT;
566 else
567 return GPIOF_INPUT;
568}
569
James Byrneb7e33d32019-11-26 11:52:04 +0000570static const char *at91_get_bank_name(uint32_t base_addr)
571{
572 switch (base_addr) {
573 case ATMEL_BASE_PIOA:
574 return "PIOA";
575 case ATMEL_BASE_PIOB:
576 return "PIOB";
577 case ATMEL_BASE_PIOC:
578 return "PIOC";
579#if (ATMEL_PIO_PORTS > 3)
580 case ATMEL_BASE_PIOD:
581 return "PIOD";
582#if (ATMEL_PIO_PORTS > 4)
583 case ATMEL_BASE_PIOE:
584 return "PIOE";
585#endif
586#endif
587 }
588
589 return "undefined";
590}
591
Simon Glass4d717882014-10-29 13:08:57 -0600592static const struct dm_gpio_ops gpio_at91_ops = {
593 .direction_input = at91_gpio_direction_input,
594 .direction_output = at91_gpio_direction_output,
595 .get_value = at91_gpio_get_value,
596 .set_value = at91_gpio_set_value,
597 .get_function = at91_gpio_get_function,
598};
599
600static int at91_gpio_probe(struct udevice *dev)
601{
602 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glassb75b15b2020-12-03 16:55:23 -0700603 struct at91_port_plat *plat = dev_get_plat(dev);
Simon Glassde0977b2015-03-05 12:25:20 -0700604 struct gpio_dev_priv *uc_priv = dev_get_uclass_priv(dev);
Wenyou Yange29b82b2017-03-23 12:46:21 +0800605 struct clk clk;
606 int ret;
607
608 ret = clk_get_by_index(dev, 0, &clk);
609 if (ret)
610 return ret;
611
612 ret = clk_enable(&clk);
613 if (ret)
614 return ret;
615
Wenyou Yangeff6d8a2017-03-23 12:46:20 +0800616#if CONFIG_IS_ENABLED(OF_CONTROL)
Masahiro Yamada5150bc72020-08-04 14:14:41 +0900617 plat->base_addr = dev_read_addr(dev);
Wenyou Yangeff6d8a2017-03-23 12:46:20 +0800618#endif
James Byrneb7e33d32019-11-26 11:52:04 +0000619 plat->bank_name = at91_get_bank_name(plat->base_addr);
Simon Glass4d717882014-10-29 13:08:57 -0600620 port->regs = (struct at91_port *)plat->base_addr;
621
James Byrneb7e33d32019-11-26 11:52:04 +0000622 uc_priv->bank_name = plat->bank_name;
623 uc_priv->gpio_count = GPIO_PER_BANK;
624
Simon Glass4d717882014-10-29 13:08:57 -0600625 return 0;
626}
627
Wenyou Yangeff6d8a2017-03-23 12:46:20 +0800628#if CONFIG_IS_ENABLED(OF_CONTROL)
629static const struct udevice_id at91_gpio_ids[] = {
630 { .compatible = "atmel,at91rm9200-gpio" },
631 { }
632};
633#endif
634
Walter Lozano2901ac62020-06-25 01:10:04 -0300635U_BOOT_DRIVER(atmel_at91rm9200_gpio) = {
636 .name = "atmel_at91rm9200_gpio",
Simon Glass4d717882014-10-29 13:08:57 -0600637 .id = UCLASS_GPIO,
Wenyou Yangeff6d8a2017-03-23 12:46:20 +0800638#if CONFIG_IS_ENABLED(OF_CONTROL)
639 .of_match = at91_gpio_ids,
Simon Glassb75b15b2020-12-03 16:55:23 -0700640 .plat_auto = sizeof(struct at91_port_plat),
Wenyou Yangeff6d8a2017-03-23 12:46:20 +0800641#endif
Simon Glass4d717882014-10-29 13:08:57 -0600642 .ops = &gpio_at91_ops,
643 .probe = at91_gpio_probe,
Simon Glass8a2b47f2020-12-03 16:55:17 -0700644 .priv_auto = sizeof(struct at91_port_priv),
Simon Glass4d717882014-10-29 13:08:57 -0600645};
646#endif