[PATCH v6 2/7] mailbox: arm_mhu: add driver for ARM MHU controller

Sudeep Holla sudeep.holla at arm.com
Thu Feb 19 09:17:25 PST 2015



On 19/02/15 16:10, Jassi Brar wrote:
> On 18 February 2015 at 16:07, Sudeep Holla <sudeep.holla at arm.com> wrote:
>
>>> +++ b/Documentation/devicetree/bindings/mailbox/arm-mhu.txt
>>> @@ -0,0 +1,35 @@
>>> +ARM MHU Mailbox Driver
>>> +======================
>>> +
>>> +The ARM's Message-Handling-Unit (MHU) is a mailbox controller that has
>>> +3 independent channels/links to communicate with remote processor(s).
>>> + MHU links are hardwired on a platform. A link raises interrupt for any
>>> +received data. However, there is no specified way of knowing if the sent
>>
>>
>> IIUC the IP as such doesn't have this restriction, it's just the way
>> currently integrated in the SoCs. So we need to provide a way for future
>> expansion.
>>
>>> +data has been read by the remote. This driver assumes the sender polls
>>> +STAT register and the remote clears it after having read the data.
>>
>> I would rather drop any specifics about what driver does. Bindings should
>> try to confine to the underlying hardware only if possible.
>>
> The behavior of local controller and remote firmware clubbed together
> is what mailbox drivers have to deal with.
>
>   For example, if the remote f/w doesn't clear the STAT register after
> reading it, this driver won't have a way to know if the last tx has
> been done or not, i.e, mhu_last_tx_done() won't work. And that
> behavior is not decided by Linux/driver, so is in a way a h/w feature.

OK then mention it as the firmware. Don't tell anything about what the
driver assumes, that's Linux/driver specific.

> Does your remote f/w not clear the STAT?
>

It does.

>>> +Mailbox Device Node:
>>> +====================
>>> +
>>> +Required properties:
>>> +--------------------
>>> +- compatible:          Shall be "arm,mhu" & "arm,primecell"
>>> +- reg:                 Contains the mailbox register address range (base
>>> +                       address and length)
>>> +- #mbox-cells          Shall be 1
>>
>> Need to explain what that one cell must represent.
>>
>>> +- interrupts:          Contains the interrupt information corresponding
>>> to
>>> +                       each of the 3 links of MHU.
>>
>> How do we handle if the middle link has no interrupt ?
>>
> The MHU chapter in my SoC manual suggests the specification requires
> an RX-IRQ per channel. Does your platform omit the irq?
>

What if there's a platform with broken irq, it just can skip that in DT
if we use named interrupts in the binding.

>> Also please that the last channel is secure only and must not be used in
>> non-secure execution.
>>
> I am aware and it is perfectly possible to not touch the secure
> channel from NS mode.
>

Sorry my statement was not clear, I wanted to ask you to add that info
to the binding.

[...]

>>> +
>>> +static bool mhu_last_tx_done(struct mbox_chan *chan)
>>> +{
>>> +       struct mhu_link *mlink = chan->con_priv;
>>> +       u32 val = readl_relaxed(mlink->tx_reg + INTR_STAT_OFS);
>>> +
>>> +       return (val == 0);
>>
>>
>> [Nit] How about just
>>          return readl_relaxed(mlink->tx_reg + INTR_STAT_OFS) == 0;
>>
> I think (val == 0) is less cluttered.
>

OK

>>> +static int mhu_send_data(struct mbox_chan *chan, void *data)
>>> +{
>>> +       struct mhu_link *mlink = chan->con_priv;
>>> +       u32 *arg = data;
>>> +
>>> +       if (!mhu_last_tx_done(chan)) {
>>> +               dev_err(chan->mbox->dev, "Last TX(%d) pending!\n",
>>> mlink->irq);
>>> +               return -EBUSY;
>>> +       }
>>
>> Why do you need the above check when the core handles that already ?
>>
> Our bootloader also uses mhu so we could have failed/pending bits in
> STAT when Linux first tries to use it. However since then the driver
> clears the STAT register upon startup, so this could probably be
> dropped. Will do.
>

Exactly, it can and you have already handled it in the probe, and it
must be that way. Else you are confusing the state machine in the
mailbox core and working it around in your driver, which is wrong.

>>> +static int mhu_startup(struct mbox_chan *chan)
>>> +{
>>> +       struct mhu_link *mlink = chan->con_priv;
>>> +       u32 val;
>>> +       int ret;
>>> +
>>> +       val = readl_relaxed(mlink->tx_reg + INTR_STAT_OFS);
>>> +       writel_relaxed(val, mlink->tx_reg + INTR_CLR_OFS);
>>> +
>>> +       ret = request_irq(mlink->irq, mhu_rx_interrupt, 0, "mhu_link",
>>> chan);
>>> +       if (ret) {
>>> +               dev_err(chan->mbox->dev,
>>> +                       "Unable to aquire IRQ %d\n", mlink->irq);
>>> +               return ret;
>>> +       }
>>
>> This proved costly(in terms of time) on Juno board in my testing,
>> requesting and setting up irq for every small packets you need to send.
>> I would prefer it to be moved to probe.
>>
> If your usage of mailbox is so frequent that request-release of irq
> proves expensive, maybe you could fix your client to hold onto the
> channel for the lifetime. I don't see why your scpi_protocol.c
> shouldn't do that, even if request_irq was in probe.
>

Agreed, and that's how it's done. But I was thinking of cases where
there are multiple clients transferring small data.

[...]

>
>>> +
>>> +       mhu->mbox.dev = dev;
>>> +       mhu->mbox.chans = &mhu->chan[0];
>>> +       mhu->mbox.num_chans = 3;
>>> +       mhu->mbox.ops = &mhu_ops;
>>> +       mhu->mbox.txdone_irq = false;
>>> +       mhu->mbox.txdone_poll = true;
>>> +       mhu->mbox.txpoll_period = 10;
>>
>> 10ms seems to high, but if that's a derived value then I am fine.
>> On Juno, typically we get response within a millisecond, so we need not
>> get blocked on Tx even after getting Rx for 10ms. I prefer it to be set
>> to 1 ms.
>>
> Similar on my platform.  However 1 isn't much meaning in milliseconds
> because mod_timer works in jiffiy which is usually atleast 5ms. If we
> are polling we can't anyway expect latency critical stuff, so 10ms
> seems like a safe bet.
>

Here you are assuming the fact about HZ value in the driver. Leave that
to timer subsystem for the conversion(msecs_to_jiffies takes care of
rounding) and put values based on what hardware expects IMO.

>>> +
>>> +static struct amba_id mhu_ids[] = {
>>> +       {
>>> +               .id     = 0x1bb098,
>>
>>
>> This is the problem. This IP has PID(0x98 0xB0 0x1B 0x00 0x04)
>> it's 5 bytes[1] . Even I had thought of AMBA initially, it doesn't fit
>> as is, may need changes to the amba core to consider this.
>>
> How is that a problem? AMBA chooses to compare 32bits of PID for the
> class. The PID4 might change across versions of MHU, which we could
> figure out in the mhu driver in future.
>

No that's wrong IMO. You mask is incorrect as it doesn't include PID4
which in this case contains
[3:0] DES_2 JEP106 continuation code that identifies the designer.
Set to 0x4 for ARM.

IMO it's not a proper comparison/match as you are not including part of
ID that identifies the designer.

Regards,
Sudeep




More information about the linux-arm-kernel mailing list