cpu_parse.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. /* cpu_parse.c
  2. * Parse CPU-related files.
  3. */
  4. #include <stdlib.h>
  5. #include <stdio.h>
  6. #include <string.h>
  7. #include <sys/types.h>
  8. #include <dirent.h>
  9. #include <limits.h>
  10. #include <ctype.h>
  11. #include <unistd.h>
  12. #include "lub/list.h"
  13. #include "cpumask.h"
  14. #include "cpu.h"
  15. #include "irq.h"
  16. #define STR(str) ( str ? str : "" )
  17. int cpu_list_compare(const void *first, const void *second)
  18. {
  19. const cpu_t *f = (const cpu_t *)first;
  20. const cpu_t *s = (const cpu_t *)second;
  21. return (f->id - s->id);
  22. }
  23. static cpu_t * cpu_new(unsigned int id)
  24. {
  25. cpu_t *new;
  26. if (!(new = malloc(sizeof(*new))))
  27. return NULL;
  28. new->id = id;
  29. new->old_load_all = 0;
  30. new->old_load_irq = 0;
  31. new->load = 0;
  32. new->irqs = lub_list_new(irq_list_compare);
  33. return new;
  34. }
  35. static void cpu_free(cpu_t *cpu)
  36. {
  37. lub_list_free(cpu->irqs);
  38. free(cpu);
  39. }
  40. static cpu_t * cpu_list_search_ht(lub_list_t *cpus,
  41. unsigned int package_id,
  42. unsigned int core_id)
  43. {
  44. lub_list_node_t *iter;
  45. for (iter = lub_list_iterator_init(cpus); iter;
  46. iter = lub_list_iterator_next(iter)) {
  47. cpu_t *cpu;
  48. cpu = (cpu_t *)lub_list_node__get_data(iter);
  49. if (cpu->package_id != package_id)
  50. continue;
  51. if (cpu->core_id != core_id)
  52. continue;
  53. return cpu;
  54. }
  55. return NULL;
  56. }
  57. cpu_t * cpu_list_search(lub_list_t *cpus, unsigned int id)
  58. {
  59. lub_list_node_t *node;
  60. cpu_t search;
  61. search.id = id;
  62. node = lub_list_search(cpus, &search);
  63. if (!node)
  64. return NULL;
  65. return (cpu_t *)lub_list_node__get_data(node);
  66. }
  67. static cpu_t * cpu_list_add(lub_list_t *cpus, cpu_t *cpu)
  68. {
  69. cpu_t *old = cpu_list_search(cpus, cpu->id);
  70. if (old) /* CPU already exists. May be renew some fields later */
  71. return old;
  72. lub_list_add(cpus, cpu);
  73. return cpu;
  74. }
  75. int cpu_list_free(lub_list_t *cpus)
  76. {
  77. lub_list_node_t *iter;
  78. while ((iter = lub_list__get_head(cpus))) {
  79. cpu_t *cpu;
  80. cpu = (cpu_t *)lub_list_node__get_data(iter);
  81. cpu_free(cpu);
  82. lub_list_del(cpus, iter);
  83. lub_list_node_free(iter);
  84. }
  85. lub_list_free(cpus);
  86. return 0;
  87. }
  88. /* Show CPU information */
  89. static void show_cpu_info(cpu_t *cpu)
  90. {
  91. char buf[NR_CPUS + 1];
  92. cpumask_scnprintf(buf, sizeof(buf), cpu->cpumask);
  93. printf("CPU %d package %d core %d mask %s\n", cpu->id, cpu->package_id, cpu->core_id, buf);
  94. }
  95. /* Show CPU list */
  96. int show_cpus(lub_list_t *cpus)
  97. {
  98. lub_list_node_t *iter;
  99. for (iter = lub_list_iterator_init(cpus); iter;
  100. iter = lub_list_iterator_next(iter)) {
  101. cpu_t *cpu;
  102. cpu = (cpu_t *)lub_list_node__get_data(iter);
  103. show_cpu_info(cpu);
  104. }
  105. return 0;
  106. }
  107. int scan_cpus(lub_list_t *cpus)
  108. {
  109. FILE *fd;
  110. char path[PATH_MAX];
  111. unsigned int id;
  112. unsigned int package_id;
  113. unsigned int core_id;
  114. cpu_t *new;
  115. for (id = 0; id < NR_CPUS; id++) {
  116. sprintf(path, "%s/cpu%d", SYSFS_CPU_PATH, id);
  117. if (access(path, F_OK))
  118. break;
  119. /* Try to get package_id */
  120. sprintf(path, "%s/cpu%d/topology/physical_package_id",
  121. SYSFS_CPU_PATH, id);
  122. if (!(fd = fopen(path, "r")))
  123. continue;
  124. if (fscanf(fd, "%u", &package_id) < 0) {
  125. fclose(fd);
  126. continue;
  127. }
  128. fclose(fd);
  129. /* Try to get core_id */
  130. sprintf(path, "%s/cpu%d/topology/core_id",
  131. SYSFS_CPU_PATH, id);
  132. if (!(fd = fopen(path, "r")))
  133. continue;
  134. if (fscanf(fd, "%u", &core_id) < 0) {
  135. fclose(fd);
  136. continue;
  137. }
  138. fclose(fd);
  139. /* Don't use second thread of Hyper Threading */
  140. if (cpu_list_search_ht(cpus, package_id, core_id))
  141. continue;
  142. new = cpu_new(id);
  143. new->package_id = package_id;
  144. new->core_id = core_id;
  145. cpus_clear(new->cpumask);
  146. cpu_set(new->id, new->cpumask);
  147. cpu_list_add(cpus, new);
  148. }
  149. return 0;
  150. }