sim_time.c 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. #include <linux/types.h>
  2. #include <linux/init.h>
  3. #include <linux/kernel_stat.h>
  4. #include <linux/sched.h>
  5. #include <linux/spinlock.h>
  6. #include <linux/interrupt.h>
  7. #include <linux/mc146818rtc.h>
  8. #include <linux/smp.h>
  9. #include <linux/timex.h>
  10. #include <asm/hardirq.h>
  11. #include <asm/div64.h>
  12. #include <asm/cpu.h>
  13. #include <asm/time.h>
  14. #include <asm/irq.h>
  15. #include <asm/mc146818-time.h>
  16. #include <asm/msc01_ic.h>
  17. #include <asm/mips-boards/generic.h>
  18. #include <asm/mips-boards/prom.h>
  19. #include <asm/mips-boards/simint.h>
  20. unsigned long cpu_khz;
  21. /*
  22. * Estimate CPU frequency. Sets mips_hpt_frequency as a side-effect
  23. */
  24. static unsigned int __init estimate_cpu_frequency(void)
  25. {
  26. unsigned int prid = read_c0_prid() & 0xffff00;
  27. unsigned int count;
  28. #if 1
  29. /*
  30. * hardwire the board frequency to 12MHz.
  31. */
  32. if ((prid == (PRID_COMP_MIPS | PRID_IMP_20KC)) ||
  33. (prid == (PRID_COMP_MIPS | PRID_IMP_25KF)))
  34. count = 12000000;
  35. else
  36. count = 6000000;
  37. #else
  38. unsigned int flags;
  39. local_irq_save(flags);
  40. /* Start counter exactly on falling edge of update flag */
  41. while (CMOS_READ(RTC_REG_A) & RTC_UIP);
  42. while (!(CMOS_READ(RTC_REG_A) & RTC_UIP));
  43. /* Start r4k counter. */
  44. write_c0_count(0);
  45. /* Read counter exactly on falling edge of update flag */
  46. while (CMOS_READ(RTC_REG_A) & RTC_UIP);
  47. while (!(CMOS_READ(RTC_REG_A) & RTC_UIP));
  48. count = read_c0_count();
  49. /* restore interrupts */
  50. local_irq_restore(flags);
  51. #endif
  52. mips_hpt_frequency = count;
  53. if ((prid != (PRID_COMP_MIPS | PRID_IMP_20KC)) &&
  54. (prid != (PRID_COMP_MIPS | PRID_IMP_25KF)))
  55. count *= 2;
  56. count += 5000; /* round */
  57. count -= count%10000;
  58. return count;
  59. }
  60. void __init plat_time_init(void)
  61. {
  62. unsigned int est_freq, flags;
  63. local_irq_save(flags);
  64. /* Set Data mode - binary. */
  65. CMOS_WRITE(CMOS_READ(RTC_CONTROL) | RTC_DM_BINARY, RTC_CONTROL);
  66. est_freq = estimate_cpu_frequency();
  67. printk(KERN_INFO "CPU frequency %d.%02d MHz\n", est_freq / 1000000,
  68. (est_freq % 1000000) * 100 / 1000000);
  69. cpu_khz = est_freq / 1000;
  70. local_irq_restore(flags);
  71. }
  72. static int mips_cpu_timer_irq;
  73. static void mips_timer_dispatch(void)
  74. {
  75. do_IRQ(mips_cpu_timer_irq);
  76. }
  77. void __init plat_timer_setup(struct irqaction *irq)
  78. {
  79. if (cpu_has_veic) {
  80. set_vi_handler(MSC01E_INT_CPUCTR, mips_timer_dispatch);
  81. mips_cpu_timer_irq = MSC01E_INT_BASE + MSC01E_INT_CPUCTR;
  82. } else {
  83. if (cpu_has_vint)
  84. set_vi_handler(cp0_compare_irq, mips_timer_dispatch);
  85. mips_cpu_timer_irq = MIPS_CPU_IRQ_BASE + cp0_compare_irq;
  86. }
  87. /* we are using the cpu counter for timer interrupts */
  88. setup_irq(mips_cpu_timer_irq, irq);
  89. #ifdef CONFIG_SMP
  90. /* irq_desc(riptor) is a global resource, when the interrupt overlaps
  91. on seperate cpu's the first one tries to handle the second interrupt.
  92. The effect is that the int remains disabled on the second cpu.
  93. Mark the interrupt with IRQ_PER_CPU to avoid any confusion */
  94. irq_desc[mips_cpu_timer_irq].flags |= IRQ_PER_CPU;
  95. set_irq_handler(mips_cpu_timer_irq, handle_percpu_irq);
  96. #endif
  97. }