Commit 1d90da95 authored by Tony Lindgren's avatar Tony Lindgren

Merge branch 'hwmod_clockevent_2.6.39' of git://git.pwsan.com/linux-2.6 into omap-for-linus

parents d6b5d01b 38698bef
/* /*
* omap_hwmod implementation for OMAP2/3/4 * omap_hwmod implementation for OMAP2/3/4
* *
* Copyright (C) 2009-2010 Nokia Corporation * Copyright (C) 2009-2011 Nokia Corporation
* *
* Paul Walmsley, Benoît Cousson, Kevin Hilman * Paul Walmsley, Benoît Cousson, Kevin Hilman
* *
...@@ -162,9 +162,6 @@ static LIST_HEAD(omap_hwmod_list); ...@@ -162,9 +162,6 @@ static LIST_HEAD(omap_hwmod_list);
/* mpu_oh: used to add/remove MPU initiator from sleepdep list */ /* mpu_oh: used to add/remove MPU initiator from sleepdep list */
static struct omap_hwmod *mpu_oh; static struct omap_hwmod *mpu_oh;
/* inited: 0 if omap_hwmod_init() has not yet been called; 1 otherwise */
static u8 inited;
/* Private functions */ /* Private functions */
...@@ -904,18 +901,16 @@ static struct omap_hwmod *_lookup(const char *name) ...@@ -904,18 +901,16 @@ static struct omap_hwmod *_lookup(const char *name)
* @oh: struct omap_hwmod * * @oh: struct omap_hwmod *
* @data: not used; pass NULL * @data: not used; pass NULL
* *
* Called by omap_hwmod_late_init() (after omap2_clk_init()). * Called by omap_hwmod_setup_*() (after omap2_clk_init()).
* Resolves all clock names embedded in the hwmod. Returns -EINVAL if * Resolves all clock names embedded in the hwmod. Returns 0 on
* the omap_hwmod has not yet been registered or if the clocks have * success, or a negative error code on failure.
* already been initialized, 0 on success, or a non-zero error on
* failure.
*/ */
static int _init_clocks(struct omap_hwmod *oh, void *data) static int _init_clocks(struct omap_hwmod *oh, void *data)
{ {
int ret = 0; int ret = 0;
if (!oh || (oh->_state != _HWMOD_STATE_REGISTERED)) if (oh->_state != _HWMOD_STATE_REGISTERED)
return -EINVAL; return 0;
pr_debug("omap_hwmod: %s: looking up clocks\n", oh->name); pr_debug("omap_hwmod: %s: looking up clocks\n", oh->name);
...@@ -1354,14 +1349,16 @@ static int _shutdown(struct omap_hwmod *oh) ...@@ -1354,14 +1349,16 @@ static int _shutdown(struct omap_hwmod *oh)
* @oh: struct omap_hwmod * * @oh: struct omap_hwmod *
* *
* Writes the CLOCKACTIVITY bits @clockact to the hwmod @oh * Writes the CLOCKACTIVITY bits @clockact to the hwmod @oh
* OCP_SYSCONFIG register. Returns -EINVAL if the hwmod is in the * OCP_SYSCONFIG register. Returns 0.
* wrong state or returns 0.
*/ */
static int _setup(struct omap_hwmod *oh, void *data) static int _setup(struct omap_hwmod *oh, void *data)
{ {
int i, r; int i, r;
u8 postsetup_state; u8 postsetup_state;
if (oh->_state != _HWMOD_STATE_CLKS_INITED)
return 0;
/* Set iclk autoidle mode */ /* Set iclk autoidle mode */
if (oh->slaves_cnt > 0) { if (oh->slaves_cnt > 0) {
for (i = 0; i < oh->slaves_cnt; i++) { for (i = 0; i < oh->slaves_cnt; i++) {
...@@ -1455,7 +1452,7 @@ static int _setup(struct omap_hwmod *oh, void *data) ...@@ -1455,7 +1452,7 @@ static int _setup(struct omap_hwmod *oh, void *data)
*/ */
static int __init _register(struct omap_hwmod *oh) static int __init _register(struct omap_hwmod *oh)
{ {
int ret, ms_id; int ms_id;
if (!oh || !oh->name || !oh->class || !oh->class->name || if (!oh || !oh->name || !oh->class || !oh->class->name ||
(oh->_state != _HWMOD_STATE_UNKNOWN)) (oh->_state != _HWMOD_STATE_UNKNOWN))
...@@ -1478,9 +1475,14 @@ static int __init _register(struct omap_hwmod *oh) ...@@ -1478,9 +1475,14 @@ static int __init _register(struct omap_hwmod *oh)
oh->_state = _HWMOD_STATE_REGISTERED; oh->_state = _HWMOD_STATE_REGISTERED;
ret = 0; /*
* XXX Rather than doing a strcmp(), this should test a flag
* set in the hwmod data, inserted by the autogenerator code.
*/
if (!strcmp(oh->name, MPU_INITIATOR_NAME))
mpu_oh = oh;
return ret; return 0;
} }
...@@ -1583,38 +1585,30 @@ int omap_hwmod_for_each(int (*fn)(struct omap_hwmod *oh, void *data), ...@@ -1583,38 +1585,30 @@ int omap_hwmod_for_each(int (*fn)(struct omap_hwmod *oh, void *data),
return ret; return ret;
} }
/** /**
* omap_hwmod_init - init omap_hwmod code and register hwmods * omap_hwmod_register - register an array of hwmods
* @ohs: pointer to an array of omap_hwmods to register * @ohs: pointer to an array of omap_hwmods to register
* *
* Intended to be called early in boot before the clock framework is * Intended to be called early in boot before the clock framework is
* initialized. If @ohs is not null, will register all omap_hwmods * initialized. If @ohs is not null, will register all omap_hwmods
* listed in @ohs that are valid for this chip. Returns -EINVAL if * listed in @ohs that are valid for this chip. Returns 0.
* omap_hwmod_init() has already been called or 0 otherwise.
*/ */
int __init omap_hwmod_init(struct omap_hwmod **ohs) int __init omap_hwmod_register(struct omap_hwmod **ohs)
{ {
struct omap_hwmod *oh; int r, i;
int r;
if (inited)
return -EINVAL;
inited = 1;
if (!ohs) if (!ohs)
return 0; return 0;
oh = *ohs; i = 0;
while (oh) { do {
if (omap_chip_is(oh->omap_chip)) { if (!omap_chip_is(ohs[i]->omap_chip))
r = _register(oh); continue;
WARN(r, "omap_hwmod: %s: _register returned "
"%d\n", oh->name, r); r = _register(ohs[i]);
} WARN(r, "omap_hwmod: %s: _register returned %d\n", ohs[i]->name,
oh = *++ohs; r);
} } while (ohs[++i]);
return 0; return 0;
} }
...@@ -1622,12 +1616,14 @@ int __init omap_hwmod_init(struct omap_hwmod **ohs) ...@@ -1622,12 +1616,14 @@ int __init omap_hwmod_init(struct omap_hwmod **ohs)
/* /*
* _populate_mpu_rt_base - populate the virtual address for a hwmod * _populate_mpu_rt_base - populate the virtual address for a hwmod
* *
* Must be called only from omap_hwmod_late_init so ioremap works properly. * Must be called only from omap_hwmod_setup_*() so ioremap works properly.
* Assumes the caller takes care of locking if needed. * Assumes the caller takes care of locking if needed.
*
*/ */
static int __init _populate_mpu_rt_base(struct omap_hwmod *oh, void *data) static int __init _populate_mpu_rt_base(struct omap_hwmod *oh, void *data)
{ {
if (oh->_state != _HWMOD_STATE_REGISTERED)
return 0;
if (oh->_int_flags & _HWMOD_NO_MPU_PORT) if (oh->_int_flags & _HWMOD_NO_MPU_PORT)
return 0; return 0;
...@@ -1640,31 +1636,81 @@ static int __init _populate_mpu_rt_base(struct omap_hwmod *oh, void *data) ...@@ -1640,31 +1636,81 @@ static int __init _populate_mpu_rt_base(struct omap_hwmod *oh, void *data)
} }
/** /**
* omap_hwmod_late_init - do some post-clock framework initialization * omap_hwmod_setup_one - set up a single hwmod
* @oh_name: const char * name of the already-registered hwmod to set up
*
* Must be called after omap2_clk_init(). Resolves the struct clk
* names to struct clk pointers for each registered omap_hwmod. Also
* calls _setup() on each hwmod. Returns -EINVAL upon error or 0 upon
* success.
*/
int __init omap_hwmod_setup_one(const char *oh_name)
{
struct omap_hwmod *oh;
int r;
pr_debug("omap_hwmod: %s: %s\n", oh_name, __func__);
if (!mpu_oh) {
pr_err("omap_hwmod: %s: cannot setup_one: MPU initiator hwmod %s not yet registered\n",
oh_name, MPU_INITIATOR_NAME);
return -EINVAL;
}
oh = _lookup(oh_name);
if (!oh) {
WARN(1, "omap_hwmod: %s: hwmod not yet registered\n", oh_name);
return -EINVAL;
}
if (mpu_oh->_state == _HWMOD_STATE_REGISTERED && oh != mpu_oh)
omap_hwmod_setup_one(MPU_INITIATOR_NAME);
r = _populate_mpu_rt_base(oh, NULL);
if (IS_ERR_VALUE(r)) {
WARN(1, "omap_hwmod: %s: couldn't set mpu_rt_base\n", oh_name);
return -EINVAL;
}
r = _init_clocks(oh, NULL);
if (IS_ERR_VALUE(r)) {
WARN(1, "omap_hwmod: %s: couldn't init clocks\n", oh_name);
return -EINVAL;
}
_setup(oh, NULL);
return 0;
}
/**
* omap_hwmod_setup - do some post-clock framework initialization
* *
* Must be called after omap2_clk_init(). Resolves the struct clk names * Must be called after omap2_clk_init(). Resolves the struct clk names
* to struct clk pointers for each registered omap_hwmod. Also calls * to struct clk pointers for each registered omap_hwmod. Also calls
* _setup() on each hwmod. Returns 0. * _setup() on each hwmod. Returns 0 upon success.
*/ */
static int __init omap_hwmod_late_init(void) static int __init omap_hwmod_setup_all(void)
{ {
int r; int r;
if (!mpu_oh) {
pr_err("omap_hwmod: %s: MPU initiator hwmod %s not yet registered\n",
__func__, MPU_INITIATOR_NAME);
return -EINVAL;
}
r = omap_hwmod_for_each(_populate_mpu_rt_base, NULL); r = omap_hwmod_for_each(_populate_mpu_rt_base, NULL);
/* XXX check return value */
r = omap_hwmod_for_each(_init_clocks, NULL); r = omap_hwmod_for_each(_init_clocks, NULL);
WARN(r, "omap_hwmod: omap_hwmod_late_init(): _init_clocks failed\n"); WARN(IS_ERR_VALUE(r),
"omap_hwmod: %s: _init_clocks failed\n", __func__);
mpu_oh = omap_hwmod_lookup(MPU_INITIATOR_NAME);
WARN(!mpu_oh, "omap_hwmod: could not find MPU initiator hwmod %s\n",
MPU_INITIATOR_NAME);
omap_hwmod_for_each(_setup, NULL); omap_hwmod_for_each(_setup, NULL);
return 0; return 0;
} }
core_initcall(omap_hwmod_late_init); core_initcall(omap_hwmod_setup_all);
/** /**
* omap_hwmod_enable - enable an omap_hwmod * omap_hwmod_enable - enable an omap_hwmod
...@@ -2183,11 +2229,11 @@ int omap_hwmod_for_each_by_class(const char *classname, ...@@ -2183,11 +2229,11 @@ int omap_hwmod_for_each_by_class(const char *classname,
* @oh: struct omap_hwmod * * @oh: struct omap_hwmod *
* @state: state that _setup() should leave the hwmod in * @state: state that _setup() should leave the hwmod in
* *
* Sets the hwmod state that @oh will enter at the end of _setup() (called by * Sets the hwmod state that @oh will enter at the end of _setup()
* omap_hwmod_late_init()). Only valid to call between calls to * (called by omap_hwmod_setup_*()). Only valid to call between
* omap_hwmod_init() and omap_hwmod_late_init(). Returns 0 upon success or * calling omap_hwmod_register() and omap_hwmod_setup_*(). Returns
* -EINVAL if there is a problem with the arguments or if the hwmod is * 0 upon success or -EINVAL if there is a problem with the arguments
* in the wrong state. * or if the hwmod is in the wrong state.
*/ */
int omap_hwmod_set_postsetup_state(struct omap_hwmod *oh, u8 state) int omap_hwmod_set_postsetup_state(struct omap_hwmod *oh, u8 state)
{ {
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
...@@ -5125,6 +5125,6 @@ static __initdata struct omap_hwmod *omap44xx_hwmods[] = { ...@@ -5125,6 +5125,6 @@ static __initdata struct omap_hwmod *omap44xx_hwmods[] = {
int __init omap44xx_hwmod_init(void) int __init omap44xx_hwmod_init(void)
{ {
return omap_hwmod_init(omap44xx_hwmods); return omap_hwmod_register(omap44xx_hwmods);
} }
...@@ -680,7 +680,7 @@ static int __init omap_serial_early_init(void) ...@@ -680,7 +680,7 @@ static int __init omap_serial_early_init(void)
num_uarts++; num_uarts++;
/* /*
* NOTE: omap_hwmod_init() has not yet been called, * NOTE: omap_hwmod_setup*() has not yet been called,
* so no hwmod functions will work yet. * so no hwmod functions will work yet.
*/ */
......
...@@ -39,10 +39,11 @@ ...@@ -39,10 +39,11 @@
#include <asm/mach/time.h> #include <asm/mach/time.h>
#include <plat/dmtimer.h> #include <plat/dmtimer.h>
#include <asm/localtimer.h> #include <asm/localtimer.h>
#include <plat/common.h>
#include <plat/omap_hwmod.h>
#include "timer-gp.h" #include "timer-gp.h"
#include <plat/common.h>
/* MAX_GPTIMER_ID: number of GPTIMERs on the chip */ /* MAX_GPTIMER_ID: number of GPTIMERs on the chip */
#define MAX_GPTIMER_ID 12 #define MAX_GPTIMER_ID 12
...@@ -132,9 +133,13 @@ static void __init omap2_gp_clockevent_init(void) ...@@ -132,9 +133,13 @@ static void __init omap2_gp_clockevent_init(void)
{ {
u32 tick_rate; u32 tick_rate;
int src; int src;
char clockevent_hwmod_name[8]; /* 8 = sizeof("timerXX0") */
inited = 1; inited = 1;
sprintf(clockevent_hwmod_name, "timer%d", gptimer_id);
omap_hwmod_setup_one(clockevent_hwmod_name);
gptimer = omap_dm_timer_request_specific(gptimer_id); gptimer = omap_dm_timer_request_specific(gptimer_id);
BUG_ON(gptimer == NULL); BUG_ON(gptimer == NULL);
gptimer_wakeup = gptimer; gptimer_wakeup = gptimer;
......
...@@ -3,6 +3,12 @@ ...@@ -3,6 +3,12 @@
* *
* OMAP Dual-Mode Timers * OMAP Dual-Mode Timers
* *
* Copyright (C) 2010 Texas Instruments Incorporated - http://www.ti.com/
* Tarun Kanti DebBarma <tarun.kanti@ti.com>
* Thara Gopinath <thara@ti.com>
*
* Platform device conversion and hwmod support.
*
* Copyright (C) 2005 Nokia Corporation * Copyright (C) 2005 Nokia Corporation
* Author: Lauri Leukkunen <lauri.leukkunen@nokia.com> * Author: Lauri Leukkunen <lauri.leukkunen@nokia.com>
* PWM and clock framwork support by Timo Teras. * PWM and clock framwork support by Timo Teras.
...@@ -44,6 +50,11 @@ ...@@ -44,6 +50,11 @@
#define OMAP_TIMER_TRIGGER_OVERFLOW 0x01 #define OMAP_TIMER_TRIGGER_OVERFLOW 0x01
#define OMAP_TIMER_TRIGGER_OVERFLOW_AND_COMPARE 0x02 #define OMAP_TIMER_TRIGGER_OVERFLOW_AND_COMPARE 0x02
/*
* IP revision identifier so that Highlander IP
* in OMAP4 can be distinguished.
*/
#define OMAP_TIMER_IP_VERSION_1 0x1
struct omap_dm_timer; struct omap_dm_timer;
extern struct omap_dm_timer *gptimer_wakeup; extern struct omap_dm_timer *gptimer_wakeup;
extern struct sys_timer omap_timer; extern struct sys_timer omap_timer;
......
/* /*
* omap_hwmod macros, structures * omap_hwmod macros, structures
* *
* Copyright (C) 2009-2010 Nokia Corporation * Copyright (C) 2009-2011 Nokia Corporation
* Paul Walmsley * Paul Walmsley
* *
* Created in collaboration with (alphabetical order): Benoît Cousson, * Created in collaboration with (alphabetical order): Benoît Cousson,
...@@ -30,6 +30,7 @@ ...@@ -30,6 +30,7 @@
#define __ARCH_ARM_PLAT_OMAP_INCLUDE_MACH_OMAP_HWMOD_H #define __ARCH_ARM_PLAT_OMAP_INCLUDE_MACH_OMAP_HWMOD_H
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/init.h>
#include <linux/list.h> #include <linux/list.h>
#include <linux/ioport.h> #include <linux/ioport.h>
#include <linux/spinlock.h> #include <linux/spinlock.h>
...@@ -370,8 +371,10 @@ struct omap_hwmod_omap4_prcm { ...@@ -370,8 +371,10 @@ struct omap_hwmod_omap4_prcm {
* of standby, rather than relying on module smart-standby * of standby, rather than relying on module smart-standby
* HWMOD_INIT_NO_RESET: don't reset this module at boot - important for * HWMOD_INIT_NO_RESET: don't reset this module at boot - important for
* SDRAM controller, etc. XXX probably belongs outside the main hwmod file * SDRAM controller, etc. XXX probably belongs outside the main hwmod file
* XXX Should be HWMOD_SETUP_NO_RESET
* HWMOD_INIT_NO_IDLE: don't idle this module at boot - important for SDRAM * HWMOD_INIT_NO_IDLE: don't idle this module at boot - important for SDRAM
* controller, etc. XXX probably belongs outside the main hwmod file * controller, etc. XXX probably belongs outside the main hwmod file
* XXX Should be HWMOD_SETUP_NO_IDLE
* HWMOD_NO_AUTOIDLE: disable module autoidle (OCP_SYSCONFIG.AUTOIDLE) * HWMOD_NO_AUTOIDLE: disable module autoidle (OCP_SYSCONFIG.AUTOIDLE)
* when module is enabled, rather than the default, which is to * when module is enabled, rather than the default, which is to
* enable autoidle * enable autoidle
...@@ -535,11 +538,13 @@ struct omap_hwmod { ...@@ -535,11 +538,13 @@ struct omap_hwmod {
const struct omap_chip_id omap_chip; const struct omap_chip_id omap_chip;
}; };
int omap_hwmod_init(struct omap_hwmod **ohs); int omap_hwmod_register(struct omap_hwmod **ohs);
struct omap_hwmod *omap_hwmod_lookup(const char *name); struct omap_hwmod *omap_hwmod_lookup(const char *name);
int omap_hwmod_for_each(int (*fn)(struct omap_hwmod *oh, void *data), int omap_hwmod_for_each(int (*fn)(struct omap_hwmod *oh, void *data),
void *data); void *data);
int __init omap_hwmod_setup_one(const char *name);
int omap_hwmod_enable(struct omap_hwmod *oh); int omap_hwmod_enable(struct omap_hwmod *oh);
int _omap_hwmod_enable(struct omap_hwmod *oh); int _omap_hwmod_enable(struct omap_hwmod *oh);
int omap_hwmod_idle(struct omap_hwmod *oh); int omap_hwmod_idle(struct omap_hwmod *oh);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment