cpu.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. #include <common/cpu.h>
  2. #include <common/kprint.h>
  3. #include <common/printk.h>
  4. // #pragma GCC optimize("O0")
  5. // cpu支持的最大cpuid指令的基础主功能号
  6. uint Cpu_cpuid_max_Basic_mop;
  7. // cpu支持的最大cpuid指令的扩展主功能号
  8. uint Cpu_cpuid_max_Extended_mop;
  9. // cpu制造商信息
  10. char Cpu_Manufacturer_Name[17] = {0};
  11. // 处理器名称信息
  12. char Cpu_BrandName[49] = {0};
  13. // 处理器家族ID
  14. uint Cpu_Family_ID;
  15. // 处理器扩展家族ID
  16. uint Cpu_Extended_Family_ID;
  17. // 处理器模式ID
  18. uint Cpu_Model_ID;
  19. // 处理器扩展模式ID
  20. uint Cpu_Extended_Model_ID;
  21. // 处理器步进ID
  22. uint Cpu_Stepping_ID;
  23. // 处理器类型
  24. uint Cpu_Processor_Type;
  25. // 处理器支持的最大物理地址可寻址地址线宽度
  26. uint Cpu_max_phys_addrline_size;
  27. // 处理器支持的最大线性地址可寻址地址线宽度
  28. uint Cpu_max_linear_addrline_size;
  29. // 处理器的tsc频率(单位:hz)(HPET定时器在测定apic频率时,顺便测定了这个值)
  30. uint64_t Cpu_tsc_freq = 0;
  31. struct cpu_core_info_t cpu_core_info[MAX_CPU_NUM];
  32. void cpu_init(void)
  33. {
  34. // 获取处理器制造商信息
  35. uint tmp_info[4] = {0};
  36. cpu_cpuid(0, 0, &tmp_info[0], &tmp_info[1], &tmp_info[2], &tmp_info[3]);
  37. // 保存CPU支持的最大cpuid指令主功能号
  38. Cpu_cpuid_max_Basic_mop = tmp_info[0];
  39. // 保存制造商名称
  40. *(uint *)&Cpu_Manufacturer_Name[0] = tmp_info[1];
  41. *(uint *)&Cpu_Manufacturer_Name[4] = tmp_info[3];
  42. *(uint *)&Cpu_Manufacturer_Name[8] = tmp_info[2];
  43. Cpu_Manufacturer_Name[12] = '\0';
  44. kinfo("CPU manufacturer: %s", Cpu_Manufacturer_Name);
  45. // 获取处理器型号信息
  46. int count = 0;
  47. for (uint i = 0x80000002; i < 0x80000005; ++i)
  48. {
  49. cpu_cpuid(i, 0, &tmp_info[0], &tmp_info[1], &tmp_info[2], &tmp_info[3]);
  50. for (int j = 0; j <= 3; ++j)
  51. {
  52. *(uint *)&Cpu_BrandName[4 * count] = tmp_info[j];
  53. ++count;
  54. }
  55. }
  56. Cpu_BrandName[48] = '\0';
  57. kinfo("CPU Brand Name: %s", Cpu_BrandName);
  58. // 使用cpuid主功能号0x01进行查询(未保存ebx ecx edx的信息,具体参见白皮书)
  59. cpu_cpuid(1, 0, &tmp_info[0], &tmp_info[1], &tmp_info[2], &tmp_info[3]);
  60. // EAX中包含 Version Informatin Type,Family,Model,and Stepping ID
  61. Cpu_Stepping_ID = tmp_info[0] & 0xf;
  62. Cpu_Model_ID = (tmp_info[0] >> 4) & 0xf;
  63. Cpu_Family_ID = (tmp_info[0] >> 8) & 0xf;
  64. Cpu_Processor_Type = (tmp_info[0] >> 12) & 0x3;
  65. // 14-15位保留
  66. Cpu_Extended_Model_ID = (tmp_info[0] >> 16) & 0xf;
  67. Cpu_Extended_Family_ID = (tmp_info[0] >> 20) & 0xff;
  68. // 31-25位保留
  69. kinfo("Family ID=%#03lx\t Extended Family ID=%#03lx\t Processor Type=%#03lx\t", Cpu_Family_ID, Cpu_Extended_Family_ID, Cpu_Processor_Type);
  70. kinfo("Model ID=%#03lx\t Extended Model ID=%#03lx\tStepping ID=%#03lx\t", Cpu_Model_ID, Cpu_Extended_Model_ID, Cpu_Stepping_ID);
  71. // 使用0x80000008主功能号,查询处理器支持的最大可寻址地址线宽度
  72. cpu_cpuid(0x80000008, 0, &tmp_info[0], &tmp_info[1], &tmp_info[2], &tmp_info[3]);
  73. Cpu_max_phys_addrline_size = tmp_info[0] & 0xff;
  74. Cpu_max_linear_addrline_size = (tmp_info[0] >> 8) & 0xff;
  75. kinfo("Cpu_max_phys_addrline_size = %d", Cpu_max_phys_addrline_size);
  76. kinfo("Cpu_max_linear_addrline_size = %d", Cpu_max_linear_addrline_size);
  77. cpu_cpuid(0x80000000, 0, &tmp_info[0], &tmp_info[1], &tmp_info[2], &tmp_info[3]);
  78. Cpu_cpuid_max_Extended_mop = tmp_info[0];
  79. kinfo("Max basic mop=%#05lx", Cpu_cpuid_max_Basic_mop);
  80. kinfo("Max extended mop=%#05lx", Cpu_cpuid_max_Extended_mop);
  81. return;
  82. }
  83. void cpu_cpuid(uint32_t mop, uint32_t sop, uint32_t *eax, uint32_t *ebx, uint32_t *ecx, uint32_t *edx)
  84. {
  85. // 向eax和ecx分别输入主功能号和子功能号
  86. // 结果输出到eax, ebx, ecx, edx
  87. __asm__ __volatile__("cpuid \n\t"
  88. : "=a"(*eax), "=b"(*ebx), "=c"(*ecx), "=d"(*edx)
  89. : "0"(mop), "2"(sop)
  90. : "memory");
  91. }
  92. /**
  93. * @brief 获取当前cpu核心晶振频率(是一个Write-on-box的值)
  94. *
  95. * hint: 某些cpu无法提供该数据,返回值为0
  96. * @return uint32_t 当前cpu核心晶振频率
  97. */
  98. uint32_t cpu_get_core_crysral_freq()
  99. {
  100. uint32_t a = 0, b = 0, c = 0, d = 0;
  101. // cpu_cpuid(0x15, 0, &a, &b, &c, &d);
  102. __asm__ __volatile__("cpuid \n\t"
  103. : "=a"(a), "=b"(b), "=c"(c), "=d"(d)
  104. : "0"(0x15), "2"(0)
  105. : "memory");
  106. // kdebug("Cpu_cpuid_max_Basic_mop = %#03x, a=%ld, b=%ld, c=%ld, d=%ld", Cpu_cpuid_max_Basic_mop, a, b, c, d);
  107. return c;
  108. }