summaryrefslogtreecommitdiffstats
path: root/c/src/lib/libbsp/arm/beagle/misc/system-clocks.c
diff options
context:
space:
mode:
Diffstat (limited to 'c/src/lib/libbsp/arm/beagle/misc/system-clocks.c')
-rw-r--r--c/src/lib/libbsp/arm/beagle/misc/system-clocks.c140
1 files changed, 140 insertions, 0 deletions
diff --git a/c/src/lib/libbsp/arm/beagle/misc/system-clocks.c b/c/src/lib/libbsp/arm/beagle/misc/system-clocks.c
new file mode 100644
index 0000000000..56b5d9aadb
--- /dev/null
+++ b/c/src/lib/libbsp/arm/beagle/misc/system-clocks.c
@@ -0,0 +1,140 @@
+/**
+ * @file
+ *
+ * @ingroup beagle
+ *
+ * @brief System clocks.
+ */
+
+/*
+ * Copyright (c) 2012 Claas Ziemke. All rights reserved.
+ *
+ * Claas Ziemke
+ * Kernerstrasse 11
+ * 70182 Stuttgart
+ * Germany
+ * <claas.ziemke@gmx.net>
+ *
+ * The license and distribution terms for this file may be
+ * found in the file LICENSE in this distribution or at
+ * http://www.rtems.com/license/LICENSE.
+ */
+
+#include <bsp.h>
+#include <bsp/beagle.h>
+
+uint32_t beagle_sysclk(void)
+{
+ uint32_t sysclk_ctrl = BEAGLE_SYSCLK_CTRL;
+
+ return (sysclk_ctrl & 0x1) == 0 ?
+ BEAGLE_OSCILLATOR_MAIN
+ : (397 * BEAGLE_OSCILLATOR_RTC);
+}
+
+uint32_t beagle_hclkpll_clk(void)
+{
+ uint32_t sysclk = beagle_sysclk();
+ uint32_t hclkpll_ctrl = BEAGLE_HCLKPLL_CTRL;
+ uint32_t m = HCLK_PLL_M_GET(hclkpll_ctrl) + 1;
+ uint32_t n = HCLK_PLL_N_GET(hclkpll_ctrl) + 1;
+ uint32_t p = 1U << HCLK_PLL_P_GET(hclkpll_ctrl);
+ uint32_t hclkpll_clk = 0;
+
+ if ((hclkpll_ctrl & HCLK_PLL_BYPASS) != 0) {
+ if ((hclkpll_ctrl & HCLK_PLL_DIRECT) != 0) {
+ hclkpll_clk = sysclk;
+ } else {
+ hclkpll_clk = sysclk / (2 * p);
+ }
+ } else {
+ if ((hclkpll_ctrl & HCLK_PLL_DIRECT) != 0) {
+ hclkpll_clk = (m * sysclk) / n;
+ } else {
+ if ((hclkpll_ctrl & HCLK_PLL_FBD_FCLKOUT) != 0) {
+ hclkpll_clk = m * (sysclk / n);
+ } else {
+ hclkpll_clk = (m / (2 * p)) * (sysclk / n);
+ }
+ }
+ }
+
+ return hclkpll_clk;
+}
+
+uint32_t beagle_periph_clk(void)
+{
+ uint32_t pwr_ctrl = BEAGLE_PWR_CTRL;
+ uint32_t periph_clk = 0;
+
+ if ((pwr_ctrl & PWR_NORMAL_RUN_MODE) != 0) {
+ uint32_t hclkdiv_ctrl = BEAGLE_HCLKDIV_CTRL;
+ uint32_t div = HCLK_DIV_PERIPH_CLK_GET(hclkdiv_ctrl) + 1;
+
+ periph_clk = beagle_hclkpll_clk() / div;
+ } else {
+ periph_clk = beagle_sysclk();
+ }
+
+ return periph_clk;
+}
+
+uint32_t beagle_hclk(void)
+{
+ uint32_t pwr_ctrl = BEAGLE_PWR_CTRL;
+ uint32_t hclk = 0;
+
+ if ((pwr_ctrl & PWR_HCLK_USES_PERIPH_CLK) != 0) {
+ hclk = beagle_periph_clk();
+ } else {
+ if ((pwr_ctrl & PWR_NORMAL_RUN_MODE) != 0) {
+ uint32_t hclkdiv_ctrl = BEAGLE_HCLKDIV_CTRL;
+ uint32_t div = 1U << HCLK_DIV_HCLK_GET(hclkdiv_ctrl);
+
+ hclk = beagle_hclkpll_clk() / div;
+ } else {
+ hclk = beagle_sysclk();
+ }
+ }
+
+ return hclk;
+}
+
+uint32_t beagle_arm_clk(void)
+{
+ uint32_t pwr_ctrl = BEAGLE_PWR_CTRL;
+ uint32_t arm_clk = 0;
+
+ if ((pwr_ctrl & PWR_HCLK_USES_PERIPH_CLK) != 0) {
+ arm_clk = beagle_periph_clk();
+ } else {
+ if ((pwr_ctrl & PWR_NORMAL_RUN_MODE) != 0) {
+ arm_clk = beagle_hclkpll_clk();
+ } else {
+ arm_clk = beagle_sysclk();
+ }
+ }
+
+ return arm_clk;
+}
+
+uint32_t beagle_dram_clk(void)
+{
+ uint32_t hclkdiv_ctrl = BEAGLE_HCLKDIV_CTRL;
+ uint32_t div = HCLK_DIV_DDRAM_CLK_GET(hclkdiv_ctrl);
+ uint32_t dram_clk = 0;
+
+ if (div != 0) {
+ uint32_t pwr_ctrl = BEAGLE_PWR_CTRL;
+
+ if ((pwr_ctrl & PWR_NORMAL_RUN_MODE) != 0) {
+ dram_clk = beagle_hclkpll_clk();
+ } else {
+ dram_clk = beagle_sysclk();
+ }
+
+ dram_clk /= div;
+ }
+
+ return dram_clk;
+}