xref: /freebsd/lib/libpmc/pmclog.h (revision cfec995c87f39e59c80554b85625b4aaa8ddf8db)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright (c) 2005-2007 Joseph Koshy
5  * Copyright (c) 2007 The FreeBSD Foundation
6  * All rights reserved.
7  *
8  * Portions of this software were developed by A. Joseph Koshy under
9  * sponsorship from the FreeBSD Foundation and Google, Inc.
10  *
11  * Redistribution and use in source and binary forms, with or without
12  * modification, are permitted provided that the following conditions
13  * are met:
14  * 1. Redistributions of source code must retain the above copyright
15  *    notice, this list of conditions and the following disclaimer.
16  * 2. Redistributions in binary form must reproduce the above copyright
17  *    notice, this list of conditions and the following disclaimer in the
18  *    documentation and/or other materials provided with the distribution.
19  *
20  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
21  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
24  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30  * SUCH DAMAGE.
31  */
32 
33 #ifndef	_PMCLOG_H_
34 #define	_PMCLOG_H_
35 
36 #include <sys/cdefs.h>
37 #include <sys/pmclog.h>
38 
39 enum pmclog_state {
40 	PMCLOG_OK,
41 	PMCLOG_EOF,
42 	PMCLOG_REQUIRE_DATA,
43 	PMCLOG_ERROR
44 };
45 
46 struct pmclog_ev_callchain {
47 	uint32_t	pl_pid;
48 	uint32_t	pl_tid;
49 	uint32_t	pl_pmcid;
50 	uint32_t	pl_cpuflags;
51 	uint32_t	pl_cpuflags2;
52 	uint32_t	pl_npc;
53 	uintfptr_t	pl_pc[PMC_CALLCHAIN_DEPTH_MAX];
54 };
55 
56 struct pmclog_ev_dropnotify {
57 };
58 
59 struct pmclog_ev_closelog {
60 };
61 
62 struct pmclog_ev_initialize {
63 	uint32_t	pl_version;
64 	uint32_t	pl_arch;
65 	uint64_t	pl_tsc_freq;
66 	struct timespec pl_ts;
67 	char		pl_cpuid[PATH_MAX];
68 };
69 
70 struct pmclog_ev_map_in {
71 	pid_t		pl_pid;
72 	uintfptr_t	pl_start;
73 	char		pl_pathname[PATH_MAX];
74 };
75 
76 struct pmclog_ev_map_out {
77 	pid_t		pl_pid;
78 	uintfptr_t	pl_start;
79 	uintfptr_t	pl_end;
80 };
81 
82 struct pmclog_ev_pcsample {
83 	uintfptr_t	pl_pc;
84 	pid_t		pl_pid;
85 	pid_t		pl_tid;
86 	pmc_id_t	pl_pmcid;
87 	uint32_t	pl_flags;
88 	uint32_t	pl_usermode;
89 };
90 
91 struct pmclog_ev_pmcallocate {
92 	const char *	pl_evname;
93 	uint64_t	pl_rate;
94 	uint32_t	pl_event;
95 	uint32_t	pl_flags;
96 	pmc_id_t	pl_pmcid;
97 };
98 
99 struct pmclog_ev_pmcallocatedyn {
100 	char 		pl_evname[PMC_NAME_MAX];
101 	uint32_t	pl_event;
102 	uint32_t	pl_flags;
103 	pmc_id_t	pl_pmcid;
104 };
105 
106 struct pmclog_ev_pmcattach {
107 	pmc_id_t	pl_pmcid;
108 	pid_t		pl_pid;
109 	char		pl_pathname[PATH_MAX];
110 };
111 
112 struct pmclog_ev_pmcdetach {
113 	pmc_id_t	pl_pmcid;
114 	pid_t		pl_pid;
115 };
116 
117 struct pmclog_ev_proccsw {
118 	pid_t		pl_pid;
119 	pid_t		pl_tid;
120 	pmc_id_t	pl_pmcid;
121 	pmc_value_t	pl_value;
122 };
123 
124 struct pmclog_ev_proccreate {
125 	pid_t		pl_pid;
126 	uint32_t	pl_flags;
127 	char		pl_pcomm[MAXCOMLEN+1];
128 };
129 
130 struct pmclog_ev_procexec {
131 	pid_t		pl_pid;
132 	pmc_id_t	pl_pmcid;
133 	uintptr_t	pl_baseaddr;
134 	uintptr_t	pl_dynaddr;
135 	char		pl_pathname[PATH_MAX];
136 };
137 
138 struct pmclog_ev_procexit {
139 	uint32_t	pl_pid;
140 	pmc_id_t	pl_pmcid;
141 	pmc_value_t	pl_value;
142 };
143 
144 struct pmclog_ev_procfork {
145 	pid_t		pl_oldpid;
146 	pid_t		pl_newpid;
147 };
148 
149 struct pmclog_ev_sysexit {
150 	pid_t		pl_pid;
151 };
152 
153 struct pmclog_ev_threadcreate {
154 	pid_t		pl_tid;
155 	pid_t		pl_pid;
156 	uint32_t	pl_flags;
157 	char		pl_tdname[MAXCOMLEN+1];
158 };
159 
160 struct pmclog_ev_threadexit {
161 	pid_t		pl_tid;
162 };
163 
164 struct pmclog_ev_userdata {
165 	uint32_t	pl_userdata;
166 };
167 
168 struct pmclog_ev {
169 	enum pmclog_state pl_state;	/* state after 'get_event()' */
170 	off_t		  pl_offset;	/* byte offset in stream */
171 	size_t		  pl_count;	/* count of records so far */
172 	union {
173 		uint64_t	  pl_tsc;	/* TSC timestamp */
174 		struct timespec pl_ts;		/* log entry timestamp (legacy) */
175 	};
176 	enum pmclog_type  pl_type;	/* type of log entry */
177 	void		 *pl_data;
178 	int		  pl_len;
179 	union { 			/* log entry data */
180 		struct pmclog_ev_callchain	pl_cc;
181 		struct pmclog_ev_closelog	pl_cl;
182 		struct pmclog_ev_dropnotify	pl_dn;
183 		struct pmclog_ev_initialize	pl_i;
184 		struct pmclog_ev_map_in		pl_mi;
185 		struct pmclog_ev_map_out	pl_mo;
186 		struct pmclog_ev_pmcallocate	pl_a;
187 		struct pmclog_ev_pmcallocatedyn	pl_ad;
188 		struct pmclog_ev_pmcattach	pl_t;
189 		struct pmclog_ev_pmcdetach	pl_d;
190 		struct pmclog_ev_proccsw	pl_c;
191 		struct pmclog_ev_proccreate	pl_pc;
192 		struct pmclog_ev_procexec	pl_x;
193 		struct pmclog_ev_procexit	pl_e;
194 		struct pmclog_ev_procfork	pl_f;
195 		struct pmclog_ev_sysexit	pl_se;
196 		struct pmclog_ev_threadcreate	pl_tc;
197 		struct pmclog_ev_threadexit	pl_te;
198 		struct pmclog_ev_userdata	pl_u;
199 	} pl_u;
200 };
201 
202 enum pmclog_parser_state {
203 	PL_STATE_NEW_RECORD,		/* in-between records */
204 	PL_STATE_EXPECTING_HEADER,	/* header being read */
205 	PL_STATE_PARTIAL_RECORD,	/* header present but not the record */
206 	PL_STATE_ERROR			/* parsing error encountered */
207 };
208 
209 struct pmclog_parse_state {
210 	enum pmclog_parser_state ps_state;
211 	enum pmc_cputype	ps_arch;	/* log file architecture */
212 	uint32_t		ps_version;	/* hwpmc version */
213 	int			ps_initialized;	/* whether initialized */
214 	int			ps_count;	/* count of records processed */
215 	off_t			ps_offset;	/* stream byte offset */
216 	union pmclog_entry	ps_saved;	/* saved partial log entry */
217 	int			ps_svcount;	/* #bytes saved */
218 	int			ps_fd;		/* active fd or -1 */
219 	char			*ps_buffer;	/* scratch buffer if fd != -1 */
220 	char			*ps_data;	/* current parse pointer */
221 	char			*ps_cpuid;	/* log cpuid */
222 	size_t			ps_len;		/* length of buffered data */
223 };
224 
225 #define	PMCLOG_FD_NONE				(-1)
226 
227 __BEGIN_DECLS
228 void	*pmclog_open(int _fd);
229 int	pmclog_feed(void *_cookie, char *_data, int _len);
230 int	pmclog_read(void *_cookie, struct pmclog_ev *_ev);
231 void	pmclog_close(void *_cookie);
232 __END_DECLS
233 
234 #endif
235