Merge tag 'locks-v3.16-2' of git://git.samba.org/jlayton/linux
[linux/fpc-iii.git] / arch / mn10300 / kernel / cevt-mn10300.c
blob60f64ca1752aff964ee6274ed066ccf9744d5034
1 /* MN10300 clockevents
3 * Copyright (C) 2010 Red Hat, Inc. All Rights Reserved.
4 * Written by Mark Salter (msalter@redhat.com)
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public Licence
8 * as published by the Free Software Foundation; either version
9 * 2 of the Licence, or (at your option) any later version.
11 #include <linux/clockchips.h>
12 #include <linux/interrupt.h>
13 #include <linux/percpu.h>
14 #include <linux/smp.h>
15 #include <asm/timex.h>
16 #include "internal.h"
18 #ifdef CONFIG_SMP
19 #if (CONFIG_NR_CPUS > 2) && !defined(CONFIG_GEENERIC_CLOCKEVENTS_BROADCAST)
20 #error "This doesn't scale well! Need per-core local timers."
21 #endif
22 #else /* CONFIG_SMP */
23 #define stop_jiffies_counter1()
24 #define reload_jiffies_counter1(x)
25 #define TMJC1IRQ TMJCIRQ
26 #endif
29 static int next_event(unsigned long delta,
30 struct clock_event_device *evt)
32 unsigned int cpu = smp_processor_id();
34 if (cpu == 0) {
35 stop_jiffies_counter();
36 reload_jiffies_counter(delta - 1);
37 } else {
38 stop_jiffies_counter1();
39 reload_jiffies_counter1(delta - 1);
41 return 0;
44 static void set_clock_mode(enum clock_event_mode mode,
45 struct clock_event_device *evt)
47 /* Nothing to do ... */
50 static DEFINE_PER_CPU(struct clock_event_device, mn10300_clockevent_device);
51 static DEFINE_PER_CPU(struct irqaction, timer_irq);
53 static irqreturn_t timer_interrupt(int irq, void *dev_id)
55 struct clock_event_device *cd;
56 unsigned int cpu = smp_processor_id();
58 if (cpu == 0)
59 stop_jiffies_counter();
60 else
61 stop_jiffies_counter1();
63 cd = &per_cpu(mn10300_clockevent_device, cpu);
64 cd->event_handler(cd);
66 return IRQ_HANDLED;
69 static void event_handler(struct clock_event_device *dev)
73 static inline void setup_jiffies_interrupt(int irq,
74 struct irqaction *action)
76 u16 tmp;
77 setup_irq(irq, action);
78 set_intr_level(irq, NUM2GxICR_LEVEL(CONFIG_TIMER_IRQ_LEVEL));
79 GxICR(irq) |= GxICR_ENABLE | GxICR_DETECT | GxICR_REQUEST;
80 tmp = GxICR(irq);
83 int __init init_clockevents(void)
85 struct clock_event_device *cd;
86 struct irqaction *iact;
87 unsigned int cpu = smp_processor_id();
89 cd = &per_cpu(mn10300_clockevent_device, cpu);
91 if (cpu == 0) {
92 stop_jiffies_counter();
93 cd->irq = TMJCIRQ;
94 } else {
95 stop_jiffies_counter1();
96 cd->irq = TMJC1IRQ;
99 cd->name = "Timestamp";
100 cd->features = CLOCK_EVT_FEAT_ONESHOT;
102 /* Calculate shift/mult. We want to spawn at least 1 second */
103 clockevents_calc_mult_shift(cd, MN10300_JCCLK, 1);
105 /* Calculate the min / max delta */
106 cd->max_delta_ns = clockevent_delta2ns(TMJCBR_MAX, cd);
107 cd->min_delta_ns = clockevent_delta2ns(100, cd);
109 cd->rating = 200;
110 cd->cpumask = cpumask_of(smp_processor_id());
111 cd->set_mode = set_clock_mode;
112 cd->event_handler = event_handler;
113 cd->set_next_event = next_event;
115 iact = &per_cpu(timer_irq, cpu);
116 iact->flags = IRQF_SHARED | IRQF_TIMER;
117 iact->handler = timer_interrupt;
119 clockevents_register_device(cd);
121 #if defined(CONFIG_SMP) && !defined(CONFIG_GENERIC_CLOCKEVENTS_BROADCAST)
122 /* setup timer irq affinity so it only runs on this cpu */
124 struct irq_data *data;
125 data = irq_get_irq_data(cd->irq);
126 cpumask_copy(data->affinity, cpumask_of(cpu));
127 iact->flags |= IRQF_NOBALANCING;
129 #endif
131 if (cpu == 0) {
132 reload_jiffies_counter(MN10300_JC_PER_HZ - 1);
133 iact->name = "CPU0 Timer";
134 } else {
135 reload_jiffies_counter1(MN10300_JC_PER_HZ - 1);
136 iact->name = "CPU1 Timer";
139 setup_jiffies_interrupt(cd->irq, iact);
141 return 0;