RE: [PATCH V6 2/4] firmware: imx: enable imx scu general irq function

2019-04-08 Thread Anson Huang


Best Regards!
Anson Huang

> -Original Message-
> From: Aisheng Dong
> Sent: 2019年4月9日 11:21
> To: Anson Huang ; robh...@kernel.org;
> mark.rutl...@arm.com; shawn...@kernel.org; s.ha...@pengutronix.de;
> ker...@pengutronix.de; feste...@gmail.com; a.zu...@towertech.it;
> alexandre.bell...@bootlin.com; ulf.hans...@linaro.org; sb...@kernel.org;
> Peng Fan ; Daniel Baluta ;
> devicet...@vger.kernel.org; linux-kernel@vger.kernel.org; linux-arm-
> ker...@lists.infradead.org; linux-...@vger.kernel.org
> Cc: dl-linux-imx 
> Subject: RE: [PATCH V6 2/4] firmware: imx: enable imx scu general irq
> function
> 
> > From: Anson Huang
> > Sent: Tuesday, April 9, 2019 10:43 AM
> > Subject: [PATCH V6 2/4] firmware: imx: enable imx scu general irq
> > function
> >
> > The System Controller Firmware (SCFW) controls RTC, thermal and WDOG
> > etc., these resources' interrupt function are managed by SCU. When any
> > IRQ pending, SCU will notify Linux via MU general interrupt channel
> > #3, and Linux kernel needs to call SCU APIs to get IRQ status and
> > notify each module to handle the interrupt.
> >
> > Since there is no data transmission for SCU IRQ notification, so
> > doorbell mode is used for this MU channel, and SCU driver will use
> > notifier mechanism to broadcast to every module which registers the SCU
> block notifier.
> >
> > Signed-off-by: Anson Huang 
> > ---
> > Changes since V5:
> > - use ATOMIC_NOTIFIER instead of BLOCKING_NOTIFIER for irq
> > notification;
> > - add memory free for failed case to avoid memory leak;
> > - add new API imx_scu_irq_enable() for modules to enable/disable
> > their own irqs.
> > ---
> >  drivers/firmware/imx/Makefile  |   2 +-
> >  drivers/firmware/imx/imx-scu-irq.c | 166
> > +
> >  drivers/firmware/imx/imx-scu.c |   6 ++
> >  include/linux/firmware/imx/sci.h   |   5 ++
> >  4 files changed, 178 insertions(+), 1 deletion(-)  create mode 100644
> > drivers/firmware/imx/imx-scu-irq.c
> >
> > diff --git a/drivers/firmware/imx/Makefile
> > b/drivers/firmware/imx/Makefile index 1b2e15b..802c4ad 100644
> > --- a/drivers/firmware/imx/Makefile
> > +++ b/drivers/firmware/imx/Makefile
> > @@ -1,3 +1,3 @@
> >  # SPDX-License-Identifier: GPL-2.0
> > -obj-$(CONFIG_IMX_SCU)  += imx-scu.o misc.o
> > +obj-$(CONFIG_IMX_SCU)  += imx-scu.o misc.o imx-scu-irq.o
> >  obj-$(CONFIG_IMX_SCU_PD)   += scu-pd.o
> > diff --git a/drivers/firmware/imx/imx-scu-irq.c
> > b/drivers/firmware/imx/imx-scu-irq.c
> > new file mode 100644
> > index 000..4000c63
> > --- /dev/null
> > +++ b/drivers/firmware/imx/imx-scu-irq.c
> > @@ -0,0 +1,166 @@
> > +// SPDX-License-Identifier: GPL-2.0+
> > +/*
> > + * Copyright 2019 NXP
> > + *
> > + * Implementation of the SCU IRQ functions using MU.
> > + *
> > + */
> > +
> > +#include  #include
> > + #include 
> > +
> > +#define IMX_SC_IRQ_FUNC_ENABLE 1
> > +#define IMX_SC_IRQ_FUNC_STATUS 2
> > +#define IMX_SC_IRQ_NUM_GROUP   4
> > +
> > +static u32 mu_resource_id;
> > +
> > +struct imx_sc_msg_irq_get_status {
> > +   struct imx_sc_rpc_msg hdr;
> > +   union {
> > +   struct {
> > +   u16 resource;
> > +   u8 group;
> > +   u8 reserved;
> > +   } __packed req;
> > +   struct {
> > +   u32 status;
> > +   } resp;
> > +   } data;
> > +};
> > +
> > +struct imx_sc_msg_irq_enable {
> > +   struct imx_sc_rpc_msg hdr;
> > +   u32 mask;
> > +   u16 resource;
> > +   u8 group;
> > +   u8 enable;
> > +} __packed;
> > +
> > +static struct imx_sc_ipc *imx_sc_irq_ipc_handle; static struct
> > +work_struct imx_sc_irq_work; static
> > +ATOMIC_NOTIFIER_HEAD(imx_scu_irq_notifier_chain);
> > +
> > +int imx_scu_irq_register_notifier(struct notifier_block *nb) {
> > +   return atomic_notifier_chain_register(
> > +   _scu_irq_notifier_chain, nb);
> > +}
> > +EXPORT_SYMBOL(imx_scu_irq_register_notifier);
> > +
> > +int imx_scu_irq_unregister_notifier(struct notifier_block *nb) {
> > +   return atomic_notifier_chain_unregister(
> > +   _scu_irq_notifier_chain, nb);
> > +}
> > +EXPORT_SYMBOL(imx_scu_irq_unregister_notifier);
> > +
> > +static int imx_scu_irq_notifier_call_chain(unsigned long status, u8
> > +

RE: [PATCH V6 2/4] firmware: imx: enable imx scu general irq function

2019-04-08 Thread Aisheng Dong
> From: Anson Huang
> Sent: Tuesday, April 9, 2019 10:43 AM
> Subject: [PATCH V6 2/4] firmware: imx: enable imx scu general irq function
> 
> The System Controller Firmware (SCFW) controls RTC, thermal and WDOG etc.,
> these resources' interrupt function are managed by SCU. When any IRQ
> pending, SCU will notify Linux via MU general interrupt channel #3, and Linux
> kernel needs to call SCU APIs to get IRQ status and notify each module to
> handle the interrupt.
> 
> Since there is no data transmission for SCU IRQ notification, so doorbell mode
> is used for this MU channel, and SCU driver will use notifier mechanism to
> broadcast to every module which registers the SCU block notifier.
> 
> Signed-off-by: Anson Huang 
> ---
> Changes since V5:
>   - use ATOMIC_NOTIFIER instead of BLOCKING_NOTIFIER for irq
> notification;
>   - add memory free for failed case to avoid memory leak;
>   - add new API imx_scu_irq_enable() for modules to enable/disable their
> own irqs.
> ---
>  drivers/firmware/imx/Makefile  |   2 +-
>  drivers/firmware/imx/imx-scu-irq.c | 166
> +
>  drivers/firmware/imx/imx-scu.c |   6 ++
>  include/linux/firmware/imx/sci.h   |   5 ++
>  4 files changed, 178 insertions(+), 1 deletion(-)  create mode 100644
> drivers/firmware/imx/imx-scu-irq.c
> 
> diff --git a/drivers/firmware/imx/Makefile b/drivers/firmware/imx/Makefile
> index 1b2e15b..802c4ad 100644
> --- a/drivers/firmware/imx/Makefile
> +++ b/drivers/firmware/imx/Makefile
> @@ -1,3 +1,3 @@
>  # SPDX-License-Identifier: GPL-2.0
> -obj-$(CONFIG_IMX_SCU)+= imx-scu.o misc.o
> +obj-$(CONFIG_IMX_SCU)+= imx-scu.o misc.o imx-scu-irq.o
>  obj-$(CONFIG_IMX_SCU_PD) += scu-pd.o
> diff --git a/drivers/firmware/imx/imx-scu-irq.c
> b/drivers/firmware/imx/imx-scu-irq.c
> new file mode 100644
> index 000..4000c63
> --- /dev/null
> +++ b/drivers/firmware/imx/imx-scu-irq.c
> @@ -0,0 +1,166 @@
> +// SPDX-License-Identifier: GPL-2.0+
> +/*
> + * Copyright 2019 NXP
> + *
> + * Implementation of the SCU IRQ functions using MU.
> + *
> + */
> +
> +#include  #include
> + #include 
> +
> +#define IMX_SC_IRQ_FUNC_ENABLE   1
> +#define IMX_SC_IRQ_FUNC_STATUS   2
> +#define IMX_SC_IRQ_NUM_GROUP 4
> +
> +static u32 mu_resource_id;
> +
> +struct imx_sc_msg_irq_get_status {
> + struct imx_sc_rpc_msg hdr;
> + union {
> + struct {
> + u16 resource;
> + u8 group;
> + u8 reserved;
> + } __packed req;
> + struct {
> + u32 status;
> + } resp;
> + } data;
> +};
> +
> +struct imx_sc_msg_irq_enable {
> + struct imx_sc_rpc_msg hdr;
> + u32 mask;
> + u16 resource;
> + u8 group;
> + u8 enable;
> +} __packed;
> +
> +static struct imx_sc_ipc *imx_sc_irq_ipc_handle; static struct
> +work_struct imx_sc_irq_work; static
> +ATOMIC_NOTIFIER_HEAD(imx_scu_irq_notifier_chain);
> +
> +int imx_scu_irq_register_notifier(struct notifier_block *nb) {
> + return atomic_notifier_chain_register(
> + _scu_irq_notifier_chain, nb);
> +}
> +EXPORT_SYMBOL(imx_scu_irq_register_notifier);
> +
> +int imx_scu_irq_unregister_notifier(struct notifier_block *nb) {
> + return atomic_notifier_chain_unregister(
> + _scu_irq_notifier_chain, nb);
> +}
> +EXPORT_SYMBOL(imx_scu_irq_unregister_notifier);
> +
> +static int imx_scu_irq_notifier_call_chain(unsigned long status, u8
> +*group) {
> + return atomic_notifier_call_chain(_scu_irq_notifier_chain,
> + status, (void *)group);
> +}
> +
> +static void imx_scu_irq_work_handler(struct work_struct *work) {
> + struct imx_sc_msg_irq_get_status msg;
> + struct imx_sc_rpc_msg *hdr = 
> + u32 irq_status;
> + int ret;
> + u8 i;
> +
> + for (i = 0; i < IMX_SC_IRQ_NUM_GROUP; i++) {
> + hdr->ver = IMX_SC_RPC_VERSION;
> + hdr->svc = IMX_SC_RPC_SVC_IRQ;
> + hdr->func = IMX_SC_IRQ_FUNC_STATUS;
> + hdr->size = 2;
> +
> + msg.data.req.resource = mu_resource_id;
> + msg.data.req.group = i;
> +
> + ret = imx_scu_call_rpc(imx_sc_irq_ipc_handle, , true);
> + if (ret) {
> + pr_err("get irq group %d status failed, ret %d\n",
> +i, ret);
> + return;
> + }
> +
> + irq_status = msg.data.resp.status;
> + if (!irq_status)
> +

[PATCH V6 2/4] firmware: imx: enable imx scu general irq function

2019-04-08 Thread Anson Huang
The System Controller Firmware (SCFW) controls RTC, thermal
and WDOG etc., these resources' interrupt function are managed
by SCU. When any IRQ pending, SCU will notify Linux via MU general
interrupt channel #3, and Linux kernel needs to call SCU APIs
to get IRQ status and notify each module to handle the interrupt.

Since there is no data transmission for SCU IRQ notification, so
doorbell mode is used for this MU channel, and SCU driver will
use notifier mechanism to broadcast to every module which registers
the SCU block notifier.

Signed-off-by: Anson Huang 
---
Changes since V5:
- use ATOMIC_NOTIFIER instead of BLOCKING_NOTIFIER for irq notification;
- add memory free for failed case to avoid memory leak;
- add new API imx_scu_irq_enable() for modules to enable/disable their 
own irqs.
---
 drivers/firmware/imx/Makefile  |   2 +-
 drivers/firmware/imx/imx-scu-irq.c | 166 +
 drivers/firmware/imx/imx-scu.c |   6 ++
 include/linux/firmware/imx/sci.h   |   5 ++
 4 files changed, 178 insertions(+), 1 deletion(-)
 create mode 100644 drivers/firmware/imx/imx-scu-irq.c

diff --git a/drivers/firmware/imx/Makefile b/drivers/firmware/imx/Makefile
index 1b2e15b..802c4ad 100644
--- a/drivers/firmware/imx/Makefile
+++ b/drivers/firmware/imx/Makefile
@@ -1,3 +1,3 @@
 # SPDX-License-Identifier: GPL-2.0
-obj-$(CONFIG_IMX_SCU)  += imx-scu.o misc.o
+obj-$(CONFIG_IMX_SCU)  += imx-scu.o misc.o imx-scu-irq.o
 obj-$(CONFIG_IMX_SCU_PD)   += scu-pd.o
diff --git a/drivers/firmware/imx/imx-scu-irq.c 
b/drivers/firmware/imx/imx-scu-irq.c
new file mode 100644
index 000..4000c63
--- /dev/null
+++ b/drivers/firmware/imx/imx-scu-irq.c
@@ -0,0 +1,166 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Copyright 2019 NXP
+ *
+ * Implementation of the SCU IRQ functions using MU.
+ *
+ */
+
+#include 
+#include 
+#include 
+
+#define IMX_SC_IRQ_FUNC_ENABLE 1
+#define IMX_SC_IRQ_FUNC_STATUS 2
+#define IMX_SC_IRQ_NUM_GROUP   4
+
+static u32 mu_resource_id;
+
+struct imx_sc_msg_irq_get_status {
+   struct imx_sc_rpc_msg hdr;
+   union {
+   struct {
+   u16 resource;
+   u8 group;
+   u8 reserved;
+   } __packed req;
+   struct {
+   u32 status;
+   } resp;
+   } data;
+};
+
+struct imx_sc_msg_irq_enable {
+   struct imx_sc_rpc_msg hdr;
+   u32 mask;
+   u16 resource;
+   u8 group;
+   u8 enable;
+} __packed;
+
+static struct imx_sc_ipc *imx_sc_irq_ipc_handle;
+static struct work_struct imx_sc_irq_work;
+static ATOMIC_NOTIFIER_HEAD(imx_scu_irq_notifier_chain);
+
+int imx_scu_irq_register_notifier(struct notifier_block *nb)
+{
+   return atomic_notifier_chain_register(
+   _scu_irq_notifier_chain, nb);
+}
+EXPORT_SYMBOL(imx_scu_irq_register_notifier);
+
+int imx_scu_irq_unregister_notifier(struct notifier_block *nb)
+{
+   return atomic_notifier_chain_unregister(
+   _scu_irq_notifier_chain, nb);
+}
+EXPORT_SYMBOL(imx_scu_irq_unregister_notifier);
+
+static int imx_scu_irq_notifier_call_chain(unsigned long status, u8 *group)
+{
+   return atomic_notifier_call_chain(_scu_irq_notifier_chain,
+   status, (void *)group);
+}
+
+static void imx_scu_irq_work_handler(struct work_struct *work)
+{
+   struct imx_sc_msg_irq_get_status msg;
+   struct imx_sc_rpc_msg *hdr = 
+   u32 irq_status;
+   int ret;
+   u8 i;
+
+   for (i = 0; i < IMX_SC_IRQ_NUM_GROUP; i++) {
+   hdr->ver = IMX_SC_RPC_VERSION;
+   hdr->svc = IMX_SC_RPC_SVC_IRQ;
+   hdr->func = IMX_SC_IRQ_FUNC_STATUS;
+   hdr->size = 2;
+
+   msg.data.req.resource = mu_resource_id;
+   msg.data.req.group = i;
+
+   ret = imx_scu_call_rpc(imx_sc_irq_ipc_handle, , true);
+   if (ret) {
+   pr_err("get irq group %d status failed, ret %d\n",
+  i, ret);
+   return;
+   }
+
+   irq_status = msg.data.resp.status;
+   if (!irq_status)
+   continue;
+
+   imx_scu_irq_notifier_call_chain(irq_status, );
+   }
+}
+
+void imx_scu_irq_enable(u8 group, u32 mask, u8 enable)
+{
+   struct imx_sc_msg_irq_enable msg;
+   struct imx_sc_rpc_msg *hdr = 
+   int ret;
+
+   hdr->ver = IMX_SC_RPC_VERSION;
+   hdr->svc = IMX_SC_RPC_SVC_IRQ;
+   hdr->func = IMX_SC_IRQ_FUNC_ENABLE;
+   hdr->size = 3;
+
+   msg.resource = mu_resource_id;
+   msg.group = group;
+   msg.mask = mask;
+   msg.enable = enable;
+
+   ret = imx_scu_call_rpc(imx_sc_irq_ipc_handle, , true);
+   if (ret)
+   pr_err("enable irq failed, group %d, mask %d, ret %d\n",
+   group, mask, ret);
+}