Add a new CLOCK_DATA feature that stores reference times when
-k/--clockid option is specified.
It contains the clock id and its reference time together with wall clock
time taken at the 'same time', both values are in nanoseconds.
The format of data is as below:
  struct {
       u32 version;  /* version = 1 */
       u32 clockid;
       u64 wall_clock_ns;
       u64 clockid_time_ns;
  };
This clock reference times will be used in following changes to display
wall clock for perf events.
It's available only for recording with clockid specified, because it's
the only case where we can get reference time to wallclock time. It's
can't do that with perf clock yet.
Committer testing:
  $ perf record -h -k
   Usage: perf record [<options>] [<command>]
      or: perf record [<options>] -- <command> [<options>]
      -k, --clockid <clockid>
                            clockid to use for events, see clock_gettime()
  $ perf record -k monotonic sleep 1
  [ perf record: Woken up 1 times to write data ]
  [ perf record: Captured and wrote 0.017 MB perf.data (8 samples) ]
  $ perf report --header-only | grep clockid -A1
  # event : name = cycles:u, , id = { 88815, 88816, 88817, 88818, 88819, 88820, 88821, 88822 }, size = 120, { sample_period, sample_freq } = 4000, sample_type = IP|TID|TIME|PERIOD, read_format = ID, disabled = 1, inherit = 1, exclude_kernel = 1, mmap = 1, comm = 1, freq = 1, enable_on_exec = 1, task = 1, precise_ip = 3, sample_id_all = 1, exclude_guest = 1, mmap2 = 1, comm_exec = 1, use_clockid = 1, ksymbol = 1, bpf_event = 1, clockid = 1
  # CPU_TOPOLOGY info available, use -I to display
  --
  # clockid frequency: 1000 MHz
  # cpu pmu capabilities: branches=32, max_precise=3, pmu_name=skylake
  # clockid: monotonic (1)
  # reference time: 2020-08-06 09:40:21.619290 = 1596717621.619290 (TOD) = 21931.077673635 (monotonic)
  $
Original-patch-by: David Ahern <dsahern@gmail.com>
Signed-off-by: Jiri Olsa <jolsa@kernel.org>
Tested-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Alexander Shishkin <alexander.shishkin@linux.intel.com>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: David Ahern <dsahern@gmail.com>
Cc: Geneviève Bastien <gbastien@versatic.net>
Cc: Ian Rogers <irogers@google.com>
Cc: Jeremie Galarneau <jgalar@efficios.com>
Cc: Michael Petlan <mpetlan@redhat.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Wang Nan <wangnan0@huawei.com>
Link: http://lore.kernel.org/lkml/20200805093444.314999-4-jolsa@kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
		
	
			
		
			
				
	
	
		
			167 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			167 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* SPDX-License-Identifier: GPL-2.0 */
 | |
| #ifndef __PERF_HEADER_H
 | |
| #define __PERF_HEADER_H
 | |
| 
 | |
| #include <linux/stddef.h>
 | |
| #include <linux/perf_event.h>
 | |
| #include <sys/types.h>
 | |
| #include <stdio.h> // FILE
 | |
| #include <stdbool.h>
 | |
| #include <linux/bitmap.h>
 | |
| #include <linux/types.h>
 | |
| #include "env.h"
 | |
| #include "pmu.h"
 | |
| 
 | |
| enum {
 | |
| 	HEADER_RESERVED		= 0,	/* always cleared */
 | |
| 	HEADER_FIRST_FEATURE	= 1,
 | |
| 	HEADER_TRACING_DATA	= 1,
 | |
| 	HEADER_BUILD_ID,
 | |
| 
 | |
| 	HEADER_HOSTNAME,
 | |
| 	HEADER_OSRELEASE,
 | |
| 	HEADER_VERSION,
 | |
| 	HEADER_ARCH,
 | |
| 	HEADER_NRCPUS,
 | |
| 	HEADER_CPUDESC,
 | |
| 	HEADER_CPUID,
 | |
| 	HEADER_TOTAL_MEM,
 | |
| 	HEADER_CMDLINE,
 | |
| 	HEADER_EVENT_DESC,
 | |
| 	HEADER_CPU_TOPOLOGY,
 | |
| 	HEADER_NUMA_TOPOLOGY,
 | |
| 	HEADER_BRANCH_STACK,
 | |
| 	HEADER_PMU_MAPPINGS,
 | |
| 	HEADER_GROUP_DESC,
 | |
| 	HEADER_AUXTRACE,
 | |
| 	HEADER_STAT,
 | |
| 	HEADER_CACHE,
 | |
| 	HEADER_SAMPLE_TIME,
 | |
| 	HEADER_MEM_TOPOLOGY,
 | |
| 	HEADER_CLOCKID,
 | |
| 	HEADER_DIR_FORMAT,
 | |
| 	HEADER_BPF_PROG_INFO,
 | |
| 	HEADER_BPF_BTF,
 | |
| 	HEADER_COMPRESSED,
 | |
| 	HEADER_CPU_PMU_CAPS,
 | |
| 	HEADER_CLOCK_DATA,
 | |
| 	HEADER_LAST_FEATURE,
 | |
| 	HEADER_FEAT_BITS	= 256,
 | |
| };
 | |
| 
 | |
| enum perf_header_version {
 | |
| 	PERF_HEADER_VERSION_1,
 | |
| 	PERF_HEADER_VERSION_2,
 | |
| };
 | |
| 
 | |
| struct perf_file_section {
 | |
| 	u64 offset;
 | |
| 	u64 size;
 | |
| };
 | |
| 
 | |
| struct perf_file_header {
 | |
| 	u64				magic;
 | |
| 	u64				size;
 | |
| 	u64				attr_size;
 | |
| 	struct perf_file_section	attrs;
 | |
| 	struct perf_file_section	data;
 | |
| 	/* event_types is ignored */
 | |
| 	struct perf_file_section	event_types;
 | |
| 	DECLARE_BITMAP(adds_features, HEADER_FEAT_BITS);
 | |
| };
 | |
| 
 | |
| struct perf_pipe_file_header {
 | |
| 	u64				magic;
 | |
| 	u64				size;
 | |
| };
 | |
| 
 | |
| struct perf_header;
 | |
| 
 | |
| int perf_file_header__read(struct perf_file_header *header,
 | |
| 			   struct perf_header *ph, int fd);
 | |
| 
 | |
| struct perf_header {
 | |
| 	enum perf_header_version	version;
 | |
| 	bool				needs_swap;
 | |
| 	u64				data_offset;
 | |
| 	u64				data_size;
 | |
| 	u64				feat_offset;
 | |
| 	DECLARE_BITMAP(adds_features, HEADER_FEAT_BITS);
 | |
| 	struct perf_env 	env;
 | |
| };
 | |
| 
 | |
| struct feat_fd {
 | |
| 	struct perf_header *ph;
 | |
| 	int		   fd;
 | |
| 	void		   *buf;	/* Either buf != NULL or fd >= 0 */
 | |
| 	ssize_t		   offset;
 | |
| 	size_t		   size;
 | |
| 	struct evsel	   *events;
 | |
| };
 | |
| 
 | |
| struct perf_header_feature_ops {
 | |
| 	int	   (*write)(struct feat_fd *ff, struct evlist *evlist);
 | |
| 	void	   (*print)(struct feat_fd *ff, FILE *fp);
 | |
| 	int	   (*process)(struct feat_fd *ff, void *data);
 | |
| 	const char *name;
 | |
| 	bool	   full_only;
 | |
| 	bool	   synthesize;
 | |
| };
 | |
| 
 | |
| struct evlist;
 | |
| struct perf_session;
 | |
| struct perf_tool;
 | |
| union perf_event;
 | |
| 
 | |
| int perf_session__read_header(struct perf_session *session);
 | |
| int perf_session__write_header(struct perf_session *session,
 | |
| 			       struct evlist *evlist,
 | |
| 			       int fd, bool at_exit);
 | |
| int perf_header__write_pipe(int fd);
 | |
| 
 | |
| void perf_header__set_feat(struct perf_header *header, int feat);
 | |
| void perf_header__clear_feat(struct perf_header *header, int feat);
 | |
| bool perf_header__has_feat(const struct perf_header *header, int feat);
 | |
| 
 | |
| int perf_header__set_cmdline(int argc, const char **argv);
 | |
| 
 | |
| int perf_header__process_sections(struct perf_header *header, int fd,
 | |
| 				  void *data,
 | |
| 				  int (*process)(struct perf_file_section *section,
 | |
| 				  struct perf_header *ph,
 | |
| 				  int feat, int fd, void *data));
 | |
| 
 | |
| int perf_header__fprintf_info(struct perf_session *s, FILE *fp, bool full);
 | |
| 
 | |
| int perf_event__process_feature(struct perf_session *session,
 | |
| 				union perf_event *event);
 | |
| int perf_event__process_attr(struct perf_tool *tool, union perf_event *event,
 | |
| 			     struct evlist **pevlist);
 | |
| int perf_event__process_event_update(struct perf_tool *tool,
 | |
| 				     union perf_event *event,
 | |
| 				     struct evlist **pevlist);
 | |
| size_t perf_event__fprintf_event_update(union perf_event *event, FILE *fp);
 | |
| int perf_event__process_tracing_data(struct perf_session *session,
 | |
| 				     union perf_event *event);
 | |
| int perf_event__process_build_id(struct perf_session *session,
 | |
| 				 union perf_event *event);
 | |
| bool is_perf_magic(u64 magic);
 | |
| 
 | |
| #define NAME_ALIGN 64
 | |
| 
 | |
| struct feat_fd;
 | |
| 
 | |
| int do_write(struct feat_fd *fd, const void *buf, size_t size);
 | |
| 
 | |
| int write_padded(struct feat_fd *fd, const void *bf,
 | |
| 		 size_t count, size_t count_aligned);
 | |
| 
 | |
| /*
 | |
|  * arch specific callback
 | |
|  */
 | |
| int get_cpuid(char *buffer, size_t sz);
 | |
| 
 | |
| char *get_cpuid_str(struct perf_pmu *pmu __maybe_unused);
 | |
| int strcmp_cpuid_str(const char *s1, const char *s2);
 | |
| #endif /* __PERF_HEADER_H */
 |