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>
19 #if (CONFIG_NR_CPUS > 2) && !defined(CONFIG_GEENERIC_CLOCKEVENTS_BROADCAST)
20 #error "This doesn't scale well! Need per-core local timers."
22 #else /* CONFIG_SMP */
23 #define stop_jiffies_counter1()
24 #define reload_jiffies_counter1(x)
25 #define TMJC1IRQ TMJCIRQ
29 static int next_event(unsigned long delta
,
30 struct clock_event_device
*evt
)
32 unsigned int cpu
= smp_processor_id();
35 stop_jiffies_counter();
36 reload_jiffies_counter(delta
- 1);
38 stop_jiffies_counter1();
39 reload_jiffies_counter1(delta
- 1);
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();
59 stop_jiffies_counter();
61 stop_jiffies_counter1();
63 cd
= &per_cpu(mn10300_clockevent_device
, cpu
);
64 cd
->event_handler(cd
);
69 static void event_handler(struct clock_event_device
*dev
)
73 static inline void setup_jiffies_interrupt(int irq
,
74 struct irqaction
*action
)
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
;
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
);
92 stop_jiffies_counter();
95 stop_jiffies_counter1();
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
);
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
;
132 reload_jiffies_counter(MN10300_JC_PER_HZ
- 1);
133 iact
->name
= "CPU0 Timer";
135 reload_jiffies_counter1(MN10300_JC_PER_HZ
- 1);
136 iact
->name
= "CPU1 Timer";
139 setup_jiffies_interrupt(cd
->irq
, iact
);