xref: /xnu-12377.1.9/san/tools/ksancov.h (revision f6217f891ac0bb64f3d375211650a4c1ff8ca1ea)
1*f6217f89SApple OSS Distributions /*
2*f6217f89SApple OSS Distributions  * Copyright (c) 2019 Apple Inc. All rights reserved.
3*f6217f89SApple OSS Distributions  *
4*f6217f89SApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*f6217f89SApple OSS Distributions  *
6*f6217f89SApple OSS Distributions  * This file contains Original Code and/or Modifications of Original Code
7*f6217f89SApple OSS Distributions  * as defined in and that are subject to the Apple Public Source License
8*f6217f89SApple OSS Distributions  * Version 2.0 (the 'License'). You may not use this file except in
9*f6217f89SApple OSS Distributions  * compliance with the License. The rights granted to you under the License
10*f6217f89SApple OSS Distributions  * may not be used to create, or enable the creation or redistribution of,
11*f6217f89SApple OSS Distributions  * unlawful or unlicensed copies of an Apple operating system, or to
12*f6217f89SApple OSS Distributions  * circumvent, violate, or enable the circumvention or violation of, any
13*f6217f89SApple OSS Distributions  * terms of an Apple operating system software license agreement.
14*f6217f89SApple OSS Distributions  *
15*f6217f89SApple OSS Distributions  * Please obtain a copy of the License at
16*f6217f89SApple OSS Distributions  * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*f6217f89SApple OSS Distributions  *
18*f6217f89SApple OSS Distributions  * The Original Code and all software distributed under the License are
19*f6217f89SApple OSS Distributions  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*f6217f89SApple OSS Distributions  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*f6217f89SApple OSS Distributions  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*f6217f89SApple OSS Distributions  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*f6217f89SApple OSS Distributions  * Please see the License for the specific language governing rights and
24*f6217f89SApple OSS Distributions  * limitations under the License.
25*f6217f89SApple OSS Distributions  *
26*f6217f89SApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*f6217f89SApple OSS Distributions  */
28*f6217f89SApple OSS Distributions 
29*f6217f89SApple OSS Distributions #ifndef _KSANCOV_H_
30*f6217f89SApple OSS Distributions #define _KSANCOV_H_
31*f6217f89SApple OSS Distributions 
32*f6217f89SApple OSS Distributions #include <stdint.h>
33*f6217f89SApple OSS Distributions #include <stdatomic.h>
34*f6217f89SApple OSS Distributions #include <sys/ioccom.h>
35*f6217f89SApple OSS Distributions #include <sys/ioctl.h>
36*f6217f89SApple OSS Distributions #include <strings.h>
37*f6217f89SApple OSS Distributions #include <assert.h>
38*f6217f89SApple OSS Distributions #include <unistd.h>
39*f6217f89SApple OSS Distributions 
40*f6217f89SApple OSS Distributions #define KSANCOV_DEVNODE "ksancov"
41*f6217f89SApple OSS Distributions #define KSANCOV_PATH "/dev/" KSANCOV_DEVNODE
42*f6217f89SApple OSS Distributions 
43*f6217f89SApple OSS Distributions /* Set mode */
44*f6217f89SApple OSS Distributions #define KSANCOV_IOC_TRACE            _IOW('K', 1, size_t) /* number of pcs */
45*f6217f89SApple OSS Distributions #define KSANCOV_IOC_COUNTERS         _IO('K', 2)
46*f6217f89SApple OSS Distributions #define KSANCOV_IOC_STKSIZE          _IOW('K', 3, size_t) /* number of pcs */
47*f6217f89SApple OSS Distributions 
48*f6217f89SApple OSS Distributions /* Establish a shared mapping of the coverage buffer. */
49*f6217f89SApple OSS Distributions #define KSANCOV_IOC_MAP              _IOWR('K', 8, struct ksancov_buf_desc)
50*f6217f89SApple OSS Distributions 
51*f6217f89SApple OSS Distributions /* Establish a shared mapping of the edge address buffer. */
52*f6217f89SApple OSS Distributions #define KSANCOV_IOC_MAP_EDGEMAP      _IOWR('K', 9, struct ksancov_buf_desc)
53*f6217f89SApple OSS Distributions 
54*f6217f89SApple OSS Distributions /* Log the current thread */
55*f6217f89SApple OSS Distributions #define KSANCOV_IOC_START            _IOW('K', 10, uintptr_t)
56*f6217f89SApple OSS Distributions #define KSANCOV_IOC_NEDGES           _IOR('K', 50, size_t)
57*f6217f89SApple OSS Distributions #define KSANCOV_IOC_TESTPANIC        _IOW('K', 20, uint64_t)
58*f6217f89SApple OSS Distributions 
59*f6217f89SApple OSS Distributions /* Operations related to on-demand instrumentation */
60*f6217f89SApple OSS Distributions #define KSANCOV_IOC_ON_DEMAND        _IOWR('K', 60, struct ksancov_on_demand_msg)
61*f6217f89SApple OSS Distributions 
62*f6217f89SApple OSS Distributions /* Set comparison log mode */
63*f6217f89SApple OSS Distributions #define KSANCOV_IOC_CMPS_TRACE       _IOW('K', 70, size_t) /* number of cmps */
64*f6217f89SApple OSS Distributions #define KSANCOV_IOC_CMPS_TRACE_FUNC  _IOW('K', 71, size_t) /* number of cmps */
65*f6217f89SApple OSS Distributions 
66*f6217f89SApple OSS Distributions /* Establish a shared mapping of the comparisons buffer. */
67*f6217f89SApple OSS Distributions #define KSANCOV_IOC_CMPS_MAP         _IOWR('K', 90, struct ksancov_buf_desc)
68*f6217f89SApple OSS Distributions 
69*f6217f89SApple OSS Distributions /*
70*f6217f89SApple OSS Distributions  * shared kernel-user mapping
71*f6217f89SApple OSS Distributions  */
72*f6217f89SApple OSS Distributions 
73*f6217f89SApple OSS Distributions #define KSANCOV_MAX_EDGES         (1 << 24)
74*f6217f89SApple OSS Distributions #define KSANCOV_MAX_HITS          UINT8_MAX
75*f6217f89SApple OSS Distributions #define KSANCOV_TRACE_MAGIC       (uint32_t)0x5AD17F5BU
76*f6217f89SApple OSS Distributions #define KSANCOV_COUNTERS_MAGIC    (uint32_t)0x5AD27F6BU
77*f6217f89SApple OSS Distributions #define KSANCOV_EDGEMAP_MAGIC     (uint32_t)0x5AD37F7BU
78*f6217f89SApple OSS Distributions #define KSANCOV_STKSIZE_MAGIC     (uint32_t)0x5AD47F8BU
79*f6217f89SApple OSS Distributions #define KSANCOV_CMPS_TRACE_MAGIC  (uint32_t)0x5AD47F9BU
80*f6217f89SApple OSS Distributions 
81*f6217f89SApple OSS Distributions /*
82*f6217f89SApple OSS Distributions  * ioctl
83*f6217f89SApple OSS Distributions  */
84*f6217f89SApple OSS Distributions 
85*f6217f89SApple OSS Distributions struct ksancov_buf_desc {
86*f6217f89SApple OSS Distributions 	uintptr_t ptr;  /* ptr to shared buffer [out] */
87*f6217f89SApple OSS Distributions 	size_t sz;      /* size of shared buffer [out] */
88*f6217f89SApple OSS Distributions };
89*f6217f89SApple OSS Distributions 
90*f6217f89SApple OSS Distributions /*
91*f6217f89SApple OSS Distributions  * Supported coverage modes.
92*f6217f89SApple OSS Distributions  */
93*f6217f89SApple OSS Distributions typedef enum {
94*f6217f89SApple OSS Distributions 	KS_MODE_NONE,
95*f6217f89SApple OSS Distributions 	KS_MODE_TRACE,
96*f6217f89SApple OSS Distributions 	KS_MODE_COUNTERS,
97*f6217f89SApple OSS Distributions 	KS_MODE_STKSIZE,
98*f6217f89SApple OSS Distributions 	KS_MODE_MAX
99*f6217f89SApple OSS Distributions } ksancov_mode_t;
100*f6217f89SApple OSS Distributions 
101*f6217f89SApple OSS Distributions /*
102*f6217f89SApple OSS Distributions  * A header that is always present in every ksancov mode shared memory structure.
103*f6217f89SApple OSS Distributions  */
104*f6217f89SApple OSS Distributions typedef struct ksancov_header {
105*f6217f89SApple OSS Distributions 	uint32_t         kh_magic;
106*f6217f89SApple OSS Distributions 	_Atomic uint32_t kh_enabled;
107*f6217f89SApple OSS Distributions } ksancov_header_t;
108*f6217f89SApple OSS Distributions 
109*f6217f89SApple OSS Distributions /*
110*f6217f89SApple OSS Distributions  * TRACE mode data structure.
111*f6217f89SApple OSS Distributions  */
112*f6217f89SApple OSS Distributions 
113*f6217f89SApple OSS Distributions /*
114*f6217f89SApple OSS Distributions  * All trace based tools share this structure.
115*f6217f89SApple OSS Distributions  */
116*f6217f89SApple OSS Distributions typedef struct ksancov_trace {
117*f6217f89SApple OSS Distributions 	ksancov_header_t kt_hdr;         /* header (must be always first) */
118*f6217f89SApple OSS Distributions 	uint32_t         kt_maxent;      /* Maximum entries in this shared buffer. */
119*f6217f89SApple OSS Distributions 	_Atomic uint32_t kt_head;        /* Pointer to the first unused element. */
120*f6217f89SApple OSS Distributions 	uint64_t         kt_entries[];   /* Trace entries in this buffer. */
121*f6217f89SApple OSS Distributions } ksancov_trace_t;
122*f6217f89SApple OSS Distributions 
123*f6217f89SApple OSS Distributions /* PC tracing only records PCs. */
124*f6217f89SApple OSS Distributions typedef uintptr_t ksancov_trace_pc_ent_t;
125*f6217f89SApple OSS Distributions 
126*f6217f89SApple OSS Distributions /* STKSIZE tracing records PCs and stack size. */
127*f6217f89SApple OSS Distributions typedef struct ksancov_trace_stksize_entry {
128*f6217f89SApple OSS Distributions 	uintptr_t pc;                      /* PC */
129*f6217f89SApple OSS Distributions 	uint32_t  stksize;                 /* associated stack size */
130*f6217f89SApple OSS Distributions } ksancov_trace_stksize_ent_t;
131*f6217f89SApple OSS Distributions 
132*f6217f89SApple OSS Distributions /*
133*f6217f89SApple OSS Distributions  * COUNTERS mode data structure.
134*f6217f89SApple OSS Distributions  */
135*f6217f89SApple OSS Distributions typedef struct ksancov_counters {
136*f6217f89SApple OSS Distributions 	ksancov_header_t kc_hdr;
137*f6217f89SApple OSS Distributions 	uint32_t         kc_nedges;       /* total number of edges */
138*f6217f89SApple OSS Distributions 	uint8_t          kc_hits[];       /* hits on each edge (8bit saturating) */
139*f6217f89SApple OSS Distributions } ksancov_counters_t;
140*f6217f89SApple OSS Distributions 
141*f6217f89SApple OSS Distributions /*
142*f6217f89SApple OSS Distributions  * Edge to PC mapping.
143*f6217f89SApple OSS Distributions  */
144*f6217f89SApple OSS Distributions typedef struct ksancov_edgemap {
145*f6217f89SApple OSS Distributions 	uint32_t  ke_magic;
146*f6217f89SApple OSS Distributions 	uint32_t  ke_nedges;
147*f6217f89SApple OSS Distributions 	uintptr_t ke_addrs[];             /* address of each edge relative to 'offset' */
148*f6217f89SApple OSS Distributions } ksancov_edgemap_t;
149*f6217f89SApple OSS Distributions 
150*f6217f89SApple OSS Distributions /*
151*f6217f89SApple OSS Distributions  * Supported comparison logging modes.
152*f6217f89SApple OSS Distributions  */
153*f6217f89SApple OSS Distributions typedef enum {
154*f6217f89SApple OSS Distributions 	KS_CMPS_MODE_NONE,
155*f6217f89SApple OSS Distributions 	KS_CMPS_MODE_TRACE,
156*f6217f89SApple OSS Distributions 	KS_CMPS_MODE_TRACE_FUNC,
157*f6217f89SApple OSS Distributions 	KS_CMPS_MODE_MAX
158*f6217f89SApple OSS Distributions } ksancov_cmps_mode_t;
159*f6217f89SApple OSS Distributions 
160*f6217f89SApple OSS Distributions /* Comparison type values */
161*f6217f89SApple OSS Distributions enum {
162*f6217f89SApple OSS Distributions 	KCOV_CMP_CONST = 1,
163*f6217f89SApple OSS Distributions 	KCOV_CMP_SIZE1 = 0,
164*f6217f89SApple OSS Distributions 	KCOV_CMP_SIZE2 = 2,
165*f6217f89SApple OSS Distributions 	KCOV_CMP_SIZE4 = 4,
166*f6217f89SApple OSS Distributions 	KCOV_CMP_SIZE8 = 6,
167*f6217f89SApple OSS Distributions 
168*f6217f89SApple OSS Distributions 	KCOV_CMP_FUNC_MEMCMP = 32,
169*f6217f89SApple OSS Distributions 	KCOV_CMP_FUNC_STRCMP = 34,
170*f6217f89SApple OSS Distributions 	KCOV_CMP_FUNC_STRNCMP = 36,
171*f6217f89SApple OSS Distributions 	KCOV_CMP_FUNC_STRBUFCMP = 38,
172*f6217f89SApple OSS Distributions };
173*f6217f89SApple OSS Distributions 
174*f6217f89SApple OSS Distributions #define KCOV_CMP_SIZE_MASK 0xfffffff6
175*f6217f89SApple OSS Distributions #define KCOV_CMP_IS_FUNC(type) (type >= KCOV_CMP_FUNC_MEMCMP)
176*f6217f89SApple OSS Distributions 
177*f6217f89SApple OSS Distributions /* CMPS TRACE mode tracks comparison values */
178*f6217f89SApple OSS Distributions typedef struct __attribute__((__packed__)) ksancov_cmps_trace_entry {
179*f6217f89SApple OSS Distributions 	uint64_t pc;
180*f6217f89SApple OSS Distributions 	uint32_t type;
181*f6217f89SApple OSS Distributions 	uint16_t len1_func;
182*f6217f89SApple OSS Distributions 	uint16_t len2_func;
183*f6217f89SApple OSS Distributions 	union {
184*f6217f89SApple OSS Distributions 		uint64_t args[2];              /* cmp instruction arguments */
185*f6217f89SApple OSS Distributions 		uint8_t args_func[0];          /* cmp function arguments (variadic) */
186*f6217f89SApple OSS Distributions 	};
187*f6217f89SApple OSS Distributions } ksancov_cmps_trace_ent_t;
188*f6217f89SApple OSS Distributions 
189*f6217f89SApple OSS Distributions /* Calculate the total space that a ksancov_cmps_trace_ent_t tracing a function takes */
190*f6217f89SApple OSS Distributions static inline size_t
ksancov_cmps_trace_func_space(size_t len1_func,size_t len2_func)191*f6217f89SApple OSS Distributions ksancov_cmps_trace_func_space(size_t len1_func, size_t len2_func)
192*f6217f89SApple OSS Distributions {
193*f6217f89SApple OSS Distributions 	static_assert(sizeof(ksancov_cmps_trace_ent_t) == sizeof(uint64_t) * 3 + sizeof(uint32_t) + sizeof(uint16_t) * 2, "ksancov_cmps_trace_ent_t invalid size");
194*f6217f89SApple OSS Distributions 
195*f6217f89SApple OSS Distributions 	size_t size = sizeof(uint64_t) + sizeof(uint32_t) + sizeof(uint16_t) * 2; // header
196*f6217f89SApple OSS Distributions 	size += len1_func + len2_func;
197*f6217f89SApple OSS Distributions 	size_t rem = size % sizeof(ksancov_cmps_trace_ent_t);
198*f6217f89SApple OSS Distributions 	if (rem == 0) {
199*f6217f89SApple OSS Distributions 		return size;
200*f6217f89SApple OSS Distributions 	}
201*f6217f89SApple OSS Distributions 	return size + sizeof(ksancov_cmps_trace_ent_t) - rem;
202*f6217f89SApple OSS Distributions }
203*f6217f89SApple OSS Distributions 
204*f6217f89SApple OSS Distributions static inline uint8_t *
ksancov_cmps_trace_func_arg1(ksancov_cmps_trace_ent_t * entry)205*f6217f89SApple OSS Distributions ksancov_cmps_trace_func_arg1(ksancov_cmps_trace_ent_t *entry)
206*f6217f89SApple OSS Distributions {
207*f6217f89SApple OSS Distributions 	return entry->args_func;
208*f6217f89SApple OSS Distributions }
209*f6217f89SApple OSS Distributions 
210*f6217f89SApple OSS Distributions static inline uint8_t *
ksancov_cmps_trace_func_arg2(ksancov_cmps_trace_ent_t * entry)211*f6217f89SApple OSS Distributions ksancov_cmps_trace_func_arg2(ksancov_cmps_trace_ent_t *entry)
212*f6217f89SApple OSS Distributions {
213*f6217f89SApple OSS Distributions 	uint8_t* func_args = entry->args_func;
214*f6217f89SApple OSS Distributions 	return &func_args[entry->len1_func];
215*f6217f89SApple OSS Distributions }
216*f6217f89SApple OSS Distributions 
217*f6217f89SApple OSS Distributions /*
218*f6217f89SApple OSS Distributions  * On-demand related functionalities
219*f6217f89SApple OSS Distributions  */
220*f6217f89SApple OSS Distributions typedef enum {
221*f6217f89SApple OSS Distributions 	KS_OD_GET_GATE = 1,
222*f6217f89SApple OSS Distributions 	KS_OD_SET_GATE = 2,
223*f6217f89SApple OSS Distributions 	KS_OD_GET_RANGE = 3,
224*f6217f89SApple OSS Distributions } ksancov_on_demand_operation_t;
225*f6217f89SApple OSS Distributions 
226*f6217f89SApple OSS Distributions struct ksancov_on_demand_msg {
227*f6217f89SApple OSS Distributions 	char bundle[/*KMOD_MAX_NAME*/ 64];
228*f6217f89SApple OSS Distributions 	ksancov_on_demand_operation_t operation;
229*f6217f89SApple OSS Distributions 	union {
230*f6217f89SApple OSS Distributions 		uint64_t gate;
231*f6217f89SApple OSS Distributions 		struct {
232*f6217f89SApple OSS Distributions 			uint32_t start;
233*f6217f89SApple OSS Distributions 			uint32_t stop;
234*f6217f89SApple OSS Distributions 		} range;
235*f6217f89SApple OSS Distributions 	};
236*f6217f89SApple OSS Distributions };
237*f6217f89SApple OSS Distributions 
238*f6217f89SApple OSS Distributions /*
239*f6217f89SApple OSS Distributions  * ksancov userspace API
240*f6217f89SApple OSS Distributions  *
241*f6217f89SApple OSS Distributions  * Usage:
242*f6217f89SApple OSS Distributions  * 1) open the ksancov device
243*f6217f89SApple OSS Distributions  * 2) set the coverage mode
244*f6217f89SApple OSS Distributions  * 3) map the coverage buffer
245*f6217f89SApple OSS Distributions  * 4) start the trace on a thread
246*f6217f89SApple OSS Distributions  * 5) flip the enable bit
247*f6217f89SApple OSS Distributions  */
248*f6217f89SApple OSS Distributions 
249*f6217f89SApple OSS Distributions static inline int
ksancov_open(void)250*f6217f89SApple OSS Distributions ksancov_open(void)
251*f6217f89SApple OSS Distributions {
252*f6217f89SApple OSS Distributions 	return open(KSANCOV_PATH, 0);
253*f6217f89SApple OSS Distributions }
254*f6217f89SApple OSS Distributions 
255*f6217f89SApple OSS Distributions static inline int
ksancov_map(int fd,uintptr_t * buf,size_t * sz)256*f6217f89SApple OSS Distributions ksancov_map(int fd, uintptr_t *buf, size_t *sz)
257*f6217f89SApple OSS Distributions {
258*f6217f89SApple OSS Distributions 	int ret;
259*f6217f89SApple OSS Distributions 	struct ksancov_buf_desc mc = {0};
260*f6217f89SApple OSS Distributions 
261*f6217f89SApple OSS Distributions 	assert(buf != NULL);
262*f6217f89SApple OSS Distributions 
263*f6217f89SApple OSS Distributions 	ret = ioctl(fd, KSANCOV_IOC_MAP, &mc);
264*f6217f89SApple OSS Distributions 	if (ret == -1) {
265*f6217f89SApple OSS Distributions 		return errno;
266*f6217f89SApple OSS Distributions 	}
267*f6217f89SApple OSS Distributions 
268*f6217f89SApple OSS Distributions 	*buf = mc.ptr;
269*f6217f89SApple OSS Distributions 	if (sz) {
270*f6217f89SApple OSS Distributions 		*sz = mc.sz;
271*f6217f89SApple OSS Distributions 	}
272*f6217f89SApple OSS Distributions 
273*f6217f89SApple OSS Distributions 	ksancov_header_t *hdr = (ksancov_header_t *)mc.ptr;
274*f6217f89SApple OSS Distributions 	assert(hdr->kh_magic == KSANCOV_TRACE_MAGIC ||
275*f6217f89SApple OSS Distributions 	    hdr->kh_magic == KSANCOV_COUNTERS_MAGIC ||
276*f6217f89SApple OSS Distributions 	    hdr->kh_magic == KSANCOV_STKSIZE_MAGIC);
277*f6217f89SApple OSS Distributions 
278*f6217f89SApple OSS Distributions 	return 0;
279*f6217f89SApple OSS Distributions }
280*f6217f89SApple OSS Distributions 
281*f6217f89SApple OSS Distributions static inline int
ksancov_map_edgemap(int fd,uintptr_t * buf,size_t * sz)282*f6217f89SApple OSS Distributions ksancov_map_edgemap(int fd, uintptr_t *buf, size_t *sz)
283*f6217f89SApple OSS Distributions {
284*f6217f89SApple OSS Distributions 	int ret;
285*f6217f89SApple OSS Distributions 	struct ksancov_buf_desc mc = {0};
286*f6217f89SApple OSS Distributions 
287*f6217f89SApple OSS Distributions 	assert(buf != NULL);
288*f6217f89SApple OSS Distributions 
289*f6217f89SApple OSS Distributions 	ret = ioctl(fd, KSANCOV_IOC_MAP_EDGEMAP, &mc);
290*f6217f89SApple OSS Distributions 	if (ret == -1) {
291*f6217f89SApple OSS Distributions 		return errno;
292*f6217f89SApple OSS Distributions 	}
293*f6217f89SApple OSS Distributions 
294*f6217f89SApple OSS Distributions 	*buf = mc.ptr;
295*f6217f89SApple OSS Distributions 	if (sz) {
296*f6217f89SApple OSS Distributions 		*sz = mc.sz;
297*f6217f89SApple OSS Distributions 	}
298*f6217f89SApple OSS Distributions 
299*f6217f89SApple OSS Distributions 	ksancov_edgemap_t *emap = (ksancov_edgemap_t *)mc.ptr;
300*f6217f89SApple OSS Distributions 	assert(emap->ke_magic == KSANCOV_EDGEMAP_MAGIC);
301*f6217f89SApple OSS Distributions 
302*f6217f89SApple OSS Distributions 	return 0;
303*f6217f89SApple OSS Distributions }
304*f6217f89SApple OSS Distributions 
305*f6217f89SApple OSS Distributions static inline size_t
ksancov_nedges(int fd)306*f6217f89SApple OSS Distributions ksancov_nedges(int fd)
307*f6217f89SApple OSS Distributions {
308*f6217f89SApple OSS Distributions 	size_t nedges;
309*f6217f89SApple OSS Distributions 	int ret = ioctl(fd, KSANCOV_IOC_NEDGES, &nedges);
310*f6217f89SApple OSS Distributions 	if (ret == -1) {
311*f6217f89SApple OSS Distributions 		return SIZE_MAX;
312*f6217f89SApple OSS Distributions 	}
313*f6217f89SApple OSS Distributions 	return nedges;
314*f6217f89SApple OSS Distributions }
315*f6217f89SApple OSS Distributions 
316*f6217f89SApple OSS Distributions static inline int
ksancov_mode_trace(int fd,size_t entries)317*f6217f89SApple OSS Distributions ksancov_mode_trace(int fd, size_t entries)
318*f6217f89SApple OSS Distributions {
319*f6217f89SApple OSS Distributions 	int ret;
320*f6217f89SApple OSS Distributions 	ret = ioctl(fd, KSANCOV_IOC_TRACE, &entries);
321*f6217f89SApple OSS Distributions 	if (ret == -1) {
322*f6217f89SApple OSS Distributions 		return errno;
323*f6217f89SApple OSS Distributions 	}
324*f6217f89SApple OSS Distributions 	return 0;
325*f6217f89SApple OSS Distributions }
326*f6217f89SApple OSS Distributions 
327*f6217f89SApple OSS Distributions static inline int
ksancov_mode_stksize(int fd,size_t entries)328*f6217f89SApple OSS Distributions ksancov_mode_stksize(int fd, size_t entries)
329*f6217f89SApple OSS Distributions {
330*f6217f89SApple OSS Distributions 	int ret;
331*f6217f89SApple OSS Distributions 	ret = ioctl(fd, KSANCOV_IOC_STKSIZE, &entries);
332*f6217f89SApple OSS Distributions 	if (ret == -1) {
333*f6217f89SApple OSS Distributions 		return errno;
334*f6217f89SApple OSS Distributions 	}
335*f6217f89SApple OSS Distributions 	return 0;
336*f6217f89SApple OSS Distributions }
337*f6217f89SApple OSS Distributions 
338*f6217f89SApple OSS Distributions static inline int
ksancov_mode_counters(int fd)339*f6217f89SApple OSS Distributions ksancov_mode_counters(int fd)
340*f6217f89SApple OSS Distributions {
341*f6217f89SApple OSS Distributions 	int ret;
342*f6217f89SApple OSS Distributions 	ret = ioctl(fd, KSANCOV_IOC_COUNTERS);
343*f6217f89SApple OSS Distributions 	if (ret == -1) {
344*f6217f89SApple OSS Distributions 		return errno;
345*f6217f89SApple OSS Distributions 	}
346*f6217f89SApple OSS Distributions 	return 0;
347*f6217f89SApple OSS Distributions }
348*f6217f89SApple OSS Distributions 
349*f6217f89SApple OSS Distributions static inline int
ksancov_thread_self(int fd)350*f6217f89SApple OSS Distributions ksancov_thread_self(int fd)
351*f6217f89SApple OSS Distributions {
352*f6217f89SApple OSS Distributions 	int ret;
353*f6217f89SApple OSS Distributions 	uintptr_t th = 0;
354*f6217f89SApple OSS Distributions 	ret = ioctl(fd, KSANCOV_IOC_START, &th);
355*f6217f89SApple OSS Distributions 	if (ret == -1) {
356*f6217f89SApple OSS Distributions 		return errno;
357*f6217f89SApple OSS Distributions 	}
358*f6217f89SApple OSS Distributions 	return 0;
359*f6217f89SApple OSS Distributions }
360*f6217f89SApple OSS Distributions 
361*f6217f89SApple OSS Distributions static inline int
ksancov_start(void * buf)362*f6217f89SApple OSS Distributions ksancov_start(void *buf)
363*f6217f89SApple OSS Distributions {
364*f6217f89SApple OSS Distributions 	ksancov_header_t *hdr = (ksancov_header_t *)buf;
365*f6217f89SApple OSS Distributions 	atomic_store_explicit(&hdr->kh_enabled, 1, memory_order_relaxed);
366*f6217f89SApple OSS Distributions 	return 0;
367*f6217f89SApple OSS Distributions }
368*f6217f89SApple OSS Distributions 
369*f6217f89SApple OSS Distributions static inline int
ksancov_stop(void * buf)370*f6217f89SApple OSS Distributions ksancov_stop(void *buf)
371*f6217f89SApple OSS Distributions {
372*f6217f89SApple OSS Distributions 	ksancov_header_t *hdr = (ksancov_header_t *)buf;
373*f6217f89SApple OSS Distributions 	atomic_store_explicit(&hdr->kh_enabled, 0, memory_order_relaxed);
374*f6217f89SApple OSS Distributions 	return 0;
375*f6217f89SApple OSS Distributions }
376*f6217f89SApple OSS Distributions 
377*f6217f89SApple OSS Distributions static inline int
ksancov_reset(void * buf)378*f6217f89SApple OSS Distributions ksancov_reset(void *buf)
379*f6217f89SApple OSS Distributions {
380*f6217f89SApple OSS Distributions 	ksancov_header_t *hdr = (ksancov_header_t *)buf;
381*f6217f89SApple OSS Distributions 	if (hdr->kh_magic == KSANCOV_TRACE_MAGIC || hdr->kh_magic == KSANCOV_STKSIZE_MAGIC || hdr->kh_magic == KSANCOV_CMPS_TRACE_MAGIC) {
382*f6217f89SApple OSS Distributions 		ksancov_trace_t *trace = (ksancov_trace_t *)buf;
383*f6217f89SApple OSS Distributions 		atomic_store_explicit(&trace->kt_head, 0, memory_order_relaxed);
384*f6217f89SApple OSS Distributions 	} else if (hdr->kh_magic == KSANCOV_COUNTERS_MAGIC) {
385*f6217f89SApple OSS Distributions 		ksancov_counters_t *counters = (ksancov_counters_t *)buf;
386*f6217f89SApple OSS Distributions 		bzero(counters->kc_hits, counters->kc_nedges);
387*f6217f89SApple OSS Distributions 	} else {
388*f6217f89SApple OSS Distributions 		return EINVAL;
389*f6217f89SApple OSS Distributions 	}
390*f6217f89SApple OSS Distributions 	return 0;
391*f6217f89SApple OSS Distributions }
392*f6217f89SApple OSS Distributions 
393*f6217f89SApple OSS Distributions static inline uintptr_t
ksancov_edge_addr(ksancov_edgemap_t * kemap,size_t idx)394*f6217f89SApple OSS Distributions ksancov_edge_addr(ksancov_edgemap_t *kemap, size_t idx)
395*f6217f89SApple OSS Distributions {
396*f6217f89SApple OSS Distributions 	assert(kemap);
397*f6217f89SApple OSS Distributions 	if (idx >= kemap->ke_nedges) {
398*f6217f89SApple OSS Distributions 		return 0;
399*f6217f89SApple OSS Distributions 	}
400*f6217f89SApple OSS Distributions 	return kemap->ke_addrs[idx];
401*f6217f89SApple OSS Distributions }
402*f6217f89SApple OSS Distributions 
403*f6217f89SApple OSS Distributions static inline size_t
ksancov_trace_max_ent(ksancov_trace_t * trace)404*f6217f89SApple OSS Distributions ksancov_trace_max_ent(ksancov_trace_t *trace)
405*f6217f89SApple OSS Distributions {
406*f6217f89SApple OSS Distributions 	assert(trace);
407*f6217f89SApple OSS Distributions 	return trace->kt_maxent;
408*f6217f89SApple OSS Distributions }
409*f6217f89SApple OSS Distributions 
410*f6217f89SApple OSS Distributions static inline size_t
ksancov_trace_head(ksancov_trace_t * trace)411*f6217f89SApple OSS Distributions ksancov_trace_head(ksancov_trace_t *trace)
412*f6217f89SApple OSS Distributions {
413*f6217f89SApple OSS Distributions 	assert(trace);
414*f6217f89SApple OSS Distributions 	size_t maxent = trace->kt_maxent;
415*f6217f89SApple OSS Distributions 	size_t head = atomic_load_explicit(&trace->kt_head, memory_order_acquire);
416*f6217f89SApple OSS Distributions 	return head < maxent ? head : maxent;
417*f6217f89SApple OSS Distributions }
418*f6217f89SApple OSS Distributions 
419*f6217f89SApple OSS Distributions static inline uintptr_t
ksancov_trace_entry(ksancov_trace_t * trace,size_t i)420*f6217f89SApple OSS Distributions ksancov_trace_entry(ksancov_trace_t *trace, size_t i)
421*f6217f89SApple OSS Distributions {
422*f6217f89SApple OSS Distributions 	assert(trace);
423*f6217f89SApple OSS Distributions 	assert(trace->kt_hdr.kh_magic == KSANCOV_TRACE_MAGIC);
424*f6217f89SApple OSS Distributions 	if (i >= trace->kt_head) {
425*f6217f89SApple OSS Distributions 		return 0;
426*f6217f89SApple OSS Distributions 	}
427*f6217f89SApple OSS Distributions 
428*f6217f89SApple OSS Distributions 	ksancov_trace_pc_ent_t *entries = (ksancov_trace_pc_ent_t *)trace->kt_entries;
429*f6217f89SApple OSS Distributions 	return entries[i];
430*f6217f89SApple OSS Distributions }
431*f6217f89SApple OSS Distributions 
432*f6217f89SApple OSS Distributions static inline uintptr_t
ksancov_stksize_pc(ksancov_trace_t * trace,size_t i)433*f6217f89SApple OSS Distributions ksancov_stksize_pc(ksancov_trace_t *trace, size_t i)
434*f6217f89SApple OSS Distributions {
435*f6217f89SApple OSS Distributions 	assert(trace);
436*f6217f89SApple OSS Distributions 	assert(trace->kt_hdr.kh_magic == KSANCOV_STKSIZE_MAGIC);
437*f6217f89SApple OSS Distributions 	if (i >= trace->kt_head) {
438*f6217f89SApple OSS Distributions 		return 0;
439*f6217f89SApple OSS Distributions 	}
440*f6217f89SApple OSS Distributions 
441*f6217f89SApple OSS Distributions 	ksancov_trace_stksize_ent_t *entries = (ksancov_trace_stksize_ent_t *)trace->kt_entries;
442*f6217f89SApple OSS Distributions 	return entries[i].pc;
443*f6217f89SApple OSS Distributions }
444*f6217f89SApple OSS Distributions 
445*f6217f89SApple OSS Distributions static inline uint32_t
ksancov_stksize_size(ksancov_trace_t * trace,size_t i)446*f6217f89SApple OSS Distributions ksancov_stksize_size(ksancov_trace_t *trace, size_t i)
447*f6217f89SApple OSS Distributions {
448*f6217f89SApple OSS Distributions 	assert(trace);
449*f6217f89SApple OSS Distributions 	assert(trace->kt_hdr.kh_magic == KSANCOV_STKSIZE_MAGIC);
450*f6217f89SApple OSS Distributions 	if (i >= trace->kt_head) {
451*f6217f89SApple OSS Distributions 		return 0;
452*f6217f89SApple OSS Distributions 	}
453*f6217f89SApple OSS Distributions 
454*f6217f89SApple OSS Distributions 	ksancov_trace_stksize_ent_t *entries = (ksancov_trace_stksize_ent_t *)trace->kt_entries;
455*f6217f89SApple OSS Distributions 	return entries[i].stksize;
456*f6217f89SApple OSS Distributions }
457*f6217f89SApple OSS Distributions 
458*f6217f89SApple OSS Distributions static inline int
ksancov_cmps_mode_trace(int fd,size_t entries,bool trace_functions)459*f6217f89SApple OSS Distributions ksancov_cmps_mode_trace(int fd, size_t entries, bool trace_functions)
460*f6217f89SApple OSS Distributions {
461*f6217f89SApple OSS Distributions 	int ret;
462*f6217f89SApple OSS Distributions 	if (trace_functions) {
463*f6217f89SApple OSS Distributions 		ret = ioctl(fd, KSANCOV_IOC_CMPS_TRACE_FUNC, &entries);
464*f6217f89SApple OSS Distributions 	} else {
465*f6217f89SApple OSS Distributions 		ret = ioctl(fd, KSANCOV_IOC_CMPS_TRACE, &entries);
466*f6217f89SApple OSS Distributions 	}
467*f6217f89SApple OSS Distributions 	if (ret == -1) {
468*f6217f89SApple OSS Distributions 		return errno;
469*f6217f89SApple OSS Distributions 	}
470*f6217f89SApple OSS Distributions 	return 0;
471*f6217f89SApple OSS Distributions }
472*f6217f89SApple OSS Distributions 
473*f6217f89SApple OSS Distributions static inline int
ksancov_cmps_map(int fd,uintptr_t * buf,size_t * sz)474*f6217f89SApple OSS Distributions ksancov_cmps_map(int fd, uintptr_t *buf, size_t *sz)
475*f6217f89SApple OSS Distributions {
476*f6217f89SApple OSS Distributions 	int ret;
477*f6217f89SApple OSS Distributions 	struct ksancov_buf_desc mc = {0};
478*f6217f89SApple OSS Distributions 
479*f6217f89SApple OSS Distributions 	assert(buf != NULL);
480*f6217f89SApple OSS Distributions 
481*f6217f89SApple OSS Distributions 	ret = ioctl(fd, KSANCOV_IOC_CMPS_MAP, &mc);
482*f6217f89SApple OSS Distributions 	if (ret == -1) {
483*f6217f89SApple OSS Distributions 		return errno;
484*f6217f89SApple OSS Distributions 	}
485*f6217f89SApple OSS Distributions 
486*f6217f89SApple OSS Distributions 	*buf = mc.ptr;
487*f6217f89SApple OSS Distributions 	if (sz) {
488*f6217f89SApple OSS Distributions 		*sz = mc.sz;
489*f6217f89SApple OSS Distributions 	}
490*f6217f89SApple OSS Distributions 
491*f6217f89SApple OSS Distributions 	ksancov_header_t *hdr = (ksancov_header_t *)mc.ptr;
492*f6217f89SApple OSS Distributions 	assert(hdr->kh_magic == KSANCOV_CMPS_TRACE_MAGIC);
493*f6217f89SApple OSS Distributions 
494*f6217f89SApple OSS Distributions 	return 0;
495*f6217f89SApple OSS Distributions }
496*f6217f89SApple OSS Distributions 
497*f6217f89SApple OSS Distributions static inline ksancov_cmps_trace_ent_t*
ksancov_cmps_trace_entry(ksancov_trace_t * trace,size_t i)498*f6217f89SApple OSS Distributions ksancov_cmps_trace_entry(ksancov_trace_t *trace, size_t i)
499*f6217f89SApple OSS Distributions {
500*f6217f89SApple OSS Distributions 	assert(trace);
501*f6217f89SApple OSS Distributions 	assert(trace->kt_hdr.kh_magic == KSANCOV_CMPS_TRACE_MAGIC);
502*f6217f89SApple OSS Distributions 	if (i >= trace->kt_head) {
503*f6217f89SApple OSS Distributions 		return NULL;
504*f6217f89SApple OSS Distributions 	}
505*f6217f89SApple OSS Distributions 
506*f6217f89SApple OSS Distributions 	ksancov_cmps_trace_ent_t *entries = (ksancov_cmps_trace_ent_t *)trace->kt_entries;
507*f6217f89SApple OSS Distributions 	return &entries[i];
508*f6217f89SApple OSS Distributions }
509*f6217f89SApple OSS Distributions 
510*f6217f89SApple OSS Distributions /*
511*f6217f89SApple OSS Distributions  * On-demand control API
512*f6217f89SApple OSS Distributions  */
513*f6217f89SApple OSS Distributions 
514*f6217f89SApple OSS Distributions static inline int
_ksancov_on_demand_operation(int fd,const char * bundle,ksancov_on_demand_operation_t op,struct ksancov_on_demand_msg * msg)515*f6217f89SApple OSS Distributions _ksancov_on_demand_operation(int fd, const char *bundle, ksancov_on_demand_operation_t op, struct ksancov_on_demand_msg *msg)
516*f6217f89SApple OSS Distributions {
517*f6217f89SApple OSS Distributions 	int ret;
518*f6217f89SApple OSS Distributions 
519*f6217f89SApple OSS Distributions 	msg->operation = op;
520*f6217f89SApple OSS Distributions 	strlcpy(msg->bundle, bundle, sizeof(msg->bundle));
521*f6217f89SApple OSS Distributions 
522*f6217f89SApple OSS Distributions 	ret = ioctl(fd, KSANCOV_IOC_ON_DEMAND, msg);
523*f6217f89SApple OSS Distributions 	if (ret == -1) {
524*f6217f89SApple OSS Distributions 		return errno;
525*f6217f89SApple OSS Distributions 	}
526*f6217f89SApple OSS Distributions 
527*f6217f89SApple OSS Distributions 	return ret;
528*f6217f89SApple OSS Distributions }
529*f6217f89SApple OSS Distributions 
530*f6217f89SApple OSS Distributions /*
531*f6217f89SApple OSS Distributions  * Retrieve the value of the gate for a given module bundle ID.
532*f6217f89SApple OSS Distributions  */
533*f6217f89SApple OSS Distributions static inline int
ksancov_on_demand_get_gate(int fd,const char * bundle,uint64_t * gate)534*f6217f89SApple OSS Distributions ksancov_on_demand_get_gate(int fd, const char *bundle, uint64_t *gate)
535*f6217f89SApple OSS Distributions {
536*f6217f89SApple OSS Distributions 	assert(gate);
537*f6217f89SApple OSS Distributions 
538*f6217f89SApple OSS Distributions 	struct ksancov_on_demand_msg msg;
539*f6217f89SApple OSS Distributions 	int ret = _ksancov_on_demand_operation(fd, bundle, KS_OD_GET_GATE, &msg);
540*f6217f89SApple OSS Distributions 	if (ret == 0) {
541*f6217f89SApple OSS Distributions 		*gate = msg.gate;
542*f6217f89SApple OSS Distributions 	}
543*f6217f89SApple OSS Distributions 	return ret;
544*f6217f89SApple OSS Distributions }
545*f6217f89SApple OSS Distributions 
546*f6217f89SApple OSS Distributions /*
547*f6217f89SApple OSS Distributions  * Set the value of the gate for a given module bundle ID.
548*f6217f89SApple OSS Distributions  *
549*f6217f89SApple OSS Distributions  * Any non-zero value enables the invocation of the sanitizer coverage callbacks
550*f6217f89SApple OSS Distributions  * inserted in the specified module.
551*f6217f89SApple OSS Distributions  */
552*f6217f89SApple OSS Distributions static inline int
ksancov_on_demand_set_gate(int fd,const char * bundle,uint64_t value)553*f6217f89SApple OSS Distributions ksancov_on_demand_set_gate(int fd, const char *bundle, uint64_t value)
554*f6217f89SApple OSS Distributions {
555*f6217f89SApple OSS Distributions 	struct ksancov_on_demand_msg msg = {};
556*f6217f89SApple OSS Distributions 	msg.gate = value;
557*f6217f89SApple OSS Distributions 	return _ksancov_on_demand_operation(fd, bundle, KS_OD_SET_GATE, &msg);
558*f6217f89SApple OSS Distributions }
559*f6217f89SApple OSS Distributions 
560*f6217f89SApple OSS Distributions /*
561*f6217f89SApple OSS Distributions  * Get the guards range for a specified module.
562*f6217f89SApple OSS Distributions  */
563*f6217f89SApple OSS Distributions static inline int
ksancov_on_demand_get_range(int fd,const char * bundle,uint32_t * start,uint32_t * stop)564*f6217f89SApple OSS Distributions ksancov_on_demand_get_range(int fd, const char *bundle, uint32_t *start, uint32_t *stop)
565*f6217f89SApple OSS Distributions {
566*f6217f89SApple OSS Distributions 	assert(start && stop);
567*f6217f89SApple OSS Distributions 
568*f6217f89SApple OSS Distributions 	struct ksancov_on_demand_msg msg = {};
569*f6217f89SApple OSS Distributions 	int ret = _ksancov_on_demand_operation(fd, bundle, KS_OD_GET_RANGE, &msg);
570*f6217f89SApple OSS Distributions 	if (ret == 0) {
571*f6217f89SApple OSS Distributions 		*start = msg.range.start;
572*f6217f89SApple OSS Distributions 		*stop = msg.range.stop;
573*f6217f89SApple OSS Distributions 	}
574*f6217f89SApple OSS Distributions 	return ret;
575*f6217f89SApple OSS Distributions }
576*f6217f89SApple OSS Distributions 
577*f6217f89SApple OSS Distributions #endif /* _KSANCOV_H_ */
578