[PATCH v2 2/3] mailbox: add support for bottom half received data
From: Karl . Li
Date: Sun Dec 29 2024 - 22:46:52 EST
From: Karl Li <karl.li@xxxxxxxxxxxx>
Within the MediaTek APU subsystem, a message passing mechanism
is constructed on top of the mailbox system.
The mailbox only has limited space for each message. The MTK APU firmware
expects the message header from the mailbox, while the message body
is passed through some fixed shared memory.
The mailbox interrupt also serves as a mutex for the shared memory.
Thus the interrupt may only be cleared after the message is handled.
To be specific, the MTK APU firmware fills the message body into a
fixed shared memory, while passing the message header via a mailbox.
--------------- --------------- ---------
| Kernel Buffer | <- | Shared Memory | <- | MTK APU |
--------------- --------------- ---------
The top-half handler first copies the data from the shared memory to the
kernel buffer. Subsequently, the bottom-half handler uses the data
in the kernel buffer for specific processing tasks.
However, it is possible for the MTK APU to send a new message while
the kernel is still processing the previous one. In such cases,
the top-half handler may overwrite the kernel buffer while the data is
still needed, leading to a race condition.
Additionally, due to performance considerations, we cannot allocate a new
kernel buffer for each message.
To prevent this, we need a mechanism to protect the kernel buffer from
such race conditions. Since we cannot use locks to protect the kernel
buffer in an atomic context, we use the mailbox interrupt as a mutex for
the shared memory to further protect our kernel buffer. Therefore,
we clear the interrupt only after the message is handled.
Since the bottom half callback might need to go to sleep, we require a
sleepable callback for mailbox clients.
This patch adds a new sleepable RX callback for mailbox clients for cases
where handling the incoming message requires sleeping.
Signed-off-by: Karl Li <karl.li@xxxxxxxxxxxx>
---
drivers/mailbox/mailbox.c | 16 ++++++++++++++++
include/linux/mailbox_client.h | 2 ++
include/linux/mailbox_controller.h | 1 +
3 files changed, 19 insertions(+)
diff --git a/drivers/mailbox/mailbox.c b/drivers/mailbox/mailbox.c
index d3d26a2c9895..d58a77fcf804 100644
--- a/drivers/mailbox/mailbox.c
+++ b/drivers/mailbox/mailbox.c
@@ -164,6 +164,22 @@ void mbox_chan_received_data(struct mbox_chan *chan, void *mssg)
}
EXPORT_SYMBOL_GPL(mbox_chan_received_data);
+/**
+ * mbox_chan_received_data_bh - A way for controller driver to push data
+ * received from remote to the upper layer.
+ * @chan: Pointer to the mailbox channel on which RX happened.
+ * @mssg: Client specific message typecasted as void *
+ *
+ * For the operations which is not atomic can be called from
+ * mbox_chan_received_data_bh().
+ */
+void mbox_chan_received_data_bh(struct mbox_chan *chan, void *mssg)
+{
+ if (chan->cl->rx_callback_bh)
+ chan->cl->rx_callback_bh(chan->cl, mssg);
+}
+EXPORT_SYMBOL_GPL(mbox_chan_received_data_bh);
+
/**
* mbox_chan_txdone - A way for controller driver to notify the
* framework that the last TX has completed.
diff --git a/include/linux/mailbox_client.h b/include/linux/mailbox_client.h
index 734694912ef7..2cc6fa4e1bf9 100644
--- a/include/linux/mailbox_client.h
+++ b/include/linux/mailbox_client.h
@@ -22,6 +22,7 @@ struct mbox_chan;
* if the client receives some ACK packet for transmission.
* Unused if the controller already has TX_Done/RTR IRQ.
* @rx_callback: Atomic callback to provide client the data received
+ * @rx_callback_bh: Non-atomic callback to provide client the data received
* @tx_prepare: Atomic callback to ask client to prepare the payload
* before initiating the transmission if required.
* @tx_done: Atomic callback to tell client of data transmission
@@ -33,6 +34,7 @@ struct mbox_client {
bool knows_txdone;
void (*rx_callback)(struct mbox_client *cl, void *mssg);
+ void (*rx_callback_bh)(struct mbox_client *cl, void *mssg);
void (*tx_prepare)(struct mbox_client *cl, void *mssg);
void (*tx_done)(struct mbox_client *cl, void *mssg, int r);
};
diff --git a/include/linux/mailbox_controller.h b/include/linux/mailbox_controller.h
index 6fee33cb52f5..74c6a31cd313 100644
--- a/include/linux/mailbox_controller.h
+++ b/include/linux/mailbox_controller.h
@@ -130,6 +130,7 @@ struct mbox_chan {
int mbox_controller_register(struct mbox_controller *mbox); /* can sleep */
void mbox_controller_unregister(struct mbox_controller *mbox); /* can sleep */
void mbox_chan_received_data(struct mbox_chan *chan, void *data); /* atomic */
+void mbox_chan_received_data_bh(struct mbox_chan *chan, void *data); /* can sleep */
void mbox_chan_txdone(struct mbox_chan *chan, int r); /* atomic */
int devm_mbox_controller_register(struct device *dev,
--
2.18.0