|  | @@ -0,0 +1,162 @@
 | 
	
		
			
				|  |  | +/*
 | 
	
		
			
				|  |  | + * OMAP3/4 - specific DPLL control functions
 | 
	
		
			
				|  |  | + *
 | 
	
		
			
				|  |  | + * Copyright (C) 2009-2010 Texas Instruments, Inc.
 | 
	
		
			
				|  |  | + * Copyright (C) 2009-2010 Nokia Corporation
 | 
	
		
			
				|  |  | + *
 | 
	
		
			
				|  |  | + * Written by Paul Walmsley
 | 
	
		
			
				|  |  | + * Testing and integration fixes by Jouni Högander
 | 
	
		
			
				|  |  | + *
 | 
	
		
			
				|  |  | + * 36xx support added by Vishwanath BS, Richard Woodruff, and Nishanth
 | 
	
		
			
				|  |  | + * Menon
 | 
	
		
			
				|  |  | + *
 | 
	
		
			
				|  |  | + * Parts of this code are based on code written by
 | 
	
		
			
				|  |  | + * Richard Woodruff, Tony Lindgren, Tuukka Tikkanen, Karthik Dasu
 | 
	
		
			
				|  |  | + *
 | 
	
		
			
				|  |  | + * 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 <linux/kernel.h>
 | 
	
		
			
				|  |  | +#include <linux/device.h>
 | 
	
		
			
				|  |  | +#include <linux/list.h>
 | 
	
		
			
				|  |  | +#include <linux/errno.h>
 | 
	
		
			
				|  |  | +#include <linux/delay.h>
 | 
	
		
			
				|  |  | +#include <linux/clk.h>
 | 
	
		
			
				|  |  | +#include <linux/io.h>
 | 
	
		
			
				|  |  | +#include <linux/bitops.h>
 | 
	
		
			
				|  |  | +#include <linux/clkdev.h>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +#include "soc.h"
 | 
	
		
			
				|  |  | +#include "clockdomain.h"
 | 
	
		
			
				|  |  | +#include "clock.h"
 | 
	
		
			
				|  |  | +#include "cm2xxx_3xxx.h"
 | 
	
		
			
				|  |  | +#include "cm-regbits-34xx.h"
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +/* CM_AUTOIDLE_PLL*.AUTO_* bit values */
 | 
	
		
			
				|  |  | +#define DPLL_AUTOIDLE_DISABLE			0x0
 | 
	
		
			
				|  |  | +#define DPLL_AUTOIDLE_LOW_POWER_STOP		0x1
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +#define MAX_DPLL_WAIT_TRIES		1000000
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +/* Private functions */
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +/* _omap3_dpll_write_clken - write clken_bits arg to a DPLL's enable bits */
 | 
	
		
			
				|  |  | +static void _omap3_dpll_write_clken(struct clk_hw_omap *clk, u8 clken_bits)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	const struct dpll_data *dd;
 | 
	
		
			
				|  |  | +	u32 v;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	dd = clk->dpll_data;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	v = __raw_readl(dd->control_reg);
 | 
	
		
			
				|  |  | +	v &= ~dd->enable_mask;
 | 
	
		
			
				|  |  | +	v |= clken_bits << __ffs(dd->enable_mask);
 | 
	
		
			
				|  |  | +	__raw_writel(v, dd->control_reg);
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +/* _omap3_wait_dpll_status: wait for a DPLL to enter a specific state */
 | 
	
		
			
				|  |  | +static int _omap3_wait_dpll_status(struct clk_hw_omap *clk, u8 state)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	const struct dpll_data *dd;
 | 
	
		
			
				|  |  | +	int i = 0;
 | 
	
		
			
				|  |  | +	int ret = -EINVAL;
 | 
	
		
			
				|  |  | +	const char *clk_name;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	dd = clk->dpll_data;
 | 
	
		
			
				|  |  | +	clk_name = __clk_get_name(clk->hw.clk);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	state <<= __ffs(dd->idlest_mask);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	while (((__raw_readl(dd->idlest_reg) & dd->idlest_mask) != state) &&
 | 
	
		
			
				|  |  | +	       i < MAX_DPLL_WAIT_TRIES) {
 | 
	
		
			
				|  |  | +		i++;
 | 
	
		
			
				|  |  | +		udelay(1);
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	if (i == MAX_DPLL_WAIT_TRIES) {
 | 
	
		
			
				|  |  | +		printk(KERN_ERR "clock: %s failed transition to '%s'\n",
 | 
	
		
			
				|  |  | +		       clk_name, (state) ? "locked" : "bypassed");
 | 
	
		
			
				|  |  | +	} else {
 | 
	
		
			
				|  |  | +		pr_debug("clock: %s transition to '%s' in %d loops\n",
 | 
	
		
			
				|  |  | +			 clk_name, (state) ? "locked" : "bypassed", i);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +		ret = 0;
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	return ret;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +/* From 3430 TRM ES2 4.7.6.2 */
 | 
	
		
			
				|  |  | +static u16 _omap3_dpll_compute_freqsel(struct clk_hw_omap *clk, u8 n)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	unsigned long fint;
 | 
	
		
			
				|  |  | +	u16 f = 0;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	fint = __clk_get_rate(clk->dpll_data->clk_ref) / n;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	pr_debug("clock: fint is %lu\n", fint);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	if (fint >= 750000 && fint <= 1000000)
 | 
	
		
			
				|  |  | +		f = 0x3;
 | 
	
		
			
				|  |  | +	else if (fint > 1000000 && fint <= 1250000)
 | 
	
		
			
				|  |  | +		f = 0x4;
 | 
	
		
			
				|  |  | +	else if (fint > 1250000 && fint <= 1500000)
 | 
	
		
			
				|  |  | +		f = 0x5;
 | 
	
		
			
				|  |  | +	else if (fint > 1500000 && fint <= 1750000)
 | 
	
		
			
				|  |  | +		f = 0x6;
 | 
	
		
			
				|  |  | +	else if (fint > 1750000 && fint <= 2100000)
 | 
	
		
			
				|  |  | +		f = 0x7;
 | 
	
		
			
				|  |  | +	else if (fint > 7500000 && fint <= 10000000)
 | 
	
		
			
				|  |  | +		f = 0xB;
 | 
	
		
			
				|  |  | +	else if (fint > 10000000 && fint <= 12500000)
 | 
	
		
			
				|  |  | +		f = 0xC;
 | 
	
		
			
				|  |  | +	else if (fint > 12500000 && fint <= 15000000)
 | 
	
		
			
				|  |  | +		f = 0xD;
 | 
	
		
			
				|  |  | +	else if (fint > 15000000 && fint <= 17500000)
 | 
	
		
			
				|  |  | +		f = 0xE;
 | 
	
		
			
				|  |  | +	else if (fint > 17500000 && fint <= 21000000)
 | 
	
		
			
				|  |  | +		f = 0xF;
 | 
	
		
			
				|  |  | +	else
 | 
	
		
			
				|  |  | +		pr_debug("clock: unknown freqsel setting for %d\n", n);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	return f;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +/*
 | 
	
		
			
				|  |  | + * _omap3_noncore_dpll_lock - instruct a DPLL to lock and wait for readiness
 | 
	
		
			
				|  |  | + * @clk: pointer to a DPLL struct clk
 | 
	
		
			
				|  |  | + *
 | 
	
		
			
				|  |  | + * Instructs a non-CORE DPLL to lock.  Waits for the DPLL to report
 | 
	
		
			
				|  |  | + * readiness before returning.  Will save and restore the DPLL's
 | 
	
		
			
				|  |  | + * autoidle state across the enable, per the CDP code.  If the DPLL
 | 
	
		
			
				|  |  | + * locked successfully, return 0; if the DPLL did not lock in the time
 | 
	
		
			
				|  |  | + * allotted, or DPLL3 was passed in, return -EINVAL.
 | 
	
		
			
				|  |  | + */
 | 
	
		
			
				|  |  | +static int _omap3_noncore_dpll_lock(struct clk_hw_omap *clk)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	const struct dpll_data *dd;
 | 
	
		
			
				|  |  | +	u8 ai;
 | 
	
		
			
				|  |  | +	u8 state = 1;
 | 
	
		
			
				|  |  | +	int r = 0;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	pr_debug("clock: locking DPLL %s\n", __clk_get_name(clk->hw.clk));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	dd = clk->dpll_data;
 | 
	
		
			
				|  |  | +	state <<= __ffs(dd->idlest_mask);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	/* Check if already locked */
 | 
	
		
			
				|  |  | +	if ((__raw_readl(dd->idlest_reg) & dd->idlest_mask) == state)
 | 
	
		
			
				|  |  | +		goto done;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	ai = omap3_dpll_autoidle_read(clk);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	if (ai)
 | 
	
		
			
				|  |  | +		omap3_dpll_deny_idle(clk);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	_omap3_dpll_write_clken(clk, DPLL_LOCKED);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	r = _omap3_wait_dpll_status(clk, 1);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	if (ai)
 |