4 * \brief Chip-specific system clock management functions.
\r
6 * Copyright (c) 2011-2012 Atmel Corporation. All rights reserved.
\r
12 * Redistribution and use in source and binary forms, with or without
\r
13 * modification, are permitted provided that the following conditions are met:
\r
15 * 1. Redistributions of source code must retain the above copyright notice,
\r
16 * this list of conditions and the following disclaimer.
\r
18 * 2. Redistributions in binary form must reproduce the above copyright notice,
\r
19 * this list of conditions and the following disclaimer in the documentation
\r
20 * and/or other materials provided with the distribution.
\r
22 * 3. The name of Atmel may not be used to endorse or promote products derived
\r
23 * from this software without specific prior written permission.
\r
25 * 4. This software may only be redistributed and used in connection with an
\r
26 * Atmel microcontroller product.
\r
28 * THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR IMPLIED
\r
29 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
\r
30 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
\r
31 * EXPRESSLY AND SPECIFICALLY DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR
\r
32 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
\r
33 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
\r
34 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
\r
35 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
\r
36 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
\r
37 * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
\r
38 * POSSIBILITY OF SUCH DAMAGE.
\r
55 * \weakgroup sysclk_group
\r
59 #if defined(CONFIG_SYSCLK_DEFAULT_RETURNS_SLOW_OSC)
\r
61 * \brief boolean signalling that the sysclk_init is done.
\r
63 uint32_t sysclk_initialized = 0;
\r
67 * \brief Set system clock prescaler configuration
\r
69 * This function will change the system clock prescaler configuration to
\r
70 * match the parameters.
\r
72 * \note The parameters to this function are device-specific.
\r
74 * \param cpu_shift The CPU clock will be divided by \f$2^{mck\_pres}\f$
\r
76 void sysclk_set_prescalers(uint32_t ul_pres)
\r
78 pmc_mck_set_prescaler(ul_pres);
\r
79 SystemCoreClockUpdate();
\r
83 * \brief Change the source of the main system clock.
\r
85 * \param src The new system clock source. Must be one of the constants
\r
86 * from the <em>System Clock Sources</em> section.
\r
88 void sysclk_set_source(uint32_t ul_src)
\r
91 case SYSCLK_SRC_SLCK_RC:
\r
92 case SYSCLK_SRC_SLCK_XTAL:
\r
93 case SYSCLK_SRC_SLCK_BYPASS:
\r
94 pmc_mck_set_source(PMC_MCKR_CSS_SLOW_CLK);
\r
97 case SYSCLK_SRC_MAINCK_4M_RC:
\r
98 case SYSCLK_SRC_MAINCK_8M_RC:
\r
99 case SYSCLK_SRC_MAINCK_12M_RC:
\r
100 case SYSCLK_SRC_MAINCK_XTAL:
\r
101 case SYSCLK_SRC_MAINCK_BYPASS:
\r
102 pmc_mck_set_source(PMC_MCKR_CSS_MAIN_CLK);
\r
105 case SYSCLK_SRC_PLLACK:
\r
106 pmc_mck_set_source(PMC_MCKR_CSS_PLLA_CLK);
\r
109 case SYSCLK_SRC_PLLBCK:
\r
110 pmc_mck_set_source(PMC_MCKR_CSS_PLLB_CLK);
\r
114 SystemCoreClockUpdate();
\r
117 #if defined(CONFIG_USBCLK_SOURCE) || defined(__DOXYGEN__)
\r
119 * \brief Enable USB clock.
\r
121 * \note The SAM3S UDP hardware interprets div as div+1. For readability the hardware div+1
\r
122 * is hidden in this implementation. Use div as div effective value.
\r
124 * \param pll_id Source of the USB clock.
\r
125 * \param div Actual clock divisor. Must be superior to 0.
\r
127 void sysclk_enable_usb(void)
\r
129 Assert(CONFIG_USBCLK_DIV > 0);
\r
131 switch (CONFIG_USBCLK_SOURCE) {
\r
132 #ifdef CONFIG_PLL0_SOURCE
\r
133 case USBCLK_SRC_PLL0: {
\r
134 struct pll_config pllcfg;
\r
136 pll_enable_source(CONFIG_PLL0_SOURCE);
\r
137 pll_config_defaults(&pllcfg, 0);
\r
138 pll_enable(&pllcfg, 0);
\r
139 pll_wait_for_lock(0);
\r
140 pmc_switch_udpck_to_pllack(CONFIG_USBCLK_DIV - 1);
\r
141 pmc_enable_udpck();
\r
146 #ifdef CONFIG_PLL1_SOURCE
\r
147 case USBCLK_SRC_PLL1: {
\r
148 struct pll_config pllcfg;
\r
150 pll_enable_source(CONFIG_PLL1_SOURCE);
\r
151 pll_config_defaults(&pllcfg, 1);
\r
152 pll_enable(&pllcfg, 1);
\r
153 pll_wait_for_lock(1);
\r
154 pmc_switch_udpck_to_pllbck(CONFIG_USBCLK_DIV - 1);
\r
155 pmc_enable_udpck();
\r
163 * \brief Disable the USB clock.
\r
165 * \note This implementation does not switch off the PLL, it just turns off the USB clock.
\r
167 void sysclk_disable_usb(void)
\r
169 pmc_disable_udpck();
\r
171 #endif // CONFIG_USBCLK_SOURCE
\r
173 void sysclk_init(void)
\r
175 struct pll_config pllcfg;
\r
177 /* Set a flash wait state depending on the new cpu frequency */
\r
178 system_init_flash(sysclk_get_cpu_hz());
\r
180 /* Config system clock setting */
\r
181 switch (CONFIG_SYSCLK_SOURCE) {
\r
182 case SYSCLK_SRC_SLCK_RC:
\r
183 osc_enable(OSC_SLCK_32K_RC);
\r
184 osc_wait_ready(OSC_SLCK_32K_RC);
\r
185 pmc_switch_mck_to_sclk(CONFIG_SYSCLK_PRES);
\r
188 case SYSCLK_SRC_SLCK_XTAL:
\r
189 osc_enable(OSC_SLCK_32K_XTAL);
\r
190 osc_wait_ready(OSC_SLCK_32K_XTAL);
\r
191 pmc_switch_mck_to_sclk(CONFIG_SYSCLK_PRES);
\r
194 case SYSCLK_SRC_SLCK_BYPASS:
\r
195 osc_enable(OSC_SLCK_32K_BYPASS);
\r
196 osc_wait_ready(OSC_SLCK_32K_BYPASS);
\r
197 pmc_switch_mck_to_sclk(CONFIG_SYSCLK_PRES);
\r
200 case SYSCLK_SRC_MAINCK_4M_RC:
\r
201 /* Already running from SYSCLK_SRC_MAINCK_4M_RC */
\r
204 case SYSCLK_SRC_MAINCK_8M_RC:
\r
205 osc_enable(OSC_MAINCK_8M_RC);
\r
206 osc_wait_ready(OSC_MAINCK_8M_RC);
\r
207 pmc_switch_mck_to_mainck(CONFIG_SYSCLK_PRES);
\r
210 case SYSCLK_SRC_MAINCK_12M_RC:
\r
211 osc_enable(OSC_MAINCK_12M_RC);
\r
212 osc_wait_ready(OSC_MAINCK_12M_RC);
\r
213 pmc_switch_mck_to_mainck(CONFIG_SYSCLK_PRES);
\r
217 case SYSCLK_SRC_MAINCK_XTAL:
\r
218 osc_enable(OSC_MAINCK_XTAL);
\r
219 osc_wait_ready(OSC_MAINCK_XTAL);
\r
220 pmc_switch_mck_to_mainck(CONFIG_SYSCLK_PRES);
\r
223 case SYSCLK_SRC_MAINCK_BYPASS:
\r
224 osc_enable(OSC_MAINCK_BYPASS);
\r
225 osc_wait_ready(OSC_MAINCK_BYPASS);
\r
226 pmc_switch_mck_to_mainck(CONFIG_SYSCLK_PRES);
\r
229 #ifdef CONFIG_PLL0_SOURCE
\r
230 case SYSCLK_SRC_PLLACK:
\r
231 pll_enable_source(CONFIG_PLL0_SOURCE);
\r
232 pll_config_defaults(&pllcfg, 0);
\r
233 pll_enable(&pllcfg, 0);
\r
234 pll_wait_for_lock(0);
\r
235 pmc_switch_mck_to_pllack(CONFIG_SYSCLK_PRES);
\r
239 #ifdef CONFIG_PLL1_SOURCE
\r
240 case SYSCLK_SRC_PLLBCK:
\r
241 pll_enable_source(CONFIG_PLL1_SOURCE);
\r
242 pll_config_defaults(&pllcfg, 1);
\r
243 pll_enable(&pllcfg, 1);
\r
244 pll_wait_for_lock(1);
\r
245 pmc_switch_mck_to_pllbck(CONFIG_SYSCLK_PRES);
\r
250 /* Update the SystemFrequency variable */
\r
251 SystemCoreClockUpdate();
\r
253 #if (defined CONFIG_SYSCLK_DEFAULT_RETURNS_SLOW_OSC)
\r
254 /* Signal that the internal frequencies are setup */
\r
255 sysclk_initialized = 1;
\r