2010-01-26 20:13:06 -07:00
/*
* OMAP2xxx APLL clock control functions
*
* Copyright ( C ) 2005 - 2008 Texas Instruments , Inc .
* Copyright ( C ) 2004 - 2010 Nokia Corporation
*
* Contacts :
* Richard Woodruff < r - woodruff2 @ ti . com >
* Paul Walmsley
*
* Based on earlier work by Tuukka Tikkanen , Tony Lindgren ,
* Gordon McNutt and RidgeRun , Inc .
*
* 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 .
*/
# undef DEBUG
# include <linux/kernel.h>
# include <linux/clk.h>
# include <linux/io.h>
# include <plat/clock.h>
# include <plat/prcm.h>
# include "clock.h"
# include "clock2xxx.h"
2010-12-21 15:30:55 -07:00
# include "cm2xxx_3xxx.h"
2010-01-26 20:13:06 -07:00
# include "cm-regbits-24xx.h"
/* CM_CLKEN_PLL.EN_{54,96}M_PLL options (24XX) */
# define EN_APLL_STOPPED 0
# define EN_APLL_LOCKED 3
/* CM_CLKSEL1_PLL.APLLS_CLKIN options (24XX) */
# define APLLS_CLKIN_19_2MHZ 0
# define APLLS_CLKIN_13MHZ 2
# define APLLS_CLKIN_12MHZ 3
OMAP2 clock: split OMAP2420, OMAP2430 clock data into their own files
In preparation for multi-OMAP2 kernels, split
mach-omap2/clock2xxx_data.c into mach-omap2/clock2420_data.c and
mach-omap2/clock2430_data.c. 2430 uses a different device space
physical memory layout than past or future OMAPs, and we use a
different virtual memory layout as well, which causes trouble for
architecture-level code/data that tries to support both. We tried
using offsets from the virtual base last year, but those patches never
made it upstream; so after some discussion with Tony about the best
all-around approach, we'll just grit our teeth and duplicate the
structures. The maintenance advantages of a single kernel config that
can compile and boot on OMAP2, 3, and 4 platforms are simply too
compelling.
This approach does have some nice benefits beyond multi-OMAP 2 kernel
support. The runtime size of OMAP2420-specific and OMAP2430-specific
kernels is smaller, since unused clocks for the other OMAP2 chip will
no longer be compiled in. (At some point we will mark the clock data
__initdata and allocate it during registration, which will eliminate
the runtime memory advantage.) It also makes the clock trees slightly
easier to read, since 2420-specific and 2430-specific clocks are no
longer mixed together.
This patch also splits 2430-specific clock code into its own file,
mach-omap2/clock2430.c, which is only compiled in for 2430 builds -
mostly for organizational clarity.
While here, fix a bug in the OMAP2430 clock tree: "emul_ck" was
incorrectly marked as being 2420-only, when actually it is present on
both OMAP2420 and OMAP2430.
Thanks to Tony for some good discussions about how to approach this
problem.
Signed-off-by: Paul Walmsley <paul@pwsan.com>
Cc: Tony Lindgren <tony@atomide.com>
Cc: Richard Woodruff <r-woodruff2@ti.com>
2010-02-22 22:09:22 -07:00
void __iomem * cm_idlest_pll ;
2010-01-26 20:13:06 -07:00
/* Private functions */
/* Enable an APLL if off */
static int omap2_clk_apll_enable ( struct clk * clk , u32 status_mask )
{
u32 cval , apll_mask ;
apll_mask = EN_APLL_LOCKED < < clk - > enable_bit ;
2010-12-21 21:05:14 -07:00
cval = omap2_cm_read_mod_reg ( PLL_MOD , CM_CLKEN ) ;
2010-01-26 20:13:06 -07:00
if ( ( cval & apll_mask ) = = apll_mask )
return 0 ; /* apll already enabled */
cval & = ~ apll_mask ;
cval | = apll_mask ;
2010-12-21 21:05:14 -07:00
omap2_cm_write_mod_reg ( cval , PLL_MOD , CM_CLKEN ) ;
2010-01-26 20:13:06 -07:00
OMAP2 clock: split OMAP2420, OMAP2430 clock data into their own files
In preparation for multi-OMAP2 kernels, split
mach-omap2/clock2xxx_data.c into mach-omap2/clock2420_data.c and
mach-omap2/clock2430_data.c. 2430 uses a different device space
physical memory layout than past or future OMAPs, and we use a
different virtual memory layout as well, which causes trouble for
architecture-level code/data that tries to support both. We tried
using offsets from the virtual base last year, but those patches never
made it upstream; so after some discussion with Tony about the best
all-around approach, we'll just grit our teeth and duplicate the
structures. The maintenance advantages of a single kernel config that
can compile and boot on OMAP2, 3, and 4 platforms are simply too
compelling.
This approach does have some nice benefits beyond multi-OMAP 2 kernel
support. The runtime size of OMAP2420-specific and OMAP2430-specific
kernels is smaller, since unused clocks for the other OMAP2 chip will
no longer be compiled in. (At some point we will mark the clock data
__initdata and allocate it during registration, which will eliminate
the runtime memory advantage.) It also makes the clock trees slightly
easier to read, since 2420-specific and 2430-specific clocks are no
longer mixed together.
This patch also splits 2430-specific clock code into its own file,
mach-omap2/clock2430.c, which is only compiled in for 2430 builds -
mostly for organizational clarity.
While here, fix a bug in the OMAP2430 clock tree: "emul_ck" was
incorrectly marked as being 2420-only, when actually it is present on
both OMAP2420 and OMAP2430.
Thanks to Tony for some good discussions about how to approach this
problem.
Signed-off-by: Paul Walmsley <paul@pwsan.com>
Cc: Tony Lindgren <tony@atomide.com>
Cc: Richard Woodruff <r-woodruff2@ti.com>
2010-02-22 22:09:22 -07:00
omap2_cm_wait_idlest ( cm_idlest_pll , status_mask ,
2010-02-24 12:05:54 -07:00
OMAP24XX_CM_IDLEST_VAL , clk - > name ) ;
2010-01-26 20:13:06 -07:00
/*
* REVISIT : Should we return an error code if omap2_wait_clock_ready ( )
* fails ?
*/
return 0 ;
}
static int omap2_clk_apll96_enable ( struct clk * clk )
{
2010-05-20 12:31:04 -06:00
return omap2_clk_apll_enable ( clk , OMAP24XX_ST_96M_APLL_MASK ) ;
2010-01-26 20:13:06 -07:00
}
static int omap2_clk_apll54_enable ( struct clk * clk )
{
2010-05-20 12:31:04 -06:00
return omap2_clk_apll_enable ( clk , OMAP24XX_ST_54M_APLL_MASK ) ;
2010-01-26 20:13:06 -07:00
}
2011-02-25 15:39:27 -07:00
static void _apll96_allow_idle ( struct clk * clk )
{
omap2xxx_cm_set_apll96_auto_low_power_stop ( ) ;
}
static void _apll96_deny_idle ( struct clk * clk )
{
omap2xxx_cm_set_apll96_disable_autoidle ( ) ;
}
static void _apll54_allow_idle ( struct clk * clk )
{
omap2xxx_cm_set_apll54_auto_low_power_stop ( ) ;
}
static void _apll54_deny_idle ( struct clk * clk )
{
omap2xxx_cm_set_apll54_disable_autoidle ( ) ;
}
2010-01-26 20:13:06 -07:00
/* Stop APLL */
static void omap2_clk_apll_disable ( struct clk * clk )
{
u32 cval ;
2010-12-21 21:05:14 -07:00
cval = omap2_cm_read_mod_reg ( PLL_MOD , CM_CLKEN ) ;
2010-01-26 20:13:06 -07:00
cval & = ~ ( EN_APLL_LOCKED < < clk - > enable_bit ) ;
2010-12-21 21:05:14 -07:00
omap2_cm_write_mod_reg ( cval , PLL_MOD , CM_CLKEN ) ;
2010-01-26 20:13:06 -07:00
}
/* Public data */
const struct clkops clkops_apll96 = {
. enable = omap2_clk_apll96_enable ,
. disable = omap2_clk_apll_disable ,
2011-02-25 15:39:27 -07:00
. allow_idle = _apll96_allow_idle ,
. deny_idle = _apll96_deny_idle ,
2010-01-26 20:13:06 -07:00
} ;
const struct clkops clkops_apll54 = {
. enable = omap2_clk_apll54_enable ,
. disable = omap2_clk_apll_disable ,
2011-02-25 15:39:27 -07:00
. allow_idle = _apll54_allow_idle ,
. deny_idle = _apll54_deny_idle ,
2010-01-26 20:13:06 -07:00
} ;
/* Public functions */
u32 omap2xxx_get_apll_clkin ( void )
{
u32 aplls , srate = 0 ;
2010-12-21 21:05:14 -07:00
aplls = omap2_cm_read_mod_reg ( PLL_MOD , CM_CLKSEL1 ) ;
2010-01-26 20:13:06 -07:00
aplls & = OMAP24XX_APLLS_CLKIN_MASK ;
aplls > > = OMAP24XX_APLLS_CLKIN_SHIFT ;
if ( aplls = = APLLS_CLKIN_19_2MHZ )
srate = 19200000 ;
else if ( aplls = = APLLS_CLKIN_13MHZ )
srate = 13000000 ;
else if ( aplls = = APLLS_CLKIN_12MHZ )
srate = 12000000 ;
return srate ;
}