Re: [PATCH] usb: musb: musb_host: Introduce postponed URB giveback

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



2017-04-27 18:35 GMT+03:00 Bin Liu <b-liu@xxxxxx>:
> Hi Matwey,
>
> On Thu, Apr 27, 2017 at 01:20:33PM +0300, Matwey V. Kornilov wrote:
>> This commit changes the order of actions undertaken in
>> musb_advance_schedule() in order to overcome issue with broken
>> isochronous transfer [1].
>>
>> There is no harm to split musb_giveback into two pieces.  The first
>> unlinks finished urb, the second givebacks it. The issue here that
>> givebacking may be quite time-consuming due to urb->complete() call.
>> As it happens in case of pwc-driven web cameras. It may take about 0.5
>> ms to call __musb_giveback() that calls urb->callback() internally.
>> Under specific circumstances setting MUSB_RXCSR_H_REQPKT in subsequent
>> musb_start_urb() for the next urb will be too late to produce physical
>> IN packet. Since auto req is not used by this module the exchange
>> would be as the following:
>>
>> [        ]   7.220456 d=  0.000997 [182   +  3.667] [  3] IN   : 4.5
>> [    T   ]   7.220459 d=  0.000003 [182   +  7.000] [800] DATA0: [skipped]
>> [        ]   7.222456 d=  0.001997 [184   +  3.667] [  3] IN   : 4.5
>> [        ]   7.222459 d=  0.000003 [184   +  7.000] [  3] DATA0: 00 00
>>
>> It is known that missed IN in isochronous mode makes some
>> perepherial broken. For instance, pwc-driven or uvc-driven
>> web cameras.
>> In order to workaround this issue we postpone calling
>> urb->callback() after setting MUSB_RXCSR_H_REQPKT for the
>> next urb if there is the next urb pending in queue.
>>
>> [1] https://www.spinics.net/lists/linux-usb/msg145747.html
>>
>> Fixes: f551e1352983 ("Revert "usb: musb: musb_host: Enable HCD_BH flag to handle urb return in bottom half"")
>> Signed-off-by: Matwey V. Kornilov <matwey@xxxxxxxxxx>
>
> Thanks for the effort of working on this long standing issue, I know you
> have spent alot of time on it, but what I am thinking is instead of
> workaround the problem we need to understand the root cause - why
> uvc-video takes longer to exec the urb callback, why only am335x
> reported this issue. This is on my backlog, just seems never got time
> for it...

Have you tried other SoCs with Invetra MUSB?

>
> Ideally MUSB driver should be just using HCD_BH flag and let the core to
> handle the urb callback, regardless the usb transfer types.

I think the only reason why everything worked before with HCD_BH is
that execution of urb->callback() was placed after musb_start(). The
order of operations matters.
However, you said that something was also wrong with HCD_BH and other
peripherals.

>
> The MUSB drivers are already messy and complicated enough for
> maintenance, I'd like to understand the root cause of the delay first
> before decide how to solve the issue.
>

I feel from playing with OpenVizsla that REQPKT should be set well in
advance. So, time window to set the flag is actually smaller than 1
ms.
urb->callback() is never takes longer than 0.4 ms for pwc driver, but
INs are skipped.
At the same time musb_host doesn't utilize AutoReq here. I think many
other USB host controllers (OHCI?) just rely on hardware to send IN
packets in time.

> Regards,
> -Bin.
>
>> ---
>>  drivers/usb/musb/musb_host.c | 54 +++++++++++++++++++++++++++++++++++++-------
>>  1 file changed, 46 insertions(+), 8 deletions(-)
>>
>> diff --git a/drivers/usb/musb/musb_host.c b/drivers/usb/musb/musb_host.c
>> index ac3a4952abb4..b590c2555dab 100644
>> --- a/drivers/usb/musb/musb_host.c
>> +++ b/drivers/usb/musb/musb_host.c
>> @@ -299,19 +299,24 @@ musb_start_urb(struct musb *musb, int is_in, struct musb_qh *qh)
>>       }
>>  }
>>
>> -/* Context: caller owns controller lock, IRQs are blocked */
>> -static void musb_giveback(struct musb *musb, struct urb *urb, int status)
>> +static void __musb_giveback(struct musb *musb, struct urb *urb, int status)
>>  __releases(musb->lock)
>>  __acquires(musb->lock)
>>  {
>> -     trace_musb_urb_gb(musb, urb);
>> -
>> -     usb_hcd_unlink_urb_from_ep(musb->hcd, urb);
>>       spin_unlock(&musb->lock);
>>       usb_hcd_giveback_urb(musb->hcd, urb, status);
>>       spin_lock(&musb->lock);
>>  }
>>
>> +/* Context: caller owns controller lock, IRQs are blocked */
>> +static void musb_giveback(struct musb *musb, struct urb *urb, int status)
>> +{
>> +     trace_musb_urb_gb(musb, urb);
>> +
>> +     usb_hcd_unlink_urb_from_ep(musb->hcd, urb);
>> +     __musb_giveback(musb, urb, status);
>> +}
>> +
>>  /* For bulk/interrupt endpoints only */
>>  static inline void musb_save_toggle(struct musb_qh *qh, int is_in,
>>                                   struct urb *urb)
>> @@ -346,6 +351,7 @@ static void musb_advance_schedule(struct musb *musb, struct urb *urb,
>>       struct musb_hw_ep       *ep = qh->hw_ep;
>>       int                     ready = qh->is_ready;
>>       int                     status;
>> +     int                     postponed_giveback = 0;
>>
>>       status = (urb->status == -EINPROGRESS) ? 0 : urb->status;
>>
>> @@ -361,9 +367,35 @@ static void musb_advance_schedule(struct musb *musb, struct urb *urb,
>>               break;
>>       }
>>
>> -     qh->is_ready = 0;
>> -     musb_giveback(musb, urb, status);
>> -     qh->is_ready = ready;
>> +     usb_hcd_unlink_urb_from_ep(musb->hcd, urb);
>> +
>> +     /* It may take about 0.5 ms to call __musb_giveback() that
>> +      * calls urb->callback() internally. Under specific circumstances
>> +      * setting MUSB_RXCSR_H_REQPKT in subsequent musb_start_urb() for the
>> +      * next urb will be too late to produce physical IN packet. Since
>> +      * auto req is not used by this module the exchange would be as the
>> +      * following:
>> +      *
>> +      * [        ]   7.220456 d=  0.000997 [182   +  3.667] [  3] IN   : 4.5
>> +      * [    T   ]   7.220459 d=  0.000003 [182   +  7.000] [800] DATA0: [skipped]
>> +      * [        ]   7.222456 d=  0.001997 [184   +  3.667] [  3] IN   : 4.5
>> +      * [        ]   7.222459 d=  0.000003 [184   +  7.000] [  3] DATA0: 00 00
>> +      *
>> +      * It is known that missed IN in isochronous mode makes some
>> +      * perepherial broken. For instance, pwc-driven or uvc-driven
>> +      * web cameras.
>> +      * In order to workaround this issue we postpone calling
>> +      * urb->callback() after setting MUSB_RXCSR_H_REQPKT for the
>> +      * next urb if there is the next urb pending in queue.
>> +      */
>> +     if (is_in && qh->type == USB_ENDPOINT_XFER_ISOC
>> +         && !list_empty(&qh->hep->urb_list)) {
>> +             postponed_giveback = 1;
>> +     } else {
>> +             qh->is_ready = 0;
>> +             __musb_giveback(musb, urb, status);
>> +             qh->is_ready = ready;
>> +     }
>>
>>       /* reclaim resources (and bandwidth) ASAP; deschedule it, and
>>        * invalidate qh as soon as list_empty(&hep->urb_list)
>> @@ -428,6 +460,12 @@ static void musb_advance_schedule(struct musb *musb, struct urb *urb,
>>                   hw_ep->epnum, is_in ? 'R' : 'T', next_urb(qh));
>>               musb_start_urb(musb, is_in, qh);
>>       }
>> +
>> +     if (postponed_giveback) {
>> +             qh->is_ready = 0;
>> +             __musb_giveback(musb, urb, status);
>> +             qh->is_ready = ready;
>> +     }
>>  }
>>
>>  static u16 musb_h_flush_rxfifo(struct musb_hw_ep *hw_ep, u16 csr)
>> --
>> 2.12.0
>>
>



-- 
With best regards,
Matwey V. Kornilov.
Sternberg Astronomical Institute, Lomonosov Moscow State University, Russia
119234, Moscow, Universitetsky pr-k 13, +7 (495) 9392382
--
To unsubscribe from this list: send the line "unsubscribe linux-usb" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [Linux Media]     [Linux Input]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [Old Linux USB Devel Archive]

  Powered by Linux