/* linux/arch/arm/plat-s5pc1xx/s5pc100-clock.c * * Copyright 2009 Samsung Electronics, Co. * Byungho Min * * S5PC100 based common clock support * * Based on plat-s3c64xx/s3c6400-clock.c * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* fin_apll, fin_mpll and fin_epll are all the same clock, which we call * ext_xtal_mux for want of an actual name from the manual. */ static struct clk clk_ext_xtal_mux = { .name = "ext_xtal", .id = -1, }; #define clk_fin_apll clk_ext_xtal_mux #define clk_fin_mpll clk_ext_xtal_mux #define clk_fin_epll clk_ext_xtal_mux #define clk_fin_hpll clk_ext_xtal_mux #define clk_fout_mpll clk_mpll struct clk_sources { unsigned int nr_sources; struct clk **sources; }; struct clksrc_clk { struct clk clk; unsigned int mask; unsigned int shift; struct clk_sources *sources; unsigned int divider_shift; void __iomem *reg_divider; void __iomem *reg_source; }; static int clk_default_setrate(struct clk *clk, unsigned long rate) { clk->rate = rate; return 1; } struct clk clk_27m = { .name = "clk_27m", .id = -1, .rate = 27000000, }; static int clk_48m_ctrl(struct clk *clk, int enable) { unsigned long flags; u32 val; /* can't rely on clock lock, this register has other usages */ local_irq_save(flags); val = __raw_readl(S5PC100_CLKSRC1); if (enable) val |= S5PC100_CLKSRC1_CLK48M_MASK; else val &= ~S5PC100_CLKSRC1_CLK48M_MASK; __raw_writel(val, S5PC100_CLKSRC1); local_irq_restore(flags); return 0; } struct clk clk_48m = { .name = "clk_48m", .id = -1, .rate = 48000000, .enable = clk_48m_ctrl, }; struct clk clk_54m = { .name = "clk_54m", .id = -1, .rate = 54000000, }; struct clk clk_hpll = { .name = "hpll", .id = -1, }; struct clk clk_hd0 = { .name = "hclkd0", .id = -1, .rate = 0, .parent = NULL, .ctrlbit = 0, .set_rate = clk_default_setrate, }; struct clk clk_pd0 = { .name = "pclkd0", .id = -1, .rate = 0, .parent = NULL, .ctrlbit = 0, .set_rate = clk_default_setrate, }; static int s5pc1xx_clk_gate(void __iomem *reg, struct clk *clk, int enable) { unsigned int ctrlbit = clk->ctrlbit; u32 con; con = __raw_readl(reg); if (enable) con |= ctrlbit; else con &= ~ctrlbit; __raw_writel(con, reg); return 0; } static int s5pc1xx_clk_d00_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_CLKGATE_D00, clk, enable); } static int s5pc1xx_clk_d01_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_CLKGATE_D01, clk, enable); } static int s5pc1xx_clk_d02_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_CLKGATE_D02, clk, enable); } static int s5pc1xx_clk_d10_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_CLKGATE_D10, clk, enable); } static int s5pc1xx_clk_d11_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_CLKGATE_D11, clk, enable); } static int s5pc1xx_clk_d12_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_CLKGATE_D12, clk, enable); } static int s5pc1xx_clk_d13_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_CLKGATE_D13, clk, enable); } static int s5pc1xx_clk_d14_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_CLKGATE_D14, clk, enable); } static int s5pc1xx_clk_d15_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_CLKGATE_D15, clk, enable); } static int s5pc1xx_clk_d20_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_CLKGATE_D20, clk, enable); } int s5pc1xx_sclk0_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_SCLKGATE0, clk, enable); } int s5pc1xx_sclk1_ctrl(struct clk *clk, int enable) { return s5pc1xx_clk_gate(S5PC100_SCLKGATE1, clk, enable); } static struct clk init_clocks_disable[] = { { .name = "dsi", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d11_ctrl, .ctrlbit = S5PC100_CLKGATE_D11_DSI, }, { .name = "csi", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d11_ctrl, .ctrlbit = S5PC100_CLKGATE_D11_CSI, }, { .name = "ccan0", .id = 0, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_CCAN0, }, { .name = "ccan1", .id = 1, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_CCAN1, }, { .name = "keypad", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_KEYIF, }, { .name = "hclkd2", .id = -1, .parent = NULL, .enable = s5pc1xx_clk_d20_ctrl, .ctrlbit = S5PC100_CLKGATE_D20_HCLKD2, }, { .name = "iis-d2", .id = -1, .parent = NULL, .enable = s5pc1xx_clk_d20_ctrl, .ctrlbit = S5PC100_CLKGATE_D20_I2SD2, }, { .name = "otg", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d10_ctrl, .ctrlbit = S5PC100_CLKGATE_D10_USBOTG, }, }; static struct clk init_clocks[] = { /* System1 (D0_0) devices */ { .name = "intc", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d00_ctrl, .ctrlbit = S5PC100_CLKGATE_D00_INTC, }, { .name = "tzic", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d00_ctrl, .ctrlbit = S5PC100_CLKGATE_D00_TZIC, }, { .name = "cf-ata", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d00_ctrl, .ctrlbit = S5PC100_CLKGATE_D00_CFCON, }, { .name = "mdma", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d00_ctrl, .ctrlbit = S5PC100_CLKGATE_D00_MDMA, }, { .name = "g2d", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d00_ctrl, .ctrlbit = S5PC100_CLKGATE_D00_G2D, }, { .name = "secss", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d00_ctrl, .ctrlbit = S5PC100_CLKGATE_D00_SECSS, }, { .name = "cssys", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d00_ctrl, .ctrlbit = S5PC100_CLKGATE_D00_CSSYS, }, /* Memory (D0_1) devices */ { .name = "dmc", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d01_ctrl, .ctrlbit = S5PC100_CLKGATE_D01_DMC, }, { .name = "sromc", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d01_ctrl, .ctrlbit = S5PC100_CLKGATE_D01_SROMC, }, { .name = "onenand", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d01_ctrl, .ctrlbit = S5PC100_CLKGATE_D01_ONENAND, }, { .name = "nand", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d01_ctrl, .ctrlbit = S5PC100_CLKGATE_D01_NFCON, }, { .name = "intmem", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d01_ctrl, .ctrlbit = S5PC100_CLKGATE_D01_INTMEM, }, { .name = "ebi", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d01_ctrl, .ctrlbit = S5PC100_CLKGATE_D01_EBI, }, /* System2 (D0_2) devices */ { .name = "seckey", .id = -1, .parent = &clk_pd0, .enable = s5pc1xx_clk_d02_ctrl, .ctrlbit = S5PC100_CLKGATE_D02_SECKEY, }, { .name = "sdm", .id = -1, .parent = &clk_hd0, .enable = s5pc1xx_clk_d02_ctrl, .ctrlbit = S5PC100_CLKGATE_D02_SDM, }, /* File (D1_0) devices */ { .name = "pdma0", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d10_ctrl, .ctrlbit = S5PC100_CLKGATE_D10_PDMA0, }, { .name = "pdma1", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d10_ctrl, .ctrlbit = S5PC100_CLKGATE_D10_PDMA1, }, { .name = "usb-host", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d10_ctrl, .ctrlbit = S5PC100_CLKGATE_D10_USBHOST, }, { .name = "modem", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d10_ctrl, .ctrlbit = S5PC100_CLKGATE_D10_MODEMIF, }, { .name = "hsmmc", .id = 0, .parent = &clk_h, .enable = s5pc1xx_clk_d10_ctrl, .ctrlbit = S5PC100_CLKGATE_D10_HSMMC0, }, { .name = "hsmmc", .id = 1, .parent = &clk_h, .enable = s5pc1xx_clk_d10_ctrl, .ctrlbit = S5PC100_CLKGATE_D10_HSMMC1, }, { .name = "hsmmc", .id = 2, .parent = &clk_h, .enable = s5pc1xx_clk_d10_ctrl, .ctrlbit = S5PC100_CLKGATE_D10_HSMMC2, }, /* Multimedia1 (D1_1) devices */ { .name = "lcd", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d11_ctrl, .ctrlbit = S5PC100_CLKGATE_D11_LCD, }, { .name = "rotator", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d11_ctrl, .ctrlbit = S5PC100_CLKGATE_D11_ROTATOR, }, { .name = "fimc", .id = 0, .parent = &clk_h, .enable = s5pc1xx_clk_d11_ctrl, .ctrlbit = S5PC100_CLKGATE_D11_FIMC0, }, { .name = "fimc", .id = 1, .parent = &clk_h, .enable = s5pc1xx_clk_d11_ctrl, .ctrlbit = S5PC100_CLKGATE_D11_FIMC1, }, { .name = "fimc", .id = 2, .parent = &clk_h, .enable = s5pc1xx_clk_d11_ctrl, .ctrlbit = S5PC100_CLKGATE_D11_FIMC2, }, { .name = "jpeg", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d11_ctrl, .ctrlbit = S5PC100_CLKGATE_D11_JPEG, }, { .name = "g3d", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d11_ctrl, .ctrlbit = S5PC100_CLKGATE_D11_G3D, }, /* Multimedia2 (D1_2) devices */ { .name = "tv", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d12_ctrl, .ctrlbit = S5PC100_CLKGATE_D12_TV, }, { .name = "vp", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d12_ctrl, .ctrlbit = S5PC100_CLKGATE_D12_VP, }, { .name = "mixer", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d12_ctrl, .ctrlbit = S5PC100_CLKGATE_D12_MIXER, }, { .name = "hdmi", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d12_ctrl, .ctrlbit = S5PC100_CLKGATE_D12_HDMI, }, { .name = "mfc", .id = -1, .parent = &clk_h, .enable = s5pc1xx_clk_d12_ctrl, .ctrlbit = S5PC100_CLKGATE_D12_MFC, }, /* System (D1_3) devices */ { .name = "chipid", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d13_ctrl, .ctrlbit = S5PC100_CLKGATE_D13_CHIPID, }, { .name = "gpio", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d13_ctrl, .ctrlbit = S5PC100_CLKGATE_D13_GPIO, }, { .name = "apc", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d13_ctrl, .ctrlbit = S5PC100_CLKGATE_D13_APC, }, { .name = "iec", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d13_ctrl, .ctrlbit = S5PC100_CLKGATE_D13_IEC, }, { .name = "timers", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d13_ctrl, .ctrlbit = S5PC100_CLKGATE_D13_PWM, }, { .name = "systimer", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d13_ctrl, .ctrlbit = S5PC100_CLKGATE_D13_SYSTIMER, }, { .name = "watchdog", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d13_ctrl, .ctrlbit = S5PC100_CLKGATE_D13_WDT, }, { .name = "rtc", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d13_ctrl, .ctrlbit = S5PC100_CLKGATE_D13_RTC, }, /* Connectivity (D1_4) devices */ { .name = "uart", .id = 0, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_UART0, }, { .name = "uart", .id = 1, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_UART1, }, { .name = "uart", .id = 2, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_UART2, }, { .name = "uart", .id = 3, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_UART3, }, { .name = "i2c", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_IIC, }, { .name = "hdmi-i2c", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_HDMI_IIC, }, { .name = "spi", .id = 0, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_SPI0, }, { .name = "spi", .id = 1, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_SPI1, }, { .name = "spi", .id = 2, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_SPI2, }, { .name = "irda", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_IRDA, }, { .name = "hsitx", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_HSITX, }, { .name = "hsirx", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d14_ctrl, .ctrlbit = S5PC100_CLKGATE_D14_HSIRX, }, /* Audio (D1_5) devices */ { .name = "iis", .id = 0, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_IIS0, }, { .name = "iis", .id = 1, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_IIS1, }, { .name = "iis", .id = 2, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_IIS2, }, { .name = "ac97", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_AC97, }, { .name = "pcm", .id = 0, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_PCM0, }, { .name = "pcm", .id = 1, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_PCM1, }, { .name = "spdif", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_SPDIF, }, { .name = "adc", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_TSADC, }, { .name = "keyif", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_KEYIF, }, { .name = "cg", .id = -1, .parent = &clk_p, .enable = s5pc1xx_clk_d15_ctrl, .ctrlbit = S5PC100_CLKGATE_D15_CG, }, /* Audio (D2_0) devices: all disabled */ /* Special Clocks 1 */ { .name = "sclk_hpm", .id = -1, .parent = NULL, .enable = s5pc1xx_sclk0_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK0_HPM, }, { .name = "sclk_onenand", .id = -1, .parent = NULL, .enable = s5pc1xx_sclk0_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK0_ONENAND, }, { .name = "sclk_spi_48", .id = 0, .parent = &clk_48m, .enable = s5pc1xx_sclk0_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK0_SPI0_48, }, { .name = "sclk_spi_48", .id = 1, .parent = &clk_48m, .enable = s5pc1xx_sclk0_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK0_SPI1_48, }, { .name = "sclk_spi_48", .id = 2, .parent = &clk_48m, .enable = s5pc1xx_sclk0_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK0_SPI2_48, }, { .name = "sclk_mmc_48", .id = 0, .parent = &clk_48m, .enable = s5pc1xx_sclk0_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK0_MMC0_48, }, { .name = "sclk_mmc_48", .id = 1, .parent = &clk_48m, .enable = s5pc1xx_sclk0_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK0_MMC1_48, }, { .name = "sclk_mmc_48", .id = 2, .parent = &clk_48m, .enable = s5pc1xx_sclk0_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK0_MMC2_48, }, /* Special Clocks 2 */ { .name = "sclk_tv_54", .id = -1, .parent = &clk_54m, .enable = s5pc1xx_sclk1_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK1_TV54, }, { .name = "sclk_vdac_54", .id = -1, .parent = &clk_54m, .enable = s5pc1xx_sclk1_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK1_VDAC54, }, { .name = "sclk_spdif", .id = -1, .parent = NULL, .enable = s5pc1xx_sclk1_ctrl, .ctrlbit = S5PC100_CLKGATE_SCLK1_SPDIF, }, }; void __init s5pc1xx_register_clocks(void) { struct clk *clkp; int ret; int ptr; clkp = init_clocks; for (ptr = 0; ptr < ARRAY_SIZE(init_clocks); ptr++, clkp++) { ret = s3c24xx_register_clock(clkp); if (ret < 0) { printk(KERN_ERR "Failed to register clock %s (%d)\n", clkp->name, ret); } } clkp = init_clocks_disable; for (ptr = 0; ptr < ARRAY_SIZE(init_clocks_disable); ptr++, clkp++) { ret = s3c24xx_register_clock(clkp); if (ret < 0) { printk(KERN_ERR "Failed to register clock %s (%d)\n", clkp->name, ret); } (clkp->enable)(clkp, 0); } s3c_pwmclk_init(); } static struct clk clk_fout_apll = { .name = "fout_apll", .id = -1, }; static struct clk *clk_src_apll_list[] = { [0] = &clk_fin_apll, [1] = &clk_fout_apll, }; static struct clk_sources clk_src_apll = { .sources = clk_src_apll_list, .nr_sources = ARRAY_SIZE(clk_src_apll_list), }; static struct clksrc_clk clk_mout_apll = { .clk = { .name = "mout_apll", .id = -1, }, .shift = S5PC100_CLKSRC0_APLL_SHIFT, .mask = S5PC100_CLKSRC0_APLL_MASK, .sources = &clk_src_apll, .reg_source = S5PC100_CLKSRC0, }; static struct clk clk_fout_epll = { .name = "fout_epll", .id = -1, }; static struct clk *clk_src_epll_list[] = { [0] = &clk_fin_epll, [1] = &clk_fout_epll, }; static struct clk_sources clk_src_epll = { .sources = clk_src_epll_list, .nr_sources = ARRAY_SIZE(clk_src_epll_list), }; static struct clksrc_clk clk_mout_epll = { .clk = { .name = "mout_epll", .id = -1, }, .shift = S5PC100_CLKSRC0_EPLL_SHIFT, .mask = S5PC100_CLKSRC0_EPLL_MASK, .sources = &clk_src_epll, .reg_source = S5PC100_CLKSRC0, }; static struct clk *clk_src_mpll_list[] = { [0] = &clk_fin_mpll, [1] = &clk_fout_mpll, }; static struct clk_sources clk_src_mpll = { .sources = clk_src_mpll_list, .nr_sources = ARRAY_SIZE(clk_src_mpll_list), }; static struct clksrc_clk clk_mout_mpll = { .clk = { .name = "mout_mpll", .id = -1, }, .shift = S5PC100_CLKSRC0_MPLL_SHIFT, .mask = S5PC100_CLKSRC0_MPLL_MASK, .sources = &clk_src_mpll, .reg_source = S5PC100_CLKSRC0, }; static unsigned long s5pc1xx_clk_doutmpll_get_rate(struct clk *clk) { unsigned long rate = clk_get_rate(clk->parent); unsigned long clkdiv; printk(KERN_DEBUG "%s: parent is %ld\n", __func__, rate); clkdiv = __raw_readl(S5PC100_CLKDIV1) & S5PC100_CLKDIV1_MPLL_MASK; rate /= (clkdiv >> S5PC100_CLKDIV1_MPLL_SHIFT) + 1; return rate; } static struct clk clk_dout_mpll = { .name = "dout_mpll", .id = -1, .parent = &clk_mout_mpll.clk, .get_rate = s5pc1xx_clk_doutmpll_get_rate, }; static unsigned long s5pc1xx_clk_doutmpll2_get_rate(struct clk *clk) { unsigned long rate = clk_get_rate(clk->parent); unsigned long clkdiv; printk(KERN_DEBUG "%s: parent is %ld\n", __func__, rate); clkdiv = __raw_readl(S5PC100_CLKDIV1) & S5PC100_CLKDIV1_MPLL2_MASK; rate /= (clkdiv >> S5PC100_CLKDIV1_MPLL2_SHIFT) + 1; return rate; } struct clk clk_dout_mpll2 = { .name = "dout_mpll2", .id = -1, .parent = &clk_mout_mpll.clk, .get_rate = s5pc1xx_clk_doutmpll2_get_rate, }; static struct clk *clkset_uart_list[] = { &clk_mout_epll.clk, &clk_dout_mpll, NULL, NULL }; static struct clk_sources clkset_uart = { .sources = clkset_uart_list, .nr_sources = ARRAY_SIZE(clkset_uart_list), }; static inline struct clksrc_clk *to_clksrc(struct clk *clk) { return container_of(clk, struct clksrc_clk, clk); } static unsigned long s5pc1xx_getrate_clksrc(struct clk *clk) { struct clksrc_clk *sclk = to_clksrc(clk); unsigned long rate = clk_get_rate(clk->parent); u32 clkdiv = __raw_readl(sclk->reg_divider); clkdiv >>= sclk->divider_shift; clkdiv &= 0xf; clkdiv++; rate /= clkdiv; return rate; } static int s5pc1xx_setrate_clksrc(struct clk *clk, unsigned long rate) { struct clksrc_clk *sclk = to_clksrc(clk); void __iomem *reg = sclk->reg_divider; unsigned int div; u32 val; rate = clk_round_rate(clk, rate); div = clk_get_rate(clk->parent) / rate; if (div > 16) return -EINVAL; val = __raw_readl(reg); val &= ~(0xf << sclk->shift); val |= (div - 1) << sclk->shift; __raw_writel(val, reg); return 0; } static int s5pc1xx_setparent_clksrc(struct clk *clk, struct clk *parent) { struct clksrc_clk *sclk = to_clksrc(clk); struct clk_sources *srcs = sclk->sources; u32 clksrc = __raw_readl(sclk->reg_source); int src_nr = -1; int ptr; for (ptr = 0; ptr < srcs->nr_sources; ptr++) if (srcs->sources[ptr] == parent) { src_nr = ptr; break; } if (src_nr >= 0) { clksrc &= ~sclk->mask; clksrc |= src_nr << sclk->shift; __raw_writel(clksrc, sclk->reg_source); return 0; } return -EINVAL; } static unsigned long s5pc1xx_roundrate_clksrc(struct clk *clk, unsigned long rate) { unsigned long parent_rate = clk_get_rate(clk->parent); int div; if (rate > parent_rate) rate = parent_rate; else { div = rate / parent_rate; if (div == 0) div = 1; if (div > 16) div = 16; rate = parent_rate / div; } return rate; } static struct clksrc_clk clk_uart_uclk1 = { .clk = { .name = "uclk1", .id = -1, .ctrlbit = S5PC100_CLKGATE_SCLK0_UART, .enable = s5pc1xx_sclk0_ctrl, .set_parent = s5pc1xx_setparent_clksrc, .get_rate = s5pc1xx_getrate_clksrc, .set_rate = s5pc1xx_setrate_clksrc, .round_rate = s5pc1xx_roundrate_clksrc, }, .shift = S5PC100_CLKSRC1_UART_SHIFT, .mask = S5PC100_CLKSRC1_UART_MASK, .sources = &clkset_uart, .divider_shift = S5PC100_CLKDIV2_UART_SHIFT, .reg_divider = S5PC100_CLKDIV2, .reg_source = S5PC100_CLKSRC1, }; /* Clock initialisation code */ static struct clksrc_clk *init_parents[] = { &clk_mout_apll, &clk_mout_epll, &clk_mout_mpll, &clk_uart_uclk1, }; static void __init_or_cpufreq s5pc1xx_set_clksrc(struct clksrc_clk *clk) { struct clk_sources *srcs = clk->sources; u32 clksrc = __raw_readl(clk->reg_source); clksrc &= clk->mask; clksrc >>= clk->shift; if (clksrc > srcs->nr_sources || !srcs->sources[clksrc]) { printk(KERN_ERR "%s: bad source %d\n", clk->clk.name, clksrc); return; } clk->clk.parent = srcs->sources[clksrc]; printk(KERN_INFO "%s: source is %s (%d), rate is %ld\n", clk->clk.name, clk->clk.parent->name, clksrc, clk_get_rate(&clk->clk)); } #define GET_DIV(clk, field) ((((clk) & field##_MASK) >> field##_SHIFT) + 1) void __init_or_cpufreq s5pc100_setup_clocks(void) { struct clk *xtal_clk; unsigned long xtal; unsigned long armclk; unsigned long hclkd0; unsigned long hclk; unsigned long pclkd0; unsigned long pclk; unsigned long apll; unsigned long mpll; unsigned long hpll; unsigned long epll; unsigned int ptr; u32 clkdiv0, clkdiv1; printk(KERN_DEBUG "%s: registering clocks\n", __func__); clkdiv0 = __raw_readl(S5PC100_CLKDIV0); clkdiv1 = __raw_readl(S5PC100_CLKDIV1); printk(KERN_DEBUG "%s: clkdiv0 = %08x, clkdiv1 = %08x\n", __func__, clkdiv0, clkdiv1); xtal_clk = clk_get(NULL, "xtal"); BUG_ON(IS_ERR(xtal_clk)); xtal = clk_get_rate(xtal_clk); clk_put(xtal_clk); printk(KERN_DEBUG "%s: xtal is %ld\n", __func__, xtal); apll = s5pc1xx_get_pll(xtal, __raw_readl(S5PC100_APLL_CON)); mpll = s5pc1xx_get_pll(xtal, __raw_readl(S5PC100_MPLL_CON)); epll = s5pc1xx_get_pll(xtal, __raw_readl(S5PC100_EPLL_CON)); hpll = s5pc1xx_get_pll(xtal, __raw_readl(S5PC100_HPLL_CON)); printk(KERN_INFO "S5PC100: PLL settings, A=%ld, M=%ld, E=%ld, H=%ld\n", apll, mpll, epll, hpll); armclk = apll / GET_DIV(clkdiv0, S5PC100_CLKDIV0_APLL); armclk = armclk / GET_DIV(clkdiv0, S5PC100_CLKDIV0_ARM); hclkd0 = armclk / GET_DIV(clkdiv0, S5PC100_CLKDIV0_D0); pclkd0 = hclkd0 / GET_DIV(clkdiv0, S5PC100_CLKDIV0_PCLKD0); hclk = mpll / GET_DIV(clkdiv1, S5PC100_CLKDIV1_D1); pclk = hclk / GET_DIV(clkdiv1, S5PC100_CLKDIV1_PCLKD1); printk(KERN_INFO "S5PC100: ARMCLK=%ld, HCLKD0=%ld, PCLKD0=%ld, HCLK=%ld, PCLK=%ld\n", armclk, hclkd0, pclkd0, hclk, pclk); clk_fout_apll.rate = apll; clk_fout_mpll.rate = mpll; clk_fout_epll.rate = epll; clk_fout_apll.rate = apll; clk_h.rate = hclk; clk_p.rate = pclk; for (ptr = 0; ptr < ARRAY_SIZE(init_parents); ptr++) s5pc1xx_set_clksrc(init_parents[ptr]); } static struct clk *clks[] __initdata = { &clk_ext_xtal_mux, &clk_mout_epll.clk, &clk_fout_epll, &clk_mout_mpll.clk, &clk_dout_mpll, &clk_uart_uclk1.clk, &clk_ext, &clk_epll, &clk_27m, &clk_48m, &clk_54m, }; void __init s5pc100_register_clocks(void) { struct clk *clkp; int ret; int ptr; for (ptr = 0; ptr < ARRAY_SIZE(clks); ptr++) { clkp = clks[ptr]; ret = s3c24xx_register_clock(clkp); if (ret < 0) { printk(KERN_ERR "Failed to register clock %s (%d)\n", clkp->name, ret); } } clk_mpll.parent = &clk_mout_mpll.clk; clk_epll.parent = &clk_mout_epll.clk; }