| /* |
| * Copyright (C) 2009 Samsung Electronics |
| * Minkyu Kang <mk7.kang@samsung.com> |
| * Heungjun Kim <riverful.kim@samsung.com> |
| * |
| * See file CREDITS for list of people who contributed to this |
| * project. |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public License as |
| * published by the Free Software Foundation; either version 2 of |
| * the License, or (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, |
| * MA 02111-1307 USA |
| */ |
| |
| #include <common.h> |
| #include <asm/io.h> |
| #include <asm/arch/clock.h> |
| #include <asm/arch/clk.h> |
| |
| #define CLK_M 0 |
| #define CLK_D 1 |
| #define CLK_P 2 |
| |
| #ifndef CONFIG_SYS_CLK_FREQ_C100 |
| #define CONFIG_SYS_CLK_FREQ_C100 12000000 |
| #endif |
| #ifndef CONFIG_SYS_CLK_FREQ_C110 |
| #define CONFIG_SYS_CLK_FREQ_C110 24000000 |
| #endif |
| |
| unsigned long (*get_pclk)(void); |
| unsigned long (*get_arm_clk)(void); |
| unsigned long (*get_pll_clk)(int); |
| |
| /* s5pc110: return pll clock frequency */ |
| static unsigned long s5pc100_get_pll_clk(int pllreg) |
| { |
| struct s5pc100_clock *clk = (struct s5pc100_clock *)S5PC1XX_CLOCK_BASE; |
| unsigned long r, m, p, s, mask, fout; |
| unsigned int freq; |
| |
| switch (pllreg) { |
| case APLL: |
| r = readl(&clk->apll_con); |
| break; |
| case MPLL: |
| r = readl(&clk->mpll_con); |
| break; |
| case EPLL: |
| r = readl(&clk->epll_con); |
| break; |
| case HPLL: |
| r = readl(&clk->hpll_con); |
| break; |
| default: |
| printf("Unsupported PLL (%d)\n", pllreg); |
| return 0; |
| } |
| |
| /* |
| * APLL_CON: MIDV [25:16] |
| * MPLL_CON: MIDV [23:16] |
| * EPLL_CON: MIDV [23:16] |
| * HPLL_CON: MIDV [23:16] |
| */ |
| if (pllreg == APLL) |
| mask = 0x3ff; |
| else |
| mask = 0x0ff; |
| |
| m = (r >> 16) & mask; |
| |
| /* PDIV [13:8] */ |
| p = (r >> 8) & 0x3f; |
| /* SDIV [2:0] */ |
| s = r & 0x7; |
| |
| /* FOUT = MDIV * FIN / (PDIV * 2^SDIV) */ |
| freq = CONFIG_SYS_CLK_FREQ_C100; |
| fout = m * (freq / (p * (1 << s))); |
| |
| return fout; |
| } |
| |
| /* s5pc100: return pll clock frequency */ |
| static unsigned long s5pc110_get_pll_clk(int pllreg) |
| { |
| struct s5pc110_clock *clk = (struct s5pc110_clock *)S5PC1XX_CLOCK_BASE; |
| unsigned long r, m, p, s, mask, fout; |
| unsigned int freq; |
| |
| switch (pllreg) { |
| case APLL: |
| r = readl(&clk->apll_con); |
| break; |
| case MPLL: |
| r = readl(&clk->mpll_con); |
| break; |
| case EPLL: |
| r = readl(&clk->epll_con); |
| break; |
| case VPLL: |
| r = readl(&clk->vpll_con); |
| break; |
| default: |
| printf("Unsupported PLL (%d)\n", pllreg); |
| return 0; |
| } |
| |
| /* |
| * APLL_CON: MIDV [25:16] |
| * MPLL_CON: MIDV [25:16] |
| * EPLL_CON: MIDV [24:16] |
| * VPLL_CON: MIDV [24:16] |
| */ |
| if (pllreg == APLL || pllreg == MPLL) |
| mask = 0x3ff; |
| else |
| mask = 0x1ff; |
| |
| m = (r >> 16) & mask; |
| |
| /* PDIV [13:8] */ |
| p = (r >> 8) & 0x3f; |
| /* SDIV [2:0] */ |
| s = r & 0x7; |
| |
| freq = CONFIG_SYS_CLK_FREQ_C110; |
| if (pllreg == APLL) { |
| if (s < 1) |
| s = 1; |
| /* FOUT = MDIV * FIN / (PDIV * 2^(SDIV - 1)) */ |
| fout = m * (freq / (p * (1 << (s - 1)))); |
| } else |
| /* FOUT = MDIV * FIN / (PDIV * 2^SDIV) */ |
| fout = m * (freq / (p * (1 << s))); |
| |
| return fout; |
| } |
| |
| /* s5pc110: return ARM clock frequency */ |
| static unsigned long s5pc110_get_arm_clk(void) |
| { |
| struct s5pc110_clock *clk = (struct s5pc110_clock *)S5PC1XX_CLOCK_BASE; |
| unsigned long div; |
| unsigned long dout_apll, armclk; |
| unsigned int apll_ratio; |
| |
| div = readl(&clk->div0); |
| |
| /* APLL_RATIO: [2:0] */ |
| apll_ratio = div & 0x7; |
| |
| dout_apll = get_pll_clk(APLL) / (apll_ratio + 1); |
| armclk = dout_apll; |
| |
| return armclk; |
| } |
| |
| /* s5pc100: return ARM clock frequency */ |
| static unsigned long s5pc100_get_arm_clk(void) |
| { |
| struct s5pc100_clock *clk = (struct s5pc100_clock *)S5PC1XX_CLOCK_BASE; |
| unsigned long div; |
| unsigned long dout_apll, armclk; |
| unsigned int apll_ratio, arm_ratio; |
| |
| div = readl(&clk->div0); |
| |
| /* ARM_RATIO: [6:4] */ |
| arm_ratio = (div >> 4) & 0x7; |
| /* APLL_RATIO: [0] */ |
| apll_ratio = div & 0x1; |
| |
| dout_apll = get_pll_clk(APLL) / (apll_ratio + 1); |
| armclk = dout_apll / (arm_ratio + 1); |
| |
| return armclk; |
| } |
| |
| /* s5pc100: return HCLKD0 frequency */ |
| static unsigned long get_hclk(void) |
| { |
| struct s5pc100_clock *clk = (struct s5pc100_clock *)S5PC1XX_CLOCK_BASE; |
| unsigned long hclkd0; |
| uint div, d0_bus_ratio; |
| |
| div = readl(&clk->div0); |
| /* D0_BUS_RATIO: [10:8] */ |
| d0_bus_ratio = (div >> 8) & 0x7; |
| |
| hclkd0 = get_arm_clk() / (d0_bus_ratio + 1); |
| |
| return hclkd0; |
| } |
| |
| /* s5pc100: return PCLKD1 frequency */ |
| static unsigned long get_pclkd1(void) |
| { |
| struct s5pc100_clock *clk = (struct s5pc100_clock *)S5PC1XX_CLOCK_BASE; |
| unsigned long d1_bus, pclkd1; |
| uint div, d1_bus_ratio, pclkd1_ratio; |
| |
| div = readl(&clk->div0); |
| /* D1_BUS_RATIO: [14:12] */ |
| d1_bus_ratio = (div >> 12) & 0x7; |
| /* PCLKD1_RATIO: [18:16] */ |
| pclkd1_ratio = (div >> 16) & 0x7; |
| |
| /* ASYNC Mode */ |
| d1_bus = get_pll_clk(MPLL) / (d1_bus_ratio + 1); |
| pclkd1 = d1_bus / (pclkd1_ratio + 1); |
| |
| return pclkd1; |
| } |
| |
| /* s5pc110: return HCLKs frequency */ |
| static unsigned long get_hclk_sys(int dom) |
| { |
| struct s5pc110_clock *clk = (struct s5pc110_clock *)S5PC1XX_CLOCK_BASE; |
| unsigned long hclk; |
| unsigned int div; |
| unsigned int offset; |
| unsigned int hclk_sys_ratio; |
| |
| if (dom == CLK_M) |
| return get_hclk(); |
| |
| div = readl(&clk->div0); |
| |
| /* |
| * HCLK_MSYS_RATIO: [10:8] |
| * HCLK_DSYS_RATIO: [19:16] |
| * HCLK_PSYS_RATIO: [27:24] |
| */ |
| offset = 8 + (dom << 0x3); |
| |
| hclk_sys_ratio = (div >> offset) & 0xf; |
| |
| hclk = get_pll_clk(MPLL) / (hclk_sys_ratio + 1); |
| |
| return hclk; |
| } |
| |
| /* s5pc110: return PCLKs frequency */ |
| static unsigned long get_pclk_sys(int dom) |
| { |
| struct s5pc110_clock *clk = (struct s5pc110_clock *)S5PC1XX_CLOCK_BASE; |
| unsigned long pclk; |
| unsigned int div; |
| unsigned int offset; |
| unsigned int pclk_sys_ratio; |
| |
| div = readl(&clk->div0); |
| |
| /* |
| * PCLK_MSYS_RATIO: [14:12] |
| * PCLK_DSYS_RATIO: [22:20] |
| * PCLK_PSYS_RATIO: [30:28] |
| */ |
| offset = 12 + (dom << 0x3); |
| |
| pclk_sys_ratio = (div >> offset) & 0x7; |
| |
| pclk = get_hclk_sys(dom) / (pclk_sys_ratio + 1); |
| |
| return pclk; |
| } |
| |
| /* s5pc110: return peripheral clock frequency */ |
| static unsigned long s5pc110_get_pclk(void) |
| { |
| return get_pclk_sys(CLK_P); |
| } |
| |
| /* s5pc100: return peripheral clock frequency */ |
| static unsigned long s5pc100_get_pclk(void) |
| { |
| return get_pclkd1(); |
| } |
| |
| void s5pc1xx_clock_init(void) |
| { |
| if (cpu_is_s5pc110()) { |
| get_pll_clk = s5pc110_get_pll_clk; |
| get_arm_clk = s5pc110_get_arm_clk; |
| get_pclk = s5pc110_get_pclk; |
| } else { |
| get_pll_clk = s5pc100_get_pll_clk; |
| get_arm_clk = s5pc100_get_arm_clk; |
| get_pclk = s5pc100_get_pclk; |
| } |
| } |