blob: 76fcd3fb930cf785687ffaf7b81f3d445682f0c4 [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}
Zixun LI760e4dd2024-11-13 11:10:54 +0100231
232static void at91_set_port_multi_drive(struct at91_port *at91_port, int offset, int is_on)
233{
234 u32 mask;
235
236 mask = 1 << offset;
237 if (is_on)
238 writel(mask, &at91_port->mder);
239 else
240 writel(mask, &at91_port->mddr);
241}
Zixun LI10766aa2024-11-13 11:11:09 +0100242
243static bool at91_get_port_multi_drive(struct at91_port *at91_port, int offset)
244{
245 u32 mask, val;
246
247 mask = 1 << offset;
248 val = readl(&at91_port->mdsr);
249 return !!(val & mask);
250}
251
252static bool at91_get_port_pullup(struct at91_port *at91_port, int offset)
253{
254 u32 mask, val;
255
256 mask = 1 << offset;
257 val = readl(&at91_port->pusr);
258 return !(val & mask);
259}
Simon Glass4d717882014-10-29 13:08:57 -0600260#endif
261
262static void at91_set_port_input(struct at91_port *at91_port, int offset,
263 int use_pullup)
264{
265 u32 mask;
266
267 mask = 1 << offset;
268 writel(mask, &at91_port->idr);
269 at91_set_port_pullup(at91_port, offset, use_pullup);
270 writel(mask, &at91_port->odr);
271 writel(mask, &at91_port->per);
272}
273
Jens Scharsig8d065462010-02-03 22:46:16 +0100274/*
275 * mux the pin to the gpio controller (instead of "A" or "B" peripheral), and
276 * configure it for an input.
277 */
278int at91_set_pio_input(unsigned port, u32 pin, int use_pullup)
279{
Bo Shen02d88142013-08-22 15:24:40 +0800280 struct at91_port *at91_port = at91_pio_get_port(port);
Jens Scharsig8d065462010-02-03 22:46:16 +0100281
Simon Glass4d717882014-10-29 13:08:57 -0600282 if (at91_port && (pin < GPIO_PER_BANK))
283 at91_set_port_input(at91_port, pin, use_pullup);
Bo Shen02d88142013-08-22 15:24:40 +0800284
Jens Scharsig8d065462010-02-03 22:46:16 +0100285 return 0;
286}
287
Simon Glass4d717882014-10-29 13:08:57 -0600288static void at91_set_port_output(struct at91_port *at91_port, int offset,
289 int value)
290{
291 u32 mask;
292
293 mask = 1 << offset;
294 writel(mask, &at91_port->idr);
295 writel(mask, &at91_port->pudr);
296 if (value)
297 writel(mask, &at91_port->sodr);
298 else
299 writel(mask, &at91_port->codr);
300 writel(mask, &at91_port->oer);
301 writel(mask, &at91_port->per);
302}
303
Jens Scharsig8d065462010-02-03 22:46:16 +0100304/*
305 * mux the pin to the gpio controller (instead of "A" or "B" peripheral),
306 * and configure it for an output.
307 */
308int at91_set_pio_output(unsigned port, u32 pin, int value)
309{
Bo Shen02d88142013-08-22 15:24:40 +0800310 struct at91_port *at91_port = at91_pio_get_port(port);
Jens Scharsig8d065462010-02-03 22:46:16 +0100311
Simon Glass4d717882014-10-29 13:08:57 -0600312 if (at91_port && (pin < GPIO_PER_BANK))
313 at91_set_port_output(at91_port, pin, value);
Bo Shen02d88142013-08-22 15:24:40 +0800314
Jens Scharsig8d065462010-02-03 22:46:16 +0100315 return 0;
316}
317
318/*
319 * enable/disable the glitch filter. mostly used with IRQ handling.
320 */
321int at91_set_pio_deglitch(unsigned port, unsigned pin, int is_on)
322{
Bo Shen02d88142013-08-22 15:24:40 +0800323 struct at91_port *at91_port = at91_pio_get_port(port);
324 u32 mask;
Jens Scharsig8d065462010-02-03 22:46:16 +0100325
Simon Glass4d717882014-10-29 13:08:57 -0600326 if (at91_port && (pin < GPIO_PER_BANK)) {
Jens Scharsig8d065462010-02-03 22:46:16 +0100327 mask = 1 << pin;
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800328 if (is_on)
329 writel(mask, &at91_port->ifer);
330 else
331 writel(mask, &at91_port->ifdr);
332 }
333
334 return 0;
335}
336
337/*
338 * enable/disable the glitch filter. mostly used with IRQ handling.
339 */
340int at91_pio3_set_pio_deglitch(unsigned port, unsigned pin, int is_on)
341{
342 struct at91_port *at91_port = at91_pio_get_port(port);
343 u32 mask;
344
345 if (at91_port && (pin < GPIO_PER_BANK)) {
346 mask = 1 << pin;
Bo Shen0ac13452012-05-20 15:50:00 +0000347 if (is_on) {
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800348 writel(mask, &at91_port->mux.pio3.ifscdr);
Bo Shen02d88142013-08-22 15:24:40 +0800349 writel(mask, &at91_port->ifer);
Bo Shen0ac13452012-05-20 15:50:00 +0000350 } else {
Bo Shen02d88142013-08-22 15:24:40 +0800351 writel(mask, &at91_port->ifdr);
Bo Shen0ac13452012-05-20 15:50:00 +0000352 }
353 }
Bo Shen02d88142013-08-22 15:24:40 +0800354
Bo Shen0ac13452012-05-20 15:50:00 +0000355 return 0;
356}
357
Bo Shen0ac13452012-05-20 15:50:00 +0000358/*
359 * enable/disable the debounce filter.
360 */
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800361int at91_pio3_set_pio_debounce(unsigned port, unsigned pin, int is_on, int div)
Bo Shen0ac13452012-05-20 15:50:00 +0000362{
Bo Shen02d88142013-08-22 15:24:40 +0800363 struct at91_port *at91_port = at91_pio_get_port(port);
364 u32 mask;
Bo Shen0ac13452012-05-20 15:50:00 +0000365
Simon Glass4d717882014-10-29 13:08:57 -0600366 if (at91_port && (pin < GPIO_PER_BANK)) {
Bo Shen0ac13452012-05-20 15:50:00 +0000367 mask = 1 << pin;
368 if (is_on) {
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800369 writel(mask, &at91_port->mux.pio3.ifscer);
370 writel(div & PIO_SCDR_DIV, &at91_port->mux.pio3.scdr);
Bo Shen02d88142013-08-22 15:24:40 +0800371 writel(mask, &at91_port->ifer);
Bo Shen0ac13452012-05-20 15:50:00 +0000372 } else {
Bo Shen02d88142013-08-22 15:24:40 +0800373 writel(mask, &at91_port->ifdr);
Bo Shen0ac13452012-05-20 15:50:00 +0000374 }
375 }
Bo Shen02d88142013-08-22 15:24:40 +0800376
Bo Shen0ac13452012-05-20 15:50:00 +0000377 return 0;
378}
379
380/*
381 * enable/disable the pull-down.
382 * If pull-up already enabled while calling the function, we disable it.
383 */
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800384int at91_pio3_set_pio_pulldown(unsigned port, unsigned pin, int is_on)
Bo Shen0ac13452012-05-20 15:50:00 +0000385{
Bo Shen02d88142013-08-22 15:24:40 +0800386 struct at91_port *at91_port = at91_pio_get_port(port);
387 u32 mask;
Bo Shen0ac13452012-05-20 15:50:00 +0000388
Simon Glass4d717882014-10-29 13:08:57 -0600389 if (at91_port && (pin < GPIO_PER_BANK)) {
Bo Shen0ac13452012-05-20 15:50:00 +0000390 mask = 1 << pin;
Marek Vasutba91bd52016-05-04 23:05:23 +0200391 if (is_on) {
392 at91_set_pio_pullup(port, pin, 0);
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800393 writel(mask, &at91_port->mux.pio3.ppder);
Marek Vasutba91bd52016-05-04 23:05:23 +0200394 } else
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800395 writel(mask, &at91_port->mux.pio3.ppddr);
Bo Shen0ac13452012-05-20 15:50:00 +0000396 }
Bo Shen02d88142013-08-22 15:24:40 +0800397
Bo Shen0ac13452012-05-20 15:50:00 +0000398 return 0;
399}
400
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800401int at91_pio3_set_pio_pullup(unsigned port, unsigned pin, int use_pullup)
402{
403 struct at91_port *at91_port = at91_pio_get_port(port);
404
405 if (use_pullup)
406 at91_pio3_set_pio_pulldown(port, pin, 0);
407
408 if (at91_port && (pin < GPIO_PER_BANK))
409 at91_set_port_pullup(at91_port, pin, use_pullup);
410
411 return 0;
412}
413
Bo Shen0ac13452012-05-20 15:50:00 +0000414/*
415 * disable Schmitt trigger
416 */
Wenyou Yang4a92a3e2017-03-23 12:44:36 +0800417int at91_pio3_set_pio_disable_schmitt_trig(unsigned port, unsigned pin)
Bo Shen0ac13452012-05-20 15:50:00 +0000418{
Bo Shen02d88142013-08-22 15:24:40 +0800419 struct at91_port *at91_port = at91_pio_get_port(port);
420 u32 mask;
Bo Shen0ac13452012-05-20 15:50:00 +0000421
Simon Glass4d717882014-10-29 13:08:57 -0600422 if (at91_port && (pin < GPIO_PER_BANK)) {
Bo Shen0ac13452012-05-20 15:50:00 +0000423 mask = 1 << pin;
Bo Shen02d88142013-08-22 15:24:40 +0800424 writel(readl(&at91_port->schmitt) | mask,
425 &at91_port->schmitt);
Jens Scharsig8d065462010-02-03 22:46:16 +0100426 }
Bo Shen02d88142013-08-22 15:24:40 +0800427
Jens Scharsig8d065462010-02-03 22:46:16 +0100428 return 0;
429}
430
431/*
432 * enable/disable the multi-driver. This is only valid for output and
433 * allows the output pin to run as an open collector output.
434 */
435int at91_set_pio_multi_drive(unsigned port, unsigned pin, int is_on)
436{
Bo Shen02d88142013-08-22 15:24:40 +0800437 struct at91_port *at91_port = at91_pio_get_port(port);
438 u32 mask;
Jens Scharsig8d065462010-02-03 22:46:16 +0100439
Simon Glass4d717882014-10-29 13:08:57 -0600440 if (at91_port && (pin < GPIO_PER_BANK)) {
Jens Scharsig8d065462010-02-03 22:46:16 +0100441 mask = 1 << pin;
442 if (is_on)
Bo Shen02d88142013-08-22 15:24:40 +0800443 writel(mask, &at91_port->mder);
Jens Scharsig8d065462010-02-03 22:46:16 +0100444 else
Bo Shen02d88142013-08-22 15:24:40 +0800445 writel(mask, &at91_port->mddr);
Jens Scharsig8d065462010-02-03 22:46:16 +0100446 }
Bo Shen02d88142013-08-22 15:24:40 +0800447
Jens Scharsig8d065462010-02-03 22:46:16 +0100448 return 0;
449}
450
Simon Glass4d717882014-10-29 13:08:57 -0600451static void at91_set_port_value(struct at91_port *at91_port, int offset,
452 int value)
453{
454 u32 mask;
455
456 mask = 1 << offset;
457 if (value)
458 writel(mask, &at91_port->sodr);
459 else
460 writel(mask, &at91_port->codr);
461}
462
Jens Scharsig8d065462010-02-03 22:46:16 +0100463/*
464 * assuming the pin is muxed as a gpio output, set its value.
465 */
466int at91_set_pio_value(unsigned port, unsigned pin, int value)
467{
Bo Shen02d88142013-08-22 15:24:40 +0800468 struct at91_port *at91_port = at91_pio_get_port(port);
Jens Scharsig8d065462010-02-03 22:46:16 +0100469
Simon Glass4d717882014-10-29 13:08:57 -0600470 if (at91_port && (pin < GPIO_PER_BANK))
471 at91_set_port_value(at91_port, pin, value);
Bo Shen02d88142013-08-22 15:24:40 +0800472
Jens Scharsig8d065462010-02-03 22:46:16 +0100473 return 0;
474}
475
Simon Glass4d717882014-10-29 13:08:57 -0600476static int at91_get_port_value(struct at91_port *at91_port, int offset)
477{
478 u32 pdsr = 0, mask;
479
480 mask = 1 << offset;
481 pdsr = readl(&at91_port->pdsr) & mask;
482
483 return pdsr != 0;
484}
Jens Scharsig8d065462010-02-03 22:46:16 +0100485/*
486 * read the pin's value (works even if it's not muxed as a gpio).
487 */
488int at91_get_pio_value(unsigned port, unsigned pin)
489{
Bo Shen02d88142013-08-22 15:24:40 +0800490 struct at91_port *at91_port = at91_pio_get_port(port);
Jens Scharsig8d065462010-02-03 22:46:16 +0100491
Simon Glass4d717882014-10-29 13:08:57 -0600492 if (at91_port && (pin < GPIO_PER_BANK))
493 return at91_get_port_value(at91_port, pin);
Bo Shen02d88142013-08-22 15:24:40 +0800494
Simon Glass4d717882014-10-29 13:08:57 -0600495 return 0;
Jens Scharsig8d065462010-02-03 22:46:16 +0100496}
Bo Shenad1d2ac2013-08-13 14:38:31 +0800497
Simon Glassfa4689a2019-12-06 21:41:35 -0700498#if !CONFIG_IS_ENABLED(DM_GPIO)
Bo Shenad1d2ac2013-08-13 14:38:31 +0800499/* Common GPIO API */
500
Bo Shenad1d2ac2013-08-13 14:38:31 +0800501int gpio_request(unsigned gpio, const char *label)
502{
503 return 0;
504}
505
506int gpio_free(unsigned gpio)
507{
508 return 0;
509}
510
511int gpio_direction_input(unsigned gpio)
512{
513 at91_set_pio_input(at91_gpio_to_port(gpio),
514 at91_gpio_to_pin(gpio), 0);
515 return 0;
516}
517
518int gpio_direction_output(unsigned gpio, int value)
519{
520 at91_set_pio_output(at91_gpio_to_port(gpio),
521 at91_gpio_to_pin(gpio), value);
522 return 0;
523}
524
525int gpio_get_value(unsigned gpio)
526{
527 return at91_get_pio_value(at91_gpio_to_port(gpio),
528 at91_gpio_to_pin(gpio));
529}
530
531int gpio_set_value(unsigned gpio, int value)
532{
533 at91_set_pio_value(at91_gpio_to_port(gpio),
534 at91_gpio_to_pin(gpio), value);
535
536 return 0;
537}
Simon Glass4d717882014-10-29 13:08:57 -0600538#endif
539
Simon Glassfa4689a2019-12-06 21:41:35 -0700540#if CONFIG_IS_ENABLED(DM_GPIO)
Simon Glass4d717882014-10-29 13:08:57 -0600541
542struct at91_port_priv {
543 struct at91_port *regs;
544};
545
546/* set GPIO pin 'gpio' as an input */
547static int at91_gpio_direction_input(struct udevice *dev, unsigned offset)
548{
Axel Lin98b9b372015-01-31 14:47:34 +0800549 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glass4d717882014-10-29 13:08:57 -0600550
551 at91_set_port_input(port->regs, offset, 0);
552
553 return 0;
554}
555
556/* set GPIO pin 'gpio' as an output, with polarity 'value' */
557static int at91_gpio_direction_output(struct udevice *dev, unsigned offset,
558 int value)
559{
Axel Lin98b9b372015-01-31 14:47:34 +0800560 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glass4d717882014-10-29 13:08:57 -0600561
562 at91_set_port_output(port->regs, offset, value);
563
564 return 0;
565}
566
567/* read GPIO IN value of pin 'gpio' */
568static int at91_gpio_get_value(struct udevice *dev, unsigned offset)
569{
Axel Lin98b9b372015-01-31 14:47:34 +0800570 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glass4d717882014-10-29 13:08:57 -0600571
572 return at91_get_port_value(port->regs, offset);
573}
574
575/* write GPIO OUT value to pin 'gpio' */
576static int at91_gpio_set_value(struct udevice *dev, unsigned offset,
577 int value)
578{
Axel Lin98b9b372015-01-31 14:47:34 +0800579 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glass4d717882014-10-29 13:08:57 -0600580
581 at91_set_port_value(port->regs, offset, value);
582
583 return 0;
584}
585
586static int at91_gpio_get_function(struct udevice *dev, unsigned offset)
587{
Axel Lin98b9b372015-01-31 14:47:34 +0800588 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glass4d717882014-10-29 13:08:57 -0600589
Zixun LI6e399502024-11-13 11:10:22 +0100590 if (!at91_is_port_gpio(port->regs, offset))
591 return GPIOF_FUNC;
592
Simon Glass4d717882014-10-29 13:08:57 -0600593 if (at91_get_port_output(port->regs, offset))
594 return GPIOF_OUTPUT;
595 else
596 return GPIOF_INPUT;
597}
598
Zixun LI760e4dd2024-11-13 11:10:54 +0100599static int at91_gpio_set_flags(struct udevice *dev, unsigned int offset,
600 ulong flags)
601{
602 struct at91_port_priv *port = dev_get_priv(dev);
603 ulong supported_mask;
604
605 supported_mask = GPIOD_OPEN_DRAIN | GPIOD_MASK_DIR | GPIOD_PULL_UP;
606 if (flags & ~supported_mask)
607 return -ENOTSUPP;
608
609 if (flags & GPIOD_IS_OUT) {
610 if (flags & GPIOD_OPEN_DRAIN)
611 at91_set_port_multi_drive(port->regs, offset, true);
612 else
613 at91_set_port_multi_drive(port->regs, offset, false);
614
615 at91_set_port_output(port->regs, offset, flags & GPIOD_IS_OUT_ACTIVE);
616
617 } else if (flags & GPIOD_IS_IN) {
618 at91_set_port_input(port->regs, offset, false);
619 }
620 if (flags & GPIOD_PULL_UP)
621 at91_set_port_pullup(port->regs, offset, true);
622
623 return 0;
624}
625
Zixun LI10766aa2024-11-13 11:11:09 +0100626static int at91_gpio_get_flags(struct udevice *dev, unsigned int offset,
627 ulong *flagsp)
628{
629 struct at91_port_priv *port = dev_get_priv(dev);
630 ulong dir_flags = 0;
631
632 if (at91_get_port_output(port->regs, offset)) {
633 dir_flags |= GPIOD_IS_OUT;
634
635 if (at91_get_port_multi_drive(port->regs, offset))
636 dir_flags |= GPIOD_OPEN_DRAIN;
637
638 if (at91_get_port_value(port->regs, offset))
639 dir_flags |= GPIOD_IS_OUT_ACTIVE;
640 } else {
641 dir_flags |= GPIOD_IS_IN;
642 }
643
644 if (at91_get_port_pullup(port->regs, offset))
645 dir_flags |= GPIOD_PULL_UP;
646
647 *flagsp = dir_flags;
648
649 return 0;
650}
651
James Byrneb7e33d32019-11-26 11:52:04 +0000652static const char *at91_get_bank_name(uint32_t base_addr)
653{
654 switch (base_addr) {
655 case ATMEL_BASE_PIOA:
656 return "PIOA";
657 case ATMEL_BASE_PIOB:
658 return "PIOB";
659 case ATMEL_BASE_PIOC:
660 return "PIOC";
661#if (ATMEL_PIO_PORTS > 3)
662 case ATMEL_BASE_PIOD:
663 return "PIOD";
664#if (ATMEL_PIO_PORTS > 4)
665 case ATMEL_BASE_PIOE:
666 return "PIOE";
667#endif
668#endif
669 }
670
671 return "undefined";
672}
673
Simon Glass4d717882014-10-29 13:08:57 -0600674static const struct dm_gpio_ops gpio_at91_ops = {
675 .direction_input = at91_gpio_direction_input,
676 .direction_output = at91_gpio_direction_output,
677 .get_value = at91_gpio_get_value,
678 .set_value = at91_gpio_set_value,
679 .get_function = at91_gpio_get_function,
Zixun LI760e4dd2024-11-13 11:10:54 +0100680 .set_flags = at91_gpio_set_flags,
Zixun LI10766aa2024-11-13 11:11:09 +0100681 .get_flags = at91_gpio_get_flags,
Simon Glass4d717882014-10-29 13:08:57 -0600682};
683
684static int at91_gpio_probe(struct udevice *dev)
685{
686 struct at91_port_priv *port = dev_get_priv(dev);
Simon Glassb75b15b2020-12-03 16:55:23 -0700687 struct at91_port_plat *plat = dev_get_plat(dev);
Simon Glassde0977b2015-03-05 12:25:20 -0700688 struct gpio_dev_priv *uc_priv = dev_get_uclass_priv(dev);
Wenyou Yange29b82b2017-03-23 12:46:21 +0800689 struct clk clk;
690 int ret;
691
692 ret = clk_get_by_index(dev, 0, &clk);
693 if (ret)
694 return ret;
695
696 ret = clk_enable(&clk);
697 if (ret)
698 return ret;
699
Wenyou Yangeff6d8a2017-03-23 12:46:20 +0800700#if CONFIG_IS_ENABLED(OF_CONTROL)
Masahiro Yamada5150bc72020-08-04 14:14:41 +0900701 plat->base_addr = dev_read_addr(dev);
Wenyou Yangeff6d8a2017-03-23 12:46:20 +0800702#endif
James Byrneb7e33d32019-11-26 11:52:04 +0000703 plat->bank_name = at91_get_bank_name(plat->base_addr);
Simon Glass4d717882014-10-29 13:08:57 -0600704 port->regs = (struct at91_port *)plat->base_addr;
705
James Byrneb7e33d32019-11-26 11:52:04 +0000706 uc_priv->bank_name = plat->bank_name;
707 uc_priv->gpio_count = GPIO_PER_BANK;
708
Simon Glass4d717882014-10-29 13:08:57 -0600709 return 0;
710}
711
Wenyou Yangeff6d8a2017-03-23 12:46:20 +0800712#if CONFIG_IS_ENABLED(OF_CONTROL)
713static const struct udevice_id at91_gpio_ids[] = {
714 { .compatible = "atmel,at91rm9200-gpio" },
715 { }
716};
717#endif
718
Walter Lozano2901ac62020-06-25 01:10:04 -0300719U_BOOT_DRIVER(atmel_at91rm9200_gpio) = {
720 .name = "atmel_at91rm9200_gpio",
Simon Glass4d717882014-10-29 13:08:57 -0600721 .id = UCLASS_GPIO,
Wenyou Yangeff6d8a2017-03-23 12:46:20 +0800722#if CONFIG_IS_ENABLED(OF_CONTROL)
723 .of_match = at91_gpio_ids,
Simon Glassb75b15b2020-12-03 16:55:23 -0700724 .plat_auto = sizeof(struct at91_port_plat),
Wenyou Yangeff6d8a2017-03-23 12:46:20 +0800725#endif
Simon Glass4d717882014-10-29 13:08:57 -0600726 .ops = &gpio_at91_ops,
727 .probe = at91_gpio_probe,
Simon Glass8a2b47f2020-12-03 16:55:17 -0700728 .priv_auto = sizeof(struct at91_port_priv),
Simon Glass4d717882014-10-29 13:08:57 -0600729};
730#endif