parse-events.y 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504
  1. %pure-parser
  2. %parse-param {void *_data}
  3. %parse-param {void *scanner}
  4. %lex-param {void* scanner}
  5. %{
  6. #define YYDEBUG 1
  7. #include <linux/compiler.h>
  8. #include <linux/list.h>
  9. #include <linux/types.h>
  10. #include "util.h"
  11. #include "parse-events.h"
  12. #include "parse-events-bison.h"
  13. extern int parse_events_lex (YYSTYPE* lvalp, void* scanner);
  14. #define ABORT_ON(val) \
  15. do { \
  16. if (val) \
  17. YYABORT; \
  18. } while (0)
  19. #define ALLOC_LIST(list) \
  20. do { \
  21. list = malloc(sizeof(*list)); \
  22. ABORT_ON(!list); \
  23. INIT_LIST_HEAD(list); \
  24. } while (0)
  25. static inc_group_count(struct list_head *list,
  26. struct parse_events_evlist *data)
  27. {
  28. /* Count groups only have more than 1 members */
  29. if (!list_is_last(list->next, list))
  30. data->nr_groups++;
  31. }
  32. %}
  33. %token PE_START_EVENTS PE_START_TERMS
  34. %token PE_VALUE PE_VALUE_SYM_HW PE_VALUE_SYM_SW PE_RAW PE_TERM
  35. %token PE_EVENT_NAME
  36. %token PE_NAME
  37. %token PE_MODIFIER_EVENT PE_MODIFIER_BP
  38. %token PE_NAME_CACHE_TYPE PE_NAME_CACHE_OP_RESULT
  39. %token PE_PREFIX_MEM PE_PREFIX_RAW PE_PREFIX_GROUP
  40. %token PE_ERROR
  41. %token PE_PMU_EVENT_PRE PE_PMU_EVENT_SUF PE_KERNEL_PMU_EVENT
  42. %type <num> PE_VALUE
  43. %type <num> PE_VALUE_SYM_HW
  44. %type <num> PE_VALUE_SYM_SW
  45. %type <num> PE_RAW
  46. %type <num> PE_TERM
  47. %type <str> PE_NAME
  48. %type <str> PE_NAME_CACHE_TYPE
  49. %type <str> PE_NAME_CACHE_OP_RESULT
  50. %type <str> PE_MODIFIER_EVENT
  51. %type <str> PE_MODIFIER_BP
  52. %type <str> PE_EVENT_NAME
  53. %type <str> PE_PMU_EVENT_PRE PE_PMU_EVENT_SUF PE_KERNEL_PMU_EVENT
  54. %type <num> value_sym
  55. %type <head> event_config
  56. %type <term> event_term
  57. %type <head> event_pmu
  58. %type <head> event_legacy_symbol
  59. %type <head> event_legacy_cache
  60. %type <head> event_legacy_mem
  61. %type <head> event_legacy_tracepoint
  62. %type <head> event_legacy_numeric
  63. %type <head> event_legacy_raw
  64. %type <head> event_def
  65. %type <head> event_mod
  66. %type <head> event_name
  67. %type <head> event
  68. %type <head> events
  69. %type <head> group_def
  70. %type <head> group
  71. %type <head> groups
  72. %union
  73. {
  74. char *str;
  75. u64 num;
  76. struct list_head *head;
  77. struct parse_events_term *term;
  78. }
  79. %%
  80. start:
  81. PE_START_EVENTS start_events
  82. |
  83. PE_START_TERMS start_terms
  84. start_events: groups
  85. {
  86. struct parse_events_evlist *data = _data;
  87. parse_events_update_lists($1, &data->list);
  88. }
  89. groups:
  90. groups ',' group
  91. {
  92. struct list_head *list = $1;
  93. struct list_head *group = $3;
  94. parse_events_update_lists(group, list);
  95. $$ = list;
  96. }
  97. |
  98. groups ',' event
  99. {
  100. struct list_head *list = $1;
  101. struct list_head *event = $3;
  102. parse_events_update_lists(event, list);
  103. $$ = list;
  104. }
  105. |
  106. group
  107. |
  108. event
  109. group:
  110. group_def ':' PE_MODIFIER_EVENT
  111. {
  112. struct list_head *list = $1;
  113. ABORT_ON(parse_events__modifier_group(list, $3));
  114. $$ = list;
  115. }
  116. |
  117. group_def
  118. group_def:
  119. PE_NAME '{' events '}'
  120. {
  121. struct list_head *list = $3;
  122. inc_group_count(list, _data);
  123. parse_events__set_leader($1, list);
  124. $$ = list;
  125. }
  126. |
  127. '{' events '}'
  128. {
  129. struct list_head *list = $2;
  130. inc_group_count(list, _data);
  131. parse_events__set_leader(NULL, list);
  132. $$ = list;
  133. }
  134. events:
  135. events ',' event
  136. {
  137. struct list_head *event = $3;
  138. struct list_head *list = $1;
  139. parse_events_update_lists(event, list);
  140. $$ = list;
  141. }
  142. |
  143. event
  144. event: event_mod
  145. event_mod:
  146. event_name PE_MODIFIER_EVENT
  147. {
  148. struct list_head *list = $1;
  149. /*
  150. * Apply modifier on all events added by single event definition
  151. * (there could be more events added for multiple tracepoint
  152. * definitions via '*?'.
  153. */
  154. ABORT_ON(parse_events__modifier_event(list, $2, false));
  155. $$ = list;
  156. }
  157. |
  158. event_name
  159. event_name:
  160. PE_EVENT_NAME event_def
  161. {
  162. ABORT_ON(parse_events_name($2, $1));
  163. free($1);
  164. $$ = $2;
  165. }
  166. |
  167. event_def
  168. event_def: event_pmu |
  169. event_legacy_symbol |
  170. event_legacy_cache sep_dc |
  171. event_legacy_mem |
  172. event_legacy_tracepoint sep_dc |
  173. event_legacy_numeric sep_dc |
  174. event_legacy_raw sep_dc
  175. event_pmu:
  176. PE_NAME '/' event_config '/'
  177. {
  178. struct parse_events_evlist *data = _data;
  179. struct list_head *list;
  180. ALLOC_LIST(list);
  181. ABORT_ON(parse_events_add_pmu(list, &data->idx, $1, $3));
  182. parse_events__free_terms($3);
  183. $$ = list;
  184. }
  185. |
  186. PE_NAME '/' '/'
  187. {
  188. struct parse_events_evlist *data = _data;
  189. struct list_head *list;
  190. ALLOC_LIST(list);
  191. ABORT_ON(parse_events_add_pmu(list, &data->idx, $1, NULL));
  192. $$ = list;
  193. }
  194. |
  195. PE_KERNEL_PMU_EVENT sep_dc
  196. {
  197. struct parse_events_evlist *data = _data;
  198. struct list_head *head;
  199. struct parse_events_term *term;
  200. struct list_head *list;
  201. ALLOC_LIST(head);
  202. ABORT_ON(parse_events_term__num(&term, PARSE_EVENTS__TERM_TYPE_USER,
  203. $1, 1));
  204. list_add_tail(&term->list, head);
  205. ALLOC_LIST(list);
  206. ABORT_ON(parse_events_add_pmu(list, &data->idx, "cpu", head));
  207. parse_events__free_terms(head);
  208. $$ = list;
  209. }
  210. |
  211. PE_PMU_EVENT_PRE '-' PE_PMU_EVENT_SUF sep_dc
  212. {
  213. struct parse_events_evlist *data = _data;
  214. struct list_head *head;
  215. struct parse_events_term *term;
  216. struct list_head *list;
  217. char pmu_name[128];
  218. snprintf(&pmu_name, 128, "%s-%s", $1, $3);
  219. ALLOC_LIST(head);
  220. ABORT_ON(parse_events_term__num(&term, PARSE_EVENTS__TERM_TYPE_USER,
  221. &pmu_name, 1));
  222. list_add_tail(&term->list, head);
  223. ALLOC_LIST(list);
  224. ABORT_ON(parse_events_add_pmu(list, &data->idx, "cpu", head));
  225. parse_events__free_terms(head);
  226. $$ = list;
  227. }
  228. value_sym:
  229. PE_VALUE_SYM_HW
  230. |
  231. PE_VALUE_SYM_SW
  232. event_legacy_symbol:
  233. value_sym '/' event_config '/'
  234. {
  235. struct parse_events_evlist *data = _data;
  236. struct list_head *list;
  237. int type = $1 >> 16;
  238. int config = $1 & 255;
  239. ALLOC_LIST(list);
  240. ABORT_ON(parse_events_add_numeric(list, &data->idx,
  241. type, config, $3));
  242. parse_events__free_terms($3);
  243. $$ = list;
  244. }
  245. |
  246. value_sym sep_slash_dc
  247. {
  248. struct parse_events_evlist *data = _data;
  249. struct list_head *list;
  250. int type = $1 >> 16;
  251. int config = $1 & 255;
  252. ALLOC_LIST(list);
  253. ABORT_ON(parse_events_add_numeric(list, &data->idx,
  254. type, config, NULL));
  255. $$ = list;
  256. }
  257. event_legacy_cache:
  258. PE_NAME_CACHE_TYPE '-' PE_NAME_CACHE_OP_RESULT '-' PE_NAME_CACHE_OP_RESULT
  259. {
  260. struct parse_events_evlist *data = _data;
  261. struct list_head *list;
  262. ALLOC_LIST(list);
  263. ABORT_ON(parse_events_add_cache(list, &data->idx, $1, $3, $5));
  264. $$ = list;
  265. }
  266. |
  267. PE_NAME_CACHE_TYPE '-' PE_NAME_CACHE_OP_RESULT
  268. {
  269. struct parse_events_evlist *data = _data;
  270. struct list_head *list;
  271. ALLOC_LIST(list);
  272. ABORT_ON(parse_events_add_cache(list, &data->idx, $1, $3, NULL));
  273. $$ = list;
  274. }
  275. |
  276. PE_NAME_CACHE_TYPE
  277. {
  278. struct parse_events_evlist *data = _data;
  279. struct list_head *list;
  280. ALLOC_LIST(list);
  281. ABORT_ON(parse_events_add_cache(list, &data->idx, $1, NULL, NULL));
  282. $$ = list;
  283. }
  284. event_legacy_mem:
  285. PE_PREFIX_MEM PE_VALUE ':' PE_MODIFIER_BP sep_dc
  286. {
  287. struct parse_events_evlist *data = _data;
  288. struct list_head *list;
  289. ALLOC_LIST(list);
  290. ABORT_ON(parse_events_add_breakpoint(list, &data->idx,
  291. (void *) $2, $4));
  292. $$ = list;
  293. }
  294. |
  295. PE_PREFIX_MEM PE_VALUE sep_dc
  296. {
  297. struct parse_events_evlist *data = _data;
  298. struct list_head *list;
  299. ALLOC_LIST(list);
  300. ABORT_ON(parse_events_add_breakpoint(list, &data->idx,
  301. (void *) $2, NULL));
  302. $$ = list;
  303. }
  304. event_legacy_tracepoint:
  305. PE_NAME '-' PE_NAME ':' PE_NAME
  306. {
  307. struct parse_events_evlist *data = _data;
  308. struct list_head *list;
  309. char sys_name[128];
  310. snprintf(&sys_name, 128, "%s-%s", $1, $3);
  311. ALLOC_LIST(list);
  312. ABORT_ON(parse_events_add_tracepoint(list, &data->idx, &sys_name, $5));
  313. $$ = list;
  314. }
  315. |
  316. PE_NAME ':' PE_NAME
  317. {
  318. struct parse_events_evlist *data = _data;
  319. struct list_head *list;
  320. ALLOC_LIST(list);
  321. ABORT_ON(parse_events_add_tracepoint(list, &data->idx, $1, $3));
  322. $$ = list;
  323. }
  324. event_legacy_numeric:
  325. PE_VALUE ':' PE_VALUE
  326. {
  327. struct parse_events_evlist *data = _data;
  328. struct list_head *list;
  329. ALLOC_LIST(list);
  330. ABORT_ON(parse_events_add_numeric(list, &data->idx, (u32)$1, $3, NULL));
  331. $$ = list;
  332. }
  333. event_legacy_raw:
  334. PE_RAW
  335. {
  336. struct parse_events_evlist *data = _data;
  337. struct list_head *list;
  338. ALLOC_LIST(list);
  339. ABORT_ON(parse_events_add_numeric(list, &data->idx,
  340. PERF_TYPE_RAW, $1, NULL));
  341. $$ = list;
  342. }
  343. start_terms: event_config
  344. {
  345. struct parse_events_terms *data = _data;
  346. data->terms = $1;
  347. }
  348. event_config:
  349. event_config ',' event_term
  350. {
  351. struct list_head *head = $1;
  352. struct parse_events_term *term = $3;
  353. ABORT_ON(!head);
  354. list_add_tail(&term->list, head);
  355. $$ = $1;
  356. }
  357. |
  358. event_term
  359. {
  360. struct list_head *head = malloc(sizeof(*head));
  361. struct parse_events_term *term = $1;
  362. ABORT_ON(!head);
  363. INIT_LIST_HEAD(head);
  364. list_add_tail(&term->list, head);
  365. $$ = head;
  366. }
  367. event_term:
  368. PE_NAME '=' PE_NAME
  369. {
  370. struct parse_events_term *term;
  371. ABORT_ON(parse_events_term__str(&term, PARSE_EVENTS__TERM_TYPE_USER,
  372. $1, $3));
  373. $$ = term;
  374. }
  375. |
  376. PE_NAME '=' PE_VALUE
  377. {
  378. struct parse_events_term *term;
  379. ABORT_ON(parse_events_term__num(&term, PARSE_EVENTS__TERM_TYPE_USER,
  380. $1, $3));
  381. $$ = term;
  382. }
  383. |
  384. PE_NAME '=' PE_VALUE_SYM_HW
  385. {
  386. struct parse_events_term *term;
  387. int config = $3 & 255;
  388. ABORT_ON(parse_events_term__sym_hw(&term, $1, config));
  389. $$ = term;
  390. }
  391. |
  392. PE_NAME
  393. {
  394. struct parse_events_term *term;
  395. ABORT_ON(parse_events_term__num(&term, PARSE_EVENTS__TERM_TYPE_USER,
  396. $1, 1));
  397. $$ = term;
  398. }
  399. |
  400. PE_VALUE_SYM_HW
  401. {
  402. struct parse_events_term *term;
  403. int config = $1 & 255;
  404. ABORT_ON(parse_events_term__sym_hw(&term, NULL, config));
  405. $$ = term;
  406. }
  407. |
  408. PE_TERM '=' PE_NAME
  409. {
  410. struct parse_events_term *term;
  411. ABORT_ON(parse_events_term__str(&term, (int)$1, NULL, $3));
  412. $$ = term;
  413. }
  414. |
  415. PE_TERM '=' PE_VALUE
  416. {
  417. struct parse_events_term *term;
  418. ABORT_ON(parse_events_term__num(&term, (int)$1, NULL, $3));
  419. $$ = term;
  420. }
  421. |
  422. PE_TERM
  423. {
  424. struct parse_events_term *term;
  425. ABORT_ON(parse_events_term__num(&term, (int)$1, NULL, 1));
  426. $$ = term;
  427. }
  428. sep_dc: ':' |
  429. sep_slash_dc: '/' | ':' |
  430. %%
  431. void parse_events_error(void *data __maybe_unused, void *scanner __maybe_unused,
  432. char const *msg __maybe_unused)
  433. {
  434. }