#ifndef LINUX_POWERPC_PERF_REQ_GEN_PERF_H_ #define LINUX_POWERPC_PERF_REQ_GEN_PERF_H_ #include #ifndef REQUEST_FILE #error "REQUEST_FILE must be defined before including" #endif #ifndef NAME_LOWER #error "NAME_LOWER must be defined before including" #endif #ifndef NAME_UPPER #error "NAME_UPPER must be defined before including" #endif #define BE_TYPE_b1 __u8 #define BE_TYPE_b2 __be16 #define BE_TYPE_b4 __be32 #define BE_TYPE_b8 __be64 #define BYTES_TO_BE_TYPE(bytes) \ BE_TYPE_b##bytes #define CAT2_(a, b) a ## b #define CAT2(a, b) CAT2_(a, b) #define CAT3_(a, b, c) a ## b ## c #define CAT3(a, b, c) CAT3_(a, b, c) /* * enumerate the request values as * _ = */ #define REQUEST_VALUE__(name_upper, r_name) name_upper ## _ ## r_name #define REQUEST_VALUE_(name_upper, r_name) REQUEST_VALUE__(name_upper, r_name) #define REQUEST_VALUE(r_name) REQUEST_VALUE_(NAME_UPPER, r_name) #include "_clear.h" #define REQUEST_(r_name, r_value, r_idx_1, r_fields) \ REQUEST_VALUE(r_name) = r_value, enum CAT2(NAME_LOWER, _requests) { #include REQUEST_FILE }; /* * For each request: * struct _ { * r_fields * }; */ #include "_clear.h" #define STRUCT_NAME__(name_lower, r_name) name_lower ## _ ## r_name #define STRUCT_NAME_(name_lower, r_name) STRUCT_NAME__(name_lower, r_name) #define STRUCT_NAME(r_name) STRUCT_NAME_(NAME_LOWER, r_name) #define REQUEST_(r_name, r_value, r_idx_1, r_fields) \ struct STRUCT_NAME(r_name) { \ r_fields \ }; #define __field_(r_name, r_value, r_idx_1, f_offset, f_bytes, f_name) \ BYTES_TO_BE_TYPE(f_bytes) f_name; #define __count_(r_name, r_value, r_idx_1, f_offset, f_bytes, f_name) \ __field_(r_name, r_value, r_idx_1, f_offset, f_bytes, f_name) #define __array_(r_name, r_value, r_idx_1, a_offset, a_bytes, a_name) \ __u8 a_name[a_bytes]; #include REQUEST_FILE /* * Generate a check of the field offsets * _assert_offsets_correct() */ #include "_clear.h" #define REQUEST_(r_name, r_value, index, r_fields) \ r_fields #define __field_(r_name, r_value, r_idx_1, f_offset, f_size, f_name) \ BUILD_BUG_ON(offsetof(struct STRUCT_NAME(r_name), f_name) != f_offset); #define __count_(r_name, r_value, r_idx_1, c_offset, c_size, c_name) \ __field_(r_name, r_value, r_idx_1, c_offset, c_size, c_name) #define __array_(r_name, r_value, r_idx_1, a_offset, a_size, a_name) \ __field_(r_name, r_value, r_idx_1, a_offset, a_size, a_name) static inline void CAT2(NAME_LOWER, _assert_offsets_correct)(void) { #include REQUEST_FILE } /* * Generate event attributes: * PMU_EVENT_ATTR_STRING(_, * _event_attr__, * "request=" * "starting_index=" * "counter_info_version=CURRENT_COUNTER_INFO_VERSION" * "length=" * "offset=") * * TODO: counter_info_version may need to vary, we should interperate the * value to some extent */ #define EVENT_ATTR_NAME__(name, r_name, c_name) \ name ## _event_attr_ ## r_name ## _ ## c_name #define EVENT_ATTR_NAME_(name, r_name, c_name) \ EVENT_ATTR_NAME__(name, r_name, c_name) #define EVENT_ATTR_NAME(r_name, c_name) \ EVENT_ATTR_NAME_(NAME_LOWER, r_name, c_name) #include "_clear.h" #define __field_(r_name, r_value, r_idx_1, f_offset, f_size, f_name) #define __array_(r_name, r_value, r_idx_1, a_offset, a_size, a_name) #define __count_(r_name, r_value, r_idx_1, c_offset, c_size, c_name) \ PMU_EVENT_ATTR_STRING( \ CAT3(r_name, _, c_name), \ EVENT_ATTR_NAME(r_name, c_name), \ "request=" __stringify(r_value) "," \ r_idx_1 "," \ "counter_info_version=" \ __stringify(COUNTER_INFO_VERSION_CURRENT) "," \ "length=" #c_size "," \ "offset=" #c_offset) #define REQUEST_(r_name, r_value, r_idx_1, r_fields) \ r_fields #include REQUEST_FILE /* * Define event attribute array * static struct attribute *hv_gpci_event_attrs[] = { * &_event_attr__.attr, * }; */ #include "_clear.h" #define __field_(r_name, r_value, r_idx_1, f_offset, f_size, f_name) #define __count_(r_name, r_value, r_idx_1, c_offset, c_size, c_name) \ &EVENT_ATTR_NAME(r_name, c_name).attr.attr, #define __array_(r_name, r_value, r_idx_1, a_offset, a_size, a_name) #define REQUEST_(r_name, r_value, r_idx_1, r_fields) \ r_fields static __maybe_unused struct attribute *hv_gpci_event_attrs[] = { #include REQUEST_FILE NULL }; /* cleanup */ #include "_clear.h" #undef EVENT_ATTR_NAME #undef EVENT_ATTR_NAME_ #undef BIT_NAME #undef BIT_NAME_ #undef STRUCT_NAME #undef REQUEST_VALUE #undef REQUEST_VALUE_ #endif