| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 1 | /* | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 2 | * Based on arm clockevents implementation and old bfin time tick. | 
|  | 3 | * | 
| Robin Getz | 96f1050 | 2009-09-24 14:11:24 +0000 | [diff] [blame] | 4 | * Copyright 2008-2009 Analog Devics Inc. | 
|  | 5 | *                2008 GeoTechnologies | 
|  | 6 | *                     Vitja Makarov | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 7 | * | 
| Robin Getz | 96f1050 | 2009-09-24 14:11:24 +0000 | [diff] [blame] | 8 | * Licensed under the GPL-2 | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 9 | */ | 
| Robin Getz | 96f1050 | 2009-09-24 14:11:24 +0000 | [diff] [blame] | 10 |  | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 11 | #include <linux/module.h> | 
|  | 12 | #include <linux/profile.h> | 
|  | 13 | #include <linux/interrupt.h> | 
|  | 14 | #include <linux/time.h> | 
| Mike Frysinger | 764cb81 | 2008-04-24 05:07:29 +0800 | [diff] [blame] | 15 | #include <linux/timex.h> | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 16 | #include <linux/irq.h> | 
|  | 17 | #include <linux/clocksource.h> | 
|  | 18 | #include <linux/clockchips.h> | 
| Michael Hennerich | e6c91b6 | 2008-04-25 04:58:29 +0800 | [diff] [blame] | 19 | #include <linux/cpufreq.h> | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 20 |  | 
|  | 21 | #include <asm/blackfin.h> | 
| Michael Hennerich | e6c91b6 | 2008-04-25 04:58:29 +0800 | [diff] [blame] | 22 | #include <asm/time.h> | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 23 | #include <asm/gptimers.h> | 
| Graf Yang | 60ffdb3 | 2010-01-20 10:56:24 +0000 | [diff] [blame^] | 24 | #include <asm/nmi.h> | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 25 |  | 
| Michael Hennerich | e6c91b6 | 2008-04-25 04:58:29 +0800 | [diff] [blame] | 26 | /* Accelerators for sched_clock() | 
|  | 27 | * convert from cycles(64bits) => nanoseconds (64bits) | 
|  | 28 | *  basic equation: | 
|  | 29 | *		ns = cycles / (freq / ns_per_sec) | 
|  | 30 | *		ns = cycles * (ns_per_sec / freq) | 
|  | 31 | *		ns = cycles * (10^9 / (cpu_khz * 10^3)) | 
|  | 32 | *		ns = cycles * (10^6 / cpu_khz) | 
|  | 33 | * | 
|  | 34 | *	Then we use scaling math (suggested by george@mvista.com) to get: | 
|  | 35 | *		ns = cycles * (10^6 * SC / cpu_khz) / SC | 
|  | 36 | *		ns = cycles * cyc2ns_scale / SC | 
|  | 37 | * | 
|  | 38 | *	And since SC is a constant power of two, we can convert the div | 
|  | 39 | *  into a shift. | 
|  | 40 | * | 
|  | 41 | *  We can use khz divisor instead of mhz to keep a better precision, since | 
|  | 42 | *  cyc2ns_scale is limited to 10^6 * 2^10, which fits in 32 bits. | 
|  | 43 | *  (mathieu.desnoyers@polymtl.ca) | 
|  | 44 | * | 
|  | 45 | *			-johnstul@us.ibm.com "math is hard, lets go shopping!" | 
|  | 46 | */ | 
|  | 47 |  | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 48 | #define CYC2NS_SCALE_FACTOR 10 /* 2^10, carefully chosen */ | 
|  | 49 |  | 
| Yi Li | ceb33be | 2009-09-15 06:50:51 +0000 | [diff] [blame] | 50 | #if defined(CONFIG_CYCLES_CLOCKSOURCE) | 
|  | 51 |  | 
| Yi Li | ceb33be | 2009-09-15 06:50:51 +0000 | [diff] [blame] | 52 | static notrace cycle_t bfin_read_cycles(struct clocksource *cs) | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 53 | { | 
| Vitja Makarov | 1bfb4b2 | 2008-05-07 11:41:26 +0800 | [diff] [blame] | 54 | return __bfin_cycles_off + (get_cycles() << __bfin_cycles_mod); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 55 | } | 
|  | 56 |  | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 57 | static struct clocksource bfin_cs_cycles = { | 
|  | 58 | .name		= "bfin_cs_cycles", | 
| Graf Yang | e78feaa | 2009-09-14 04:41:00 +0000 | [diff] [blame] | 59 | .rating		= 400, | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 60 | .read		= bfin_read_cycles, | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 61 | .mask		= CLOCKSOURCE_MASK(64), | 
| Yi Li | 2985712 | 2009-09-15 08:55:47 +0000 | [diff] [blame] | 62 | .shift		= CYC2NS_SCALE_FACTOR, | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 63 | .flags		= CLOCK_SOURCE_IS_CONTINUOUS, | 
|  | 64 | }; | 
|  | 65 |  | 
| Yi Li | ceb33be | 2009-09-15 06:50:51 +0000 | [diff] [blame] | 66 | static inline unsigned long long bfin_cs_cycles_sched_clock(void) | 
| Magnus Damm | 8e19608 | 2009-04-21 12:24:00 -0700 | [diff] [blame] | 67 | { | 
| Mike Frysinger | c768a94 | 2009-12-04 03:32:11 +0000 | [diff] [blame] | 68 | return clocksource_cyc2ns(bfin_read_cycles(&bfin_cs_cycles), | 
|  | 69 | bfin_cs_cycles.mult, bfin_cs_cycles.shift); | 
| Magnus Damm | 8e19608 | 2009-04-21 12:24:00 -0700 | [diff] [blame] | 70 | } | 
|  | 71 |  | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 72 | static int __init bfin_cs_cycles_init(void) | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 73 | { | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 74 | bfin_cs_cycles.mult = \ | 
|  | 75 | clocksource_hz2mult(get_cclk(), bfin_cs_cycles.shift); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 76 |  | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 77 | if (clocksource_register(&bfin_cs_cycles)) | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 78 | panic("failed to register clocksource"); | 
|  | 79 |  | 
|  | 80 | return 0; | 
|  | 81 | } | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 82 | #else | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 83 | # define bfin_cs_cycles_init() | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 84 | #endif | 
|  | 85 |  | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 86 | #ifdef CONFIG_GPTMR0_CLOCKSOURCE | 
|  | 87 |  | 
|  | 88 | void __init setup_gptimer0(void) | 
|  | 89 | { | 
|  | 90 | disable_gptimers(TIMER0bit); | 
|  | 91 |  | 
|  | 92 | set_gptimer_config(TIMER0_id, \ | 
|  | 93 | TIMER_OUT_DIS | TIMER_PERIOD_CNT | TIMER_MODE_PWM); | 
|  | 94 | set_gptimer_period(TIMER0_id, -1); | 
|  | 95 | set_gptimer_pwidth(TIMER0_id, -2); | 
|  | 96 | SSYNC(); | 
|  | 97 | enable_gptimers(TIMER0bit); | 
|  | 98 | } | 
|  | 99 |  | 
| Yi Li | f7036d6 | 2009-09-15 02:08:50 +0000 | [diff] [blame] | 100 | static cycle_t bfin_read_gptimer0(struct clocksource *cs) | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 101 | { | 
|  | 102 | return bfin_read_TIMER0_COUNTER(); | 
|  | 103 | } | 
|  | 104 |  | 
|  | 105 | static struct clocksource bfin_cs_gptimer0 = { | 
|  | 106 | .name		= "bfin_cs_gptimer0", | 
| Graf Yang | e78feaa | 2009-09-14 04:41:00 +0000 | [diff] [blame] | 107 | .rating		= 350, | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 108 | .read		= bfin_read_gptimer0, | 
|  | 109 | .mask		= CLOCKSOURCE_MASK(32), | 
| Yi Li | 2985712 | 2009-09-15 08:55:47 +0000 | [diff] [blame] | 110 | .shift		= CYC2NS_SCALE_FACTOR, | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 111 | .flags		= CLOCK_SOURCE_IS_CONTINUOUS, | 
|  | 112 | }; | 
|  | 113 |  | 
| Yi Li | ceb33be | 2009-09-15 06:50:51 +0000 | [diff] [blame] | 114 | static inline unsigned long long bfin_cs_gptimer0_sched_clock(void) | 
|  | 115 | { | 
| Mike Frysinger | c768a94 | 2009-12-04 03:32:11 +0000 | [diff] [blame] | 116 | return clocksource_cyc2ns(bfin_read_TIMER0_COUNTER(), | 
|  | 117 | bfin_cs_gptimer0.mult, bfin_cs_gptimer0.shift); | 
| Yi Li | ceb33be | 2009-09-15 06:50:51 +0000 | [diff] [blame] | 118 | } | 
|  | 119 |  | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 120 | static int __init bfin_cs_gptimer0_init(void) | 
|  | 121 | { | 
|  | 122 | setup_gptimer0(); | 
|  | 123 |  | 
|  | 124 | bfin_cs_gptimer0.mult = \ | 
|  | 125 | clocksource_hz2mult(get_sclk(), bfin_cs_gptimer0.shift); | 
|  | 126 |  | 
|  | 127 | if (clocksource_register(&bfin_cs_gptimer0)) | 
|  | 128 | panic("failed to register clocksource"); | 
|  | 129 |  | 
|  | 130 | return 0; | 
|  | 131 | } | 
|  | 132 | #else | 
|  | 133 | # define bfin_cs_gptimer0_init() | 
|  | 134 | #endif | 
|  | 135 |  | 
| Yi Li | ceb33be | 2009-09-15 06:50:51 +0000 | [diff] [blame] | 136 | #if defined(CONFIG_GPTMR0_CLOCKSOURCE) || defined(CONFIG_CYCLES_CLOCKSOURCE) | 
|  | 137 | /* prefer to use cycles since it has higher rating */ | 
|  | 138 | notrace unsigned long long sched_clock(void) | 
|  | 139 | { | 
|  | 140 | #if defined(CONFIG_CYCLES_CLOCKSOURCE) | 
|  | 141 | return bfin_cs_cycles_sched_clock(); | 
|  | 142 | #else | 
|  | 143 | return bfin_cs_gptimer0_sched_clock(); | 
|  | 144 | #endif | 
|  | 145 | } | 
|  | 146 | #endif | 
|  | 147 |  | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 148 | #if defined(CONFIG_TICKSOURCE_GPTMR0) | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 149 | static int bfin_gptmr0_set_next_event(unsigned long cycles, | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 150 | struct clock_event_device *evt) | 
|  | 151 | { | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 152 | disable_gptimers(TIMER0bit); | 
|  | 153 |  | 
|  | 154 | /* it starts counting three SCLK cycles after the TIMENx bit is set */ | 
|  | 155 | set_gptimer_pwidth(TIMER0_id, cycles - 3); | 
|  | 156 | enable_gptimers(TIMER0bit); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 157 | return 0; | 
|  | 158 | } | 
|  | 159 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 160 | static void bfin_gptmr0_set_mode(enum clock_event_mode mode, | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 161 | struct clock_event_device *evt) | 
|  | 162 | { | 
|  | 163 | switch (mode) { | 
|  | 164 | case CLOCK_EVT_MODE_PERIODIC: { | 
|  | 165 | set_gptimer_config(TIMER0_id, \ | 
|  | 166 | TIMER_OUT_DIS | TIMER_IRQ_ENA | \ | 
|  | 167 | TIMER_PERIOD_CNT | TIMER_MODE_PWM); | 
|  | 168 | set_gptimer_period(TIMER0_id, get_sclk() / HZ); | 
|  | 169 | set_gptimer_pwidth(TIMER0_id, get_sclk() / HZ - 1); | 
|  | 170 | enable_gptimers(TIMER0bit); | 
|  | 171 | break; | 
|  | 172 | } | 
|  | 173 | case CLOCK_EVT_MODE_ONESHOT: | 
|  | 174 | disable_gptimers(TIMER0bit); | 
|  | 175 | set_gptimer_config(TIMER0_id, \ | 
|  | 176 | TIMER_OUT_DIS | TIMER_IRQ_ENA | TIMER_MODE_PWM); | 
|  | 177 | set_gptimer_period(TIMER0_id, 0); | 
|  | 178 | break; | 
|  | 179 | case CLOCK_EVT_MODE_UNUSED: | 
|  | 180 | case CLOCK_EVT_MODE_SHUTDOWN: | 
|  | 181 | disable_gptimers(TIMER0bit); | 
|  | 182 | break; | 
|  | 183 | case CLOCK_EVT_MODE_RESUME: | 
|  | 184 | break; | 
|  | 185 | } | 
|  | 186 | } | 
|  | 187 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 188 | static void bfin_gptmr0_ack(void) | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 189 | { | 
|  | 190 | set_gptimer_status(TIMER_GROUP1, TIMER_STATUS_TIMIL0); | 
|  | 191 | } | 
|  | 192 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 193 | static void __init bfin_gptmr0_init(void) | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 194 | { | 
|  | 195 | disable_gptimers(TIMER0bit); | 
|  | 196 | } | 
|  | 197 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 198 | #ifdef CONFIG_CORE_TIMER_IRQ_L1 | 
|  | 199 | __attribute__((l1_text)) | 
|  | 200 | #endif | 
|  | 201 | irqreturn_t bfin_gptmr0_interrupt(int irq, void *dev_id) | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 202 | { | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 203 | struct clock_event_device *evt = dev_id; | 
|  | 204 | smp_mb(); | 
|  | 205 | evt->event_handler(evt); | 
|  | 206 | bfin_gptmr0_ack(); | 
|  | 207 | return IRQ_HANDLED; | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 208 | } | 
|  | 209 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 210 | static struct irqaction gptmr0_irq = { | 
|  | 211 | .name		= "Blackfin GPTimer0", | 
|  | 212 | .flags		= IRQF_DISABLED | IRQF_TIMER | \ | 
|  | 213 | IRQF_IRQPOLL | IRQF_PERCPU, | 
|  | 214 | .handler	= bfin_gptmr0_interrupt, | 
|  | 215 | }; | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 216 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 217 | static struct clock_event_device clockevent_gptmr0 = { | 
|  | 218 | .name		= "bfin_gptimer0", | 
|  | 219 | .rating		= 300, | 
|  | 220 | .irq		= IRQ_TIMER0, | 
|  | 221 | .shift		= 32, | 
|  | 222 | .features	= CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT, | 
|  | 223 | .set_next_event = bfin_gptmr0_set_next_event, | 
|  | 224 | .set_mode	= bfin_gptmr0_set_mode, | 
|  | 225 | }; | 
|  | 226 |  | 
|  | 227 | static void __init bfin_gptmr0_clockevent_init(struct clock_event_device *evt) | 
|  | 228 | { | 
|  | 229 | unsigned long clock_tick; | 
|  | 230 |  | 
|  | 231 | clock_tick = get_sclk(); | 
|  | 232 | evt->mult = div_sc(clock_tick, NSEC_PER_SEC, evt->shift); | 
|  | 233 | evt->max_delta_ns = clockevent_delta2ns(-1, evt); | 
|  | 234 | evt->min_delta_ns = clockevent_delta2ns(100, evt); | 
|  | 235 |  | 
|  | 236 | evt->cpumask = cpumask_of(0); | 
|  | 237 |  | 
|  | 238 | clockevents_register_device(evt); | 
|  | 239 | } | 
|  | 240 | #endif /* CONFIG_TICKSOURCE_GPTMR0 */ | 
|  | 241 |  | 
|  | 242 | #if defined(CONFIG_TICKSOURCE_CORETMR) | 
|  | 243 | /* per-cpu local core timer */ | 
|  | 244 | static DEFINE_PER_CPU(struct clock_event_device, coretmr_events); | 
|  | 245 |  | 
|  | 246 | static int bfin_coretmr_set_next_event(unsigned long cycles, | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 247 | struct clock_event_device *evt) | 
|  | 248 | { | 
|  | 249 | bfin_write_TCNTL(TMPWR); | 
|  | 250 | CSYNC(); | 
|  | 251 | bfin_write_TCOUNT(cycles); | 
|  | 252 | CSYNC(); | 
|  | 253 | bfin_write_TCNTL(TMPWR | TMREN); | 
|  | 254 | return 0; | 
|  | 255 | } | 
|  | 256 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 257 | static void bfin_coretmr_set_mode(enum clock_event_mode mode, | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 258 | struct clock_event_device *evt) | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 259 | { | 
|  | 260 | switch (mode) { | 
|  | 261 | case CLOCK_EVT_MODE_PERIODIC: { | 
| Michael Hennerich | e6c91b6 | 2008-04-25 04:58:29 +0800 | [diff] [blame] | 262 | unsigned long tcount = ((get_cclk() / (HZ * TIME_SCALE)) - 1); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 263 | bfin_write_TCNTL(TMPWR); | 
|  | 264 | CSYNC(); | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 265 | bfin_write_TSCALE(TIME_SCALE - 1); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 266 | bfin_write_TPERIOD(tcount); | 
|  | 267 | bfin_write_TCOUNT(tcount); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 268 | CSYNC(); | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 269 | bfin_write_TCNTL(TMPWR | TMREN | TAUTORLD); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 270 | break; | 
|  | 271 | } | 
|  | 272 | case CLOCK_EVT_MODE_ONESHOT: | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 273 | bfin_write_TCNTL(TMPWR); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 274 | CSYNC(); | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 275 | bfin_write_TSCALE(TIME_SCALE - 1); | 
|  | 276 | bfin_write_TPERIOD(0); | 
|  | 277 | bfin_write_TCOUNT(0); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 278 | break; | 
|  | 279 | case CLOCK_EVT_MODE_UNUSED: | 
|  | 280 | case CLOCK_EVT_MODE_SHUTDOWN: | 
|  | 281 | bfin_write_TCNTL(0); | 
|  | 282 | CSYNC(); | 
|  | 283 | break; | 
|  | 284 | case CLOCK_EVT_MODE_RESUME: | 
|  | 285 | break; | 
|  | 286 | } | 
|  | 287 | } | 
|  | 288 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 289 | void bfin_coretmr_init(void) | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 290 | { | 
|  | 291 | /* power up the timer, but don't enable it just yet */ | 
|  | 292 | bfin_write_TCNTL(TMPWR); | 
|  | 293 | CSYNC(); | 
|  | 294 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 295 | /* the TSCALE prescaler counter. */ | 
| Michael Hennerich | e6c91b6 | 2008-04-25 04:58:29 +0800 | [diff] [blame] | 296 | bfin_write_TSCALE(TIME_SCALE - 1); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 297 | bfin_write_TPERIOD(0); | 
|  | 298 | bfin_write_TCOUNT(0); | 
|  | 299 |  | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 300 | CSYNC(); | 
|  | 301 | } | 
|  | 302 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 303 | #ifdef CONFIG_CORE_TIMER_IRQ_L1 | 
|  | 304 | __attribute__((l1_text)) | 
|  | 305 | #endif | 
|  | 306 | irqreturn_t bfin_coretmr_interrupt(int irq, void *dev_id) | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 307 | { | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 308 | int cpu = smp_processor_id(); | 
|  | 309 | struct clock_event_device *evt = &per_cpu(coretmr_events, cpu); | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 310 |  | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 311 | smp_mb(); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 312 | evt->event_handler(evt); | 
| Graf Yang | 60ffdb3 | 2010-01-20 10:56:24 +0000 | [diff] [blame^] | 313 |  | 
|  | 314 | touch_nmi_watchdog(); | 
|  | 315 |  | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 316 | return IRQ_HANDLED; | 
|  | 317 | } | 
|  | 318 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 319 | static struct irqaction coretmr_irq = { | 
|  | 320 | .name		= "Blackfin CoreTimer", | 
|  | 321 | .flags		= IRQF_DISABLED | IRQF_TIMER | \ | 
|  | 322 | IRQF_IRQPOLL | IRQF_PERCPU, | 
|  | 323 | .handler	= bfin_coretmr_interrupt, | 
|  | 324 | }; | 
|  | 325 |  | 
|  | 326 | void bfin_coretmr_clockevent_init(void) | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 327 | { | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 328 | unsigned long clock_tick; | 
|  | 329 | unsigned int cpu = smp_processor_id(); | 
|  | 330 | struct clock_event_device *evt = &per_cpu(coretmr_events, cpu); | 
| Vitja Makarov | 1bfb4b2 | 2008-05-07 11:41:26 +0800 | [diff] [blame] | 331 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 332 | evt->name = "bfin_core_timer"; | 
|  | 333 | evt->rating = 350; | 
|  | 334 | evt->irq = -1; | 
|  | 335 | evt->shift = 32; | 
|  | 336 | evt->features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT; | 
|  | 337 | evt->set_next_event = bfin_coretmr_set_next_event; | 
|  | 338 | evt->set_mode = bfin_coretmr_set_mode; | 
| Vitja Makarov | 1bfb4b2 | 2008-05-07 11:41:26 +0800 | [diff] [blame] | 339 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 340 | clock_tick = get_cclk() / TIME_SCALE; | 
|  | 341 | evt->mult = div_sc(clock_tick, NSEC_PER_SEC, evt->shift); | 
|  | 342 | evt->max_delta_ns = clockevent_delta2ns(-1, evt); | 
|  | 343 | evt->min_delta_ns = clockevent_delta2ns(100, evt); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 344 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 345 | evt->cpumask = cpumask_of(cpu); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 346 |  | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 347 | clockevents_register_device(evt); | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 348 | } | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 349 | #endif /* CONFIG_TICKSOURCE_CORETMR */ | 
|  | 350 |  | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 351 |  | 
|  | 352 | void __init time_init(void) | 
|  | 353 | { | 
|  | 354 | time_t secs_since_1970 = (365 * 37 + 9) * 24 * 60 * 60;	/* 1 Jan 2007 */ | 
|  | 355 |  | 
|  | 356 | #ifdef CONFIG_RTC_DRV_BFIN | 
|  | 357 | /* [#2663] hack to filter junk RTC values that would cause | 
|  | 358 | * userspace to have to deal with time values greater than | 
|  | 359 | * 2^31 seconds (which uClibc cannot cope with yet) | 
|  | 360 | */ | 
|  | 361 | if ((bfin_read_RTC_STAT() & 0xC0000000) == 0xC0000000) { | 
|  | 362 | printk(KERN_NOTICE "bfin-rtc: invalid date; resetting\n"); | 
|  | 363 | bfin_write_RTC_STAT(0); | 
|  | 364 | } | 
|  | 365 | #endif | 
|  | 366 |  | 
|  | 367 | /* Initialize xtime. From now on, xtime is updated with timer interrupts */ | 
|  | 368 | xtime.tv_sec = secs_since_1970; | 
|  | 369 | xtime.tv_nsec = 0; | 
|  | 370 | set_normalized_timespec(&wall_to_monotonic, -xtime.tv_sec, -xtime.tv_nsec); | 
|  | 371 |  | 
| Graf Yang | 1fa9be7 | 2009-05-15 11:01:59 +0000 | [diff] [blame] | 372 | bfin_cs_cycles_init(); | 
|  | 373 | bfin_cs_gptimer0_init(); | 
| Yi Li | 0d152c2 | 2009-12-28 10:21:49 +0000 | [diff] [blame] | 374 |  | 
|  | 375 | #if defined(CONFIG_TICKSOURCE_CORETMR) | 
|  | 376 | bfin_coretmr_init(); | 
|  | 377 | setup_irq(IRQ_CORETMR, &coretmr_irq); | 
|  | 378 | bfin_coretmr_clockevent_init(); | 
|  | 379 | #endif | 
|  | 380 |  | 
|  | 381 | #if defined(CONFIG_TICKSOURCE_GPTMR0) | 
|  | 382 | bfin_gptmr0_init(); | 
|  | 383 | setup_irq(IRQ_TIMER0, &gptmr0_irq); | 
|  | 384 | gptmr0_irq.dev_id = &clockevent_gptmr0; | 
|  | 385 | bfin_gptmr0_clockevent_init(&clockevent_gptmr0); | 
|  | 386 | #endif | 
|  | 387 |  | 
|  | 388 | #if !defined(CONFIG_TICKSOURCE_CORETMR) && !defined(CONFIG_TICKSOURCE_GPTMR0) | 
|  | 389 | # error at least one clock event device is required | 
|  | 390 | #endif | 
| Vitja Makarov | 8b5f79f | 2008-02-29 12:24:23 +0800 | [diff] [blame] | 391 | } |