blob: 874a18e8ac24b4ba2867457c80c2ddf88d3dc029 [file] [log] [blame]
Ralf Baechlec78cbf42005-09-30 13:59:37 +01001#include <linux/types.h>
Ralf Baechlec78cbf42005-09-30 13:59:37 +01002#include <linux/init.h>
3#include <linux/kernel_stat.h>
4#include <linux/sched.h>
5#include <linux/spinlock.h>
Nicolas Kaiserdd6bfd62006-11-07 09:56:24 +01006#include <linux/interrupt.h>
7#include <linux/mc146818rtc.h>
Ralf Baechlef6e23732007-07-10 17:32:56 +01008#include <linux/mipsregs.h>
9#include <linux/smp.h>
Nicolas Kaiserdd6bfd62006-11-07 09:56:24 +010010#include <linux/timex.h>
Ralf Baechlec78cbf42005-09-30 13:59:37 +010011
Ralf Baechlec78cbf42005-09-30 13:59:37 +010012#include <asm/hardirq.h>
13#include <asm/div64.h>
14#include <asm/cpu.h>
15#include <asm/time.h>
Ralf Baechlec78cbf42005-09-30 13:59:37 +010016#include <asm/irq.h>
Ralf Baechlec78cbf42005-09-30 13:59:37 +010017#include <asm/mc146818-time.h>
18#include <asm/msc01_ic.h>
19
20#include <asm/mips-boards/generic.h>
21#include <asm/mips-boards/prom.h>
22#include <asm/mips-boards/simint.h>
Ralf Baechlec78cbf42005-09-30 13:59:37 +010023
24
25unsigned long cpu_khz;
26
Ralf Baechle937a8012006-10-07 19:44:33 +010027irqreturn_t sim_timer_interrupt(int irq, void *dev_id)
Ralf Baechlec78cbf42005-09-30 13:59:37 +010028{
29#ifdef CONFIG_SMP
30 int cpu = smp_processor_id();
31
32 /*
33 * CPU 0 handles the global timer interrupt job
34 * resets count/compare registers to trigger next timer int.
35 */
36#ifndef CONFIG_MIPS_MT_SMTC
37 if (cpu == 0) {
Ralf Baechle937a8012006-10-07 19:44:33 +010038 timer_interrupt(irq, dev_id);
Ralf Baechlef6e23732007-07-10 17:32:56 +010039 } else {
Ralf Baechlec78cbf42005-09-30 13:59:37 +010040 /* Everyone else needs to reset the timer int here as
41 ll_local_timer_interrupt doesn't */
42 /*
43 * FIXME: need to cope with counter underflow.
44 * More support needs to be added to kernel/time for
45 * counter/timer interrupts on multiple CPU's
46 */
47 write_c0_compare (read_c0_count() + ( mips_hpt_frequency/HZ));
48 }
49#else /* SMTC */
50 /*
51 * In SMTC system, one Count/Compare set exists per VPE.
52 * Which TC within a VPE gets the interrupt is essentially
53 * random - we only know that it shouldn't be one with
54 * IXMT set. Whichever TC gets the interrupt needs to
55 * send special interprocessor interrupts to the other
56 * TCs to make sure that they schedule, etc.
57 *
58 * That code is specific to the SMTC kernel, not to
59 * the simulation platform, so it's invoked from
60 * the general MIPS timer_interrupt routine.
61 *
62 * We have a problem in that the interrupt vector code
63 * had to turn off the timer IM bit to avoid redundant
64 * entries, but we may never get to mips_cpu_irq_end
65 * to turn it back on again if the scheduler gets
66 * involved. So we clear the pending timer here,
67 * and re-enable the mask...
68 */
69
70 int vpflags = dvpe();
71 write_c0_compare (read_c0_count() - 1);
Ralf Baechle3b1d4ed2007-06-20 22:27:10 +010072 clear_c0_cause(0x100 << cp0_compare_irq);
73 set_c0_status(0x100 << cp0_compare_irq);
Ralf Baechlec78cbf42005-09-30 13:59:37 +010074 irq_enable_hazard();
75 evpe(vpflags);
76
Ralf Baechlef6e23732007-07-10 17:32:56 +010077 if (cpu_data[cpu].vpe_id == 0)
78 timer_interrupt(irq, dev_id);
79 else
80 write_c0_compare (read_c0_count() + ( mips_hpt_frequency/HZ));
Ralf Baechlec78cbf42005-09-30 13:59:37 +010081 smtc_timer_broadcast(cpu_data[cpu].vpe_id);
82
83#endif /* CONFIG_MIPS_MT_SMTC */
84
85 /*
86 * every CPU should do profiling and process accounting
87 */
Ralf Baechlef6e23732007-07-10 17:32:56 +010088 local_timer_interrupt (irq, dev_id);
89
Ralf Baechlec78cbf42005-09-30 13:59:37 +010090 return IRQ_HANDLED;
91#else
Ralf Baechle937a8012006-10-07 19:44:33 +010092 return timer_interrupt (irq, dev_id);
Ralf Baechlec78cbf42005-09-30 13:59:37 +010093#endif
94}
95
96
97
98/*
Ralf Baechle224dc502006-10-21 02:05:20 +010099 * Estimate CPU frequency. Sets mips_hpt_frequency as a side-effect
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100100 */
101static unsigned int __init estimate_cpu_frequency(void)
102{
103 unsigned int prid = read_c0_prid() & 0xffff00;
104 unsigned int count;
105
106#if 1
107 /*
108 * hardwire the board frequency to 12MHz.
109 */
110
111 if ((prid == (PRID_COMP_MIPS | PRID_IMP_20KC)) ||
112 (prid == (PRID_COMP_MIPS | PRID_IMP_25KF)))
113 count = 12000000;
114 else
115 count = 6000000;
116#else
117 unsigned int flags;
118
119 local_irq_save(flags);
120
121 /* Start counter exactly on falling edge of update flag */
122 while (CMOS_READ(RTC_REG_A) & RTC_UIP);
123 while (!(CMOS_READ(RTC_REG_A) & RTC_UIP));
124
125 /* Start r4k counter. */
126 write_c0_count(0);
127
128 /* Read counter exactly on falling edge of update flag */
129 while (CMOS_READ(RTC_REG_A) & RTC_UIP);
130 while (!(CMOS_READ(RTC_REG_A) & RTC_UIP));
131
132 count = read_c0_count();
133
134 /* restore interrupts */
135 local_irq_restore(flags);
136#endif
137
138 mips_hpt_frequency = count;
139
140 if ((prid != (PRID_COMP_MIPS | PRID_IMP_20KC)) &&
141 (prid != (PRID_COMP_MIPS | PRID_IMP_25KF)))
142 count *= 2;
143
144 count += 5000; /* round */
145 count -= count%10000;
146
147 return count;
148}
149
150void __init sim_time_init(void)
151{
152 unsigned int est_freq, flags;
153
154 local_irq_save(flags);
155
Ralf Baechlef6e23732007-07-10 17:32:56 +0100156 /* Set Data mode - binary. */
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100157 CMOS_WRITE(CMOS_READ(RTC_CONTROL) | RTC_DM_BINARY, RTC_CONTROL);
158
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100159 est_freq = estimate_cpu_frequency ();
160
Ralf Baechlef6e23732007-07-10 17:32:56 +0100161 printk(KERN_INFO "CPU frequency %d.%02d MHz\n", est_freq / 1000000,
162 (est_freq % 1000000) * 100 / 1000000);
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100163
Ralf Baechlef6e23732007-07-10 17:32:56 +0100164 cpu_khz = est_freq / 1000;
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100165
166 local_irq_restore(flags);
167}
168
169static int mips_cpu_timer_irq;
170
Ralf Baechle937a8012006-10-07 19:44:33 +0100171static void mips_timer_dispatch(void)
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100172{
Ralf Baechle937a8012006-10-07 19:44:33 +0100173 do_IRQ(mips_cpu_timer_irq);
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100174}
175
176
Ralf Baechle5fd32652006-07-10 02:37:21 +0100177void __init plat_timer_setup(struct irqaction *irq)
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100178{
179 if (cpu_has_veic) {
180 set_vi_handler(MSC01E_INT_CPUCTR, mips_timer_dispatch);
181 mips_cpu_timer_irq = MSC01E_INT_BASE + MSC01E_INT_CPUCTR;
Ralf Baechlef6e23732007-07-10 17:32:56 +0100182 } else {
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100183 if (cpu_has_vint)
Ralf Baechle3b1d4ed2007-06-20 22:27:10 +0100184 set_vi_handler(cp0_compare_irq, mips_timer_dispatch);
185 mips_cpu_timer_irq = MIPS_CPU_IRQ_BASE + cp0_compare_irq;
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100186 }
187
188 /* we are using the cpu counter for timer interrupts */
189 irq->handler = sim_timer_interrupt;
190 setup_irq(mips_cpu_timer_irq, irq);
191
192#ifdef CONFIG_SMP
193 /* irq_desc(riptor) is a global resource, when the interrupt overlaps
194 on seperate cpu's the first one tries to handle the second interrupt.
195 The effect is that the int remains disabled on the second cpu.
196 Mark the interrupt with IRQ_PER_CPU to avoid any confusion */
Atsushi Nemoto14178362006-11-14 01:13:18 +0900197 irq_desc[mips_cpu_timer_irq].flags |= IRQ_PER_CPU;
198 set_irq_handler(mips_cpu_timer_irq, handle_percpu_irq);
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100199#endif
Ralf Baechlec78cbf42005-09-30 13:59:37 +0100200}