pmu.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. #include "parse-events.h"
  2. #include "pmu.h"
  3. #include "util.h"
  4. #include "tests.h"
  5. #include <errno.h>
  6. #include <linux/kernel.h>
  7. /* Simulated format definitions. */
  8. static struct test_format {
  9. const char *name;
  10. const char *value;
  11. } test_formats[] = {
  12. { "krava01", "config:0-1,62-63\n", },
  13. { "krava02", "config:10-17\n", },
  14. { "krava03", "config:5\n", },
  15. { "krava11", "config1:0,2,4,6,8,20-28\n", },
  16. { "krava12", "config1:63\n", },
  17. { "krava13", "config1:45-47\n", },
  18. { "krava21", "config2:0-3,10-13,20-23,30-33,40-43,50-53,60-63\n", },
  19. { "krava22", "config2:8,18,48,58\n", },
  20. { "krava23", "config2:28-29,38\n", },
  21. };
  22. /* Simulated users input. */
  23. static struct parse_events_term test_terms[] = {
  24. {
  25. .config = (char *) "krava01",
  26. .val.num = 15,
  27. .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
  28. .type_term = PARSE_EVENTS__TERM_TYPE_USER,
  29. },
  30. {
  31. .config = (char *) "krava02",
  32. .val.num = 170,
  33. .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
  34. .type_term = PARSE_EVENTS__TERM_TYPE_USER,
  35. },
  36. {
  37. .config = (char *) "krava03",
  38. .val.num = 1,
  39. .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
  40. .type_term = PARSE_EVENTS__TERM_TYPE_USER,
  41. },
  42. {
  43. .config = (char *) "krava11",
  44. .val.num = 27,
  45. .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
  46. .type_term = PARSE_EVENTS__TERM_TYPE_USER,
  47. },
  48. {
  49. .config = (char *) "krava12",
  50. .val.num = 1,
  51. .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
  52. .type_term = PARSE_EVENTS__TERM_TYPE_USER,
  53. },
  54. {
  55. .config = (char *) "krava13",
  56. .val.num = 2,
  57. .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
  58. .type_term = PARSE_EVENTS__TERM_TYPE_USER,
  59. },
  60. {
  61. .config = (char *) "krava21",
  62. .val.num = 119,
  63. .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
  64. .type_term = PARSE_EVENTS__TERM_TYPE_USER,
  65. },
  66. {
  67. .config = (char *) "krava22",
  68. .val.num = 11,
  69. .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
  70. .type_term = PARSE_EVENTS__TERM_TYPE_USER,
  71. },
  72. {
  73. .config = (char *) "krava23",
  74. .val.num = 2,
  75. .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
  76. .type_term = PARSE_EVENTS__TERM_TYPE_USER,
  77. },
  78. };
  79. /*
  80. * Prepare format directory data, exported by kernel
  81. * at /sys/bus/event_source/devices/<dev>/format.
  82. */
  83. static char *test_format_dir_get(void)
  84. {
  85. static char dir[PATH_MAX];
  86. unsigned int i;
  87. snprintf(dir, PATH_MAX, "/tmp/perf-pmu-test-format-XXXXXX");
  88. if (!mkdtemp(dir))
  89. return NULL;
  90. for (i = 0; i < ARRAY_SIZE(test_formats); i++) {
  91. static char name[PATH_MAX];
  92. struct test_format *format = &test_formats[i];
  93. FILE *file;
  94. snprintf(name, PATH_MAX, "%s/%s", dir, format->name);
  95. file = fopen(name, "w");
  96. if (!file)
  97. return NULL;
  98. if (1 != fwrite(format->value, strlen(format->value), 1, file))
  99. break;
  100. fclose(file);
  101. }
  102. return dir;
  103. }
  104. /* Cleanup format directory. */
  105. static int test_format_dir_put(char *dir)
  106. {
  107. char buf[PATH_MAX];
  108. snprintf(buf, PATH_MAX, "rm -f %s/*\n", dir);
  109. if (system(buf))
  110. return -1;
  111. snprintf(buf, PATH_MAX, "rmdir %s\n", dir);
  112. return system(buf);
  113. }
  114. static struct list_head *test_terms_list(void)
  115. {
  116. static LIST_HEAD(terms);
  117. unsigned int i;
  118. for (i = 0; i < ARRAY_SIZE(test_terms); i++)
  119. list_add_tail(&test_terms[i].list, &terms);
  120. return &terms;
  121. }
  122. int test__pmu(int subtest __maybe_unused)
  123. {
  124. char *format = test_format_dir_get();
  125. LIST_HEAD(formats);
  126. struct list_head *terms = test_terms_list();
  127. int ret;
  128. if (!format)
  129. return -EINVAL;
  130. do {
  131. struct perf_event_attr attr;
  132. memset(&attr, 0, sizeof(attr));
  133. ret = perf_pmu__format_parse(format, &formats);
  134. if (ret)
  135. break;
  136. ret = perf_pmu__config_terms(&formats, &attr, terms,
  137. false, NULL);
  138. if (ret)
  139. break;
  140. ret = -EINVAL;
  141. if (attr.config != 0xc00000000002a823)
  142. break;
  143. if (attr.config1 != 0x8000400000000145)
  144. break;
  145. if (attr.config2 != 0x0400000020041d07)
  146. break;
  147. ret = 0;
  148. } while (0);
  149. test_format_dir_put(format);
  150. return ret;
  151. }