pmu.c 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470
  1. #include <linux/list.h>
  2. #include <sys/types.h>
  3. #include <sys/stat.h>
  4. #include <unistd.h>
  5. #include <stdio.h>
  6. #include <dirent.h>
  7. #include "sysfs.h"
  8. #include "util.h"
  9. #include "pmu.h"
  10. #include "parse-events.h"
  11. int perf_pmu_parse(struct list_head *list, char *name);
  12. extern FILE *perf_pmu_in;
  13. static LIST_HEAD(pmus);
  14. /*
  15. * Parse & process all the sysfs attributes located under
  16. * the directory specified in 'dir' parameter.
  17. */
  18. static int pmu_format_parse(char *dir, struct list_head *head)
  19. {
  20. struct dirent *evt_ent;
  21. DIR *format_dir;
  22. int ret = 0;
  23. format_dir = opendir(dir);
  24. if (!format_dir)
  25. return -EINVAL;
  26. while (!ret && (evt_ent = readdir(format_dir))) {
  27. char path[PATH_MAX];
  28. char *name = evt_ent->d_name;
  29. FILE *file;
  30. if (!strcmp(name, ".") || !strcmp(name, ".."))
  31. continue;
  32. snprintf(path, PATH_MAX, "%s/%s", dir, name);
  33. ret = -EINVAL;
  34. file = fopen(path, "r");
  35. if (!file)
  36. break;
  37. perf_pmu_in = file;
  38. ret = perf_pmu_parse(head, name);
  39. fclose(file);
  40. }
  41. closedir(format_dir);
  42. return ret;
  43. }
  44. /*
  45. * Reading/parsing the default pmu format definition, which should be
  46. * located at:
  47. * /sys/bus/event_source/devices/<dev>/format as sysfs group attributes.
  48. */
  49. static int pmu_format(char *name, struct list_head *format)
  50. {
  51. struct stat st;
  52. char path[PATH_MAX];
  53. const char *sysfs;
  54. sysfs = sysfs_find_mountpoint();
  55. if (!sysfs)
  56. return -1;
  57. snprintf(path, PATH_MAX,
  58. "%s/bus/event_source/devices/%s/format", sysfs, name);
  59. if (stat(path, &st) < 0)
  60. return -1;
  61. if (pmu_format_parse(path, format))
  62. return -1;
  63. return 0;
  64. }
  65. /*
  66. * Reading/parsing the default pmu type value, which should be
  67. * located at:
  68. * /sys/bus/event_source/devices/<dev>/type as sysfs attribute.
  69. */
  70. static int pmu_type(char *name, __u32 *type)
  71. {
  72. struct stat st;
  73. char path[PATH_MAX];
  74. const char *sysfs;
  75. FILE *file;
  76. int ret = 0;
  77. sysfs = sysfs_find_mountpoint();
  78. if (!sysfs)
  79. return -1;
  80. snprintf(path, PATH_MAX,
  81. "%s/bus/event_source/devices/%s/type", sysfs, name);
  82. if (stat(path, &st) < 0)
  83. return -1;
  84. file = fopen(path, "r");
  85. if (!file)
  86. return -EINVAL;
  87. if (1 != fscanf(file, "%u", type))
  88. ret = -1;
  89. fclose(file);
  90. return ret;
  91. }
  92. static struct perf_pmu *pmu_lookup(char *name)
  93. {
  94. struct perf_pmu *pmu;
  95. LIST_HEAD(format);
  96. __u32 type;
  97. /*
  98. * The pmu data we store & need consists of the pmu
  99. * type value and format definitions. Load both right
  100. * now.
  101. */
  102. if (pmu_format(name, &format))
  103. return NULL;
  104. if (pmu_type(name, &type))
  105. return NULL;
  106. pmu = zalloc(sizeof(*pmu));
  107. if (!pmu)
  108. return NULL;
  109. INIT_LIST_HEAD(&pmu->format);
  110. list_splice(&format, &pmu->format);
  111. pmu->name = strdup(name);
  112. pmu->type = type;
  113. return pmu;
  114. }
  115. static struct perf_pmu *pmu_find(char *name)
  116. {
  117. struct perf_pmu *pmu;
  118. list_for_each_entry(pmu, &pmus, list)
  119. if (!strcmp(pmu->name, name))
  120. return pmu;
  121. return NULL;
  122. }
  123. struct perf_pmu *perf_pmu__find(char *name)
  124. {
  125. struct perf_pmu *pmu;
  126. /*
  127. * Once PMU is loaded it stays in the list,
  128. * so we keep us from multiple reading/parsing
  129. * the pmu format definitions.
  130. */
  131. pmu = pmu_find(name);
  132. if (pmu)
  133. return pmu;
  134. return pmu_lookup(name);
  135. }
  136. static struct perf_pmu__format*
  137. pmu_find_format(struct list_head *formats, char *name)
  138. {
  139. struct perf_pmu__format *format;
  140. list_for_each_entry(format, formats, list)
  141. if (!strcmp(format->name, name))
  142. return format;
  143. return NULL;
  144. }
  145. /*
  146. * Returns value based on the format definition (format parameter)
  147. * and unformated value (value parameter).
  148. *
  149. * TODO maybe optimize a little ;)
  150. */
  151. static __u64 pmu_format_value(unsigned long *format, __u64 value)
  152. {
  153. unsigned long fbit, vbit;
  154. __u64 v = 0;
  155. for (fbit = 0, vbit = 0; fbit < PERF_PMU_FORMAT_BITS; fbit++) {
  156. if (!test_bit(fbit, format))
  157. continue;
  158. if (!(value & (1llu << vbit++)))
  159. continue;
  160. v |= (1llu << fbit);
  161. }
  162. return v;
  163. }
  164. /*
  165. * Setup one of config[12] attr members based on the
  166. * user input data - temr parameter.
  167. */
  168. static int pmu_config_term(struct list_head *formats,
  169. struct perf_event_attr *attr,
  170. struct parse_events__term *term)
  171. {
  172. struct perf_pmu__format *format;
  173. __u64 *vp;
  174. /*
  175. * Support only for hardcoded and numnerial terms.
  176. * Hardcoded terms should be already in, so nothing
  177. * to be done for them.
  178. */
  179. if (parse_events__is_hardcoded_term(term))
  180. return 0;
  181. if (term->type != PARSE_EVENTS__TERM_TYPE_NUM)
  182. return -EINVAL;
  183. format = pmu_find_format(formats, term->config);
  184. if (!format)
  185. return -EINVAL;
  186. switch (format->value) {
  187. case PERF_PMU_FORMAT_VALUE_CONFIG:
  188. vp = &attr->config;
  189. break;
  190. case PERF_PMU_FORMAT_VALUE_CONFIG1:
  191. vp = &attr->config1;
  192. break;
  193. case PERF_PMU_FORMAT_VALUE_CONFIG2:
  194. vp = &attr->config2;
  195. break;
  196. default:
  197. return -EINVAL;
  198. }
  199. *vp |= pmu_format_value(format->bits, term->val.num);
  200. return 0;
  201. }
  202. static int pmu_config(struct list_head *formats, struct perf_event_attr *attr,
  203. struct list_head *head_terms)
  204. {
  205. struct parse_events__term *term, *h;
  206. list_for_each_entry_safe(term, h, head_terms, list)
  207. if (pmu_config_term(formats, attr, term))
  208. return -EINVAL;
  209. return 0;
  210. }
  211. /*
  212. * Configures event's 'attr' parameter based on the:
  213. * 1) users input - specified in terms parameter
  214. * 2) pmu format definitions - specified by pmu parameter
  215. */
  216. int perf_pmu__config(struct perf_pmu *pmu, struct perf_event_attr *attr,
  217. struct list_head *head_terms)
  218. {
  219. attr->type = pmu->type;
  220. return pmu_config(&pmu->format, attr, head_terms);
  221. }
  222. int perf_pmu__new_format(struct list_head *list, char *name,
  223. int config, unsigned long *bits)
  224. {
  225. struct perf_pmu__format *format;
  226. format = zalloc(sizeof(*format));
  227. if (!format)
  228. return -ENOMEM;
  229. format->name = strdup(name);
  230. format->value = config;
  231. memcpy(format->bits, bits, sizeof(format->bits));
  232. list_add_tail(&format->list, list);
  233. return 0;
  234. }
  235. void perf_pmu__set_format(unsigned long *bits, long from, long to)
  236. {
  237. long b;
  238. if (!to)
  239. to = from;
  240. memset(bits, 0, BITS_TO_LONGS(PERF_PMU_FORMAT_BITS));
  241. for (b = from; b <= to; b++)
  242. set_bit(b, bits);
  243. }
  244. /* Simulated format definitions. */
  245. static struct test_format {
  246. const char *name;
  247. const char *value;
  248. } test_formats[] = {
  249. { "krava01", "config:0-1,62-63\n", },
  250. { "krava02", "config:10-17\n", },
  251. { "krava03", "config:5\n", },
  252. { "krava11", "config1:0,2,4,6,8,20-28\n", },
  253. { "krava12", "config1:63\n", },
  254. { "krava13", "config1:45-47\n", },
  255. { "krava21", "config2:0-3,10-13,20-23,30-33,40-43,50-53,60-63\n", },
  256. { "krava22", "config2:8,18,48,58\n", },
  257. { "krava23", "config2:28-29,38\n", },
  258. };
  259. #define TEST_FORMATS_CNT (sizeof(test_formats) / sizeof(struct test_format))
  260. /* Simulated users input. */
  261. static struct parse_events__term test_terms[] = {
  262. {
  263. .config = (char *) "krava01",
  264. .val.num = 15,
  265. .type = PARSE_EVENTS__TERM_TYPE_NUM,
  266. },
  267. {
  268. .config = (char *) "krava02",
  269. .val.num = 170,
  270. .type = PARSE_EVENTS__TERM_TYPE_NUM,
  271. },
  272. {
  273. .config = (char *) "krava03",
  274. .val.num = 1,
  275. .type = PARSE_EVENTS__TERM_TYPE_NUM,
  276. },
  277. {
  278. .config = (char *) "krava11",
  279. .val.num = 27,
  280. .type = PARSE_EVENTS__TERM_TYPE_NUM,
  281. },
  282. {
  283. .config = (char *) "krava12",
  284. .val.num = 1,
  285. .type = PARSE_EVENTS__TERM_TYPE_NUM,
  286. },
  287. {
  288. .config = (char *) "krava13",
  289. .val.num = 2,
  290. .type = PARSE_EVENTS__TERM_TYPE_NUM,
  291. },
  292. {
  293. .config = (char *) "krava21",
  294. .val.num = 119,
  295. .type = PARSE_EVENTS__TERM_TYPE_NUM,
  296. },
  297. {
  298. .config = (char *) "krava22",
  299. .val.num = 11,
  300. .type = PARSE_EVENTS__TERM_TYPE_NUM,
  301. },
  302. {
  303. .config = (char *) "krava23",
  304. .val.num = 2,
  305. .type = PARSE_EVENTS__TERM_TYPE_NUM,
  306. },
  307. };
  308. #define TERMS_CNT (sizeof(test_terms) / sizeof(struct parse_events__term))
  309. /*
  310. * Prepare format directory data, exported by kernel
  311. * at /sys/bus/event_source/devices/<dev>/format.
  312. */
  313. static char *test_format_dir_get(void)
  314. {
  315. static char dir[PATH_MAX];
  316. unsigned int i;
  317. snprintf(dir, PATH_MAX, "/tmp/perf-pmu-test-format-XXXXXX");
  318. if (!mkdtemp(dir))
  319. return NULL;
  320. for (i = 0; i < TEST_FORMATS_CNT; i++) {
  321. static char name[PATH_MAX];
  322. struct test_format *format = &test_formats[i];
  323. FILE *file;
  324. snprintf(name, PATH_MAX, "%s/%s", dir, format->name);
  325. file = fopen(name, "w");
  326. if (!file)
  327. return NULL;
  328. if (1 != fwrite(format->value, strlen(format->value), 1, file))
  329. break;
  330. fclose(file);
  331. }
  332. return dir;
  333. }
  334. /* Cleanup format directory. */
  335. static int test_format_dir_put(char *dir)
  336. {
  337. char buf[PATH_MAX];
  338. snprintf(buf, PATH_MAX, "rm -f %s/*\n", dir);
  339. if (system(buf))
  340. return -1;
  341. snprintf(buf, PATH_MAX, "rmdir %s\n", dir);
  342. return system(buf);
  343. }
  344. static struct list_head *test_terms_list(void)
  345. {
  346. static LIST_HEAD(terms);
  347. unsigned int i;
  348. for (i = 0; i < TERMS_CNT; i++)
  349. list_add_tail(&test_terms[i].list, &terms);
  350. return &terms;
  351. }
  352. #undef TERMS_CNT
  353. int perf_pmu__test(void)
  354. {
  355. char *format = test_format_dir_get();
  356. LIST_HEAD(formats);
  357. struct list_head *terms = test_terms_list();
  358. int ret;
  359. if (!format)
  360. return -EINVAL;
  361. do {
  362. struct perf_event_attr attr;
  363. memset(&attr, 0, sizeof(attr));
  364. ret = pmu_format_parse(format, &formats);
  365. if (ret)
  366. break;
  367. ret = pmu_config(&formats, &attr, terms);
  368. if (ret)
  369. break;
  370. ret = -EINVAL;
  371. if (attr.config != 0xc00000000002a823)
  372. break;
  373. if (attr.config1 != 0x8000400000000145)
  374. break;
  375. if (attr.config2 != 0x0400000020041d07)
  376. break;
  377. ret = 0;
  378. } while (0);
  379. test_format_dir_put(format);
  380. return ret;
  381. }