1*0f4c859eSApple OSS Distributions /*
2*0f4c859eSApple OSS Distributions * CDDL HEADER START
3*0f4c859eSApple OSS Distributions *
4*0f4c859eSApple OSS Distributions * The contents of this file are subject to the terms of the
5*0f4c859eSApple OSS Distributions * Common Development and Distribution License (the "License").
6*0f4c859eSApple OSS Distributions * You may not use this file except in compliance with the License.
7*0f4c859eSApple OSS Distributions *
8*0f4c859eSApple OSS Distributions * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9*0f4c859eSApple OSS Distributions * or http://www.opensolaris.org/os/licensing.
10*0f4c859eSApple OSS Distributions * See the License for the specific language governing permissions
11*0f4c859eSApple OSS Distributions * and limitations under the License.
12*0f4c859eSApple OSS Distributions *
13*0f4c859eSApple OSS Distributions * When distributing Covered Code, include this CDDL HEADER in each
14*0f4c859eSApple OSS Distributions * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15*0f4c859eSApple OSS Distributions * If applicable, add the following below this CDDL HEADER, with the
16*0f4c859eSApple OSS Distributions * fields enclosed by brackets "[]" replaced with your own identifying
17*0f4c859eSApple OSS Distributions * information: Portions Copyright [yyyy] [name of copyright owner]
18*0f4c859eSApple OSS Distributions *
19*0f4c859eSApple OSS Distributions * CDDL HEADER END
20*0f4c859eSApple OSS Distributions */
21*0f4c859eSApple OSS Distributions /*
22*0f4c859eSApple OSS Distributions * Copyright 2009 Sun Microsystems, Inc. All rights reserved.
23*0f4c859eSApple OSS Distributions * Use is subject to license terms.
24*0f4c859eSApple OSS Distributions */
25*0f4c859eSApple OSS Distributions
26*0f4c859eSApple OSS Distributions #include <mach-o/loader.h>
27*0f4c859eSApple OSS Distributions #include <libkern/kernel_mach_header.h>
28*0f4c859eSApple OSS Distributions
29*0f4c859eSApple OSS Distributions #include <sys/param.h>
30*0f4c859eSApple OSS Distributions #include <sys/systm.h>
31*0f4c859eSApple OSS Distributions #include <sys/sysctl.h>
32*0f4c859eSApple OSS Distributions #include <sys/errno.h>
33*0f4c859eSApple OSS Distributions #include <sys/stat.h>
34*0f4c859eSApple OSS Distributions #include <sys/ioctl.h>
35*0f4c859eSApple OSS Distributions #include <sys/conf.h>
36*0f4c859eSApple OSS Distributions #include <sys/fcntl.h>
37*0f4c859eSApple OSS Distributions #include <miscfs/devfs/devfs.h>
38*0f4c859eSApple OSS Distributions #include <pexpert/pexpert.h>
39*0f4c859eSApple OSS Distributions
40*0f4c859eSApple OSS Distributions #include <sys/dtrace.h>
41*0f4c859eSApple OSS Distributions #include <sys/dtrace_impl.h>
42*0f4c859eSApple OSS Distributions #include <sys/fbt.h>
43*0f4c859eSApple OSS Distributions
44*0f4c859eSApple OSS Distributions #include <sys/dtrace_glue.h>
45*0f4c859eSApple OSS Distributions #include <san/kasan.h>
46*0f4c859eSApple OSS Distributions
47*0f4c859eSApple OSS Distributions #include <ptrauth.h>
48*0f4c859eSApple OSS Distributions
49*0f4c859eSApple OSS Distributions /* #include <machine/trap.h> */
50*0f4c859eSApple OSS Distributions struct savearea_t; /* Used anonymously */
51*0f4c859eSApple OSS Distributions
52*0f4c859eSApple OSS Distributions #if defined(__arm64__)
53*0f4c859eSApple OSS Distributions typedef kern_return_t (*perfCallback)(int, struct savearea_t *, __unused int, __unused int);
54*0f4c859eSApple OSS Distributions extern perfCallback tempDTraceTrapHook;
55*0f4c859eSApple OSS Distributions extern kern_return_t fbt_perfCallback(int, struct savearea_t *, __unused int, __unused int);
56*0f4c859eSApple OSS Distributions #elif defined(__x86_64__)
57*0f4c859eSApple OSS Distributions typedef kern_return_t (*perfCallback)(int, struct savearea_t *, uintptr_t *, __unused int);
58*0f4c859eSApple OSS Distributions extern perfCallback tempDTraceTrapHook;
59*0f4c859eSApple OSS Distributions extern kern_return_t fbt_perfCallback(int, struct savearea_t *, uintptr_t *, __unused int);
60*0f4c859eSApple OSS Distributions #else
61*0f4c859eSApple OSS Distributions #error Unknown architecture
62*0f4c859eSApple OSS Distributions #endif
63*0f4c859eSApple OSS Distributions
64*0f4c859eSApple OSS Distributions __private_extern__
65*0f4c859eSApple OSS Distributions void
66*0f4c859eSApple OSS Distributions qsort(void *a, size_t n, size_t es, int (*cmp)(const void *, const void *));
67*0f4c859eSApple OSS Distributions
68*0f4c859eSApple OSS Distributions #define FBT_ADDR2NDX(addr) ((((uintptr_t)(addr)) >> 4) & fbt_probetab_mask)
69*0f4c859eSApple OSS Distributions #define FBT_PROBETAB_SIZE 0x8000 /* 32k entries -- 128K total */
70*0f4c859eSApple OSS Distributions
71*0f4c859eSApple OSS Distributions static int fbt_probetab_size;
72*0f4c859eSApple OSS Distributions dtrace_provider_id_t fbt_id;
73*0f4c859eSApple OSS Distributions fbt_probe_t **fbt_probetab;
74*0f4c859eSApple OSS Distributions int fbt_probetab_mask;
75*0f4c859eSApple OSS Distributions static int fbt_verbose = 0;
76*0f4c859eSApple OSS Distributions
77*0f4c859eSApple OSS Distributions extern int ignore_fbt_blacklist;
78*0f4c859eSApple OSS Distributions
79*0f4c859eSApple OSS Distributions extern int dtrace_kernel_symbol_mode;
80*0f4c859eSApple OSS Distributions
81*0f4c859eSApple OSS Distributions
82*0f4c859eSApple OSS Distributions void fbt_init( void );
83*0f4c859eSApple OSS Distributions
84*0f4c859eSApple OSS Distributions /*ARGSUSED*/
85*0f4c859eSApple OSS Distributions static void
fbt_destroy(void * arg,dtrace_id_t id,void * parg)86*0f4c859eSApple OSS Distributions fbt_destroy(void *arg, dtrace_id_t id, void *parg)
87*0f4c859eSApple OSS Distributions {
88*0f4c859eSApple OSS Distributions #pragma unused(arg,id)
89*0f4c859eSApple OSS Distributions fbt_probe_t *fbt = parg, *next, *hash, *last;
90*0f4c859eSApple OSS Distributions int ndx;
91*0f4c859eSApple OSS Distributions
92*0f4c859eSApple OSS Distributions do {
93*0f4c859eSApple OSS Distributions /*
94*0f4c859eSApple OSS Distributions * Now we need to remove this probe from the fbt_probetab.
95*0f4c859eSApple OSS Distributions */
96*0f4c859eSApple OSS Distributions ndx = FBT_ADDR2NDX(fbt->fbtp_patchpoint);
97*0f4c859eSApple OSS Distributions last = NULL;
98*0f4c859eSApple OSS Distributions hash = fbt_probetab[ndx];
99*0f4c859eSApple OSS Distributions
100*0f4c859eSApple OSS Distributions while (hash != fbt) {
101*0f4c859eSApple OSS Distributions ASSERT(hash != NULL);
102*0f4c859eSApple OSS Distributions last = hash;
103*0f4c859eSApple OSS Distributions hash = hash->fbtp_hashnext;
104*0f4c859eSApple OSS Distributions }
105*0f4c859eSApple OSS Distributions
106*0f4c859eSApple OSS Distributions if (last != NULL) {
107*0f4c859eSApple OSS Distributions last->fbtp_hashnext = fbt->fbtp_hashnext;
108*0f4c859eSApple OSS Distributions } else {
109*0f4c859eSApple OSS Distributions fbt_probetab[ndx] = fbt->fbtp_hashnext;
110*0f4c859eSApple OSS Distributions }
111*0f4c859eSApple OSS Distributions
112*0f4c859eSApple OSS Distributions next = fbt->fbtp_next;
113*0f4c859eSApple OSS Distributions kmem_free(fbt, sizeof(fbt_probe_t));
114*0f4c859eSApple OSS Distributions
115*0f4c859eSApple OSS Distributions fbt = next;
116*0f4c859eSApple OSS Distributions } while (fbt != NULL);
117*0f4c859eSApple OSS Distributions }
118*0f4c859eSApple OSS Distributions
119*0f4c859eSApple OSS Distributions /*ARGSUSED*/
120*0f4c859eSApple OSS Distributions int
fbt_enable(void * arg,dtrace_id_t id,void * parg)121*0f4c859eSApple OSS Distributions fbt_enable(void *arg, dtrace_id_t id, void *parg)
122*0f4c859eSApple OSS Distributions {
123*0f4c859eSApple OSS Distributions #pragma unused(arg,id)
124*0f4c859eSApple OSS Distributions fbt_probe_t *fbt = parg;
125*0f4c859eSApple OSS Distributions struct modctl *ctl = NULL;
126*0f4c859eSApple OSS Distributions
127*0f4c859eSApple OSS Distributions for (; fbt != NULL; fbt = fbt->fbtp_next) {
128*0f4c859eSApple OSS Distributions ctl = fbt->fbtp_ctl;
129*0f4c859eSApple OSS Distributions
130*0f4c859eSApple OSS Distributions if (!ctl->mod_loaded) {
131*0f4c859eSApple OSS Distributions if (fbt_verbose) {
132*0f4c859eSApple OSS Distributions cmn_err(CE_NOTE, "fbt is failing for probe %s "
133*0f4c859eSApple OSS Distributions "(module %s unloaded)",
134*0f4c859eSApple OSS Distributions fbt->fbtp_name, ctl->mod_modname);
135*0f4c859eSApple OSS Distributions }
136*0f4c859eSApple OSS Distributions
137*0f4c859eSApple OSS Distributions continue;
138*0f4c859eSApple OSS Distributions }
139*0f4c859eSApple OSS Distributions
140*0f4c859eSApple OSS Distributions /*
141*0f4c859eSApple OSS Distributions * Now check that our modctl has the expected load count. If it
142*0f4c859eSApple OSS Distributions * doesn't, this module must have been unloaded and reloaded -- and
143*0f4c859eSApple OSS Distributions * we're not going to touch it.
144*0f4c859eSApple OSS Distributions */
145*0f4c859eSApple OSS Distributions if (ctl->mod_loadcnt != fbt->fbtp_loadcnt) {
146*0f4c859eSApple OSS Distributions if (fbt_verbose) {
147*0f4c859eSApple OSS Distributions cmn_err(CE_NOTE, "fbt is failing for probe %s "
148*0f4c859eSApple OSS Distributions "(module %s reloaded)",
149*0f4c859eSApple OSS Distributions fbt->fbtp_name, ctl->mod_modname);
150*0f4c859eSApple OSS Distributions }
151*0f4c859eSApple OSS Distributions
152*0f4c859eSApple OSS Distributions continue;
153*0f4c859eSApple OSS Distributions }
154*0f4c859eSApple OSS Distributions
155*0f4c859eSApple OSS Distributions dtrace_casptr(&tempDTraceTrapHook, NULL, ptrauth_nop_cast(void *, &fbt_perfCallback));
156*0f4c859eSApple OSS Distributions if (tempDTraceTrapHook != (perfCallback)fbt_perfCallback) {
157*0f4c859eSApple OSS Distributions if (fbt_verbose) {
158*0f4c859eSApple OSS Distributions cmn_err(CE_NOTE, "fbt_enable is failing for probe %s "
159*0f4c859eSApple OSS Distributions "in module %s: tempDTraceTrapHook already occupied.",
160*0f4c859eSApple OSS Distributions fbt->fbtp_name, ctl->mod_modname);
161*0f4c859eSApple OSS Distributions }
162*0f4c859eSApple OSS Distributions continue;
163*0f4c859eSApple OSS Distributions }
164*0f4c859eSApple OSS Distributions
165*0f4c859eSApple OSS Distributions if (fbt->fbtp_currentval != fbt->fbtp_patchval) {
166*0f4c859eSApple OSS Distributions #if KASAN_CLASSIC
167*0f4c859eSApple OSS Distributions /* Since dtrace probes can call into KASan and vice versa, things can get
168*0f4c859eSApple OSS Distributions * very slow if we have a lot of probes. This call will disable the KASan
169*0f4c859eSApple OSS Distributions * fakestack after a threshold of probes is reached. */
170*0f4c859eSApple OSS Distributions kasan_fakestack_suspend();
171*0f4c859eSApple OSS Distributions #endif /* KASAN_CLASSIC */
172*0f4c859eSApple OSS Distributions
173*0f4c859eSApple OSS Distributions (void)ml_nofault_copy((vm_offset_t)&fbt->fbtp_patchval, (vm_offset_t)fbt->fbtp_patchpoint,
174*0f4c859eSApple OSS Distributions sizeof(fbt->fbtp_patchval));
175*0f4c859eSApple OSS Distributions /*
176*0f4c859eSApple OSS Distributions * Make the patched instruction visible via a data + instruction
177*0f4c859eSApple OSS Distributions * cache flush for the platforms that need it
178*0f4c859eSApple OSS Distributions */
179*0f4c859eSApple OSS Distributions flush_dcache((vm_offset_t)fbt->fbtp_patchpoint, (vm_size_t)sizeof(fbt->fbtp_patchval), 0);
180*0f4c859eSApple OSS Distributions invalidate_icache((vm_offset_t)fbt->fbtp_patchpoint, (vm_size_t)sizeof(fbt->fbtp_patchval), 0);
181*0f4c859eSApple OSS Distributions fbt->fbtp_currentval = fbt->fbtp_patchval;
182*0f4c859eSApple OSS Distributions
183*0f4c859eSApple OSS Distributions ctl->mod_nenabled++;
184*0f4c859eSApple OSS Distributions }
185*0f4c859eSApple OSS Distributions }
186*0f4c859eSApple OSS Distributions
187*0f4c859eSApple OSS Distributions dtrace_membar_consumer();
188*0f4c859eSApple OSS Distributions
189*0f4c859eSApple OSS Distributions return 0;
190*0f4c859eSApple OSS Distributions }
191*0f4c859eSApple OSS Distributions
192*0f4c859eSApple OSS Distributions /*ARGSUSED*/
193*0f4c859eSApple OSS Distributions static void
fbt_disable(void * arg,dtrace_id_t id,void * parg)194*0f4c859eSApple OSS Distributions fbt_disable(void *arg, dtrace_id_t id, void *parg)
195*0f4c859eSApple OSS Distributions {
196*0f4c859eSApple OSS Distributions #pragma unused(arg,id)
197*0f4c859eSApple OSS Distributions fbt_probe_t *fbt = parg;
198*0f4c859eSApple OSS Distributions struct modctl *ctl = NULL;
199*0f4c859eSApple OSS Distributions
200*0f4c859eSApple OSS Distributions for (; fbt != NULL; fbt = fbt->fbtp_next) {
201*0f4c859eSApple OSS Distributions ctl = fbt->fbtp_ctl;
202*0f4c859eSApple OSS Distributions
203*0f4c859eSApple OSS Distributions if (!ctl->mod_loaded || (ctl->mod_loadcnt != fbt->fbtp_loadcnt)) {
204*0f4c859eSApple OSS Distributions continue;
205*0f4c859eSApple OSS Distributions }
206*0f4c859eSApple OSS Distributions
207*0f4c859eSApple OSS Distributions if (fbt->fbtp_currentval != fbt->fbtp_savedval) {
208*0f4c859eSApple OSS Distributions (void)ml_nofault_copy((vm_offset_t)&fbt->fbtp_savedval, (vm_offset_t)fbt->fbtp_patchpoint,
209*0f4c859eSApple OSS Distributions sizeof(fbt->fbtp_savedval));
210*0f4c859eSApple OSS Distributions /*
211*0f4c859eSApple OSS Distributions * Make the patched instruction visible via a data + instruction
212*0f4c859eSApple OSS Distributions * cache flush for the platforms that need it
213*0f4c859eSApple OSS Distributions */
214*0f4c859eSApple OSS Distributions flush_dcache((vm_offset_t)fbt->fbtp_patchpoint, (vm_size_t)sizeof(fbt->fbtp_patchval), 0);
215*0f4c859eSApple OSS Distributions invalidate_icache((vm_offset_t)fbt->fbtp_patchpoint, (vm_size_t)sizeof(fbt->fbtp_patchval), 0);
216*0f4c859eSApple OSS Distributions
217*0f4c859eSApple OSS Distributions fbt->fbtp_currentval = fbt->fbtp_savedval;
218*0f4c859eSApple OSS Distributions ASSERT(ctl->mod_nenabled > 0);
219*0f4c859eSApple OSS Distributions ctl->mod_nenabled--;
220*0f4c859eSApple OSS Distributions
221*0f4c859eSApple OSS Distributions #if KASAN && KASAN_CLASSIC
222*0f4c859eSApple OSS Distributions kasan_fakestack_resume();
223*0f4c859eSApple OSS Distributions #endif /* KASAN && KASAN_CLASSIC */
224*0f4c859eSApple OSS Distributions }
225*0f4c859eSApple OSS Distributions }
226*0f4c859eSApple OSS Distributions dtrace_membar_consumer();
227*0f4c859eSApple OSS Distributions }
228*0f4c859eSApple OSS Distributions
229*0f4c859eSApple OSS Distributions /*ARGSUSED*/
230*0f4c859eSApple OSS Distributions static void
fbt_suspend(void * arg,dtrace_id_t id,void * parg)231*0f4c859eSApple OSS Distributions fbt_suspend(void *arg, dtrace_id_t id, void *parg)
232*0f4c859eSApple OSS Distributions {
233*0f4c859eSApple OSS Distributions #pragma unused(arg,id)
234*0f4c859eSApple OSS Distributions fbt_probe_t *fbt = parg;
235*0f4c859eSApple OSS Distributions struct modctl *ctl = NULL;
236*0f4c859eSApple OSS Distributions
237*0f4c859eSApple OSS Distributions for (; fbt != NULL; fbt = fbt->fbtp_next) {
238*0f4c859eSApple OSS Distributions ctl = fbt->fbtp_ctl;
239*0f4c859eSApple OSS Distributions
240*0f4c859eSApple OSS Distributions ASSERT(ctl->mod_nenabled > 0);
241*0f4c859eSApple OSS Distributions if (!ctl->mod_loaded || (ctl->mod_loadcnt != fbt->fbtp_loadcnt)) {
242*0f4c859eSApple OSS Distributions continue;
243*0f4c859eSApple OSS Distributions }
244*0f4c859eSApple OSS Distributions
245*0f4c859eSApple OSS Distributions (void)ml_nofault_copy((vm_offset_t)&fbt->fbtp_savedval, (vm_offset_t)fbt->fbtp_patchpoint,
246*0f4c859eSApple OSS Distributions sizeof(fbt->fbtp_savedval));
247*0f4c859eSApple OSS Distributions
248*0f4c859eSApple OSS Distributions /*
249*0f4c859eSApple OSS Distributions * Make the patched instruction visible via a data + instruction
250*0f4c859eSApple OSS Distributions * cache flush for the platforms that need it
251*0f4c859eSApple OSS Distributions */
252*0f4c859eSApple OSS Distributions flush_dcache((vm_offset_t)fbt->fbtp_patchpoint, (vm_size_t)sizeof(fbt->fbtp_savedval), 0);
253*0f4c859eSApple OSS Distributions invalidate_icache((vm_offset_t)fbt->fbtp_patchpoint, (vm_size_t)sizeof(fbt->fbtp_savedval), 0);
254*0f4c859eSApple OSS Distributions
255*0f4c859eSApple OSS Distributions fbt->fbtp_currentval = fbt->fbtp_savedval;
256*0f4c859eSApple OSS Distributions }
257*0f4c859eSApple OSS Distributions
258*0f4c859eSApple OSS Distributions dtrace_membar_consumer();
259*0f4c859eSApple OSS Distributions }
260*0f4c859eSApple OSS Distributions
261*0f4c859eSApple OSS Distributions /*ARGSUSED*/
262*0f4c859eSApple OSS Distributions static void
fbt_resume(void * arg,dtrace_id_t id,void * parg)263*0f4c859eSApple OSS Distributions fbt_resume(void *arg, dtrace_id_t id, void *parg)
264*0f4c859eSApple OSS Distributions {
265*0f4c859eSApple OSS Distributions #pragma unused(arg,id)
266*0f4c859eSApple OSS Distributions fbt_probe_t *fbt = parg;
267*0f4c859eSApple OSS Distributions struct modctl *ctl = NULL;
268*0f4c859eSApple OSS Distributions
269*0f4c859eSApple OSS Distributions for (; fbt != NULL; fbt = fbt->fbtp_next) {
270*0f4c859eSApple OSS Distributions ctl = fbt->fbtp_ctl;
271*0f4c859eSApple OSS Distributions
272*0f4c859eSApple OSS Distributions ASSERT(ctl->mod_nenabled > 0);
273*0f4c859eSApple OSS Distributions if (!ctl->mod_loaded || (ctl->mod_loadcnt != fbt->fbtp_loadcnt)) {
274*0f4c859eSApple OSS Distributions continue;
275*0f4c859eSApple OSS Distributions }
276*0f4c859eSApple OSS Distributions
277*0f4c859eSApple OSS Distributions dtrace_casptr(&tempDTraceTrapHook, NULL, ptrauth_nop_cast(void *, &fbt_perfCallback));
278*0f4c859eSApple OSS Distributions if (tempDTraceTrapHook != (perfCallback)fbt_perfCallback) {
279*0f4c859eSApple OSS Distributions if (fbt_verbose) {
280*0f4c859eSApple OSS Distributions cmn_err(CE_NOTE, "fbt_resume is failing for probe %s "
281*0f4c859eSApple OSS Distributions "in module %s: tempDTraceTrapHook already occupied.",
282*0f4c859eSApple OSS Distributions fbt->fbtp_name, ctl->mod_modname);
283*0f4c859eSApple OSS Distributions }
284*0f4c859eSApple OSS Distributions return;
285*0f4c859eSApple OSS Distributions }
286*0f4c859eSApple OSS Distributions
287*0f4c859eSApple OSS Distributions (void)ml_nofault_copy((vm_offset_t)&fbt->fbtp_patchval, (vm_offset_t)fbt->fbtp_patchpoint,
288*0f4c859eSApple OSS Distributions sizeof(fbt->fbtp_patchval));
289*0f4c859eSApple OSS Distributions
290*0f4c859eSApple OSS Distributions /*
291*0f4c859eSApple OSS Distributions * Make the patched instruction visible via a data + instruction cache flush.
292*0f4c859eSApple OSS Distributions */
293*0f4c859eSApple OSS Distributions flush_dcache((vm_offset_t)fbt->fbtp_patchpoint, (vm_size_t)sizeof(fbt->fbtp_patchval), 0);
294*0f4c859eSApple OSS Distributions invalidate_icache((vm_offset_t)fbt->fbtp_patchpoint, (vm_size_t)sizeof(fbt->fbtp_patchval), 0);
295*0f4c859eSApple OSS Distributions
296*0f4c859eSApple OSS Distributions fbt->fbtp_currentval = fbt->fbtp_patchval;
297*0f4c859eSApple OSS Distributions }
298*0f4c859eSApple OSS Distributions
299*0f4c859eSApple OSS Distributions dtrace_membar_consumer();
300*0f4c859eSApple OSS Distributions }
301*0f4c859eSApple OSS Distributions
302*0f4c859eSApple OSS Distributions static void
fbt_provide_module_user_syms(struct modctl * ctl)303*0f4c859eSApple OSS Distributions fbt_provide_module_user_syms(struct modctl *ctl)
304*0f4c859eSApple OSS Distributions {
305*0f4c859eSApple OSS Distributions unsigned int i;
306*0f4c859eSApple OSS Distributions char *modname = ctl->mod_modname;
307*0f4c859eSApple OSS Distributions
308*0f4c859eSApple OSS Distributions dtrace_module_symbols_t* module_symbols = ctl->mod_user_symbols;
309*0f4c859eSApple OSS Distributions if (module_symbols) {
310*0f4c859eSApple OSS Distributions for (i = 0; i < module_symbols->dtmodsyms_count; i++) {
311*0f4c859eSApple OSS Distributions /*
312*0f4c859eSApple OSS Distributions * symbol->dtsym_addr (the symbol address) passed in from
313*0f4c859eSApple OSS Distributions * user space, is already slid for both kexts and kernel.
314*0f4c859eSApple OSS Distributions */
315*0f4c859eSApple OSS Distributions dtrace_symbol_t* symbol = &module_symbols->dtmodsyms_symbols[i];
316*0f4c859eSApple OSS Distributions
317*0f4c859eSApple OSS Distributions char* name = symbol->dtsym_name;
318*0f4c859eSApple OSS Distributions
319*0f4c859eSApple OSS Distributions /* Lop off omnipresent leading underscore. */
320*0f4c859eSApple OSS Distributions if (*name == '_') {
321*0f4c859eSApple OSS Distributions name += 1;
322*0f4c859eSApple OSS Distributions }
323*0f4c859eSApple OSS Distributions
324*0f4c859eSApple OSS Distributions if (fbt_excluded(name)) {
325*0f4c859eSApple OSS Distributions continue;
326*0f4c859eSApple OSS Distributions }
327*0f4c859eSApple OSS Distributions
328*0f4c859eSApple OSS Distributions /*
329*0f4c859eSApple OSS Distributions * Ignore symbols with a null address
330*0f4c859eSApple OSS Distributions */
331*0f4c859eSApple OSS Distributions if (!symbol->dtsym_addr) {
332*0f4c859eSApple OSS Distributions continue;
333*0f4c859eSApple OSS Distributions }
334*0f4c859eSApple OSS Distributions
335*0f4c859eSApple OSS Distributions /*
336*0f4c859eSApple OSS Distributions * Ignore symbols not part of this module
337*0f4c859eSApple OSS Distributions */
338*0f4c859eSApple OSS Distributions if (!dtrace_addr_in_module((void*)symbol->dtsym_addr, ctl)) {
339*0f4c859eSApple OSS Distributions continue;
340*0f4c859eSApple OSS Distributions }
341*0f4c859eSApple OSS Distributions
342*0f4c859eSApple OSS Distributions fbt_provide_probe(ctl, modname, name, (machine_inst_t*)(uintptr_t)symbol->dtsym_addr, (machine_inst_t*)(uintptr_t)(symbol->dtsym_addr + symbol->dtsym_size));
343*0f4c859eSApple OSS Distributions }
344*0f4c859eSApple OSS Distributions }
345*0f4c859eSApple OSS Distributions }
346*0f4c859eSApple OSS Distributions static void
fbt_provide_kernel_section(struct modctl * ctl,kernel_section_t * sect,kernel_nlist_t * sym,uint32_t nsyms,const char * strings)347*0f4c859eSApple OSS Distributions fbt_provide_kernel_section(struct modctl *ctl, kernel_section_t *sect, kernel_nlist_t *sym, uint32_t nsyms, const char *strings)
348*0f4c859eSApple OSS Distributions {
349*0f4c859eSApple OSS Distributions uintptr_t sect_start = (uintptr_t)sect->addr;
350*0f4c859eSApple OSS Distributions uintptr_t sect_end = (uintptr_t)sect->size + sect->addr;
351*0f4c859eSApple OSS Distributions unsigned int i;
352*0f4c859eSApple OSS Distributions
353*0f4c859eSApple OSS Distributions if ((sect->flags & S_ATTR_PURE_INSTRUCTIONS) != S_ATTR_PURE_INSTRUCTIONS) {
354*0f4c859eSApple OSS Distributions return;
355*0f4c859eSApple OSS Distributions }
356*0f4c859eSApple OSS Distributions
357*0f4c859eSApple OSS Distributions for (i = 0; i < nsyms; i++) {
358*0f4c859eSApple OSS Distributions uint8_t n_type = sym[i].n_type & (N_TYPE | N_EXT);
359*0f4c859eSApple OSS Distributions const char *name = strings + sym[i].n_un.n_strx;
360*0f4c859eSApple OSS Distributions uint64_t limit;
361*0f4c859eSApple OSS Distributions
362*0f4c859eSApple OSS Distributions if (sym[i].n_value < sect_start || sym[i].n_value > sect_end) {
363*0f4c859eSApple OSS Distributions continue;
364*0f4c859eSApple OSS Distributions }
365*0f4c859eSApple OSS Distributions
366*0f4c859eSApple OSS Distributions /* Check that the symbol is a global and that it has a name. */
367*0f4c859eSApple OSS Distributions if (((N_SECT | N_EXT) != n_type && (N_ABS | N_EXT) != n_type)) {
368*0f4c859eSApple OSS Distributions continue;
369*0f4c859eSApple OSS Distributions }
370*0f4c859eSApple OSS Distributions
371*0f4c859eSApple OSS Distributions if (0 == sym[i].n_un.n_strx) { /* iff a null, "", name. */
372*0f4c859eSApple OSS Distributions continue;
373*0f4c859eSApple OSS Distributions }
374*0f4c859eSApple OSS Distributions
375*0f4c859eSApple OSS Distributions /* Lop off omnipresent leading underscore. */
376*0f4c859eSApple OSS Distributions if (*name == '_') {
377*0f4c859eSApple OSS Distributions name += 1;
378*0f4c859eSApple OSS Distributions }
379*0f4c859eSApple OSS Distributions
380*0f4c859eSApple OSS Distributions if (fbt_excluded(name)) {
381*0f4c859eSApple OSS Distributions continue;
382*0f4c859eSApple OSS Distributions }
383*0f4c859eSApple OSS Distributions
384*0f4c859eSApple OSS Distributions /*
385*0f4c859eSApple OSS Distributions * Find the function boundary by looking at either the
386*0f4c859eSApple OSS Distributions * end of the section or the beginning of the next symbol
387*0f4c859eSApple OSS Distributions */
388*0f4c859eSApple OSS Distributions if (i == nsyms - 1) {
389*0f4c859eSApple OSS Distributions limit = sect_end;
390*0f4c859eSApple OSS Distributions } else {
391*0f4c859eSApple OSS Distributions limit = sym[i + 1].n_value;
392*0f4c859eSApple OSS Distributions }
393*0f4c859eSApple OSS Distributions
394*0f4c859eSApple OSS Distributions fbt_provide_probe(ctl, ctl->mod_modname, name, (machine_inst_t*)sym[i].n_value, (machine_inst_t*)limit);
395*0f4c859eSApple OSS Distributions }
396*0f4c859eSApple OSS Distributions }
397*0f4c859eSApple OSS Distributions
398*0f4c859eSApple OSS Distributions static int
fbt_sym_cmp(const void * ap,const void * bp)399*0f4c859eSApple OSS Distributions fbt_sym_cmp(const void *ap, const void *bp)
400*0f4c859eSApple OSS Distributions {
401*0f4c859eSApple OSS Distributions return (int)(((const kernel_nlist_t*)ap)->n_value - ((const kernel_nlist_t*)bp)->n_value);
402*0f4c859eSApple OSS Distributions }
403*0f4c859eSApple OSS Distributions
404*0f4c859eSApple OSS Distributions static void
fbt_provide_module_kernel_syms(struct modctl * ctl)405*0f4c859eSApple OSS Distributions fbt_provide_module_kernel_syms(struct modctl *ctl)
406*0f4c859eSApple OSS Distributions {
407*0f4c859eSApple OSS Distributions kernel_mach_header_t *mh = (kernel_mach_header_t *)(ctl->mod_address);
408*0f4c859eSApple OSS Distributions kernel_segment_command_t *seg;
409*0f4c859eSApple OSS Distributions struct load_command *cmd;
410*0f4c859eSApple OSS Distributions kernel_segment_command_t *linkedit = NULL;
411*0f4c859eSApple OSS Distributions struct symtab_command *symtab = NULL;
412*0f4c859eSApple OSS Distributions kernel_nlist_t *syms = NULL, *sorted_syms = NULL;
413*0f4c859eSApple OSS Distributions const char *strings;
414*0f4c859eSApple OSS Distributions unsigned int i;
415*0f4c859eSApple OSS Distributions size_t symlen;
416*0f4c859eSApple OSS Distributions
417*0f4c859eSApple OSS Distributions if (mh->magic != MH_MAGIC_KERNEL) {
418*0f4c859eSApple OSS Distributions return;
419*0f4c859eSApple OSS Distributions }
420*0f4c859eSApple OSS Distributions
421*0f4c859eSApple OSS Distributions cmd = (struct load_command *) &mh[1];
422*0f4c859eSApple OSS Distributions for (i = 0; i < mh->ncmds; i++) {
423*0f4c859eSApple OSS Distributions if (cmd->cmd == LC_SEGMENT_KERNEL) {
424*0f4c859eSApple OSS Distributions kernel_segment_command_t *orig_sg = (kernel_segment_command_t *) cmd;
425*0f4c859eSApple OSS Distributions if (LIT_STRNEQL(orig_sg->segname, SEG_LINKEDIT)) {
426*0f4c859eSApple OSS Distributions linkedit = orig_sg;
427*0f4c859eSApple OSS Distributions }
428*0f4c859eSApple OSS Distributions } else if (cmd->cmd == LC_SYMTAB) {
429*0f4c859eSApple OSS Distributions symtab = (struct symtab_command *) cmd;
430*0f4c859eSApple OSS Distributions }
431*0f4c859eSApple OSS Distributions if (symtab && linkedit) {
432*0f4c859eSApple OSS Distributions break;
433*0f4c859eSApple OSS Distributions }
434*0f4c859eSApple OSS Distributions cmd = (struct load_command *) ((caddr_t) cmd + cmd->cmdsize);
435*0f4c859eSApple OSS Distributions }
436*0f4c859eSApple OSS Distributions
437*0f4c859eSApple OSS Distributions if ((symtab == NULL) || (linkedit == NULL)) {
438*0f4c859eSApple OSS Distributions return;
439*0f4c859eSApple OSS Distributions }
440*0f4c859eSApple OSS Distributions
441*0f4c859eSApple OSS Distributions syms = (kernel_nlist_t *)(linkedit->vmaddr + symtab->symoff - linkedit->fileoff);
442*0f4c859eSApple OSS Distributions strings = (const char *)(linkedit->vmaddr + symtab->stroff - linkedit->fileoff);
443*0f4c859eSApple OSS Distributions
444*0f4c859eSApple OSS Distributions /*
445*0f4c859eSApple OSS Distributions * Make a copy of the symbol table and sort it to not cross into the next function
446*0f4c859eSApple OSS Distributions * when disassembling the function
447*0f4c859eSApple OSS Distributions */
448*0f4c859eSApple OSS Distributions symlen = sizeof(kernel_nlist_t) * symtab->nsyms;
449*0f4c859eSApple OSS Distributions sorted_syms = kmem_alloc(symlen, KM_SLEEP);
450*0f4c859eSApple OSS Distributions bcopy(syms, sorted_syms, symlen);
451*0f4c859eSApple OSS Distributions qsort(sorted_syms, symtab->nsyms, sizeof(kernel_nlist_t), fbt_sym_cmp);
452*0f4c859eSApple OSS Distributions
453*0f4c859eSApple OSS Distributions for (seg = firstsegfromheader(mh); seg != NULL; seg = nextsegfromheader(mh, seg)) {
454*0f4c859eSApple OSS Distributions kernel_section_t *sect = firstsect(seg);
455*0f4c859eSApple OSS Distributions
456*0f4c859eSApple OSS Distributions if (strcmp(seg->segname, "__KLD") == 0 || strcmp(seg->segname, "__KLDDATA") == 0) {
457*0f4c859eSApple OSS Distributions continue;
458*0f4c859eSApple OSS Distributions }
459*0f4c859eSApple OSS Distributions
460*0f4c859eSApple OSS Distributions for (sect = firstsect(seg); sect != NULL; sect = nextsect(seg, sect)) {
461*0f4c859eSApple OSS Distributions fbt_provide_kernel_section(ctl, sect, sorted_syms, symtab->nsyms, strings);
462*0f4c859eSApple OSS Distributions }
463*0f4c859eSApple OSS Distributions }
464*0f4c859eSApple OSS Distributions
465*0f4c859eSApple OSS Distributions kmem_free(sorted_syms, symlen);
466*0f4c859eSApple OSS Distributions }
467*0f4c859eSApple OSS Distributions
468*0f4c859eSApple OSS Distributions void
fbt_provide_module(void * arg,struct modctl * ctl)469*0f4c859eSApple OSS Distributions fbt_provide_module(void *arg, struct modctl *ctl)
470*0f4c859eSApple OSS Distributions {
471*0f4c859eSApple OSS Distributions #pragma unused(arg)
472*0f4c859eSApple OSS Distributions ASSERT(ctl != NULL);
473*0f4c859eSApple OSS Distributions ASSERT(dtrace_kernel_symbol_mode != DTRACE_KERNEL_SYMBOLS_NEVER);
474*0f4c859eSApple OSS Distributions LCK_MTX_ASSERT(&mod_lock, LCK_MTX_ASSERT_OWNED);
475*0f4c859eSApple OSS Distributions
476*0f4c859eSApple OSS Distributions if (dtrace_fbt_probes_restricted()) {
477*0f4c859eSApple OSS Distributions return;
478*0f4c859eSApple OSS Distributions }
479*0f4c859eSApple OSS Distributions
480*0f4c859eSApple OSS Distributions // Update the "ignore blacklist" bit
481*0f4c859eSApple OSS Distributions if (ignore_fbt_blacklist) {
482*0f4c859eSApple OSS Distributions ctl->mod_flags |= MODCTL_FBT_PROVIDE_BLACKLISTED_PROBES;
483*0f4c859eSApple OSS Distributions }
484*0f4c859eSApple OSS Distributions
485*0f4c859eSApple OSS Distributions if (MOD_FBT_DONE(ctl)) {
486*0f4c859eSApple OSS Distributions return;
487*0f4c859eSApple OSS Distributions }
488*0f4c859eSApple OSS Distributions
489*0f4c859eSApple OSS Distributions if (fbt_module_excluded(ctl)) {
490*0f4c859eSApple OSS Distributions ctl->mod_flags |= MODCTL_FBT_INVALID;
491*0f4c859eSApple OSS Distributions return;
492*0f4c859eSApple OSS Distributions }
493*0f4c859eSApple OSS Distributions
494*0f4c859eSApple OSS Distributions if (MOD_HAS_KERNEL_SYMBOLS(ctl)) {
495*0f4c859eSApple OSS Distributions fbt_provide_module_kernel_syms(ctl);
496*0f4c859eSApple OSS Distributions ctl->mod_flags |= MODCTL_FBT_PROBES_PROVIDED;
497*0f4c859eSApple OSS Distributions if (MOD_FBT_PROVIDE_BLACKLISTED_PROBES(ctl)) {
498*0f4c859eSApple OSS Distributions ctl->mod_flags |= MODCTL_FBT_BLACKLISTED_PROBES_PROVIDED;
499*0f4c859eSApple OSS Distributions }
500*0f4c859eSApple OSS Distributions return;
501*0f4c859eSApple OSS Distributions }
502*0f4c859eSApple OSS Distributions
503*0f4c859eSApple OSS Distributions if (MOD_HAS_USERSPACE_SYMBOLS(ctl)) {
504*0f4c859eSApple OSS Distributions fbt_provide_module_user_syms(ctl);
505*0f4c859eSApple OSS Distributions ctl->mod_flags |= MODCTL_FBT_PROBES_PROVIDED;
506*0f4c859eSApple OSS Distributions if (MOD_FBT_PROVIDE_BLACKLISTED_PROBES(ctl)) {
507*0f4c859eSApple OSS Distributions ctl->mod_flags |= MODCTL_FBT_BLACKLISTED_PROBES_PROVIDED;
508*0f4c859eSApple OSS Distributions }
509*0f4c859eSApple OSS Distributions return;
510*0f4c859eSApple OSS Distributions }
511*0f4c859eSApple OSS Distributions }
512*0f4c859eSApple OSS Distributions
513*0f4c859eSApple OSS Distributions static dtrace_pattr_t fbt_attr = {
514*0f4c859eSApple OSS Distributions { DTRACE_STABILITY_EVOLVING, DTRACE_STABILITY_EVOLVING, DTRACE_CLASS_ISA },
515*0f4c859eSApple OSS Distributions { DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_UNKNOWN },
516*0f4c859eSApple OSS Distributions { DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_UNKNOWN },
517*0f4c859eSApple OSS Distributions { DTRACE_STABILITY_EVOLVING, DTRACE_STABILITY_EVOLVING, DTRACE_CLASS_ISA },
518*0f4c859eSApple OSS Distributions { DTRACE_STABILITY_PRIVATE, DTRACE_STABILITY_PRIVATE, DTRACE_CLASS_ISA },
519*0f4c859eSApple OSS Distributions };
520*0f4c859eSApple OSS Distributions
521*0f4c859eSApple OSS Distributions static dtrace_pops_t fbt_pops = {
522*0f4c859eSApple OSS Distributions .dtps_provide = NULL,
523*0f4c859eSApple OSS Distributions .dtps_provide_module = fbt_provide_module,
524*0f4c859eSApple OSS Distributions .dtps_enable = fbt_enable,
525*0f4c859eSApple OSS Distributions .dtps_disable = fbt_disable,
526*0f4c859eSApple OSS Distributions .dtps_suspend = fbt_suspend,
527*0f4c859eSApple OSS Distributions .dtps_resume = fbt_resume,
528*0f4c859eSApple OSS Distributions .dtps_getargdesc = NULL, /* APPLE NOTE: fbt_getargdesc implemented in userspace */
529*0f4c859eSApple OSS Distributions .dtps_getargval = NULL,
530*0f4c859eSApple OSS Distributions .dtps_usermode = NULL,
531*0f4c859eSApple OSS Distributions .dtps_destroy = fbt_destroy
532*0f4c859eSApple OSS Distributions };
533*0f4c859eSApple OSS Distributions
534*0f4c859eSApple OSS Distributions static void
fbt_cleanup(dev_info_t * devi)535*0f4c859eSApple OSS Distributions fbt_cleanup(dev_info_t *devi)
536*0f4c859eSApple OSS Distributions {
537*0f4c859eSApple OSS Distributions dtrace_invop_remove(fbt_invop);
538*0f4c859eSApple OSS Distributions ddi_remove_minor_node(devi, NULL);
539*0f4c859eSApple OSS Distributions kmem_free(fbt_probetab, fbt_probetab_size * sizeof(fbt_probe_t *));
540*0f4c859eSApple OSS Distributions fbt_probetab = NULL;
541*0f4c859eSApple OSS Distributions fbt_probetab_mask = 0;
542*0f4c859eSApple OSS Distributions }
543*0f4c859eSApple OSS Distributions
544*0f4c859eSApple OSS Distributions static int
fbt_attach(dev_info_t * devi)545*0f4c859eSApple OSS Distributions fbt_attach(dev_info_t *devi)
546*0f4c859eSApple OSS Distributions {
547*0f4c859eSApple OSS Distributions if (fbt_probetab_size == 0) {
548*0f4c859eSApple OSS Distributions fbt_probetab_size = FBT_PROBETAB_SIZE;
549*0f4c859eSApple OSS Distributions }
550*0f4c859eSApple OSS Distributions
551*0f4c859eSApple OSS Distributions fbt_probetab_mask = fbt_probetab_size - 1;
552*0f4c859eSApple OSS Distributions fbt_probetab =
553*0f4c859eSApple OSS Distributions kmem_zalloc(fbt_probetab_size * sizeof(fbt_probe_t *), KM_SLEEP);
554*0f4c859eSApple OSS Distributions
555*0f4c859eSApple OSS Distributions dtrace_invop_add(fbt_invop);
556*0f4c859eSApple OSS Distributions
557*0f4c859eSApple OSS Distributions if (ddi_create_minor_node(devi, "fbt", S_IFCHR, 0,
558*0f4c859eSApple OSS Distributions DDI_PSEUDO, 0) == DDI_FAILURE ||
559*0f4c859eSApple OSS Distributions dtrace_register("fbt", &fbt_attr, DTRACE_PRIV_KERNEL, NULL,
560*0f4c859eSApple OSS Distributions &fbt_pops, NULL, &fbt_id) != 0) {
561*0f4c859eSApple OSS Distributions fbt_cleanup(devi);
562*0f4c859eSApple OSS Distributions return DDI_FAILURE;
563*0f4c859eSApple OSS Distributions }
564*0f4c859eSApple OSS Distributions
565*0f4c859eSApple OSS Distributions return DDI_SUCCESS;
566*0f4c859eSApple OSS Distributions }
567*0f4c859eSApple OSS Distributions
568*0f4c859eSApple OSS Distributions static d_open_t _fbt_open;
569*0f4c859eSApple OSS Distributions
570*0f4c859eSApple OSS Distributions static int
_fbt_open(dev_t dev,int flags,int devtype,struct proc * p)571*0f4c859eSApple OSS Distributions _fbt_open(dev_t dev, int flags, int devtype, struct proc *p)
572*0f4c859eSApple OSS Distributions {
573*0f4c859eSApple OSS Distributions #pragma unused(dev,flags,devtype,p)
574*0f4c859eSApple OSS Distributions return 0;
575*0f4c859eSApple OSS Distributions }
576*0f4c859eSApple OSS Distributions
577*0f4c859eSApple OSS Distributions #define FBT_MAJOR -24 /* let the kernel pick the device number */
578*0f4c859eSApple OSS Distributions
579*0f4c859eSApple OSS Distributions static const struct cdevsw fbt_cdevsw =
580*0f4c859eSApple OSS Distributions {
581*0f4c859eSApple OSS Distributions .d_open = _fbt_open,
582*0f4c859eSApple OSS Distributions .d_close = eno_opcl,
583*0f4c859eSApple OSS Distributions .d_read = eno_rdwrt,
584*0f4c859eSApple OSS Distributions .d_write = eno_rdwrt,
585*0f4c859eSApple OSS Distributions .d_ioctl = eno_ioctl,
586*0f4c859eSApple OSS Distributions .d_stop = eno_stop,
587*0f4c859eSApple OSS Distributions .d_reset = eno_reset,
588*0f4c859eSApple OSS Distributions .d_select = eno_select,
589*0f4c859eSApple OSS Distributions .d_mmap = eno_mmap,
590*0f4c859eSApple OSS Distributions .d_strategy = eno_strat,
591*0f4c859eSApple OSS Distributions .d_reserved_1 = eno_getc,
592*0f4c859eSApple OSS Distributions .d_reserved_2 = eno_putc,
593*0f4c859eSApple OSS Distributions };
594*0f4c859eSApple OSS Distributions
595*0f4c859eSApple OSS Distributions #undef kmem_alloc /* from its binding to dt_kmem_alloc glue */
596*0f4c859eSApple OSS Distributions #undef kmem_free /* from its binding to dt_kmem_free glue */
597*0f4c859eSApple OSS Distributions #include <vm/vm_kern.h>
598*0f4c859eSApple OSS Distributions
599*0f4c859eSApple OSS Distributions
600*0f4c859eSApple OSS Distributions void
fbt_init(void)601*0f4c859eSApple OSS Distributions fbt_init( void )
602*0f4c859eSApple OSS Distributions {
603*0f4c859eSApple OSS Distributions int majdevno = cdevsw_add(FBT_MAJOR, &fbt_cdevsw);
604*0f4c859eSApple OSS Distributions
605*0f4c859eSApple OSS Distributions if (majdevno < 0) {
606*0f4c859eSApple OSS Distributions printf("fbt_init: failed to allocate a major number!\n");
607*0f4c859eSApple OSS Distributions return;
608*0f4c859eSApple OSS Distributions }
609*0f4c859eSApple OSS Distributions
610*0f4c859eSApple OSS Distributions fbt_blacklist_init();
611*0f4c859eSApple OSS Distributions fbt_attach((dev_info_t*)(uintptr_t)majdevno);
612*0f4c859eSApple OSS Distributions }
613*0f4c859eSApple OSS Distributions #undef FBT_MAJOR
614