[U-Boot] [PATCH 1/2] at91sam9: add watchdog support

Giulio Benetti giulio.benetti at micronovasrl.com
Sat Jun 13 12:44:34 CEST 2009


Jean-Christophe PLAGNIOL-VILLARD wrote:

> Signed-off-by: Jean-Christophe PLAGNIOL-VILLARD <plagnioj at jcrosoft.com>
> ---
>  Makefile                             |    1 +
>  doc/README.at91                      |    9 ++++
>  drivers/serial/atmel_usart.c         |    4 +-
>  drivers/watchdog/Makefile            |   46 ++++++++++++++++++++
>  drivers/watchdog/at91sam9_wdt.c      |   79
>  ++++++++++++++++++++++++++++++++++
>  include/asm-arm/arch-at91/at91_wdt.h |   38 ++++++++++++++++
>  6 files changed, 176 insertions(+), 1 deletions(-)
>  create mode 100644 drivers/watchdog/Makefile
>  create mode 100644 drivers/watchdog/at91sam9_wdt.c
>  create mode 100644 include/asm-arm/arch-at91/at91_wdt.h
> 
> diff --git a/Makefile b/Makefile
> index fcf57f3..3e61417 100644
> --- a/Makefile
> +++ b/Makefile
> @@ -262,6 +262,7 @@ LIBS += drivers/rtc/librtc.a
>  LIBS += drivers/serial/libserial.a
>  LIBS += drivers/usb/libusb.a
>  LIBS += drivers/video/libvideo.a
> +LIBS += drivers/watchdog/libwatchdog.a
>  LIBS += common/libcommon.a
>  LIBS += libfdt/libfdt.a
>  LIBS += api/libapi.a
> diff --git a/doc/README.at91 b/doc/README.at91
> index 4e3928a..c883c7c 100644
> --- a/doc/README.at91
> +++ b/doc/README.at91
> @@ -2,6 +2,7 @@ Atmel AT91 Evaluation kits
>  
>  http://atmel.com/dyn/products/tools.asp?family_id=605#1443
>  
> +I. Board mapping & boot media
>  ------------------------------------------------------------------------------
>  AT91SAM9260EK & AT91SAM9XEEK
>  ------------------------------------------------------------------------------
> @@ -86,3 +87,11 @@ Environment variables
>  make at91sam9263ek_config            - use data flash (spi cs0) (default)
>  make at91sam9263ek_nandflash_config  - use nand flash
>  make at91sam9263ek_dataflash_cs0_config      - use data flash (spi cs0)
> +
> +II. Watchdog support
> +
> +     The watchdog wan only be activate once so you need to activate in the
> +     bootloader and in the bootstrap if you use it
> +
> +     The you can activate with
> +     CONFIG_AT91SAM9_WATCHDOG and CONFIG_HW_WATCHDOG
> diff --git a/drivers/serial/atmel_usart.c b/drivers/serial/atmel_usart.c
> index f3b146c..f50552a 100644
> --- a/drivers/serial/atmel_usart.c
> +++ b/drivers/serial/atmel_usart.c
> @@ -16,6 +16,7 @@
>   * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 
>   USA */
>  #include <common.h>
> +#include <watchdog.h>
>  
>  #include <asm/io.h>
>  #include <asm/arch/clk.h>
> @@ -87,7 +88,8 @@ void serial_puts(const char *s)
>  
>  int serial_getc(void)
>  {
> -     while (!(usart3_readl(CSR) & USART3_BIT(RXRDY))) ;
> +     while (!(usart3_readl(CSR) & USART3_BIT(RXRDY)))
> +              WATCHDOG_RESET();
>  return usart3_readl(RHR);
>  }
>  
> diff --git a/drivers/watchdog/Makefile b/drivers/watchdog/Makefile
> new file mode 100644
> index 0000000..200968d
> --- /dev/null
> +++ b/drivers/watchdog/Makefile
> @@ -0,0 +1,46 @@
> +#
> +# (C) Copyright 2008
> +# Wolfgang Denk, DENX Software Engineering, wd at denx.de.
> +#
> +# See file CREDITS for list of people who contributed to this
> +# project.
> +#
> +# This program is free software; you can redistribute it and/or
> +# modify it under the terms of the GNU General Public License as
> +# published by the Free Software Foundation; either version 2 of
> +# the License, or (at your option) any later version.
> +#
> +# This program is distributed in the hope that it will be useful,
> +# but WITHOUT ANY WARRANTY; without even the implied warranty of
> +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.       See the
> +# GNU General Public License for more details.
> +#
> +# You should have received a copy of the GNU General Public License
> +# along with this program; if not, write to the Free Software
> +# Foundation, Inc., 59 Temple Place, Suite 330, Boston,
> +# MA 02111-1307 USA
> +#
> +
> +include $(TOPDIR)/config.mk
> +
> +LIB  := $(obj)libwatchdog.a
> +
> +COBJS-$(CONFIG_AT91SAM9_WATCHDOG) += at91sam9_wdt.o
> +
> +COBJS        := $(COBJS-y)
> +SRCS := $(COBJS:.o=.c)
> +OBJS := $(addprefix $(obj),$(COBJS))
> +
> +all: $(LIB)
> +
> +$(LIB):      $(obj).depend $(OBJS)
> +     $(AR) $(ARFLAGS) $@ $(OBJS)
> +
> +#########################################################################
> +
> +# defines $(obj).depend target
> +include $(SRCTREE)/rules.mk
> +
> +sinclude $(obj).depend
> +
> +#########################################################################
> diff --git a/drivers/watchdog/at91sam9_wdt.c
> b/drivers/watchdog/at91sam9_wdt.c new file mode 100644
> index 0000000..5bb8b77
> --- /dev/null
> +++ b/drivers/watchdog/at91sam9_wdt.c
> @@ -0,0 +1,79 @@
> +/*
> + * [origin: Linux kernel drivers/watchdog/at91sam9_wdt.c]
> + *
> + * Watchdog driver for Atmel AT91SAM9x processors.
> + *
> + * Copyright (C) 2008 Jean-Christophe PLAGNIOL-VILLARD
> <plagnioj at jcrosoft.com> + * Copyright (C) 2008 Renaud CERRATO
> r.cerrato at til-technologies.fr + *
> + * This program is free software; you can redistribute it and/or
> + * modify it under the terms of the GNU General Public License
> + * as published by the Free Software Foundation; either version
> + * 2 of the License, or (at your option) any later version.
> + */
> +
> +/*
> + * The Watchdog Timer Mode Register can be only written to once. If the
> + * timeout need to be set from U-Boot, be sure that the bootstrap doesn't
> + * write to this register. Inform Linux to it too
> + */
> +
> +#include <common.h>
> +#include <watchdog.h>
> +#include <asm/arch/hardware.h>
> +#include <asm/arch/io.h>
> +#include <asm/arch/at91_wdt.h>
> +
> +/*
> + * AT91SAM9 watchdog runs a 12bit counter @ 256Hz,
> + * use this to convert a watchdog
> + * value from/to milliseconds.
> + */
> +#define ms_to_ticks(t)       (((t << 8) / 1000) - 1)
> +#define ticks_to_ms(t)       (((t + 1) * 1000) >> 8)
> +
> +/* Hardware timeout in seconds */
> +#define WDT_HW_TIMEOUT 2
> +
> +/*
> + * Set the watchdog time interval in 1/256Hz (write-once)
> + * Counter is 12 bit.
> + */
> +static int at91_wdt_settimeout(unsigned int timeout)
> +{
> +     unsigned int reg;
> +     unsigned int mr;
> +
> +     /* Check if disabled */
> +     mr = at91_sys_read(AT91_WDT_MR);
> +     if (mr & AT91_WDT_WDDIS) {
> +             printf("sorry, watchdog is disabled\n");
> +             return -1;
> +     }
> +
> +     /*
> +      * All counting occurs at SLOW_CLOCK / 128 = 256 Hz
> +      *
> +      * Since WDV is a 12-bit counter, the maximum period is
> +      * 4096 / 256 = 16 seconds.
> +      */
> +     reg = AT91_WDT_WDRSTEN  /* causes watchdog reset */
> +             /* | AT91_WDT_WDRPROC   causes processor reset only */
> +             | AT91_WDT_WDDBGHLT             /* disabled in debug mode */
> +             | AT91_WDT_WDD                  /* restart at any time */
> +             | (timeout & AT91_WDT_WDV);     /* timer value */
> +     at91_sys_write(AT91_WDT_MR, reg);
> +
> +     return 0;
> +}
> +
> +void hw_watchdog_reset(void)
> +{
> +     at91_sys_write(AT91_WDT_CR, AT91_WDT_KEY | AT91_WDT_WDRSTT);
> +}
> +
> +void hw_watchdog_init(void)
> +{
> +     /* 16 seconds timer, resets enabled */
> +     at91_wdt_settimeout(ms_to_ticks(WDT_HW_TIMEOUT * 1000));
> +}
> diff --git a/include/asm-arm/arch-at91/at91_wdt.h
> b/include/asm-arm/arch-at91/at91_wdt.h new file mode 100644
> index 0000000..7e18537
> --- /dev/null
> +++ b/include/asm-arm/arch-at91/at91_wdt.h
> @@ -0,0 +1,38 @@
> +/*
> + * [origin: Linux kernel arch/arm/mach-at91/include/mach/at91_wdt.h]
> + *
> + * Copyright (C) 2008 Jean-Christophe PLAGNIOL-VILLARD
> <plagnioj at jcrosoft.com> + * Copyright (C) 2007 Andrew Victor
> + * Copyright (C) 2007 Atmel Corporation.
> + *
> + * Watchdog Timer (WDT) - System peripherals regsters.
> + * Based on AT91SAM9261 datasheet revision D.
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License as published by
> + * the Free Software Foundation; either version 2 of the License, or
> + * (at your option) any later version.
> + */
> +
> +#ifndef AT91_WDT_H
> +#define AT91_WDT_H
> +
> +#define AT91_WDT_CR          (AT91_WDT + 0x00)       /* Watchdog Control Register */
> +#define              AT91_WDT_WDRSTT         (1    << 0)             /* Restart */
> +#define              AT91_WDT_KEY            (0xa5 << 24)            /* KEY Password */
> +
> +#define AT91_WDT_MR          (AT91_WDT + 0x04)       /* Watchdog Mode Register */
> +#define              AT91_WDT_WDV            (0xfff << 0)            /* Counter Value */
> +#define              AT91_WDT_WDFIEN         (1     << 12)           /* Fault Interrupt Enable */
> +#define              AT91_WDT_WDRSTEN        (1     << 13)           /* Reset Processor */
> +#define              AT91_WDT_WDRPROC        (1     << 14)           /* Timer Restart */
> +#define              AT91_WDT_WDDIS          (1     << 15)           /* Watchdog Disable */
> +#define              AT91_WDT_WDD            (0xfff << 16)           /* Delta Value */
> +#define              AT91_WDT_WDDBGHLT       (1     << 28)           /* Debug Halt */
> +#define              AT91_WDT_WDIDLEHLT      (1     << 29)           /* Idle Halt */
> +
> +#define AT91_WDT_SR          (AT91_WDT + 0x08)       /* Watchdog Status Register */
> +#define              AT91_WDT_WDUNF          (1 << 0)                /* Watchdog Underflow */
> +#define              AT91_WDT_WDERR          (1 << 1)                /* Watchdog Error */
> +
> +#endif

In this way the linux driver won't work(the one by Renaud CERRATO).
Because it expects MR register not to be set, or to be set enabled.

After I don't agree refreshing watchdog inside atmel_usart.c
If u-boot hangs there instead of execute bootcmd, then the machine lies
there forever.
In my opinion we should put WATCHDOG_RESET() inside main_loop and nand write
and read.
What do you think?

Kind regards

-- 
Giulio Benetti
R&D
Micronova srl



More information about the U-Boot mailing list