mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-23 06:17:21 +08:00
168 lines
5.2 KiB
C
168 lines
5.2 KiB
C
|
/*
|
||
|
* @brief LPC5410X System & Control driver
|
||
|
*
|
||
|
* @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 licenser 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.
|
||
|
*/
|
||
|
|
||
|
#include "chip.h"
|
||
|
|
||
|
/*****************************************************************************
|
||
|
* Private types/enumerations/variables
|
||
|
****************************************************************************/
|
||
|
|
||
|
/*****************************************************************************
|
||
|
* Public types/enumerations/variables
|
||
|
****************************************************************************/
|
||
|
|
||
|
/*****************************************************************************
|
||
|
* Private functions
|
||
|
****************************************************************************/
|
||
|
|
||
|
/*****************************************************************************
|
||
|
* Public functions
|
||
|
****************************************************************************/
|
||
|
|
||
|
/* Set source for non-maskable interrupt (NMI) */
|
||
|
void Chip_SYSCON_SetNMISource(uint32_t intsrc)
|
||
|
{
|
||
|
uint32_t reg;
|
||
|
|
||
|
reg = LPC_SYSCON->NMISRC;
|
||
|
#if defined(CORE_M4)
|
||
|
reg &= ~SYSCON_NMISRC_M4_ENABLE;
|
||
|
#else
|
||
|
reg &= ~SYSCON_NMISRC_M0_ENABLE;
|
||
|
intsrc = (intsrc << 8);
|
||
|
#endif
|
||
|
|
||
|
/* First write without NMI bit, and then write source */
|
||
|
LPC_SYSCON->NMISRC = reg;
|
||
|
LPC_SYSCON->NMISRC = reg | intsrc;
|
||
|
}
|
||
|
|
||
|
/* Enable interrupt used for NMI source */
|
||
|
void Chip_SYSCON_EnableNMISource(void)
|
||
|
{
|
||
|
#if defined(CORE_M4)
|
||
|
LPC_SYSCON->NMISRC |= SYSCON_NMISRC_M4_ENABLE;
|
||
|
#else
|
||
|
LPC_SYSCON->NMISRC |= SYSCON_NMISRC_M0_ENABLE;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
/* Disable interrupt used for NMI source */
|
||
|
void Chip_SYSCON_DisableNMISource(void)
|
||
|
{
|
||
|
#if defined(CORE_M4)
|
||
|
LPC_SYSCON->NMISRC &= ~SYSCON_NMISRC_M4_ENABLE;
|
||
|
#else
|
||
|
LPC_SYSCON->NMISRC &= ~SYSCON_NMISRC_M0_ENABLE;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
/* Enable or disable asynchronous APB bridge and subsystem */
|
||
|
void Chip_SYSCON_Enable_ASYNC_Syscon(bool enable)
|
||
|
{
|
||
|
if (enable) {
|
||
|
LPC_SYSCON->ASYNCAPBCTRL = 0x01;
|
||
|
}
|
||
|
else {
|
||
|
LPC_SYSCON->ASYNCAPBCTRL = 0x00;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Resets a peripheral */
|
||
|
void Chip_SYSCON_PeriphReset(CHIP_SYSCON_PERIPH_RESET_T periph)
|
||
|
{
|
||
|
uint32_t pid = (uint32_t) periph;
|
||
|
|
||
|
if (pid >= 128) {
|
||
|
/* Async resets mapped to 128 and above, offset for peripheral bit index */
|
||
|
pid = 1 << (((uint32_t) periph) - 128);
|
||
|
LPC_ASYNC_SYSCON->ASYNCPRESETCTRLSET = pid;
|
||
|
LPC_ASYNC_SYSCON->ASYNCPRESETCTRLCLR = pid;
|
||
|
}
|
||
|
else if (periph >= 32) {
|
||
|
pid = 1 << (((uint32_t) periph) - 32);
|
||
|
LPC_SYSCON->PRESETCTRLSET[1] = pid;
|
||
|
LPC_SYSCON->PRESETCTRLCLR[1] = pid;
|
||
|
}
|
||
|
else {
|
||
|
pid = 1 << ((uint32_t) periph);
|
||
|
LPC_SYSCON->PRESETCTRLSET[0] = pid;
|
||
|
LPC_SYSCON->PRESETCTRLCLR[0] = pid;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Returns the computed value for a frequency measurement cycle */
|
||
|
uint32_t Chip_SYSCON_GetCompFreqMeas(uint32_t refClockRate)
|
||
|
{
|
||
|
uint32_t capval;
|
||
|
uint64_t clkrate = 0;
|
||
|
|
||
|
/* Get raw capture value */
|
||
|
capval = Chip_SYSCON_GetRawFreqMeasCapval();
|
||
|
|
||
|
/* Limit CAPVAL check */
|
||
|
if (capval > 2) {
|
||
|
clkrate = (((uint64_t) capval - 2) * (uint64_t) refClockRate) / 0x4000;
|
||
|
}
|
||
|
|
||
|
return (uint32_t) clkrate;
|
||
|
}
|
||
|
|
||
|
void Chip_SYSCON_PowerUp(uint32_t powerupmask)
|
||
|
{
|
||
|
/* If turning the PLL back on, perform the following sequence to accelerate PLL lock */
|
||
|
if (powerupmask & SYSCON_PDRUNCFG_PD_SYS_PLL) {
|
||
|
volatile uint32_t delayX;
|
||
|
uint32_t maxCCO = (1 << 18) | 0x3fff;
|
||
|
uint32_t curSSCTRL = LPC_SYSCON->SYSPLLSSCTRL[0];
|
||
|
|
||
|
/* If NOT using spread spectrum mode */
|
||
|
if (curSSCTRL & (1 << 18)) {
|
||
|
|
||
|
/* Turn on PLL */
|
||
|
LPC_SYSCON->PDRUNCFGCLR = SYSCON_PDRUNCFG_PD_SYS_PLL;
|
||
|
|
||
|
/* this sequence acclerates the PLL lock time */
|
||
|
LPC_SYSCON->SYSPLLSSCTRL[0] = maxCCO | (1 << 17); /* Set mreq to activate */
|
||
|
LPC_SYSCON->SYSPLLSSCTRL[0] = maxCCO; /* clear mreq to prepare for restoring mreq */
|
||
|
|
||
|
/* Delay for 20 uSec @ 12Mhz*/
|
||
|
for (delayX = 0; delayX < 48; ++delayX) {}
|
||
|
|
||
|
/* set original value back with mreq */
|
||
|
LPC_SYSCON->SYSPLLSSCTRL[0] = curSSCTRL | (1 << 17);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Enable peripheral states by setting low */
|
||
|
LPC_SYSCON->PDRUNCFGCLR = powerupmask;
|
||
|
}
|