diff options
-rw-r--r-- | arch/m68knommu/Kconfig | 11 | ||||
-rw-r--r-- | arch/m68knommu/kernel/time.c | 2 | ||||
-rw-r--r-- | arch/m68knommu/platform/coldfire/pit.c | 91 |
3 files changed, 91 insertions, 13 deletions
diff --git a/arch/m68knommu/Kconfig b/arch/m68knommu/Kconfig index bfd35304d586..2e7515e8db98 100644 --- a/arch/m68knommu/Kconfig +++ b/arch/m68knommu/Kconfig | |||
@@ -66,6 +66,10 @@ config TIME_LOW_RES | |||
66 | bool | 66 | bool |
67 | default y | 67 | default y |
68 | 68 | ||
69 | config GENERIC_CLOCKEVENTS | ||
70 | bool | ||
71 | default n | ||
72 | |||
69 | config NO_IOPORT | 73 | config NO_IOPORT |
70 | def_bool y | 74 | def_bool y |
71 | 75 | ||
@@ -112,11 +116,13 @@ config M5206e | |||
112 | 116 | ||
113 | config M520x | 117 | config M520x |
114 | bool "MCF520x" | 118 | bool "MCF520x" |
119 | select GENERIC_CLOCKEVENTS | ||
115 | help | 120 | help |
116 | Freescale Coldfire 5207/5208 processor support. | 121 | Freescale Coldfire 5207/5208 processor support. |
117 | 122 | ||
118 | config M523x | 123 | config M523x |
119 | bool "MCF523x" | 124 | bool "MCF523x" |
125 | select GENERIC_CLOCKEVENTS | ||
120 | help | 126 | help |
121 | Freescale Coldfire 5230/1/2/4/5 processor support | 127 | Freescale Coldfire 5230/1/2/4/5 processor support |
122 | 128 | ||
@@ -142,6 +148,7 @@ config M5275 | |||
142 | 148 | ||
143 | config M528x | 149 | config M528x |
144 | bool "MCF528x" | 150 | bool "MCF528x" |
151 | select GENERIC_CLOCKEVENTS | ||
145 | help | 152 | help |
146 | Motorola ColdFire 5280/5282 processor support. | 153 | Motorola ColdFire 5280/5282 processor support. |
147 | 154 | ||
@@ -165,6 +172,7 @@ endchoice | |||
165 | config M527x | 172 | config M527x |
166 | bool | 173 | bool |
167 | depends on (M5271 || M5275) | 174 | depends on (M5271 || M5275) |
175 | select GENERIC_CLOCKEVENTS | ||
168 | default y | 176 | default y |
169 | 177 | ||
170 | config COLDFIRE | 178 | config COLDFIRE |
@@ -678,6 +686,9 @@ endchoice | |||
678 | if COLDFIRE | 686 | if COLDFIRE |
679 | source "kernel/Kconfig.preempt" | 687 | source "kernel/Kconfig.preempt" |
680 | endif | 688 | endif |
689 | |||
690 | source "kernel/time/Kconfig" | ||
691 | |||
681 | source "mm/Kconfig" | 692 | source "mm/Kconfig" |
682 | 693 | ||
683 | endmenu | 694 | endmenu |
diff --git a/arch/m68knommu/kernel/time.c b/arch/m68knommu/kernel/time.c index 67944aa27280..d182b2f72211 100644 --- a/arch/m68knommu/kernel/time.c +++ b/arch/m68knommu/kernel/time.c | |||
@@ -33,6 +33,7 @@ static inline int set_rtc_mmss(unsigned long nowtime) | |||
33 | return -1; | 33 | return -1; |
34 | } | 34 | } |
35 | 35 | ||
36 | #ifndef CONFIG_GENERIC_CLOCKEVENTS | ||
36 | /* | 37 | /* |
37 | * timer_interrupt() needs to keep up the real-time clock, | 38 | * timer_interrupt() needs to keep up the real-time clock, |
38 | * as well as call the "do_timer()" routine every clocktick | 39 | * as well as call the "do_timer()" routine every clocktick |
@@ -54,6 +55,7 @@ irqreturn_t arch_timer_interrupt(int irq, void *dummy) | |||
54 | #endif | 55 | #endif |
55 | return(IRQ_HANDLED); | 56 | return(IRQ_HANDLED); |
56 | } | 57 | } |
58 | #endif | ||
57 | 59 | ||
58 | static unsigned long read_rtc_mmss(void) | 60 | static unsigned long read_rtc_mmss(void) |
59 | { | 61 | { |
diff --git a/arch/m68knommu/platform/coldfire/pit.c b/arch/m68knommu/platform/coldfire/pit.c index 4290638012e0..c5b916700b22 100644 --- a/arch/m68knommu/platform/coldfire/pit.c +++ b/arch/m68knommu/platform/coldfire/pit.c | |||
@@ -18,7 +18,7 @@ | |||
18 | #include <linux/init.h> | 18 | #include <linux/init.h> |
19 | #include <linux/interrupt.h> | 19 | #include <linux/interrupt.h> |
20 | #include <linux/irq.h> | 20 | #include <linux/irq.h> |
21 | #include <linux/clocksource.h> | 21 | #include <linux/clockchips.h> |
22 | #include <asm/machdep.h> | 22 | #include <asm/machdep.h> |
23 | #include <asm/io.h> | 23 | #include <asm/io.h> |
24 | #include <asm/coldfire.h> | 24 | #include <asm/coldfire.h> |
@@ -33,22 +33,86 @@ | |||
33 | #define FREQ ((MCF_CLK / 2) / 64) | 33 | #define FREQ ((MCF_CLK / 2) / 64) |
34 | #define TA(a) (MCF_IPSBAR + MCFPIT_BASE1 + (a)) | 34 | #define TA(a) (MCF_IPSBAR + MCFPIT_BASE1 + (a)) |
35 | #define INTC0 (MCF_IPSBAR + MCFICM_INTC0) | 35 | #define INTC0 (MCF_IPSBAR + MCFICM_INTC0) |
36 | #define PIT_CYCLES_PER_JIFFY (FREQ / HZ) | ||
36 | 37 | ||
37 | static u32 pit_cycles_per_jiffy; | ||
38 | static u32 pit_cnt; | 38 | static u32 pit_cnt; |
39 | 39 | ||
40 | /* | ||
41 | * Initialize the PIT timer. | ||
42 | * | ||
43 | * This is also called after resume to bring the PIT into operation again. | ||
44 | */ | ||
45 | |||
46 | static void init_cf_pit_timer(enum clock_event_mode mode, | ||
47 | struct clock_event_device *evt) | ||
48 | { | ||
49 | switch (mode) { | ||
50 | case CLOCK_EVT_MODE_PERIODIC: | ||
51 | |||
52 | __raw_writew(MCFPIT_PCSR_DISABLE, TA(MCFPIT_PCSR)); | ||
53 | __raw_writew(PIT_CYCLES_PER_JIFFY, TA(MCFPIT_PMR)); | ||
54 | __raw_writew(MCFPIT_PCSR_EN | MCFPIT_PCSR_PIE | \ | ||
55 | MCFPIT_PCSR_OVW | MCFPIT_PCSR_RLD | \ | ||
56 | MCFPIT_PCSR_CLK64, TA(MCFPIT_PCSR)); | ||
57 | break; | ||
58 | |||
59 | case CLOCK_EVT_MODE_SHUTDOWN: | ||
60 | case CLOCK_EVT_MODE_UNUSED: | ||
61 | |||
62 | __raw_writew(MCFPIT_PCSR_DISABLE, TA(MCFPIT_PCSR)); | ||
63 | break; | ||
64 | |||
65 | case CLOCK_EVT_MODE_ONESHOT: | ||
66 | |||
67 | __raw_writew(MCFPIT_PCSR_DISABLE, TA(MCFPIT_PCSR)); | ||
68 | __raw_writew(MCFPIT_PCSR_EN | MCFPIT_PCSR_PIE | \ | ||
69 | MCFPIT_PCSR_OVW | MCFPIT_PCSR_CLK64, \ | ||
70 | TA(MCFPIT_PCSR)); | ||
71 | break; | ||
72 | |||
73 | case CLOCK_EVT_MODE_RESUME: | ||
74 | /* Nothing to do here */ | ||
75 | break; | ||
76 | } | ||
77 | } | ||
78 | |||
79 | /* | ||
80 | * Program the next event in oneshot mode | ||
81 | * | ||
82 | * Delta is given in PIT ticks | ||
83 | */ | ||
84 | static int cf_pit_next_event(unsigned long delta, | ||
85 | struct clock_event_device *evt) | ||
86 | { | ||
87 | __raw_writew(delta, TA(MCFPIT_PMR)); | ||
88 | return 0; | ||
89 | } | ||
90 | |||
91 | struct clock_event_device cf_pit_clockevent = { | ||
92 | .name = "pit", | ||
93 | .features = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT, | ||
94 | .set_mode = init_cf_pit_timer, | ||
95 | .set_next_event = cf_pit_next_event, | ||
96 | .shift = 32, | ||
97 | .irq = MCFINT_VECBASE + MCFINT_PIT1, | ||
98 | }; | ||
99 | |||
100 | |||
101 | |||
40 | /***************************************************************************/ | 102 | /***************************************************************************/ |
41 | 103 | ||
42 | static irqreturn_t pit_tick(int irq, void *dummy) | 104 | static irqreturn_t pit_tick(int irq, void *dummy) |
43 | { | 105 | { |
106 | struct clock_event_device *evt = &cf_pit_clockevent; | ||
44 | u16 pcsr; | 107 | u16 pcsr; |
45 | 108 | ||
46 | /* Reset the ColdFire timer */ | 109 | /* Reset the ColdFire timer */ |
47 | pcsr = __raw_readw(TA(MCFPIT_PCSR)); | 110 | pcsr = __raw_readw(TA(MCFPIT_PCSR)); |
48 | __raw_writew(pcsr | MCFPIT_PCSR_PIF, TA(MCFPIT_PCSR)); | 111 | __raw_writew(pcsr | MCFPIT_PCSR_PIF, TA(MCFPIT_PCSR)); |
49 | 112 | ||
50 | pit_cnt += pit_cycles_per_jiffy; | 113 | pit_cnt += PIT_CYCLES_PER_JIFFY; |
51 | return arch_timer_interrupt(irq, dummy); | 114 | evt->event_handler(evt); |
115 | return IRQ_HANDLED; | ||
52 | } | 116 | } |
53 | 117 | ||
54 | /***************************************************************************/ | 118 | /***************************************************************************/ |
@@ -72,14 +136,14 @@ static cycle_t pit_read_clk(void) | |||
72 | cycles = pit_cnt; | 136 | cycles = pit_cnt; |
73 | local_irq_restore(flags); | 137 | local_irq_restore(flags); |
74 | 138 | ||
75 | return cycles + pit_cycles_per_jiffy - pcntr; | 139 | return cycles + PIT_CYCLES_PER_JIFFY - pcntr; |
76 | } | 140 | } |
77 | 141 | ||
78 | /***************************************************************************/ | 142 | /***************************************************************************/ |
79 | 143 | ||
80 | static struct clocksource pit_clk = { | 144 | static struct clocksource pit_clk = { |
81 | .name = "pit", | 145 | .name = "pit", |
82 | .rating = 250, | 146 | .rating = 100, |
83 | .read = pit_read_clk, | 147 | .read = pit_read_clk, |
84 | .shift = 20, | 148 | .shift = 20, |
85 | .mask = CLOCKSOURCE_MASK(32), | 149 | .mask = CLOCKSOURCE_MASK(32), |
@@ -92,6 +156,14 @@ void hw_timer_init(void) | |||
92 | { | 156 | { |
93 | u32 imr; | 157 | u32 imr; |
94 | 158 | ||
159 | cf_pit_clockevent.cpumask = cpumask_of_cpu(smp_processor_id()); | ||
160 | cf_pit_clockevent.mult = div_sc(FREQ, NSEC_PER_SEC, 32); | ||
161 | cf_pit_clockevent.max_delta_ns = | ||
162 | clockevent_delta2ns(0xFFFF, &cf_pit_clockevent); | ||
163 | cf_pit_clockevent.min_delta_ns = | ||
164 | clockevent_delta2ns(0x3f, &cf_pit_clockevent); | ||
165 | clockevents_register_device(&cf_pit_clockevent); | ||
166 | |||
95 | setup_irq(MCFINT_VECBASE + MCFINT_PIT1, &pit_irq); | 167 | setup_irq(MCFINT_VECBASE + MCFINT_PIT1, &pit_irq); |
96 | 168 | ||
97 | __raw_writeb(ICR_INTRCONF, INTC0 + MCFINTC_ICR0 + MCFINT_PIT1); | 169 | __raw_writeb(ICR_INTRCONF, INTC0 + MCFINTC_ICR0 + MCFINT_PIT1); |
@@ -99,13 +171,6 @@ void hw_timer_init(void) | |||
99 | imr &= ~MCFPIT_IMR_IBIT; | 171 | imr &= ~MCFPIT_IMR_IBIT; |
100 | __raw_writel(imr, INTC0 + MCFPIT_IMR); | 172 | __raw_writel(imr, INTC0 + MCFPIT_IMR); |
101 | 173 | ||
102 | /* Set up PIT timer 1 as poll clock */ | ||
103 | pit_cycles_per_jiffy = FREQ / HZ; | ||
104 | __raw_writew(MCFPIT_PCSR_DISABLE, TA(MCFPIT_PCSR)); | ||
105 | __raw_writew(pit_cycles_per_jiffy, TA(MCFPIT_PMR)); | ||
106 | __raw_writew(MCFPIT_PCSR_EN | MCFPIT_PCSR_PIE | MCFPIT_PCSR_OVW | | ||
107 | MCFPIT_PCSR_RLD | MCFPIT_PCSR_CLK64, TA(MCFPIT_PCSR)); | ||
108 | |||
109 | pit_clk.mult = clocksource_hz2mult(FREQ, pit_clk.shift); | 174 | pit_clk.mult = clocksource_hz2mult(FREQ, pit_clk.shift); |
110 | clocksource_register(&pit_clk); | 175 | clocksource_register(&pit_clk); |
111 | } | 176 | } |