mirror of
https://github.com/torvalds/linux
synced 2024-11-05 18:23:50 +00:00
mxc timer: make compile time independent
Currently we depend on hardcoded base addresses for the timer. This prevents us from compiling in more than one i.MX architecture at a time. This patch changes the base address to a runtime calculated one. Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de>
This commit is contained in:
parent
cd4a05f9df
commit
ec996ba9b5
2 changed files with 135 additions and 178 deletions
|
@ -1,158 +0,0 @@
|
||||||
/*
|
|
||||||
* mxc_timer.h
|
|
||||||
*
|
|
||||||
* Copyright (C) 2008 Juergen Beisert (kernel@pengutronix.de)
|
|
||||||
*
|
|
||||||
* Platform independent (i.MX1, i.MX2, i.MX3) definition for timer handling.
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or
|
|
||||||
* modify it under the terms of the GNU General Public License
|
|
||||||
* as published by the Free Software Foundation; either version 2
|
|
||||||
* of the License, or (at your option) any later version.
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program; if not, write to the Free Software
|
|
||||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
||||||
* Boston, MA 02110-1301, USA.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef __PLAT_MXC_TIMER_H
|
|
||||||
#define __PLAT_MXC_TIMER_H
|
|
||||||
|
|
||||||
#include <linux/clk.h>
|
|
||||||
#include <mach/hardware.h>
|
|
||||||
|
|
||||||
#ifdef CONFIG_ARCH_MX1
|
|
||||||
#define TIMER_BASE IO_ADDRESS(TIM1_BASE_ADDR)
|
|
||||||
#define TIMER_INTERRUPT TIM1_INT
|
|
||||||
|
|
||||||
#define TCTL_VAL TCTL_CLK_PCLK1
|
|
||||||
#define TCTL_IRQEN (1<<4)
|
|
||||||
#define TCTL_FRR (1<<8)
|
|
||||||
#define TCTL_CLK_PCLK1 (1<<1)
|
|
||||||
#define TCTL_CLK_PCLK1_4 (2<<1)
|
|
||||||
#define TCTL_CLK_TIN (3<<1)
|
|
||||||
#define TCTL_CLK_32 (4<<1)
|
|
||||||
|
|
||||||
#define MXC_TCTL 0x00
|
|
||||||
#define MXC_TPRER 0x04
|
|
||||||
#define MXC_TCMP 0x08
|
|
||||||
#define MXC_TCR 0x0c
|
|
||||||
#define MXC_TCN 0x10
|
|
||||||
#define MXC_TSTAT 0x14
|
|
||||||
#define TSTAT_CAPT (1<<1)
|
|
||||||
#define TSTAT_COMP (1<<0)
|
|
||||||
|
|
||||||
static inline void gpt_irq_disable(void)
|
|
||||||
{
|
|
||||||
unsigned int tmp;
|
|
||||||
|
|
||||||
tmp = __raw_readl(TIMER_BASE + MXC_TCTL);
|
|
||||||
__raw_writel(tmp & ~TCTL_IRQEN, TIMER_BASE + MXC_TCTL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void gpt_irq_enable(void)
|
|
||||||
{
|
|
||||||
__raw_writel(__raw_readl(TIMER_BASE + MXC_TCTL) | TCTL_IRQEN,
|
|
||||||
TIMER_BASE + MXC_TCTL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void gpt_irq_acknowledge(void)
|
|
||||||
{
|
|
||||||
__raw_writel(0, TIMER_BASE + MXC_TSTAT);
|
|
||||||
}
|
|
||||||
#endif /* CONFIG_ARCH_MX1 */
|
|
||||||
|
|
||||||
#ifdef CONFIG_ARCH_MX2
|
|
||||||
#define TIMER_BASE IO_ADDRESS(GPT1_BASE_ADDR)
|
|
||||||
#define TIMER_INTERRUPT MXC_INT_GPT1
|
|
||||||
|
|
||||||
#define MXC_TCTL 0x00
|
|
||||||
#define TCTL_VAL TCTL_CLK_PCLK1
|
|
||||||
#define TCTL_CLK_PCLK1 (1<<1)
|
|
||||||
#define TCTL_CLK_PCLK1_4 (2<<1)
|
|
||||||
#define TCTL_IRQEN (1<<4)
|
|
||||||
#define TCTL_FRR (1<<8)
|
|
||||||
#define MXC_TPRER 0x04
|
|
||||||
#define MXC_TCMP 0x08
|
|
||||||
#define MXC_TCR 0x0c
|
|
||||||
#define MXC_TCN 0x10
|
|
||||||
#define MXC_TSTAT 0x14
|
|
||||||
#define TSTAT_CAPT (1<<1)
|
|
||||||
#define TSTAT_COMP (1<<0)
|
|
||||||
|
|
||||||
static inline void gpt_irq_disable(void)
|
|
||||||
{
|
|
||||||
unsigned int tmp;
|
|
||||||
|
|
||||||
tmp = __raw_readl(TIMER_BASE + MXC_TCTL);
|
|
||||||
__raw_writel(tmp & ~TCTL_IRQEN, TIMER_BASE + MXC_TCTL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void gpt_irq_enable(void)
|
|
||||||
{
|
|
||||||
__raw_writel(__raw_readl(TIMER_BASE + MXC_TCTL) | TCTL_IRQEN,
|
|
||||||
TIMER_BASE + MXC_TCTL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void gpt_irq_acknowledge(void)
|
|
||||||
{
|
|
||||||
__raw_writel(TSTAT_CAPT | TSTAT_COMP, TIMER_BASE + MXC_TSTAT);
|
|
||||||
}
|
|
||||||
#endif /* CONFIG_ARCH_MX2 */
|
|
||||||
|
|
||||||
#ifdef CONFIG_ARCH_MX3
|
|
||||||
#define TIMER_BASE IO_ADDRESS(GPT1_BASE_ADDR)
|
|
||||||
#define TIMER_INTERRUPT MXC_INT_GPT
|
|
||||||
|
|
||||||
#define MXC_TCTL 0x00
|
|
||||||
#define TCTL_VAL (TCTL_CLK_IPG | TCTL_WAITEN)
|
|
||||||
#define TCTL_CLK_IPG (1<<6)
|
|
||||||
#define TCTL_FRR (1<<9)
|
|
||||||
#define TCTL_WAITEN (1<<3)
|
|
||||||
|
|
||||||
#define MXC_TPRER 0x04
|
|
||||||
#define MXC_TSTAT 0x08
|
|
||||||
#define TSTAT_OF1 (1<<0)
|
|
||||||
#define TSTAT_OF2 (1<<1)
|
|
||||||
#define TSTAT_OF3 (1<<2)
|
|
||||||
#define TSTAT_IF1 (1<<3)
|
|
||||||
#define TSTAT_IF2 (1<<4)
|
|
||||||
#define TSTAT_ROV (1<<5)
|
|
||||||
#define MXC_IR 0x0c
|
|
||||||
#define MXC_TCMP 0x10
|
|
||||||
#define MXC_TCMP2 0x14
|
|
||||||
#define MXC_TCMP3 0x18
|
|
||||||
#define MXC_TCR 0x1c
|
|
||||||
#define MXC_TCN 0x24
|
|
||||||
|
|
||||||
static inline void gpt_irq_disable(void)
|
|
||||||
{
|
|
||||||
__raw_writel(0, TIMER_BASE + MXC_IR);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void gpt_irq_enable(void)
|
|
||||||
{
|
|
||||||
__raw_writel(1<<0, TIMER_BASE + MXC_IR);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void gpt_irq_acknowledge(void)
|
|
||||||
{
|
|
||||||
__raw_writel(TSTAT_OF1, TIMER_BASE + MXC_TSTAT);
|
|
||||||
}
|
|
||||||
#endif /* CONFIG_ARCH_MX3 */
|
|
||||||
|
|
||||||
#define TCTL_SWR (1<<15)
|
|
||||||
#define TCTL_CC (1<<10)
|
|
||||||
#define TCTL_OM (1<<9)
|
|
||||||
#define TCTL_CAP_RIS (1<<6)
|
|
||||||
#define TCTL_CAP_FAL (2<<6)
|
|
||||||
#define TCTL_CAP_RIS_FAL (3<<6)
|
|
||||||
#define TCTL_CAP_ENA (1<<5)
|
|
||||||
#define TCTL_TEN (1<<0)
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -29,22 +29,85 @@
|
||||||
#include <mach/hardware.h>
|
#include <mach/hardware.h>
|
||||||
#include <asm/mach/time.h>
|
#include <asm/mach/time.h>
|
||||||
#include <mach/common.h>
|
#include <mach/common.h>
|
||||||
#include <mach/mxc_timer.h>
|
|
||||||
|
/* defines common for all i.MX */
|
||||||
|
#define MXC_TCTL 0x00
|
||||||
|
#define MXC_TCTL_TEN (1 << 0)
|
||||||
|
#define MXC_TPRER 0x04
|
||||||
|
|
||||||
|
/* MX1, MX21, MX27 */
|
||||||
|
#define MX1_2_TCTL_CLK_PCLK1 (1 << 1)
|
||||||
|
#define MX1_2_TCTL_IRQEN (1 << 4)
|
||||||
|
#define MX1_2_TCTL_FRR (1 << 8)
|
||||||
|
#define MX1_2_TCMP 0x08
|
||||||
|
#define MX1_2_TCN 0x10
|
||||||
|
#define MX1_2_TSTAT 0x14
|
||||||
|
|
||||||
|
/* MX21, MX27 */
|
||||||
|
#define MX2_TSTAT_CAPT (1 << 1)
|
||||||
|
#define MX2_TSTAT_COMP (1 << 0)
|
||||||
|
|
||||||
|
/* MX31, MX35 */
|
||||||
|
#define MX3_TCTL_WAITEN (1 << 3)
|
||||||
|
#define MX3_TCTL_CLK_IPG (1 << 6)
|
||||||
|
#define MX3_TCTL_FRR (1 << 9)
|
||||||
|
#define MX3_IR 0x0c
|
||||||
|
#define MX3_TSTAT 0x08
|
||||||
|
#define MX3_TSTAT_OF1 (1 << 0)
|
||||||
|
#define MX3_TCN 0x24
|
||||||
|
#define MX3_TCMP 0x10
|
||||||
|
|
||||||
static struct clock_event_device clockevent_mxc;
|
static struct clock_event_device clockevent_mxc;
|
||||||
static enum clock_event_mode clockevent_mode = CLOCK_EVT_MODE_UNUSED;
|
static enum clock_event_mode clockevent_mode = CLOCK_EVT_MODE_UNUSED;
|
||||||
|
|
||||||
/* clock source */
|
static void __iomem *timer_base;
|
||||||
|
|
||||||
static cycle_t mxc_get_cycles(struct clocksource *cs)
|
static inline void gpt_irq_disable(void)
|
||||||
{
|
{
|
||||||
return __raw_readl(TIMER_BASE + MXC_TCN);
|
unsigned int tmp;
|
||||||
|
|
||||||
|
if (cpu_is_mx3())
|
||||||
|
__raw_writel(0, timer_base + MX3_IR);
|
||||||
|
else {
|
||||||
|
tmp = __raw_readl(timer_base + MXC_TCTL);
|
||||||
|
__raw_writel(tmp & ~MX1_2_TCTL_IRQEN, timer_base + MXC_TCTL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void gpt_irq_enable(void)
|
||||||
|
{
|
||||||
|
if (cpu_is_mx3())
|
||||||
|
__raw_writel(1<<0, timer_base + MX3_IR);
|
||||||
|
else {
|
||||||
|
__raw_writel(__raw_readl(timer_base + MXC_TCTL) | MX1_2_TCTL_IRQEN,
|
||||||
|
timer_base + MXC_TCTL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gpt_irq_acknowledge(void)
|
||||||
|
{
|
||||||
|
if (cpu_is_mx1())
|
||||||
|
__raw_writel(0, timer_base + MX1_2_TSTAT);
|
||||||
|
if (cpu_is_mx2())
|
||||||
|
__raw_writel(MX2_TSTAT_CAPT | MX2_TSTAT_COMP, timer_base + MX1_2_TSTAT);
|
||||||
|
if (cpu_is_mx3())
|
||||||
|
__raw_writel(MX3_TSTAT_OF1, timer_base + MX3_TSTAT);
|
||||||
|
}
|
||||||
|
|
||||||
|
static cycle_t mx1_2_get_cycles(struct clocksource *cs)
|
||||||
|
{
|
||||||
|
return __raw_readl(timer_base + MX1_2_TCN);
|
||||||
|
}
|
||||||
|
|
||||||
|
static cycle_t mx3_get_cycles(struct clocksource *cs)
|
||||||
|
{
|
||||||
|
return __raw_readl(timer_base + MX3_TCN);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct clocksource clocksource_mxc = {
|
static struct clocksource clocksource_mxc = {
|
||||||
.name = "mxc_timer1",
|
.name = "mxc_timer1",
|
||||||
.rating = 200,
|
.rating = 200,
|
||||||
.read = mxc_get_cycles,
|
.read = mx1_2_get_cycles,
|
||||||
.mask = CLOCKSOURCE_MASK(32),
|
.mask = CLOCKSOURCE_MASK(32),
|
||||||
.shift = 20,
|
.shift = 20,
|
||||||
.flags = CLOCK_SOURCE_IS_CONTINUOUS,
|
.flags = CLOCK_SOURCE_IS_CONTINUOUS,
|
||||||
|
@ -54,6 +117,9 @@ static int __init mxc_clocksource_init(struct clk *timer_clk)
|
||||||
{
|
{
|
||||||
unsigned int c = clk_get_rate(timer_clk);
|
unsigned int c = clk_get_rate(timer_clk);
|
||||||
|
|
||||||
|
if (cpu_is_mx3())
|
||||||
|
clocksource_mxc.read = mx3_get_cycles;
|
||||||
|
|
||||||
clocksource_mxc.mult = clocksource_hz2mult(c,
|
clocksource_mxc.mult = clocksource_hz2mult(c,
|
||||||
clocksource_mxc.shift);
|
clocksource_mxc.shift);
|
||||||
clocksource_register(&clocksource_mxc);
|
clocksource_register(&clocksource_mxc);
|
||||||
|
@ -63,15 +129,29 @@ static int __init mxc_clocksource_init(struct clk *timer_clk)
|
||||||
|
|
||||||
/* clock event */
|
/* clock event */
|
||||||
|
|
||||||
static int mxc_set_next_event(unsigned long evt,
|
static int mx1_2_set_next_event(unsigned long evt,
|
||||||
struct clock_event_device *unused)
|
struct clock_event_device *unused)
|
||||||
{
|
{
|
||||||
unsigned long tcmp;
|
unsigned long tcmp;
|
||||||
|
|
||||||
tcmp = __raw_readl(TIMER_BASE + MXC_TCN) + evt;
|
tcmp = __raw_readl(timer_base + MX1_2_TCN) + evt;
|
||||||
__raw_writel(tcmp, TIMER_BASE + MXC_TCMP);
|
|
||||||
|
|
||||||
return (int)(tcmp - __raw_readl(TIMER_BASE + MXC_TCN)) < 0 ?
|
__raw_writel(tcmp, timer_base + MX1_2_TCMP);
|
||||||
|
|
||||||
|
return (int)(tcmp - __raw_readl(timer_base + MX1_2_TCN)) < 0 ?
|
||||||
|
-ETIME : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mx3_set_next_event(unsigned long evt,
|
||||||
|
struct clock_event_device *unused)
|
||||||
|
{
|
||||||
|
unsigned long tcmp;
|
||||||
|
|
||||||
|
tcmp = __raw_readl(timer_base + MX3_TCN) + evt;
|
||||||
|
|
||||||
|
__raw_writel(tcmp, timer_base + MX3_TCMP);
|
||||||
|
|
||||||
|
return (int)(tcmp - __raw_readl(timer_base + MX3_TCN)) < 0 ?
|
||||||
-ETIME : 0;
|
-ETIME : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -100,8 +180,13 @@ static void mxc_set_mode(enum clock_event_mode mode,
|
||||||
|
|
||||||
if (mode != clockevent_mode) {
|
if (mode != clockevent_mode) {
|
||||||
/* Set event time into far-far future */
|
/* Set event time into far-far future */
|
||||||
__raw_writel(__raw_readl(TIMER_BASE + MXC_TCN) - 3,
|
if (cpu_is_mx3())
|
||||||
TIMER_BASE + MXC_TCMP);
|
__raw_writel(__raw_readl(timer_base + MX3_TCN) - 3,
|
||||||
|
timer_base + MX3_TCMP);
|
||||||
|
else
|
||||||
|
__raw_writel(__raw_readl(timer_base + MX1_2_TCN) - 3,
|
||||||
|
timer_base + MX1_2_TCMP);
|
||||||
|
|
||||||
/* Clear pending interrupt */
|
/* Clear pending interrupt */
|
||||||
gpt_irq_acknowledge();
|
gpt_irq_acknowledge();
|
||||||
}
|
}
|
||||||
|
@ -148,7 +233,10 @@ static irqreturn_t mxc_timer_interrupt(int irq, void *dev_id)
|
||||||
struct clock_event_device *evt = &clockevent_mxc;
|
struct clock_event_device *evt = &clockevent_mxc;
|
||||||
uint32_t tstat;
|
uint32_t tstat;
|
||||||
|
|
||||||
tstat = __raw_readl(TIMER_BASE + MXC_TSTAT);
|
if (cpu_is_mx3())
|
||||||
|
tstat = __raw_readl(timer_base + MX1_2_TSTAT);
|
||||||
|
else
|
||||||
|
tstat = __raw_readl(timer_base + MX3_TSTAT);
|
||||||
|
|
||||||
gpt_irq_acknowledge();
|
gpt_irq_acknowledge();
|
||||||
|
|
||||||
|
@ -168,7 +256,7 @@ static struct clock_event_device clockevent_mxc = {
|
||||||
.features = CLOCK_EVT_FEAT_ONESHOT,
|
.features = CLOCK_EVT_FEAT_ONESHOT,
|
||||||
.shift = 32,
|
.shift = 32,
|
||||||
.set_mode = mxc_set_mode,
|
.set_mode = mxc_set_mode,
|
||||||
.set_next_event = mxc_set_next_event,
|
.set_next_event = mx1_2_set_next_event,
|
||||||
.rating = 200,
|
.rating = 200,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -176,6 +264,9 @@ static int __init mxc_clockevent_init(struct clk *timer_clk)
|
||||||
{
|
{
|
||||||
unsigned int c = clk_get_rate(timer_clk);
|
unsigned int c = clk_get_rate(timer_clk);
|
||||||
|
|
||||||
|
if (cpu_is_mx3())
|
||||||
|
clockevent_mxc.set_next_event = mx3_set_next_event;
|
||||||
|
|
||||||
clockevent_mxc.mult = div_sc(c, NSEC_PER_SEC,
|
clockevent_mxc.mult = div_sc(c, NSEC_PER_SEC,
|
||||||
clockevent_mxc.shift);
|
clockevent_mxc.shift);
|
||||||
clockevent_mxc.max_delta_ns =
|
clockevent_mxc.max_delta_ns =
|
||||||
|
@ -192,23 +283,47 @@ static int __init mxc_clockevent_init(struct clk *timer_clk)
|
||||||
|
|
||||||
void __init mxc_timer_init(struct clk *timer_clk)
|
void __init mxc_timer_init(struct clk *timer_clk)
|
||||||
{
|
{
|
||||||
|
uint32_t tctl_val;
|
||||||
|
int irq;
|
||||||
|
|
||||||
clk_enable(timer_clk);
|
clk_enable(timer_clk);
|
||||||
|
|
||||||
|
if (cpu_is_mx1()) {
|
||||||
|
#ifdef CONFIG_ARCH_MX1
|
||||||
|
timer_base = IO_ADDRESS(TIM1_BASE_ADDR);
|
||||||
|
irq = TIM1_INT;
|
||||||
|
#endif
|
||||||
|
} else if (cpu_is_mx2()) {
|
||||||
|
#ifdef CONFIG_ARCH_MX2
|
||||||
|
timer_base = IO_ADDRESS(GPT1_BASE_ADDR);
|
||||||
|
irq = MXC_INT_GPT1;
|
||||||
|
#endif
|
||||||
|
} else if (cpu_is_mx3()) {
|
||||||
|
#ifdef CONFIG_ARCH_MX3
|
||||||
|
timer_base = IO_ADDRESS(GPT1_BASE_ADDR);
|
||||||
|
irq = MXC_INT_GPT;
|
||||||
|
#endif
|
||||||
|
} else
|
||||||
|
BUG();
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Initialise to a known state (all timers off, and timing reset)
|
* Initialise to a known state (all timers off, and timing reset)
|
||||||
*/
|
*/
|
||||||
__raw_writel(0, TIMER_BASE + MXC_TCTL);
|
|
||||||
__raw_writel(0, TIMER_BASE + MXC_TPRER); /* see datasheet note */
|
|
||||||
|
|
||||||
__raw_writel(TCTL_FRR | /* free running */
|
__raw_writel(0, timer_base + MXC_TCTL);
|
||||||
TCTL_VAL | /* set clocksource and arch specific bits */
|
__raw_writel(0, timer_base + MXC_TPRER); /* see datasheet note */
|
||||||
TCTL_TEN, /* start the timer */
|
|
||||||
TIMER_BASE + MXC_TCTL);
|
if (cpu_is_mx3())
|
||||||
|
tctl_val = MX3_TCTL_CLK_IPG | MX3_TCTL_FRR | MX3_TCTL_WAITEN | MXC_TCTL_TEN;
|
||||||
|
else
|
||||||
|
tctl_val = MX1_2_TCTL_FRR | MX1_2_TCTL_CLK_PCLK1 | MXC_TCTL_TEN;
|
||||||
|
|
||||||
|
__raw_writel(tctl_val, timer_base + MXC_TCTL);
|
||||||
|
|
||||||
/* init and register the timer to the framework */
|
/* init and register the timer to the framework */
|
||||||
mxc_clocksource_init(timer_clk);
|
mxc_clocksource_init(timer_clk);
|
||||||
mxc_clockevent_init(timer_clk);
|
mxc_clockevent_init(timer_clk);
|
||||||
|
|
||||||
/* Make irqs happen */
|
/* Make irqs happen */
|
||||||
setup_irq(TIMER_INTERRUPT, &mxc_timer_irq);
|
setup_irq(irq, &mxc_timer_irq);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue