diff options
Diffstat (limited to 'arch/arm/plat-nomadik/timer.c')
-rw-r--r-- | arch/arm/plat-nomadik/timer.c | 145 |
1 files changed, 86 insertions, 59 deletions
diff --git a/arch/arm/plat-nomadik/timer.c b/arch/arm/plat-nomadik/timer.c index fa7cb3a57cbf..0ff3798769ab 100644 --- a/arch/arm/plat-nomadik/timer.c +++ b/arch/arm/plat-nomadik/timer.c | |||
@@ -2,7 +2,7 @@ | |||
2 | * linux/arch/arm/mach-nomadik/timer.c | 2 | * linux/arch/arm/mach-nomadik/timer.c |
3 | * | 3 | * |
4 | * Copyright (C) 2008 STMicroelectronics | 4 | * Copyright (C) 2008 STMicroelectronics |
5 | * Copyright (C) 2009 Alessandro Rubini, somewhat based on at91sam926x | 5 | * Copyright (C) 2010 Alessandro Rubini |
6 | * | 6 | * |
7 | * This program is free software; you can redistribute it and/or modify | 7 | * This program is free software; you can redistribute it and/or modify |
8 | * it under the terms of the GNU General Public License version 2, as | 8 | * it under the terms of the GNU General Public License version 2, as |
@@ -18,123 +18,150 @@ | |||
18 | 18 | ||
19 | #include <plat/mtu.h> | 19 | #include <plat/mtu.h> |
20 | 20 | ||
21 | static u32 nmdk_count; /* accumulated count */ | 21 | void __iomem *mtu_base; /* ssigned by machine code */ |
22 | static u32 nmdk_cycle; /* write-once */ | ||
23 | |||
24 | /* setup by the platform code */ | ||
25 | void __iomem *mtu_base; | ||
26 | 22 | ||
27 | /* | 23 | /* |
28 | * clocksource: the MTU device is a decrementing counters, so we negate | 24 | * Kernel assumes that sched_clock can be called early |
29 | * the value being read. | 25 | * but the MTU may not yet be initialized. |
30 | */ | 26 | */ |
31 | static cycle_t nmdk_read_timer(struct clocksource *cs) | 27 | static cycle_t nmdk_read_timer_dummy(struct clocksource *cs) |
32 | { | 28 | { |
33 | u32 count = readl(mtu_base + MTU_VAL(0)); | 29 | return 0; |
34 | return nmdk_count + nmdk_cycle - count; | 30 | } |
35 | 31 | ||
32 | /* clocksource: MTU decrements, so we negate the value being read. */ | ||
33 | static cycle_t nmdk_read_timer(struct clocksource *cs) | ||
34 | { | ||
35 | return -readl(mtu_base + MTU_VAL(0)); | ||
36 | } | 36 | } |
37 | 37 | ||
38 | static struct clocksource nmdk_clksrc = { | 38 | static struct clocksource nmdk_clksrc = { |
39 | .name = "mtu_0", | 39 | .name = "mtu_0", |
40 | .rating = 120, | 40 | .rating = 200, |
41 | .read = nmdk_read_timer, | 41 | .read = nmdk_read_timer_dummy, |
42 | .mask = CLOCKSOURCE_MASK(32), | ||
42 | .shift = 20, | 43 | .shift = 20, |
43 | .flags = CLOCK_SOURCE_IS_CONTINUOUS, | 44 | .flags = CLOCK_SOURCE_IS_CONTINUOUS, |
44 | }; | 45 | }; |
45 | 46 | ||
46 | /* | 47 | /* |
47 | * Clockevent device: currently only periodic mode is supported | 48 | * Override the global weak sched_clock symbol with this |
49 | * local implementation which uses the clocksource to get some | ||
50 | * better resolution when scheduling the kernel. We accept that | ||
51 | * this wraps around for now, since it is just a relative time | ||
52 | * stamp. (Inspired by OMAP implementation.) | ||
48 | */ | 53 | */ |
54 | unsigned long long notrace sched_clock(void) | ||
55 | { | ||
56 | return clocksource_cyc2ns(nmdk_clksrc.read( | ||
57 | &nmdk_clksrc), | ||
58 | nmdk_clksrc.mult, | ||
59 | nmdk_clksrc.shift); | ||
60 | } | ||
61 | |||
62 | /* Clockevent device: use one-shot mode */ | ||
49 | static void nmdk_clkevt_mode(enum clock_event_mode mode, | 63 | static void nmdk_clkevt_mode(enum clock_event_mode mode, |
50 | struct clock_event_device *dev) | 64 | struct clock_event_device *dev) |
51 | { | 65 | { |
66 | u32 cr; | ||
67 | |||
52 | switch (mode) { | 68 | switch (mode) { |
53 | case CLOCK_EVT_MODE_PERIODIC: | 69 | case CLOCK_EVT_MODE_PERIODIC: |
54 | /* count current value? */ | 70 | pr_err("%s: periodic mode not supported\n", __func__); |
55 | writel(readl(mtu_base + MTU_IMSC) | 1, mtu_base + MTU_IMSC); | ||
56 | break; | 71 | break; |
57 | case CLOCK_EVT_MODE_ONESHOT: | 72 | case CLOCK_EVT_MODE_ONESHOT: |
58 | BUG(); /* Not supported, yet */ | 73 | /* Load highest value, enable device, enable interrupts */ |
59 | /* FALLTHROUGH */ | 74 | cr = readl(mtu_base + MTU_CR(1)); |
75 | writel(0, mtu_base + MTU_LR(1)); | ||
76 | writel(cr | MTU_CRn_ENA, mtu_base + MTU_CR(1)); | ||
77 | writel(0x2, mtu_base + MTU_IMSC); | ||
78 | break; | ||
60 | case CLOCK_EVT_MODE_SHUTDOWN: | 79 | case CLOCK_EVT_MODE_SHUTDOWN: |
61 | case CLOCK_EVT_MODE_UNUSED: | 80 | case CLOCK_EVT_MODE_UNUSED: |
62 | writel(readl(mtu_base + MTU_IMSC) & ~1, mtu_base + MTU_IMSC); | 81 | /* disable irq */ |
82 | writel(0, mtu_base + MTU_IMSC); | ||
63 | break; | 83 | break; |
64 | case CLOCK_EVT_MODE_RESUME: | 84 | case CLOCK_EVT_MODE_RESUME: |
65 | break; | 85 | break; |
66 | } | 86 | } |
67 | } | 87 | } |
68 | 88 | ||
89 | static int nmdk_clkevt_next(unsigned long evt, struct clock_event_device *ev) | ||
90 | { | ||
91 | /* writing the value has immediate effect */ | ||
92 | writel(evt, mtu_base + MTU_LR(1)); | ||
93 | return 0; | ||
94 | } | ||
95 | |||
69 | static struct clock_event_device nmdk_clkevt = { | 96 | static struct clock_event_device nmdk_clkevt = { |
70 | .name = "mtu_0", | 97 | .name = "mtu_1", |
71 | .features = CLOCK_EVT_FEAT_PERIODIC, | 98 | .features = CLOCK_EVT_FEAT_ONESHOT, |
72 | .shift = 32, | 99 | .shift = 32, |
73 | .rating = 100, | 100 | .rating = 200, |
74 | .set_mode = nmdk_clkevt_mode, | 101 | .set_mode = nmdk_clkevt_mode, |
102 | .set_next_event = nmdk_clkevt_next, | ||
75 | }; | 103 | }; |
76 | 104 | ||
77 | /* | 105 | /* |
78 | * IRQ Handler for the timer 0 of the MTU block. The irq is not shared | 106 | * IRQ Handler for timer 1 of the MTU block. |
79 | * as we are the only users of mtu0 by now. | ||
80 | */ | 107 | */ |
81 | static irqreturn_t nmdk_timer_interrupt(int irq, void *dev_id) | 108 | static irqreturn_t nmdk_timer_interrupt(int irq, void *dev_id) |
82 | { | 109 | { |
83 | /* ack: "interrupt clear register" */ | 110 | struct clock_event_device *evdev = dev_id; |
84 | writel(1 << 0, mtu_base + MTU_ICR); | ||
85 | |||
86 | /* we can't count lost ticks, unfortunately */ | ||
87 | nmdk_count += nmdk_cycle; | ||
88 | nmdk_clkevt.event_handler(&nmdk_clkevt); | ||
89 | 111 | ||
112 | writel(1 << 1, mtu_base + MTU_ICR); /* Interrupt clear reg */ | ||
113 | evdev->event_handler(evdev); | ||
90 | return IRQ_HANDLED; | 114 | return IRQ_HANDLED; |
91 | } | 115 | } |
92 | 116 | ||
93 | /* | ||
94 | * Set up timer interrupt, and return the current time in seconds. | ||
95 | */ | ||
96 | static struct irqaction nmdk_timer_irq = { | 117 | static struct irqaction nmdk_timer_irq = { |
97 | .name = "Nomadik Timer Tick", | 118 | .name = "Nomadik Timer Tick", |
98 | .flags = IRQF_DISABLED | IRQF_TIMER, | 119 | .flags = IRQF_DISABLED | IRQF_TIMER, |
99 | .handler = nmdk_timer_interrupt, | 120 | .handler = nmdk_timer_interrupt, |
121 | .dev_id = &nmdk_clkevt, | ||
100 | }; | 122 | }; |
101 | 123 | ||
102 | static void nmdk_timer_reset(void) | ||
103 | { | ||
104 | u32 cr; | ||
105 | |||
106 | writel(0, mtu_base + MTU_CR(0)); /* off */ | ||
107 | |||
108 | /* configure load and background-load, and fire it up */ | ||
109 | writel(nmdk_cycle, mtu_base + MTU_LR(0)); | ||
110 | writel(nmdk_cycle, mtu_base + MTU_BGLR(0)); | ||
111 | cr = MTU_CRn_PERIODIC | MTU_CRn_PRESCALE_1 | MTU_CRn_32BITS; | ||
112 | writel(cr, mtu_base + MTU_CR(0)); | ||
113 | writel(cr | MTU_CRn_ENA, mtu_base + MTU_CR(0)); | ||
114 | } | ||
115 | |||
116 | void __init nmdk_timer_init(void) | 124 | void __init nmdk_timer_init(void) |
117 | { | 125 | { |
118 | unsigned long rate; | 126 | unsigned long rate; |
119 | int bits; | 127 | u32 cr = MTU_CRn_32BITS;; |
120 | 128 | ||
121 | rate = CLOCK_TICK_RATE; /* 2.4MHz */ | 129 | /* |
122 | nmdk_cycle = (rate + HZ/2) / HZ; | 130 | * Tick rate is 2.4MHz for Nomadik and 110MHz for ux500: |
131 | * use a divide-by-16 counter if it's more than 16MHz | ||
132 | */ | ||
133 | rate = CLOCK_TICK_RATE; | ||
134 | if (rate > 16 << 20) { | ||
135 | rate /= 16; | ||
136 | cr |= MTU_CRn_PRESCALE_16; | ||
137 | } else { | ||
138 | cr |= MTU_CRn_PRESCALE_1; | ||
139 | } | ||
123 | 140 | ||
124 | /* Init the timer and register clocksource */ | 141 | /* Timer 0 is the free running clocksource */ |
125 | nmdk_timer_reset(); | 142 | writel(cr, mtu_base + MTU_CR(0)); |
143 | writel(0, mtu_base + MTU_LR(0)); | ||
144 | writel(0, mtu_base + MTU_BGLR(0)); | ||
145 | writel(cr | MTU_CRn_ENA, mtu_base + MTU_CR(0)); | ||
126 | 146 | ||
127 | nmdk_clksrc.mult = clocksource_hz2mult(rate, nmdk_clksrc.shift); | 147 | nmdk_clksrc.mult = clocksource_hz2mult(rate, nmdk_clksrc.shift); |
128 | bits = 8*sizeof(nmdk_count); | 148 | /* Now the scheduling clock is ready */ |
129 | nmdk_clksrc.mask = CLOCKSOURCE_MASK(bits); | 149 | nmdk_clksrc.read = nmdk_read_timer; |
130 | 150 | ||
131 | if (clocksource_register(&nmdk_clksrc)) | 151 | if (clocksource_register(&nmdk_clksrc)) |
132 | printk(KERN_ERR "timer: failed to initialize clock " | 152 | pr_err("timer: failed to initialize clock source %s\n", |
133 | "source %s\n", nmdk_clksrc.name); | 153 | nmdk_clksrc.name); |
154 | |||
155 | /* Timer 1 is used for events, fix according to rate */ | ||
156 | writel(cr | MTU_CRn_ONESHOT, mtu_base + MTU_CR(1)); /* off, currently */ | ||
157 | nmdk_clkevt.mult = div_sc(rate, NSEC_PER_SEC, nmdk_clkevt.shift); | ||
158 | nmdk_clkevt.max_delta_ns = | ||
159 | clockevent_delta2ns(0xffffffff, &nmdk_clkevt); | ||
160 | nmdk_clkevt.min_delta_ns = | ||
161 | clockevent_delta2ns(0x00000002, &nmdk_clkevt); | ||
162 | nmdk_clkevt.cpumask = cpumask_of(0); | ||
134 | 163 | ||
135 | /* Register irq and clockevents */ | 164 | /* Register irq and clockevents */ |
136 | setup_irq(IRQ_MTU0, &nmdk_timer_irq); | 165 | setup_irq(IRQ_MTU0, &nmdk_timer_irq); |
137 | nmdk_clkevt.mult = div_sc(rate, NSEC_PER_SEC, nmdk_clkevt.shift); | ||
138 | nmdk_clkevt.cpumask = cpumask_of(0); | ||
139 | clockevents_register_device(&nmdk_clkevt); | 166 | clockevents_register_device(&nmdk_clkevt); |
140 | } | 167 | } |