mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-22 09:39:20 +08:00
341 lines
9.6 KiB
C
341 lines
9.6 KiB
C
|
/*
|
||
|
* @brief LPC5410X Multi-Rate Timer (MRT) registers and driver functions
|
||
|
*
|
||
|
* @note
|
||
|
* Copyright(C) NXP Semiconductors, 2014
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* @par
|
||
|
* Software that is described herein is for illustrative purposes only
|
||
|
* which provides customers with programming information regarding the
|
||
|
* LPC products. This software is supplied "AS IS" without any warranties of
|
||
|
* any kind, and NXP Semiconductors and its licensor disclaim any and
|
||
|
* all warranties, express or implied, including all implied warranties of
|
||
|
* merchantability, fitness for a particular purpose and non-infringement of
|
||
|
* intellectual property rights. NXP Semiconductors assumes no responsibility
|
||
|
* or liability for the use of the software, conveys no license or rights under any
|
||
|
* patent, copyright, mask work right, or any other intellectual property rights in
|
||
|
* or to any products. NXP Semiconductors reserves the right to make changes
|
||
|
* in the software without notification. NXP Semiconductors also makes no
|
||
|
* representation or warranty that such application will be suitable for the
|
||
|
* specified use without further testing or modification.
|
||
|
*
|
||
|
* @par
|
||
|
* Permission to use, copy, modify, and distribute this software and its
|
||
|
* documentation is hereby granted, under NXP Semiconductors' and its
|
||
|
* licensor's relevant copyrights 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.
|
||
|
*/
|
||
|
|
||
|
#ifndef __MRT_5410X_H_
|
||
|
#define __MRT_5410X_H_
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
extern "C" {
|
||
|
#endif
|
||
|
|
||
|
/** @defgroup MRT_5410X CHIP: LPC5410X Multi-Rate Timer driver
|
||
|
* @ingroup CHIP_5410X_DRIVERS
|
||
|
* @{
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* @brief LPC5410X MRT chip configuration
|
||
|
*/
|
||
|
#define MRT_CHANNELS_NUM (4)
|
||
|
#define MRT_NO_IDLE_CHANNEL (0x40)
|
||
|
|
||
|
/**
|
||
|
* @brief MRT register block structure
|
||
|
*/
|
||
|
typedef struct {
|
||
|
__IO uint32_t INTVAL; /*!< Timer interval register */
|
||
|
__O uint32_t TIMER; /*!< Timer register */
|
||
|
__IO uint32_t CTRL; /*!< Timer control register */
|
||
|
__IO uint32_t STAT; /*!< Timer status register */
|
||
|
} LPC_MRT_CH_T;
|
||
|
|
||
|
/**
|
||
|
* @brief MRT register block structure
|
||
|
*/
|
||
|
typedef struct {
|
||
|
LPC_MRT_CH_T CHANNEL[MRT_CHANNELS_NUM];
|
||
|
uint32_t unused[44];
|
||
|
__IO uint32_t MODCFG;
|
||
|
__O uint32_t IDLE_CH;
|
||
|
__IO uint32_t IRQ_FLAG;
|
||
|
} LPC_MRT_T;
|
||
|
|
||
|
/**
|
||
|
* @brief MRT Interrupt Modes enum
|
||
|
*/
|
||
|
typedef enum MRT_MODE {
|
||
|
MRT_MODE_REPEAT = (0 << 1), /*!< MRT Repeat interrupt mode */
|
||
|
MRT_MODE_ONESHOT = (1 << 1) /*!< MRT One-shot interrupt mode */
|
||
|
} MRT_MODE_T;
|
||
|
|
||
|
/**
|
||
|
* @brief MRT register bit fields & masks
|
||
|
*/
|
||
|
/* MRT Time interval register bit fields */
|
||
|
#define MRT_INTVAL_IVALUE (0x7FFFFFFF) /* Maximum interval load value and mask */
|
||
|
#define MRT_INTVAL_LOAD (0x80000000UL) /* Force immediate load of timer interval register bit */
|
||
|
|
||
|
/* MRT Control register bit fields & masks */
|
||
|
#define MRT_CTRL_INTEN_MASK (0x01)
|
||
|
#define MRT_CTRL_MODE_MASK (0x06)
|
||
|
|
||
|
/* MRT Status register bit fields & masks */
|
||
|
#define MRT_STAT_INTFLAG (0x01)
|
||
|
#define MRT_STAT_RUNNING (0x02)
|
||
|
|
||
|
/* Pointer to individual MR register blocks */
|
||
|
#define LPC_MRT_CH0 ((LPC_MRT_CH_T *) &LPC_MRT->CHANNEL[0])
|
||
|
#define LPC_MRT_CH1 ((LPC_MRT_CH_T *) &LPC_MRT->CHANNEL[1])
|
||
|
#define LPC_MRT_CH2 ((LPC_MRT_CH_T *) &LPC_MRT->CHANNEL[2])
|
||
|
#define LPC_MRT_CH3 ((LPC_MRT_CH_T *) &LPC_MRT->CHANNEL[3])
|
||
|
#define LPC_MRT_CH(ch) ((LPC_MRT_CH_T *) &LPC_MRT->CHANNEL[(ch)])
|
||
|
|
||
|
/* Global interrupt flag register interrupt mask/clear values */
|
||
|
#define MRT0_INTFLAG (1)
|
||
|
#define MRT1_INTFLAG (2)
|
||
|
#define MRT2_INTFLAG (4)
|
||
|
#define MRT3_INTFLAG (8)
|
||
|
#define MRTn_INTFLAG(ch) (1 << (ch))
|
||
|
|
||
|
/**
|
||
|
* @brief Initializes the MRT
|
||
|
* @return Nothing
|
||
|
*/
|
||
|
STATIC INLINE void Chip_MRT_Init(void)
|
||
|
{
|
||
|
/* Enable the clock to the register interface */
|
||
|
Chip_Clock_EnablePeriphClock(SYSCON_CLOCK_MRT);
|
||
|
|
||
|
/* Reset MRT */
|
||
|
Chip_SYSCON_PeriphReset(RESET_MRT);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief De-initializes the MRT Channel
|
||
|
* @return Nothing
|
||
|
*/
|
||
|
STATIC INLINE void Chip_MRT_DeInit(void)
|
||
|
{
|
||
|
/* Disable the clock to the MRT */
|
||
|
Chip_Clock_DisablePeriphClock(SYSCON_CLOCK_MRT);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Returns a pointer to the register block for a MRT channel
|
||
|
* @param ch : MRT channel tog et register block for (0..3)
|
||
|
* @return Pointer to the MRT register block for the channel
|
||
|
*/
|
||
|
STATIC INLINE LPC_MRT_CH_T *Chip_MRT_GetRegPtr(uint8_t ch)
|
||
|
{
|
||
|
return LPC_MRT_CH(ch);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Returns the timer time interval value
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return Timer time interval value (IVALUE)
|
||
|
*/
|
||
|
STATIC INLINE uint32_t Chip_MRT_GetInterval(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
return pMRT->INTVAL;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Sets the timer time interval value
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @param interval : The interval timeout (31-bits)
|
||
|
* @return Nothing
|
||
|
* @note Setting bit 31 in timer time interval register causes the time interval value
|
||
|
* to load immediately, otherwise the time interval value will be loaded in
|
||
|
* next timer cycle.<br>
|
||
|
* Example: Chip_MRT_SetInterval(pMRT, 0x500 | MRT_INTVAL_LOAD); // Will load timer interval immediately<br>
|
||
|
* Example: Chip_MRT_SetInterval(pMRT, 0x500); // Will load timer interval after internal expires
|
||
|
*/
|
||
|
STATIC INLINE void Chip_MRT_SetInterval(LPC_MRT_CH_T *pMRT, uint32_t interval)
|
||
|
{
|
||
|
pMRT->INTVAL = interval;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Returns the current timer value
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return The current timer value
|
||
|
*/
|
||
|
STATIC INLINE uint32_t Chip_MRT_GetTimer(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
return pMRT->TIMER;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Returns true if the timer is enabled
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return True if enabled, Flase if not enabled
|
||
|
*/
|
||
|
STATIC INLINE bool Chip_MRT_GetEnabled(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
return (bool) ((pMRT->CTRL & MRT_CTRL_INTEN_MASK) != 0);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Enables the timer
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return Nothing
|
||
|
*/
|
||
|
STATIC INLINE void Chip_MRT_SetEnabled(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
pMRT->CTRL |= MRT_CTRL_INTEN_MASK;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Disables the timer
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return Nothing
|
||
|
*/
|
||
|
STATIC INLINE void Chip_MRT_SetDisabled(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
pMRT->CTRL &= ~MRT_CTRL_INTEN_MASK;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Returns the timer mode (repeat or one-shot)
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return The current timer mode
|
||
|
*/
|
||
|
STATIC INLINE MRT_MODE_T Chip_MRT_GetMode(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
return (MRT_MODE_T) (pMRT->CTRL & MRT_CTRL_MODE_MASK);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Sets the timer mode (repeat or one-shot)
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @param mode : Timer mode
|
||
|
* @return Nothing
|
||
|
*/
|
||
|
STATIC INLINE void Chip_MRT_SetMode(LPC_MRT_CH_T *pMRT, MRT_MODE_T mode)
|
||
|
{
|
||
|
uint32_t reg;
|
||
|
|
||
|
reg = pMRT->CTRL & ~MRT_CTRL_MODE_MASK;
|
||
|
pMRT->CTRL = reg | (uint32_t) mode;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Check if the timer is configured in repeat mode
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return True if in repeat mode, False if in one-shot mode
|
||
|
*/
|
||
|
STATIC INLINE bool Chip_MRT_IsRepeatMode(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
return ((pMRT->CTRL & MRT_CTRL_MODE_MASK) != 0) ? false : true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Check if the timer is configured in one-shot mode
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return True if in one-shot mode, False if in repeat mode
|
||
|
*/
|
||
|
STATIC INLINE bool Chip_MRT_IsOneShotMode(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
return ((pMRT->CTRL & MRT_CTRL_MODE_MASK) != 0) ? true : false;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Check if the timer has an interrupt pending
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return True if interrupt is pending, False if no interrupt is pending
|
||
|
*/
|
||
|
STATIC INLINE bool Chip_MRT_IntPending(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
return (bool) ((pMRT->STAT & MRT_STAT_INTFLAG) != 0);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Clears the pending interrupt (if any)
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return Nothing
|
||
|
*/
|
||
|
STATIC INLINE void Chip_MRT_IntClear(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
pMRT->STAT |= MRT_STAT_INTFLAG;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Check if the timer is running
|
||
|
* @param pMRT : Pointer to selected MRT Channel
|
||
|
* @return True if running, False if stopped
|
||
|
*/
|
||
|
STATIC INLINE bool Chip_MRT_Running(LPC_MRT_CH_T *pMRT)
|
||
|
{
|
||
|
return (bool) ((pMRT->STAT & MRT_STAT_RUNNING) != 0);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Returns the IDLE channel value
|
||
|
* @return IDLE channel value (unshifted in bits 7..4)
|
||
|
*/
|
||
|
STATIC INLINE uint8_t Chip_MRT_GetIdleChannel(void)
|
||
|
{
|
||
|
return (uint8_t) (LPC_MRT->IDLE_CH);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Returns the IDLE channel value
|
||
|
* @return IDLE channel value (shifted in bits 3..0)
|
||
|
*/
|
||
|
STATIC INLINE uint8_t Chip_MRT_GetIdleChannelShifted(void)
|
||
|
{
|
||
|
return (uint8_t) (Chip_MRT_GetIdleChannel() >> 4);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Returns the interrupt pending status for all MRT channels
|
||
|
* @return IRQ pending channel bitfield(bit 0 = MRT0, bit 1 = MRT1, etc.)
|
||
|
*/
|
||
|
STATIC INLINE uint32_t Chip_MRT_GetIntPending(void)
|
||
|
{
|
||
|
return LPC_MRT->IRQ_FLAG;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Returns the interrupt pending status for a singel MRT channel
|
||
|
* @param ch : Channel to check pending interrupt status for
|
||
|
* @return IRQ pending channel number
|
||
|
*/
|
||
|
STATIC INLINE bool Chip_MRT_GetIntPendingByChannel(uint8_t ch)
|
||
|
{
|
||
|
return (bool) (((LPC_MRT->IRQ_FLAG >> ch) & 1) != 0);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Clears the interrupt pending status for one or more MRT channels
|
||
|
* @param mask : Channels to clear (bit 0 = MRT0, bit 1 = MRT1, etc.)
|
||
|
* @return Nothing
|
||
|
* @note Use this function to clear multiple interrupt pending states in
|
||
|
* a single call via the IRQ_FLAG register. Performs the same function for
|
||
|
* all MRT channels in a single call as the Chip_MRT_IntClear() does for a
|
||
|
* single channel.
|
||
|
*/
|
||
|
STATIC INLINE void Chip_MRT_ClearIntPending(uint32_t mask)
|
||
|
{
|
||
|
LPC_MRT->IRQ_FLAG = mask;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @}
|
||
|
*/
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#endif /* __MRT_5410X_H_ */
|