diff mbox

[U-Boot] dfu: cmd: trigger watchdog before calling usb_gadget_handle_interrupts

Message ID 1426170850-25688-1-git-send-email-hs@denx.de
State Accepted
Delegated to: Łukasz Majewski
Headers show

Commit Message

Heiko Schocher March 12, 2015, 2:34 p.m. UTC
trigger watchdog before calling usb_gadget_handle_interrupts()
This prevents board resets when calling dfu command on boards
which have a watchdog.

Signed-off-by: Heiko Schocher <hs@denx.de>
---

 common/cmd_dfu.c | 2 ++
 1 file changed, 2 insertions(+)

Comments

Łukasz Majewski March 17, 2015, 9:52 a.m. UTC | #1
Hi Heiko,

> trigger watchdog before calling usb_gadget_handle_interrupts()
> This prevents board resets when calling dfu command on boards
> which have a watchdog.
> 
> Signed-off-by: Heiko Schocher <hs@denx.de>
> ---
> 
>  common/cmd_dfu.c | 2 ++
>  1 file changed, 2 insertions(+)
> 
> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
> index e975abe..46af4cf 100644
> --- a/common/cmd_dfu.c
> +++ b/common/cmd_dfu.c
> @@ -9,6 +9,7 @@
>   */
>  
>  #include <common.h>
> +#include <watchdog.h>
>  #include <dfu.h>
>  #include <g_dnl.h>
>  #include <usb.h>
> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int flag, int
> argc, char * const argv[]) if (ctrlc())
>  			goto exit;
>  
> +		WATCHDOG_RESET();
>  		usb_gadget_handle_interrupts();
>  	}
>  exit:

It seems strange for me, that we must reset watchdog when looping in
the dfu.

What is the WATCHDOG interval on the affected board?
Heiko Schocher March 17, 2015, 10:48 a.m. UTC | #2
Hello Lukasz,

Am 17.03.2015 10:52, schrieb Lukasz Majewski:
> Hi Heiko,
>
>> trigger watchdog before calling usb_gadget_handle_interrupts()
>> This prevents board resets when calling dfu command on boards
>> which have a watchdog.
>>
>> Signed-off-by: Heiko Schocher <hs@denx.de>
>> ---
>>
>>   common/cmd_dfu.c | 2 ++
>>   1 file changed, 2 insertions(+)
>>
>> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
>> index e975abe..46af4cf 100644
>> --- a/common/cmd_dfu.c
>> +++ b/common/cmd_dfu.c
>> @@ -9,6 +9,7 @@
>>    */
>>
>>   #include <common.h>
>> +#include <watchdog.h>
>>   #include <dfu.h>
>>   #include <g_dnl.h>
>>   #include <usb.h>
>> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int flag, int
>> argc, char * const argv[]) if (ctrlc())
>>   			goto exit;
>>
>> +		WATCHDOG_RESET();
>>   		usb_gadget_handle_interrupts();
>>   	}
>>   exit:
>
> It seems strange for me, that we must reset watchdog when looping in
> the dfu.

Hmm.. maybe I overlook something, but If you look into this while(1)
loop, there is no trigger of the watchdog ... and if I start the dfu
command without a USB cable on the board, what triggers the boards
watchdog?

> What is the WATCHDOG interval on the affected board?

~5 seconds

Ah, this is on an at91 board .. and in the
drivers/serial/atmel_usart.c atmel_serial_tstc() is no WATCHDOG_RESET...

So ctrlc() does not trigger the watchdog

bye,
Heiko
Łukasz Majewski March 17, 2015, 12:56 p.m. UTC | #3
Hi Heiko,

> Hello Lukasz,
> 
> Am 17.03.2015 10:52, schrieb Lukasz Majewski:
> > Hi Heiko,
> >
> >> trigger watchdog before calling usb_gadget_handle_interrupts()
> >> This prevents board resets when calling dfu command on boards
> >> which have a watchdog.
> >>
> >> Signed-off-by: Heiko Schocher <hs@denx.de>
> >> ---
> >>
> >>   common/cmd_dfu.c | 2 ++
> >>   1 file changed, 2 insertions(+)
> >>
> >> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
> >> index e975abe..46af4cf 100644
> >> --- a/common/cmd_dfu.c
> >> +++ b/common/cmd_dfu.c
> >> @@ -9,6 +9,7 @@
> >>    */
> >>
> >>   #include <common.h>
> >> +#include <watchdog.h>
> >>   #include <dfu.h>
> >>   #include <g_dnl.h>
> >>   #include <usb.h>
> >> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int flag, int
> >> argc, char * const argv[]) if (ctrlc())
> >>   			goto exit;
> >>
> >> +		WATCHDOG_RESET();
> >>   		usb_gadget_handle_interrupts();
> >>   	}
> >>   exit:
> >
> > It seems strange for me, that we must reset watchdog when looping in
> > the dfu.
> 
> Hmm.. maybe I overlook something, but If you look into this while(1)
> loop, there is no trigger of the watchdog ... and if I start the dfu
> command without a USB cable on the board, what triggers the boards
> watchdog?

So the problem is when cable is not attached to the board and you call
dfu command to execute?

> 
> > What is the WATCHDOG interval on the affected board?
> 
> ~5 seconds
> 
> Ah, this is on an at91 board .. and in the
> drivers/serial/atmel_usart.c atmel_serial_tstc() is no
> WATCHDOG_RESET...
> 
> So ctrlc() does not trigger the watchdog

Could you be a bit more specific here. Does your board expect ctrlc()
to update watchdog, so it won't reset?

> 
> bye,
> Heiko
Heiko Schocher March 17, 2015, 1:27 p.m. UTC | #4
Hello Lukasz,

Am 17.03.2015 13:56, schrieb Lukasz Majewski:
> Hi Heiko,
>
>> Hello Lukasz,
>>
>> Am 17.03.2015 10:52, schrieb Lukasz Majewski:
>>> Hi Heiko,
>>>
>>>> trigger watchdog before calling usb_gadget_handle_interrupts()
>>>> This prevents board resets when calling dfu command on boards
>>>> which have a watchdog.
>>>>
>>>> Signed-off-by: Heiko Schocher <hs@denx.de>
>>>> ---
>>>>
>>>>    common/cmd_dfu.c | 2 ++
>>>>    1 file changed, 2 insertions(+)
>>>>
>>>> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
>>>> index e975abe..46af4cf 100644
>>>> --- a/common/cmd_dfu.c
>>>> +++ b/common/cmd_dfu.c
>>>> @@ -9,6 +9,7 @@
>>>>     */
>>>>
>>>>    #include <common.h>
>>>> +#include <watchdog.h>
>>>>    #include <dfu.h>
>>>>    #include <g_dnl.h>
>>>>    #include <usb.h>
>>>> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int flag, int
>>>> argc, char * const argv[]) if (ctrlc())
>>>>    			goto exit;
>>>>
>>>> +		WATCHDOG_RESET();
>>>>    		usb_gadget_handle_interrupts();
>>>>    	}
>>>>    exit:
>>>
>>> It seems strange for me, that we must reset watchdog when looping in
>>> the dfu.
>>
>> Hmm.. maybe I overlook something, but If you look into this while(1)
>> loop, there is no trigger of the watchdog ... and if I start the dfu
>> command without a USB cable on the board, what triggers the boards
>> watchdog?
>
> So the problem is when cable is not attached to the board and you call
> dfu command to execute?

Yes.

>>> What is the WATCHDOG interval on the affected board?
>>
>> ~5 seconds
>>
>> Ah, this is on an at91 board .. and in the
>> drivers/serial/atmel_usart.c atmel_serial_tstc() is no
>> WATCHDOG_RESET...
>>
>> So ctrlc() does not trigger the watchdog
>
> Could you be a bit more specific here. Does your board expect ctrlc()
> to update watchdog, so it won't reset?

I do not know, if ctrlc() must trigger the WDT ... I just looked into
the while(1) loop in common/cmd_dfu.c. There is only ctrlc() which
could trigger the WDT... and on at91 it does not trigger it ...

If dfu transfer is started there is some output on the console, which
triggers the WDT too ... do you have a board with a running WDT?

bye,
Heiko
Łukasz Majewski March 17, 2015, 3:16 p.m. UTC | #5
Hi Heiko,

> Hello Lukasz,
> 
> Am 17.03.2015 13:56, schrieb Lukasz Majewski:
> > Hi Heiko,
> >
> >> Hello Lukasz,
> >>
> >> Am 17.03.2015 10:52, schrieb Lukasz Majewski:
> >>> Hi Heiko,
> >>>
> >>>> trigger watchdog before calling usb_gadget_handle_interrupts()
> >>>> This prevents board resets when calling dfu command on boards
> >>>> which have a watchdog.
> >>>>
> >>>> Signed-off-by: Heiko Schocher <hs@denx.de>
> >>>> ---
> >>>>
> >>>>    common/cmd_dfu.c | 2 ++
> >>>>    1 file changed, 2 insertions(+)
> >>>>
> >>>> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
> >>>> index e975abe..46af4cf 100644
> >>>> --- a/common/cmd_dfu.c
> >>>> +++ b/common/cmd_dfu.c
> >>>> @@ -9,6 +9,7 @@
> >>>>     */
> >>>>
> >>>>    #include <common.h>
> >>>> +#include <watchdog.h>
> >>>>    #include <dfu.h>
> >>>>    #include <g_dnl.h>
> >>>>    #include <usb.h>
> >>>> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int flag,
> >>>> int argc, char * const argv[]) if (ctrlc())
> >>>>    			goto exit;
> >>>>
> >>>> +		WATCHDOG_RESET();
> >>>>    		usb_gadget_handle_interrupts();
> >>>>    	}
> >>>>    exit:
> >>>
> >>> It seems strange for me, that we must reset watchdog when looping
> >>> in the dfu.
> >>
> >> Hmm.. maybe I overlook something, but If you look into this
> >> while(1) loop, there is no trigger of the watchdog ... and if I
> >> start the dfu command without a USB cable on the board, what
> >> triggers the boards watchdog?
> >
> > So the problem is when cable is not attached to the board and you
> > call dfu command to execute?
> 
> Yes.

For UMS gadget there is defined g_dnl_board_usb_cable_connected()
function.

However, it is not yet supported in dfu and requires working MUIC
block, which might be not available at your board.

> 
> >>> What is the WATCHDOG interval on the affected board?
> >>
> >> ~5 seconds
> >>
> >> Ah, this is on an at91 board .. and in the
> >> drivers/serial/atmel_usart.c atmel_serial_tstc() is no
> >> WATCHDOG_RESET...
> >>
> >> So ctrlc() does not trigger the watchdog
> >
> > Could you be a bit more specific here. Does your board expect
> > ctrlc() to update watchdog, so it won't reset?
> 
> I do not know, if ctrlc() must trigger the WDT ... I just looked into
> the while(1) loop in common/cmd_dfu.c. There is only ctrlc() which
> could trigger the WDT... and on at91 it does not trigger it ...

By trigger you mean reset WDT core and don't reset the board?

> 
> If dfu transfer is started there is some output on the console, which
> triggers the WDT too ... do you have a board with a running WDT?

On trats/trats2 we disable WDT when we enter the u-boot.

I can imagine following situation when WDT is enabled when u-boot
starts (its timeout is ~5sec) and then you start dfu transfer with not
connected cable.
Then 5sec pass since cable is not connected and no transfer is ongoing.
This causes board reset by WDT.

Am I right?

> 
> bye,
> Heiko
Tom Rini March 17, 2015, 4:09 p.m. UTC | #6
On Tue, Mar 17, 2015 at 04:16:14PM +0100, Lukasz Majewski wrote:
> Hi Heiko,
> 
> > Hello Lukasz,
> > 
> > Am 17.03.2015 13:56, schrieb Lukasz Majewski:
> > > Hi Heiko,
> > >
> > >> Hello Lukasz,
> > >>
> > >> Am 17.03.2015 10:52, schrieb Lukasz Majewski:
> > >>> Hi Heiko,
> > >>>
> > >>>> trigger watchdog before calling usb_gadget_handle_interrupts()
> > >>>> This prevents board resets when calling dfu command on boards
> > >>>> which have a watchdog.
> > >>>>
> > >>>> Signed-off-by: Heiko Schocher <hs@denx.de>
> > >>>> ---
> > >>>>
> > >>>>    common/cmd_dfu.c | 2 ++
> > >>>>    1 file changed, 2 insertions(+)
> > >>>>
> > >>>> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
> > >>>> index e975abe..46af4cf 100644
> > >>>> --- a/common/cmd_dfu.c
> > >>>> +++ b/common/cmd_dfu.c
> > >>>> @@ -9,6 +9,7 @@
> > >>>>     */
> > >>>>
> > >>>>    #include <common.h>
> > >>>> +#include <watchdog.h>
> > >>>>    #include <dfu.h>
> > >>>>    #include <g_dnl.h>
> > >>>>    #include <usb.h>
> > >>>> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int flag,
> > >>>> int argc, char * const argv[]) if (ctrlc())
> > >>>>    			goto exit;
> > >>>>
> > >>>> +		WATCHDOG_RESET();
> > >>>>    		usb_gadget_handle_interrupts();
> > >>>>    	}
> > >>>>    exit:
> > >>>
> > >>> It seems strange for me, that we must reset watchdog when looping
> > >>> in the dfu.
> > >>
> > >> Hmm.. maybe I overlook something, but If you look into this
> > >> while(1) loop, there is no trigger of the watchdog ... and if I
> > >> start the dfu command without a USB cable on the board, what
> > >> triggers the boards watchdog?
> > >
> > > So the problem is when cable is not attached to the board and you
> > > call dfu command to execute?
> > 
> > Yes.
> 
> For UMS gadget there is defined g_dnl_board_usb_cable_connected()
> function.
> 
> However, it is not yet supported in dfu and requires working MUIC
> block, which might be not available at your board.
> 
> > 
> > >>> What is the WATCHDOG interval on the affected board?
> > >>
> > >> ~5 seconds
> > >>
> > >> Ah, this is on an at91 board .. and in the
> > >> drivers/serial/atmel_usart.c atmel_serial_tstc() is no
> > >> WATCHDOG_RESET...
> > >>
> > >> So ctrlc() does not trigger the watchdog
> > >
> > > Could you be a bit more specific here. Does your board expect
> > > ctrlc() to update watchdog, so it won't reset?
> > 
> > I do not know, if ctrlc() must trigger the WDT ... I just looked into
> > the while(1) loop in common/cmd_dfu.c. There is only ctrlc() which
> > could trigger the WDT... and on at91 it does not trigger it ...
> 
> By trigger you mean reset WDT core and don't reset the board?
> 
> > 
> > If dfu transfer is started there is some output on the console, which
> > triggers the WDT too ... do you have a board with a running WDT?
> 
> On trats/trats2 we disable WDT when we enter the u-boot.

This seems wrong.  We should enable the WDT and then be "petting" or
whatever the right term is for saying "Hey WDT, I'm alive!" which is
what WATCHDOG_RESET() is really about.
Lukasz Majewski March 17, 2015, 9:36 p.m. UTC | #7
On Tue, 17 Mar 2015 12:09:17 -0400
Tom Rini <trini@konsulko.com> wrote:

> On Tue, Mar 17, 2015 at 04:16:14PM +0100, Lukasz Majewski wrote:
> > Hi Heiko,
> > 
> > > Hello Lukasz,
> > > 
> > > Am 17.03.2015 13:56, schrieb Lukasz Majewski:
> > > > Hi Heiko,
> > > >
> > > >> Hello Lukasz,
> > > >>
> > > >> Am 17.03.2015 10:52, schrieb Lukasz Majewski:
> > > >>> Hi Heiko,
> > > >>>
> > > >>>> trigger watchdog before calling
> > > >>>> usb_gadget_handle_interrupts() This prevents board resets
> > > >>>> when calling dfu command on boards which have a watchdog.
> > > >>>>
> > > >>>> Signed-off-by: Heiko Schocher <hs@denx.de>
> > > >>>> ---
> > > >>>>
> > > >>>>    common/cmd_dfu.c | 2 ++
> > > >>>>    1 file changed, 2 insertions(+)
> > > >>>>
> > > >>>> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
> > > >>>> index e975abe..46af4cf 100644
> > > >>>> --- a/common/cmd_dfu.c
> > > >>>> +++ b/common/cmd_dfu.c
> > > >>>> @@ -9,6 +9,7 @@
> > > >>>>     */
> > > >>>>
> > > >>>>    #include <common.h>
> > > >>>> +#include <watchdog.h>
> > > >>>>    #include <dfu.h>
> > > >>>>    #include <g_dnl.h>
> > > >>>>    #include <usb.h>
> > > >>>> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int
> > > >>>> flag, int argc, char * const argv[]) if (ctrlc())
> > > >>>>    			goto exit;
> > > >>>>
> > > >>>> +		WATCHDOG_RESET();
> > > >>>>    		usb_gadget_handle_interrupts();
> > > >>>>    	}
> > > >>>>    exit:
> > > >>>
> > > >>> It seems strange for me, that we must reset watchdog when
> > > >>> looping in the dfu.
> > > >>
> > > >> Hmm.. maybe I overlook something, but If you look into this
> > > >> while(1) loop, there is no trigger of the watchdog ... and if I
> > > >> start the dfu command without a USB cable on the board, what
> > > >> triggers the boards watchdog?
> > > >
> > > > So the problem is when cable is not attached to the board and
> > > > you call dfu command to execute?
> > > 
> > > Yes.
> > 
> > For UMS gadget there is defined g_dnl_board_usb_cable_connected()
> > function.
> > 
> > However, it is not yet supported in dfu and requires working MUIC
> > block, which might be not available at your board.
> > 
> > > 
> > > >>> What is the WATCHDOG interval on the affected board?
> > > >>
> > > >> ~5 seconds
> > > >>
> > > >> Ah, this is on an at91 board .. and in the
> > > >> drivers/serial/atmel_usart.c atmel_serial_tstc() is no
> > > >> WATCHDOG_RESET...
> > > >>
> > > >> So ctrlc() does not trigger the watchdog
> > > >
> > > > Could you be a bit more specific here. Does your board expect
> > > > ctrlc() to update watchdog, so it won't reset?
> > > 
> > > I do not know, if ctrlc() must trigger the WDT ... I just looked
> > > into the while(1) loop in common/cmd_dfu.c. There is only ctrlc()
> > > which could trigger the WDT... and on at91 it does not trigger
> > > it ...
> > 
> > By trigger you mean reset WDT core and don't reset the board?
> > 
> > > 
> > > If dfu transfer is started there is some output on the console,
> > > which triggers the WDT too ... do you have a board with a running
> > > WDT?
> > 
> > On trats/trats2 we disable WDT when we enter the u-boot.
> 
> This seems wrong.  We should enable the WDT and then be "petting" or
> whatever the right term is for saying "Hey WDT, I'm alive!" which is
> what WATCHDOG_RESET() is really about.

Tom, I agree with the above. However in trats/trats2 case it is not
necessary.

Best regards,
Lukasz
Heiko Schocher March 18, 2015, 6:32 a.m. UTC | #8
Hello Lukasz,

Am 17.03.2015 16:16, schrieb Lukasz Majewski:
> Hi Heiko,
>
>> Hello Lukasz,
>>
>> Am 17.03.2015 13:56, schrieb Lukasz Majewski:
>>> Hi Heiko,
>>>
>>>> Hello Lukasz,
>>>>
>>>> Am 17.03.2015 10:52, schrieb Lukasz Majewski:
>>>>> Hi Heiko,
>>>>>
>>>>>> trigger watchdog before calling usb_gadget_handle_interrupts()
>>>>>> This prevents board resets when calling dfu command on boards
>>>>>> which have a watchdog.
>>>>>>
>>>>>> Signed-off-by: Heiko Schocher <hs@denx.de>
>>>>>> ---
>>>>>>
>>>>>>     common/cmd_dfu.c | 2 ++
>>>>>>     1 file changed, 2 insertions(+)
>>>>>>
>>>>>> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
>>>>>> index e975abe..46af4cf 100644
>>>>>> --- a/common/cmd_dfu.c
>>>>>> +++ b/common/cmd_dfu.c
>>>>>> @@ -9,6 +9,7 @@
>>>>>>      */
>>>>>>
>>>>>>     #include <common.h>
>>>>>> +#include <watchdog.h>
>>>>>>     #include <dfu.h>
>>>>>>     #include <g_dnl.h>
>>>>>>     #include <usb.h>
>>>>>> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int flag,
>>>>>> int argc, char * const argv[]) if (ctrlc())
>>>>>>     			goto exit;
>>>>>>
>>>>>> +		WATCHDOG_RESET();
>>>>>>     		usb_gadget_handle_interrupts();
>>>>>>     	}
>>>>>>     exit:
>>>>>
>>>>> It seems strange for me, that we must reset watchdog when looping
>>>>> in the dfu.
>>>>
>>>> Hmm.. maybe I overlook something, but If you look into this
>>>> while(1) loop, there is no trigger of the watchdog ... and if I
>>>> start the dfu command without a USB cable on the board, what
>>>> triggers the boards watchdog?
>>>
>>> So the problem is when cable is not attached to the board and you
>>> call dfu command to execute?
>>
>> Yes.
>
> For UMS gadget there is defined g_dnl_board_usb_cable_connected()
> function.
>
> However, it is not yet supported in dfu and requires working MUIC
> block, which might be not available at your board.

Maybe, I must check this ...

>>>>> What is the WATCHDOG interval on the affected board?
>>>>
>>>> ~5 seconds
>>>>
>>>> Ah, this is on an at91 board .. and in the
>>>> drivers/serial/atmel_usart.c atmel_serial_tstc() is no
>>>> WATCHDOG_RESET...
>>>>
>>>> So ctrlc() does not trigger the watchdog
>>>
>>> Could you be a bit more specific here. Does your board expect
>>> ctrlc() to update watchdog, so it won't reset?
>>
>> I do not know, if ctrlc() must trigger the WDT ... I just looked into
>> the while(1) loop in common/cmd_dfu.c. There is only ctrlc() which
>> could trigger the WDT... and on at91 it does not trigger it ...
>
> By trigger you mean reset WDT core and don't reset the board?

I mean, the code must trigger the WDT somewhere in this while(1).
If not, the WDT resets the board.

>> If dfu transfer is started there is some output on the console, which
>> triggers the WDT too ... do you have a board with a running WDT?
>
> On trats/trats2 we disable WDT when we enter the u-boot.

Why?

So you can test this behaviour. Do not disable the WDT and start
the dfu command ... the board should reset when you start the dfu
command (without starting dfu-util on the host, and if ctrl() does not
call WATCHDOG_RESET for your hw) ... with my patch it should work ...

BTW: I could not disable the WDT on this board, once it is enabled.
So disabling the WDT is no option ... :-(

> I can imagine following situation when WDT is enabled when u-boot
> starts (its timeout is ~5sec) and then you start dfu transfer with not
> connected cable.
> Then 5sec pass since cable is not connected and no transfer is ongoing.
> This causes board reset by WDT.
>
> Am I right?

Yes, because the WDT gets not triggered in this while(1). And maybe
if you start the dfu command with connected cable, but not starting
dfu-util on the host also, but I have not yet running the usb gadget
on this at91 board (at91sam9g20 based) ...

bye,
Heiko
Łukasz Majewski March 18, 2015, 7:44 a.m. UTC | #9
Hi Heiko,

> Hello Lukasz,
> 
> Am 17.03.2015 16:16, schrieb Lukasz Majewski:
> > Hi Heiko,
> >
> >> Hello Lukasz,
> >>
> >> Am 17.03.2015 13:56, schrieb Lukasz Majewski:
> >>> Hi Heiko,
> >>>
> >>>> Hello Lukasz,
> >>>>
> >>>> Am 17.03.2015 10:52, schrieb Lukasz Majewski:
> >>>>> Hi Heiko,
> >>>>>
> >>>>>> trigger watchdog before calling usb_gadget_handle_interrupts()
> >>>>>> This prevents board resets when calling dfu command on boards
> >>>>>> which have a watchdog.
> >>>>>>
> >>>>>> Signed-off-by: Heiko Schocher <hs@denx.de>
> >>>>>> ---
> >>>>>>
> >>>>>>     common/cmd_dfu.c | 2 ++
> >>>>>>     1 file changed, 2 insertions(+)
> >>>>>>
> >>>>>> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
> >>>>>> index e975abe..46af4cf 100644
> >>>>>> --- a/common/cmd_dfu.c
> >>>>>> +++ b/common/cmd_dfu.c
> >>>>>> @@ -9,6 +9,7 @@
> >>>>>>      */
> >>>>>>
> >>>>>>     #include <common.h>
> >>>>>> +#include <watchdog.h>
> >>>>>>     #include <dfu.h>
> >>>>>>     #include <g_dnl.h>
> >>>>>>     #include <usb.h>
> >>>>>> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int flag,
> >>>>>> int argc, char * const argv[]) if (ctrlc())
> >>>>>>     			goto exit;
> >>>>>>
> >>>>>> +		WATCHDOG_RESET();
> >>>>>>     		usb_gadget_handle_interrupts();
> >>>>>>     	}
> >>>>>>     exit:
> >>>>>
> >>>>> It seems strange for me, that we must reset watchdog when
> >>>>> looping in the dfu.
> >>>>
> >>>> Hmm.. maybe I overlook something, but If you look into this
> >>>> while(1) loop, there is no trigger of the watchdog ... and if I
> >>>> start the dfu command without a USB cable on the board, what
> >>>> triggers the boards watchdog?
> >>>
> >>> So the problem is when cable is not attached to the board and you
> >>> call dfu command to execute?
> >>
> >> Yes.
> >
> > For UMS gadget there is defined g_dnl_board_usb_cable_connected()
> > function.
> >
> > However, it is not yet supported in dfu and requires working MUIC
> > block, which might be not available at your board.
> 
> Maybe, I must check this ...
> 
> >>>>> What is the WATCHDOG interval on the affected board?
> >>>>
> >>>> ~5 seconds
> >>>>
> >>>> Ah, this is on an at91 board .. and in the
> >>>> drivers/serial/atmel_usart.c atmel_serial_tstc() is no
> >>>> WATCHDOG_RESET...
> >>>>
> >>>> So ctrlc() does not trigger the watchdog
> >>>
> >>> Could you be a bit more specific here. Does your board expect
> >>> ctrlc() to update watchdog, so it won't reset?
> >>
> >> I do not know, if ctrlc() must trigger the WDT ... I just looked
> >> into the while(1) loop in common/cmd_dfu.c. There is only ctrlc()
> >> which could trigger the WDT... and on at91 it does not trigger
> >> it ...
> >
> > By trigger you mean reset WDT core and don't reset the board?
> 
> I mean, the code must trigger the WDT somewhere in this while(1).
> If not, the WDT resets the board.
> 
> >> If dfu transfer is started there is some output on the console,
> >> which triggers the WDT too ... do you have a board with a running
> >> WDT?
> >
> > On trats/trats2 we disable WDT when we enter the u-boot.
> 
> Why?
> 
> So you can test this behaviour. Do not disable the WDT and start
> the dfu command ... the board should reset when you start the dfu
> command (without starting dfu-util on the host, and if ctrl() does not
> call WATCHDOG_RESET for your hw) ... with my patch it should work ...
> 
> BTW: I could not disable the WDT on this board, once it is enabled.
> So disabling the WDT is no option ... :-(

Ok, I see. Then I will apply the proposed patch.

> 
> > I can imagine following situation when WDT is enabled when u-boot
> > starts (its timeout is ~5sec) and then you start dfu transfer with
> > not connected cable.
> > Then 5sec pass since cable is not connected and no transfer is
> > ongoing. This causes board reset by WDT.
> >
> > Am I right?
> 
> Yes, because the WDT gets not triggered in this while(1). And maybe
> if you start the dfu command with connected cable, but not starting
> dfu-util on the host also, but I have not yet running the usb gadget
> on this at91 board (at91sam9g20 based) ...

Thanks for explanation. I just wanted to have bigger picture of the
problem.

> 
> bye,
> Heiko
Heiko Schocher March 18, 2015, 8:10 a.m. UTC | #10
Hello Lukasz,

Am 18.03.2015 08:44, schrieb Lukasz Majewski:
> Hi Heiko,
>
>> Hello Lukasz,
>>
>> Am 17.03.2015 16:16, schrieb Lukasz Majewski:
>>> Hi Heiko,
>>>
>>>> Hello Lukasz,
>>>>
>>>> Am 17.03.2015 13:56, schrieb Lukasz Majewski:
>>>>> Hi Heiko,
>>>>>
>>>>>> Hello Lukasz,
>>>>>>
>>>>>> Am 17.03.2015 10:52, schrieb Lukasz Majewski:
>>>>>>> Hi Heiko,
>>>>>>>
>>>>>>>> trigger watchdog before calling usb_gadget_handle_interrupts()
>>>>>>>> This prevents board resets when calling dfu command on boards
>>>>>>>> which have a watchdog.
>>>>>>>>
>>>>>>>> Signed-off-by: Heiko Schocher <hs@denx.de>
>>>>>>>> ---
>>>>>>>>
>>>>>>>>      common/cmd_dfu.c | 2 ++
>>>>>>>>      1 file changed, 2 insertions(+)
>>>>>>>>
>>>>>>>> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
>>>>>>>> index e975abe..46af4cf 100644
>>>>>>>> --- a/common/cmd_dfu.c
>>>>>>>> +++ b/common/cmd_dfu.c
>>>>>>>> @@ -9,6 +9,7 @@
>>>>>>>>       */
>>>>>>>>
>>>>>>>>      #include <common.h>
>>>>>>>> +#include <watchdog.h>
>>>>>>>>      #include <dfu.h>
>>>>>>>>      #include <g_dnl.h>
>>>>>>>>      #include <usb.h>
>>>>>>>> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int flag,
>>>>>>>> int argc, char * const argv[]) if (ctrlc())
>>>>>>>>      			goto exit;
>>>>>>>>
>>>>>>>> +		WATCHDOG_RESET();
>>>>>>>>      		usb_gadget_handle_interrupts();
>>>>>>>>      	}
>>>>>>>>      exit:
>>>>>>>
>>>>>>> It seems strange for me, that we must reset watchdog when
>>>>>>> looping in the dfu.
>>>>>>
>>>>>> Hmm.. maybe I overlook something, but If you look into this
>>>>>> while(1) loop, there is no trigger of the watchdog ... and if I
>>>>>> start the dfu command without a USB cable on the board, what
>>>>>> triggers the boards watchdog?
>>>>>
>>>>> So the problem is when cable is not attached to the board and you
>>>>> call dfu command to execute?
>>>>
>>>> Yes.
>>>
>>> For UMS gadget there is defined g_dnl_board_usb_cable_connected()
>>> function.
>>>
>>> However, it is not yet supported in dfu and requires working MUIC
>>> block, which might be not available at your board.
>>
>> Maybe, I must check this ...
>>
>>>>>>> What is the WATCHDOG interval on the affected board?
>>>>>>
>>>>>> ~5 seconds
>>>>>>
>>>>>> Ah, this is on an at91 board .. and in the
>>>>>> drivers/serial/atmel_usart.c atmel_serial_tstc() is no
>>>>>> WATCHDOG_RESET...
>>>>>>
>>>>>> So ctrlc() does not trigger the watchdog
>>>>>
>>>>> Could you be a bit more specific here. Does your board expect
>>>>> ctrlc() to update watchdog, so it won't reset?
>>>>
>>>> I do not know, if ctrlc() must trigger the WDT ... I just looked
>>>> into the while(1) loop in common/cmd_dfu.c. There is only ctrlc()
>>>> which could trigger the WDT... and on at91 it does not trigger
>>>> it ...
>>>
>>> By trigger you mean reset WDT core and don't reset the board?
>>
>> I mean, the code must trigger the WDT somewhere in this while(1).
>> If not, the WDT resets the board.
>>
>>>> If dfu transfer is started there is some output on the console,
>>>> which triggers the WDT too ... do you have a board with a running
>>>> WDT?
>>>
>>> On trats/trats2 we disable WDT when we enter the u-boot.
>>
>> Why?
>>
>> So you can test this behaviour. Do not disable the WDT and start
>> the dfu command ... the board should reset when you start the dfu
>> command (without starting dfu-util on the host, and if ctrl() does not
>> call WATCHDOG_RESET for your hw) ... with my patch it should work ...
>>
>> BTW: I could not disable the WDT on this board, once it is enabled.
>> So disabling the WDT is no option ... :-(
>
> Ok, I see. Then I will apply the proposed patch.

great!

>>> I can imagine following situation when WDT is enabled when u-boot
>>> starts (its timeout is ~5sec) and then you start dfu transfer with
>>> not connected cable.
>>> Then 5sec pass since cable is not connected and no transfer is
>>> ongoing. This causes board reset by WDT.
>>>
>>> Am I right?
>>
>> Yes, because the WDT gets not triggered in this while(1). And maybe
>> if you start the dfu command with connected cable, but not starting
>> dfu-util on the host also, but I have not yet running the usb gadget
>> on this at91 board (at91sam9g20 based) ...
>
> Thanks for explanation. I just wanted to have bigger picture of the
> problem.

all right.

bye,
Heiko
Łukasz Majewski March 19, 2015, 11:11 a.m. UTC | #11
Hi Heiko,

> trigger watchdog before calling usb_gadget_handle_interrupts()
> This prevents board resets when calling dfu command on boards
> which have a watchdog.
> 
> Signed-off-by: Heiko Schocher <hs@denx.de>
> ---
> 
>  common/cmd_dfu.c | 2 ++
>  1 file changed, 2 insertions(+)
> 
> diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
> index e975abe..46af4cf 100644
> --- a/common/cmd_dfu.c
> +++ b/common/cmd_dfu.c
> @@ -9,6 +9,7 @@
>   */
>  
>  #include <common.h>
> +#include <watchdog.h>
>  #include <dfu.h>
>  #include <g_dnl.h>
>  #include <usb.h>
> @@ -64,6 +65,7 @@ static int do_dfu(cmd_tbl_t *cmdtp, int flag, int
> argc, char * const argv[]) if (ctrlc())
>  			goto exit;
>  
> +		WATCHDOG_RESET();
>  		usb_gadget_handle_interrupts();
>  	}
>  exit:

Applied to u-boot-dfu tree! Thanks.
diff mbox

Patch

diff --git a/common/cmd_dfu.c b/common/cmd_dfu.c
index e975abe..46af4cf 100644
--- a/common/cmd_dfu.c
+++ b/common/cmd_dfu.c
@@ -9,6 +9,7 @@ 
  */
 
 #include <common.h>
+#include <watchdog.h>
 #include <dfu.h>
 #include <g_dnl.h>
 #include <usb.h>
@@ -64,6 +65,7 @@  static int do_dfu(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
 		if (ctrlc())
 			goto exit;
 
+		WATCHDOG_RESET();
 		usb_gadget_handle_interrupts();
 	}
 exit: