?? clock.c
字號:
/* linux/arch/arm/mach-s3c2412/clock.c * * Copyright (c) 2006 Simtec Electronics * Ben Dooks <ben@simtec.co.uk> * * S3C2412,S3C2413 Clock control support * * 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 <linux/init.h>#include <linux/module.h>#include <linux/kernel.h>#include <linux/list.h>#include <linux/errno.h>#include <linux/err.h>#include <linux/sysdev.h>#include <linux/clk.h>#include <linux/mutex.h>#include <linux/delay.h>#include <linux/serial_core.h>#include <asm/mach/map.h>#include <asm/hardware.h>#include <asm/io.h>#include <asm/plat-s3c/regs-serial.h>#include <asm/arch/regs-clock.h>#include <asm/arch/regs-gpio.h>#include <asm/plat-s3c24xx/s3c2412.h>#include <asm/plat-s3c24xx/clock.h>#include <asm/plat-s3c24xx/cpu.h>/* We currently have to assume that the system is running * from the XTPll input, and that all ***REFCLKs are being * fed from it, as we cannot read the state of OM[4] from * software. * * It would be possible for each board initialisation to * set the correct muxing at initialisation*/static int s3c2412_clkcon_enable(struct clk *clk, int enable){ unsigned int clocks = clk->ctrlbit; unsigned long clkcon; clkcon = __raw_readl(S3C2410_CLKCON); if (enable) clkcon |= clocks; else clkcon &= ~clocks; __raw_writel(clkcon, S3C2410_CLKCON); return 0;}static int s3c2412_upll_enable(struct clk *clk, int enable){ unsigned long upllcon = __raw_readl(S3C2410_UPLLCON); unsigned long orig = upllcon; if (!enable) upllcon |= S3C2412_PLLCON_OFF; else upllcon &= ~S3C2412_PLLCON_OFF; __raw_writel(upllcon, S3C2410_UPLLCON); /* allow ~150uS for the PLL to settle and lock */ if (enable && (orig & S3C2412_PLLCON_OFF)) udelay(150); return 0;}/* clock selections *//* CPU EXTCLK input */static struct clk clk_ext = { .name = "extclk", .id = -1,};static struct clk clk_erefclk = { .name = "erefclk", .id = -1,};static struct clk clk_urefclk = { .name = "urefclk", .id = -1,};static int s3c2412_setparent_usysclk(struct clk *clk, struct clk *parent){ unsigned long clksrc = __raw_readl(S3C2412_CLKSRC); if (parent == &clk_urefclk) clksrc &= ~S3C2412_CLKSRC_USYSCLK_UPLL; else if (parent == &clk_upll) clksrc |= S3C2412_CLKSRC_USYSCLK_UPLL; else return -EINVAL; clk->parent = parent; __raw_writel(clksrc, S3C2412_CLKSRC); return 0;}static struct clk clk_usysclk = { .name = "usysclk", .id = -1, .parent = &clk_xtal, .set_parent = s3c2412_setparent_usysclk,};static struct clk clk_mrefclk = { .name = "mrefclk", .parent = &clk_xtal, .id = -1,};static struct clk clk_mdivclk = { .name = "mdivclk", .parent = &clk_xtal, .id = -1,};static int s3c2412_setparent_usbsrc(struct clk *clk, struct clk *parent){ unsigned long clksrc = __raw_readl(S3C2412_CLKSRC); if (parent == &clk_usysclk) clksrc &= ~S3C2412_CLKSRC_USBCLK_HCLK; else if (parent == &clk_h) clksrc |= S3C2412_CLKSRC_USBCLK_HCLK; else return -EINVAL; clk->parent = parent; __raw_writel(clksrc, S3C2412_CLKSRC); return 0;}static unsigned long s3c2412_roundrate_usbsrc(struct clk *clk, unsigned long rate){ unsigned long parent_rate = clk_get_rate(clk->parent); int div; if (rate > parent_rate) return parent_rate; div = parent_rate / rate; if (div > 2) div = 2; return parent_rate / div;}static unsigned long s3c2412_getrate_usbsrc(struct clk *clk){ unsigned long parent_rate = clk_get_rate(clk->parent); unsigned long div = __raw_readl(S3C2410_CLKDIVN); return parent_rate / ((div & S3C2412_CLKDIVN_USB48DIV) ? 2 : 1);}static int s3c2412_setrate_usbsrc(struct clk *clk, unsigned long rate){ unsigned long parent_rate = clk_get_rate(clk->parent); unsigned long clkdivn = __raw_readl(S3C2410_CLKDIVN); rate = s3c2412_roundrate_usbsrc(clk, rate); if ((parent_rate / rate) == 2) clkdivn |= S3C2412_CLKDIVN_USB48DIV; else clkdivn &= ~S3C2412_CLKDIVN_USB48DIV; __raw_writel(clkdivn, S3C2410_CLKDIVN); return 0;}static struct clk clk_usbsrc = { .name = "usbsrc", .id = -1, .get_rate = s3c2412_getrate_usbsrc, .set_rate = s3c2412_setrate_usbsrc, .round_rate = s3c2412_roundrate_usbsrc, .set_parent = s3c2412_setparent_usbsrc,};static int s3c2412_setparent_msysclk(struct clk *clk, struct clk *parent){ unsigned long clksrc = __raw_readl(S3C2412_CLKSRC); if (parent == &clk_mdivclk) clksrc &= ~S3C2412_CLKSRC_MSYSCLK_MPLL; else if (parent == &clk_upll) clksrc |= S3C2412_CLKSRC_MSYSCLK_MPLL; else return -EINVAL; clk->parent = parent; __raw_writel(clksrc, S3C2412_CLKSRC); return 0;}static struct clk clk_msysclk = { .name = "msysclk", .id = -1, .set_parent = s3c2412_setparent_msysclk,};/* these next clocks have an divider immediately after them, * so we can register them with their divider and leave out the * intermediate clock stage*/static unsigned long s3c2412_roundrate_clksrc(struct clk *clk, unsigned long rate){ unsigned long parent_rate = clk_get_rate(clk->parent); int div; if (rate > parent_rate) return parent_rate; /* note, we remove the +/- 1 calculations as they cancel out */ div = (rate / parent_rate); if (div < 1) div = 1; else if (div > 16) div = 16; return parent_rate / div;}static int s3c2412_setparent_uart(struct clk *clk, struct clk *parent){ unsigned long clksrc = __raw_readl(S3C2412_CLKSRC); if (parent == &clk_erefclk) clksrc &= ~S3C2412_CLKSRC_UARTCLK_MPLL; else if (parent == &clk_mpll) clksrc |= S3C2412_CLKSRC_UARTCLK_MPLL; else return -EINVAL; clk->parent = parent; __raw_writel(clksrc, S3C2412_CLKSRC); return 0;}static unsigned long s3c2412_getrate_uart(struct clk *clk){ unsigned long parent_rate = clk_get_rate(clk->parent); unsigned long div = __raw_readl(S3C2410_CLKDIVN); div &= S3C2412_CLKDIVN_UARTDIV_MASK; div >>= S3C2412_CLKDIVN_UARTDIV_SHIFT; return parent_rate / (div + 1);}static int s3c2412_setrate_uart(struct clk *clk, unsigned long rate){ unsigned long parent_rate = clk_get_rate(clk->parent); unsigned long clkdivn = __raw_readl(S3C2410_CLKDIVN); rate = s3c2412_roundrate_clksrc(clk, rate); clkdivn &= ~S3C2412_CLKDIVN_UARTDIV_MASK; clkdivn |= ((parent_rate / rate) - 1) << S3C2412_CLKDIVN_UARTDIV_SHIFT; __raw_writel(clkdivn, S3C2410_CLKDIVN); return 0;}static struct clk clk_uart = { .name = "uartclk", .id = -1, .get_rate = s3c2412_getrate_uart, .set_rate = s3c2412_setrate_uart, .set_parent = s3c2412_setparent_uart, .round_rate = s3c2412_roundrate_clksrc,};static int s3c2412_setparent_i2s(struct clk *clk, struct clk *parent){ unsigned long clksrc = __raw_readl(S3C2412_CLKSRC); if (parent == &clk_erefclk) clksrc &= ~S3C2412_CLKSRC_I2SCLK_MPLL; else if (parent == &clk_mpll) clksrc |= S3C2412_CLKSRC_I2SCLK_MPLL; else return -EINVAL; clk->parent = parent; __raw_writel(clksrc, S3C2412_CLKSRC); return 0;}static unsigned long s3c2412_getrate_i2s(struct clk *clk){ unsigned long parent_rate = clk_get_rate(clk->parent); unsigned long div = __raw_readl(S3C2410_CLKDIVN); div &= S3C2412_CLKDIVN_I2SDIV_MASK; div >>= S3C2412_CLKDIVN_I2SDIV_SHIFT; return parent_rate / (div + 1);}static int s3c2412_setrate_i2s(struct clk *clk, unsigned long rate){ unsigned long parent_rate = clk_get_rate(clk->parent); unsigned long clkdivn = __raw_readl(S3C2410_CLKDIVN); rate = s3c2412_roundrate_clksrc(clk, rate); clkdivn &= ~S3C2412_CLKDIVN_I2SDIV_MASK; clkdivn |= ((parent_rate / rate) - 1) << S3C2412_CLKDIVN_I2SDIV_SHIFT; __raw_writel(clkdivn, S3C2410_CLKDIVN); return 0;}static struct clk clk_i2s = { .name = "i2sclk", .id = -1, .get_rate = s3c2412_getrate_i2s, .set_rate = s3c2412_setrate_i2s,
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -