[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Message-ID: <86bf1252-9b2b-4001-830d-2746403539e6@foss.st.com>
Date: Fri, 3 Oct 2025 09:40:07 +0200
From: Arnaud POULIQUEN <arnaud.pouliquen@...s.st.com>
To: Shenwei Wang <shenwei.wang@....com>,
Bjorn Andersson
<andersson@...nel.org>,
Mathieu Poirier <mathieu.poirier@...aro.org>,
"Rob
Herring" <robh@...nel.org>,
Krzysztof Kozlowski <krzk+dt@...nel.org>,
"Conor
Dooley" <conor+dt@...nel.org>, Shawn Guo <shawnguo@...nel.org>,
Sascha Hauer
<s.hauer@...gutronix.de>,
Linus Walleij <linus.walleij@...aro.org>,
"Bartosz
Golaszewski" <brgl@...ev.pl>
CC: Pengutronix Kernel Team <kernel@...gutronix.de>,
Fabio Estevam
<festevam@...il.com>, Peng Fan <peng.fan@....com>,
<linux-remoteproc@...r.kernel.org>, <devicetree@...r.kernel.org>,
<imx@...ts.linux.dev>, <linux-arm-kernel@...ts.infradead.org>,
<linux-kernel@...r.kernel.org>, <linux-imx@....com>,
<openamp-rp@...ts.openampproject.org>
Subject: Re: [PATCH v2 3/4] gpio: imx-rpmsg: add imx-rpmsg GPIO driver
Hello Shenwei,
On 9/22/25 22:04, Shenwei Wang wrote:
> On i.MX SoCs, the system may include two processors:
> - An MCU running an RTOS
> - An MPU running Linux
>
> These processors communicate via the RPMSG protocol.
> The driver implements the standard GPIO interface, allowing
> the Linux side to control GPIO controllers which reside in
> the remote processor via RPMSG protocol.
What about my request in previous version to make this driver generic?
In ST we have similar driver in downstream, we would be interested in
reusing it if generic. Indeed we need some rpmsg mechanism for
gpio-interrupt. Today we have a downstream rpmsg driver [1][2] that could
migrate on a generic rpmsg-gpio driver.
[1]Documentation/devicetree/bindings/interrupt-controller/rpmsg,intc.yaml
[2]https://github.com/STMicroelectronics/linux/blob/v6.6-stm32mp/drivers/irqchip/irq-rpmsg.c
My comment below is based on the assumption that it would become generic.
> Signed-off-by: Shenwei Wang <shenwei.wang@....com>
> ---
> drivers/gpio/Kconfig | 17 ++
> drivers/gpio/Makefile | 1 +
> drivers/gpio/gpio-imx-rpmsg.c | 488 ++++++++++++++++++++++++++++++++++
> 3 files changed, 506 insertions(+)
> create mode 100644 drivers/gpio/gpio-imx-rpmsg.c
>
> diff --git a/drivers/gpio/Kconfig b/drivers/gpio/Kconfig
> index a437fe652dbc..97eda94b0ba1 100644
> --- a/drivers/gpio/Kconfig
> +++ b/drivers/gpio/Kconfig
> @@ -1847,6 +1847,23 @@ config GPIO_SODAVILLE
>
> endmenu
>
> +menu "RPMSG GPIO drivers"
> + depends on RPMSG
> +
> +config GPIO_IMX_RPMSG
> + tristate "NXP i.MX SoC RPMSG GPIO support"
> + depends on IMX_REMOTEPROC
> + select GPIOLIB_IRQCHIP
> + default IMX_REMOTEPROC
> + help
> + Say yes here to support the RPMSG GPIO functions on i.MX SoC based
> + platform. Currently supported devices: i.MX7ULP, i.MX8ULP, i.MX8x,
> + and i.MX9x.
> +
> + If unsure, say N.
> +
> +endmenu
> +
> menu "SPI GPIO expanders"
> depends on SPI_MASTER
>
> diff --git a/drivers/gpio/Makefile b/drivers/gpio/Makefile
> index 379f55e9ed1e..e01465c03431 100644
> --- a/drivers/gpio/Makefile
> +++ b/drivers/gpio/Makefile
> @@ -83,6 +83,7 @@ obj-$(CONFIG_GPIO_I8255) += gpio-i8255.o
> obj-$(CONFIG_GPIO_ICH) += gpio-ich.o
> obj-$(CONFIG_GPIO_IDIO_16) += gpio-idio-16.o
> obj-$(CONFIG_GPIO_IDT3243X) += gpio-idt3243x.o
> +obj-$(CONFIG_GPIO_IMX_RPMSG) += gpio-imx-rpmsg.o
> obj-$(CONFIG_GPIO_IMX_SCU) += gpio-imx-scu.o
> obj-$(CONFIG_GPIO_IT87) += gpio-it87.o
> obj-$(CONFIG_GPIO_IXP4XX) += gpio-ixp4xx.o
> diff --git a/drivers/gpio/gpio-imx-rpmsg.c b/drivers/gpio/gpio-imx-rpmsg.c
> new file mode 100644
> index 000000000000..46b1b6b798c8
> --- /dev/null
> +++ b/drivers/gpio/gpio-imx-rpmsg.c
> @@ -0,0 +1,488 @@
> +// SPDX-License-Identifier: GPL-2.0-only
> +/*
> + * Copyright 2025 NXP
> + *
> + * The driver exports a standard gpiochip interface to control
> + * the GPIO controllers via RPMSG on a remote processor.
> + */
> +
> +#include <linux/err.h>
> +#include <linux/gpio/driver.h>
> +#include <linux/rpmsg/imx_rpmsg.h>
> +#include <linux/init.h>
> +#include <linux/irqdomain.h>
> +#include <linux/of.h>
> +#include <linux/platform_device.h>
> +#include <linux/rpmsg.h>
> +
> +#define IMX_RPMSG_GPIO_PER_PORT 32
> +#define RPMSG_TIMEOUT 1000
> +
> +enum gpio_input_trigger_type {
> + GPIO_RPMSG_TRI_IGNORE,
> + GPIO_RPMSG_TRI_RISING,
> + GPIO_RPMSG_TRI_FALLING,
> + GPIO_RPMSG_TRI_BOTH_EDGE,
> + GPIO_RPMSG_TRI_LOW_LEVEL,
> + GPIO_RPMSG_TRI_HIGH_LEVEL,
> +};
What about taking inspiration from the|IRQ_TYPE|bitfield defined in|irq.h|?
For instance:
GPIO_RPMSG_TRI_BOTH_EDGE = GPIO_RPMSG_TRI_FALLING | GPIO_RPMSG_TRI_RISING,
> +
> +enum gpio_rpmsg_header_type {
> + GPIO_RPMSG_SETUP,
> + GPIO_RPMSG_REPLY,
> + GPIO_RPMSG_NOTIFY,
> +};
> +
> +enum gpio_rpmsg_header_cmd {
> + GPIO_RPMSG_INPUT_INIT,
> + GPIO_RPMSG_OUTPUT_INIT,
> + GPIO_RPMSG_INPUT_GET,
> +};
> +
> +struct gpio_rpmsg_data {
> + struct imx_rpmsg_head header;
> + u8 pin_idx;
> + u8 port_idx;
> + union {
> + u8 event;
> + u8 retcode;
> + u8 value;
> + } out;
> + union {
> + u8 wakeup;
> + u8 value;
nitpicking put "value" field out of union as common
> + } in;
> +} __packed __aligned(8);
Any reason to pack it an align it?
This structure will be copied in the RPMSg payload, right?
I also wonder if that definition should not be in a header file
with double licensing that the DT. Indeed this need to be
common with the remote side implementation that can
be non GPL implementation.
> +
> +struct imx_rpmsg_gpio_pin {
> + u8 irq_shutdown;
> + u8 irq_unmask;
> + u8 irq_mask;
> + u32 irq_wake_enable;
> + u32 irq_type;
> + struct gpio_rpmsg_data msg;
> +};
> +
> +struct imx_gpio_rpmsg_info {
> + struct rpmsg_device *rpdev;
> + struct gpio_rpmsg_data *notify_msg;
> + struct gpio_rpmsg_data *reply_msg;
> + struct completion cmd_complete;
> + struct mutex lock;
> + void **port_store;
> +};
> +
> +struct imx_rpmsg_gpio_port {
> + struct gpio_chip gc;
> + struct imx_rpmsg_gpio_pin gpio_pins[IMX_RPMSG_GPIO_PER_PORT];
> + struct imx_gpio_rpmsg_info info;
> + int idx;
> +};
> +
> +static int gpio_send_message(struct imx_rpmsg_gpio_port *port,
> + struct gpio_rpmsg_data *msg,
> + bool sync)
> +{
> + struct imx_gpio_rpmsg_info *info = &port->info;
> + int err;
> +
> + if (!info->rpdev) {
> + dev_dbg(&info->rpdev->dev,
> + "rpmsg channel not ready, m4 image ready?\n");
> + return -EINVAL;
> + }
> +
> + reinit_completion(&info->cmd_complete);
> + err = rpmsg_send(info->rpdev->ept, (void *)msg,
> + sizeof(struct gpio_rpmsg_data));
> + if (err) {
> + dev_err(&info->rpdev->dev, "rpmsg_send failed: %d\n", err);
> + return err;
> + }
> +
> + if (sync) {
> + err = wait_for_completion_timeout(&info->cmd_complete,
> + msecs_to_jiffies(RPMSG_TIMEOUT));
> + if (!err) {
> + dev_err(&info->rpdev->dev, "rpmsg_send timeout!\n");
> + return -ETIMEDOUT;
> + }
> +
> + if (info->reply_msg->out.retcode != 0) {
> + dev_err(&info->rpdev->dev, "rpmsg not ack %d!\n",
> + info->reply_msg->out.retcode);
> + return -EINVAL;
> + }
> +
> + /* copy the reply message */
> + memcpy(&port->gpio_pins[info->reply_msg->pin_idx].msg,
> + info->reply_msg, sizeof(*info->reply_msg));
> + }
> +
> + return 0;
> +}
> +
> +static struct gpio_rpmsg_data *gpio_get_pin_msg(struct imx_rpmsg_gpio_port *port,
> + unsigned int offset)
> +{
> + struct gpio_rpmsg_data *msg = &port->gpio_pins[offset].msg;
> +
> + memset(msg, 0, sizeof(struct gpio_rpmsg_data));
> +
> + return msg;
> +};
> +
> +static int imx_rpmsg_gpio_get(struct gpio_chip *gc, unsigned int gpio)
> +{
> + struct imx_rpmsg_gpio_port *port = gpiochip_get_data(gc);
> + struct gpio_rpmsg_data *msg = NULL;
> + int ret;
> +
> + guard(mutex)(&port->info.lock);
> +
> + msg = gpio_get_pin_msg(port, gpio);
> + msg->header.cate = IMX_RPMSG_GPIO;
> + msg->header.major = IMX_RMPSG_MAJOR;
> + msg->header.minor = IMX_RMPSG_MINOR;
> + msg->header.type = GPIO_RPMSG_SETUP;
> + msg->header.cmd = GPIO_RPMSG_INPUT_GET;
> + msg->pin_idx = gpio;
> + msg->port_idx = port->idx;
> +
> + ret = gpio_send_message(port, msg, true);
> + if (!ret)
> + ret = !!port->gpio_pins[gpio].msg.in.value;
Does this code is save? !! return a boolean right?
why not force to 1 if greater that 1?
> +
> + return ret;
> +}
> +
> +static int imx_rpmsg_gpio_direction_input(struct gpio_chip *gc,
> + unsigned int gpio)
> +{
> + struct imx_rpmsg_gpio_port *port = gpiochip_get_data(gc);
> + struct gpio_rpmsg_data *msg = NULL;
> +
> + guard(mutex)(&port->info.lock);
> +
> + msg = gpio_get_pin_msg(port, gpio);
> + msg->header.cate = IMX_RPMSG_GPIO;
Do you use a single rpmsg channel for several feature?
Any reason to not use one rpmsg channel per feature category?
> + msg->header.major = IMX_RMPSG_MAJOR;
> + msg->header.minor = IMX_RMPSG_MINOR;
> + msg->header.type = GPIO_RPMSG_SETUP;
> + msg->header.cmd = GPIO_RPMSG_INPUT_INIT;
> + msg->pin_idx = gpio;
> + msg->port_idx = port->idx;
> +
> + msg->out.event = GPIO_RPMSG_TRI_IGNORE;
> + msg->in.wakeup = 0;
> +
> + return gpio_send_message(port, msg, true);
> +}
> +
> +static inline void imx_rpmsg_gpio_direction_output_init(struct gpio_chip *gc,
> + unsigned int gpio, int val, struct gpio_rpmsg_data *msg)
> +{
> + struct imx_rpmsg_gpio_port *port = gpiochip_get_data(gc);
> +
> + msg->header.cate = IMX_RPMSG_GPIO;
> + msg->header.major = IMX_RMPSG_MAJOR;
> + msg->header.minor = IMX_RMPSG_MINOR;
> + msg->header.type = GPIO_RPMSG_SETUP;
> + msg->header.cmd = GPIO_RPMSG_OUTPUT_INIT;
> + msg->pin_idx = gpio;
> + msg->port_idx = port->idx;
> + msg->out.value = val;
> +}
> +
> +static int imx_rpmsg_gpio_set(struct gpio_chip *gc, unsigned int gpio, int val)
> +{
> + struct imx_rpmsg_gpio_port *port = gpiochip_get_data(gc);
> + struct gpio_rpmsg_data *msg = NULL;
> +
> + guard(mutex)(&port->info.lock);
> +
> + msg = gpio_get_pin_msg(port, gpio);
> + imx_rpmsg_gpio_direction_output_init(gc, gpio, val, msg);
> +
> + return gpio_send_message(port, msg, true);
> +}
> +
> +static int imx_rpmsg_gpio_direction_output(struct gpio_chip *gc,
> + unsigned int gpio, int val)
> +{
> + struct imx_rpmsg_gpio_port *port = gpiochip_get_data(gc);
> + struct gpio_rpmsg_data *msg = NULL;
> +
> + guard(mutex)(&port->info.lock);
> +
> + msg = gpio_get_pin_msg(port, gpio);
> + imx_rpmsg_gpio_direction_output_init(gc, gpio, val, msg);
> +
> + return gpio_send_message(port, msg, true);
> +}
> +
> +static int imx_rpmsg_irq_set_type(struct irq_data *d, u32 type)
> +{
> + struct imx_rpmsg_gpio_port *port = irq_data_get_irq_chip_data(d);
> + u32 gpio_idx = d->hwirq;
> + int edge = 0;
> + int ret = 0;
> +
> + switch (type) {
> + case IRQ_TYPE_EDGE_RISING:
> + edge = GPIO_RPMSG_TRI_RISING;
> + break;
> + case IRQ_TYPE_EDGE_FALLING:
> + edge = GPIO_RPMSG_TRI_FALLING;
> + break;
> + case IRQ_TYPE_EDGE_BOTH:
> + edge = GPIO_RPMSG_TRI_BOTH_EDGE;
> + break;
> + case IRQ_TYPE_LEVEL_LOW:
> + edge = GPIO_RPMSG_TRI_LOW_LEVEL;
> + break;
> + case IRQ_TYPE_LEVEL_HIGH:
> + edge = GPIO_RPMSG_TRI_HIGH_LEVEL;
> + break;
> + default:
> + ret = -EINVAL;
> + break;
> + }
> +
> + port->gpio_pins[gpio_idx].irq_type = edge;
> +
> + return ret;
> +}
> +
> +static int imx_rpmsg_irq_set_wake(struct irq_data *d, u32 enable)
> +{
> + struct imx_rpmsg_gpio_port *port = irq_data_get_irq_chip_data(d);
> + u32 gpio_idx = d->hwirq;
> +
> + port->gpio_pins[gpio_idx].irq_wake_enable = enable;
> +
> + return 0;
> +}
> +
> +/*
> + * This function will be called at:
> + * - one interrupt setup.
> + * - the end of one interrupt happened
> + * The gpio over rpmsg driver will not write the real register, so save
> + * all infos before this function and then send all infos to M core in this
> + * step.
> + */
> +static void imx_rpmsg_unmask_irq(struct irq_data *d)
> +{
> + struct imx_rpmsg_gpio_port *port = irq_data_get_irq_chip_data(d);
> + u32 gpio_idx = d->hwirq;
> +
> + port->gpio_pins[gpio_idx].irq_unmask = 1;
> +}
> +
> +static void imx_rpmsg_mask_irq(struct irq_data *d)
> +{
> + struct imx_rpmsg_gpio_port *port = irq_data_get_irq_chip_data(d);
> + u32 gpio_idx = d->hwirq;
> + /*
> + * No need to implement the callback at A core side.
> + * M core will mask interrupt after a interrupt occurred, and then
> + * sends a notify to A core.
> + * After A core dealt with the notify, A core will send a rpmsg to
> + * M core to unmask this interrupt again.
> + */
> + port->gpio_pins[gpio_idx].irq_mask = 1;
> +}
> +
> +static void imx_rpmsg_irq_shutdown(struct irq_data *d)
> +{
> + struct imx_rpmsg_gpio_port *port = irq_data_get_irq_chip_data(d);
> + u32 gpio_idx = d->hwirq;
> +
> + port->gpio_pins[gpio_idx].irq_shutdown = 1;
> +}
> +
> +static void imx_rpmsg_irq_bus_lock(struct irq_data *d)
> +{
> + struct imx_rpmsg_gpio_port *port = irq_data_get_irq_chip_data(d);
> +
> + mutex_lock(&port->info.lock);
> +}
> +
> +static void imx_rpmsg_irq_bus_sync_unlock(struct irq_data *d)
> +{
> + struct imx_rpmsg_gpio_port *port = irq_data_get_irq_chip_data(d);
> + struct gpio_rpmsg_data *msg = NULL;
> + u32 gpio_idx = d->hwirq;
> +
> + if (port == NULL) {
> + mutex_unlock(&port->info.lock);
> + return;
> + }
> +
> + /*
> + * For mask irq, do nothing here.
> + * M core will mask interrupt after a interrupt occurred, and then
> + * sends a notify to A core.
> + * After A core dealt with the notify, A core will send a rpmsg to
> + * M core to unmask this interrupt again.
> + */
> +
> + if (port->gpio_pins[gpio_idx].irq_mask && !port->gpio_pins[gpio_idx].irq_unmask) {
> + port->gpio_pins[gpio_idx].irq_mask = 0;
> + mutex_unlock(&port->info.lock);
> + return;
> + }
> +
> + msg = gpio_get_pin_msg(port, gpio_idx);
> + msg->header.cate = IMX_RPMSG_GPIO;
> + msg->header.major = IMX_RMPSG_MAJOR;
> + msg->header.minor = IMX_RMPSG_MINOR;
> + msg->header.type = GPIO_RPMSG_SETUP;
> + msg->header.cmd = GPIO_RPMSG_INPUT_INIT;
> + msg->pin_idx = gpio_idx;
> + msg->port_idx = port->idx;
> +
> + if (port->gpio_pins[gpio_idx].irq_shutdown) {
> + msg->out.event = GPIO_RPMSG_TRI_IGNORE;
> + msg->in.wakeup = 0;
> + port->gpio_pins[gpio_idx].irq_shutdown = 0;
> + } else {
> + /* if not set irq type, then use low level as trigger type */
> + msg->out.event = port->gpio_pins[gpio_idx].irq_type;
> + if (!msg->out.event)
> + msg->out.event = GPIO_RPMSG_TRI_LOW_LEVEL;
> + if (port->gpio_pins[gpio_idx].irq_unmask) {
> + msg->in.wakeup = 0;
> + port->gpio_pins[gpio_idx].irq_unmask = 0;
> + } else /* irq set wake */
> + msg->in.wakeup = port->gpio_pins[gpio_idx].irq_wake_enable;
> + }
> +
> + gpio_send_message(port, msg, false);
> + mutex_unlock(&port->info.lock);
> +}
> +
> +static const struct irq_chip imx_rpmsg_irq_chip = {
> + .irq_mask = imx_rpmsg_mask_irq,
> + .irq_unmask = imx_rpmsg_unmask_irq,
> + .irq_set_wake = imx_rpmsg_irq_set_wake,
> + .irq_set_type = imx_rpmsg_irq_set_type,
> + .irq_shutdown = imx_rpmsg_irq_shutdown,
> + .irq_bus_lock = imx_rpmsg_irq_bus_lock,
> + .irq_bus_sync_unlock = imx_rpmsg_irq_bus_sync_unlock,
> + .flags = IRQCHIP_IMMUTABLE,
> +};
> +
> +static int imx_rpmsg_gpio_callback(struct rpmsg_device *rpdev,
> + void *data, int len, void *priv, u32 src)
> +{
> + struct gpio_rpmsg_data *msg = (struct gpio_rpmsg_data *)data;
> + struct imx_rpmsg_gpio_port *port = NULL;
> + struct imx_rpmsg_driver_data *drvdata;
> + unsigned long flags;
> +
> + drvdata = dev_get_drvdata(&rpdev->dev);
> + if (msg)
> + port = drvdata->channel_devices[msg->port_idx];
> +
> + if (!port)
> + return -ENODEV;
> +
> + if (msg->header.type == GPIO_RPMSG_REPLY) {
> + port->info.reply_msg = msg;
> + complete(&port->info.cmd_complete);
> + } else if (msg->header.type == GPIO_RPMSG_NOTIFY) {
> + port->info.notify_msg = msg;
> + local_irq_save(flags);
> + generic_handle_domain_irq(port->gc.irq.domain, msg->pin_idx);
> + local_irq_restore(flags);
> + } else
> + dev_err(&rpdev->dev, "wrong command type!\n");
> +
> + return 0;
> +}
> +
> +static void imx_rpmsg_gpio_remove_action(void *data)
> +{
> + struct imx_rpmsg_gpio_port *port = data;
> +
> + port->info.port_store[port->idx] = 0;
> +}
> +
> +static int imx_rpmsg_gpio_probe(struct platform_device *pdev)
> +{
> + struct imx_rpmsg_driver_data *pltdata = pdev->dev.platform_data;
> + struct device_node *np = pdev->dev.of_node;
> + struct imx_rpmsg_gpio_port *port;
> + struct gpio_irq_chip *girq;
> + struct gpio_chip *gc;
> + int ret;
> +
> + if (!pltdata)
> + return -EPROBE_DEFER;
> +
> + port = devm_kzalloc(&pdev->dev, sizeof(*port), GFP_KERNEL);
> + if (!port)
> + return -ENOMEM;
> +
> + ret = of_property_read_u32(np, "reg", &port->idx);
> + if (ret)
> + return ret;
> +
> + if (port->idx > MAX_DEV_PER_CHANNEL)
> + return -EINVAL;
> +
> + mutex_init(&port->info.lock);
> + init_completion(&port->info.cmd_complete);
> + port->info.rpdev = pltdata->rpdev;
> + port->info.port_store = pltdata->channel_devices;
> + port->info.port_store[port->idx] = port;
> + if (!pltdata->rx_callback)
> + pltdata->rx_callback = imx_rpmsg_gpio_callback;
> +
> + gc = &port->gc;
> + gc->owner = THIS_MODULE;
> + gc->parent = &pltdata->rpdev->dev;
> + gc->label = devm_kasprintf(&pdev->dev, GFP_KERNEL, "%s-gpio%d",
> + pltdata->rproc_name, port->idx);
> + gc->ngpio = IMX_RPMSG_GPIO_PER_PORT;
> + gc->base = -1;
> +
> + gc->direction_input = imx_rpmsg_gpio_direction_input;
> + gc->direction_output = imx_rpmsg_gpio_direction_output;
> + gc->get = imx_rpmsg_gpio_get;
> + gc->set = imx_rpmsg_gpio_set;
> +
> + platform_set_drvdata(pdev, port);
> + girq = &gc->irq;
> + gpio_irq_chip_set_chip(girq, &imx_rpmsg_irq_chip);
> + girq->parent_handler = NULL;
> + girq->num_parents = 0;
> + girq->parents = NULL;
> + girq->chip->name = devm_kasprintf(&pdev->dev, GFP_KERNEL, "%s-gpio%d",
> + pltdata->rproc_name, port->idx);
> +
> + devm_add_action_or_reset(&pdev->dev, imx_rpmsg_gpio_remove_action, port);
> +
> + return devm_gpiochip_add_data(&pdev->dev, gc, port);
> +}
> +
> +static const struct of_device_id imx_rpmsg_gpio_dt_ids[] = {
> + { .compatible = "fsl,imx-rpmsg-gpio" },
> + { /* sentinel */ }
> +};
> +
> +static struct platform_driver imx_rpmsg_gpio_driver = {
> + .driver = {
> + .name = "gpio-imx-rpmsg",
> + .of_match_table = imx_rpmsg_gpio_dt_ids,
> + },
> + .probe = imx_rpmsg_gpio_probe,
> +};
> +
> +module_platform_driver(imx_rpmsg_gpio_driver);
This implementation seems strange to me.
You have a platform driver, but your RPMsg driver appears split between
this driver
and the remoteproc driver, especially regarding the
imx_rpmsg_endpoint_probe() function.
From my point of view, this driver should declare both a
platform_driver and an
rpmsg_driver structures.
Your imx_rpmsg_gpio_driver platform driver should be probed by your
remoteproc platform.
This platform driver would be responsible for:
- Parsing the device tree node
- Registering the RPMsg driver
Then, the RPMsg device should be probed either by the remote processor
using the name service
announcement mechanism or if not possible by your remoteproc driver.
To better understand my proposal you can have a look to [1]and [2].
Here is another example for an rpmsg_i2c( ST downstream implementation):
https://github.com/STMicroelectronics/linux/blob/v6.6-stm32mp/drivers/i2c/busses/i2c-rpmsg.c
https://github.com/STMicroelectronics/linux/blob/v6.6-stm32mp/Documentation/devicetree/bindings/i2c/i2c-rpmsg.yaml
Thanks and Regards,
Arnaud
> +
> +MODULE_AUTHOR("NXP Semiconductor");
> +MODULE_DESCRIPTION("NXP i.MX SoC rpmsg gpio driver");
> +MODULE_LICENSE("GPL");
Powered by blists - more mailing lists