diff mbox series

[4/5] pinctrl: Add pin controller driver for AAEON UP boards

Message ID 20241211-aaeon-up-board-pinctrl-support-v1-4-24719be27631@bootlin.com
State New
Headers show
Series Add support for the AAEON UP board FPGA | expand

Commit Message

Thomas Richard Dec. 11, 2024, 4:27 p.m. UTC
This enables the pin control support of the onboard FPGA on AAEON UP
boards.
Due to the hardware design, the driver shall control its pins in tandem
with their corresponding Intel SoC GPIOs.

UP boards and UP Squared boards are supported.

Based on the work done by Gary Wang <garywang@aaeon.com.tw>, largely
rewritten.

Signed-off-by: Thomas Richard <thomas.richard@bootlin.com>
---
 drivers/pinctrl/Kconfig           |   14 +
 drivers/pinctrl/Makefile          |    1 +
 drivers/pinctrl/pinctrl-upboard.c | 1090 +++++++++++++++++++++++++++++++++++++
 3 files changed, 1105 insertions(+)

Comments

Linus Walleij Dec. 20, 2024, 12:22 p.m. UTC | #1
Hi Thomas,

thanks for your patch!

On Wed, Dec 11, 2024 at 5:27 PM Thomas Richard
<thomas.richard@bootlin.com> wrote:

> This enables the pin control support of the onboard FPGA on AAEON UP
> boards.
> Due to the hardware design, the driver shall control its pins in tandem
> with their corresponding Intel SoC GPIOs.
>
> UP boards and UP Squared boards are supported.
>
> Based on the work done by Gary Wang <garywang@aaeon.com.tw>, largely
> rewritten.
>
> Signed-off-by: Thomas Richard <thomas.richard@bootlin.com>

Overall this looks as a good start, some comments below.

> +config PINCTRL_UPBOARD
> +       tristate "AAeon UP board FPGA pin controller"
> +       depends on MFD_UPBOARD_FPGA
> +       select PINMUX
> +       select GENERIC_PINCTRL_GROUPS
> +       select GENERIC_PINMUX_FUNCTIONS

This implements GPIO so you need:
select GPIOLIB

But I'm not sure because of some oddities, see below.

> +#include <linux/device.h>
> +#include <linux/gpio/consumer.h>

Questionable include, see below.

> +static int __upboard_pinctrl_gpio_request_enable(struct pinctrl_dev *pctldev,
> +                                                unsigned int offset)

I'm not a fan of functions named with __inner_function() double-underscore
convention. The reason is that double underscore is also used for
compiler intrinsics. Can you just name it

committ_upboard_pinctrl_gpio_request_enable()?

> +static void __upboard_pinctrl_gpio_disable_free(struct pinctrl_dev *pctldev, unsigned int offset)

Dito

> +static int __upboard_pinctrl_gpio_set_direction(struct pinctrl_dev *pctldev,
> +                                               unsigned int offset, bool input)

Dito

The pinmux code is very straight forward otherwise, good job!

> +static int upboard_gpio_get_direction(struct gpio_chip *gc, unsigned int offset)
> +{
> +       struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
> +       unsigned int pin = pctrl->pctrl_data->pin_header[offset];
> +       int mode;
> +
> +       if (pctrl->gpio[offset])
> +               return gpiod_get_direction(pctrl->gpio[offset]);

See below.

> +       /*
> +        * GPIO was not requested so SoC pin is probably not in GPIO mode.
> +        * When a gpio_chip is registered, the core calls get_direction() for all lines.
> +        * At this time, upboard_gpio_request() was not yet called, so the driver didn't
> +        * request the corresponding SoC pin. So the SoC pin is probably in function (not in
> +        * GPIO mode).
> +        *
> +        * To get the direction of the SoC pin, it shall be requested in GPIO mode.
> +        * Once a SoC pin is set in GPIO mode, there is no way to set it back to its
> +        * function mode.
> +        * Instead of returning the SoC pin direction, the direction of the FPGA pin is
> +        * returned (only for the get_direction() called during the gpio_chip registration).
> +        */
> +       mode = upboard_pinctrl_pin_get_mode(pctrl->pctldev, pin);

Fair enough I guess it's the best we can do here.

> +static int upboard_gpio_get(struct gpio_chip *gc, unsigned int offset)
> +{
> +       struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
> +
> +       return gpiod_get_value(pctrl->gpio[offset]);
> +}
> +
> +static void upboard_gpio_set(struct gpio_chip *gc, unsigned int offset, int value)
> +{
> +       struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
> +
> +       gpiod_set_value(pctrl->gpio[offset], value);
> +}
> +
> +static int upboard_gpio_direction_input(struct gpio_chip *gc, unsigned int offset)
> +{
> +       struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
> +       int ret;
> +
> +       ret = pinctrl_gpio_direction_input(gc, offset);
> +       if (ret)
> +               return ret;
> +
> +       return gpiod_direction_input(pctrl->gpio[offset]);
> +}
> +
> +static int upboard_gpio_direction_output(struct gpio_chip *gc, unsigned int offset, int value)
> +{
> +       struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
> +       int ret;
> +
> +       ret = pinctrl_gpio_direction_output(gc, offset);
> +       if (ret)
> +               return ret;
> +
> +       return gpiod_direction_output(pctrl->gpio[offset], value);
> +}

This looks dangerous and I guess also the reason you are including consumer.h.

Explain with a comment in the code what is going on here, like if this
GPIO comes from a completely different hardware unit, it looks like
a recepie for an eternal loop if it would point back to the same GPIO.

All of these have the same "loop out to another hardware" feature
that looks weird to me, but explain what's going on so I understand
it.

To me usually pin control works like this:

linux gpio <-> gpio driver <-> pin control driver

so the pin control driver is a pure "backend" for GPIO,
typically implements in struct pinmux_ops:
.gpio_request_enable()
.gpio_disable_free()
.gpio_set_direction()

that just set up the pin in the corresponding way. If your hardware
cannot mux back a pin from GPIO mode (as a comment says)
I would say that gpio_disable_free() can just return -ENODEV
or something if the pin has been put into gpio mode, maybe
some experimentation is needed there.

The corresponding GPIO driver typically uses GPIO ranges
to access the corresponding pin. It usually call
gpiochip_add_pin_range() to map its pins to the pin control
driver (if e.g. device tree is not used for the ranges).

What you do here is confusing to me, it looks like:

linux gpio <-> this gpio shim <-> pin control <-> other gpio driver

I think it is better to try to keep things separate if you can,
the current design seems to come from an attempt to be
"complete" and protect users from themselves, but we can
never protect users from themselves.

> +static int upboard_gpio_to_irq(struct gpio_chip *gc, unsigned int offset)
> +{
> +       struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
> +
> +       return gpiod_to_irq(pctrl->gpio[offset]);
> +}

If you use the GPIOLIB_IRQCHIP, you do not need to define this function
at all, it is handled by gpiolib.

> +       ret = gpiochip_add_pinlist_range(chip, dev_name(dev), 0, pctrl->pctrl_data->pin_header,
> +                                        pctrl->pctrl_data->ngpio);

I would rather have it that the actual gpio chip (the one that write
something into hardware registers) do this without another gpio chip
inbetween if you see what I mean.

But explain what's going on! I'm curious.

Yours,
Linus Walleij
Thomas Richard Dec. 20, 2024, 1:50 p.m. UTC | #2
On 12/20/24 13:22, Linus Walleij wrote:
> Hi Thomas,
> 
> thanks for your patch!

Hi Linus,

Thanks for the review !!

> 
> On Wed, Dec 11, 2024 at 5:27 PM Thomas Richard
> <thomas.richard@bootlin.com> wrote:
> 
>> This enables the pin control support of the onboard FPGA on AAEON UP
>> boards.
>> Due to the hardware design, the driver shall control its pins in tandem
>> with their corresponding Intel SoC GPIOs.
>>
>> UP boards and UP Squared boards are supported.
>>
>> Based on the work done by Gary Wang <garywang@aaeon.com.tw>, largely
>> rewritten.
>>
>> Signed-off-by: Thomas Richard <thomas.richard@bootlin.com>
> 
> Overall this looks as a good start, some comments below.
> 
>> +config PINCTRL_UPBOARD
>> +       tristate "AAeon UP board FPGA pin controller"
>> +       depends on MFD_UPBOARD_FPGA
>> +       select PINMUX
>> +       select GENERIC_PINCTRL_GROUPS
>> +       select GENERIC_PINMUX_FUNCTIONS
> 
> This implements GPIO so you need:
> select GPIOLIB
> 
> But I'm not sure because of some oddities, see below.
> 
>> +#include <linux/device.h>
>> +#include <linux/gpio/consumer.h>
> 
> Questionable include, see below.
> 
>> +static int __upboard_pinctrl_gpio_request_enable(struct pinctrl_dev *pctldev,
>> +                                                unsigned int offset)
> 
> I'm not a fan of functions named with __inner_function() double-underscore
> convention. The reason is that double underscore is also used for
> compiler intrinsics. Can you just name it
> 
> committ_upboard_pinctrl_gpio_request_enable()?

ack

> 
>> +static void __upboard_pinctrl_gpio_disable_free(struct pinctrl_dev *pctldev, unsigned int offset)
> 

[...]

>> +static int upboard_gpio_direction_output(struct gpio_chip *gc, unsigned int offset, int value)
>> +{
>> +       struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
>> +       int ret;
>> +
>> +       ret = pinctrl_gpio_direction_output(gc, offset);
>> +       if (ret)
>> +               return ret;
>> +
>> +       return gpiod_direction_output(pctrl->gpio[offset], value);
>> +}
> 
> This looks dangerous and I guess also the reason you are including consumer.h.
> 
> Explain with a comment in the code what is going on here, like if this
> GPIO comes from a completely different hardware unit, it looks like
> a recepie for an eternal loop if it would point back to the same GPIO.
> 
> All of these have the same "loop out to another hardware" feature
> that looks weird to me, but explain what's going on so I understand
> it.
> 
> To me usually pin control works like this:
> 
> linux gpio <-> gpio driver <-> pin control driver
> 
> so the pin control driver is a pure "backend" for GPIO,
> typically implements in struct pinmux_ops:
> .gpio_request_enable()
> .gpio_disable_free()
> .gpio_set_direction()
> 
> that just set up the pin in the corresponding way. If your hardware
> cannot mux back a pin from GPIO mode (as a comment says)
> I would say that gpio_disable_free() can just return -ENODEV
> or something if the pin has been put into gpio mode, maybe
> some experimentation is needed there.
> 
> The corresponding GPIO driver typically uses GPIO ranges
> to access the corresponding pin. It usually call
> gpiochip_add_pin_range() to map its pins to the pin control
> driver (if e.g. device tree is not used for the ranges).
> 
> What you do here is confusing to me, it looks like:
> 
> linux gpio <-> this gpio shim <-> pin control <-> other gpio driver
> 
> I think it is better to try to keep things separate if you can,
> the current design seems to come from an attempt to be
> "complete" and protect users from themselves, but we can
> never protect users from themselves.
> 
>> +static int upboard_gpio_to_irq(struct gpio_chip *gc, unsigned int offset)
>> +{
>> +       struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
>> +
>> +       return gpiod_to_irq(pctrl->gpio[offset]);
>> +}
> 
> If you use the GPIOLIB_IRQCHIP, you do not need to define this function
> at all, it is handled by gpiolib.
> 
>> +       ret = gpiochip_add_pinlist_range(chip, dev_name(dev), 0, pctrl->pctrl_data->pin_header,
>> +                                        pctrl->pctrl_data->ngpio);
> 
> I would rather have it that the actual gpio chip (the one that write
> something into hardware registers) do this without another gpio chip
> inbetween if you see what I mean.
> 
> But explain what's going on! I'm curious.

Yes my cover letter was a bit short, and maybe some context was missing.

This FPGA acts as a level shifter between the Intel SoC pins and the pin
header, and also makes a kind of switch/mux.

+---------+         +--------------+             +---+
          |         |              |             | H |
          |---------|              |-------------| E |
          |         |              |             | A |
Intel Soc |---------|    FPGA      |-------------| D |
          |         |              |             | E |
          |---------|              |-------------| R |
          |         |              |             |   |
----------+         +--------------+             +---+


For most of the pins, the FPGA opens/closes a switch to enable/disable
the access to the SoC pin from a pin header.
Each "switch", has a direction flag that shall be set in tandem with the
status of the SoC pin.
For example, if the SoC pin is in PWM mode, the "switch" shall be
configured in output direction.
If the SoC pin is set in GPIO mode, the direction of the "switch" shall
corresponds to the GPIO direction.

+---------+              +--------------+             +---+
          |              |              |             | H |
          |              |      \       |             | E |
          |   PWM1       |       \      |             | A |
Intel Soc |--------------|-----   \-----|-------------| D |
          |              |              |             | E |
          |              |              |             | R |
          |              |    FPGA      |             |   |
----------+              +--------------+             +---+

(PWM1 pin from Intel SoC can be used as PWM, and also in GPIO mode,
thanks to the Intel pinctrl driver).


Few pins (PINMUX_* pins) work differently. The FPGA acts as a mux and
routes for example the I2C0_SDA pin or GPIOX (of the SoC) to the pin header.

+---------+           +--------------+             +---+
          | I2C0_SDA  |              |             | H |
          |-----------|----- \       |             | E |
          |           |       \      |             | A |
Intel Soc |           |        \-----|-------------| D |
          | GPIOX     |              |             | E |
          |-----------|-----         |             | R |
          |           |    FPGA      |             |   |
----------+           +--------------+             +---+

The pin header looks like this:
+--------------------+--------------------+
|      3.3V	     |       5V           |
| GPIO2 / I2C1_SDA   |       5V           |
| GPIO3 / I2C1_SCL   |       GND          |
| GPIO4 / ADC0       | GPIO14 / UART1_TX  |
|      GND	     | GPIO15 / UART1_RX  |
| GPIO17 / UART1_RTS | GPIO18 / I2S_CLK   |
|     GPIO27         |       GND          |
|     GPIO22         |      GPIO23        |
|      3.3V          |      GPIO24        |
| GPIO10 / SPI_MOSI  |       GND          |
| GPIO9 / SPI_MISO   |      GPIO25        |
| GPIO11 / SPI_CLK   | GPIO8 / SPI_CS0    |
|      GND	     | GPIO7 / SPI_CS1    |
| GPIO0 / I2C0_SDA   | GPIO1 / I2C0_SCL   |
|     GPIO5          |       GND          |
|     GPIO6	     | GPIO12 / PWM0      |
| GPIO13 / PWM1      |       GND          |
| GPIO19 / I2S_FRM   | GPIO16 / UART1_CTS |
|     GPIO26	     | GPIO20 / I2S_DIN   |
|      GND	     | GPIO21 / I2S_DOUT  |
+--------------------+--------------------+

The GPIOs in the pin header corresponds to the gpiochip I declare in
this driver.
So when I want to use a pin in GPIO mode, the upboard pinctrl driver
requests the corresponding SoC GPIO to the Intel pinctrl driver.
The SoC pins connected to the FPGA, are identified with "external" id.

The hardware and the FPGA were designed in tandem, so you know for
example that for the GPIOX you need to request the Nth "external" GPIO.

When you drive your GPIO, the upboard gpiochip manages in the same time
the direction of the "switch" and the value/direction of the
corresponding SoC pin.

+------------------+         +--------------+             +---+
                   |---------|              |-------------| H |
                   |---------|   GPIOCHIP   |-------------| E |
   Intel gpiochip  |---------|              |-------------| A |
 provided by Intel |---------|    FPGA      |-------------| D |
  pinctrl driver   |---------|              |-------------| E |
                   |---------|              |-------------| R |
                   |---------|              |-------------|   |
+------------------+         +--------------+             +---+


About gpiochip_add_pinlist_range(), I added it because the FPGA pins
used by the gpiochip are not consecutive.

Please let me know if it is not clear.
And sorry I'm not very good to make ascii art.

Best Regards,

Thomas
Linus Walleij Dec. 21, 2024, 11:43 p.m. UTC | #3
Hi Thomas,

thanks for your detailed reply!

On Fri, Dec 20, 2024 at 2:50 PM Thomas Richard
<thomas.richard@bootlin.com> wrote:

> Yes my cover letter was a bit short, and maybe some context was missing.

The text and graphics below explain it very well, so please include them
into the commit message so we have it there!

> This FPGA acts as a level shifter between the Intel SoC pins and the pin
> header, and also makes a kind of switch/mux.

Since it's Intel we need to notify Andy to help out with this so that
it gets done in a way that works with how he think consumers
should interact with Intel pin control and GPIO.

> +---------+         +--------------+             +---+
>           |         |              |             | H |
>           |---------|              |-------------| E |
>           |         |              |             | A |
> Intel Soc |---------|    FPGA      |-------------| D |
>           |         |              |             | E |
>           |---------|              |-------------| R |
>           |         |              |             |   |
> ----------+         +--------------+             +---+
>
>
> For most of the pins, the FPGA opens/closes a switch to enable/disable
> the access to the SoC pin from a pin header.
> Each "switch", has a direction flag that shall be set in tandem with the
> status of the SoC pin.
> For example, if the SoC pin is in PWM mode, the "switch" shall be
> configured in output direction.
> If the SoC pin is set in GPIO mode, the direction of the "switch" shall
> corresponds to the GPIO direction.
>
> +---------+              +--------------+             +---+
>           |              |              |             | H |
>           |              |      \       |             | E |
>           |   PWM1       |       \      |             | A |
> Intel Soc |--------------|-----   \-----|-------------| D |
>           |              |              |             | E |
>           |              |              |             | R |
>           |              |    FPGA      |             |   |
> ----------+              +--------------+             +---+
>
> (PWM1 pin from Intel SoC can be used as PWM, and also in GPIO mode,
> thanks to the Intel pinctrl driver).
>
>
> Few pins (PINMUX_* pins) work differently. The FPGA acts as a mux and
> routes for example the I2C0_SDA pin or GPIOX (of the SoC) to the pin header.
>
> +---------+           +--------------+             +---+
>           | I2C0_SDA  |              |             | H |
>           |-----------|----- \       |             | E |
>           |           |       \      |             | A |
> Intel Soc |           |        \-----|-------------| D |
>           | GPIOX     |              |             | E |
>           |-----------|-----         |             | R |
>           |           |    FPGA      |             |   |
> ----------+           +--------------+             +---+
>
> The pin header looks like this:
> +--------------------+--------------------+
> |      3.3V          |       5V           |
> | GPIO2 / I2C1_SDA   |       5V           |
> | GPIO3 / I2C1_SCL   |       GND          |
> | GPIO4 / ADC0       | GPIO14 / UART1_TX  |
> |      GND           | GPIO15 / UART1_RX  |
> | GPIO17 / UART1_RTS | GPIO18 / I2S_CLK   |
> |     GPIO27         |       GND          |
> |     GPIO22         |      GPIO23        |
> |      3.3V          |      GPIO24        |
> | GPIO10 / SPI_MOSI  |       GND          |
> | GPIO9 / SPI_MISO   |      GPIO25        |
> | GPIO11 / SPI_CLK   | GPIO8 / SPI_CS0    |
> |      GND           | GPIO7 / SPI_CS1    |
> | GPIO0 / I2C0_SDA   | GPIO1 / I2C0_SCL   |
> |     GPIO5          |       GND          |
> |     GPIO6          | GPIO12 / PWM0      |
> | GPIO13 / PWM1      |       GND          |
> | GPIO19 / I2S_FRM   | GPIO16 / UART1_CTS |
> |     GPIO26         | GPIO20 / I2S_DIN   |
> |      GND           | GPIO21 / I2S_DOUT  |
> +--------------------+--------------------+
>
> The GPIOs in the pin header corresponds to the gpiochip I declare in
> this driver.
> So when I want to use a pin in GPIO mode, the upboard pinctrl driver
> requests the corresponding SoC GPIO to the Intel pinctrl driver.
> The SoC pins connected to the FPGA, are identified with "external" id.
>
> The hardware and the FPGA were designed in tandem, so you know for
> example that for the GPIOX you need to request the Nth "external" GPIO.
>
> When you drive your GPIO, the upboard gpiochip manages in the same time
> the direction of the "switch" and the value/direction of the
> corresponding SoC pin.
>
> +------------------+         +--------------+             +---+
>                    |---------|              |-------------| H |
>                    |---------|   GPIOCHIP   |-------------| E |
>    Intel gpiochip  |---------|              |-------------| A |
>  provided by Intel |---------|    FPGA      |-------------| D |
>   pinctrl driver   |---------|              |-------------| E |
>                    |---------|              |-------------| R |
>                    |---------|              |-------------|   |
> +------------------+         +--------------+             +---+
>
>
> About gpiochip_add_pinlist_range(), I added it because the FPGA pins
> used by the gpiochip are not consecutive.
>
> Please let me know if it is not clear.
> And sorry I'm not very good to make ascii art.

I get it! We have a similar driver in the kernel already, look into:
drivers/gpio/gpio-aggregator.c

The aggregator abstraction is however just software. What you
need here is a gpio-aggregator that adds some hardware
control on top. But it has a very nice design using a bitmap
to keep track of the GPIOs etc, and it supports operations
on multiple GPIOs (many man-hours of hard coding and
design went into that driver, ask Geert and Andy...)

So I would proceed like this:

- The pin control part of the driver looks sound, except
  for the way you add ranges.

- The gpiochip part needs to be refactored using the
  ideas from gpio-aggregator.c.

- Look closely at aggregator and see what you can do
  based on that code, if you can mimic how it picks up
  and forwards all GPIO functions. Maybe part of it
  needs to be made into a library?
 <linux/gpio/gpio-aggregator.h>?
  For example if you start to feel like "I would really like
  to just call gpio_fwd_get_multiple() then this is what
  you want to do. The library can probably still be
  inside gpio-aggregator.c the way we do it in
  e.g. gpio-mmio.c, just export and keep library functions
  separately.

- The way you split up gpiochip_add_pin_range() I
  still do not understand at all, in my view you just want
  this gpiochip to refer to the pin controller pins in the
  same file so I don't see it. How can e.g.
  pinctrl-sx150x.c do this trick but not your driver?
  I might be missing something though.

Yours,
Linus Walleij
Thomas Richard Jan. 3, 2025, 10:28 a.m. UTC | #4
On 12/22/24 00:43, Linus Walleij wrote:
> Hi Thomas,
> 
> thanks for your detailed reply!
> 
> On Fri, Dec 20, 2024 at 2:50 PM Thomas Richard
> <thomas.richard@bootlin.com> wrote:
> 
>> Yes my cover letter was a bit short, and maybe some context was missing.
> 
> The text and graphics below explain it very well, so please include them
> into the commit message so we have it there!
> 
>> This FPGA acts as a level shifter between the Intel SoC pins and the pin
>> header, and also makes a kind of switch/mux.
> 
> Since it's Intel we need to notify Andy to help out with this so that
> it gets done in a way that works with how he think consumers
> should interact with Intel pin control and GPIO.
> 
>> +---------+         +--------------+             +---+
>>           |         |              |             | H |
>>           |---------|              |-------------| E |
>>           |         |              |             | A |
>> Intel Soc |---------|    FPGA      |-------------| D |
>>           |         |              |             | E |
>>           |---------|              |-------------| R |
>>           |         |              |             |   |
>> ----------+         +--------------+             +---+
>>
>>
>> For most of the pins, the FPGA opens/closes a switch to enable/disable
>> the access to the SoC pin from a pin header.
>> Each "switch", has a direction flag that shall be set in tandem with the
>> status of the SoC pin.
>> For example, if the SoC pin is in PWM mode, the "switch" shall be
>> configured in output direction.
>> If the SoC pin is set in GPIO mode, the direction of the "switch" shall
>> corresponds to the GPIO direction.
>>
>> +---------+              +--------------+             +---+
>>           |              |              |             | H |
>>           |              |      \       |             | E |
>>           |   PWM1       |       \      |             | A |
>> Intel Soc |--------------|-----   \-----|-------------| D |
>>           |              |              |             | E |
>>           |              |              |             | R |
>>           |              |    FPGA      |             |   |
>> ----------+              +--------------+             +---+
>>
>> (PWM1 pin from Intel SoC can be used as PWM, and also in GPIO mode,
>> thanks to the Intel pinctrl driver).
>>
>>
>> Few pins (PINMUX_* pins) work differently. The FPGA acts as a mux and
>> routes for example the I2C0_SDA pin or GPIOX (of the SoC) to the pin header.
>>
>> +---------+           +--------------+             +---+
>>           | I2C0_SDA  |              |             | H |
>>           |-----------|----- \       |             | E |
>>           |           |       \      |             | A |
>> Intel Soc |           |        \-----|-------------| D |
>>           | GPIOX     |              |             | E |
>>           |-----------|-----         |             | R |
>>           |           |    FPGA      |             |   |
>> ----------+           +--------------+             +---+
>>
>> The pin header looks like this:
>> +--------------------+--------------------+
>> |      3.3V          |       5V           |
>> | GPIO2 / I2C1_SDA   |       5V           |
>> | GPIO3 / I2C1_SCL   |       GND          |
>> | GPIO4 / ADC0       | GPIO14 / UART1_TX  |
>> |      GND           | GPIO15 / UART1_RX  |
>> | GPIO17 / UART1_RTS | GPIO18 / I2S_CLK   |
>> |     GPIO27         |       GND          |
>> |     GPIO22         |      GPIO23        |
>> |      3.3V          |      GPIO24        |
>> | GPIO10 / SPI_MOSI  |       GND          |
>> | GPIO9 / SPI_MISO   |      GPIO25        |
>> | GPIO11 / SPI_CLK   | GPIO8 / SPI_CS0    |
>> |      GND           | GPIO7 / SPI_CS1    |
>> | GPIO0 / I2C0_SDA   | GPIO1 / I2C0_SCL   |
>> |     GPIO5          |       GND          |
>> |     GPIO6          | GPIO12 / PWM0      |
>> | GPIO13 / PWM1      |       GND          |
>> | GPIO19 / I2S_FRM   | GPIO16 / UART1_CTS |
>> |     GPIO26         | GPIO20 / I2S_DIN   |
>> |      GND           | GPIO21 / I2S_DOUT  |
>> +--------------------+--------------------+
>>
>> The GPIOs in the pin header corresponds to the gpiochip I declare in
>> this driver.
>> So when I want to use a pin in GPIO mode, the upboard pinctrl driver
>> requests the corresponding SoC GPIO to the Intel pinctrl driver.
>> The SoC pins connected to the FPGA, are identified with "external" id.
>>
>> The hardware and the FPGA were designed in tandem, so you know for
>> example that for the GPIOX you need to request the Nth "external" GPIO.
>>
>> When you drive your GPIO, the upboard gpiochip manages in the same time
>> the direction of the "switch" and the value/direction of the
>> corresponding SoC pin.
>>
>> +------------------+         +--------------+             +---+
>>                    |---------|              |-------------| H |
>>                    |---------|   GPIOCHIP   |-------------| E |
>>    Intel gpiochip  |---------|              |-------------| A |
>>  provided by Intel |---------|    FPGA      |-------------| D |
>>   pinctrl driver   |---------|              |-------------| E |
>>                    |---------|              |-------------| R |
>>                    |---------|              |-------------|   |
>> +------------------+         +--------------+             +---+
>>
>>
>> About gpiochip_add_pinlist_range(), I added it because the FPGA pins
>> used by the gpiochip are not consecutive.
>>
>> Please let me know if it is not clear.
>> And sorry I'm not very good to make ascii art.
> 
> I get it! We have a similar driver in the kernel already, look into:
> drivers/gpio/gpio-aggregator.c
> 
> The aggregator abstraction is however just software. What you
> need here is a gpio-aggregator that adds some hardware
> control on top. But it has a very nice design using a bitmap
> to keep track of the GPIOs etc, and it supports operations
> on multiple GPIOs (many man-hours of hard coding and
> design went into that driver, ask Geert and Andy...)
> 
> So I would proceed like this:
> 
> - The pin control part of the driver looks sound, except
>   for the way you add ranges.
> 
> - The gpiochip part needs to be refactored using the
>   ideas from gpio-aggregator.c.
> 
> - Look closely at aggregator and see what you can do
>   based on that code, if you can mimic how it picks up
>   and forwards all GPIO functions. Maybe part of it
>   needs to be made into a library?
>  <linux/gpio/gpio-aggregator.h>?
>   For example if you start to feel like "I would really like
>   to just call gpio_fwd_get_multiple() then this is what
>   you want to do. The library can probably still be
>   inside gpio-aggregator.c the way we do it in
>   e.g. gpio-mmio.c, just export and keep library functions
>   separately.

Hi Linus,

Ok I think I understand what you expect.
I started to look at the gpio-aggregator code, play a bit with it, and
refactor it to use it from my driver.

My main issue is about the request of the SoC GPIOs done by the aggregator.
If from my driver I call the aggregator library to create a gpiochip,
the SoC pins will be requested. So the SoC pins will be set in GPIO
mode, and the pins will never be in function mode.
There is no way to set the pins back to function mode (even if the GPIO
is free).

I tried to add a feature in the aggregator to defer the request of the gpio.
So at the beginning of each ops the gpio_desc is checked. If it is
valid, the gpio can be used. Otherwise, the gpio is requested.
For example:

gpio_fwd_get() {
	if (!gpio_desc_is_valid(desc))
		desc = request_gpio()

	return gpiod_get_value(desc)
}

But when a gpiochip is registered, the core calls get_direction() or
direction_input(), so all GPIOs are requested and it does not solve my
problem.

I expect to register a gpiochip without setting all pins in GPIO mode at
probe time (like all pinctrl driver do).
But I did not find a solution.

Best Regards,

Thomas
Andy Shevchenko Jan. 13, 2025, 9:46 a.m. UTC | #5
On Fri, Jan 03, 2025 at 11:28:30AM +0100, Thomas Richard wrote:
> On 12/22/24 00:43, Linus Walleij wrote:
> > Hi Thomas,
> > 
> > thanks for your detailed reply!
> > 
> > On Fri, Dec 20, 2024 at 2:50 PM Thomas Richard
> > <thomas.richard@bootlin.com> wrote:
> > 
> >> Yes my cover letter was a bit short, and maybe some context was missing.
> > 
> > The text and graphics below explain it very well, so please include them
> > into the commit message so we have it there!
> > 
> >> This FPGA acts as a level shifter between the Intel SoC pins and the pin
> >> header, and also makes a kind of switch/mux.
> > 
> > Since it's Intel we need to notify Andy to help out with this so that
> > it gets done in a way that works with how he think consumers
> > should interact with Intel pin control and GPIO.
> > 
> >> +---------+         +--------------+             +---+
> >>           |         |              |             | H |
> >>           |---------|              |-------------| E |
> >>           |         |              |             | A |
> >> Intel Soc |---------|    FPGA      |-------------| D |
> >>           |         |              |             | E |
> >>           |---------|              |-------------| R |
> >>           |         |              |             |   |
> >> ----------+         +--------------+             +---+
> >>
> >>
> >> For most of the pins, the FPGA opens/closes a switch to enable/disable
> >> the access to the SoC pin from a pin header.
> >> Each "switch", has a direction flag that shall be set in tandem with the
> >> status of the SoC pin.
> >> For example, if the SoC pin is in PWM mode, the "switch" shall be
> >> configured in output direction.
> >> If the SoC pin is set in GPIO mode, the direction of the "switch" shall
> >> corresponds to the GPIO direction.
> >>
> >> +---------+              +--------------+             +---+
> >>           |              |              |             | H |
> >>           |              |      \       |             | E |
> >>           |   PWM1       |       \      |             | A |
> >> Intel Soc |--------------|-----   \-----|-------------| D |
> >>           |              |              |             | E |
> >>           |              |              |             | R |
> >>           |              |    FPGA      |             |   |
> >> ----------+              +--------------+             +---+
> >>
> >> (PWM1 pin from Intel SoC can be used as PWM, and also in GPIO mode,
> >> thanks to the Intel pinctrl driver).
> >>
> >>
> >> Few pins (PINMUX_* pins) work differently. The FPGA acts as a mux and
> >> routes for example the I2C0_SDA pin or GPIOX (of the SoC) to the pin header.
> >>
> >> +---------+           +--------------+             +---+
> >>           | I2C0_SDA  |              |             | H |
> >>           |-----------|----- \       |             | E |
> >>           |           |       \      |             | A |
> >> Intel Soc |           |        \-----|-------------| D |
> >>           | GPIOX     |              |             | E |
> >>           |-----------|-----         |             | R |
> >>           |           |    FPGA      |             |   |
> >> ----------+           +--------------+             +---+
> >>
> >> The pin header looks like this:
> >> +--------------------+--------------------+
> >> |      3.3V          |       5V           |
> >> | GPIO2 / I2C1_SDA   |       5V           |
> >> | GPIO3 / I2C1_SCL   |       GND          |
> >> | GPIO4 / ADC0       | GPIO14 / UART1_TX  |
> >> |      GND           | GPIO15 / UART1_RX  |
> >> | GPIO17 / UART1_RTS | GPIO18 / I2S_CLK   |
> >> |     GPIO27         |       GND          |
> >> |     GPIO22         |      GPIO23        |
> >> |      3.3V          |      GPIO24        |
> >> | GPIO10 / SPI_MOSI  |       GND          |
> >> | GPIO9 / SPI_MISO   |      GPIO25        |
> >> | GPIO11 / SPI_CLK   | GPIO8 / SPI_CS0    |
> >> |      GND           | GPIO7 / SPI_CS1    |
> >> | GPIO0 / I2C0_SDA   | GPIO1 / I2C0_SCL   |
> >> |     GPIO5          |       GND          |
> >> |     GPIO6          | GPIO12 / PWM0      |
> >> | GPIO13 / PWM1      |       GND          |
> >> | GPIO19 / I2S_FRM   | GPIO16 / UART1_CTS |
> >> |     GPIO26         | GPIO20 / I2S_DIN   |
> >> |      GND           | GPIO21 / I2S_DOUT  |
> >> +--------------------+--------------------+
> >>
> >> The GPIOs in the pin header corresponds to the gpiochip I declare in
> >> this driver.
> >> So when I want to use a pin in GPIO mode, the upboard pinctrl driver
> >> requests the corresponding SoC GPIO to the Intel pinctrl driver.
> >> The SoC pins connected to the FPGA, are identified with "external" id.
> >>
> >> The hardware and the FPGA were designed in tandem, so you know for
> >> example that for the GPIOX you need to request the Nth "external" GPIO.
> >>
> >> When you drive your GPIO, the upboard gpiochip manages in the same time
> >> the direction of the "switch" and the value/direction of the
> >> corresponding SoC pin.
> >>
> >> +------------------+         +--------------+             +---+
> >>                    |---------|              |-------------| H |
> >>                    |---------|   GPIOCHIP   |-------------| E |
> >>    Intel gpiochip  |---------|              |-------------| A |
> >>  provided by Intel |---------|    FPGA      |-------------| D |
> >>   pinctrl driver   |---------|              |-------------| E |
> >>                    |---------|              |-------------| R |
> >>                    |---------|              |-------------|   |
> >> +------------------+         +--------------+             +---+
> >>
> >>
> >> About gpiochip_add_pinlist_range(), I added it because the FPGA pins
> >> used by the gpiochip are not consecutive.
> >>
> >> Please let me know if it is not clear.
> >> And sorry I'm not very good to make ascii art.
> > 
> > I get it! We have a similar driver in the kernel already, look into:
> > drivers/gpio/gpio-aggregator.c
> > 
> > The aggregator abstraction is however just software. What you
> > need here is a gpio-aggregator that adds some hardware
> > control on top. But it has a very nice design using a bitmap
> > to keep track of the GPIOs etc, and it supports operations
> > on multiple GPIOs (many man-hours of hard coding and
> > design went into that driver, ask Geert and Andy...)
> > 
> > So I would proceed like this:
> > 
> > - The pin control part of the driver looks sound, except
> >   for the way you add ranges.
> > 
> > - The gpiochip part needs to be refactored using the
> >   ideas from gpio-aggregator.c.
> > 
> > - Look closely at aggregator and see what you can do
> >   based on that code, if you can mimic how it picks up
> >   and forwards all GPIO functions. Maybe part of it
> >   needs to be made into a library?
> >  <linux/gpio/gpio-aggregator.h>?
> >   For example if you start to feel like "I would really like
> >   to just call gpio_fwd_get_multiple() then this is what
> >   you want to do. The library can probably still be
> >   inside gpio-aggregator.c the way we do it in
> >   e.g. gpio-mmio.c, just export and keep library functions
> >   separately.
> 
> Hi Linus,
> 
> Ok I think I understand what you expect.
> I started to look at the gpio-aggregator code, play a bit with it, and
> refactor it to use it from my driver.
> 
> My main issue is about the request of the SoC GPIOs done by the aggregator.
> If from my driver I call the aggregator library to create a gpiochip,
> the SoC pins will be requested. So the SoC pins will be set in GPIO
> mode, and the pins will never be in function mode.
> There is no way to set the pins back to function mode (even if the GPIO
> is free).
> 
> I tried to add a feature in the aggregator to defer the request of the gpio.
> So at the beginning of each ops the gpio_desc is checked. If it is
> valid, the gpio can be used. Otherwise, the gpio is requested.
> For example:
> 
> gpio_fwd_get() {
> 	if (!gpio_desc_is_valid(desc))
> 		desc = request_gpio()
> 
> 	return gpiod_get_value(desc)
> }
> 
> But when a gpiochip is registered, the core calls get_direction() or
> direction_input(), so all GPIOs are requested and it does not solve my
> problem.
> 
> I expect to register a gpiochip without setting all pins in GPIO mode at
> probe time (like all pinctrl driver do).
> But I did not find a solution.

Basically what you need is a pinctrl-aggregattor (an analogue for the pin
muxing and configuration).
Thomas Richard Jan. 14, 2025, 10:28 a.m. UTC | #6
On 1/13/25 10:46, Andy Shevchenko wrote:
> On Fri, Jan 03, 2025 at 11:28:30AM +0100, Thomas Richard wrote:
>> On 12/22/24 00:43, Linus Walleij wrote:
>>> Hi Thomas,
>>>
>>> thanks for your detailed reply!
>>>
>>> On Fri, Dec 20, 2024 at 2:50 PM Thomas Richard
>>> <thomas.richard@bootlin.com> wrote:
>>>
>>>> Yes my cover letter was a bit short, and maybe some context was missing.
>>>
>>> The text and graphics below explain it very well, so please include them
>>> into the commit message so we have it there!
>>>
>>>> This FPGA acts as a level shifter between the Intel SoC pins and the pin
>>>> header, and also makes a kind of switch/mux.
>>>
>>> Since it's Intel we need to notify Andy to help out with this so that
>>> it gets done in a way that works with how he think consumers
>>> should interact with Intel pin control and GPIO.
>>>
>>>> +---------+         +--------------+             +---+
>>>>           |         |              |             | H |
>>>>           |---------|              |-------------| E |
>>>>           |         |              |             | A |
>>>> Intel Soc |---------|    FPGA      |-------------| D |
>>>>           |         |              |             | E |
>>>>           |---------|              |-------------| R |
>>>>           |         |              |             |   |
>>>> ----------+         +--------------+             +---+
>>>>
>>>>
>>>> For most of the pins, the FPGA opens/closes a switch to enable/disable
>>>> the access to the SoC pin from a pin header.
>>>> Each "switch", has a direction flag that shall be set in tandem with the
>>>> status of the SoC pin.
>>>> For example, if the SoC pin is in PWM mode, the "switch" shall be
>>>> configured in output direction.
>>>> If the SoC pin is set in GPIO mode, the direction of the "switch" shall
>>>> corresponds to the GPIO direction.
>>>>
>>>> +---------+              +--------------+             +---+
>>>>           |              |              |             | H |
>>>>           |              |      \       |             | E |
>>>>           |   PWM1       |       \      |             | A |
>>>> Intel Soc |--------------|-----   \-----|-------------| D |
>>>>           |              |              |             | E |
>>>>           |              |              |             | R |
>>>>           |              |    FPGA      |             |   |
>>>> ----------+              +--------------+             +---+
>>>>
>>>> (PWM1 pin from Intel SoC can be used as PWM, and also in GPIO mode,
>>>> thanks to the Intel pinctrl driver).
>>>>
>>>>
>>>> Few pins (PINMUX_* pins) work differently. The FPGA acts as a mux and
>>>> routes for example the I2C0_SDA pin or GPIOX (of the SoC) to the pin header.
>>>>
>>>> +---------+           +--------------+             +---+
>>>>           | I2C0_SDA  |              |             | H |
>>>>           |-----------|----- \       |             | E |
>>>>           |           |       \      |             | A |
>>>> Intel Soc |           |        \-----|-------------| D |
>>>>           | GPIOX     |              |             | E |
>>>>           |-----------|-----         |             | R |
>>>>           |           |    FPGA      |             |   |
>>>> ----------+           +--------------+             +---+
>>>>
>>>> The pin header looks like this:
>>>> +--------------------+--------------------+
>>>> |      3.3V          |       5V           |
>>>> | GPIO2 / I2C1_SDA   |       5V           |
>>>> | GPIO3 / I2C1_SCL   |       GND          |
>>>> | GPIO4 / ADC0       | GPIO14 / UART1_TX  |
>>>> |      GND           | GPIO15 / UART1_RX  |
>>>> | GPIO17 / UART1_RTS | GPIO18 / I2S_CLK   |
>>>> |     GPIO27         |       GND          |
>>>> |     GPIO22         |      GPIO23        |
>>>> |      3.3V          |      GPIO24        |
>>>> | GPIO10 / SPI_MOSI  |       GND          |
>>>> | GPIO9 / SPI_MISO   |      GPIO25        |
>>>> | GPIO11 / SPI_CLK   | GPIO8 / SPI_CS0    |
>>>> |      GND           | GPIO7 / SPI_CS1    |
>>>> | GPIO0 / I2C0_SDA   | GPIO1 / I2C0_SCL   |
>>>> |     GPIO5          |       GND          |
>>>> |     GPIO6          | GPIO12 / PWM0      |
>>>> | GPIO13 / PWM1      |       GND          |
>>>> | GPIO19 / I2S_FRM   | GPIO16 / UART1_CTS |
>>>> |     GPIO26         | GPIO20 / I2S_DIN   |
>>>> |      GND           | GPIO21 / I2S_DOUT  |
>>>> +--------------------+--------------------+
>>>>
>>>> The GPIOs in the pin header corresponds to the gpiochip I declare in
>>>> this driver.
>>>> So when I want to use a pin in GPIO mode, the upboard pinctrl driver
>>>> requests the corresponding SoC GPIO to the Intel pinctrl driver.
>>>> The SoC pins connected to the FPGA, are identified with "external" id.
>>>>
>>>> The hardware and the FPGA were designed in tandem, so you know for
>>>> example that for the GPIOX you need to request the Nth "external" GPIO.
>>>>
>>>> When you drive your GPIO, the upboard gpiochip manages in the same time
>>>> the direction of the "switch" and the value/direction of the
>>>> corresponding SoC pin.
>>>>
>>>> +------------------+         +--------------+             +---+
>>>>                    |---------|              |-------------| H |
>>>>                    |---------|   GPIOCHIP   |-------------| E |
>>>>    Intel gpiochip  |---------|              |-------------| A |
>>>>  provided by Intel |---------|    FPGA      |-------------| D |
>>>>   pinctrl driver   |---------|              |-------------| E |
>>>>                    |---------|              |-------------| R |
>>>>                    |---------|              |-------------|   |
>>>> +------------------+         +--------------+             +---+
>>>>
>>>>
>>>> About gpiochip_add_pinlist_range(), I added it because the FPGA pins
>>>> used by the gpiochip are not consecutive.
>>>>
>>>> Please let me know if it is not clear.
>>>> And sorry I'm not very good to make ascii art.
>>>
>>> I get it! We have a similar driver in the kernel already, look into:
>>> drivers/gpio/gpio-aggregator.c
>>>
>>> The aggregator abstraction is however just software. What you
>>> need here is a gpio-aggregator that adds some hardware
>>> control on top. But it has a very nice design using a bitmap
>>> to keep track of the GPIOs etc, and it supports operations
>>> on multiple GPIOs (many man-hours of hard coding and
>>> design went into that driver, ask Geert and Andy...)
>>>
>>> So I would proceed like this:
>>>
>>> - The pin control part of the driver looks sound, except
>>>   for the way you add ranges.
>>>
>>> - The gpiochip part needs to be refactored using the
>>>   ideas from gpio-aggregator.c.
>>>
>>> - Look closely at aggregator and see what you can do
>>>   based on that code, if you can mimic how it picks up
>>>   and forwards all GPIO functions. Maybe part of it
>>>   needs to be made into a library?
>>>  <linux/gpio/gpio-aggregator.h>?
>>>   For example if you start to feel like "I would really like
>>>   to just call gpio_fwd_get_multiple() then this is what
>>>   you want to do. The library can probably still be
>>>   inside gpio-aggregator.c the way we do it in
>>>   e.g. gpio-mmio.c, just export and keep library functions
>>>   separately.
>>
>> Hi Linus,
>>
>> Ok I think I understand what you expect.
>> I started to look at the gpio-aggregator code, play a bit with it, and
>> refactor it to use it from my driver.
>>
>> My main issue is about the request of the SoC GPIOs done by the aggregator.
>> If from my driver I call the aggregator library to create a gpiochip,
>> the SoC pins will be requested. So the SoC pins will be set in GPIO
>> mode, and the pins will never be in function mode.
>> There is no way to set the pins back to function mode (even if the GPIO
>> is free).
>>
>> I tried to add a feature in the aggregator to defer the request of the gpio.
>> So at the beginning of each ops the gpio_desc is checked. If it is
>> valid, the gpio can be used. Otherwise, the gpio is requested.
>> For example:
>>
>> gpio_fwd_get() {
>> 	if (!gpio_desc_is_valid(desc))
>> 		desc = request_gpio()
>>
>> 	return gpiod_get_value(desc)
>> }
>>
>> But when a gpiochip is registered, the core calls get_direction() or
>> direction_input(), so all GPIOs are requested and it does not solve my
>> problem.
>>
>> I expect to register a gpiochip without setting all pins in GPIO mode at
>> probe time (like all pinctrl driver do).
>> But I did not find a solution.
> 
> Basically what you need is a pinctrl-aggregattor (an analogue for the pin
> muxing and configuration).
> 

Hi Andy,

I found a trick to workaround the get_direction() issue in the
gpio-aggregator.

I added a "request on first use" feature on the aggregator.
The GPIO is requested during the request() operation of the fowarder.

static int gpio_fwd_request(struct gpio_chip *chip, unsigned int offset)
{
	struct gpiochip_fwd *fwd = gpiochip_get_data(chip);

	if (!IS_ERR_OR_NULL(fwd->descs[offset]))
		return 0;

	fwd->descs[offset] = devm_gpiod_get_index(fwd->dev, NULL,
						offset, GPIOD_ASIS);

	return PTR_ERR_OR_ZERO(fwd->descs[offset]);
}

The remaining problem is that the get_direction() callback is called
during gpiochip registration. For now if the gpio_desc is not valid (so
the GPIO was not yet requested) I return GPIO_LINE_DIRECTION_OUT by
default. But I'm not very convinced by this hack.
Maybe I could retrieve the gpio_chip and call its get_direction()
callback, but it seems not to be a better idea.

For the pinctrl-aggregator you mentioned, if I understand correctly the
idea to aggregate the SoC pins in a pinctrl aggregator (with a
gpio_chip) which just forwards gpio_request_enable(),
gpio_disable_free(), gpio_set_direction() and also all gpio_chip operations.

But how to deal with the pinctrl of my FPGA ? For one of its fake pin
the dummy pinctrl drives the corresponding SoC pin and FPGA pin ? So for
each pin, the aggregator may have multiple pins to drive ?

Was it your idea Andy ?

Other challenge is to retrieve all the pins to add in the
pinctrl-aggregator. As input I have only GPIO descriptors, but I guess
it should be feasible.

Best Regards,

Thomas
Andy Shevchenko Jan. 16, 2025, 9:12 a.m. UTC | #7
On Tue, Jan 14, 2025 at 11:28:26AM +0100, Thomas Richard wrote:
> On 1/13/25 10:46, Andy Shevchenko wrote:
> > On Fri, Jan 03, 2025 at 11:28:30AM +0100, Thomas Richard wrote:
> >> On 12/22/24 00:43, Linus Walleij wrote:
> >>> On Fri, Dec 20, 2024 at 2:50 PM Thomas Richard
> >>> <thomas.richard@bootlin.com> wrote:
> >>>
> >>>> Yes my cover letter was a bit short, and maybe some context was missing.
> >>>
> >>> The text and graphics below explain it very well, so please include them
> >>> into the commit message so we have it there!
> >>>
> >>>> This FPGA acts as a level shifter between the Intel SoC pins and the pin
> >>>> header, and also makes a kind of switch/mux.
> >>>
> >>> Since it's Intel we need to notify Andy to help out with this so that
> >>> it gets done in a way that works with how he think consumers
> >>> should interact with Intel pin control and GPIO.
> >>>
> >>>> +---------+         +--------------+             +---+
> >>>>           |         |              |             | H |
> >>>>           |---------|              |-------------| E |
> >>>>           |         |              |             | A |
> >>>> Intel Soc |---------|    FPGA      |-------------| D |
> >>>>           |         |              |             | E |
> >>>>           |---------|              |-------------| R |
> >>>>           |         |              |             |   |
> >>>> ----------+         +--------------+             +---+
> >>>>
> >>>>
> >>>> For most of the pins, the FPGA opens/closes a switch to enable/disable
> >>>> the access to the SoC pin from a pin header.
> >>>> Each "switch", has a direction flag that shall be set in tandem with the
> >>>> status of the SoC pin.
> >>>> For example, if the SoC pin is in PWM mode, the "switch" shall be
> >>>> configured in output direction.
> >>>> If the SoC pin is set in GPIO mode, the direction of the "switch" shall
> >>>> corresponds to the GPIO direction.
> >>>>
> >>>> +---------+              +--------------+             +---+
> >>>>           |              |              |             | H |
> >>>>           |              |      \       |             | E |
> >>>>           |   PWM1       |       \      |             | A |
> >>>> Intel Soc |--------------|-----   \-----|-------------| D |
> >>>>           |              |              |             | E |
> >>>>           |              |              |             | R |
> >>>>           |              |    FPGA      |             |   |
> >>>> ----------+              +--------------+             +---+
> >>>>
> >>>> (PWM1 pin from Intel SoC can be used as PWM, and also in GPIO mode,
> >>>> thanks to the Intel pinctrl driver).
> >>>>
> >>>> Few pins (PINMUX_* pins) work differently. The FPGA acts as a mux and
> >>>> routes for example the I2C0_SDA pin or GPIOX (of the SoC) to the pin header.

Yep, that's clear.

> >>>> +---------+           +--------------+             +---+
> >>>>           | I2C0_SDA  |              |             | H |
> >>>>           |-----------|----- \       |             | E |
> >>>>           |           |       \      |             | A |
> >>>> Intel Soc |           |        \-----|-------------| D |
> >>>>           | GPIOX     |              |             | E |
> >>>>           |-----------|-----         |             | R |
> >>>>           |           |    FPGA      |             |   |
> >>>> ----------+           +--------------+             +---+
> >>>>
> >>>> The pin header looks like this:
> >>>> +--------------------+--------------------+
> >>>> |      3.3V          |       5V           |
> >>>> | GPIO2 / I2C1_SDA   |       5V           |
> >>>> | GPIO3 / I2C1_SCL   |       GND          |
> >>>> | GPIO4 / ADC0       | GPIO14 / UART1_TX  |
> >>>> |      GND           | GPIO15 / UART1_RX  |
> >>>> | GPIO17 / UART1_RTS | GPIO18 / I2S_CLK   |
> >>>> |     GPIO27         |       GND          |
> >>>> |     GPIO22         |      GPIO23        |
> >>>> |      3.3V          |      GPIO24        |
> >>>> | GPIO10 / SPI_MOSI  |       GND          |
> >>>> | GPIO9 / SPI_MISO   |      GPIO25        |
> >>>> | GPIO11 / SPI_CLK   | GPIO8 / SPI_CS0    |
> >>>> |      GND           | GPIO7 / SPI_CS1    |
> >>>> | GPIO0 / I2C0_SDA   | GPIO1 / I2C0_SCL   |
> >>>> |     GPIO5          |       GND          |
> >>>> |     GPIO6          | GPIO12 / PWM0      |
> >>>> | GPIO13 / PWM1      |       GND          |
> >>>> | GPIO19 / I2S_FRM   | GPIO16 / UART1_CTS |
> >>>> |     GPIO26         | GPIO20 / I2S_DIN   |
> >>>> |      GND           | GPIO21 / I2S_DOUT  |
> >>>> +--------------------+--------------------+
> >>>>
> >>>> The GPIOs in the pin header corresponds to the gpiochip I declare in
> >>>> this driver.
> >>>> So when I want to use a pin in GPIO mode, the upboard pinctrl driver
> >>>> requests the corresponding SoC GPIO to the Intel pinctrl driver.
> >>>> The SoC pins connected to the FPGA, are identified with "external" id.
> >>>>
> >>>> The hardware and the FPGA were designed in tandem, so you know for
> >>>> example that for the GPIOX you need to request the Nth "external" GPIO.
> >>>>
> >>>> When you drive your GPIO, the upboard gpiochip manages in the same time
> >>>> the direction of the "switch" and the value/direction of the
> >>>> corresponding SoC pin.
> >>>>
> >>>> +------------------+         +--------------+             +---+
> >>>>                    |---------|              |-------------| H |
> >>>>                    |---------|   GPIOCHIP   |-------------| E |
> >>>>    Intel gpiochip  |---------|              |-------------| A |
> >>>>  provided by Intel |---------|    FPGA      |-------------| D |
> >>>>   pinctrl driver   |---------|              |-------------| E |
> >>>>                    |---------|              |-------------| R |
> >>>>                    |---------|              |-------------|   |
> >>>> +------------------+         +--------------+             +---+
> >>>>
> >>>> About gpiochip_add_pinlist_range(), I added it because the FPGA pins
> >>>> used by the gpiochip are not consecutive.
> >>>>
> >>>> Please let me know if it is not clear.
> >>>> And sorry I'm not very good to make ascii art.
> >>>
> >>> I get it! We have a similar driver in the kernel already, look into:
> >>> drivers/gpio/gpio-aggregator.c
> >>>
> >>> The aggregator abstraction is however just software. What you
> >>> need here is a gpio-aggregator that adds some hardware
> >>> control on top. But it has a very nice design using a bitmap
> >>> to keep track of the GPIOs etc, and it supports operations
> >>> on multiple GPIOs (many man-hours of hard coding and
> >>> design went into that driver, ask Geert and Andy...)
> >>>
> >>> So I would proceed like this:
> >>>
> >>> - The pin control part of the driver looks sound, except
> >>>   for the way you add ranges.
> >>>
> >>> - The gpiochip part needs to be refactored using the
> >>>   ideas from gpio-aggregator.c.
> >>>
> >>> - Look closely at aggregator and see what you can do
> >>>   based on that code, if you can mimic how it picks up
> >>>   and forwards all GPIO functions. Maybe part of it
> >>>   needs to be made into a library?
> >>>  <linux/gpio/gpio-aggregator.h>?
> >>>   For example if you start to feel like "I would really like
> >>>   to just call gpio_fwd_get_multiple() then this is what
> >>>   you want to do. The library can probably still be
> >>>   inside gpio-aggregator.c the way we do it in
> >>>   e.g. gpio-mmio.c, just export and keep library functions
> >>>   separately.
> >>
> >> Ok I think I understand what you expect.
> >> I started to look at the gpio-aggregator code, play a bit with it, and
> >> refactor it to use it from my driver.
> >>
> >> My main issue is about the request of the SoC GPIOs done by the aggregator.
> >> If from my driver I call the aggregator library to create a gpiochip,
> >> the SoC pins will be requested. So the SoC pins will be set in GPIO
> >> mode, and the pins will never be in function mode.
> >> There is no way to set the pins back to function mode (even if the GPIO
> >> is free).
> >>
> >> I tried to add a feature in the aggregator to defer the request of the gpio.
> >> So at the beginning of each ops the gpio_desc is checked. If it is
> >> valid, the gpio can be used. Otherwise, the gpio is requested.
> >> For example:
> >>
> >> gpio_fwd_get() {
> >> 	if (!gpio_desc_is_valid(desc))
> >> 		desc = request_gpio()
> >>
> >> 	return gpiod_get_value(desc)
> >> }
> >>
> >> But when a gpiochip is registered, the core calls get_direction() or
> >> direction_input(), so all GPIOs are requested and it does not solve my
> >> problem.
> >>
> >> I expect to register a gpiochip without setting all pins in GPIO mode at
> >> probe time (like all pinctrl driver do).
> >> But I did not find a solution.
> > 
> > Basically what you need is a pinctrl-aggregattor (an analogue for the pin
> > muxing and configuration).
> 
> I found a trick to workaround the get_direction() issue in the
> gpio-aggregator.
> 
> I added a "request on first use" feature on the aggregator.
> The GPIO is requested during the request() operation of the fowarder.
> 
> static int gpio_fwd_request(struct gpio_chip *chip, unsigned int offset)
> {
> 	struct gpiochip_fwd *fwd = gpiochip_get_data(chip);
> 
> 	if (!IS_ERR_OR_NULL(fwd->descs[offset]))
> 		return 0;
> 
> 	fwd->descs[offset] = devm_gpiod_get_index(fwd->dev, NULL,
> 						offset, GPIOD_ASIS);
> 
> 	return PTR_ERR_OR_ZERO(fwd->descs[offset]);
> }
> 
> The remaining problem is that the get_direction() callback is called
> during gpiochip registration. For now if the gpio_desc is not valid (so
> the GPIO was not yet requested) I return GPIO_LINE_DIRECTION_OUT by
> default. But I'm not very convinced by this hack.
> Maybe I could retrieve the gpio_chip and call its get_direction()
> callback, but it seems not to be a better idea.
> 
> For the pinctrl-aggregator you mentioned, if I understand correctly the
> idea to aggregate the SoC pins in a pinctrl aggregator (with a
> gpio_chip) which just forwards gpio_request_enable(),
> gpio_disable_free(), gpio_set_direction() and also all gpio_chip operations.

No only these, all of the pin mux, pin configuration, and GPIO operations
should be proxied.

> But how to deal with the pinctrl of my FPGA ? For one of its fake pin
> the dummy pinctrl drives the corresponding SoC pin and FPGA pin ?

What's the "fake pin"? I can't find the one up in your schemas.

> So for each pin, the aggregator may have multiple pins to drive ?

Depending on the case, but yes. Currently we have implementations of
the discrete pin controls on Intel platforms based on ACPI (see Intel Minnow,
Intel Galileo, Intel Edison/Arduino boards in meta-acpi project [1]).
You probably want something similar to be written in C.

> Was it your idea Andy ?
> 
> Other challenge is to retrieve all the pins to add in the
> pinctrl-aggregator. As input I have only GPIO descriptors, but I guess
> it should be feasible.

In general your "proxy" (FPGA) pin control driver should be consumer of all SoC
pins (and their respective muxing and configurations) and be provider of the
pins that are available to the user.

[1]: https://github.com/westeri/meta-acpi/tree/master/recipes-bsp/acpi-tables/samples
Thomas Richard Jan. 16, 2025, 12:21 p.m. UTC | #8
On 1/16/25 10:12, Andy Shevchenko wrote:
> On Tue, Jan 14, 2025 at 11:28:26AM +0100, Thomas Richard wrote:
>> On 1/13/25 10:46, Andy Shevchenko wrote:
>>> On Fri, Jan 03, 2025 at 11:28:30AM +0100, Thomas Richard wrote:
>>>> On 12/22/24 00:43, Linus Walleij wrote:
>>>>> On Fri, Dec 20, 2024 at 2:50 PM Thomas Richard
>>>>> <thomas.richard@bootlin.com> wrote:
>>>>>
>>>>>> Yes my cover letter was a bit short, and maybe some context was missing.
>>>>>
>>>>> The text and graphics below explain it very well, so please include them
>>>>> into the commit message so we have it there!
>>>>>
>>>>>> This FPGA acts as a level shifter between the Intel SoC pins and the pin
>>>>>> header, and also makes a kind of switch/mux.
>>>>>
>>>>> Since it's Intel we need to notify Andy to help out with this so that
>>>>> it gets done in a way that works with how he think consumers
>>>>> should interact with Intel pin control and GPIO.
>>>>>
>>>>>> +---------+         +--------------+             +---+
>>>>>>           |         |              |             | H |
>>>>>>           |---------|              |-------------| E |
>>>>>>           |         |              |             | A |
>>>>>> Intel Soc |---------|    FPGA      |-------------| D |
>>>>>>           |         |              |             | E |
>>>>>>           |---------|              |-------------| R |
>>>>>>           |         |              |             |   |
>>>>>> ----------+         +--------------+             +---+
>>>>>>
>>>>>>
>>>>>> For most of the pins, the FPGA opens/closes a switch to enable/disable
>>>>>> the access to the SoC pin from a pin header.
>>>>>> Each "switch", has a direction flag that shall be set in tandem with the
>>>>>> status of the SoC pin.
>>>>>> For example, if the SoC pin is in PWM mode, the "switch" shall be
>>>>>> configured in output direction.
>>>>>> If the SoC pin is set in GPIO mode, the direction of the "switch" shall
>>>>>> corresponds to the GPIO direction.
>>>>>>
>>>>>> +---------+              +--------------+             +---+
>>>>>>           |              |              |             | H |
>>>>>>           |              |      \       |             | E |
>>>>>>           |   PWM1       |       \      |             | A |
>>>>>> Intel Soc |--------------|-----   \-----|-------------| D |
>>>>>>           |              |              |             | E |
>>>>>>           |              |              |             | R |
>>>>>>           |              |    FPGA      |             |   |
>>>>>> ----------+              +--------------+             +---+
>>>>>>
>>>>>> (PWM1 pin from Intel SoC can be used as PWM, and also in GPIO mode,
>>>>>> thanks to the Intel pinctrl driver).
>>>>>>
>>>>>> Few pins (PINMUX_* pins) work differently. The FPGA acts as a mux and
>>>>>> routes for example the I2C0_SDA pin or GPIOX (of the SoC) to the pin header.
> 
> Yep, that's clear.
> 
>>>>>> +---------+           +--------------+             +---+
>>>>>>           | I2C0_SDA  |              |             | H |
>>>>>>           |-----------|----- \       |             | E |
>>>>>>           |           |       \      |             | A |
>>>>>> Intel Soc |           |        \-----|-------------| D |
>>>>>>           | GPIOX     |              |             | E |
>>>>>>           |-----------|-----         |             | R |
>>>>>>           |           |    FPGA      |             |   |
>>>>>> ----------+           +--------------+             +---+
>>>>>>
>>>>>> The pin header looks like this:
>>>>>> +--------------------+--------------------+
>>>>>> |      3.3V          |       5V           |
>>>>>> | GPIO2 / I2C1_SDA   |       5V           |
>>>>>> | GPIO3 / I2C1_SCL   |       GND          |
>>>>>> | GPIO4 / ADC0       | GPIO14 / UART1_TX  |
>>>>>> |      GND           | GPIO15 / UART1_RX  |
>>>>>> | GPIO17 / UART1_RTS | GPIO18 / I2S_CLK   |
>>>>>> |     GPIO27         |       GND          |
>>>>>> |     GPIO22         |      GPIO23        |
>>>>>> |      3.3V          |      GPIO24        |
>>>>>> | GPIO10 / SPI_MOSI  |       GND          |
>>>>>> | GPIO9 / SPI_MISO   |      GPIO25        |
>>>>>> | GPIO11 / SPI_CLK   | GPIO8 / SPI_CS0    |
>>>>>> |      GND           | GPIO7 / SPI_CS1    |
>>>>>> | GPIO0 / I2C0_SDA   | GPIO1 / I2C0_SCL   |
>>>>>> |     GPIO5          |       GND          |
>>>>>> |     GPIO6          | GPIO12 / PWM0      |
>>>>>> | GPIO13 / PWM1      |       GND          |
>>>>>> | GPIO19 / I2S_FRM   | GPIO16 / UART1_CTS |
>>>>>> |     GPIO26         | GPIO20 / I2S_DIN   |
>>>>>> |      GND           | GPIO21 / I2S_DOUT  |
>>>>>> +--------------------+--------------------+
>>>>>>
>>>>>> The GPIOs in the pin header corresponds to the gpiochip I declare in
>>>>>> this driver.
>>>>>> So when I want to use a pin in GPIO mode, the upboard pinctrl driver
>>>>>> requests the corresponding SoC GPIO to the Intel pinctrl driver.
>>>>>> The SoC pins connected to the FPGA, are identified with "external" id.
>>>>>>
>>>>>> The hardware and the FPGA were designed in tandem, so you know for
>>>>>> example that for the GPIOX you need to request the Nth "external" GPIO.
>>>>>>
>>>>>> When you drive your GPIO, the upboard gpiochip manages in the same time
>>>>>> the direction of the "switch" and the value/direction of the
>>>>>> corresponding SoC pin.
>>>>>>
>>>>>> +------------------+         +--------------+             +---+
>>>>>>                    |---------|              |-------------| H |
>>>>>>                    |---------|   GPIOCHIP   |-------------| E |
>>>>>>    Intel gpiochip  |---------|              |-------------| A |
>>>>>>  provided by Intel |---------|    FPGA      |-------------| D |
>>>>>>   pinctrl driver   |---------|              |-------------| E |
>>>>>>                    |---------|              |-------------| R |
>>>>>>                    |---------|              |-------------|   |
>>>>>> +------------------+         +--------------+             +---+
>>>>>>
>>>>>> About gpiochip_add_pinlist_range(), I added it because the FPGA pins
>>>>>> used by the gpiochip are not consecutive.
>>>>>>
>>>>>> Please let me know if it is not clear.
>>>>>> And sorry I'm not very good to make ascii art.
>>>>>
>>>>> I get it! We have a similar driver in the kernel already, look into:
>>>>> drivers/gpio/gpio-aggregator.c
>>>>>
>>>>> The aggregator abstraction is however just software. What you
>>>>> need here is a gpio-aggregator that adds some hardware
>>>>> control on top. But it has a very nice design using a bitmap
>>>>> to keep track of the GPIOs etc, and it supports operations
>>>>> on multiple GPIOs (many man-hours of hard coding and
>>>>> design went into that driver, ask Geert and Andy...)
>>>>>
>>>>> So I would proceed like this:
>>>>>
>>>>> - The pin control part of the driver looks sound, except
>>>>>   for the way you add ranges.
>>>>>
>>>>> - The gpiochip part needs to be refactored using the
>>>>>   ideas from gpio-aggregator.c.
>>>>>
>>>>> - Look closely at aggregator and see what you can do
>>>>>   based on that code, if you can mimic how it picks up
>>>>>   and forwards all GPIO functions. Maybe part of it
>>>>>   needs to be made into a library?
>>>>>  <linux/gpio/gpio-aggregator.h>?
>>>>>   For example if you start to feel like "I would really like
>>>>>   to just call gpio_fwd_get_multiple() then this is what
>>>>>   you want to do. The library can probably still be
>>>>>   inside gpio-aggregator.c the way we do it in
>>>>>   e.g. gpio-mmio.c, just export and keep library functions
>>>>>   separately.
>>>>
>>>> Ok I think I understand what you expect.
>>>> I started to look at the gpio-aggregator code, play a bit with it, and
>>>> refactor it to use it from my driver.
>>>>
>>>> My main issue is about the request of the SoC GPIOs done by the aggregator.
>>>> If from my driver I call the aggregator library to create a gpiochip,
>>>> the SoC pins will be requested. So the SoC pins will be set in GPIO
>>>> mode, and the pins will never be in function mode.
>>>> There is no way to set the pins back to function mode (even if the GPIO
>>>> is free).
>>>>
>>>> I tried to add a feature in the aggregator to defer the request of the gpio.
>>>> So at the beginning of each ops the gpio_desc is checked. If it is
>>>> valid, the gpio can be used. Otherwise, the gpio is requested.
>>>> For example:
>>>>
>>>> gpio_fwd_get() {
>>>> 	if (!gpio_desc_is_valid(desc))
>>>> 		desc = request_gpio()
>>>>
>>>> 	return gpiod_get_value(desc)
>>>> }
>>>>
>>>> But when a gpiochip is registered, the core calls get_direction() or
>>>> direction_input(), so all GPIOs are requested and it does not solve my
>>>> problem.
>>>>
>>>> I expect to register a gpiochip without setting all pins in GPIO mode at
>>>> probe time (like all pinctrl driver do).
>>>> But I did not find a solution.
>>>
>>> Basically what you need is a pinctrl-aggregattor (an analogue for the pin
>>> muxing and configuration).
>>
>> I found a trick to workaround the get_direction() issue in the
>> gpio-aggregator.
>>
>> I added a "request on first use" feature on the aggregator.
>> The GPIO is requested during the request() operation of the fowarder.
>>
>> static int gpio_fwd_request(struct gpio_chip *chip, unsigned int offset)
>> {
>> 	struct gpiochip_fwd *fwd = gpiochip_get_data(chip);
>>
>> 	if (!IS_ERR_OR_NULL(fwd->descs[offset]))
>> 		return 0;
>>
>> 	fwd->descs[offset] = devm_gpiod_get_index(fwd->dev, NULL,
>> 						offset, GPIOD_ASIS);
>>
>> 	return PTR_ERR_OR_ZERO(fwd->descs[offset]);
>> }
>>
>> The remaining problem is that the get_direction() callback is called
>> during gpiochip registration. For now if the gpio_desc is not valid (so
>> the GPIO was not yet requested) I return GPIO_LINE_DIRECTION_OUT by
>> default. But I'm not very convinced by this hack.
>> Maybe I could retrieve the gpio_chip and call its get_direction()
>> callback, but it seems not to be a better idea.
>>
>> For the pinctrl-aggregator you mentioned, if I understand correctly the
>> idea to aggregate the SoC pins in a pinctrl aggregator (with a
>> gpio_chip) which just forwards gpio_request_enable(),
>> gpio_disable_free(), gpio_set_direction() and also all gpio_chip operations.
> 
> No only these, all of the pin mux, pin configuration, and GPIO operations
> should be proxied.

Why should I proxy operations that will never be used (pin mux, pin
conf) ? I mean there will never be a driver that will configure FPGA pins.

> 
>> But how to deal with the pinctrl of my FPGA ? For one of its fake pin
>> the dummy pinctrl drives the corresponding SoC pin and FPGA pin ?
> 
> What's the "fake pin"? I can't find the one up in your schemas.

I thought that the idea was a virtual pinctrl which drive the SoC
pinctrl and the FPGA pinctrl. But what you mean is the FPGA pinctrl acts
as a proxy.

> 
>> So for each pin, the aggregator may have multiple pins to drive ?
> 
> Depending on the case, but yes. Currently we have implementations of
> the discrete pin controls on Intel platforms based on ACPI (see Intel Minnow,
> Intel Galileo, Intel Edison/Arduino boards in meta-acpi project [1]).
> You probably want something similar to be written in C.
> 
>> Was it your idea Andy ?
>>
>> Other challenge is to retrieve all the pins to add in the
>> pinctrl-aggregator. As input I have only GPIO descriptors, but I guess
>> it should be feasible.
> 
> In general your "proxy" (FPGA) pin control driver should be consumer of all SoC
> pins (and their respective muxing and configurations) and be provider of the
> pins that are available to the user.

Isn't that a bit over-engineered for my use case ?

For the pinconf / pinmux, the FPGA is just a voltage translator. It is
transparent. The only relevant thing for the FPGA is the direction to
set for the switch of each pin. And the drivers knows which directions
to apply during the probe. This direction will only change in GPIO mode,
but in GPIO mode we know which direction to set.

For the GPIO part, the FPGA provides GPIOs. And in this case it is a
consumer of SoC GPIOs, it is already a kind of aggregator.

> 
> [1]: https://github.com/westeri/meta-acpi/tree/master/recipes-bsp/acpi-tables/samples
> 

Best Regards,

Thomas
Andy Shevchenko Jan. 16, 2025, 12:31 p.m. UTC | #9
On Thu, Jan 16, 2025 at 01:21:16PM +0100, Thomas Richard wrote:
> On 1/16/25 10:12, Andy Shevchenko wrote:
> > On Tue, Jan 14, 2025 at 11:28:26AM +0100, Thomas Richard wrote:
> >> On 1/13/25 10:46, Andy Shevchenko wrote:
> >>> On Fri, Jan 03, 2025 at 11:28:30AM +0100, Thomas Richard wrote:
> >>>> On 12/22/24 00:43, Linus Walleij wrote:
> >>>>> On Fri, Dec 20, 2024 at 2:50 PM Thomas Richard
> >>>>> <thomas.richard@bootlin.com> wrote:

> >>>>>> Yes my cover letter was a bit short, and maybe some context was missing.
> >>>>>
> >>>>> The text and graphics below explain it very well, so please include them
> >>>>> into the commit message so we have it there!
> >>>>>
> >>>>>> This FPGA acts as a level shifter between the Intel SoC pins and the pin
> >>>>>> header, and also makes a kind of switch/mux.
> >>>>>
> >>>>> Since it's Intel we need to notify Andy to help out with this so that
> >>>>> it gets done in a way that works with how he think consumers
> >>>>> should interact with Intel pin control and GPIO.
> >>>>>
> >>>>>> +---------+         +--------------+             +---+
> >>>>>>           |         |              |             | H |
> >>>>>>           |---------|              |-------------| E |
> >>>>>>           |         |              |             | A |
> >>>>>> Intel Soc |---------|    FPGA      |-------------| D |
> >>>>>>           |         |              |             | E |
> >>>>>>           |---------|              |-------------| R |
> >>>>>>           |         |              |             |   |
> >>>>>> ----------+         +--------------+             +---+
> >>>>>>
> >>>>>>
> >>>>>> For most of the pins, the FPGA opens/closes a switch to enable/disable
> >>>>>> the access to the SoC pin from a pin header.
> >>>>>> Each "switch", has a direction flag that shall be set in tandem with the
> >>>>>> status of the SoC pin.
> >>>>>> For example, if the SoC pin is in PWM mode, the "switch" shall be
> >>>>>> configured in output direction.
> >>>>>> If the SoC pin is set in GPIO mode, the direction of the "switch" shall
> >>>>>> corresponds to the GPIO direction.
> >>>>>>
> >>>>>> +---------+              +--------------+             +---+
> >>>>>>           |              |              |             | H |
> >>>>>>           |              |      \       |             | E |
> >>>>>>           |   PWM1       |       \      |             | A |
> >>>>>> Intel Soc |--------------|-----   \-----|-------------| D |
> >>>>>>           |              |              |             | E |
> >>>>>>           |              |              |             | R |
> >>>>>>           |              |    FPGA      |             |   |
> >>>>>> ----------+              +--------------+             +---+
> >>>>>>
> >>>>>> (PWM1 pin from Intel SoC can be used as PWM, and also in GPIO mode,
> >>>>>> thanks to the Intel pinctrl driver).
> >>>>>>
> >>>>>> Few pins (PINMUX_* pins) work differently. The FPGA acts as a mux and
> >>>>>> routes for example the I2C0_SDA pin or GPIOX (of the SoC) to the pin header.
> > 
> > Yep, that's clear.
> > 
> >>>>>> +---------+           +--------------+             +---+
> >>>>>>           | I2C0_SDA  |              |             | H |
> >>>>>>           |-----------|----- \       |             | E |
> >>>>>>           |           |       \      |             | A |
> >>>>>> Intel Soc |           |        \-----|-------------| D |
> >>>>>>           | GPIOX     |              |             | E |
> >>>>>>           |-----------|-----         |             | R |
> >>>>>>           |           |    FPGA      |             |   |
> >>>>>> ----------+           +--------------+             +---+
> >>>>>>
> >>>>>> The pin header looks like this:
> >>>>>> +--------------------+--------------------+
> >>>>>> |      3.3V          |       5V           |
> >>>>>> | GPIO2 / I2C1_SDA   |       5V           |
> >>>>>> | GPIO3 / I2C1_SCL   |       GND          |
> >>>>>> | GPIO4 / ADC0       | GPIO14 / UART1_TX  |
> >>>>>> |      GND           | GPIO15 / UART1_RX  |
> >>>>>> | GPIO17 / UART1_RTS | GPIO18 / I2S_CLK   |
> >>>>>> |     GPIO27         |       GND          |
> >>>>>> |     GPIO22         |      GPIO23        |
> >>>>>> |      3.3V          |      GPIO24        |
> >>>>>> | GPIO10 / SPI_MOSI  |       GND          |
> >>>>>> | GPIO9 / SPI_MISO   |      GPIO25        |
> >>>>>> | GPIO11 / SPI_CLK   | GPIO8 / SPI_CS0    |
> >>>>>> |      GND           | GPIO7 / SPI_CS1    |
> >>>>>> | GPIO0 / I2C0_SDA   | GPIO1 / I2C0_SCL   |
> >>>>>> |     GPIO5          |       GND          |
> >>>>>> |     GPIO6          | GPIO12 / PWM0      |
> >>>>>> | GPIO13 / PWM1      |       GND          |
> >>>>>> | GPIO19 / I2S_FRM   | GPIO16 / UART1_CTS |
> >>>>>> |     GPIO26         | GPIO20 / I2S_DIN   |
> >>>>>> |      GND           | GPIO21 / I2S_DOUT  |
> >>>>>> +--------------------+--------------------+
> >>>>>>
> >>>>>> The GPIOs in the pin header corresponds to the gpiochip I declare in
> >>>>>> this driver.
> >>>>>> So when I want to use a pin in GPIO mode, the upboard pinctrl driver
> >>>>>> requests the corresponding SoC GPIO to the Intel pinctrl driver.
> >>>>>> The SoC pins connected to the FPGA, are identified with "external" id.
> >>>>>>
> >>>>>> The hardware and the FPGA were designed in tandem, so you know for
> >>>>>> example that for the GPIOX you need to request the Nth "external" GPIO.
> >>>>>>
> >>>>>> When you drive your GPIO, the upboard gpiochip manages in the same time
> >>>>>> the direction of the "switch" and the value/direction of the
> >>>>>> corresponding SoC pin.
> >>>>>>
> >>>>>> +------------------+         +--------------+             +---+
> >>>>>>                    |---------|              |-------------| H |
> >>>>>>                    |---------|   GPIOCHIP   |-------------| E |
> >>>>>>    Intel gpiochip  |---------|              |-------------| A |
> >>>>>>  provided by Intel |---------|    FPGA      |-------------| D |
> >>>>>>   pinctrl driver   |---------|              |-------------| E |
> >>>>>>                    |---------|              |-------------| R |
> >>>>>>                    |---------|              |-------------|   |
> >>>>>> +------------------+         +--------------+             +---+
> >>>>>>
> >>>>>> About gpiochip_add_pinlist_range(), I added it because the FPGA pins
> >>>>>> used by the gpiochip are not consecutive.
> >>>>>>
> >>>>>> Please let me know if it is not clear.
> >>>>>> And sorry I'm not very good to make ascii art.
> >>>>>
> >>>>> I get it! We have a similar driver in the kernel already, look into:
> >>>>> drivers/gpio/gpio-aggregator.c
> >>>>>
> >>>>> The aggregator abstraction is however just software. What you
> >>>>> need here is a gpio-aggregator that adds some hardware
> >>>>> control on top. But it has a very nice design using a bitmap
> >>>>> to keep track of the GPIOs etc, and it supports operations
> >>>>> on multiple GPIOs (many man-hours of hard coding and
> >>>>> design went into that driver, ask Geert and Andy...)
> >>>>>
> >>>>> So I would proceed like this:
> >>>>>
> >>>>> - The pin control part of the driver looks sound, except
> >>>>>   for the way you add ranges.
> >>>>>
> >>>>> - The gpiochip part needs to be refactored using the
> >>>>>   ideas from gpio-aggregator.c.
> >>>>>
> >>>>> - Look closely at aggregator and see what you can do
> >>>>>   based on that code, if you can mimic how it picks up
> >>>>>   and forwards all GPIO functions. Maybe part of it
> >>>>>   needs to be made into a library?
> >>>>>  <linux/gpio/gpio-aggregator.h>?
> >>>>>   For example if you start to feel like "I would really like
> >>>>>   to just call gpio_fwd_get_multiple() then this is what
> >>>>>   you want to do. The library can probably still be
> >>>>>   inside gpio-aggregator.c the way we do it in
> >>>>>   e.g. gpio-mmio.c, just export and keep library functions
> >>>>>   separately.
> >>>>
> >>>> Ok I think I understand what you expect.
> >>>> I started to look at the gpio-aggregator code, play a bit with it, and
> >>>> refactor it to use it from my driver.
> >>>>
> >>>> My main issue is about the request of the SoC GPIOs done by the aggregator.
> >>>> If from my driver I call the aggregator library to create a gpiochip,
> >>>> the SoC pins will be requested. So the SoC pins will be set in GPIO
> >>>> mode, and the pins will never be in function mode.
> >>>> There is no way to set the pins back to function mode (even if the GPIO
> >>>> is free).
> >>>>
> >>>> I tried to add a feature in the aggregator to defer the request of the gpio.
> >>>> So at the beginning of each ops the gpio_desc is checked. If it is
> >>>> valid, the gpio can be used. Otherwise, the gpio is requested.
> >>>> For example:
> >>>>
> >>>> gpio_fwd_get() {
> >>>> 	if (!gpio_desc_is_valid(desc))
> >>>> 		desc = request_gpio()
> >>>>
> >>>> 	return gpiod_get_value(desc)
> >>>> }
> >>>>
> >>>> But when a gpiochip is registered, the core calls get_direction() or
> >>>> direction_input(), so all GPIOs are requested and it does not solve my
> >>>> problem.
> >>>>
> >>>> I expect to register a gpiochip without setting all pins in GPIO mode at
> >>>> probe time (like all pinctrl driver do).
> >>>> But I did not find a solution.
> >>>
> >>> Basically what you need is a pinctrl-aggregattor (an analogue for the pin
> >>> muxing and configuration).
> >>
> >> I found a trick to workaround the get_direction() issue in the
> >> gpio-aggregator.
> >>
> >> I added a "request on first use" feature on the aggregator.
> >> The GPIO is requested during the request() operation of the fowarder.
> >>
> >> static int gpio_fwd_request(struct gpio_chip *chip, unsigned int offset)
> >> {
> >> 	struct gpiochip_fwd *fwd = gpiochip_get_data(chip);
> >>
> >> 	if (!IS_ERR_OR_NULL(fwd->descs[offset]))
> >> 		return 0;
> >>
> >> 	fwd->descs[offset] = devm_gpiod_get_index(fwd->dev, NULL,
> >> 						offset, GPIOD_ASIS);
> >>
> >> 	return PTR_ERR_OR_ZERO(fwd->descs[offset]);
> >> }
> >>
> >> The remaining problem is that the get_direction() callback is called
> >> during gpiochip registration. For now if the gpio_desc is not valid (so
> >> the GPIO was not yet requested) I return GPIO_LINE_DIRECTION_OUT by
> >> default. But I'm not very convinced by this hack.
> >> Maybe I could retrieve the gpio_chip and call its get_direction()
> >> callback, but it seems not to be a better idea.
> >>
> >> For the pinctrl-aggregator you mentioned, if I understand correctly the
> >> idea to aggregate the SoC pins in a pinctrl aggregator (with a
> >> gpio_chip) which just forwards gpio_request_enable(),
> >> gpio_disable_free(), gpio_set_direction() and also all gpio_chip operations.
> > 
> > No only these, all of the pin mux, pin configuration, and GPIO operations
> > should be proxied.
> 
> Why should I proxy operations that will never be used (pin mux, pin
> conf) ? I mean there will never be a driver that will configure FPGA pins.

According to your picture above the FPGA works as a pin function selector,
which also implies different pin configuration (e.g., slew rate for I²C pins).
Did I get it wrong?

> >> But how to deal with the pinctrl of my FPGA ? For one of its fake pin
> >> the dummy pinctrl drives the corresponding SoC pin and FPGA pin ?
> > 
> > What's the "fake pin"? I can't find the one up in your schemas.
> 
> I thought that the idea was a virtual pinctrl which drive the SoC
> pinctrl and the FPGA pinctrl. But what you mean is the FPGA pinctrl acts
> as a proxy.

When one wants to configure the pin parameters (let's say pin bias) the driver
delegates that to the SoC, in case the FPGA doesn't repeat this itself. Or both,
but this makes things too complicated already.

> >> So for each pin, the aggregator may have multiple pins to drive ?
> > 
> > Depending on the case, but yes. Currently we have implementations of
> > the discrete pin controls on Intel platforms based on ACPI (see Intel Minnow,
> > Intel Galileo, Intel Edison/Arduino boards in meta-acpi project [1]).
> > You probably want something similar to be written in C.
> > 
> >> Was it your idea Andy ?
> >>
> >> Other challenge is to retrieve all the pins to add in the
> >> pinctrl-aggregator. As input I have only GPIO descriptors, but I guess
> >> it should be feasible.
> > 
> > In general your "proxy" (FPGA) pin control driver should be consumer of all SoC
> > pins (and their respective muxing and configurations) and be provider of the
> > pins that are available to the user.
> 
> Isn't that a bit over-engineered for my use case ?

Yes, but it's already over-engineered in the HW, no?

> For the pinconf / pinmux, the FPGA is just a voltage translator.

It doesn't correspond to your picture where the pin function selector is depicted.

> It is transparent. The only relevant thing for the FPGA is the direction to
> set for the switch of each pin. And the drivers knows which directions
> to apply during the probe. This direction will only change in GPIO mode,
> but in GPIO mode we know which direction to set.
> 
> For the GPIO part, the FPGA provides GPIOs. And in this case it is a
> consumer of SoC GPIOs, it is already a kind of aggregator.
> 
> > [1]: https://github.com/westeri/meta-acpi/tree/master/recipes-bsp/acpi-tables/samples
Thomas Richard Jan. 16, 2025, 1:23 p.m. UTC | #10
On 1/16/25 13:31, Andy Shevchenko wrote:
> On Thu, Jan 16, 2025 at 01:21:16PM +0100, Thomas Richard wrote:
>> On 1/16/25 10:12, Andy Shevchenko wrote:
>>> On Tue, Jan 14, 2025 at 11:28:26AM +0100, Thomas Richard wrote:
>>>> On 1/13/25 10:46, Andy Shevchenko wrote:
>>>>> On Fri, Jan 03, 2025 at 11:28:30AM +0100, Thomas Richard wrote:
>>>>>> On 12/22/24 00:43, Linus Walleij wrote:
>>>>>>> On Fri, Dec 20, 2024 at 2:50 PM Thomas Richard
>>>>>>> <thomas.richard@bootlin.com> wrote:
> 
>>>>>>>> Yes my cover letter was a bit short, and maybe some context was missing.
>>>>>>>
>>>>>>> The text and graphics below explain it very well, so please include them
>>>>>>> into the commit message so we have it there!
>>>>>>>
>>>>>>>> This FPGA acts as a level shifter between the Intel SoC pins and the pin
>>>>>>>> header, and also makes a kind of switch/mux.
>>>>>>>
>>>>>>> Since it's Intel we need to notify Andy to help out with this so that
>>>>>>> it gets done in a way that works with how he think consumers
>>>>>>> should interact with Intel pin control and GPIO.
>>>>>>>
>>>>>>>> +---------+         +--------------+             +---+
>>>>>>>>           |         |              |             | H |
>>>>>>>>           |---------|              |-------------| E |
>>>>>>>>           |         |              |             | A |
>>>>>>>> Intel Soc |---------|    FPGA      |-------------| D |
>>>>>>>>           |         |              |             | E |
>>>>>>>>           |---------|              |-------------| R |
>>>>>>>>           |         |              |             |   |
>>>>>>>> ----------+         +--------------+             +---+
>>>>>>>>
>>>>>>>>
>>>>>>>> For most of the pins, the FPGA opens/closes a switch to enable/disable
>>>>>>>> the access to the SoC pin from a pin header.
>>>>>>>> Each "switch", has a direction flag that shall be set in tandem with the
>>>>>>>> status of the SoC pin.
>>>>>>>> For example, if the SoC pin is in PWM mode, the "switch" shall be
>>>>>>>> configured in output direction.
>>>>>>>> If the SoC pin is set in GPIO mode, the direction of the "switch" shall
>>>>>>>> corresponds to the GPIO direction.
>>>>>>>>
>>>>>>>> +---------+              +--------------+             +---+
>>>>>>>>           |              |              |             | H |
>>>>>>>>           |              |      \       |             | E |
>>>>>>>>           |   PWM1       |       \      |             | A |
>>>>>>>> Intel Soc |--------------|-----   \-----|-------------| D |
>>>>>>>>           |              |              |             | E |
>>>>>>>>           |              |              |             | R |
>>>>>>>>           |              |    FPGA      |             |   |
>>>>>>>> ----------+              +--------------+             +---+
>>>>>>>>
>>>>>>>> (PWM1 pin from Intel SoC can be used as PWM, and also in GPIO mode,
>>>>>>>> thanks to the Intel pinctrl driver).
>>>>>>>>
>>>>>>>> Few pins (PINMUX_* pins) work differently. The FPGA acts as a mux and
>>>>>>>> routes for example the I2C0_SDA pin or GPIOX (of the SoC) to the pin header.
>>>
>>> Yep, that's clear.
>>>
>>>>>>>> +---------+           +--------------+             +---+
>>>>>>>>           | I2C0_SDA  |              |             | H |
>>>>>>>>           |-----------|----- \       |             | E |
>>>>>>>>           |           |       \      |             | A |
>>>>>>>> Intel Soc |           |        \-----|-------------| D |
>>>>>>>>           | GPIOX     |              |             | E |
>>>>>>>>           |-----------|-----         |             | R |
>>>>>>>>           |           |    FPGA      |             |   |
>>>>>>>> ----------+           +--------------+             +---+
>>>>>>>>
>>>>>>>> The pin header looks like this:
>>>>>>>> +--------------------+--------------------+
>>>>>>>> |      3.3V          |       5V           |
>>>>>>>> | GPIO2 / I2C1_SDA   |       5V           |
>>>>>>>> | GPIO3 / I2C1_SCL   |       GND          |
>>>>>>>> | GPIO4 / ADC0       | GPIO14 / UART1_TX  |
>>>>>>>> |      GND           | GPIO15 / UART1_RX  |
>>>>>>>> | GPIO17 / UART1_RTS | GPIO18 / I2S_CLK   |
>>>>>>>> |     GPIO27         |       GND          |
>>>>>>>> |     GPIO22         |      GPIO23        |
>>>>>>>> |      3.3V          |      GPIO24        |
>>>>>>>> | GPIO10 / SPI_MOSI  |       GND          |
>>>>>>>> | GPIO9 / SPI_MISO   |      GPIO25        |
>>>>>>>> | GPIO11 / SPI_CLK   | GPIO8 / SPI_CS0    |
>>>>>>>> |      GND           | GPIO7 / SPI_CS1    |
>>>>>>>> | GPIO0 / I2C0_SDA   | GPIO1 / I2C0_SCL   |
>>>>>>>> |     GPIO5          |       GND          |
>>>>>>>> |     GPIO6          | GPIO12 / PWM0      |
>>>>>>>> | GPIO13 / PWM1      |       GND          |
>>>>>>>> | GPIO19 / I2S_FRM   | GPIO16 / UART1_CTS |
>>>>>>>> |     GPIO26         | GPIO20 / I2S_DIN   |
>>>>>>>> |      GND           | GPIO21 / I2S_DOUT  |
>>>>>>>> +--------------------+--------------------+
>>>>>>>>
>>>>>>>> The GPIOs in the pin header corresponds to the gpiochip I declare in
>>>>>>>> this driver.
>>>>>>>> So when I want to use a pin in GPIO mode, the upboard pinctrl driver
>>>>>>>> requests the corresponding SoC GPIO to the Intel pinctrl driver.
>>>>>>>> The SoC pins connected to the FPGA, are identified with "external" id.
>>>>>>>>
>>>>>>>> The hardware and the FPGA were designed in tandem, so you know for
>>>>>>>> example that for the GPIOX you need to request the Nth "external" GPIO.
>>>>>>>>
>>>>>>>> When you drive your GPIO, the upboard gpiochip manages in the same time
>>>>>>>> the direction of the "switch" and the value/direction of the
>>>>>>>> corresponding SoC pin.
>>>>>>>>
>>>>>>>> +------------------+         +--------------+             +---+
>>>>>>>>                    |---------|              |-------------| H |
>>>>>>>>                    |---------|   GPIOCHIP   |-------------| E |
>>>>>>>>    Intel gpiochip  |---------|              |-------------| A |
>>>>>>>>  provided by Intel |---------|    FPGA      |-------------| D |
>>>>>>>>   pinctrl driver   |---------|              |-------------| E |
>>>>>>>>                    |---------|              |-------------| R |
>>>>>>>>                    |---------|              |-------------|   |
>>>>>>>> +------------------+         +--------------+             +---+
>>>>>>>>
>>>>>>>> About gpiochip_add_pinlist_range(), I added it because the FPGA pins
>>>>>>>> used by the gpiochip are not consecutive.
>>>>>>>>
>>>>>>>> Please let me know if it is not clear.
>>>>>>>> And sorry I'm not very good to make ascii art.
>>>>>>>
>>>>>>> I get it! We have a similar driver in the kernel already, look into:
>>>>>>> drivers/gpio/gpio-aggregator.c
>>>>>>>
>>>>>>> The aggregator abstraction is however just software. What you
>>>>>>> need here is a gpio-aggregator that adds some hardware
>>>>>>> control on top. But it has a very nice design using a bitmap
>>>>>>> to keep track of the GPIOs etc, and it supports operations
>>>>>>> on multiple GPIOs (many man-hours of hard coding and
>>>>>>> design went into that driver, ask Geert and Andy...)
>>>>>>>
>>>>>>> So I would proceed like this:
>>>>>>>
>>>>>>> - The pin control part of the driver looks sound, except
>>>>>>>   for the way you add ranges.
>>>>>>>
>>>>>>> - The gpiochip part needs to be refactored using the
>>>>>>>   ideas from gpio-aggregator.c.
>>>>>>>
>>>>>>> - Look closely at aggregator and see what you can do
>>>>>>>   based on that code, if you can mimic how it picks up
>>>>>>>   and forwards all GPIO functions. Maybe part of it
>>>>>>>   needs to be made into a library?
>>>>>>>  <linux/gpio/gpio-aggregator.h>?
>>>>>>>   For example if you start to feel like "I would really like
>>>>>>>   to just call gpio_fwd_get_multiple() then this is what
>>>>>>>   you want to do. The library can probably still be
>>>>>>>   inside gpio-aggregator.c the way we do it in
>>>>>>>   e.g. gpio-mmio.c, just export and keep library functions
>>>>>>>   separately.
>>>>>>
>>>>>> Ok I think I understand what you expect.
>>>>>> I started to look at the gpio-aggregator code, play a bit with it, and
>>>>>> refactor it to use it from my driver.
>>>>>>
>>>>>> My main issue is about the request of the SoC GPIOs done by the aggregator.
>>>>>> If from my driver I call the aggregator library to create a gpiochip,
>>>>>> the SoC pins will be requested. So the SoC pins will be set in GPIO
>>>>>> mode, and the pins will never be in function mode.
>>>>>> There is no way to set the pins back to function mode (even if the GPIO
>>>>>> is free).
>>>>>>
>>>>>> I tried to add a feature in the aggregator to defer the request of the gpio.
>>>>>> So at the beginning of each ops the gpio_desc is checked. If it is
>>>>>> valid, the gpio can be used. Otherwise, the gpio is requested.
>>>>>> For example:
>>>>>>
>>>>>> gpio_fwd_get() {
>>>>>> 	if (!gpio_desc_is_valid(desc))
>>>>>> 		desc = request_gpio()
>>>>>>
>>>>>> 	return gpiod_get_value(desc)
>>>>>> }
>>>>>>
>>>>>> But when a gpiochip is registered, the core calls get_direction() or
>>>>>> direction_input(), so all GPIOs are requested and it does not solve my
>>>>>> problem.
>>>>>>
>>>>>> I expect to register a gpiochip without setting all pins in GPIO mode at
>>>>>> probe time (like all pinctrl driver do).
>>>>>> But I did not find a solution.
>>>>>
>>>>> Basically what you need is a pinctrl-aggregattor (an analogue for the pin
>>>>> muxing and configuration).
>>>>
>>>> I found a trick to workaround the get_direction() issue in the
>>>> gpio-aggregator.
>>>>
>>>> I added a "request on first use" feature on the aggregator.
>>>> The GPIO is requested during the request() operation of the fowarder.
>>>>
>>>> static int gpio_fwd_request(struct gpio_chip *chip, unsigned int offset)
>>>> {
>>>> 	struct gpiochip_fwd *fwd = gpiochip_get_data(chip);
>>>>
>>>> 	if (!IS_ERR_OR_NULL(fwd->descs[offset]))
>>>> 		return 0;
>>>>
>>>> 	fwd->descs[offset] = devm_gpiod_get_index(fwd->dev, NULL,
>>>> 						offset, GPIOD_ASIS);
>>>>
>>>> 	return PTR_ERR_OR_ZERO(fwd->descs[offset]);
>>>> }
>>>>
>>>> The remaining problem is that the get_direction() callback is called
>>>> during gpiochip registration. For now if the gpio_desc is not valid (so
>>>> the GPIO was not yet requested) I return GPIO_LINE_DIRECTION_OUT by
>>>> default. But I'm not very convinced by this hack.
>>>> Maybe I could retrieve the gpio_chip and call its get_direction()
>>>> callback, but it seems not to be a better idea.
>>>>
>>>> For the pinctrl-aggregator you mentioned, if I understand correctly the
>>>> idea to aggregate the SoC pins in a pinctrl aggregator (with a
>>>> gpio_chip) which just forwards gpio_request_enable(),
>>>> gpio_disable_free(), gpio_set_direction() and also all gpio_chip operations.
>>>
>>> No only these, all of the pin mux, pin configuration, and GPIO operations
>>> should be proxied.
>>
>> Why should I proxy operations that will never be used (pin mux, pin
>> conf) ? I mean there will never be a driver that will configure FPGA pins.
> 
> According to your picture above the FPGA works as a pin function selector,
> which also implies different pin configuration (e.g., slew rate for I²C pins).
> Did I get it wrong?

Yes you are right. In function mode the FPGA selects the I2C pin. When
the FPGA GPIO is requested, the FPGA changes the mux to select a SoC
GPIO pin (a pure GPIO pin).

> 
>>>> But how to deal with the pinctrl of my FPGA ? For one of its fake pin
>>>> the dummy pinctrl drives the corresponding SoC pin and FPGA pin ?
>>>
>>> What's the "fake pin"? I can't find the one up in your schemas.
>>
>> I thought that the idea was a virtual pinctrl which drive the SoC
>> pinctrl and the FPGA pinctrl. But what you mean is the FPGA pinctrl acts
>> as a proxy.
> 
> When one wants to configure the pin parameters (let's say pin bias) the driver
> delegates that to the SoC, in case the FPGA doesn't repeat this itself. Or both,
> but this makes things too complicated already.

The FPGA cannot configure pin parameter. It can only configures the
forward direction.
For example, for UART_TX pin the FPGA will set direction as "output" and
for UART_RX pin the FPGA will set direction as "input".
It can also enable/disable the forward of the SoC pin. If the forward of
a pin is disabled, the FPGA pin is in HIGH-Z.

> 
>>>> So for each pin, the aggregator may have multiple pins to drive ?
>>>
>>> Depending on the case, but yes. Currently we have implementations of
>>> the discrete pin controls on Intel platforms based on ACPI (see Intel Minnow,
>>> Intel Galileo, Intel Edison/Arduino boards in meta-acpi project [1]).
>>> You probably want something similar to be written in C.
>>>
>>>> Was it your idea Andy ?
>>>>
>>>> Other challenge is to retrieve all the pins to add in the
>>>> pinctrl-aggregator. As input I have only GPIO descriptors, but I guess
>>>> it should be feasible.
>>>
>>> In general your "proxy" (FPGA) pin control driver should be consumer of all SoC
>>> pins (and their respective muxing and configurations) and be provider of the
>>> pins that are available to the user.
>>
>> Isn't that a bit over-engineered for my use case ?
> 
> Yes, but it's already over-engineered in the HW, no?
> 
>> For the pinconf / pinmux, the FPGA is just a voltage translator.
> 
> It doesn't correspond to your picture where the pin function selector is depicted.

Yes you're right. I mean the mux can only select one function or GPIO mode.
I don't know why there is a mux to select a GPIO only pin, I'm pretty
sure the I2C pins can be set in GPIO mode.
It's probably for an hardware reason that a mux was added for only few pins.

> 
>> It is transparent. The only relevant thing for the FPGA is the direction to
>> set for the switch of each pin. And the drivers knows which directions
>> to apply during the probe. This direction will only change in GPIO mode,
>> but in GPIO mode we know which direction to set.
>>
>> For the GPIO part, the FPGA provides GPIOs. And in this case it is a
>> consumer of SoC GPIOs, it is already a kind of aggregator.
>>
>>> [1]: https://github.com/westeri/meta-acpi/tree/master/recipes-bsp/acpi-tables/samples
>
Andy Shevchenko Jan. 16, 2025, 2:14 p.m. UTC | #11
On Thu, Jan 16, 2025 at 02:23:57PM +0100, Thomas Richard wrote:
> On 1/16/25 13:31, Andy Shevchenko wrote:
> > On Thu, Jan 16, 2025 at 01:21:16PM +0100, Thomas Richard wrote:
> >> On 1/16/25 10:12, Andy Shevchenko wrote:
> >>> On Tue, Jan 14, 2025 at 11:28:26AM +0100, Thomas Richard wrote:
> >>>> On 1/13/25 10:46, Andy Shevchenko wrote:
> >>>>> On Fri, Jan 03, 2025 at 11:28:30AM +0100, Thomas Richard wrote:
> >>>>>> On 12/22/24 00:43, Linus Walleij wrote:
> >>>>>>> On Fri, Dec 20, 2024 at 2:50 PM Thomas Richard
> >>>>>>> <thomas.richard@bootlin.com> wrote:

...

> >>>>>>>> Yes my cover letter was a bit short, and maybe some context was missing.
> >>>>>>>
> >>>>>>> The text and graphics below explain it very well, so please include them
> >>>>>>> into the commit message so we have it there!
> >>>>>>>
> >>>>>>>> This FPGA acts as a level shifter between the Intel SoC pins and the pin
> >>>>>>>> header, and also makes a kind of switch/mux.
> >>>>>>>
> >>>>>>> Since it's Intel we need to notify Andy to help out with this so that
> >>>>>>> it gets done in a way that works with how he think consumers
> >>>>>>> should interact with Intel pin control and GPIO.
> >>>>>>>
> >>>>>>>> +---------+         +--------------+             +---+
> >>>>>>>>           |         |              |             | H |
> >>>>>>>>           |---------|              |-------------| E |
> >>>>>>>>           |         |              |             | A |
> >>>>>>>> Intel Soc |---------|    FPGA      |-------------| D |
> >>>>>>>>           |         |              |             | E |
> >>>>>>>>           |---------|              |-------------| R |
> >>>>>>>>           |         |              |             |   |
> >>>>>>>> ----------+         +--------------+             +---+
> >>>>>>>>
> >>>>>>>>
> >>>>>>>> For most of the pins, the FPGA opens/closes a switch to enable/disable
> >>>>>>>> the access to the SoC pin from a pin header.
> >>>>>>>> Each "switch", has a direction flag that shall be set in tandem with the
> >>>>>>>> status of the SoC pin.
> >>>>>>>> For example, if the SoC pin is in PWM mode, the "switch" shall be
> >>>>>>>> configured in output direction.
> >>>>>>>> If the SoC pin is set in GPIO mode, the direction of the "switch" shall
> >>>>>>>> corresponds to the GPIO direction.
> >>>>>>>>
> >>>>>>>> +---------+              +--------------+             +---+
> >>>>>>>>           |              |              |             | H |
> >>>>>>>>           |              |      \       |             | E |
> >>>>>>>>           |   PWM1       |       \      |             | A |
> >>>>>>>> Intel Soc |--------------|-----   \-----|-------------| D |
> >>>>>>>>           |              |              |             | E |
> >>>>>>>>           |              |              |             | R |
> >>>>>>>>           |              |    FPGA      |             |   |
> >>>>>>>> ----------+              +--------------+             +---+
> >>>>>>>>
> >>>>>>>> (PWM1 pin from Intel SoC can be used as PWM, and also in GPIO mode,
> >>>>>>>> thanks to the Intel pinctrl driver).
> >>>>>>>>
> >>>>>>>> Few pins (PINMUX_* pins) work differently. The FPGA acts as a mux and
> >>>>>>>> routes for example the I2C0_SDA pin or GPIOX (of the SoC) to the pin header.
> >>>
> >>> Yep, that's clear.
> >>>
> >>>>>>>> +---------+           +--------------+             +---+
> >>>>>>>>           | I2C0_SDA  |              |             | H |
> >>>>>>>>           |-----------|----- \       |             | E |
> >>>>>>>>           |           |       \      |             | A |
> >>>>>>>> Intel Soc |           |        \-----|-------------| D |
> >>>>>>>>           | GPIOX     |              |             | E |
> >>>>>>>>           |-----------|-----         |             | R |
> >>>>>>>>           |           |    FPGA      |             |   |
> >>>>>>>> ----------+           +--------------+             +---+
> >>>>>>>>
> >>>>>>>> The pin header looks like this:
> >>>>>>>> +--------------------+--------------------+
> >>>>>>>> |      3.3V          |       5V           |
> >>>>>>>> | GPIO2 / I2C1_SDA   |       5V           |
> >>>>>>>> | GPIO3 / I2C1_SCL   |       GND          |
> >>>>>>>> | GPIO4 / ADC0       | GPIO14 / UART1_TX  |
> >>>>>>>> |      GND           | GPIO15 / UART1_RX  |
> >>>>>>>> | GPIO17 / UART1_RTS | GPIO18 / I2S_CLK   |
> >>>>>>>> |     GPIO27         |       GND          |
> >>>>>>>> |     GPIO22         |      GPIO23        |
> >>>>>>>> |      3.3V          |      GPIO24        |
> >>>>>>>> | GPIO10 / SPI_MOSI  |       GND          |
> >>>>>>>> | GPIO9 / SPI_MISO   |      GPIO25        |
> >>>>>>>> | GPIO11 / SPI_CLK   | GPIO8 / SPI_CS0    |
> >>>>>>>> |      GND           | GPIO7 / SPI_CS1    |
> >>>>>>>> | GPIO0 / I2C0_SDA   | GPIO1 / I2C0_SCL   |
> >>>>>>>> |     GPIO5          |       GND          |
> >>>>>>>> |     GPIO6          | GPIO12 / PWM0      |
> >>>>>>>> | GPIO13 / PWM1      |       GND          |
> >>>>>>>> | GPIO19 / I2S_FRM   | GPIO16 / UART1_CTS |
> >>>>>>>> |     GPIO26         | GPIO20 / I2S_DIN   |
> >>>>>>>> |      GND           | GPIO21 / I2S_DOUT  |
> >>>>>>>> +--------------------+--------------------+
> >>>>>>>>
> >>>>>>>> The GPIOs in the pin header corresponds to the gpiochip I declare in
> >>>>>>>> this driver.
> >>>>>>>> So when I want to use a pin in GPIO mode, the upboard pinctrl driver
> >>>>>>>> requests the corresponding SoC GPIO to the Intel pinctrl driver.
> >>>>>>>> The SoC pins connected to the FPGA, are identified with "external" id.
> >>>>>>>>
> >>>>>>>> The hardware and the FPGA were designed in tandem, so you know for
> >>>>>>>> example that for the GPIOX you need to request the Nth "external" GPIO.
> >>>>>>>>
> >>>>>>>> When you drive your GPIO, the upboard gpiochip manages in the same time
> >>>>>>>> the direction of the "switch" and the value/direction of the
> >>>>>>>> corresponding SoC pin.
> >>>>>>>>
> >>>>>>>> +------------------+         +--------------+             +---+
> >>>>>>>>                    |---------|              |-------------| H |
> >>>>>>>>                    |---------|   GPIOCHIP   |-------------| E |
> >>>>>>>>    Intel gpiochip  |---------|              |-------------| A |
> >>>>>>>>  provided by Intel |---------|    FPGA      |-------------| D |
> >>>>>>>>   pinctrl driver   |---------|              |-------------| E |
> >>>>>>>>                    |---------|              |-------------| R |
> >>>>>>>>                    |---------|              |-------------|   |
> >>>>>>>> +------------------+         +--------------+             +---+
> >>>>>>>>
> >>>>>>>> About gpiochip_add_pinlist_range(), I added it because the FPGA pins
> >>>>>>>> used by the gpiochip are not consecutive.
> >>>>>>>>
> >>>>>>>> Please let me know if it is not clear.
> >>>>>>>> And sorry I'm not very good to make ascii art.
> >>>>>>>
> >>>>>>> I get it! We have a similar driver in the kernel already, look into:
> >>>>>>> drivers/gpio/gpio-aggregator.c
> >>>>>>>
> >>>>>>> The aggregator abstraction is however just software. What you
> >>>>>>> need here is a gpio-aggregator that adds some hardware
> >>>>>>> control on top. But it has a very nice design using a bitmap
> >>>>>>> to keep track of the GPIOs etc, and it supports operations
> >>>>>>> on multiple GPIOs (many man-hours of hard coding and
> >>>>>>> design went into that driver, ask Geert and Andy...)
> >>>>>>>
> >>>>>>> So I would proceed like this:
> >>>>>>>
> >>>>>>> - The pin control part of the driver looks sound, except
> >>>>>>>   for the way you add ranges.
> >>>>>>>
> >>>>>>> - The gpiochip part needs to be refactored using the
> >>>>>>>   ideas from gpio-aggregator.c.
> >>>>>>>
> >>>>>>> - Look closely at aggregator and see what you can do
> >>>>>>>   based on that code, if you can mimic how it picks up
> >>>>>>>   and forwards all GPIO functions. Maybe part of it
> >>>>>>>   needs to be made into a library?
> >>>>>>>  <linux/gpio/gpio-aggregator.h>?
> >>>>>>>   For example if you start to feel like "I would really like
> >>>>>>>   to just call gpio_fwd_get_multiple() then this is what
> >>>>>>>   you want to do. The library can probably still be
> >>>>>>>   inside gpio-aggregator.c the way we do it in
> >>>>>>>   e.g. gpio-mmio.c, just export and keep library functions
> >>>>>>>   separately.
> >>>>>>
> >>>>>> Ok I think I understand what you expect.
> >>>>>> I started to look at the gpio-aggregator code, play a bit with it, and
> >>>>>> refactor it to use it from my driver.
> >>>>>>
> >>>>>> My main issue is about the request of the SoC GPIOs done by the aggregator.
> >>>>>> If from my driver I call the aggregator library to create a gpiochip,
> >>>>>> the SoC pins will be requested. So the SoC pins will be set in GPIO
> >>>>>> mode, and the pins will never be in function mode.
> >>>>>> There is no way to set the pins back to function mode (even if the GPIO
> >>>>>> is free).
> >>>>>>
> >>>>>> I tried to add a feature in the aggregator to defer the request of the gpio.
> >>>>>> So at the beginning of each ops the gpio_desc is checked. If it is
> >>>>>> valid, the gpio can be used. Otherwise, the gpio is requested.
> >>>>>> For example:
> >>>>>>
> >>>>>> gpio_fwd_get() {
> >>>>>> 	if (!gpio_desc_is_valid(desc))
> >>>>>> 		desc = request_gpio()
> >>>>>>
> >>>>>> 	return gpiod_get_value(desc)
> >>>>>> }
> >>>>>>
> >>>>>> But when a gpiochip is registered, the core calls get_direction() or
> >>>>>> direction_input(), so all GPIOs are requested and it does not solve my
> >>>>>> problem.
> >>>>>>
> >>>>>> I expect to register a gpiochip without setting all pins in GPIO mode at
> >>>>>> probe time (like all pinctrl driver do).
> >>>>>> But I did not find a solution.
> >>>>>
> >>>>> Basically what you need is a pinctrl-aggregattor (an analogue for the pin
> >>>>> muxing and configuration).
> >>>>
> >>>> I found a trick to workaround the get_direction() issue in the
> >>>> gpio-aggregator.
> >>>>
> >>>> I added a "request on first use" feature on the aggregator.
> >>>> The GPIO is requested during the request() operation of the fowarder.
> >>>>
> >>>> static int gpio_fwd_request(struct gpio_chip *chip, unsigned int offset)
> >>>> {
> >>>> 	struct gpiochip_fwd *fwd = gpiochip_get_data(chip);
> >>>>
> >>>> 	if (!IS_ERR_OR_NULL(fwd->descs[offset]))
> >>>> 		return 0;
> >>>>
> >>>> 	fwd->descs[offset] = devm_gpiod_get_index(fwd->dev, NULL,
> >>>> 						offset, GPIOD_ASIS);
> >>>>
> >>>> 	return PTR_ERR_OR_ZERO(fwd->descs[offset]);
> >>>> }
> >>>>
> >>>> The remaining problem is that the get_direction() callback is called
> >>>> during gpiochip registration. For now if the gpio_desc is not valid (so
> >>>> the GPIO was not yet requested) I return GPIO_LINE_DIRECTION_OUT by
> >>>> default. But I'm not very convinced by this hack.
> >>>> Maybe I could retrieve the gpio_chip and call its get_direction()
> >>>> callback, but it seems not to be a better idea.
> >>>>
> >>>> For the pinctrl-aggregator you mentioned, if I understand correctly the
> >>>> idea to aggregate the SoC pins in a pinctrl aggregator (with a
> >>>> gpio_chip) which just forwards gpio_request_enable(),
> >>>> gpio_disable_free(), gpio_set_direction() and also all gpio_chip operations.
> >>>
> >>> No only these, all of the pin mux, pin configuration, and GPIO operations
> >>> should be proxied.
> >>
> >> Why should I proxy operations that will never be used (pin mux, pin
> >> conf) ? I mean there will never be a driver that will configure FPGA pins.
> > 
> > According to your picture above the FPGA works as a pin function selector,
> > which also implies different pin configuration (e.g., slew rate for I²C pins).
> > Did I get it wrong?
> 
> Yes you are right. In function mode the FPGA selects the I2C pin. When
> the FPGA GPIO is requested, the FPGA changes the mux to select a SoC
> GPIO pin (a pure GPIO pin).
> 
> >>>> But how to deal with the pinctrl of my FPGA ? For one of its fake pin
> >>>> the dummy pinctrl drives the corresponding SoC pin and FPGA pin ?
> >>>
> >>> What's the "fake pin"? I can't find the one up in your schemas.
> >>
> >> I thought that the idea was a virtual pinctrl which drive the SoC
> >> pinctrl and the FPGA pinctrl. But what you mean is the FPGA pinctrl acts
> >> as a proxy.
> > 
> > When one wants to configure the pin parameters (let's say pin bias) the driver
> > delegates that to the SoC, in case the FPGA doesn't repeat this itself. Or both,
> > but this makes things too complicated already.
> 
> The FPGA cannot configure pin parameter. It can only configures the
> forward direction.
> For example, for UART_TX pin the FPGA will set direction as "output" and
> for UART_RX pin the FPGA will set direction as "input".
> It can also enable/disable the forward of the SoC pin. If the forward of
> a pin is disabled, the FPGA pin is in HIGH-Z.

So, which is exactly why proxying is needed:
1) it delegates pin configuration when user asks for a change;
2) it properly configures SoC pin directions, etc in accordance to the signal
on the header and the user ask.

> >>>> So for each pin, the aggregator may have multiple pins to drive ?
> >>>
> >>> Depending on the case, but yes. Currently we have implementations of
> >>> the discrete pin controls on Intel platforms based on ACPI (see Intel Minnow,
> >>> Intel Galileo, Intel Edison/Arduino boards in meta-acpi project [1]).
> >>> You probably want something similar to be written in C.
> >>>
> >>>> Was it your idea Andy ?
> >>>>
> >>>> Other challenge is to retrieve all the pins to add in the
> >>>> pinctrl-aggregator. As input I have only GPIO descriptors, but I guess
> >>>> it should be feasible.
> >>>
> >>> In general your "proxy" (FPGA) pin control driver should be consumer of all SoC
> >>> pins (and their respective muxing and configurations) and be provider of the
> >>> pins that are available to the user.
> >>
> >> Isn't that a bit over-engineered for my use case ?
> > 
> > Yes, but it's already over-engineered in the HW, no?
> > 
> >> For the pinconf / pinmux, the FPGA is just a voltage translator.
> > 
> > It doesn't correspond to your picture where the pin function selector is depicted.
> 
> Yes you're right. I mean the mux can only select one function or GPIO mode.

Which is fine, but it is still a pin muxing.

> I don't know why there is a mux to select a GPIO only pin, I'm pretty
> sure the I2C pins can be set in GPIO mode.
> It's probably for an hardware reason that a mux was added for only few pins.
> 
> >> It is transparent. The only relevant thing for the FPGA is the direction to
> >> set for the switch of each pin. And the drivers knows which directions
> >> to apply during the probe. This direction will only change in GPIO mode,
> >> but in GPIO mode we know which direction to set.
> >>
> >> For the GPIO part, the FPGA provides GPIOs. And in this case it is a
> >> consumer of SoC GPIOs, it is already a kind of aggregator.
> >>
> >>> [1]: https://github.com/westeri/meta-acpi/tree/master/recipes-bsp/acpi-tables/samples
diff mbox series

Patch

diff --git a/drivers/pinctrl/Kconfig b/drivers/pinctrl/Kconfig
index 95a8e2b9a614..0289a0a6fb6d 100644
--- a/drivers/pinctrl/Kconfig
+++ b/drivers/pinctrl/Kconfig
@@ -576,6 +576,20 @@  config PINCTRL_TH1520
 	  This driver is needed for RISC-V development boards like
 	  the BeagleV Ahead and the LicheePi 4A.
 
+config PINCTRL_UPBOARD
+	tristate "AAeon UP board FPGA pin controller"
+	depends on MFD_UPBOARD_FPGA
+	select PINMUX
+	select GENERIC_PINCTRL_GROUPS
+	select GENERIC_PINMUX_FUNCTIONS
+	help
+	  Pin controller for the FPGA GPIO lines on UP boards. Due to the
+	  hardware layout, the driver control the FPGA pins in tandem with
+	  their corresponding Intel SoC GPIOs.
+
+	  To compile this driver as a module, choose M here: the module
+	  will be called pinctrl-upboard.
+
 config PINCTRL_ZYNQ
 	bool "Pinctrl driver for Xilinx Zynq"
 	depends on ARCH_ZYNQ
diff --git a/drivers/pinctrl/Makefile b/drivers/pinctrl/Makefile
index fba1c56624c0..989b8d28ecac 100644
--- a/drivers/pinctrl/Makefile
+++ b/drivers/pinctrl/Makefile
@@ -56,6 +56,7 @@  obj-$(CONFIG_PINCTRL_SX150X)	+= pinctrl-sx150x.o
 obj-$(CONFIG_PINCTRL_TB10X)	+= pinctrl-tb10x.o
 obj-$(CONFIG_PINCTRL_TPS6594)	+= pinctrl-tps6594.o
 obj-$(CONFIG_PINCTRL_TH1520)	+= pinctrl-th1520.o
+obj-$(CONFIG_PINCTRL_UPBOARD)	+= pinctrl-upboard.o
 obj-$(CONFIG_PINCTRL_ZYNQMP)	+= pinctrl-zynqmp.o
 obj-$(CONFIG_PINCTRL_ZYNQ)	+= pinctrl-zynq.o
 
diff --git a/drivers/pinctrl/pinctrl-upboard.c b/drivers/pinctrl/pinctrl-upboard.c
new file mode 100644
index 000000000000..14bf64b9587d
--- /dev/null
+++ b/drivers/pinctrl/pinctrl-upboard.c
@@ -0,0 +1,1090 @@ 
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * UP board pin control driver.
+ *
+ * FPGA provides more GPIO driving power, LEDS and pin mux function.
+ *
+ * Copyright (c) AAEON. All rights reserved.
+ * Copyright (C) 2024 Bootlin
+ *
+ * Author: Gary Wang <garywang@aaeon.com.tw>
+ * Author: Thomas Richard <thomas.richard@bootlin.com>
+ */
+
+#include <linux/device.h>
+#include <linux/gpio/consumer.h>
+#include <linux/gpio/driver.h>
+#include <linux/mfd/upboard-fpga.h>
+#include <linux/module.h>
+#include <linux/pinctrl/consumer.h>
+#include <linux/pinctrl/pinctrl.h>
+#include <linux/pinctrl/pinmux.h>
+#include <linux/platform_device.h>
+#include <linux/regmap.h>
+#include <linux/seq_file.h>
+
+#include "core.h"
+#include "pinmux.h"
+
+enum upboard_pin_mode {
+	UPBOARD_PIN_MODE_FUNCTION = 1,
+	UPBOARD_PIN_MODE_GPIO_IN,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_DISABLED,
+};
+
+struct upboard_pin {
+	struct regmap_field *funcbit;
+	struct regmap_field *enbit;
+	struct regmap_field *dirbit;
+};
+
+struct upboard_pingroup {
+	struct pingroup grp;
+	enum upboard_pin_mode mode;
+	const enum upboard_pin_mode *modes;
+};
+
+struct upboard_pinctrl_data {
+	const struct upboard_pingroup *groups;
+	size_t ngroups;
+	const struct pinfunction *funcs;
+	size_t nfuncs;
+	const struct pinctrl_map *maps;
+	size_t nmaps;
+	const unsigned int *pin_header;
+	size_t ngpio;
+};
+
+struct upboard_pinctrl {
+	struct gpio_chip chip;
+	struct device *dev;
+	struct pinctrl_dev *pctldev;
+	const struct upboard_pinctrl_data *pctrl_data;
+	struct gpio_pin_range pin_range;
+	struct upboard_pin *pins;
+	struct gpio_desc **gpio;
+};
+
+enum upboard_func0_fpgabit {
+	UPBOARD_FUNC_I2C0_EN = 8,
+	UPBOARD_FUNC_I2C1_EN = 9,
+	UPBOARD_FUNC_CEC0_EN = 12,
+	UPBOARD_FUNC_ADC0_EN = 14,
+};
+
+static const struct reg_field upboard_i2c0_reg =
+	REG_FIELD(UPBOARD_REG_FUNC_EN0, UPBOARD_FUNC_I2C0_EN, UPBOARD_FUNC_I2C0_EN);
+
+static const struct reg_field upboard_i2c1_reg =
+	REG_FIELD(UPBOARD_REG_FUNC_EN0, UPBOARD_FUNC_I2C1_EN, UPBOARD_FUNC_I2C1_EN);
+
+static const struct reg_field upboard_adc0_reg =
+	REG_FIELD(UPBOARD_REG_FUNC_EN0, UPBOARD_FUNC_ADC0_EN, UPBOARD_FUNC_ADC0_EN);
+
+#define UPBOARD_UP_BIT_TO_PIN(bit) UPBOARD_UP_BIT_##bit
+
+#define UPBOARD_UP_PIN_NAME(id)					\
+	{							\
+		.number = UPBOARD_UP_BIT_##id,			\
+		.name = #id,					\
+	}
+
+#define UPBOARD_UP_PIN_MUX(bit, data)				\
+	{							\
+		.number = UPBOARD_UP_BIT_##bit,			\
+		.name = "PINMUX_"#bit,				\
+		.drv_data = (void *)(data),			\
+	}
+
+#define UPBOARD_UP_PIN_FUNC(id, data)				\
+	{							\
+		.number = UPBOARD_UP_BIT_##id,			\
+		.name = #id,					\
+		.drv_data = (void *)(data),			\
+	}
+
+enum upboard_up_fpgabit {
+	UPBOARD_UP_BIT_I2C1_SDA,
+	UPBOARD_UP_BIT_I2C1_SCL,
+	UPBOARD_UP_BIT_ADC0,
+	UPBOARD_UP_BIT_UART1_RTS,
+	UPBOARD_UP_BIT_GPIO27,
+	UPBOARD_UP_BIT_GPIO22,
+	UPBOARD_UP_BIT_SPI_MOSI,
+	UPBOARD_UP_BIT_SPI_MISO,
+	UPBOARD_UP_BIT_SPI_CLK,
+	UPBOARD_UP_BIT_I2C0_SDA,
+	UPBOARD_UP_BIT_GPIO5,
+	UPBOARD_UP_BIT_GPIO6,
+	UPBOARD_UP_BIT_PWM1,
+	UPBOARD_UP_BIT_I2S_FRM,
+	UPBOARD_UP_BIT_GPIO26,
+	UPBOARD_UP_BIT_UART1_TX,
+	UPBOARD_UP_BIT_UART1_RX,
+	UPBOARD_UP_BIT_I2S_CLK,
+	UPBOARD_UP_BIT_GPIO23,
+	UPBOARD_UP_BIT_GPIO24,
+	UPBOARD_UP_BIT_GPIO25,
+	UPBOARD_UP_BIT_SPI_CS0,
+	UPBOARD_UP_BIT_SPI_CS1,
+	UPBOARD_UP_BIT_I2C0_SCL,
+	UPBOARD_UP_BIT_PWM0,
+	UPBOARD_UP_BIT_UART1_CTS,
+	UPBOARD_UP_BIT_I2S_DIN,
+	UPBOARD_UP_BIT_I2S_DOUT,
+};
+
+static const struct pinctrl_pin_desc upboard_up_pins[] = {
+	UPBOARD_UP_PIN_FUNC(I2C1_SDA, &upboard_i2c1_reg),
+	UPBOARD_UP_PIN_FUNC(I2C1_SCL, &upboard_i2c1_reg),
+	UPBOARD_UP_PIN_FUNC(ADC0, &upboard_adc0_reg),
+	UPBOARD_UP_PIN_NAME(UART1_RTS),
+	UPBOARD_UP_PIN_NAME(GPIO27),
+	UPBOARD_UP_PIN_NAME(GPIO22),
+	UPBOARD_UP_PIN_NAME(SPI_MOSI),
+	UPBOARD_UP_PIN_NAME(SPI_MISO),
+	UPBOARD_UP_PIN_NAME(SPI_CLK),
+	UPBOARD_UP_PIN_FUNC(I2C0_SDA, &upboard_i2c0_reg),
+	UPBOARD_UP_PIN_NAME(GPIO5),
+	UPBOARD_UP_PIN_NAME(GPIO6),
+	UPBOARD_UP_PIN_NAME(PWM1),
+	UPBOARD_UP_PIN_NAME(I2S_FRM),
+	UPBOARD_UP_PIN_NAME(GPIO26),
+	UPBOARD_UP_PIN_NAME(UART1_TX),
+	UPBOARD_UP_PIN_NAME(UART1_RX),
+	UPBOARD_UP_PIN_NAME(I2S_CLK),
+	UPBOARD_UP_PIN_NAME(GPIO23),
+	UPBOARD_UP_PIN_NAME(GPIO24),
+	UPBOARD_UP_PIN_NAME(GPIO25),
+	UPBOARD_UP_PIN_NAME(SPI_CS0),
+	UPBOARD_UP_PIN_NAME(SPI_CS1),
+	UPBOARD_UP_PIN_FUNC(I2C0_SCL, &upboard_i2c0_reg),
+	UPBOARD_UP_PIN_NAME(PWM0),
+	UPBOARD_UP_PIN_NAME(UART1_CTS),
+	UPBOARD_UP_PIN_NAME(I2S_DIN),
+	UPBOARD_UP_PIN_NAME(I2S_DOUT),
+};
+
+static const unsigned int upboard_up_pin_header[] = {
+	UPBOARD_UP_BIT_TO_PIN(I2C0_SDA),
+	UPBOARD_UP_BIT_TO_PIN(I2C0_SCL),
+	UPBOARD_UP_BIT_TO_PIN(I2C1_SDA),
+	UPBOARD_UP_BIT_TO_PIN(I2C1_SCL),
+	UPBOARD_UP_BIT_TO_PIN(ADC0),
+	UPBOARD_UP_BIT_TO_PIN(GPIO5),
+	UPBOARD_UP_BIT_TO_PIN(GPIO6),
+	UPBOARD_UP_BIT_TO_PIN(SPI_CS1),
+	UPBOARD_UP_BIT_TO_PIN(SPI_CS0),
+	UPBOARD_UP_BIT_TO_PIN(SPI_MISO),
+	UPBOARD_UP_BIT_TO_PIN(SPI_MOSI),
+	UPBOARD_UP_BIT_TO_PIN(SPI_CLK),
+	UPBOARD_UP_BIT_TO_PIN(PWM0),
+	UPBOARD_UP_BIT_TO_PIN(PWM1),
+	UPBOARD_UP_BIT_TO_PIN(UART1_TX),
+	UPBOARD_UP_BIT_TO_PIN(UART1_RX),
+	UPBOARD_UP_BIT_TO_PIN(UART1_CTS),
+	UPBOARD_UP_BIT_TO_PIN(UART1_RTS),
+	UPBOARD_UP_BIT_TO_PIN(I2S_CLK),
+	UPBOARD_UP_BIT_TO_PIN(I2S_FRM),
+	UPBOARD_UP_BIT_TO_PIN(I2S_DIN),
+	UPBOARD_UP_BIT_TO_PIN(I2S_DOUT),
+	UPBOARD_UP_BIT_TO_PIN(GPIO22),
+	UPBOARD_UP_BIT_TO_PIN(GPIO23),
+	UPBOARD_UP_BIT_TO_PIN(GPIO24),
+	UPBOARD_UP_BIT_TO_PIN(GPIO25),
+	UPBOARD_UP_BIT_TO_PIN(GPIO26),
+	UPBOARD_UP_BIT_TO_PIN(GPIO27),
+};
+
+static const unsigned int upboard_up_uart1_pins[] = {
+	UPBOARD_UP_BIT_TO_PIN(UART1_TX),
+	UPBOARD_UP_BIT_TO_PIN(UART1_RX),
+	UPBOARD_UP_BIT_TO_PIN(UART1_RTS),
+	UPBOARD_UP_BIT_TO_PIN(UART1_CTS),
+};
+
+static const enum upboard_pin_mode upboard_up_uart1_modes[] = {
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_IN,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_IN,
+};
+
+static_assert(ARRAY_SIZE(upboard_up_uart1_modes) ==
+	      ARRAY_SIZE(upboard_up_uart1_pins));
+
+static const unsigned int upboard_up_i2c0_pins[] = {
+	UPBOARD_UP_BIT_TO_PIN(I2C0_SCL),
+	UPBOARD_UP_BIT_TO_PIN(I2C0_SDA),
+};
+
+static const unsigned int upboard_up_i2c1_pins[] = {
+	UPBOARD_UP_BIT_TO_PIN(I2C1_SCL),
+	UPBOARD_UP_BIT_TO_PIN(I2C1_SDA),
+};
+
+static const unsigned int upboard_up_spi2_pins[] = {
+	UPBOARD_UP_BIT_TO_PIN(SPI_MOSI),
+	UPBOARD_UP_BIT_TO_PIN(SPI_MISO),
+	UPBOARD_UP_BIT_TO_PIN(SPI_CLK),
+	UPBOARD_UP_BIT_TO_PIN(SPI_CS0),
+	UPBOARD_UP_BIT_TO_PIN(SPI_CS1),
+};
+
+static const enum upboard_pin_mode upboard_up_spi2_modes[] = {
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_IN,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+};
+
+static_assert(ARRAY_SIZE(upboard_up_spi2_modes) ==
+	      ARRAY_SIZE(upboard_up_spi2_pins));
+
+static const unsigned int upboard_up_i2s0_pins[]  = {
+	UPBOARD_UP_BIT_TO_PIN(I2S_FRM),
+	UPBOARD_UP_BIT_TO_PIN(I2S_CLK),
+	UPBOARD_UP_BIT_TO_PIN(I2S_DIN),
+	UPBOARD_UP_BIT_TO_PIN(I2S_DOUT),
+};
+
+static const enum upboard_pin_mode upboard_up_i2s0_modes[] = {
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_IN,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+};
+
+static_assert(ARRAY_SIZE(upboard_up_i2s0_pins) ==
+	      ARRAY_SIZE(upboard_up_i2s0_modes));
+
+static const unsigned int upboard_up_pwm0_pins[] = {
+	UPBOARD_UP_BIT_TO_PIN(PWM0),
+};
+
+static const unsigned int upboard_up_pwm1_pins[] = {
+	UPBOARD_UP_BIT_TO_PIN(PWM1),
+};
+
+static const unsigned int upboard_up_adc0_pins[] = {
+	UPBOARD_UP_BIT_TO_PIN(ADC0),
+};
+
+#define UPBOARD_PINGROUP_MODE(_grp, _pins, _mode)			\
+{									\
+	.grp = PINCTRL_PINGROUP(_grp, _pins, ARRAY_SIZE(_pins)),	\
+	.mode = _mode,							\
+}
+
+#define UPBOARD_PINGROUP_MODES(_grp, _pins, _modes)			\
+{									\
+	.grp = PINCTRL_PINGROUP(_grp, _pins, ARRAY_SIZE(_pins)),	\
+	.modes = _modes,							\
+}
+
+static const struct upboard_pingroup upboard_up_pin_groups[] = {
+	UPBOARD_PINGROUP_MODES("uart1_grp", upboard_up_uart1_pins, upboard_up_uart1_modes),
+	UPBOARD_PINGROUP_MODE("i2c0_grp", upboard_up_i2c0_pins, UPBOARD_PIN_MODE_GPIO_OUT),
+	UPBOARD_PINGROUP_MODE("i2c1_grp", upboard_up_i2c1_pins, UPBOARD_PIN_MODE_GPIO_OUT),
+	UPBOARD_PINGROUP_MODES("spi2_grp", upboard_up_spi2_pins, &upboard_up_spi2_modes[0]),
+	UPBOARD_PINGROUP_MODES("i2s0_grp", upboard_up_i2s0_pins, &upboard_up_i2s0_modes[0]),
+	UPBOARD_PINGROUP_MODE("pwm0_grp", upboard_up_pwm0_pins, UPBOARD_PIN_MODE_GPIO_OUT),
+	UPBOARD_PINGROUP_MODE("pwm1_grp", upboard_up_pwm1_pins, UPBOARD_PIN_MODE_GPIO_OUT),
+	UPBOARD_PINGROUP_MODE("adc0_grp", upboard_up_adc0_pins, UPBOARD_PIN_MODE_GPIO_IN),
+};
+
+static const char * const upboard_up_uart1_groups[] = { "uart1_grp" };
+static const char * const upboard_up_i2c0_groups[]  = { "i2c0_grp" };
+static const char * const upboard_up_i2c1_groups[]  = { "i2c1_grp" };
+static const char * const upboard_up_spi2_groups[]  = { "spi2_grp" };
+static const char * const upboard_up_i2s0_groups[]  = { "i2s0_grp" };
+static const char * const upboard_up_pwm0_groups[]  = { "pwm0_grp" };
+static const char * const upboard_up_pwm1_groups[]  = { "pwm1_grp" };
+static const char * const upboard_up_adc0_groups[]  = { "adc0_grp" };
+
+#define UPBOARD_FUNCTION(func, groups)	PINCTRL_PINFUNCTION(func, groups, ARRAY_SIZE(groups))
+
+static const struct pinfunction upboard_up_pin_functions[] = {
+	UPBOARD_FUNCTION("uart1", upboard_up_uart1_groups),
+	UPBOARD_FUNCTION("i2c0", upboard_up_i2c0_groups),
+	UPBOARD_FUNCTION("i2c1", upboard_up_i2c1_groups),
+	UPBOARD_FUNCTION("spi2", upboard_up_spi2_groups),
+	UPBOARD_FUNCTION("i2s0", upboard_up_i2s0_groups),
+	UPBOARD_FUNCTION("pwm0", upboard_up_pwm0_groups),
+	UPBOARD_FUNCTION("pwm1", upboard_up_pwm1_groups),
+	UPBOARD_FUNCTION("adc0", upboard_up_adc0_groups),
+};
+
+static const struct pinctrl_map upboard_up_pin_mapping[] = {
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "uart1_grp", "uart1"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "i2c0_grp", "i2c0"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "i2c1_grp", "i2c1"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "spi2_grp", "spi2"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "i2s0_grp", "i2s0"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "pwm0_grp", "pwm0"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "pwm1_grp", "pwm1"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "adc0_grp", "adc0"),
+};
+
+static const struct upboard_pinctrl_data upboard_up_pinctrl_data = {
+	.groups = &upboard_up_pin_groups[0],
+	.ngroups = ARRAY_SIZE(upboard_up_pin_groups),
+	.funcs = &upboard_up_pin_functions[0],
+	.nfuncs = ARRAY_SIZE(upboard_up_pin_functions),
+	.maps = &upboard_up_pin_mapping[0],
+	.nmaps = ARRAY_SIZE(upboard_up_pin_mapping),
+	.pin_header = &upboard_up_pin_header[0],
+	.ngpio = ARRAY_SIZE(upboard_up_pin_header),
+};
+
+#define UPBOARD_UP2_BIT_TO_PIN(bit) UPBOARD_UP2_BIT_##bit
+
+#define UPBOARD_UP2_PIN_NAME(id)					\
+	{								\
+		.number = UPBOARD_UP2_BIT_##id,				\
+		.name = #id,						\
+	}
+
+#define UPBOARD_UP2_PIN_MUX(bit, data)					\
+	{								\
+		.number = UPBOARD_UP2_BIT_##bit,			\
+		.name = "PINMUX_"#bit,					\
+		.drv_data = (void *)(data),				\
+	}
+
+#define UPBOARD_UP2_PIN_FUNC(id, data)					\
+	{								\
+		.number = UPBOARD_UP2_BIT_##id,				\
+		.name = #id,						\
+		.drv_data = (void *)(data),				\
+	}
+
+enum upboard_up2_fpgabit {
+	UPBOARD_UP2_BIT_UART1_TXD,
+	UPBOARD_UP2_BIT_UART1_RXD,
+	UPBOARD_UP2_BIT_UART1_RTS,
+	UPBOARD_UP2_BIT_UART1_CTS,
+	UPBOARD_UP2_BIT_GPIO3_ADC0,
+	UPBOARD_UP2_BIT_GPIO5_ADC2,
+	UPBOARD_UP2_BIT_GPIO6_ADC3,
+	UPBOARD_UP2_BIT_GPIO11,
+	UPBOARD_UP2_BIT_EXHAT_LVDS1n,
+	UPBOARD_UP2_BIT_EXHAT_LVDS1p,
+	UPBOARD_UP2_BIT_SPI2_TXD,
+	UPBOARD_UP2_BIT_SPI2_RXD,
+	UPBOARD_UP2_BIT_SPI2_FS1,
+	UPBOARD_UP2_BIT_SPI2_FS0,
+	UPBOARD_UP2_BIT_SPI2_CLK,
+	UPBOARD_UP2_BIT_SPI1_TXD,
+	UPBOARD_UP2_BIT_SPI1_RXD,
+	UPBOARD_UP2_BIT_SPI1_FS1,
+	UPBOARD_UP2_BIT_SPI1_FS0,
+	UPBOARD_UP2_BIT_SPI1_CLK,
+	UPBOARD_UP2_BIT_I2C0_SCL,
+	UPBOARD_UP2_BIT_I2C0_SDA,
+	UPBOARD_UP2_BIT_I2C1_SCL,
+	UPBOARD_UP2_BIT_I2C1_SDA,
+	UPBOARD_UP2_BIT_PWM1,
+	UPBOARD_UP2_BIT_PWM0,
+	UPBOARD_UP2_BIT_EXHAT_LVDS0n,
+	UPBOARD_UP2_BIT_EXHAT_LVDS0p,
+	UPBOARD_UP2_BIT_GPIO24,
+	UPBOARD_UP2_BIT_GPIO10,
+	UPBOARD_UP2_BIT_GPIO2,
+	UPBOARD_UP2_BIT_GPIO1,
+	UPBOARD_UP2_BIT_EXHAT_LVDS3n,
+	UPBOARD_UP2_BIT_EXHAT_LVDS3p,
+	UPBOARD_UP2_BIT_EXHAT_LVDS4n,
+	UPBOARD_UP2_BIT_EXHAT_LVDS4p,
+	UPBOARD_UP2_BIT_EXHAT_LVDS5n,
+	UPBOARD_UP2_BIT_EXHAT_LVDS5p,
+	UPBOARD_UP2_BIT_I2S_SDO,
+	UPBOARD_UP2_BIT_I2S_SDI,
+	UPBOARD_UP2_BIT_I2S_WS_SYNC,
+	UPBOARD_UP2_BIT_I2S_BCLK,
+	UPBOARD_UP2_BIT_EXHAT_LVDS6n,
+	UPBOARD_UP2_BIT_EXHAT_LVDS6p,
+	UPBOARD_UP2_BIT_EXHAT_LVDS7n,
+	UPBOARD_UP2_BIT_EXHAT_LVDS7p,
+	UPBOARD_UP2_BIT_EXHAT_LVDS2n,
+	UPBOARD_UP2_BIT_EXHAT_LVDS2p,
+};
+
+static const struct pinctrl_pin_desc upboard_up2_pins[] = {
+	UPBOARD_UP2_PIN_NAME(UART1_TXD),
+	UPBOARD_UP2_PIN_NAME(UART1_RXD),
+	UPBOARD_UP2_PIN_NAME(UART1_RTS),
+	UPBOARD_UP2_PIN_NAME(UART1_CTS),
+	UPBOARD_UP2_PIN_NAME(GPIO3_ADC0),
+	UPBOARD_UP2_PIN_NAME(GPIO5_ADC2),
+	UPBOARD_UP2_PIN_NAME(GPIO6_ADC3),
+	UPBOARD_UP2_PIN_NAME(GPIO11),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS1n),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS1p),
+	UPBOARD_UP2_PIN_NAME(SPI2_TXD),
+	UPBOARD_UP2_PIN_NAME(SPI2_RXD),
+	UPBOARD_UP2_PIN_NAME(SPI2_FS1),
+	UPBOARD_UP2_PIN_NAME(SPI2_FS0),
+	UPBOARD_UP2_PIN_NAME(SPI2_CLK),
+	UPBOARD_UP2_PIN_NAME(SPI1_TXD),
+	UPBOARD_UP2_PIN_NAME(SPI1_RXD),
+	UPBOARD_UP2_PIN_NAME(SPI1_FS1),
+	UPBOARD_UP2_PIN_NAME(SPI1_FS0),
+	UPBOARD_UP2_PIN_NAME(SPI1_CLK),
+	UPBOARD_UP2_PIN_MUX(I2C0_SCL, &upboard_i2c0_reg),
+	UPBOARD_UP2_PIN_MUX(I2C0_SDA, &upboard_i2c0_reg),
+	UPBOARD_UP2_PIN_MUX(I2C1_SCL, &upboard_i2c1_reg),
+	UPBOARD_UP2_PIN_MUX(I2C1_SDA, &upboard_i2c1_reg),
+	UPBOARD_UP2_PIN_NAME(PWM1),
+	UPBOARD_UP2_PIN_NAME(PWM0),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS0n),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS0p),
+	UPBOARD_UP2_PIN_MUX(GPIO24, &upboard_i2c0_reg),
+	UPBOARD_UP2_PIN_MUX(GPIO10, &upboard_i2c0_reg),
+	UPBOARD_UP2_PIN_MUX(GPIO2, &upboard_i2c1_reg),
+	UPBOARD_UP2_PIN_MUX(GPIO1, &upboard_i2c1_reg),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS3n),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS3p),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS4n),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS4p),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS5n),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS5p),
+	UPBOARD_UP2_PIN_NAME(I2S_SDO),
+	UPBOARD_UP2_PIN_NAME(I2S_SDI),
+	UPBOARD_UP2_PIN_NAME(I2S_WS_SYNC),
+	UPBOARD_UP2_PIN_NAME(I2S_BCLK),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS6n),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS6p),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS7n),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS7p),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS2n),
+	UPBOARD_UP2_PIN_NAME(EXHAT_LVDS2p),
+};
+
+static const unsigned int upboard_up2_pin_header[] = {
+	UPBOARD_UP2_BIT_TO_PIN(GPIO10),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO24),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO1),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO2),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO3_ADC0),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO11),
+	UPBOARD_UP2_BIT_TO_PIN(SPI2_CLK),
+	UPBOARD_UP2_BIT_TO_PIN(SPI1_FS1),
+	UPBOARD_UP2_BIT_TO_PIN(SPI1_FS0),
+	UPBOARD_UP2_BIT_TO_PIN(SPI1_RXD),
+	UPBOARD_UP2_BIT_TO_PIN(SPI1_TXD),
+	UPBOARD_UP2_BIT_TO_PIN(SPI1_CLK),
+	UPBOARD_UP2_BIT_TO_PIN(PWM0),
+	UPBOARD_UP2_BIT_TO_PIN(PWM1),
+	UPBOARD_UP2_BIT_TO_PIN(UART1_TXD),
+	UPBOARD_UP2_BIT_TO_PIN(UART1_RXD),
+	UPBOARD_UP2_BIT_TO_PIN(UART1_CTS),
+	UPBOARD_UP2_BIT_TO_PIN(UART1_RTS),
+	UPBOARD_UP2_BIT_TO_PIN(I2S_BCLK),
+	UPBOARD_UP2_BIT_TO_PIN(I2S_WS_SYNC),
+	UPBOARD_UP2_BIT_TO_PIN(I2S_SDI),
+	UPBOARD_UP2_BIT_TO_PIN(I2S_SDO),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO6_ADC3),
+	UPBOARD_UP2_BIT_TO_PIN(SPI2_FS1),
+	UPBOARD_UP2_BIT_TO_PIN(SPI2_RXD),
+	UPBOARD_UP2_BIT_TO_PIN(SPI2_TXD),
+	UPBOARD_UP2_BIT_TO_PIN(SPI2_FS0),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO5_ADC2),
+};
+
+static const unsigned int upboard_up2_uart1_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(UART1_TXD),
+	UPBOARD_UP2_BIT_TO_PIN(UART1_RXD),
+	UPBOARD_UP2_BIT_TO_PIN(UART1_RTS),
+	UPBOARD_UP2_BIT_TO_PIN(UART1_CTS),
+};
+
+static const enum upboard_pin_mode upboard_up2_uart1_modes[] = {
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_IN,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_IN
+};
+
+static_assert(ARRAY_SIZE(upboard_up2_uart1_modes) ==
+	      ARRAY_SIZE(upboard_up2_uart1_pins));
+
+static const unsigned int upboard_up2_i2c0_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(I2C0_SCL),
+	UPBOARD_UP2_BIT_TO_PIN(I2C0_SDA),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO24),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO10),
+};
+
+static const unsigned int upboard_up2_i2c1_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(I2C1_SCL),
+	UPBOARD_UP2_BIT_TO_PIN(I2C1_SDA),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO2),
+	UPBOARD_UP2_BIT_TO_PIN(GPIO1),
+};
+
+static const unsigned int upboard_up2_spi1_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(SPI1_TXD),
+	UPBOARD_UP2_BIT_TO_PIN(SPI1_RXD),
+	UPBOARD_UP2_BIT_TO_PIN(SPI1_FS1),
+	UPBOARD_UP2_BIT_TO_PIN(SPI1_FS0),
+	UPBOARD_UP2_BIT_TO_PIN(SPI1_CLK),
+};
+
+static const unsigned int upboard_up2_spi2_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(SPI2_TXD),
+	UPBOARD_UP2_BIT_TO_PIN(SPI2_RXD),
+	UPBOARD_UP2_BIT_TO_PIN(SPI2_FS1),
+	UPBOARD_UP2_BIT_TO_PIN(SPI2_FS0),
+	UPBOARD_UP2_BIT_TO_PIN(SPI2_CLK),
+};
+
+static const enum upboard_pin_mode upboard_up2_spi_modes[] = {
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_IN,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+};
+
+static_assert(ARRAY_SIZE(upboard_up2_spi_modes) ==
+	      ARRAY_SIZE(upboard_up2_spi1_pins));
+
+static_assert(ARRAY_SIZE(upboard_up2_spi_modes) ==
+	      ARRAY_SIZE(upboard_up2_spi2_pins));
+
+static const unsigned int upboard_up2_i2s0_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(I2S_BCLK),
+	UPBOARD_UP2_BIT_TO_PIN(I2S_WS_SYNC),
+	UPBOARD_UP2_BIT_TO_PIN(I2S_SDI),
+	UPBOARD_UP2_BIT_TO_PIN(I2S_SDO),
+};
+
+static const enum upboard_pin_mode upboard_up2_i2s0_modes[] = {
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_OUT,
+	UPBOARD_PIN_MODE_GPIO_IN,
+	UPBOARD_PIN_MODE_GPIO_OUT
+};
+
+static_assert(ARRAY_SIZE(upboard_up2_i2s0_modes) ==
+	      ARRAY_SIZE(upboard_up2_i2s0_pins));
+
+static const unsigned int upboard_up2_pwm0_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(PWM0),
+};
+
+static const unsigned int upboard_up2_pwm1_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(PWM1),
+};
+
+static const unsigned int upboard_up2_adc0_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(GPIO3_ADC0),
+};
+
+static const unsigned int upboard_up2_adc2_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(GPIO5_ADC2),
+};
+
+static const unsigned int upboard_up2_adc3_pins[] = {
+	UPBOARD_UP2_BIT_TO_PIN(GPIO6_ADC3),
+};
+
+static const struct upboard_pingroup upboard_up2_pin_groups[] = {
+	UPBOARD_PINGROUP_MODES("uart1_grp", upboard_up2_uart1_pins, upboard_up2_uart1_modes),
+	UPBOARD_PINGROUP_MODE("i2c0_grp", upboard_up2_i2c0_pins, UPBOARD_PIN_MODE_FUNCTION),
+	UPBOARD_PINGROUP_MODE("i2c1_grp", upboard_up2_i2c1_pins, UPBOARD_PIN_MODE_FUNCTION),
+	UPBOARD_PINGROUP_MODES("spi1_grp", upboard_up2_spi1_pins, upboard_up2_spi_modes),
+	UPBOARD_PINGROUP_MODES("spi2_grp", upboard_up2_spi2_pins, upboard_up2_spi_modes),
+	UPBOARD_PINGROUP_MODES("i2s0_grp", upboard_up2_i2s0_pins, upboard_up2_i2s0_modes),
+	UPBOARD_PINGROUP_MODE("pwm0_grp", upboard_up2_pwm0_pins, UPBOARD_PIN_MODE_GPIO_OUT),
+	UPBOARD_PINGROUP_MODE("pwm1_grp", upboard_up2_pwm1_pins, UPBOARD_PIN_MODE_GPIO_OUT),
+	UPBOARD_PINGROUP_MODE("adc0_grp", upboard_up2_adc0_pins, UPBOARD_PIN_MODE_GPIO_IN),
+	UPBOARD_PINGROUP_MODE("adc2_grp", upboard_up2_adc2_pins, UPBOARD_PIN_MODE_GPIO_IN),
+	UPBOARD_PINGROUP_MODE("adc3_grp", upboard_up2_adc3_pins, UPBOARD_PIN_MODE_GPIO_IN),
+};
+
+static const char * const upboard_up2_uart1_groups[] = { "uart1_grp" };
+static const char * const upboard_up2_i2c0_groups[]  = { "i2c0_grp" };
+static const char * const upboard_up2_i2c1_groups[]  = { "i2c1_grp" };
+static const char * const upboard_up2_spi1_groups[]  = { "spi1_grp" };
+static const char * const upboard_up2_spi2_groups[]  = { "spi2_grp" };
+static const char * const upboard_up2_i2s0_groups[]  = { "i2s0_grp" };
+static const char * const upboard_up2_pwm0_groups[]  = { "pwm0_grp" };
+static const char * const upboard_up2_pwm1_groups[]  = { "pwm1_grp" };
+static const char * const upboard_up2_adc0_groups[]  = { "adc0_grp" };
+static const char * const upboard_up2_adc2_groups[]  = { "adc2_grp" };
+static const char * const upboard_up2_adc3_groups[]  = { "adc3_grp" };
+
+static const struct pinfunction upboard_up2_pin_functions[] = {
+	UPBOARD_FUNCTION("uart1", upboard_up2_uart1_groups),
+	UPBOARD_FUNCTION("i2c0", upboard_up2_i2c0_groups),
+	UPBOARD_FUNCTION("i2c1", upboard_up2_i2c1_groups),
+	UPBOARD_FUNCTION("spi1", upboard_up2_spi1_groups),
+	UPBOARD_FUNCTION("spi2", upboard_up2_spi2_groups),
+	UPBOARD_FUNCTION("i2s0", upboard_up2_i2s0_groups),
+	UPBOARD_FUNCTION("pwm0", upboard_up2_pwm0_groups),
+	UPBOARD_FUNCTION("pwm1", upboard_up2_pwm1_groups),
+	UPBOARD_FUNCTION("adc0", upboard_up2_adc0_groups),
+	UPBOARD_FUNCTION("adc2", upboard_up2_adc2_groups),
+	UPBOARD_FUNCTION("adc3", upboard_up2_adc3_groups),
+};
+
+static const struct pinctrl_map upboard_up2_pin_mapping[] = {
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "uart1_grp", "uart1"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "i2c0_grp", "i2c0"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "i2c1_grp", "i2c1"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "spi1_grp", "spi1"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "spi2_grp", "spi2"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "i2s0_grp", "i2s0"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "pwm0_grp", "pwm0"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "pwm1_grp", "pwm1"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "adc0_grp", "adc0"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "adc2_grp", "adc2"),
+	PIN_MAP_MUX_GROUP_HOG_DEFAULT("upboard-pinctrl", "adc3_grp", "adc3"),
+};
+
+static const struct upboard_pinctrl_data upboard_up2_pinctrl_data = {
+	.groups = &upboard_up2_pin_groups[0],
+	.ngroups = ARRAY_SIZE(upboard_up2_pin_groups),
+	.funcs = &upboard_up2_pin_functions[0],
+	.nfuncs = ARRAY_SIZE(upboard_up2_pin_functions),
+	.maps = &upboard_up2_pin_mapping[0],
+	.nmaps = ARRAY_SIZE(upboard_up2_pin_mapping),
+	.pin_header = &upboard_up2_pin_header[0],
+	.ngpio = ARRAY_SIZE(upboard_up2_pin_header),
+};
+
+static int __upboard_pinctrl_gpio_request_enable(struct pinctrl_dev *pctldev,
+						 unsigned int offset)
+{
+	struct upboard_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev);
+	struct upboard_pin *p = &pctrl->pins[offset];
+
+	if (p->funcbit) {
+		int ret = regmap_field_write(p->funcbit, 0);
+
+		if (ret)
+			return ret;
+	}
+
+	return regmap_field_write(p->enbit, 1);
+}
+
+static int upboard_pinctrl_gpio_request_enable(struct pinctrl_dev *pctldev,
+					       struct pinctrl_gpio_range *range,
+					       unsigned int offset)
+{
+	return __upboard_pinctrl_gpio_request_enable(pctldev, offset);
+}
+
+static void __upboard_pinctrl_gpio_disable_free(struct pinctrl_dev *pctldev, unsigned int offset)
+{
+	struct upboard_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev);
+	struct upboard_pin *p = &pctrl->pins[offset];
+
+	regmap_field_write(p->enbit, 0);
+
+	if (p->funcbit)
+		regmap_field_write(p->funcbit, 1);
+};
+
+static void upboard_pinctrl_gpio_disable_free(struct pinctrl_dev *pctldev,
+					      struct pinctrl_gpio_range *range, unsigned int offset)
+{
+	return __upboard_pinctrl_gpio_disable_free(pctldev, offset);
+}
+
+static int __upboard_pinctrl_gpio_set_direction(struct pinctrl_dev *pctldev,
+						unsigned int offset, bool input)
+{
+	struct upboard_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev);
+	struct upboard_pin *p = &pctrl->pins[offset];
+
+	return regmap_field_write(p->dirbit, input);
+}
+
+static int upboard_pinctrl_gpio_set_direction(struct pinctrl_dev *pctldev,
+					      struct pinctrl_gpio_range *range,
+					      unsigned int offset, bool input)
+{
+	return __upboard_pinctrl_gpio_set_direction(pctldev, offset, input);
+}
+
+static int upboard_pinctrl_set_mux(struct pinctrl_dev *pctldev, unsigned int func_selector,
+				   unsigned int group_selector)
+{
+	struct upboard_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev);
+	const struct upboard_pinctrl_data *pctrl_data = pctrl->pctrl_data;
+	const struct upboard_pingroup *upgroups = pctrl_data->groups;
+	struct group_desc *grp;
+	int mode, ret, i;
+
+	grp = pinctrl_generic_get_group(pctldev, group_selector);
+	if (!grp)
+		return -EINVAL;
+
+	for (i = 0; i < grp->grp.npins; i++) {
+		mode = upgroups[group_selector].mode ? upgroups[group_selector].mode :
+			upgroups[group_selector].modes[i];
+
+		if (mode == UPBOARD_PIN_MODE_FUNCTION) {
+			__upboard_pinctrl_gpio_disable_free(pctldev, grp->grp.pins[i]);
+			continue;
+		}
+
+		ret = __upboard_pinctrl_gpio_request_enable(pctldev, grp->grp.pins[i]);
+		if (ret)
+			return ret;
+
+		ret = __upboard_pinctrl_gpio_set_direction(pctldev, grp->grp.pins[i],
+							   mode == UPBOARD_PIN_MODE_GPIO_IN ?
+							   true : false);
+		if (ret)
+			return ret;
+	}
+
+	return 0;
+}
+
+static const struct pinmux_ops upboard_pinmux_ops = {
+	.get_functions_count = pinmux_generic_get_function_count,
+	.get_function_name = pinmux_generic_get_function_name,
+	.get_function_groups = pinmux_generic_get_function_groups,
+	.set_mux = upboard_pinctrl_set_mux,
+	.gpio_request_enable = upboard_pinctrl_gpio_request_enable,
+	.gpio_disable_free = upboard_pinctrl_gpio_disable_free,
+	.gpio_set_direction = upboard_pinctrl_gpio_set_direction,
+};
+
+static int upboard_pinctrl_pin_get_mode(struct pinctrl_dev *pctldev, unsigned int pin)
+{
+	struct upboard_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev);
+	struct upboard_pin *p = &pctrl->pins[pin];
+
+	unsigned int val;
+	int ret;
+
+	if (p->funcbit) {
+		ret = regmap_field_read(p->funcbit, &val);
+		if (ret)
+			return ret;
+		if (val)
+			return UPBOARD_PIN_MODE_FUNCTION;
+	}
+
+	ret = regmap_field_read(p->enbit, &val);
+	if (ret)
+		return ret;
+	if (!val)
+		return UPBOARD_PIN_MODE_DISABLED;
+
+	ret = regmap_field_read(p->dirbit, &val);
+	if (ret)
+		return ret;
+
+	return val ? UPBOARD_PIN_MODE_GPIO_IN : UPBOARD_PIN_MODE_GPIO_OUT;
+}
+
+static void upboard_pinctrl_dbg_show(struct pinctrl_dev *pctldev, struct seq_file *s,
+				     unsigned int offset)
+{
+	int ret = upboard_pinctrl_pin_get_mode(pctldev, offset);
+
+	if (ret == UPBOARD_PIN_MODE_FUNCTION)
+		seq_puts(s, "mode function ");
+	else if (ret == UPBOARD_PIN_MODE_DISABLED)
+		seq_puts(s, "HIGH-Z");
+	else
+		seq_printf(s, "GPIO (%s) ", ret == UPBOARD_PIN_MODE_GPIO_IN ? "input" : "output");
+}
+
+static const struct pinctrl_ops upboard_pinctrl_ops = {
+	.get_groups_count = pinctrl_generic_get_group_count,
+	.get_group_name = pinctrl_generic_get_group_name,
+	.get_group_pins = pinctrl_generic_get_group_pins,
+	.pin_dbg_show = upboard_pinctrl_dbg_show,
+};
+
+static int upboard_gpio_request(struct gpio_chip *gc, unsigned int offset)
+{
+	struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
+	unsigned int pin = pctrl->pctrl_data->pin_header[offset];
+	int ret;
+
+	ret = pinctrl_gpio_request(gc, offset);
+	if (ret)
+		return ret;
+
+	pctrl->gpio[offset] = gpiod_get_index(pctrl->dev, "external", pin, 0);
+
+	if (IS_ERR(pctrl->gpio[offset])) {
+		pinctrl_gpio_free(gc, offset);
+		return PTR_ERR(pctrl->gpio[offset]);
+	}
+
+	return 0;
+}
+
+static void upboard_gpio_free(struct gpio_chip *gc, unsigned int offset)
+{
+	struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
+
+	gpiod_put(pctrl->gpio[offset]);
+
+	pinctrl_gpio_free(gc, offset);
+}
+
+static int upboard_gpio_get_direction(struct gpio_chip *gc, unsigned int offset)
+{
+	struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
+	unsigned int pin = pctrl->pctrl_data->pin_header[offset];
+	int mode;
+
+	if (pctrl->gpio[offset])
+		return gpiod_get_direction(pctrl->gpio[offset]);
+
+	/*
+	 * GPIO was not requested so SoC pin is probably not in GPIO mode.
+	 * When a gpio_chip is registered, the core calls get_direction() for all lines.
+	 * At this time, upboard_gpio_request() was not yet called, so the driver didn't
+	 * request the corresponding SoC pin. So the SoC pin is probably in function (not in
+	 * GPIO mode).
+	 *
+	 * To get the direction of the SoC pin, it shall be requested in GPIO mode.
+	 * Once a SoC pin is set in GPIO mode, there is no way to set it back to its
+	 * function mode.
+	 * Instead of returning the SoC pin direction, the direction of the FPGA pin is
+	 * returned (only for the get_direction() called during the gpio_chip registration).
+	 */
+	mode = upboard_pinctrl_pin_get_mode(pctrl->pctldev, pin);
+
+	/* If the pin is in function mode or high-z, input direction is returned */
+	if (mode < 0)
+		return mode;
+	if (mode == UPBOARD_PIN_MODE_GPIO_OUT)
+		return GPIO_LINE_DIRECTION_OUT;
+	else
+		return GPIO_LINE_DIRECTION_IN;
+}
+
+static int upboard_gpio_get(struct gpio_chip *gc, unsigned int offset)
+{
+	struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
+
+	return gpiod_get_value(pctrl->gpio[offset]);
+}
+
+static void upboard_gpio_set(struct gpio_chip *gc, unsigned int offset, int value)
+{
+	struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
+
+	gpiod_set_value(pctrl->gpio[offset], value);
+}
+
+static int upboard_gpio_direction_input(struct gpio_chip *gc, unsigned int offset)
+{
+	struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
+	int ret;
+
+	ret = pinctrl_gpio_direction_input(gc, offset);
+	if (ret)
+		return ret;
+
+	return gpiod_direction_input(pctrl->gpio[offset]);
+}
+
+static int upboard_gpio_direction_output(struct gpio_chip *gc, unsigned int offset, int value)
+{
+	struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
+	int ret;
+
+	ret = pinctrl_gpio_direction_output(gc, offset);
+	if (ret)
+		return ret;
+
+	return gpiod_direction_output(pctrl->gpio[offset], value);
+}
+
+static int upboard_gpio_to_irq(struct gpio_chip *gc, unsigned int offset)
+{
+	struct upboard_pinctrl *pctrl = container_of(gc, struct upboard_pinctrl, chip);
+
+	return gpiod_to_irq(pctrl->gpio[offset]);
+}
+
+static int upboard_pinctrl_register_groups(struct upboard_pinctrl *pctrl)
+{
+	const struct upboard_pingroup *groups = pctrl->pctrl_data->groups;
+	size_t ngroups = pctrl->pctrl_data->ngroups;
+	int ret, i;
+
+	for (i = 0; i < ngroups; i++) {
+		ret = pinctrl_generic_add_group(pctrl->pctldev, groups[i].grp.name,
+						groups[i].grp.pins, groups[i].grp.npins, pctrl);
+		if (ret < 0)
+			return ret;
+	}
+
+	return 0;
+}
+
+static int upboard_pinctrl_register_functions(struct upboard_pinctrl *pctrl)
+{
+	const struct pinfunction *funcs = pctrl->pctrl_data->funcs;
+	size_t nfuncs = pctrl->pctrl_data->nfuncs;
+	int ret, i;
+
+	for (i = 0; i < nfuncs ; ++i) {
+		ret = pinmux_generic_add_function(pctrl->pctldev, funcs[i].name, funcs[i].groups,
+						  funcs[i].ngroups, NULL);
+		if (ret < 0)
+			return ret;
+	}
+
+	return 0;
+}
+
+static int upboard_pinctrl_probe(struct platform_device *pdev)
+{
+	struct device *dev = &pdev->dev;
+	struct upboard_fpga *fpga = dev_get_drvdata(dev->parent);
+	struct pinctrl_desc *pctldesc;
+	struct upboard_pinctrl *pctrl;
+	struct upboard_pin *pins;
+	struct gpio_chip *chip;
+	int ret, i;
+
+	pctldesc = devm_kzalloc(dev, sizeof(*pctldesc), GFP_KERNEL);
+	if (!pctldesc)
+		return -ENOMEM;
+
+	pctrl = devm_kzalloc(dev, sizeof(*pctrl), GFP_KERNEL);
+	if (!pctrl)
+		return -ENOMEM;
+
+	switch (fpga->fpga_data->type) {
+	case UPBOARD_UP_FPGA:
+		pctldesc->pins = upboard_up_pins;
+		pctldesc->npins = ARRAY_SIZE(upboard_up_pins);
+		pctrl->pctrl_data = &upboard_up_pinctrl_data;
+		break;
+	case UPBOARD_UP2_FPGA:
+		pctldesc->pins = upboard_up2_pins;
+		pctldesc->npins = ARRAY_SIZE(upboard_up2_pins);
+		pctrl->pctrl_data = &upboard_up2_pinctrl_data;
+		break;
+	default:
+		return dev_err_probe(dev, -ENODEV, "Unsupported device type %d\n",
+				     fpga->fpga_data->type);
+	}
+
+	pctldesc->name = dev_name(dev);
+	pctldesc->owner = THIS_MODULE;
+	pctldesc->pctlops = &upboard_pinctrl_ops;
+	pctldesc->pmxops = &upboard_pinmux_ops;
+
+	pctrl->dev = dev;
+
+	pins = devm_kcalloc(dev, pctldesc->npins, sizeof(*pins), GFP_KERNEL);
+	if (!pins)
+		return -ENOMEM;
+
+	/* Initialize pins */
+	for (i = 0; i < pctldesc->npins; i++) {
+		const struct pinctrl_pin_desc *pin_desc = &pctldesc->pins[i];
+		unsigned int regoff = pin_desc->number / UPBOARD_REGISTER_SIZE;
+		unsigned int lsb = pin_desc->number % UPBOARD_REGISTER_SIZE;
+		struct reg_field * const fld_func = pin_desc->drv_data;
+		struct upboard_pin *pin = &pins[i];
+		struct reg_field fldconf = {};
+
+		if (fld_func) {
+			pin->funcbit = devm_regmap_field_alloc(dev, fpga->regmap, *fld_func);
+
+			if (IS_ERR(pin->funcbit))
+				return PTR_ERR(pin->funcbit);
+		}
+
+		fldconf.reg = UPBOARD_REG_GPIO_EN0 + regoff;
+		fldconf.lsb = lsb;
+		fldconf.msb = lsb;
+		pin->enbit = devm_regmap_field_alloc(dev, fpga->regmap, fldconf);
+		if (IS_ERR(pin->enbit))
+			return PTR_ERR(pin->enbit);
+
+		fldconf.reg = UPBOARD_REG_GPIO_DIR0 + regoff;
+		fldconf.lsb = lsb;
+		fldconf.msb = lsb;
+		pin->dirbit = devm_regmap_field_alloc(dev, fpga->regmap, fldconf);
+		if (IS_ERR(pin->dirbit))
+			return PTR_ERR(pin->dirbit);
+	}
+
+	pctrl->pins = pins;
+
+	ret = pinctrl_register_mappings(pctrl->pctrl_data->maps, pctrl->pctrl_data->nmaps);
+
+	ret = devm_pinctrl_register_and_init(dev, pctldesc, pctrl, &pctrl->pctldev);
+	if (ret)
+		return ret;
+
+	ret = upboard_pinctrl_register_groups(pctrl);
+	if (ret)
+		return dev_err_probe(dev, ret, "Failed to register groups");
+
+	ret = upboard_pinctrl_register_functions(pctrl);
+	if (ret)
+		return dev_err_probe(dev, ret, "Failed to register functions");
+
+	ret = pinctrl_enable(pctrl->pctldev);
+	if (ret)
+		return dev_err_probe(dev, ret, "Failed to enable pinctrl");
+
+	chip = &pctrl->chip;
+	chip->owner = THIS_MODULE;
+	chip->label = dev_name(&pdev->dev);
+	chip->parent = &pdev->dev;
+	chip->ngpio = pctrl->pctrl_data->ngpio;
+	chip->base = -1;
+	chip->request = upboard_gpio_request;
+	chip->free = upboard_gpio_free;
+	chip->get = upboard_gpio_get;
+	chip->set = upboard_gpio_set;
+	chip->get_direction = upboard_gpio_get_direction;
+	chip->direction_input = upboard_gpio_direction_input;
+	chip->direction_output = upboard_gpio_direction_output;
+	chip->to_irq = upboard_gpio_to_irq;
+
+	pctrl->gpio = devm_kcalloc(dev, pctrl->pctrl_data->ngpio, sizeof(*pctrl->gpio), GFP_KERNEL);
+	if (!pctrl->gpio)
+		return -ENOMEM;
+
+	ret = devm_gpiochip_add_data(&pdev->dev, chip, pctrl);
+	if (ret)
+		return ret;
+
+	ret = gpiochip_add_pinlist_range(chip, dev_name(dev), 0, pctrl->pctrl_data->pin_header,
+					 pctrl->pctrl_data->ngpio);
+	if (ret)
+		gpiochip_remove(chip);
+
+	return ret;
+}
+
+static struct platform_driver upboard_pinctrl_driver = {
+	.driver = {
+		.name = "upboard-pinctrl",
+	},
+	.probe = upboard_pinctrl_probe,
+};
+
+module_platform_driver(upboard_pinctrl_driver);
+
+MODULE_AUTHOR("Gary Wang <garywang@aaeon.com.tw>");
+MODULE_AUTHOR("Thomas Richard <thomas.richard@bootlin.com");
+MODULE_DESCRIPTION("UP Board HAT pin controller driver");
+MODULE_LICENSE("GPL");
+MODULE_ALIAS("platform:upboard-pinctrl");