cpu_64.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. /* cpu.c: Dinky routines to look for the kind of Sparc cpu
  2. * we are on.
  3. *
  4. * Copyright (C) 1996, 2007, 2008 David S. Miller (davem@davemloft.net)
  5. */
  6. #include <linux/kernel.h>
  7. #include <linux/init.h>
  8. #include <linux/sched.h>
  9. #include <linux/smp.h>
  10. #include <asm/asi.h>
  11. #include <asm/system.h>
  12. #include <asm/fpumacro.h>
  13. #include <asm/cpudata.h>
  14. #include <asm/spitfire.h>
  15. #include <asm/oplib.h>
  16. #include "entry.h"
  17. #include "kernel.h"
  18. DEFINE_PER_CPU(cpuinfo_sparc, __cpu_data) = { 0 };
  19. struct cpu_chip_info {
  20. unsigned short manuf;
  21. unsigned short impl;
  22. const char *cpu_name;
  23. const char *fp_name;
  24. };
  25. static const struct cpu_chip_info cpu_chips[] = {
  26. {
  27. .manuf = 0x17,
  28. .impl = 0x10,
  29. .cpu_name = "TI UltraSparc I (SpitFire)",
  30. .fp_name = "UltraSparc I integrated FPU",
  31. },
  32. {
  33. .manuf = 0x22,
  34. .impl = 0x10,
  35. .cpu_name = "TI UltraSparc I (SpitFire)",
  36. .fp_name = "UltraSparc I integrated FPU",
  37. },
  38. {
  39. .manuf = 0x17,
  40. .impl = 0x11,
  41. .cpu_name = "TI UltraSparc II (BlackBird)",
  42. .fp_name = "UltraSparc II integrated FPU",
  43. },
  44. {
  45. .manuf = 0x17,
  46. .impl = 0x12,
  47. .cpu_name = "TI UltraSparc IIi (Sabre)",
  48. .fp_name = "UltraSparc IIi integrated FPU",
  49. },
  50. {
  51. .manuf = 0x17,
  52. .impl = 0x13,
  53. .cpu_name = "TI UltraSparc IIe (Hummingbird)",
  54. .fp_name = "UltraSparc IIe integrated FPU",
  55. },
  56. {
  57. .manuf = 0x3e,
  58. .impl = 0x14,
  59. .cpu_name = "TI UltraSparc III (Cheetah)",
  60. .fp_name = "UltraSparc III integrated FPU",
  61. },
  62. {
  63. .manuf = 0x3e,
  64. .impl = 0x15,
  65. .cpu_name = "TI UltraSparc III+ (Cheetah+)",
  66. .fp_name = "UltraSparc III+ integrated FPU",
  67. },
  68. {
  69. .manuf = 0x3e,
  70. .impl = 0x16,
  71. .cpu_name = "TI UltraSparc IIIi (Jalapeno)",
  72. .fp_name = "UltraSparc IIIi integrated FPU",
  73. },
  74. {
  75. .manuf = 0x3e,
  76. .impl = 0x18,
  77. .cpu_name = "TI UltraSparc IV (Jaguar)",
  78. .fp_name = "UltraSparc IV integrated FPU",
  79. },
  80. {
  81. .manuf = 0x3e,
  82. .impl = 0x19,
  83. .cpu_name = "TI UltraSparc IV+ (Panther)",
  84. .fp_name = "UltraSparc IV+ integrated FPU",
  85. },
  86. {
  87. .manuf = 0x3e,
  88. .impl = 0x22,
  89. .cpu_name = "TI UltraSparc IIIi+ (Serrano)",
  90. .fp_name = "UltraSparc IIIi+ integrated FPU",
  91. },
  92. };
  93. #define NSPARCCHIPS ARRAY_SIZE(linux_sparc_chips)
  94. const char *sparc_cpu_type;
  95. const char *sparc_fpu_type;
  96. static void __init sun4v_cpu_probe(void)
  97. {
  98. switch (sun4v_chip_type) {
  99. case SUN4V_CHIP_NIAGARA1:
  100. sparc_cpu_type = "UltraSparc T1 (Niagara)";
  101. sparc_fpu_type = "UltraSparc T1 integrated FPU";
  102. break;
  103. case SUN4V_CHIP_NIAGARA2:
  104. sparc_cpu_type = "UltraSparc T2 (Niagara2)";
  105. sparc_fpu_type = "UltraSparc T2 integrated FPU";
  106. break;
  107. default:
  108. printk(KERN_WARNING "CPU: Unknown sun4v cpu type [%s]\n",
  109. prom_cpu_compatible);
  110. sparc_cpu_type = "Unknown SUN4V CPU";
  111. sparc_fpu_type = "Unknown SUN4V FPU";
  112. break;
  113. }
  114. }
  115. static const struct cpu_chip_info * __init find_cpu_chip(unsigned short manuf,
  116. unsigned short impl)
  117. {
  118. int i;
  119. for (i = 0; i < ARRAY_SIZE(cpu_chips); i++) {
  120. const struct cpu_chip_info *p = &cpu_chips[i];
  121. if (p->manuf == manuf && p->impl == impl)
  122. return p;
  123. }
  124. return NULL;
  125. }
  126. static int __init cpu_type_probe(void)
  127. {
  128. if (tlb_type == hypervisor) {
  129. sun4v_cpu_probe();
  130. } else {
  131. unsigned long ver, manuf, impl;
  132. const struct cpu_chip_info *p;
  133. __asm__ __volatile__("rdpr %%ver, %0" : "=r" (ver));
  134. manuf = ((ver >> 48) & 0xffff);
  135. impl = ((ver >> 32) & 0xffff);
  136. p = find_cpu_chip(manuf, impl);
  137. if (p) {
  138. sparc_cpu_type = p->cpu_name;
  139. sparc_fpu_type = p->fp_name;
  140. } else {
  141. printk(KERN_ERR "CPU: Unknown chip, manuf[%lx] impl[%lx]\n",
  142. manuf, impl);
  143. sparc_cpu_type = "Unknown CPU";
  144. sparc_fpu_type = "Unknown FPU";
  145. }
  146. }
  147. return 0;
  148. }
  149. arch_initcall(cpu_type_probe);