2 * Copyright (c) 2005-2018 Apple Computer, Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
29 #include <arm/caches_internal.h>
30 #include <arm/proc_reg.h>
32 #include <kern/thread.h>
33 #include <mach/thread_status.h>
37 #include <sys/malloc.h>
39 #include <sys/systm.h>
41 #include <sys/proc_internal.h>
42 #include <sys/kauth.h>
43 #include <sys/dtrace.h>
44 #include <sys/dtrace_impl.h>
45 #include <machine/atomic.h>
46 #include <kern/simple_lock.h>
47 #include <kern/sched_prim.h> /* for thread_wakeup() */
48 #include <kern/thread_call.h>
49 #include <kern/task.h>
50 #include <miscfs/devfs/devfs.h>
51 #include <mach/vm_param.h>
53 extern struct arm_saved_state
*find_kern_regs(thread_t
);
55 extern dtrace_id_t dtrace_probeid_error
; /* special ERROR probe */
56 typedef arm_saved_state_t savearea_t
;
58 extern lck_attr_t
*dtrace_lck_attr
;
59 extern lck_grp_t
*dtrace_lck_grp
;
61 int dtrace_arm_condition_true(int condition
, int cpsr
);
64 * Atomicity and synchronization
67 dtrace_membar_producer(void)
69 __asm__
volatile ("dmb ish" : : : "memory");
73 dtrace_membar_consumer(void)
75 __asm__
volatile ("dmb ish" : : : "memory");
79 * Interrupt manipulation
80 * XXX dtrace_getipl() can be called from probe context.
86 * XXX Drat, get_interrupt_level is MACH_KERNEL_PRIVATE
87 * in osfmk/kern/cpu_data.h
89 /* return get_interrupt_level(); */
90 return ml_at_interrupt_context() ? 1 : 0;
97 decl_lck_mtx_data(static, dt_xc_lock
);
98 static uint32_t dt_xc_sync
;
100 typedef struct xcArg
{
109 xcArg_t
*pArg
= (xcArg_t
*) foo
;
111 if (pArg
->cpu
== CPU
->cpu_id
|| pArg
->cpu
== DTRACE_CPUALL
) {
112 (pArg
->f
)(pArg
->arg
);
115 if (os_atomic_dec(&dt_xc_sync
, relaxed
) == 0) {
116 thread_wakeup((event_t
) &dt_xc_sync
);
121 * dtrace_xcall() is not called from probe context.
124 dtrace_xcall(processorid_t cpu
, dtrace_xcall_t f
, void *arg
)
126 /* Only one dtrace_xcall in flight allowed */
127 lck_mtx_lock(&dt_xc_lock
);
135 cpu_broadcast_xcall(&dt_xc_sync
, TRUE
, xcRemote
, (void*) &xcArg
);
137 lck_mtx_unlock(&dt_xc_lock
);
145 dtrace_isa_init(void)
147 lck_mtx_init(&dt_xc_lock
, dtrace_lck_grp
, dtrace_lck_attr
);
155 dtrace_getreg(struct regs
* savearea
, uint_t reg
)
157 struct arm_saved_state
*regs
= (struct arm_saved_state
*) savearea
;
159 DTRACE_CPUFLAG_SET(CPU_DTRACE_ILLOP
);
162 /* beyond register limit? */
163 if (reg
> ARM_SAVED_STATE32_COUNT
- 1) {
164 DTRACE_CPUFLAG_SET(CPU_DTRACE_ILLOP
);
168 return (uint64_t) ((unsigned int *) (&(regs
->r
)))[reg
];
172 dtrace_getvmreg(uint_t ndx
)
175 DTRACE_CPUFLAG_SET(CPU_DTRACE_ILLOP
);
179 #define RETURN_OFFSET 4
182 dtrace_getustack_common(uint64_t * pcstack
, int pcstack_limit
, user_addr_t pc
,
185 volatile uint16_t *flags
= (volatile uint16_t *) &cpu_core
[CPU
->cpu_id
].cpuc_dtrace_flags
;
188 ASSERT(pcstack
== NULL
|| pcstack_limit
> 0);
192 if (pcstack
!= NULL
) {
193 *pcstack
++ = (uint64_t) pc
;
195 if (pcstack_limit
<= 0) {
204 pc
= dtrace_fuword32((sp
+ RETURN_OFFSET
));
205 sp
= dtrace_fuword32(sp
);
207 /* Truncate ustack if the iterator causes fault. */
208 if (*flags
& CPU_DTRACE_FAULT
) {
209 *flags
&= ~CPU_DTRACE_FAULT
;
218 dtrace_getupcstack(uint64_t * pcstack
, int pcstack_limit
)
220 thread_t thread
= current_thread();
223 volatile uint16_t *flags
= (volatile uint16_t *) &cpu_core
[CPU
->cpu_id
].cpuc_dtrace_flags
;
226 if (*flags
& CPU_DTRACE_FAULT
) {
230 if (pcstack_limit
<= 0) {
235 * If there's no user context we still need to zero the stack.
237 if (thread
== NULL
) {
241 regs
= (savearea_t
*) find_user_regs(thread
);
246 *pcstack
++ = (uint64_t)dtrace_proc_selfpid();
249 if (pcstack_limit
<= 0) {
256 if (DTRACE_CPUFLAG_ISSET(CPU_DTRACE_ENTRY
)) {
257 *pcstack
++ = (uint64_t) pc
;
259 if (pcstack_limit
<= 0) {
266 n
= dtrace_getustack_common(pcstack
, pcstack_limit
, pc
, regs
->r
[7]);
269 ASSERT(n
<= pcstack_limit
);
275 while (pcstack_limit
-- > 0) {
281 dtrace_getustackdepth(void)
283 thread_t thread
= current_thread();
288 if (thread
== NULL
) {
292 if (DTRACE_CPUFLAG_ISSET(CPU_DTRACE_FAULT
)) {
296 regs
= (savearea_t
*) find_user_regs(thread
);
304 if (DTRACE_CPUFLAG_ISSET(CPU_DTRACE_ENTRY
)) {
310 * Note that unlike ppc, the arm code does not use
311 * CPU_DTRACE_USTACK_FP. This is because arm always
312 * traces from the sp, even in syscall/profile/fbt
316 n
+= dtrace_getustack_common(NULL
, 0, pc
, regs
->r
[7]);
322 dtrace_getufpstack(uint64_t * pcstack
, uint64_t * fpstack
, int pcstack_limit
)
324 /* XXX ARMTODO 64vs32 */
325 thread_t thread
= current_thread();
329 volatile uint16_t *flags
= (volatile uint16_t *) &cpu_core
[CPU
->cpu_id
].cpuc_dtrace_flags
;
332 uintptr_t oldcontext
;
336 if (*flags
& CPU_DTRACE_FAULT
) {
340 if (pcstack_limit
<= 0) {
345 * If there's no user context we still need to zero the stack.
347 if (thread
== NULL
) {
351 regs
= (savearea_t
*) find_user_regs(thread
);
356 *pcstack
++ = (uint64_t)dtrace_proc_selfpid();
359 if (pcstack_limit
<= 0) {
366 #if 0 /* XXX signal stack crawl */
367 oldcontext
= lwp
->lwp_oldcontext
;
369 if (p
->p_model
== DATAMODEL_NATIVE
) {
370 s1
= sizeof(struct frame
) + 2 * sizeof(long);
371 s2
= s1
+ sizeof(siginfo_t
);
373 s1
= sizeof(struct frame32
) + 3 * sizeof(int);
374 s2
= s1
+ sizeof(siginfo32_t
);
378 if (DTRACE_CPUFLAG_ISSET(CPU_DTRACE_ENTRY
)) {
379 *pcstack
++ = (uint64_t) pc
;
382 if (pcstack_limit
<= 0) {
386 pc
= dtrace_fuword32(sp
);
388 while (pc
!= 0 && sp
!= 0) {
389 *pcstack
++ = (uint64_t) pc
;
392 if (pcstack_limit
<= 0) {
396 #if 0 /* XXX signal stack crawl */
397 if (oldcontext
== sp
+ s1
|| oldcontext
== sp
+ s2
) {
398 if (p
->p_model
== DATAMODEL_NATIVE
) {
399 ucontext_t
*ucp
= (ucontext_t
*) oldcontext
;
400 greg_t
*gregs
= ucp
->uc_mcontext
.gregs
;
402 sp
= dtrace_fulword(&gregs
[REG_FP
]);
403 pc
= dtrace_fulword(&gregs
[REG_PC
]);
405 oldcontext
= dtrace_fulword(&ucp
->uc_link
);
407 ucontext_t
*ucp
= (ucontext_t
*) oldcontext
;
408 greg_t
*gregs
= ucp
->uc_mcontext
.gregs
;
410 sp
= dtrace_fuword32(&gregs
[EBP
]);
411 pc
= dtrace_fuword32(&gregs
[EIP
]);
413 oldcontext
= dtrace_fuword32(&ucp
->uc_link
);
418 pc
= dtrace_fuword32((sp
+ RETURN_OFFSET
));
419 sp
= dtrace_fuword32(sp
);
422 /* Truncate ustack if the iterator causes fault. */
423 if (*flags
& CPU_DTRACE_FAULT
) {
424 *flags
&= ~CPU_DTRACE_FAULT
;
430 while (pcstack_limit
-- > 0) {
436 dtrace_getpcstack(pc_t
* pcstack
, int pcstack_limit
, int aframes
,
439 struct frame
*fp
= (struct frame
*) __builtin_frame_address(0);
440 struct frame
*nextfp
, *minfp
, *stacktop
;
445 uintptr_t caller
= CPU
->cpu_dtrace_caller
;
447 if ((on_intr
= CPU_ON_INTR(CPU
)) != 0) {
448 stacktop
= (struct frame
*) dtrace_get_cpu_int_stack_top();
450 stacktop
= (struct frame
*) (dtrace_get_kernel_stack(current_thread()) + kernel_stack_size
);
457 if (intrpc
!= NULL
&& depth
< pcstack_limit
) {
458 pcstack
[depth
++] = (pc_t
) intrpc
;
461 while (depth
< pcstack_limit
) {
462 nextfp
= *(struct frame
**) fp
;
463 pc
= *(uintptr_t *) (((uint32_t) fp
) + RETURN_OFFSET
);
465 if (nextfp
<= minfp
|| nextfp
>= stacktop
) {
468 * Hop from interrupt stack to thread stack.
470 arm_saved_state_t
*arm_kern_regs
= (arm_saved_state_t
*) find_kern_regs(current_thread());
472 nextfp
= (struct frame
*)arm_kern_regs
->r
[7];
474 vm_offset_t kstack_base
= dtrace_get_kernel_stack(current_thread());
476 minfp
= (struct frame
*)kstack_base
;
477 stacktop
= (struct frame
*)(kstack_base
+ kernel_stack_size
);
481 if (nextfp
<= minfp
|| nextfp
>= stacktop
) {
486 * If this thread was on the interrupt stack, but did not
487 * take an interrupt (i.e, the idle thread), there is no
488 * explicit saved state for us to use.
494 * This is the last frame we can process; indicate
495 * that we should return after processing this frame.
501 if (--aframes
== 0 && caller
!= (uintptr_t)NULL
) {
503 * We've just run out of artificial frames,
504 * and we have a valid caller -- fill it in
507 ASSERT(depth
< pcstack_limit
);
508 pcstack
[depth
++] = (pc_t
) caller
;
509 caller
= (uintptr_t)NULL
;
512 if (depth
< pcstack_limit
) {
513 pcstack
[depth
++] = (pc_t
) pc
;
518 while (depth
< pcstack_limit
) {
519 pcstack
[depth
++] = (pc_t
) NULL
;
529 dtrace_instr_size(uint32_t instr
, int thumb_mode
)
532 uint16_t instr16
= *(uint16_t*) &instr
;
533 if (((instr16
>> 11) & 0x1F) > 0x1C) {
544 dtrace_getarg(int arg
, int aframes
, dtrace_mstate_t
*mstate
, dtrace_vstate_t
*vstate
)
546 #pragma unused(arg, aframes, mstate, vstate)
550 uintptr_t *fp
= (uintptr_t *)__builtin_frame_address(0);
555 for (i
= 1; i
<= aframes
; i
++) {
559 if (dtrace_invop_callsite_pre
!= NULL
560 && pc
> (uintptr_t)dtrace_invop_callsite_pre
561 && pc
<= (uintptr_t)dtrace_invop_callsite_post
) {
563 * If we pass through the invalid op handler, we will
564 * use the pointer that it passed to the stack as the
565 * second argument to dtrace_invop() as the pointer to
566 * the frame we're hunting for.
569 stack
= (uintptr_t *)&fp
[1]; /* Find marshalled arguments */
570 fp
= (struct frame
*)stack
[1]; /* Grab *second* argument */
571 stack
= (uintptr_t *)&fp
[1]; /* Find marshalled arguments */
572 DTRACE_CPUFLAG_SET(CPU_DTRACE_NOFAULT
);
573 val
= (uint64_t)(stack
[arg
]);
574 DTRACE_CPUFLAG_CLEAR(CPU_DTRACE_NOFAULT
);
580 * Arrive here when provider has called dtrace_probe directly.
582 stack
= (uintptr_t *)&fp
[1]; /* Find marshalled arguments */
583 stack
++; /* Advance past probeID */
585 DTRACE_CPUFLAG_SET(CPU_DTRACE_NOFAULT
);
586 val
= *(((uint64_t *)stack
) + arg
); /* dtrace_probe arguments arg0 .. arg4 are 64bits wide */
587 DTRACE_CPUFLAG_CLEAR(CPU_DTRACE_NOFAULT
);
590 return 0xfeedfacedeafbeadLL
;
594 dtrace_probe_error(dtrace_state_t
*state
, dtrace_epid_t epid
, int which
,
595 int fltoffs
, int fault
, uint64_t illval
)
599 * For the case of the error probe firing lets
600 * stash away "illval" here, and special-case retrieving it in DIF_VARIABLE_ARG.
602 state
->dts_arg_error_illval
= illval
;
603 dtrace_probe( dtrace_probeid_error
, (uint64_t)(uintptr_t)state
, epid
, which
, fltoffs
, fault
);
607 dtrace_toxic_ranges(void (*func
)(uintptr_t base
, uintptr_t limit
))
609 /* XXX ARMTODO check copied from ppc/x86*/
611 * "base" is the smallest toxic address in the range, "limit" is the first
612 * VALID address greater than "base".
614 func(0x0, VM_MIN_KERNEL_ADDRESS
);
615 if (VM_MAX_KERNEL_ADDRESS
< ~(uintptr_t)0) {
616 func(VM_MAX_KERNEL_ADDRESS
+ 1, ~(uintptr_t)0);
621 dtrace_arm_condition_true(int cond
, int cpsr
)
624 int zf
= (cpsr
& PSR_ZF
) ? 1 : 0,
625 nf
= (cpsr
& PSR_NF
) ? 1 : 0,
626 cf
= (cpsr
& PSR_CF
) ? 1 : 0,
627 vf
= (cpsr
& PSR_VF
) ? 1 : 0;
630 case 0: taken
= zf
; break;
631 case 1: taken
= !zf
; break;
632 case 2: taken
= cf
; break;
633 case 3: taken
= !cf
; break;
634 case 4: taken
= nf
; break;
635 case 5: taken
= !nf
; break;
636 case 6: taken
= vf
; break;
637 case 7: taken
= !vf
; break;
638 case 8: taken
= (cf
&& !zf
); break;
639 case 9: taken
= (!cf
|| zf
); break;
640 case 10: taken
= (nf
== vf
); break;
641 case 11: taken
= (nf
!= vf
); break;
642 case 12: taken
= (!zf
&& (nf
== vf
)); break;
643 case 13: taken
= (zf
|| (nf
!= vf
)); break;
644 case 14: taken
= 1; break;
645 case 15: taken
= 1; break; /* always "true" for ARM, unpredictable for THUMB. */
652 dtrace_flush_caches(void)
654 /* TODO There were some problems with flushing just the cache line that had been modified.
655 * For now, we'll flush the entire cache, until we figure out how to flush just the patched block.
658 InvalidatePoU_Icache();