/* * Copyright (C) 2008-2009, Freie Universitaet Berlin * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup cpu_lpc2387 * @{ * @file * @brief LPC2387 clock initialization * * @author Heiko Will * @author Kaspar Schleiser * @author Michael Baar * @author Zakaria Kasmi * @note $Id$ * * @} */ #include #include "board.h" #include "cpu.h" #include "lpc2387.h" #define CL_CPU_DIV 4 #define WD_INTERVAL 10 /**< number of seconds before WD triggers */ /** * @brief Feed sequence for PLL register */ static inline void pllfeed(void) { PLLFEED = 0xAA; PLLFEED = 0x55; } /** * @brief Enabling MAM and setting number of clocks used for Flash memory fetch * @internal */ static inline void init_mam(void) { MAMCR = 0x0000; MAMTIM = 0x0003; MAMCR = 0x0002; } static void init_clks1(void) { /* Disconnect PLL */ PLLCON &= ~0x0002; pllfeed(); while (PLLSTAT & BIT25); /* wait until PLL is disconnected before * disabling - deadlock otherwise */ /* Disable PLL */ PLLCON &= ~0x0001; pllfeed(); while (PLLSTAT & BIT24); /* wait until PLL is disabled */ SCS |= 0x10; /* main OSC between 15MHz and 24MHz (more stable in tests) */ SCS |= 0x20; /* Enable main OSC */ while (!(SCS & 0x40)); /* Wait until main OSC is usable */ /* select main OSC, 16MHz, as the PLL clock source */ CLKSRCSEL = 0x0001; /* Setting Multiplier and Divider values */ PLLCFG = 0x0008; /* M=9 N=1 Fcco = 288 MHz */ pllfeed(); /* Enabling the PLL */ PLLCON = 0x0001; pllfeed(); /* Set clock divider to 4 (value+1) */ CCLKCFG = CL_CPU_DIV - 1; /* Fcpu = 72 MHz */ } static void init_clks2(void) { /* Wait for the PLL to lock to set frequency */ while (!(PLLSTAT & BIT26)); /* Connect the PLL as the clock source */ PLLCON = 0x0003; pllfeed(); /* Check connect bit status */ while (!(PLLSTAT & BIT25)); } static void watchdog_init(void) { WDCLKSEL = 0; /* clock source: RC oscillator */ WDMOD &= ~WDTOF; /* clear time-out flag */ WDTC = (F_RC_OSCILLATOR / 4) * WD_INTERVAL; } void cpu_init_clks(void) { watchdog_init(); PCONP = PCRTC; /* switch off everything except RTC */ init_clks1(); init_clks2(); init_mam(); }