You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

cpuid_power.c 7.9 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283
  1. /*********************************************************************/
  2. /* Copyright 2009, 2010 The University of Texas at Austin. */
  3. /* All rights reserved. */
  4. /* */
  5. /* Redistribution and use in source and binary forms, with or */
  6. /* without modification, are permitted provided that the following */
  7. /* conditions are met: */
  8. /* */
  9. /* 1. Redistributions of source code must retain the above */
  10. /* copyright notice, this list of conditions and the following */
  11. /* disclaimer. */
  12. /* */
  13. /* 2. Redistributions in binary form must reproduce the above */
  14. /* copyright notice, this list of conditions and the following */
  15. /* disclaimer in the documentation and/or other materials */
  16. /* provided with the distribution. */
  17. /* */
  18. /* THIS SOFTWARE IS PROVIDED BY THE UNIVERSITY OF TEXAS AT */
  19. /* AUSTIN ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, */
  20. /* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF */
  21. /* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE */
  22. /* DISCLAIMED. IN NO EVENT SHALL THE UNIVERSITY OF TEXAS AT */
  23. /* AUSTIN OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, */
  24. /* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES */
  25. /* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE */
  26. /* GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR */
  27. /* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF */
  28. /* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT */
  29. /* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT */
  30. /* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE */
  31. /* POSSIBILITY OF SUCH DAMAGE. */
  32. /* */
  33. /* The views and conclusions contained in the software and */
  34. /* documentation are those of the authors and should not be */
  35. /* interpreted as representing official policies, either expressed */
  36. /* or implied, of The University of Texas at Austin. */
  37. /*********************************************************************/
  38. #include <sys/utsname.h>
  39. #ifdef _AIX
  40. #include <sys/vminfo.h>
  41. #endif
  42. #ifdef __APPLE__
  43. #include <mach/mach.h>
  44. #include <mach/mach_host.h>
  45. #include <mach/host_info.h>
  46. #include <mach/machine.h>
  47. #endif
  48. #define CPUTYPE_UNKNOWN 0
  49. #define CPUTYPE_POWER3 1
  50. #define CPUTYPE_POWER4 2
  51. #define CPUTYPE_PPC970 3
  52. #define CPUTYPE_POWER5 4
  53. #define CPUTYPE_POWER6 5
  54. #define CPUTYPE_CELL 6
  55. #define CPUTYPE_PPCG4 7
  56. #define CPUTYPE_POWER8 8
  57. #define CPUTYPE_POWER9 9
  58. #define CPUTYPE_POWER10 10
  59. char *cpuname[] = {
  60. "UNKNOWN",
  61. "POWER3",
  62. "POWER4",
  63. "PPC970",
  64. "POWER5",
  65. "POWER6",
  66. "CELL",
  67. "PPCG4",
  68. "POWER8",
  69. "POWER9",
  70. "POWER10"
  71. };
  72. char *lowercpuname[] = {
  73. "unknown",
  74. "power3",
  75. "power4",
  76. "ppc970",
  77. "power5",
  78. "power6",
  79. "cell",
  80. "ppcg4",
  81. "power8",
  82. "power9",
  83. "power10"
  84. };
  85. char *corename[] = {
  86. "UNKNOWN",
  87. "POWER3",
  88. "POWER4",
  89. "POWER4",
  90. "POWER4",
  91. "POWER6",
  92. "CELL",
  93. "PPCG4",
  94. "POWER8",
  95. "POWER9",
  96. "POWER10"
  97. };
  98. int detect(void){
  99. #ifdef linux
  100. FILE *infile;
  101. char buffer[512], *p;
  102. p = (char *)NULL;
  103. infile = fopen("/proc/cpuinfo", "r");
  104. while (fgets(buffer, sizeof(buffer), infile)){
  105. if (!strncmp("cpu", buffer, 3)){
  106. p = strchr(buffer, ':') + 2;
  107. #if 0
  108. fprintf(stderr, "%s\n", p);
  109. #endif
  110. break;
  111. }
  112. }
  113. fclose(infile);
  114. if (!strncasecmp(p, "POWER3", 6)) return CPUTYPE_POWER3;
  115. if (!strncasecmp(p, "POWER4", 6)) return CPUTYPE_POWER4;
  116. if (!strncasecmp(p, "PPC970", 6)) return CPUTYPE_PPC970;
  117. if (!strncasecmp(p, "POWER5", 6)) return CPUTYPE_POWER5;
  118. if (!strncasecmp(p, "POWER6", 6)) return CPUTYPE_POWER6;
  119. if (!strncasecmp(p, "POWER7", 6)) return CPUTYPE_POWER6;
  120. if (!strncasecmp(p, "POWER8", 6)) return CPUTYPE_POWER8;
  121. if (!strncasecmp(p, "POWER9", 6)) return CPUTYPE_POWER9;
  122. if (!strncasecmp(p, "POWER10", 7)) return CPUTYPE_POWER10;
  123. if (!strncasecmp(p, "Cell", 4)) return CPUTYPE_CELL;
  124. if (!strncasecmp(p, "7447", 4)) return CPUTYPE_PPCG4;
  125. return CPUTYPE_UNKNOWN;
  126. #endif
  127. #ifdef _AIX
  128. FILE *infile;
  129. char buffer[512], *p;
  130. p = (char *)NULL;
  131. infile = popen("prtconf|grep 'Processor Type'", "r");
  132. while (fgets(buffer, sizeof(buffer), infile)){
  133. if (!strncmp("Pro", buffer, 3)){
  134. p = strchr(buffer, ':') + 2;
  135. #if 0
  136. fprintf(stderr, "%s\n", p);
  137. #endif
  138. break;
  139. }
  140. }
  141. pclose(infile);
  142. if (!strncasecmp(p, "POWER3", 6)) return CPUTYPE_POWER3;
  143. if (!strncasecmp(p, "POWER4", 6)) return CPUTYPE_POWER4;
  144. if (!strncasecmp(p, "PPC970", 6)) return CPUTYPE_PPC970;
  145. if (!strncasecmp(p, "POWER5", 6)) return CPUTYPE_POWER5;
  146. if (!strncasecmp(p, "POWER6", 6)) return CPUTYPE_POWER6;
  147. if (!strncasecmp(p, "POWER7", 6)) return CPUTYPE_POWER6;
  148. if (!strncasecmp(p, "POWER8", 6)) return CPUTYPE_POWER8;
  149. if (!strncasecmp(p, "POWER9", 6)) return CPUTYPE_POWER9;
  150. if (!strncasecmp(p, "POWER10", 7)) return CPUTYPE_POWER10;
  151. if (!strncasecmp(p, "Cell", 4)) return CPUTYPE_CELL;
  152. if (!strncasecmp(p, "7447", 4)) return CPUTYPE_PPCG4;
  153. return CPUTYPE_POWER5;
  154. #endif
  155. #ifdef __APPLE__
  156. host_basic_info_data_t hostInfo;
  157. mach_msg_type_number_t infoCount;
  158. infoCount = HOST_BASIC_INFO_COUNT;
  159. host_info(mach_host_self(), HOST_BASIC_INFO, (host_info_t)&hostInfo, &infoCount);
  160. if (hostInfo.cpu_subtype == CPU_SUBTYPE_POWERPC_7450) return CPUTYPE_PPCG4;
  161. if (hostInfo.cpu_subtype == CPU_SUBTYPE_POWERPC_970) return CPUTYPE_PPC970;
  162. return CPUTYPE_PPC970;
  163. #endif
  164. #if defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__NetBSD__)
  165. int id;
  166. __asm __volatile("mfpvr %0" : "=r"(id));
  167. switch ( id >> 16 ) {
  168. case 0x80: // POWER10
  169. return CPUTYPE_POWER10;
  170. break;
  171. case 0x4e: // POWER9
  172. return CPUTYPE_POWER9;
  173. break;
  174. case 0x4d:
  175. case 0x4b: // POWER8/8E
  176. return CPUTYPE_POWER8;
  177. break;
  178. case 0x4a:
  179. case 0x3f: // POWER7/7E
  180. return CPUTYPE_POWER6;
  181. break;
  182. case 0x3e:
  183. return CPUTYPE_POWER6;
  184. break;
  185. case 0x3a:
  186. return CPUTYPE_POWER5;
  187. break;
  188. case 0x35:
  189. case 0x38: // POWER4 /4+
  190. return CPUTYPE_POWER4;
  191. break;
  192. case 0x40:
  193. case 0x41: // POWER3 /3+
  194. return CPUTYPE_POWER3;
  195. break;
  196. case 0x39:
  197. case 0x3c:
  198. case 0x44:
  199. case 0x45:
  200. return CPUTYPE_PPC970;
  201. break;
  202. case 0x70:
  203. return CPUTYPE_CELL;
  204. break;
  205. case 0x8003:
  206. return CPUTYPE_PPCG4;
  207. break;
  208. default:
  209. return CPUTYPE_UNKNOWN;
  210. }
  211. #endif
  212. }
  213. void get_architecture(void){
  214. printf("POWER");
  215. }
  216. void get_subdirname(void){
  217. printf("power");
  218. }
  219. void get_subarchitecture(void){
  220. printf("%s", cpuname[detect()]);
  221. }
  222. void get_cpuconfig(void){
  223. #if 0
  224. #ifdef _AIX
  225. struct vminfo info;
  226. #endif
  227. #endif
  228. printf("#define %s\n", cpuname[detect()]);
  229. printf("#define CORE_%s\n", corename[detect()]);
  230. printf("#define L1_DATA_SIZE 32768\n");
  231. printf("#define L1_DATA_LINESIZE 128\n");
  232. printf("#define L2_SIZE 524288\n");
  233. printf("#define L2_LINESIZE 128 \n");
  234. printf("#define DTB_DEFAULT_ENTRIES 128\n");
  235. printf("#define DTB_SIZE 4096\n");
  236. printf("#define L2_ASSOCIATIVE 8\n");
  237. #if 0
  238. #ifdef _AIX
  239. if (vmgetinfo(&info, VMINFO, 0) == 0) {
  240. if ((info.lgpg_size >> 20) >= 1024) {
  241. printf("#define ALLOC_HUGETLB\n");
  242. }
  243. }
  244. #endif
  245. #endif
  246. }
  247. void get_libname(void){
  248. printf("%s", lowercpuname[detect()]);
  249. }
  250. char *get_corename(void){
  251. return cpuname[detect()];
  252. }