162306a36Sopenharmony_ci// SPDX-License-Identifier: LGPL-2.1
262306a36Sopenharmony_ci#ifndef PERF_FLAG_FD_NO_GROUP
362306a36Sopenharmony_ci# define PERF_FLAG_FD_NO_GROUP		(1UL << 0)
462306a36Sopenharmony_ci#endif
562306a36Sopenharmony_ci
662306a36Sopenharmony_ci#ifndef PERF_FLAG_FD_OUTPUT
762306a36Sopenharmony_ci# define PERF_FLAG_FD_OUTPUT		(1UL << 1)
862306a36Sopenharmony_ci#endif
962306a36Sopenharmony_ci
1062306a36Sopenharmony_ci#ifndef PERF_FLAG_PID_CGROUP
1162306a36Sopenharmony_ci# define PERF_FLAG_PID_CGROUP		(1UL << 2) /* pid=cgroup id, per-cpu mode only */
1262306a36Sopenharmony_ci#endif
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci#ifndef PERF_FLAG_FD_CLOEXEC
1562306a36Sopenharmony_ci# define PERF_FLAG_FD_CLOEXEC		(1UL << 3) /* O_CLOEXEC */
1662306a36Sopenharmony_ci#endif
1762306a36Sopenharmony_ci
1862306a36Sopenharmony_cistatic size_t syscall_arg__scnprintf_perf_flags(char *bf, size_t size,
1962306a36Sopenharmony_ci						struct syscall_arg *arg)
2062306a36Sopenharmony_ci{
2162306a36Sopenharmony_ci	bool show_prefix = arg->show_string_prefix;
2262306a36Sopenharmony_ci	const char *prefix = "PERF_";
2362306a36Sopenharmony_ci	int printed = 0, flags = arg->val;
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_ci	if (flags == 0)
2662306a36Sopenharmony_ci		return 0;
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci#define	P_FLAG(n) \
2962306a36Sopenharmony_ci	if (flags & PERF_FLAG_##n) { \
3062306a36Sopenharmony_ci		printed += scnprintf(bf + printed, size - printed, "%s%s%s", printed ? "|" : "", show_prefix ? prefix : "", #n); \
3162306a36Sopenharmony_ci		flags &= ~PERF_FLAG_##n; \
3262306a36Sopenharmony_ci	}
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_ci	P_FLAG(FD_NO_GROUP);
3562306a36Sopenharmony_ci	P_FLAG(FD_OUTPUT);
3662306a36Sopenharmony_ci	P_FLAG(PID_CGROUP);
3762306a36Sopenharmony_ci	P_FLAG(FD_CLOEXEC);
3862306a36Sopenharmony_ci#undef P_FLAG
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_ci	if (flags)
4162306a36Sopenharmony_ci		printed += scnprintf(bf + printed, size - printed, "%s%#x", printed ? "|" : "", flags);
4262306a36Sopenharmony_ci
4362306a36Sopenharmony_ci	return printed;
4462306a36Sopenharmony_ci}
4562306a36Sopenharmony_ci
4662306a36Sopenharmony_ci#define SCA_PERF_FLAGS syscall_arg__scnprintf_perf_flags
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_cistruct attr_fprintf_args {
4962306a36Sopenharmony_ci	size_t size, printed;
5062306a36Sopenharmony_ci	char *bf;
5162306a36Sopenharmony_ci	bool first;
5262306a36Sopenharmony_ci};
5362306a36Sopenharmony_ci
5462306a36Sopenharmony_cistatic int attr__fprintf(FILE *fp __maybe_unused, const char *name, const char *val, void *priv)
5562306a36Sopenharmony_ci{
5662306a36Sopenharmony_ci	struct attr_fprintf_args *args = priv;
5762306a36Sopenharmony_ci	size_t printed = scnprintf(args->bf + args->printed , args->size - args->printed, "%s%s: %s", args->first ? "" : ", ", name, val);
5862306a36Sopenharmony_ci
5962306a36Sopenharmony_ci	args->first = false;
6062306a36Sopenharmony_ci	args->printed += printed;
6162306a36Sopenharmony_ci	return printed;
6262306a36Sopenharmony_ci}
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_cistatic size_t perf_event_attr___scnprintf(struct perf_event_attr *attr, char *bf, size_t size, bool show_zeros __maybe_unused)
6562306a36Sopenharmony_ci{
6662306a36Sopenharmony_ci	struct attr_fprintf_args args = {
6762306a36Sopenharmony_ci		.printed = scnprintf(bf, size, "{ "),
6862306a36Sopenharmony_ci		.size    = size,
6962306a36Sopenharmony_ci		.first   = true,
7062306a36Sopenharmony_ci		.bf	 = bf,
7162306a36Sopenharmony_ci	};
7262306a36Sopenharmony_ci
7362306a36Sopenharmony_ci	perf_event_attr__fprintf(stdout, attr, attr__fprintf, &args);
7462306a36Sopenharmony_ci	return args.printed + scnprintf(bf + args.printed, size - args.printed, " }");
7562306a36Sopenharmony_ci}
7662306a36Sopenharmony_ci
7762306a36Sopenharmony_cistatic size_t syscall_arg__scnprintf_augmented_perf_event_attr(struct syscall_arg *arg, char *bf, size_t size)
7862306a36Sopenharmony_ci{
7962306a36Sopenharmony_ci	return perf_event_attr___scnprintf((void *)arg->augmented.args, bf, size, arg->trace->show_zeros);
8062306a36Sopenharmony_ci}
8162306a36Sopenharmony_ci
8262306a36Sopenharmony_cistatic size_t syscall_arg__scnprintf_perf_event_attr(char *bf, size_t size, struct syscall_arg *arg)
8362306a36Sopenharmony_ci{
8462306a36Sopenharmony_ci	if (arg->augmented.args)
8562306a36Sopenharmony_ci		return syscall_arg__scnprintf_augmented_perf_event_attr(arg, bf, size);
8662306a36Sopenharmony_ci
8762306a36Sopenharmony_ci	return scnprintf(bf, size, "%#lx", arg->val);
8862306a36Sopenharmony_ci}
8962306a36Sopenharmony_ci
9062306a36Sopenharmony_ci#define SCA_PERF_ATTR syscall_arg__scnprintf_perf_event_attr
91