xref: /rk3399_rockchip-uboot/arch/arm/cpu/armv7/bcm235xx/clk-core.c (revision 65e8ce29e4efc7f1716828ec845bd294a942e1ca)
143486e4cSSteve Rae /*
243486e4cSSteve Rae  * Copyright 2013 Broadcom Corporation.
343486e4cSSteve Rae  *
443486e4cSSteve Rae  * SPDX-License-Identifier:	GPL-2.0+
543486e4cSSteve Rae  */
643486e4cSSteve Rae 
743486e4cSSteve Rae /*
843486e4cSSteve Rae  *
943486e4cSSteve Rae  * bcm235xx architecture clock framework
1043486e4cSSteve Rae  *
1143486e4cSSteve Rae  */
1243486e4cSSteve Rae 
1343486e4cSSteve Rae #include <common.h>
1443486e4cSSteve Rae #include <asm/io.h>
151221ce45SMasahiro Yamada #include <linux/errno.h>
1643486e4cSSteve Rae #include <bitfield.h>
1743486e4cSSteve Rae #include <asm/arch/sysmap.h>
1843486e4cSSteve Rae #include <asm/kona-common/clk.h>
1943486e4cSSteve Rae #include "clk-core.h"
2043486e4cSSteve Rae 
2143486e4cSSteve Rae #define CLK_WR_ACCESS_PASSWORD	0x00a5a501
2243486e4cSSteve Rae #define WR_ACCESS_OFFSET	0	/* common to all clock blocks */
2343486e4cSSteve Rae #define POLICY_CTL_GO		1	/* Load and refresh policy masks */
2443486e4cSSteve Rae #define POLICY_CTL_GO_ATL	4	/* Active Load */
2543486e4cSSteve Rae 
2643486e4cSSteve Rae /* Helper function */
clk_get_and_enable(char * clkstr)2743486e4cSSteve Rae int clk_get_and_enable(char *clkstr)
2843486e4cSSteve Rae {
2943486e4cSSteve Rae 	int ret = 0;
3043486e4cSSteve Rae 	struct clk *c;
3143486e4cSSteve Rae 
3243486e4cSSteve Rae 	debug("%s: %s\n", __func__, clkstr);
3343486e4cSSteve Rae 
3443486e4cSSteve Rae 	c = clk_get(clkstr);
3543486e4cSSteve Rae 	if (c) {
3643486e4cSSteve Rae 		ret = clk_enable(c);
3743486e4cSSteve Rae 		if (ret)
3843486e4cSSteve Rae 			return ret;
3943486e4cSSteve Rae 	} else {
4043486e4cSSteve Rae 		printf("%s: Couldn't find %s\n", __func__, clkstr);
4143486e4cSSteve Rae 		return -EINVAL;
4243486e4cSSteve Rae 	}
4343486e4cSSteve Rae 	return ret;
4443486e4cSSteve Rae }
4543486e4cSSteve Rae 
4643486e4cSSteve Rae /*
4743486e4cSSteve Rae  * Poll a register in a CCU's address space, returning when the
4843486e4cSSteve Rae  * specified bit in that register's value is set (or clear). Delay
4943486e4cSSteve Rae  * a microsecond after each read of the register. Returns true if
5043486e4cSSteve Rae  * successful, or false if we gave up trying.
5143486e4cSSteve Rae  *
5243486e4cSSteve Rae  * Caller must ensure the CCU lock is held.
5343486e4cSSteve Rae  */
5443486e4cSSteve Rae #define CLK_GATE_DELAY_USEC 2000
wait_bit(void * base,u32 offset,u32 bit,bool want)5543486e4cSSteve Rae static inline int wait_bit(void *base, u32 offset, u32 bit, bool want)
5643486e4cSSteve Rae {
5743486e4cSSteve Rae 	unsigned int tries;
5843486e4cSSteve Rae 	u32 bit_mask = 1 << bit;
5943486e4cSSteve Rae 
6043486e4cSSteve Rae 	for (tries = 0; tries < CLK_GATE_DELAY_USEC; tries++) {
6143486e4cSSteve Rae 		u32 val;
6243486e4cSSteve Rae 		bool bit_val;
6343486e4cSSteve Rae 
6443486e4cSSteve Rae 		val = readl(base + offset);
6543486e4cSSteve Rae 		bit_val = (val & bit_mask) ? 1 : 0;
6643486e4cSSteve Rae 		if (bit_val == want)
6743486e4cSSteve Rae 			return 0;	/* success */
6843486e4cSSteve Rae 		udelay(1);
6943486e4cSSteve Rae 	}
7043486e4cSSteve Rae 
7143486e4cSSteve Rae 	debug("%s: timeout on addr 0x%p, waiting for bit %d to go to %d\n",
7243486e4cSSteve Rae 	      __func__, base + offset, bit, want);
7343486e4cSSteve Rae 
7443486e4cSSteve Rae 	return -ETIMEDOUT;
7543486e4cSSteve Rae }
7643486e4cSSteve Rae 
7743486e4cSSteve Rae /* Enable a peripheral clock */
peri_clk_enable(struct clk * c,int enable)7843486e4cSSteve Rae static int peri_clk_enable(struct clk *c, int enable)
7943486e4cSSteve Rae {
8043486e4cSSteve Rae 	int ret = 0;
8143486e4cSSteve Rae 	u32 reg;
8243486e4cSSteve Rae 	struct peri_clock *peri_clk = to_peri_clk(c);
8343486e4cSSteve Rae 	struct peri_clk_data *cd = peri_clk->data;
8443486e4cSSteve Rae 	struct bcm_clk_gate *gate = &cd->gate;
8543486e4cSSteve Rae 	void *base = (void *)c->ccu_clk_mgr_base;
8643486e4cSSteve Rae 
8743486e4cSSteve Rae 
8843486e4cSSteve Rae 	debug("%s: %s\n", __func__, c->name);
8943486e4cSSteve Rae 
9043486e4cSSteve Rae 	clk_get_rate(c);	/* Make sure rate and sel are filled in */
9143486e4cSSteve Rae 
9243486e4cSSteve Rae 	/* enable access */
9343486e4cSSteve Rae 	writel(CLK_WR_ACCESS_PASSWORD, base + WR_ACCESS_OFFSET);
9443486e4cSSteve Rae 
9543486e4cSSteve Rae 	if (enable) {
9643486e4cSSteve Rae 		debug("%s %s set rate %lu div %lu sel %d parent %lu\n",
9743486e4cSSteve Rae 		      __func__, c->name, c->rate, c->div, c->sel,
9843486e4cSSteve Rae 		      c->parent->rate);
9943486e4cSSteve Rae 
10043486e4cSSteve Rae 		/*
10143486e4cSSteve Rae 		 * clkgate - only software controllable gates are
10243486e4cSSteve Rae 		 * supported by u-boot which includes all clocks
10343486e4cSSteve Rae 		 * that matter. This avoids bringing in a lot of extra
10443486e4cSSteve Rae 		 * complexity as done in the kernel framework.
10543486e4cSSteve Rae 		 */
10643486e4cSSteve Rae 		if (gate_exists(gate)) {
10743486e4cSSteve Rae 			reg = readl(base + cd->gate.offset);
10843486e4cSSteve Rae 			reg |= (1 << cd->gate.en_bit);
10943486e4cSSteve Rae 			writel(reg, base + cd->gate.offset);
11043486e4cSSteve Rae 		}
11143486e4cSSteve Rae 
11243486e4cSSteve Rae 		/* div and pll select */
11343486e4cSSteve Rae 		if (divider_exists(&cd->div)) {
11443486e4cSSteve Rae 			reg = readl(base + cd->div.offset);
11543486e4cSSteve Rae 			bitfield_replace(reg, cd->div.shift, cd->div.width,
11643486e4cSSteve Rae 					 c->div - 1);
11743486e4cSSteve Rae 			writel(reg, base + cd->div.offset);
11843486e4cSSteve Rae 		}
11943486e4cSSteve Rae 
12043486e4cSSteve Rae 		/* frequency selector */
12143486e4cSSteve Rae 		if (selector_exists(&cd->sel)) {
12243486e4cSSteve Rae 			reg = readl(base + cd->sel.offset);
12343486e4cSSteve Rae 			bitfield_replace(reg, cd->sel.shift, cd->sel.width,
12443486e4cSSteve Rae 					 c->sel);
12543486e4cSSteve Rae 			writel(reg, base + cd->sel.offset);
12643486e4cSSteve Rae 		}
12743486e4cSSteve Rae 
12843486e4cSSteve Rae 		/* trigger */
12943486e4cSSteve Rae 		if (trigger_exists(&cd->trig)) {
13043486e4cSSteve Rae 			writel((1 << cd->trig.bit), base + cd->trig.offset);
13143486e4cSSteve Rae 
13243486e4cSSteve Rae 			/* wait for trigger status bit to go to 0 */
13343486e4cSSteve Rae 			ret = wait_bit(base, cd->trig.offset, cd->trig.bit, 0);
13443486e4cSSteve Rae 			if (ret)
13543486e4cSSteve Rae 				return ret;
13643486e4cSSteve Rae 		}
13743486e4cSSteve Rae 
13843486e4cSSteve Rae 		/* wait for running (status_bit = 1) */
13943486e4cSSteve Rae 		ret = wait_bit(base, cd->gate.offset, cd->gate.status_bit, 1);
14043486e4cSSteve Rae 		if (ret)
14143486e4cSSteve Rae 			return ret;
14243486e4cSSteve Rae 	} else {
14343486e4cSSteve Rae 		debug("%s disable clock %s\n", __func__, c->name);
14443486e4cSSteve Rae 
14543486e4cSSteve Rae 		/* clkgate */
14643486e4cSSteve Rae 		reg = readl(base + cd->gate.offset);
14743486e4cSSteve Rae 		reg &= ~(1 << cd->gate.en_bit);
14843486e4cSSteve Rae 		writel(reg, base + cd->gate.offset);
14943486e4cSSteve Rae 
15043486e4cSSteve Rae 		/* wait for stop (status_bit = 0) */
15143486e4cSSteve Rae 		ret = wait_bit(base, cd->gate.offset, cd->gate.status_bit, 0);
15243486e4cSSteve Rae 	}
15343486e4cSSteve Rae 
15443486e4cSSteve Rae 	/* disable access */
15543486e4cSSteve Rae 	writel(0, base + WR_ACCESS_OFFSET);
15643486e4cSSteve Rae 
15743486e4cSSteve Rae 	return ret;
15843486e4cSSteve Rae }
15943486e4cSSteve Rae 
16043486e4cSSteve Rae /* Set the rate of a peripheral clock */
peri_clk_set_rate(struct clk * c,unsigned long rate)16143486e4cSSteve Rae static int peri_clk_set_rate(struct clk *c, unsigned long rate)
16243486e4cSSteve Rae {
16343486e4cSSteve Rae 	int ret = 0;
16443486e4cSSteve Rae 	int i;
16543486e4cSSteve Rae 	unsigned long diff;
16643486e4cSSteve Rae 	unsigned long new_rate = 0, div = 1;
16743486e4cSSteve Rae 	struct peri_clock *peri_clk = to_peri_clk(c);
16843486e4cSSteve Rae 	struct peri_clk_data *cd = peri_clk->data;
16943486e4cSSteve Rae 	const char **clock;
17043486e4cSSteve Rae 
17143486e4cSSteve Rae 	debug("%s: %s\n", __func__, c->name);
17243486e4cSSteve Rae 	diff = rate;
17343486e4cSSteve Rae 
17443486e4cSSteve Rae 	i = 0;
17543486e4cSSteve Rae 	for (clock = cd->clocks; *clock; clock++, i++) {
17643486e4cSSteve Rae 		struct refclk *ref = refclk_str_to_clk(*clock);
17743486e4cSSteve Rae 		if (!ref) {
17843486e4cSSteve Rae 			printf("%s: Lookup of %s failed\n", __func__, *clock);
17943486e4cSSteve Rae 			return -EINVAL;
18043486e4cSSteve Rae 		}
18143486e4cSSteve Rae 
18243486e4cSSteve Rae 		/* round to the new rate */
18343486e4cSSteve Rae 		div = ref->clk.rate / rate;
18443486e4cSSteve Rae 		if (div == 0)
18543486e4cSSteve Rae 			div = 1;
18643486e4cSSteve Rae 
18743486e4cSSteve Rae 		new_rate = ref->clk.rate / div;
18843486e4cSSteve Rae 
18943486e4cSSteve Rae 		/* get the min diff */
19043486e4cSSteve Rae 		if (abs(new_rate - rate) < diff) {
19143486e4cSSteve Rae 			diff = abs(new_rate - rate);
19243486e4cSSteve Rae 			c->sel = i;
19343486e4cSSteve Rae 			c->parent = &ref->clk;
19443486e4cSSteve Rae 			c->rate = new_rate;
19543486e4cSSteve Rae 			c->div = div;
19643486e4cSSteve Rae 		}
19743486e4cSSteve Rae 	}
19843486e4cSSteve Rae 
19943486e4cSSteve Rae 	debug("%s %s set rate %lu div %lu sel %d parent %lu\n", __func__,
20043486e4cSSteve Rae 	      c->name, c->rate, c->div, c->sel, c->parent->rate);
20143486e4cSSteve Rae 	return ret;
20243486e4cSSteve Rae }
20343486e4cSSteve Rae 
20443486e4cSSteve Rae /* Get the rate of a peripheral clock */
peri_clk_get_rate(struct clk * c)20543486e4cSSteve Rae static unsigned long peri_clk_get_rate(struct clk *c)
20643486e4cSSteve Rae {
20743486e4cSSteve Rae 	struct peri_clock *peri_clk = to_peri_clk(c);
20843486e4cSSteve Rae 	struct peri_clk_data *cd = peri_clk->data;
20943486e4cSSteve Rae 	void *base = (void *)c->ccu_clk_mgr_base;
21043486e4cSSteve Rae 	int div = 1;
21143486e4cSSteve Rae 	const char **clock;
21243486e4cSSteve Rae 	struct refclk *ref;
21343486e4cSSteve Rae 	u32 reg;
21443486e4cSSteve Rae 
21543486e4cSSteve Rae 	debug("%s: %s\n", __func__, c->name);
21643486e4cSSteve Rae 	if (selector_exists(&cd->sel)) {
21743486e4cSSteve Rae 		reg = readl(base + cd->sel.offset);
21843486e4cSSteve Rae 		c->sel = bitfield_extract(reg, cd->sel.shift, cd->sel.width);
21943486e4cSSteve Rae 	} else {
22043486e4cSSteve Rae 		/*
22143486e4cSSteve Rae 		 * For peri clocks that don't have a selector, the single
22243486e4cSSteve Rae 		 * reference clock will always exist at index 0.
22343486e4cSSteve Rae 		 */
22443486e4cSSteve Rae 		c->sel = 0;
22543486e4cSSteve Rae 	}
22643486e4cSSteve Rae 
22743486e4cSSteve Rae 	if (divider_exists(&cd->div)) {
22843486e4cSSteve Rae 		reg = readl(base + cd->div.offset);
22943486e4cSSteve Rae 		div = bitfield_extract(reg, cd->div.shift, cd->div.width);
23043486e4cSSteve Rae 		div += 1;
23143486e4cSSteve Rae 	}
23243486e4cSSteve Rae 
23343486e4cSSteve Rae 	clock = cd->clocks;
23443486e4cSSteve Rae 	ref = refclk_str_to_clk(clock[c->sel]);
23543486e4cSSteve Rae 	if (!ref) {
23643486e4cSSteve Rae 		printf("%s: Can't lookup %s\n", __func__, clock[c->sel]);
23743486e4cSSteve Rae 		return 0;
23843486e4cSSteve Rae 	}
23943486e4cSSteve Rae 
24043486e4cSSteve Rae 	c->parent = &ref->clk;
24143486e4cSSteve Rae 	c->div = div;
24243486e4cSSteve Rae 	c->rate = c->parent->rate / c->div;
24343486e4cSSteve Rae 	debug("%s parent rate %lu div %d sel %d rate %lu\n", __func__,
24443486e4cSSteve Rae 	      c->parent->rate, div, c->sel, c->rate);
24543486e4cSSteve Rae 
24643486e4cSSteve Rae 	return c->rate;
24743486e4cSSteve Rae }
24843486e4cSSteve Rae 
24943486e4cSSteve Rae /* Peripheral clock operations */
25043486e4cSSteve Rae struct clk_ops peri_clk_ops = {
25143486e4cSSteve Rae 	.enable = peri_clk_enable,
25243486e4cSSteve Rae 	.set_rate = peri_clk_set_rate,
25343486e4cSSteve Rae 	.get_rate = peri_clk_get_rate,
25443486e4cSSteve Rae };
25543486e4cSSteve Rae 
25643486e4cSSteve Rae /* Enable a CCU clock */
ccu_clk_enable(struct clk * c,int enable)25743486e4cSSteve Rae static int ccu_clk_enable(struct clk *c, int enable)
25843486e4cSSteve Rae {
25943486e4cSSteve Rae 	struct ccu_clock *ccu_clk = to_ccu_clk(c);
26043486e4cSSteve Rae 	void *base = (void *)c->ccu_clk_mgr_base;
26143486e4cSSteve Rae 	int ret = 0;
26243486e4cSSteve Rae 	u32 reg;
26343486e4cSSteve Rae 
26443486e4cSSteve Rae 	debug("%s: %s\n", __func__, c->name);
26543486e4cSSteve Rae 	if (!enable)
26643486e4cSSteve Rae 		return -EINVAL;	/* CCU clock cannot shutdown */
26743486e4cSSteve Rae 
26843486e4cSSteve Rae 	/* enable access */
26943486e4cSSteve Rae 	writel(CLK_WR_ACCESS_PASSWORD, base + WR_ACCESS_OFFSET);
27043486e4cSSteve Rae 
27143486e4cSSteve Rae 	/* config enable for policy engine */
27243486e4cSSteve Rae 	writel(1, base + ccu_clk->lvm_en_offset);
27343486e4cSSteve Rae 
27443486e4cSSteve Rae 	/* wait for bit to go to 0 */
27543486e4cSSteve Rae 	ret = wait_bit(base, ccu_clk->lvm_en_offset, 0, 0);
27643486e4cSSteve Rae 	if (ret)
27743486e4cSSteve Rae 		return ret;
27843486e4cSSteve Rae 
27943486e4cSSteve Rae 	/* freq ID */
28043486e4cSSteve Rae 	if (!ccu_clk->freq_bit_shift)
28143486e4cSSteve Rae 		ccu_clk->freq_bit_shift = 8;
28243486e4cSSteve Rae 
28343486e4cSSteve Rae 	/* Set frequency id for each of the 4 policies */
28443486e4cSSteve Rae 	reg = ccu_clk->freq_id |
28543486e4cSSteve Rae 	    (ccu_clk->freq_id << (ccu_clk->freq_bit_shift)) |
28643486e4cSSteve Rae 	    (ccu_clk->freq_id << (ccu_clk->freq_bit_shift * 2)) |
28743486e4cSSteve Rae 	    (ccu_clk->freq_id << (ccu_clk->freq_bit_shift * 3));
28843486e4cSSteve Rae 	writel(reg, base + ccu_clk->policy_freq_offset);
28943486e4cSSteve Rae 
29043486e4cSSteve Rae 	/* enable all clock mask */
29143486e4cSSteve Rae 	writel(0x7fffffff, base + ccu_clk->policy0_mask_offset);
29243486e4cSSteve Rae 	writel(0x7fffffff, base + ccu_clk->policy1_mask_offset);
29343486e4cSSteve Rae 	writel(0x7fffffff, base + ccu_clk->policy2_mask_offset);
29443486e4cSSteve Rae 	writel(0x7fffffff, base + ccu_clk->policy3_mask_offset);
29543486e4cSSteve Rae 
29643486e4cSSteve Rae 	if (ccu_clk->num_policy_masks == 2) {
29743486e4cSSteve Rae 		writel(0x7fffffff, base + ccu_clk->policy0_mask2_offset);
29843486e4cSSteve Rae 		writel(0x7fffffff, base + ccu_clk->policy1_mask2_offset);
29943486e4cSSteve Rae 		writel(0x7fffffff, base + ccu_clk->policy2_mask2_offset);
30043486e4cSSteve Rae 		writel(0x7fffffff, base + ccu_clk->policy3_mask2_offset);
30143486e4cSSteve Rae 	}
30243486e4cSSteve Rae 
30343486e4cSSteve Rae 	/* start policy engine */
30443486e4cSSteve Rae 	reg = readl(base + ccu_clk->policy_ctl_offset);
30543486e4cSSteve Rae 	reg |= (POLICY_CTL_GO + POLICY_CTL_GO_ATL);
30643486e4cSSteve Rae 	writel(reg, base + ccu_clk->policy_ctl_offset);
30743486e4cSSteve Rae 
30843486e4cSSteve Rae 	/* wait till started */
30943486e4cSSteve Rae 	ret = wait_bit(base, ccu_clk->policy_ctl_offset, 0, 0);
31043486e4cSSteve Rae 	if (ret)
31143486e4cSSteve Rae 		return ret;
31243486e4cSSteve Rae 
31343486e4cSSteve Rae 	/* disable access */
31443486e4cSSteve Rae 	writel(0, base + WR_ACCESS_OFFSET);
31543486e4cSSteve Rae 
31643486e4cSSteve Rae 	return ret;
31743486e4cSSteve Rae }
31843486e4cSSteve Rae 
31943486e4cSSteve Rae /* Get the CCU clock rate */
ccu_clk_get_rate(struct clk * c)32043486e4cSSteve Rae static unsigned long ccu_clk_get_rate(struct clk *c)
32143486e4cSSteve Rae {
32243486e4cSSteve Rae 	struct ccu_clock *ccu_clk = to_ccu_clk(c);
32343486e4cSSteve Rae 	debug("%s: %s\n", __func__, c->name);
32443486e4cSSteve Rae 	c->rate = ccu_clk->freq_tbl[ccu_clk->freq_id];
32543486e4cSSteve Rae 	return c->rate;
32643486e4cSSteve Rae }
32743486e4cSSteve Rae 
32843486e4cSSteve Rae /* CCU clock operations */
32943486e4cSSteve Rae struct clk_ops ccu_clk_ops = {
33043486e4cSSteve Rae 	.enable = ccu_clk_enable,
33143486e4cSSteve Rae 	.get_rate = ccu_clk_get_rate,
33243486e4cSSteve Rae };
33343486e4cSSteve Rae 
33443486e4cSSteve Rae /* Enable a bus clock */
bus_clk_enable(struct clk * c,int enable)33543486e4cSSteve Rae static int bus_clk_enable(struct clk *c, int enable)
33643486e4cSSteve Rae {
33743486e4cSSteve Rae 	struct bus_clock *bus_clk = to_bus_clk(c);
33843486e4cSSteve Rae 	struct bus_clk_data *cd = bus_clk->data;
33943486e4cSSteve Rae 	void *base = (void *)c->ccu_clk_mgr_base;
34043486e4cSSteve Rae 	int ret = 0;
34143486e4cSSteve Rae 	u32 reg;
34243486e4cSSteve Rae 
34343486e4cSSteve Rae 	debug("%s: %s\n", __func__, c->name);
34443486e4cSSteve Rae 	/* enable access */
34543486e4cSSteve Rae 	writel(CLK_WR_ACCESS_PASSWORD, base + WR_ACCESS_OFFSET);
34643486e4cSSteve Rae 
34743486e4cSSteve Rae 	/* enable gating */
34843486e4cSSteve Rae 	reg = readl(base + cd->gate.offset);
34943486e4cSSteve Rae 	if (!!(reg & (1 << cd->gate.status_bit)) == !!enable)
35043486e4cSSteve Rae 		debug("%s already %s\n", c->name,
35143486e4cSSteve Rae 		      enable ? "enabled" : "disabled");
35243486e4cSSteve Rae 	else {
35343486e4cSSteve Rae 		int want = (enable) ? 1 : 0;
35443486e4cSSteve Rae 		reg |= (1 << cd->gate.hw_sw_sel_bit);
35543486e4cSSteve Rae 
35643486e4cSSteve Rae 		if (enable)
35743486e4cSSteve Rae 			reg |= (1 << cd->gate.en_bit);
35843486e4cSSteve Rae 		else
35943486e4cSSteve Rae 			reg &= ~(1 << cd->gate.en_bit);
36043486e4cSSteve Rae 
36143486e4cSSteve Rae 		writel(reg, base + cd->gate.offset);
36243486e4cSSteve Rae 		ret = wait_bit(base, cd->gate.offset, cd->gate.status_bit,
36343486e4cSSteve Rae 			       want);
36443486e4cSSteve Rae 		if (ret)
36543486e4cSSteve Rae 			return ret;
36643486e4cSSteve Rae 	}
36743486e4cSSteve Rae 
36843486e4cSSteve Rae 	/* disable access */
36943486e4cSSteve Rae 	writel(0, base + WR_ACCESS_OFFSET);
37043486e4cSSteve Rae 
37143486e4cSSteve Rae 	return ret;
37243486e4cSSteve Rae }
37343486e4cSSteve Rae 
37443486e4cSSteve Rae /* Get the rate of a bus clock */
bus_clk_get_rate(struct clk * c)37543486e4cSSteve Rae static unsigned long bus_clk_get_rate(struct clk *c)
37643486e4cSSteve Rae {
37743486e4cSSteve Rae 	struct bus_clock *bus_clk = to_bus_clk(c);
37843486e4cSSteve Rae 	struct ccu_clock *ccu_clk;
37943486e4cSSteve Rae 
38043486e4cSSteve Rae 	debug("%s: %s\n", __func__, c->name);
38143486e4cSSteve Rae 	ccu_clk = to_ccu_clk(c->parent);
38243486e4cSSteve Rae 
38343486e4cSSteve Rae 	c->rate = bus_clk->freq_tbl[ccu_clk->freq_id];
38443486e4cSSteve Rae 	c->div = ccu_clk->freq_tbl[ccu_clk->freq_id] / c->rate;
38543486e4cSSteve Rae 	return c->rate;
38643486e4cSSteve Rae }
38743486e4cSSteve Rae 
38843486e4cSSteve Rae /* Bus clock operations */
38943486e4cSSteve Rae struct clk_ops bus_clk_ops = {
39043486e4cSSteve Rae 	.enable = bus_clk_enable,
39143486e4cSSteve Rae 	.get_rate = bus_clk_get_rate,
39243486e4cSSteve Rae };
39343486e4cSSteve Rae 
39443486e4cSSteve Rae /* Enable a reference clock */
ref_clk_enable(struct clk * c,int enable)39543486e4cSSteve Rae static int ref_clk_enable(struct clk *c, int enable)
39643486e4cSSteve Rae {
39743486e4cSSteve Rae 	debug("%s: %s\n", __func__, c->name);
39843486e4cSSteve Rae 	return 0;
39943486e4cSSteve Rae }
40043486e4cSSteve Rae 
40143486e4cSSteve Rae /* Reference clock operations */
40243486e4cSSteve Rae struct clk_ops ref_clk_ops = {
40343486e4cSSteve Rae 	.enable = ref_clk_enable,
40443486e4cSSteve Rae };
40543486e4cSSteve Rae 
40643486e4cSSteve Rae /*
40743486e4cSSteve Rae  * clk.h implementation follows
40843486e4cSSteve Rae  */
40943486e4cSSteve Rae 
41043486e4cSSteve Rae /* Initialize the clock framework */
clk_init(void)41143486e4cSSteve Rae int clk_init(void)
41243486e4cSSteve Rae {
41343486e4cSSteve Rae 	debug("%s:\n", __func__);
41443486e4cSSteve Rae 	return 0;
41543486e4cSSteve Rae }
41643486e4cSSteve Rae 
41743486e4cSSteve Rae /* Get a clock handle, give a name string */
clk_get(const char * con_id)41843486e4cSSteve Rae struct clk *clk_get(const char *con_id)
41943486e4cSSteve Rae {
42043486e4cSSteve Rae 	int i;
42143486e4cSSteve Rae 	struct clk_lookup *clk_tblp;
42243486e4cSSteve Rae 
42343486e4cSSteve Rae 	debug("%s: %s\n", __func__, con_id);
42443486e4cSSteve Rae 
42543486e4cSSteve Rae 	clk_tblp = arch_clk_tbl;
42643486e4cSSteve Rae 	for (i = 0; i < arch_clk_tbl_array_size; i++, clk_tblp++) {
42743486e4cSSteve Rae 		if (clk_tblp->con_id) {
42843486e4cSSteve Rae 			if (!con_id || strcmp(clk_tblp->con_id, con_id))
42943486e4cSSteve Rae 				continue;
43043486e4cSSteve Rae 			return clk_tblp->clk;
43143486e4cSSteve Rae 		}
43243486e4cSSteve Rae 	}
43343486e4cSSteve Rae 	return NULL;
43443486e4cSSteve Rae }
43543486e4cSSteve Rae 
43643486e4cSSteve Rae /* Enable a clock */
clk_enable(struct clk * c)43743486e4cSSteve Rae int clk_enable(struct clk *c)
43843486e4cSSteve Rae {
43943486e4cSSteve Rae 	int ret = 0;
44043486e4cSSteve Rae 
44143486e4cSSteve Rae 	debug("%s: %s\n", __func__, c->name);
44243486e4cSSteve Rae 	if (!c->ops || !c->ops->enable)
44343486e4cSSteve Rae 		return -1;
44443486e4cSSteve Rae 
44543486e4cSSteve Rae 	/* enable parent clock first */
44643486e4cSSteve Rae 	if (c->parent)
44743486e4cSSteve Rae 		ret = clk_enable(c->parent);
44843486e4cSSteve Rae 
44943486e4cSSteve Rae 	if (ret)
45043486e4cSSteve Rae 		return ret;
45143486e4cSSteve Rae 
4528ada4e0eSSteve Rae 	if (!c->use_cnt)
45343486e4cSSteve Rae 		ret = c->ops->enable(c, 1);
4548ada4e0eSSteve Rae 	c->use_cnt++;
45543486e4cSSteve Rae 
45643486e4cSSteve Rae 	return ret;
45743486e4cSSteve Rae }
45843486e4cSSteve Rae 
45943486e4cSSteve Rae /* Disable a clock */
clk_disable(struct clk * c)46043486e4cSSteve Rae void clk_disable(struct clk *c)
46143486e4cSSteve Rae {
46243486e4cSSteve Rae 	debug("%s: %s\n", __func__, c->name);
46343486e4cSSteve Rae 	if (!c->ops || !c->ops->enable)
46443486e4cSSteve Rae 		return;
46543486e4cSSteve Rae 
4668ada4e0eSSteve Rae 	if (c->use_cnt > 0) {
46743486e4cSSteve Rae 		c->use_cnt--;
4688ada4e0eSSteve Rae 		if (c->use_cnt == 0)
46943486e4cSSteve Rae 			c->ops->enable(c, 0);
47043486e4cSSteve Rae 	}
47143486e4cSSteve Rae 
47243486e4cSSteve Rae 	/* disable parent */
47343486e4cSSteve Rae 	if (c->parent)
47443486e4cSSteve Rae 		clk_disable(c->parent);
47543486e4cSSteve Rae }
47643486e4cSSteve Rae 
47743486e4cSSteve Rae /* Get the clock rate */
clk_get_rate(struct clk * c)47843486e4cSSteve Rae unsigned long clk_get_rate(struct clk *c)
47943486e4cSSteve Rae {
48043486e4cSSteve Rae 	unsigned long rate;
48143486e4cSSteve Rae 
48243486e4cSSteve Rae 	if (!c || !c->ops || !c->ops->get_rate)
48343486e4cSSteve Rae 		return 0;
484b69a0849Sxypron.glpk@gmx.de 	debug("%s: %s\n", __func__, c->name);
48543486e4cSSteve Rae 
48643486e4cSSteve Rae 	rate = c->ops->get_rate(c);
48743486e4cSSteve Rae 	debug("%s: rate = %ld\n", __func__, rate);
48843486e4cSSteve Rae 	return rate;
48943486e4cSSteve Rae }
49043486e4cSSteve Rae 
49143486e4cSSteve Rae /* Set the clock rate */
clk_set_rate(struct clk * c,unsigned long rate)49243486e4cSSteve Rae int clk_set_rate(struct clk *c, unsigned long rate)
49343486e4cSSteve Rae {
49443486e4cSSteve Rae 	int ret;
49543486e4cSSteve Rae 
49643486e4cSSteve Rae 	if (!c || !c->ops || !c->ops->set_rate)
49743486e4cSSteve Rae 		return -EINVAL;
498*65e8ce29Sxypron.glpk@gmx.de 	debug("%s: %s rate=%ld\n", __func__, c->name, rate);
49943486e4cSSteve Rae 
50043486e4cSSteve Rae 	if (c->use_cnt)
50143486e4cSSteve Rae 		return -EINVAL;
50243486e4cSSteve Rae 
50343486e4cSSteve Rae 	ret = c->ops->set_rate(c, rate);
50443486e4cSSteve Rae 
50543486e4cSSteve Rae 	return ret;
50643486e4cSSteve Rae }
50743486e4cSSteve Rae 
50843486e4cSSteve Rae /* Not required for this arch */
50943486e4cSSteve Rae /*
51043486e4cSSteve Rae long clk_round_rate(struct clk *clk, unsigned long rate);
51143486e4cSSteve Rae int clk_set_parent(struct clk *clk, struct clk *parent);
51243486e4cSSteve Rae struct clk *clk_get_parent(struct clk *clk);
51343486e4cSSteve Rae */
514