aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm
diff options
context:
space:
mode:
authorKevin Hilman <khilman@ti.com>2012-06-18 14:12:23 -0400
committerPaul Walmsley <paul@pwsan.com>2012-06-18 14:12:23 -0400
commit9ebfd285371835b1c0243d15aaacd72d5def76f8 (patch)
treeff0f3674a11b5a8e23c9eb4e800ca294bd5a7fcc /arch/arm
parent3d9f032724e72851c6de38d026b2417501a2f240 (diff)
ARM: OMAP2+: hwmod: use init-time function ptrs for enable/disable module
The enable/disable module functions are specific to SoCs with OMAP4-class PRCM. Rather than use cpu_is* checks at runtime inside the enable/disable module functions, use cpu_is at init time to initialize function pointers only for SoCs that need them. NOTE: the cpu_is* check for _enable_module was different than the one for _disable_module, and this patch uses cpu_is_omap44xx() for both. Signed-off-by: Kevin Hilman <khilman@ti.com> [paul@pwsan.com: moved soc_ops function pointers to be per-kernel rather than per-hwmod since they do not vary by hwmod; added kerneldoc] Signed-off-by: Paul Walmsley <paul@pwsan.com>
Diffstat (limited to 'arch/arm')
-rw-r--r--arch/arm/mach-omap2/omap_hwmod.c62
-rw-r--r--arch/arm/mach-omap2/omap_hwmod_2420_data.c1
-rw-r--r--arch/arm/mach-omap2/omap_hwmod_2430_data.c1
-rw-r--r--arch/arm/mach-omap2/omap_hwmod_3xxx_data.c2
-rw-r--r--arch/arm/mach-omap2/omap_hwmod_44xx_data.c1
-rw-r--r--arch/arm/plat-omap/include/plat/omap_hwmod.h2
6 files changed, 57 insertions, 12 deletions
diff --git a/arch/arm/mach-omap2/omap_hwmod.c b/arch/arm/mach-omap2/omap_hwmod.c
index 939032a427fd..634a79836c64 100644
--- a/arch/arm/mach-omap2/omap_hwmod.c
+++ b/arch/arm/mach-omap2/omap_hwmod.c
@@ -166,6 +166,23 @@
166 */ 166 */
167#define LINKS_PER_OCP_IF 2 167#define LINKS_PER_OCP_IF 2
168 168
169/**
170 * struct omap_hwmod_soc_ops - fn ptrs for some SoC-specific operations
171 * @enable_module: function to enable a module (via MODULEMODE)
172 * @disable_module: function to disable a module (via MODULEMODE)
173 *
174 * XXX Eventually this functionality will be hidden inside the PRM/CM
175 * device drivers. Until then, this should avoid huge blocks of cpu_is_*()
176 * conditionals in this code.
177 */
178struct omap_hwmod_soc_ops {
179 void (*enable_module)(struct omap_hwmod *oh);
180 int (*disable_module)(struct omap_hwmod *oh);
181};
182
183/* soc_ops: adapts the omap_hwmod code to the currently-booted SoC */
184static struct omap_hwmod_soc_ops soc_ops;
185
169/* omap_hwmod_list contains all registered struct omap_hwmods */ 186/* omap_hwmod_list contains all registered struct omap_hwmods */
170static LIST_HEAD(omap_hwmod_list); 187static LIST_HEAD(omap_hwmod_list);
171 188
@@ -186,6 +203,9 @@ static struct omap_hwmod_link *linkspace;
186 */ 203 */
187static unsigned short free_ls, max_ls, ls_supp; 204static unsigned short free_ls, max_ls, ls_supp;
188 205
206/* inited: set to true once the hwmod code is initialized */
207static bool inited;
208
189/* Private functions */ 209/* Private functions */
190 210
191/** 211/**
@@ -779,10 +799,6 @@ static void _disable_optional_clocks(struct omap_hwmod *oh)
779 */ 799 */
780static void _omap4_enable_module(struct omap_hwmod *oh) 800static void _omap4_enable_module(struct omap_hwmod *oh)
781{ 801{
782 /* The module mode does not exist prior OMAP4 */
783 if (cpu_is_omap24xx() || cpu_is_omap34xx())
784 return;
785
786 if (!oh->clkdm || !oh->prcm.omap4.modulemode) 802 if (!oh->clkdm || !oh->prcm.omap4.modulemode)
787 return; 803 return;
788 804
@@ -1571,10 +1587,6 @@ static int _omap4_disable_module(struct omap_hwmod *oh)
1571{ 1587{
1572 int v; 1588 int v;
1573 1589
1574 /* The module mode does not exist prior OMAP4 */
1575 if (!cpu_is_omap44xx())
1576 return -EINVAL;
1577
1578 if (!oh->clkdm || !oh->prcm.omap4.modulemode) 1590 if (!oh->clkdm || !oh->prcm.omap4.modulemode)
1579 return -EINVAL; 1591 return -EINVAL;
1580 1592
@@ -1814,7 +1826,8 @@ static int _enable(struct omap_hwmod *oh)
1814 } 1826 }
1815 1827
1816 _enable_clocks(oh); 1828 _enable_clocks(oh);
1817 _omap4_enable_module(oh); 1829 if (soc_ops.enable_module)
1830 soc_ops.enable_module(oh);
1818 1831
1819 r = _wait_target_ready(oh); 1832 r = _wait_target_ready(oh);
1820 if (!r) { 1833 if (!r) {
@@ -1870,7 +1883,8 @@ static int _idle(struct omap_hwmod *oh)
1870 _idle_sysc(oh); 1883 _idle_sysc(oh);
1871 _del_initiator_dep(oh, mpu_oh); 1884 _del_initiator_dep(oh, mpu_oh);
1872 1885
1873 _omap4_disable_module(oh); 1886 if (soc_ops.disable_module)
1887 soc_ops.disable_module(oh);
1874 1888
1875 /* 1889 /*
1876 * The module must be in idle mode before disabling any parents 1890 * The module must be in idle mode before disabling any parents
@@ -1975,7 +1989,8 @@ static int _shutdown(struct omap_hwmod *oh)
1975 if (oh->_state == _HWMOD_STATE_ENABLED) { 1989 if (oh->_state == _HWMOD_STATE_ENABLED) {
1976 _del_initiator_dep(oh, mpu_oh); 1990 _del_initiator_dep(oh, mpu_oh);
1977 /* XXX what about the other system initiators here? dma, dsp */ 1991 /* XXX what about the other system initiators here? dma, dsp */
1978 _omap4_disable_module(oh); 1992 if (soc_ops.disable_module)
1993 soc_ops.disable_module(oh);
1979 _disable_clocks(oh); 1994 _disable_clocks(oh);
1980 if (oh->clkdm) 1995 if (oh->clkdm)
1981 clkdm_hwmod_disable(oh->clkdm, oh); 1996 clkdm_hwmod_disable(oh->clkdm, oh);
@@ -2563,12 +2578,18 @@ int omap_hwmod_for_each(int (*fn)(struct omap_hwmod *oh, void *data),
2563 * 2578 *
2564 * Intended to be called early in boot before the clock framework is 2579 * Intended to be called early in boot before the clock framework is
2565 * initialized. If @ois is not null, will register all omap_hwmods 2580 * initialized. If @ois is not null, will register all omap_hwmods
2566 * listed in @ois that are valid for this chip. Returns 0. 2581 * listed in @ois that are valid for this chip. Returns -EINVAL if
2582 * omap_hwmod_init() hasn't been called before calling this function,
2583 * -ENOMEM if the link memory area can't be allocated, or 0 upon
2584 * success.
2567 */ 2585 */
2568int __init omap_hwmod_register_links(struct omap_hwmod_ocp_if **ois) 2586int __init omap_hwmod_register_links(struct omap_hwmod_ocp_if **ois)
2569{ 2587{
2570 int r, i; 2588 int r, i;
2571 2589
2590 if (!inited)
2591 return -EINVAL;
2592
2572 if (!ois) 2593 if (!ois)
2573 return 0; 2594 return 0;
2574 2595
@@ -3401,3 +3422,20 @@ int omap_hwmod_pad_route_irq(struct omap_hwmod *oh, int pad_idx, int irq_idx)
3401 3422
3402 return 0; 3423 return 0;
3403} 3424}
3425
3426/**
3427 * omap_hwmod_init - initialize the hwmod code
3428 *
3429 * Sets up some function pointers needed by the hwmod code to operate on the
3430 * currently-booted SoC. Intended to be called once during kernel init
3431 * before any hwmods are registered. No return value.
3432 */
3433void __init omap_hwmod_init(void)
3434{
3435 if (cpu_is_omap44xx()) {
3436 soc_ops.enable_module = _omap4_enable_module;
3437 soc_ops.disable_module = _omap4_disable_module;
3438 }
3439
3440 inited = true;
3441}
diff --git a/arch/arm/mach-omap2/omap_hwmod_2420_data.c b/arch/arm/mach-omap2/omap_hwmod_2420_data.c
index a7640d1b215e..5b1938b52d09 100644
--- a/arch/arm/mach-omap2/omap_hwmod_2420_data.c
+++ b/arch/arm/mach-omap2/omap_hwmod_2420_data.c
@@ -585,5 +585,6 @@ static struct omap_hwmod_ocp_if *omap2420_hwmod_ocp_ifs[] __initdata = {
585 585
586int __init omap2420_hwmod_init(void) 586int __init omap2420_hwmod_init(void)
587{ 587{
588 omap_hwmod_init();
588 return omap_hwmod_register_links(omap2420_hwmod_ocp_ifs); 589 return omap_hwmod_register_links(omap2420_hwmod_ocp_ifs);
589} 590}
diff --git a/arch/arm/mach-omap2/omap_hwmod_2430_data.c b/arch/arm/mach-omap2/omap_hwmod_2430_data.c
index 4d7264981230..71f199783c63 100644
--- a/arch/arm/mach-omap2/omap_hwmod_2430_data.c
+++ b/arch/arm/mach-omap2/omap_hwmod_2430_data.c
@@ -938,5 +938,6 @@ static struct omap_hwmod_ocp_if *omap2430_hwmod_ocp_ifs[] __initdata = {
938 938
939int __init omap2430_hwmod_init(void) 939int __init omap2430_hwmod_init(void)
940{ 940{
941 omap_hwmod_init();
941 return omap_hwmod_register_links(omap2430_hwmod_ocp_ifs); 942 return omap_hwmod_register_links(omap2430_hwmod_ocp_ifs);
942} 943}
diff --git a/arch/arm/mach-omap2/omap_hwmod_3xxx_data.c b/arch/arm/mach-omap2/omap_hwmod_3xxx_data.c
index b26d3c9bca16..abb1f6b56ee5 100644
--- a/arch/arm/mach-omap2/omap_hwmod_3xxx_data.c
+++ b/arch/arm/mach-omap2/omap_hwmod_3xxx_data.c
@@ -3283,6 +3283,8 @@ int __init omap3xxx_hwmod_init(void)
3283 struct omap_hwmod_ocp_if **h = NULL; 3283 struct omap_hwmod_ocp_if **h = NULL;
3284 unsigned int rev; 3284 unsigned int rev;
3285 3285
3286 omap_hwmod_init();
3287
3286 /* Register hwmod links common to all OMAP3 */ 3288 /* Register hwmod links common to all OMAP3 */
3287 r = omap_hwmod_register_links(omap3xxx_hwmod_ocp_ifs); 3289 r = omap_hwmod_register_links(omap3xxx_hwmod_ocp_ifs);
3288 if (r < 0) 3290 if (r < 0)
diff --git a/arch/arm/mach-omap2/omap_hwmod_44xx_data.c b/arch/arm/mach-omap2/omap_hwmod_44xx_data.c
index 950454a3fa31..d3c48dc26af1 100644
--- a/arch/arm/mach-omap2/omap_hwmod_44xx_data.c
+++ b/arch/arm/mach-omap2/omap_hwmod_44xx_data.c
@@ -6144,6 +6144,7 @@ static struct omap_hwmod_ocp_if *omap44xx_hwmod_ocp_ifs[] __initdata = {
6144 6144
6145int __init omap44xx_hwmod_init(void) 6145int __init omap44xx_hwmod_init(void)
6146{ 6146{
6147 omap_hwmod_init();
6147 return omap_hwmod_register_links(omap44xx_hwmod_ocp_ifs); 6148 return omap_hwmod_register_links(omap44xx_hwmod_ocp_ifs);
6148} 6149}
6149 6150
diff --git a/arch/arm/plat-omap/include/plat/omap_hwmod.h b/arch/arm/plat-omap/include/plat/omap_hwmod.h
index c835b7194ff5..a8ecc53b3670 100644
--- a/arch/arm/plat-omap/include/plat/omap_hwmod.h
+++ b/arch/arm/plat-omap/include/plat/omap_hwmod.h
@@ -629,6 +629,8 @@ int omap_hwmod_no_setup_reset(struct omap_hwmod *oh);
629 629
630int omap_hwmod_pad_route_irq(struct omap_hwmod *oh, int pad_idx, int irq_idx); 630int omap_hwmod_pad_route_irq(struct omap_hwmod *oh, int pad_idx, int irq_idx);
631 631
632extern void __init omap_hwmod_init(void);
633
632/* 634/*
633 * Chip variant-specific hwmod init routines - XXX should be converted 635 * Chip variant-specific hwmod init routines - XXX should be converted
634 * to use initcalls once the initial boot ordering is straightened out 636 * to use initcalls once the initial boot ordering is straightened out