diff options
Diffstat (limited to 'arch/arm/mach-realview/localtimer.c')
-rw-r--r-- | arch/arm/mach-realview/localtimer.c | 144 |
1 files changed, 111 insertions, 33 deletions
diff --git a/arch/arm/mach-realview/localtimer.c b/arch/arm/mach-realview/localtimer.c index c7bdf04ab094..50604360479f 100644 --- a/arch/arm/mach-realview/localtimer.c +++ b/arch/arm/mach-realview/localtimer.c | |||
@@ -14,19 +14,75 @@ | |||
14 | #include <linux/device.h> | 14 | #include <linux/device.h> |
15 | #include <linux/smp.h> | 15 | #include <linux/smp.h> |
16 | #include <linux/jiffies.h> | 16 | #include <linux/jiffies.h> |
17 | #include <linux/percpu.h> | ||
18 | #include <linux/clockchips.h> | ||
19 | #include <linux/irq.h> | ||
17 | 20 | ||
18 | #include <asm/mach/time.h> | ||
19 | #include <asm/hardware/arm_twd.h> | 21 | #include <asm/hardware/arm_twd.h> |
20 | #include <asm/hardware/gic.h> | 22 | #include <asm/hardware/gic.h> |
21 | #include <asm/hardware.h> | 23 | #include <asm/hardware.h> |
22 | #include <asm/io.h> | 24 | #include <asm/io.h> |
23 | #include <asm/irq.h> | 25 | #include <asm/irq.h> |
24 | 26 | ||
25 | #define TWD_BASE(cpu) (__io_address(REALVIEW_TWD_BASE) + \ | 27 | static DEFINE_PER_CPU(struct clock_event_device, local_clockevent); |
26 | ((cpu) * REALVIEW_TWD_SIZE)) | 28 | |
29 | /* | ||
30 | * Used on SMP for either the local timer or IPI_TIMER | ||
31 | */ | ||
32 | void local_timer_interrupt(void) | ||
33 | { | ||
34 | struct clock_event_device *clk = &__get_cpu_var(local_clockevent); | ||
35 | |||
36 | clk->event_handler(clk); | ||
37 | } | ||
38 | |||
39 | #ifdef CONFIG_LOCAL_TIMERS | ||
40 | |||
41 | #define TWD_BASE(cpu) (twd_base_addr + (cpu) * twd_size) | ||
42 | |||
43 | /* set up by the platform code */ | ||
44 | void __iomem *twd_base_addr; | ||
45 | unsigned int twd_size; | ||
27 | 46 | ||
28 | static unsigned long mpcore_timer_rate; | 47 | static unsigned long mpcore_timer_rate; |
29 | 48 | ||
49 | static void local_timer_set_mode(enum clock_event_mode mode, | ||
50 | struct clock_event_device *clk) | ||
51 | { | ||
52 | void __iomem *base = TWD_BASE(smp_processor_id()); | ||
53 | unsigned long ctrl; | ||
54 | |||
55 | switch(mode) { | ||
56 | case CLOCK_EVT_MODE_PERIODIC: | ||
57 | /* timer load already set up */ | ||
58 | ctrl = TWD_TIMER_CONTROL_ENABLE | TWD_TIMER_CONTROL_IT_ENABLE | ||
59 | | TWD_TIMER_CONTROL_PERIODIC; | ||
60 | break; | ||
61 | case CLOCK_EVT_MODE_ONESHOT: | ||
62 | /* period set, and timer enabled in 'next_event' hook */ | ||
63 | ctrl = TWD_TIMER_CONTROL_IT_ENABLE | TWD_TIMER_CONTROL_ONESHOT; | ||
64 | break; | ||
65 | case CLOCK_EVT_MODE_UNUSED: | ||
66 | case CLOCK_EVT_MODE_SHUTDOWN: | ||
67 | default: | ||
68 | ctrl = 0; | ||
69 | } | ||
70 | |||
71 | __raw_writel(ctrl, base + TWD_TIMER_CONTROL); | ||
72 | } | ||
73 | |||
74 | static int local_timer_set_next_event(unsigned long evt, | ||
75 | struct clock_event_device *unused) | ||
76 | { | ||
77 | void __iomem *base = TWD_BASE(smp_processor_id()); | ||
78 | unsigned long ctrl = __raw_readl(base + TWD_TIMER_CONTROL); | ||
79 | |||
80 | __raw_writel(evt, base + TWD_TIMER_COUNTER); | ||
81 | __raw_writel(ctrl | TWD_TIMER_CONTROL_ENABLE, base + TWD_TIMER_CONTROL); | ||
82 | |||
83 | return 0; | ||
84 | } | ||
85 | |||
30 | /* | 86 | /* |
31 | * local_timer_ack: checks for a local timer interrupt. | 87 | * local_timer_ack: checks for a local timer interrupt. |
32 | * | 88 | * |
@@ -45,12 +101,11 @@ int local_timer_ack(void) | |||
45 | return 0; | 101 | return 0; |
46 | } | 102 | } |
47 | 103 | ||
48 | void __cpuinit local_timer_setup(unsigned int cpu) | 104 | static void __cpuinit twd_calibrate_rate(unsigned int cpu) |
49 | { | 105 | { |
50 | void __iomem *base = TWD_BASE(cpu); | 106 | void __iomem *base = TWD_BASE(cpu); |
51 | unsigned int load, offset; | 107 | unsigned long load, count; |
52 | u64 waitjiffies; | 108 | u64 waitjiffies; |
53 | unsigned int count; | ||
54 | 109 | ||
55 | /* | 110 | /* |
56 | * If this is the first time round, we need to work out how fast | 111 | * If this is the first time round, we need to work out how fast |
@@ -88,36 +143,36 @@ void __cpuinit local_timer_setup(unsigned int cpu) | |||
88 | load = mpcore_timer_rate / HZ; | 143 | load = mpcore_timer_rate / HZ; |
89 | 144 | ||
90 | __raw_writel(load, base + TWD_TIMER_LOAD); | 145 | __raw_writel(load, base + TWD_TIMER_LOAD); |
91 | __raw_writel(0x7, base + TWD_TIMER_CONTROL); | 146 | } |
92 | |||
93 | /* | ||
94 | * Now maneuver our local tick into the right part of the jiffy. | ||
95 | * Start by working out where within the tick our local timer | ||
96 | * interrupt should go. | ||
97 | */ | ||
98 | offset = ((mpcore_timer_rate / HZ) / (NR_CPUS + 1)) * (cpu + 1); | ||
99 | |||
100 | /* | ||
101 | * gettimeoffset() will return a number of us since the last tick. | ||
102 | * Convert this number of us to a local timer tick count. | ||
103 | * Be careful of integer overflow whilst keeping maximum precision. | ||
104 | * | ||
105 | * with HZ=100 and 1MHz (fpga) ~ 1GHz processor: | ||
106 | * load = 1 ~ 10,000 | ||
107 | * mpcore_timer_rate/10000 = 100 ~ 100,000 | ||
108 | * | ||
109 | * so the multiply value will be less than 10^9 always. | ||
110 | */ | ||
111 | load = (system_timer->offset() * (mpcore_timer_rate / 10000)) / 100; | ||
112 | |||
113 | /* Add on our offset to get the load value */ | ||
114 | load = (load + offset) % (mpcore_timer_rate / HZ); | ||
115 | 147 | ||
116 | __raw_writel(load, base + TWD_TIMER_COUNTER); | 148 | /* |
149 | * Setup the local clock events for a CPU. | ||
150 | */ | ||
151 | void __cpuinit local_timer_setup(unsigned int cpu) | ||
152 | { | ||
153 | struct clock_event_device *clk = &per_cpu(local_clockevent, cpu); | ||
154 | unsigned long flags; | ||
155 | |||
156 | twd_calibrate_rate(cpu); | ||
157 | |||
158 | clk->name = "local_timer"; | ||
159 | clk->features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT; | ||
160 | clk->rating = 350; | ||
161 | clk->set_mode = local_timer_set_mode; | ||
162 | clk->set_next_event = local_timer_set_next_event; | ||
163 | clk->irq = IRQ_LOCALTIMER; | ||
164 | clk->cpumask = cpumask_of_cpu(cpu); | ||
165 | clk->shift = 20; | ||
166 | clk->mult = div_sc(mpcore_timer_rate, NSEC_PER_SEC, clk->shift); | ||
167 | clk->max_delta_ns = clockevent_delta2ns(0xffffffff, clk); | ||
168 | clk->min_delta_ns = clockevent_delta2ns(0xf, clk); | ||
117 | 169 | ||
118 | /* Make sure our local interrupt controller has this enabled */ | 170 | /* Make sure our local interrupt controller has this enabled */ |
119 | __raw_writel(1 << IRQ_LOCALTIMER, | 171 | local_irq_save(flags); |
120 | __io_address(REALVIEW_GIC_DIST_BASE) + GIC_DIST_ENABLE_SET); | 172 | get_irq_chip(IRQ_LOCALTIMER)->unmask(IRQ_LOCALTIMER); |
173 | local_irq_restore(flags); | ||
174 | |||
175 | clockevents_register_device(clk); | ||
121 | } | 176 | } |
122 | 177 | ||
123 | /* | 178 | /* |
@@ -127,3 +182,26 @@ void __cpuexit local_timer_stop(unsigned int cpu) | |||
127 | { | 182 | { |
128 | __raw_writel(0, TWD_BASE(cpu) + TWD_TIMER_CONTROL); | 183 | __raw_writel(0, TWD_BASE(cpu) + TWD_TIMER_CONTROL); |
129 | } | 184 | } |
185 | |||
186 | #else /* CONFIG_LOCAL_TIMERS */ | ||
187 | |||
188 | static void dummy_timer_set_mode(enum clock_event_mode mode, | ||
189 | struct clock_event_device *clk) | ||
190 | { | ||
191 | } | ||
192 | |||
193 | void __cpuinit local_timer_setup(unsigned int cpu) | ||
194 | { | ||
195 | struct clock_event_device *clk = &per_cpu(local_clockevent, cpu); | ||
196 | |||
197 | clk->name = "dummy_timer"; | ||
198 | clk->features = CLOCK_EVT_FEAT_DUMMY; | ||
199 | clk->rating = 200; | ||
200 | clk->set_mode = dummy_timer_set_mode; | ||
201 | clk->broadcast = smp_timer_broadcast; | ||
202 | clk->cpumask = cpumask_of_cpu(cpu); | ||
203 | |||
204 | clockevents_register_device(clk); | ||
205 | } | ||
206 | |||
207 | #endif /* !CONFIG_LOCAL_TIMERS */ | ||