[PATCH 07/11] perf tools: Add perf_read_tsc_conv_for_clockid()

From: Adrian Hunter
Date: Wed Feb 09 2022 - 03:53:01 EST


Add a function to read TSC conversion information for a particular clock
ID. It will be used in a subsequent patch.

Signed-off-by: Adrian Hunter <adrian.hunter@xxxxxxxxx>
---
tools/perf/util/tsc.c | 56 +++++++++++++++++++++++++++++++++++++++++++
tools/perf/util/tsc.h | 1 +
2 files changed, 57 insertions(+)

diff --git a/tools/perf/util/tsc.c b/tools/perf/util/tsc.c
index f19791d46e99..175a6e43ad94 100644
--- a/tools/perf/util/tsc.c
+++ b/tools/perf/util/tsc.c
@@ -3,6 +3,8 @@
#include <inttypes.h>
#include <string.h>

+#include <sys/mman.h>
+
#include <linux/compiler.h>
#include <linux/perf_event.h>
#include <linux/stddef.h>
@@ -14,6 +16,9 @@
#include "synthetic-events.h"
#include "debug.h"
#include "tsc.h"
+#include "cpumap.h"
+#include "perf-sys.h"
+#include <internal/lib.h> /* page_size */

u64 perf_time_to_tsc(u64 ns, struct perf_tsc_conversion *tc)
{
@@ -71,6 +76,57 @@ int perf_read_tsc_conversion(const struct perf_event_mmap_page *pc,
return 0;
}

+static int perf_read_tsc_conv_attr_cpu(struct perf_event_attr *attr, int cpu,
+ struct perf_tsc_conversion *tc)
+{
+ size_t len = 2 * page_size;
+ int fd, err = -EINVAL;
+ void *addr;
+
+ fd = sys_perf_event_open(attr, 0, cpu, -1, 0);
+ if (fd == -1)
+ return -EINVAL;
+
+ addr = mmap(NULL, len, PROT_READ, MAP_SHARED, fd, 0);
+ if (addr == MAP_FAILED)
+ goto out_close;
+
+ err = perf_read_tsc_conversion(addr, tc);
+
+ munmap(addr, len);
+out_close:
+ close(fd);
+ return err;
+}
+
+static int find_a_cpu(void)
+{
+ struct perf_cpu_map *cpus;
+ int cpu;
+
+ cpus = perf_cpu_map__new(NULL);
+ if (!cpus)
+ return 0;
+ cpu = cpus->map[0];
+ perf_cpu_map__put(cpus);
+ return cpu;
+}
+
+int perf_read_tsc_conv_for_clockid(s32 clockid, struct perf_tsc_conversion *tc)
+{
+ struct perf_event_attr attr = {
+ .size = sizeof(attr),
+ .type = PERF_TYPE_SOFTWARE,
+ .config = PERF_COUNT_SW_DUMMY,
+ .exclude_kernel = 1,
+ .use_clockid = 1,
+ .clockid = clockid,
+ };
+ int cpu = find_a_cpu();
+
+ return perf_read_tsc_conv_attr_cpu(&attr, cpu, tc);
+}
+
int perf_event__synth_time_conv(const struct perf_event_mmap_page *pc,
struct perf_tool *tool,
perf_event__handler_t process,
diff --git a/tools/perf/util/tsc.h b/tools/perf/util/tsc.h
index 7d83a31732a7..ba9a52a9d70f 100644
--- a/tools/perf/util/tsc.h
+++ b/tools/perf/util/tsc.h
@@ -21,6 +21,7 @@ struct perf_event_mmap_page;

int perf_read_tsc_conversion(const struct perf_event_mmap_page *pc,
struct perf_tsc_conversion *tc);
+int perf_read_tsc_conv_for_clockid(s32 clockid, struct perf_tsc_conversion *tc);

u64 perf_time_to_tsc(u64 ns, struct perf_tsc_conversion *tc);
u64 tsc_to_perf_time(u64 cyc, struct perf_tsc_conversion *tc);
--
2.25.1