cpumap.c 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. #include "util.h"
  2. #include "../perf.h"
  3. #include "cpumap.h"
  4. #include <assert.h>
  5. #include <stdio.h>
  6. int cpumap[MAX_NR_CPUS];
  7. static int default_cpu_map(void)
  8. {
  9. int nr_cpus, i;
  10. nr_cpus = sysconf(_SC_NPROCESSORS_ONLN);
  11. assert(nr_cpus <= MAX_NR_CPUS);
  12. assert((int)nr_cpus >= 0);
  13. for (i = 0; i < nr_cpus; ++i)
  14. cpumap[i] = i;
  15. return nr_cpus;
  16. }
  17. static int read_all_cpu_map(void)
  18. {
  19. FILE *onlnf;
  20. int nr_cpus = 0;
  21. int n, cpu, prev;
  22. char sep;
  23. onlnf = fopen("/sys/devices/system/cpu/online", "r");
  24. if (!onlnf)
  25. return default_cpu_map();
  26. sep = 0;
  27. prev = -1;
  28. for (;;) {
  29. n = fscanf(onlnf, "%u%c", &cpu, &sep);
  30. if (n <= 0)
  31. break;
  32. if (prev >= 0) {
  33. assert(nr_cpus + cpu - prev - 1 < MAX_NR_CPUS);
  34. while (++prev < cpu)
  35. cpumap[nr_cpus++] = prev;
  36. }
  37. assert (nr_cpus < MAX_NR_CPUS);
  38. cpumap[nr_cpus++] = cpu;
  39. if (n == 2 && sep == '-')
  40. prev = cpu;
  41. else
  42. prev = -1;
  43. if (n == 1 || sep == '\n')
  44. break;
  45. }
  46. fclose(onlnf);
  47. if (nr_cpus > 0)
  48. return nr_cpus;
  49. return default_cpu_map();
  50. }
  51. int read_cpu_map(const char *cpu_list)
  52. {
  53. unsigned long start_cpu, end_cpu = 0;
  54. char *p = NULL;
  55. int i, nr_cpus = 0;
  56. if (!cpu_list)
  57. return read_all_cpu_map();
  58. if (!isdigit(*cpu_list))
  59. goto invalid;
  60. while (isdigit(*cpu_list)) {
  61. p = NULL;
  62. start_cpu = strtoul(cpu_list, &p, 0);
  63. if (start_cpu >= INT_MAX
  64. || (*p != '\0' && *p != ',' && *p != '-'))
  65. goto invalid;
  66. if (*p == '-') {
  67. cpu_list = ++p;
  68. p = NULL;
  69. end_cpu = strtoul(cpu_list, &p, 0);
  70. if (end_cpu >= INT_MAX || (*p != '\0' && *p != ','))
  71. goto invalid;
  72. if (end_cpu < start_cpu)
  73. goto invalid;
  74. } else {
  75. end_cpu = start_cpu;
  76. }
  77. for (; start_cpu <= end_cpu; start_cpu++) {
  78. /* check for duplicates */
  79. for (i = 0; i < nr_cpus; i++)
  80. if (cpumap[i] == (int)start_cpu)
  81. goto invalid;
  82. assert(nr_cpus < MAX_NR_CPUS);
  83. cpumap[nr_cpus++] = (int)start_cpu;
  84. }
  85. if (*p)
  86. ++p;
  87. cpu_list = p;
  88. }
  89. if (nr_cpus > 0)
  90. return nr_cpus;
  91. return default_cpu_map();
  92. invalid:
  93. return -1;
  94. }