mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-22 06:59:47 +08:00
315 lines
8.8 KiB
C
315 lines
8.8 KiB
C
/*
|
|
* Copyright 2020 NXP
|
|
* All rights reserved.
|
|
*
|
|
* SPDX-License-Identifier: BSD-3-Clause
|
|
*/
|
|
|
|
#include "fsl_cdog.h"
|
|
|
|
/*******************************************************************************
|
|
* Definitions
|
|
*******************************************************************************/
|
|
|
|
/* Component ID definition, used by tools. */
|
|
#ifndef FSL_COMPONENT_ID
|
|
#define FSL_COMPONENT_ID "platform.drivers.cdog"
|
|
#endif
|
|
|
|
/*******************************************************************************
|
|
* Prototypes
|
|
******************************************************************************/
|
|
|
|
/*******************************************************************************
|
|
* Code
|
|
******************************************************************************/
|
|
|
|
/*!
|
|
* brief Sets the default configuration of CDOG
|
|
*
|
|
* This function initialize CDOG config structure to default values.
|
|
*
|
|
* param conf CDOG configuration structure
|
|
*/
|
|
void CDOG_GetDefaultConfig(cdog_config_t *conf)
|
|
{
|
|
/* Default configuration after reset */
|
|
conf->lock = (uint8_t)kCDOG_LockCtrl_Unlock; /* Lock control */
|
|
conf->timeout = (uint8_t)kCDOG_FaultCtrl_NoAction; /* Timeout control */
|
|
conf->miscompare = (uint8_t)kCDOG_FaultCtrl_NoAction; /* Miscompare control */
|
|
conf->sequence = (uint8_t)kCDOG_FaultCtrl_NoAction; /* Sequence control */
|
|
conf->state = (uint8_t)kCDOG_FaultCtrl_NoAction; /* State control */
|
|
conf->address = (uint8_t)kCDOG_FaultCtrl_NoAction; /* Address control */
|
|
conf->irq_pause = (uint8_t)kCDOG_IrqPauseCtrl_Run; /* IRQ pause control */
|
|
conf->debug_halt = (uint8_t)kCDOG_DebugHaltCtrl_Run; /* Debug halt control */
|
|
return;
|
|
}
|
|
|
|
/*!
|
|
* brief Sets secure counter and instruction timer values
|
|
*
|
|
* This function sets value in RELOAD and START registers for instruction timer.
|
|
*
|
|
* param base CDOG peripheral base address
|
|
* param reload reload value
|
|
* param start start value
|
|
*/
|
|
void CDOG_Start(CDOG_Type *base, uint32_t reload, uint32_t start)
|
|
{
|
|
base->RELOAD = reload;
|
|
base->START = start;
|
|
}
|
|
|
|
/*!
|
|
* brief Stops secure counter and instruction timer
|
|
*
|
|
* This function stops instruction timer and secure counter.
|
|
* This also change state of CDOG to IDLE.
|
|
*
|
|
* param base CDOG peripheral base address
|
|
* param stop expected value which will be compared with value of secure counter
|
|
*/
|
|
void CDOG_Stop(CDOG_Type *base, uint32_t stop)
|
|
{
|
|
base->STOP = stop;
|
|
}
|
|
|
|
/*!
|
|
* brief Sets secure counter and instruction timer values
|
|
*
|
|
* This function sets value in STOP, RELOAD and START registers
|
|
* for instruction timer and secure counter.
|
|
*
|
|
* param base CDOG peripheral base address
|
|
* param stop expected value which will be compared with value of secure counter
|
|
* param reload reload value for instruction timer
|
|
* param start start value for secure timer
|
|
*/
|
|
void CDOG_Set(CDOG_Type *base, uint32_t stop, uint32_t reload, uint32_t start)
|
|
{
|
|
base->STOP = stop;
|
|
base->RELOAD = reload;
|
|
base->START = start;
|
|
}
|
|
|
|
/*!
|
|
* brief Add value to secure counter
|
|
*
|
|
* This function add specified value to secure counter.
|
|
*
|
|
* param base CDOG peripheral base address.
|
|
* param add Value to be added.
|
|
*/
|
|
void CDOG_Add(CDOG_Type *base, uint32_t add)
|
|
{
|
|
base->ADD = (secure_counter_t)add;
|
|
}
|
|
|
|
/*!
|
|
* brief Add 1 to secure counter
|
|
*
|
|
* This function add 1 to secure counter.
|
|
*
|
|
* param base CDOG peripheral base address.
|
|
* param add Value to be added.
|
|
*/
|
|
void CDOG_Add1(CDOG_Type *base)
|
|
{
|
|
base->ADD1 = (secure_counter_t)0x1U;
|
|
}
|
|
|
|
/*!
|
|
* brief Add 16 to secure counter
|
|
*
|
|
* This function add 16 to secure counter.
|
|
*
|
|
* param base CDOG peripheral base address.
|
|
* param add Value to be added.
|
|
*/
|
|
void CDOG_Add16(CDOG_Type *base)
|
|
{
|
|
base->ADD16 = (secure_counter_t)0x1U;
|
|
}
|
|
|
|
/*!
|
|
* brief Add 256 to secure counter
|
|
*
|
|
* This function add 256 to secure counter.
|
|
*
|
|
* param base CDOG peripheral base address.
|
|
* param add Value to be added.
|
|
*/
|
|
void CDOG_Add256(CDOG_Type *base)
|
|
{
|
|
base->ADD256 = (secure_counter_t)0x1U;
|
|
}
|
|
|
|
/*!
|
|
* brief Substract value to secure counter
|
|
*
|
|
* This function substract specified value to secure counter.
|
|
*
|
|
* param base CDOG peripheral base address.
|
|
* param sub Value to be substracted.
|
|
*/
|
|
void CDOG_Sub(CDOG_Type *base, uint32_t sub)
|
|
{
|
|
base->SUB = (secure_counter_t)sub;
|
|
}
|
|
|
|
/*!
|
|
* brief Substract 1 from secure counter
|
|
*
|
|
* This function substract specified 1 from secure counter.
|
|
*
|
|
* param base CDOG peripheral base address.
|
|
*/
|
|
void CDOG_Sub1(CDOG_Type *base)
|
|
{
|
|
base->SUB1 = (secure_counter_t)0x1U;
|
|
}
|
|
|
|
/*!
|
|
* brief Substract 16 from secure counter
|
|
*
|
|
* This function substract specified 16 from secure counter.
|
|
*
|
|
* param base CDOG peripheral base address.
|
|
*/
|
|
void CDOG_Sub16(CDOG_Type *base)
|
|
{
|
|
base->SUB16 = (secure_counter_t)0x1U;
|
|
}
|
|
|
|
/*!
|
|
* brief Substract 256 from secure counter
|
|
*
|
|
* This function substract specified 256 from secure counter.
|
|
*
|
|
* param base CDOG peripheral base address.
|
|
*/
|
|
void CDOG_Sub256(CDOG_Type *base)
|
|
{
|
|
base->SUB256 = (secure_counter_t)0x1U;
|
|
}
|
|
|
|
/*!
|
|
* brief Checks secure counter.
|
|
*
|
|
* This function compares stop value with secure counter value
|
|
* by writting to RELOAD refister.
|
|
*
|
|
* param base CDOG peripheral base address
|
|
* param check expected (stop) value.
|
|
*/
|
|
void CDOG_Check(CDOG_Type *base, uint32_t check)
|
|
{
|
|
base->RESTART = check;
|
|
}
|
|
|
|
/*!
|
|
* brief Set the CDOG persistent word.
|
|
*
|
|
* param base CDOG peripheral base address.
|
|
* param value The value to be written.
|
|
*/
|
|
void CDOG_WritePersistent(CDOG_Type *base, uint32_t value)
|
|
{
|
|
base->PERSISTENT = value;
|
|
}
|
|
|
|
/*!
|
|
* brief Get the CDOG persistent word.
|
|
*
|
|
* param base CDOG peripheral base address.
|
|
* return The persistent word.
|
|
*/
|
|
uint32_t CDOG_ReadPersistent(CDOG_Type *base)
|
|
{
|
|
return base->PERSISTENT;
|
|
}
|
|
|
|
/*!
|
|
* brief Initialize CDOG
|
|
*
|
|
* This function initializes CDOG block and setting.
|
|
*
|
|
* param base CDOG peripheral base address
|
|
* param conf CDOG configuration structure
|
|
* return Status of the init operation
|
|
*/
|
|
status_t CDOG_Init(CDOG_Type *base, cdog_config_t *conf)
|
|
{
|
|
/* Ungate clock to CDOG engine and reset it */
|
|
#if !(defined(FSL_SDK_DISABLE_DRIVER_CLOCK_CONTROL) && FSL_SDK_DISABLE_DRIVER_CLOCK_CONTROL)
|
|
#ifdef CDOG_CLOCKS
|
|
CLOCK_EnableClock(kCLOCK_Cdog);
|
|
#endif /* CDOG_CLOCKS */
|
|
#endif /* !FSL_SDK_DISABLE_DRIVER_CLOCK_CONTROL */
|
|
|
|
#if !(defined(FSL_FEATURE_CDOG_HAS_NO_RESET) && FSL_FEATURE_CDOG_HAS_NO_RESET)
|
|
RESET_PeripheralReset(kCDOG_RST_SHIFT_RSTn);
|
|
#endif /* !FSL_FEATURE_CDOG_HAS_NO_RESET */
|
|
|
|
if (base->CONTROL == 0x0U)
|
|
{
|
|
/* CDOG is not in IDLE mode, which may be cause after SW reset. */
|
|
/* Writing to CONTROL register will trigger fault. */
|
|
return kStatus_Fail;
|
|
}
|
|
|
|
/* Clear pending errors, otherwise the device will reset */
|
|
/* itself immediately after enable Code Watchdog */
|
|
if ((uint32_t)kCDOG_LockCtrl_Lock ==
|
|
((base->CONTROL & CDOG_CONTROL_LOCK_CTRL_MASK) >> CDOG_CONTROL_LOCK_CTRL_SHIFT))
|
|
|
|
{
|
|
CDOG->FLAGS = CDOG_FLAGS_TO_FLAG(1U) | CDOG_FLAGS_MISCOM_FLAG(1U) | CDOG_FLAGS_SEQ_FLAG(1U) |
|
|
CDOG_FLAGS_CNT_FLAG(1U) | CDOG_FLAGS_STATE_FLAG(1U) | CDOG_FLAGS_ADDR_FLAG(1U) |
|
|
CDOG_FLAGS_POR_FLAG(1U);
|
|
}
|
|
else
|
|
{
|
|
CDOG->FLAGS = CDOG_FLAGS_TO_FLAG(0U) | CDOG_FLAGS_MISCOM_FLAG(0U) | CDOG_FLAGS_SEQ_FLAG(0U) |
|
|
CDOG_FLAGS_CNT_FLAG(0U) | CDOG_FLAGS_STATE_FLAG(0U) | CDOG_FLAGS_ADDR_FLAG(0U) |
|
|
CDOG_FLAGS_POR_FLAG(0U);
|
|
}
|
|
|
|
base->CONTROL =
|
|
CDOG_CONTROL_TIMEOUT_CTRL(conf->timeout) | /* Action if the timeout event is triggered */
|
|
CDOG_CONTROL_MISCOMPARE_CTRL(conf->miscompare) | /* Action if the miscompare error event is triggered */
|
|
CDOG_CONTROL_SEQUENCE_CTRL(conf->sequence) | /* Action if the sequence error event is triggered */
|
|
CDOG_CONTROL_STATE_CTRL(conf->state) | /* Action if the state error event is triggered */
|
|
CDOG_CONTROL_ADDRESS_CTRL(conf->address) | /* Action if the address error event is triggered */
|
|
CDOG_CONTROL_IRQ_PAUSE(conf->irq_pause) | /* Pause running during interrupts setup */
|
|
CDOG_CONTROL_DEBUG_HALT_CTRL(
|
|
conf->debug_halt) | /* Halt CDOG timer during debug so we have chance to debug code */
|
|
CDOG_CONTROL_LOCK_CTRL(conf->lock); /* Lock control register */
|
|
|
|
NVIC_EnableIRQ(CDOG_IRQn);
|
|
|
|
return kStatus_Success;
|
|
}
|
|
|
|
/*!
|
|
* brief Deinitialize CDOG
|
|
*
|
|
* This function stops CDOG secure counter.
|
|
*
|
|
* param base CDOG peripheral base address
|
|
*/
|
|
void CDOG_Deinit(CDOG_Type *base)
|
|
{
|
|
NVIC_DisableIRQ(CDOG_IRQn);
|
|
|
|
#if !(defined(FSL_FEATURE_CDOG_HAS_NO_RESET) && FSL_FEATURE_CDOG_HAS_NO_RESET)
|
|
RESET_SetPeripheralReset(kCDOG_RST_SHIFT_RSTn);
|
|
#endif /* !FSL_FEATURE_CDOG_HAS_NO_RESET */
|
|
|
|
#if !(defined(FSL_SDK_DISABLE_DRIVER_CLOCK_CONTROL) && FSL_SDK_DISABLE_DRIVER_CLOCK_CONTROL)
|
|
#ifdef CDOG_CLOCKS
|
|
CLOCK_DisableClock(kCLOCK_Cdog);
|
|
#endif /* CDOG_CLOCKS */
|
|
#endif /* !FSL_SDK_DISABLE_DRIVER_CLOCK_CONTROL */
|
|
}
|