[PATCH v8 11/17] h8300: clock driver

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



h8300 clock generator drivers.
H8/3069 is simple oscillator.
H8S2678 is PLL multiplier support.

Signed-off-by: Yoshinori Sato <ysato@xxxxxxxxxxxxxxxxxxxx>
---
 drivers/clk/Makefile            |   1 +
 drivers/clk/h8300/Makefile      |   2 +
 drivers/clk/h8300/clk-h83069.c  |  80 +++++++++++++++++++
 drivers/clk/h8300/clk-h8s2678.c | 171 ++++++++++++++++++++++++++++++++++++++++
 include/linux/clk-provider.h    |  12 +++
 5 files changed, 266 insertions(+)
 create mode 100644 drivers/clk/h8300/Makefile
 create mode 100644 drivers/clk/h8300/clk-h83069.c
 create mode 100644 drivers/clk/h8300/clk-h8s2678.c

diff --git a/drivers/clk/Makefile b/drivers/clk/Makefile
index d478ceb..593b09c 100644
--- a/drivers/clk/Makefile
+++ b/drivers/clk/Makefile
@@ -70,3 +70,4 @@ obj-$(CONFIG_ARCH_U8500)		+= ux500/
 obj-$(CONFIG_COMMON_CLK_VERSATILE)	+= versatile/
 obj-$(CONFIG_X86)			+= x86/
 obj-$(CONFIG_ARCH_ZYNQ)			+= zynq/
+obj-$(CONFIG_H8300)		+= h8300/
diff --git a/drivers/clk/h8300/Makefile b/drivers/clk/h8300/Makefile
new file mode 100644
index 0000000..82eab42
--- /dev/null
+++ b/drivers/clk/h8300/Makefile
@@ -0,0 +1,2 @@
+obj-$(CONFIG_H83069) += clk-h83069.o
+obj-$(CONFIG_H8S2678) += clk-h8s2678.o
diff --git a/drivers/clk/h8300/clk-h83069.c b/drivers/clk/h8300/clk-h83069.c
new file mode 100644
index 0000000..0a20dd5
--- /dev/null
+++ b/drivers/clk/h8300/clk-h83069.c
@@ -0,0 +1,80 @@
+/*
+ * H8/3069 clock driver
+ *
+ * Copyright 2015 Yoshinori Sato <ysato@xxxxxxxxxxxxxxxxxxxx>
+ */
+
+#include <linux/clk.h>
+#include <linux/clkdev.h>
+#include <linux/clk-provider.h>
+#include <linux/err.h>
+#include <linux/device.h>
+#include <linux/platform_device.h>
+
+static DEFINE_SPINLOCK(clklock);
+
+#define DIVCR ((unsigned char *)0xfee01b)
+#define DEVNAME "h83069-cpg"
+
+static int clk_probe(struct platform_device *pdev)
+{
+	struct clk *clk;
+	int *hz = dev_get_platdata(&pdev->dev);
+
+	clk = clk_register_fixed_rate(&pdev->dev, "master_clk", NULL,
+				      CLK_IS_ROOT, *hz);
+	if (IS_ERR(clk)) {
+		dev_err(&pdev->dev, "failed to register clock");
+		return PTR_ERR(clk);
+	}
+	clk_register_clkdev(clk, "master_clk", DEVNAME ".%d", 0);
+
+	clk = clk_register_divider(&pdev->dev, "core_clk", "master_clk",
+				   CLK_SET_RATE_GATE, DIVCR, 0, 2,
+				   CLK_DIVIDER_POWER_OF_TWO, &clklock);
+	if (IS_ERR(clk)) {
+		dev_err(&pdev->dev, "failed to register clock");
+		return PTR_ERR(clk);
+	}
+	clk_register_clkdev(clk, "core_clk", DEVNAME ".%d", 0);
+
+	clk = clk_register_fixed_factor(&pdev->dev, "peripheral_clk", "core_clk",
+					0, 1, 1);
+	if (IS_ERR(clk)) {
+		dev_err(&pdev->dev, "failed to register clock");
+		return PTR_ERR(clk);
+	}
+	clk_register_clkdev(clk, "peripheral_clk", NULL);
+	return 0;
+}
+
+static struct platform_driver cpg_driver = {
+	.driver = {
+		.name = DEVNAME,
+	},
+	.probe = clk_probe,
+};
+
+early_platform_init(DEVNAME, &cpg_driver);
+
+static struct platform_device clk_device = {
+	.name		= DEVNAME,
+	.id		= 0,
+};
+
+static struct platform_device *devices[] __initdata = {
+	&clk_device,
+};
+
+int __init h8300_clk_init(int hz)
+{
+	static int master_hz;
+
+	master_hz = hz;
+	clk_device.dev.platform_data = &master_hz;
+	early_platform_add_devices(devices,
+				   ARRAY_SIZE(devices));
+	early_platform_driver_register_all(DEVNAME);
+	early_platform_driver_probe(DEVNAME, 1, 0);
+	return 0;
+}
diff --git a/drivers/clk/h8300/clk-h8s2678.c b/drivers/clk/h8300/clk-h8s2678.c
new file mode 100644
index 0000000..55b490c
--- /dev/null
+++ b/drivers/clk/h8300/clk-h8s2678.c
@@ -0,0 +1,171 @@
+/*
+ * H8S2678 clock driver
+ *
+ * Copyright 2015 Yoshinori Sato <ysato@xxxxxxxxxxxxxxxxxxxx>
+ */
+
+#include <linux/clk.h>
+#include <linux/clkdev.h>
+#include <linux/clk-provider.h>
+#include <linux/err.h>
+#include <linux/device.h>
+#include <linux/platform_device.h>
+
+static DEFINE_SPINLOCK(clklock);
+
+#define SCKCR 0xffff3b
+#define PLLCR 0xffff45
+#define DEVNAME "h8s2678-cpg"
+#define MAX_FREQ 33333333
+#define MIN_FREQ  8000000
+
+static unsigned long pll_recalc_rate(struct clk_hw *hw,
+		unsigned long parent_rate)
+{
+	int mul = 1 << (ctrl_inb(PLLCR) & 3);
+
+	return parent_rate * mul;
+}
+
+static long pll_round_rate(struct clk_hw *hw, unsigned long rate,
+				unsigned long *prate)
+{
+	int i, m = -1;
+	long offset[3];
+
+	if (rate > MAX_FREQ)
+		rate = MAX_FREQ;
+	if (rate < MIN_FREQ)
+		rate = MIN_FREQ;
+
+	for (i = 0; i < 3; i++)
+		offset[i] = abs(rate - (*prate * (1 << i)));
+	for (i = 0; i < 3; i++)
+		if (m < 0)
+			m = i;
+		else
+			m = (offset[i] < offset[m])?i:m;
+
+	return *prate * (1 << m);
+}
+
+static int pll_set_rate(struct clk_hw *hw, unsigned long rate,
+			unsigned long parent_rate)
+{
+	int pll;
+	unsigned char val;
+	unsigned long flags;
+
+	pll = ((rate / parent_rate) / 2) & 0x03;
+	spin_lock_irqsave(&clklock, flags);
+	val = ctrl_inb(SCKCR);
+	val |= 0x08;
+	ctrl_outb(val, SCKCR);
+	val = ctrl_inb(PLLCR);
+	val &= ~0x03;
+	val |= pll;
+	ctrl_outb(val, PLLCR);
+	spin_unlock_irqrestore(&clklock, flags);
+	return 0;
+}
+
+static const struct clk_ops pll_ops = {
+	.recalc_rate = pll_recalc_rate,
+	.round_rate = pll_round_rate,
+	.set_rate = pll_set_rate,
+};
+
+static struct clk *pll_clk_register(struct device *dev, const char *name,
+				const char *parent)
+{
+	struct clk_hw *hw;
+	struct clk *clk;
+	struct clk_init_data init;
+
+	hw = kzalloc(sizeof(struct clk_hw), GFP_KERNEL);
+	if (!hw)
+		return ERR_PTR(-ENOMEM);
+
+	init.name = name;
+	init.ops = &pll_ops;
+	init.flags = CLK_IS_BASIC;
+	init.parent_names = &parent;
+	init.num_parents = 1;
+	hw->init = &init;
+
+	clk = clk_register(dev, hw);
+	if (IS_ERR(clk))
+		kfree(hw);
+
+	return clk;
+}
+
+static int clk_probe(struct platform_device *pdev)
+{
+	struct clk *clk;
+	int *hz = dev_get_platdata(&pdev->dev);
+
+	clk = clk_register_fixed_rate(&pdev->dev, "master_clk", NULL,
+				      CLK_IS_ROOT, *hz);
+	if (IS_ERR(clk)) {
+		dev_err(&pdev->dev, "failed to register clock");
+		return PTR_ERR(clk);
+	}
+	clk_register_clkdev(clk, "master_clk", DEVNAME ".%d", 0);
+
+	clk = pll_clk_register(&pdev->dev, "pll_clk", "master_clk");
+	if (IS_ERR(clk)) {
+		dev_err(&pdev->dev, "failed to register clock");
+		return PTR_ERR(clk);
+	}
+	clk_register_clkdev(clk, "pll_clk", DEVNAME ".%d", 0);
+
+	clk = clk_register_divider(&pdev->dev, "core_clk", "pll_clk",
+				   CLK_SET_RATE_GATE, (unsigned char *)SCKCR,
+				   0, 3, CLK_DIVIDER_POWER_OF_TWO, &clklock);
+	if (IS_ERR(clk)) {
+		dev_err(&pdev->dev, "failed to register clock");
+		return PTR_ERR(clk);
+	}
+	clk_register_clkdev(clk, "core_clk", DEVNAME ".%d", 0);
+
+	clk = clk_register_fixed_factor(&pdev->dev, "peripheral_clk", "core_clk",
+					0, 1, 1);
+	if (IS_ERR(clk)) {
+		dev_err(&pdev->dev, "failed to register clock");
+		return PTR_ERR(clk);
+	}
+	clk_register_clkdev(clk, "peripheral_clk", NULL);
+	return 0;
+}
+
+static struct platform_driver cpg_driver = {
+	.driver = {
+		.name = DEVNAME,
+	},
+	.probe = clk_probe,
+};
+
+early_platform_init(DEVNAME, &cpg_driver);
+
+static struct platform_device clk_device = {
+	.name		= DEVNAME,
+	.id		= 0,
+};
+
+static struct platform_device *devices[] __initdata = {
+	&clk_device,
+};
+
+int __init h8300_clk_init(int hz)
+{
+	static int master_hz;
+
+	master_hz = hz;
+	clk_device.dev.platform_data = &master_hz;
+	early_platform_add_devices(devices,
+				   ARRAY_SIZE(devices));
+	early_platform_driver_register_all(DEVNAME);
+	early_platform_driver_probe(DEVNAME, 1, 0);
+	return 0;
+}
diff --git a/include/linux/clk-provider.h b/include/linux/clk-provider.h
index 5591ea7..6059414 100644
--- a/include/linux/clk-provider.h
+++ b/include/linux/clk-provider.h
@@ -675,6 +675,18 @@ static inline void clk_writel(u32 val, u32 __iomem *reg)
 	iowrite32be(val, reg);
 }
 
+#elif IS_ENABLED(CONFIG_H8300)
+
+static inline u32 clk_readl(u32 __iomem *reg)
+{
+	return __raw_readb(reg);
+}
+
+static inline void clk_writel(u32 val, u32 __iomem *reg)
+{
+	__raw_writeb(val, reg);
+}
+
 #else	/* platform dependent I/O accessors */
 
 static inline u32 clk_readl(u32 __iomem *reg)
-- 
2.1.4

--
To unsubscribe from this list: send the line "unsubscribe linux-arch" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html




[Index of Archives]     [Linux Kernel]     [Kernel Newbies]     [x86 Platform Driver]     [Netdev]     [Linux Wireless]     [Netfilter]     [Bugtraq]     [Linux Filesystems]     [Yosemite Discussion]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Samba]     [Device Mapper]

  Powered by Linux