mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-25 18:07:22 +08:00
485 lines
14 KiB
C
485 lines
14 KiB
C
/**********************************************************************
|
|
* $Id$ lpc_wwdt.c 2011-06-02
|
|
*//**
|
|
* @file lpc_wwdt.c
|
|
* @brief Contains all functions support for Wachtdog Timer
|
|
* firmware library on LPC
|
|
* @version 1.0
|
|
* @date 02. June. 2011
|
|
* @author NXP MCU SW Application Team
|
|
*
|
|
* Copyright(C) 2011, NXP Semiconductor
|
|
* All rights reserved.
|
|
*
|
|
***********************************************************************
|
|
* Software that is described herein is for illustrative purposes only
|
|
* which provides customers with programming information regarding the
|
|
* products. This software is supplied "AS IS" without any warranties.
|
|
* NXP Semiconductors assumes no responsibility or liability for the
|
|
* use of the software, conveys no license or title under any patent,
|
|
* copyright, or mask work right to the product. NXP Semiconductors
|
|
* reserves the right to make changes in the software without
|
|
* notification. NXP Semiconductors also make no representation or
|
|
* warranty that such application will be suitable for the specified
|
|
* use without further testing or modification.
|
|
* Permission to use, copy, modify, and distribute this software and its
|
|
* documentation is hereby granted, under NXP Semiconductors'
|
|
* relevant copyright in the software, without fee, provided that it
|
|
* is used in conjunction with NXP Semiconductors microcontrollers. This
|
|
* copyright, permission, and disclaimer notice must appear in all copies of
|
|
* this code.
|
|
**********************************************************************/
|
|
|
|
/* Peripheral group ----------------------------------------------------------- */
|
|
/** @addtogroup WWDT
|
|
* @{
|
|
*/
|
|
#ifdef __BUILD_WITH_EXAMPLE__
|
|
#include "lpc_libcfg.h"
|
|
#else
|
|
#include "lpc_libcfg_default.h"
|
|
#endif /* __BUILD_WITH_EXAMPLE__ */
|
|
#ifdef _WDT
|
|
|
|
/* Includes ------------------------------------------------------------------- */
|
|
#include "lpc_wwdt.h"
|
|
#include "lpc_clkpwr.h"
|
|
#include "lpc_pinsel.h"
|
|
|
|
|
|
/* Public Functions ----------------------------------------------------------- */
|
|
/** @addtogroup WDT_Public_Functions
|
|
* @{
|
|
*/
|
|
|
|
/********************************************************************//**
|
|
* @brief Set timeout value to Timer Constant register
|
|
* (if enable) to generate a WatchDog event if match
|
|
*
|
|
* @param[in] timeoutVal The value (counter) will write directly to
|
|
* Register w/o pre-calc
|
|
*
|
|
* @return None
|
|
*********************************************************************/
|
|
int8_t WWDT_SetTimeOutRaw(uint32_t timeoutVal)
|
|
{
|
|
int8_t retval = 0;
|
|
|
|
if(timeoutVal < WWDT_TIMEOUT_MIN)
|
|
{
|
|
timeoutVal = WWDT_TIMEOUT_MIN;
|
|
retval = WWDT_FUNC_BAD_PARAM;
|
|
}
|
|
else if (timeoutVal > WWDT_TIMEOUT_MAX)
|
|
{
|
|
timeoutVal = WWDT_TIMEOUT_MAX;
|
|
retval = WWDT_FUNC_BAD_PARAM;
|
|
}
|
|
|
|
LPC_WDT->TC = timeoutVal;
|
|
|
|
return retval;
|
|
}
|
|
|
|
|
|
/********************************************************************//**
|
|
* @brief Set WDT timeout (cal by usec) to TC register
|
|
*
|
|
* @param[in] timeout The time (usec) to generate watchdog event if
|
|
* the watchdog counter reach this value
|
|
*
|
|
* @return WWDT_FUNC_OK if success
|
|
*********************************************************************/
|
|
int8_t WWDT_SetTimeOut(uint32_t timeout)
|
|
{
|
|
return WWDT_SetTimeOutRaw(WDT_GET_FROM_USEC(timeout));
|
|
}
|
|
|
|
|
|
/*********************************************************************//**
|
|
* @brief Initial for Watchdog function by setting timeout
|
|
*
|
|
* @param[in] TimeOut time out value, should be in range:
|
|
* 2048 .. 134217728
|
|
*
|
|
* @return WWDT_FUNC_OK if success
|
|
**********************************************************************/
|
|
int8_t WWDT_Init(uint32_t TimeOut)
|
|
{
|
|
return WWDT_SetTimeOut(TimeOut);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
* @brief Configure the WatchDog by initialization all the timing
|
|
* value for register (Warning value, Window value,...)
|
|
*
|
|
* @param wdtCfg a the st_Wdt_Config type value
|
|
*
|
|
* @return None
|
|
**********************************************************************/
|
|
void WWDT_Configure(st_Wdt_Config wdtCfg)
|
|
{
|
|
WWDT_SetTimeOut(wdtCfg.wdtTmrConst);
|
|
|
|
if(wdtCfg.wdtEnable)
|
|
{
|
|
LPC_WDT->MOD |= WWDT_WDMOD_WDEN;
|
|
}
|
|
else
|
|
{
|
|
LPC_WDT->MOD &= ~WWDT_WDMOD_WDEN;
|
|
}
|
|
|
|
if(wdtCfg.wdtReset)
|
|
{
|
|
LPC_WDT->MOD |= WWDT_WDMOD_WDRESET;
|
|
}
|
|
else
|
|
{
|
|
LPC_WDT->MOD &= ~WWDT_WDMOD_WDRESET;
|
|
}
|
|
|
|
if(wdtCfg.wdtProtect)
|
|
{
|
|
LPC_WDT->MOD |= WWDT_WDMOD_WDPROTECT;
|
|
}
|
|
else
|
|
{
|
|
LPC_WDT->MOD &= ~WWDT_WDMOD_WDPROTECT;
|
|
}
|
|
|
|
WWDT_SetWarning(wdtCfg.wdtWarningVal);
|
|
|
|
WWDT_SetWindow(wdtCfg.wdtWindowVal);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
* @brief Start WatchDog with specific Timeout
|
|
*
|
|
* @param TimeOut specific Timeout for WatchDog event
|
|
*
|
|
* @return WWDT_FUNC_OK if success
|
|
**********************************************************************/
|
|
int8_t WWDT_Start(uint32_t TimeOut)
|
|
{
|
|
int8_t retval = WWDT_FUNC_OK;
|
|
|
|
retval = WWDT_SetTimeOut(TimeOut);
|
|
|
|
WWDT_Cmd(ENABLE);
|
|
|
|
return retval;
|
|
}
|
|
|
|
|
|
/********************************************************************//**
|
|
* @brief Update WDT timeout value and feed
|
|
*
|
|
* @param[in] WarnTime time to generate watchdog warning interrupt(us)
|
|
* should be in range: 2048 .. 134217728
|
|
*
|
|
* @return None
|
|
*********************************************************************/
|
|
void WWDT_SetTimerConstant(uint32_t constVal)
|
|
{
|
|
LPC_WDT->TC = constVal;
|
|
}
|
|
|
|
|
|
/*********************************************************************//**
|
|
* @brief Enable/Disable WDT function
|
|
*
|
|
* @param[in] Mode WWDT mode that will be enabled/disabled, should be:
|
|
* - WWDT_PROTECT_MODE : protect mode
|
|
* - WWDT_RESET_MODE : reset mode
|
|
*
|
|
* @param[in] NewState new state of protection function, should be:
|
|
* - ENABLE: The watchdog reload value can be changed at any time
|
|
* - DISABLE: The watchdog reload value can be changed only after
|
|
* the counter is below the value of WDWARNINT and WDWINDOW
|
|
*
|
|
* @return None
|
|
**********************************************************************/
|
|
void WWDT_SetMode(uint8_t mode, FunctionalState NewState)
|
|
{
|
|
if (mode == WWDT_PROTECT_MODE )
|
|
{
|
|
if(NewState == ENABLE)
|
|
LPC_WDT->MOD |= WWDT_WDMOD_WDPROTECT;
|
|
else
|
|
LPC_WDT->MOD &= ~WWDT_WDMOD_WDPROTECT;
|
|
}
|
|
else if(mode == WWDT_RESET_MODE)
|
|
{
|
|
if(NewState == ENABLE)
|
|
LPC_WDT->MOD |= WWDT_WDMOD_WDRESET;
|
|
else
|
|
LPC_WDT->MOD &= ~WWDT_WDMOD_WDRESET;
|
|
}
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
* @brief Enable/Disable WWDT activity
|
|
*
|
|
* @param[in] NewState To enable/disable the WatchDog
|
|
*
|
|
* @return None
|
|
**********************************************************************/
|
|
void WWDT_Enable(FunctionalState NewState)
|
|
{
|
|
if(NewState == ENABLE)
|
|
{
|
|
LPC_WDT->MOD |= WWDT_WDMOD_WDEN;
|
|
}
|
|
else
|
|
{
|
|
LPC_WDT->MOD &= ~WWDT_WDMOD_WDEN;
|
|
}
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
* @brief Enable/Disable WWDT activity. In case of Enable, it will
|
|
* do feeding WatchDog for its normal operation.
|
|
*
|
|
* @param[in] NewState To enable/disable the WatchDog
|
|
*
|
|
* @return None
|
|
**********************************************************************/
|
|
void WWDT_Cmd(FunctionalState NewState)
|
|
{
|
|
if(NewState == ENABLE)
|
|
{
|
|
LPC_WDT->MOD |= WWDT_WDMOD_WDEN;
|
|
|
|
//Load the Feed register to start using WDT
|
|
WWDT_Feed();
|
|
}
|
|
else
|
|
{
|
|
LPC_WDT->MOD &= ~WWDT_WDMOD_WDEN;
|
|
}
|
|
}
|
|
|
|
/********************************************************************//**
|
|
* @brief Set the warning value to register to generate interrupt
|
|
* (if enable) if the watchdog timer matches this value.
|
|
*
|
|
* @param[in] warnVal The value (counter) will write directly to Warning
|
|
* Register w/o pre-calc
|
|
*
|
|
* @return WWDT_FUNC_OK if success
|
|
*********************************************************************/
|
|
int8_t WWDT_SetWarningRaw(uint32_t warnVal)
|
|
{
|
|
int8_t retval = WWDT_FUNC_OK;
|
|
|
|
if(warnVal < WWDT_WARNINT_MIN)
|
|
{
|
|
warnVal = WWDT_WARNINT_MIN;
|
|
retval = WWDT_FUNC_BAD_PARAM;
|
|
}
|
|
else if (warnVal > WWDT_WARNINT_MAX)
|
|
{
|
|
warnVal = WWDT_WARNINT_MAX;
|
|
retval = WWDT_FUNC_BAD_PARAM;
|
|
}
|
|
|
|
LPC_WDT->WARNINT = warnVal;
|
|
|
|
return retval;
|
|
}
|
|
|
|
/********************************************************************//**
|
|
* @brief Update WDT warning time (cal by usec) to warning register
|
|
*
|
|
* @param[in] WarnTime The time (usec) to generate watchdog warning
|
|
* interrupt should be in range: 2048 .. 8192
|
|
*
|
|
* @return WWDT_FUNC_OK if success
|
|
*********************************************************************/
|
|
int8_t WWDT_SetWarning(uint32_t WarnTime)
|
|
{
|
|
return WWDT_SetWarningRaw(WDT_GET_FROM_USEC(WarnTime));
|
|
}
|
|
|
|
|
|
/********************************************************************//**
|
|
* @brief Update WDT Windows value (by counter) to Window Register
|
|
* of WatchDog
|
|
*
|
|
* @param[in] wndVal The value (counter) will write directly to Window
|
|
* Register w/o pre-calc
|
|
*
|
|
* @return WWDT_FUNC_OK if success
|
|
*********************************************************************/
|
|
int8_t WWDT_SetWindowRaw(uint32_t wndVal)
|
|
{
|
|
int8_t retval = WWDT_FUNC_OK;
|
|
|
|
if(wndVal < WWDT_WINDOW_MIN)
|
|
{
|
|
wndVal = WWDT_WINDOW_MIN;
|
|
retval = WWDT_FUNC_BAD_PARAM;
|
|
}
|
|
else if (wndVal > WWDT_WINDOW_MAX)
|
|
{
|
|
wndVal = WWDT_WINDOW_MAX;
|
|
retval = WWDT_FUNC_BAD_PARAM;
|
|
}
|
|
|
|
LPC_WDT->WINDOW = wndVal;
|
|
|
|
return retval;
|
|
}
|
|
|
|
|
|
/********************************************************************//**
|
|
* @brief Update WDT Windows value (by usec) to Window Register
|
|
* of WatchDog
|
|
*
|
|
* @param[in] WindowedTime Expected time (usec) to set watchdog window event
|
|
*
|
|
* @return WWDT_FUNC_OK if success
|
|
*********************************************************************/
|
|
int8_t WWDT_SetWindow(uint32_t WindowedTime)
|
|
{
|
|
return WWDT_SetWindowRaw(WDT_GET_FROM_USEC(WindowedTime));
|
|
}
|
|
|
|
|
|
/********************************************************************//**
|
|
* @brief Update WDT timeout value for WatchDog event(s)
|
|
*
|
|
* @param[in] TimeOut Time Out value (usec) to be updated, should be
|
|
* in range: 2048 .. 134217728
|
|
*
|
|
* @return None
|
|
*********************************************************************/
|
|
void WDT_UpdateTimeOut(uint32_t TimeOut)
|
|
{
|
|
/* check WDPROTECT,
|
|
* if it is enable, wait until the counter is below the value of
|
|
* WDWARNINT and WDWINDOW
|
|
*/
|
|
if(LPC_WDT->MOD & (1<<4))
|
|
{
|
|
while((LPC_WDT->TV <(LPC_WDT->WARNINT & WWDT_WDWARNINT_MASK))\
|
|
&&(LPC_WDT->TV <(LPC_WDT->WINDOW & WWDT_WDTC_MASK)));
|
|
}
|
|
|
|
WWDT_SetTimeOut(TimeOut);
|
|
}
|
|
|
|
|
|
/********************************************************************//**
|
|
* @brief Read WWDT status flag
|
|
*
|
|
* @param[in] Status kind of status flag that you want to get, should be:
|
|
* - WWDT_WARNINT_FLAG: watchdog interrupt flag
|
|
* - WWDT_TIMEOUT_FLAG: watchdog time-out flag
|
|
*
|
|
* @return Time out flag status of WDT
|
|
*********************************************************************/
|
|
FlagStatus WWDT_GetStatus (uint8_t Status)
|
|
{
|
|
if(Status == WWDT_WARNINT_FLAG)
|
|
{
|
|
return ((FlagStatus)(LPC_WDT->MOD & (1<<3)));
|
|
}
|
|
else if (Status == WWDT_TIMEOUT_FLAG)
|
|
{
|
|
return ((FlagStatus)(LPC_WDT->MOD & (1<<2)));
|
|
}
|
|
return RESET;
|
|
}
|
|
|
|
/********************************************************************//**
|
|
* @brief Clear WWDT status flag
|
|
*
|
|
* @param[in] Status kind of status flag that you want to get, should be:
|
|
* - WWDT_WARNINT_FLAG: watchdog interrupt flag
|
|
* - WWDT_TIMEOUT_FLAG: watchdog time-out flag
|
|
*
|
|
* @return None
|
|
*********************************************************************/
|
|
void WWDT_ClearStatusFlag (uint8_t flag)
|
|
{
|
|
if(flag == WWDT_WARNINT_FLAG)
|
|
{
|
|
// Write 1 to this bit to clear itself
|
|
LPC_WDT->MOD |= WWDT_WDMOD_WDINT;
|
|
}
|
|
else if(flag == WWDT_TIMEOUT_FLAG)
|
|
{
|
|
// Write 0 to this bit to clear itself
|
|
LPC_WDT->MOD &= ~ WWDT_WDMOD_WDTOF;
|
|
}
|
|
}
|
|
|
|
/********************************************************************//**
|
|
* @brief Clear WDT Time out flag
|
|
* @param None
|
|
* @return None
|
|
*********************************************************************/
|
|
void WWDT_ClrTimeOutFlag (void)
|
|
{
|
|
LPC_WDT->MOD &= ~ WWDT_WDMOD_WDTOF;
|
|
}
|
|
|
|
|
|
/********************************************************************//**
|
|
* @brief Following the standard sequence to Feed the WatchDog Timer
|
|
*
|
|
* @param None
|
|
*
|
|
* @return None
|
|
*********************************************************************/
|
|
void WWDT_FeedStdSeq (void)
|
|
{
|
|
LPC_WDT->FEED = 0xAA;
|
|
|
|
LPC_WDT->FEED = 0x55;
|
|
}
|
|
|
|
/********************************************************************//**
|
|
* @brief After set WDTEN, call this function to start Watchdog
|
|
* or reload the Watchdog timer
|
|
*
|
|
* @param None
|
|
*
|
|
* @return None
|
|
*********************************************************************/
|
|
void WWDT_Feed (void)
|
|
{
|
|
// Disable irq interrupt
|
|
__disable_irq();
|
|
|
|
WWDT_FeedStdSeq();
|
|
|
|
// Then enable irq interrupt
|
|
__enable_irq();
|
|
}
|
|
|
|
/********************************************************************//**
|
|
* @brief Get the current value of WDT
|
|
*
|
|
* @param None
|
|
*
|
|
* @return Current value of WDT
|
|
*********************************************************************/
|
|
uint32_t WWDT_GetCurrentCount(void)
|
|
{
|
|
return LPC_WDT->TV;
|
|
}
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
#endif /*_WDT*/
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/* --------------------------------- End Of File ------------------------------ */
|