Skip to content

Commit 11ee87b

Browse files
authored
i#5383: Fix macOS arm64 test build/run (#7171)
This makes tests build and run properly on macOS (14.4) arm64 (M3). Many tests do not pass, but they can be at least run now. Specifically, we: - disable selfmod tests due to build failures on M3; - fix macOS build for mangle_pauth test (test still fails for some reason); - mark dump_ucontext as NYI on macOS; - disable `reachable_client` option by default on macOS due to lack of a private loader; - fix NULL sigcontext_t in `os_forge_exception()` (the `mangle_pauth` test was hitting this). Issue: #5383 #7296
1 parent cdf427c commit 11ee87b

File tree

8 files changed

+42
-13
lines changed

8 files changed

+42
-13
lines changed

api/docs/intro.dox

+3-1
Original file line numberDiff line numberDiff line change
@@ -470,7 +470,9 @@ preferred client library base in the low 4GB. The \ref op_reachable_client
470470
"-reachable_client runtime option" can be used to remove this guarantee.
471471
The option is automatically turned off when DynamoRIO and clients are
472472
statically linked with the application: for such static usage, 32-bit
473-
reachability to the code cache by clients is not guaranteed.
473+
reachability to the code cache by clients is not guaranteed. It is also
474+
disabled on macOS, where the lack of a private loader means we cannot
475+
control where the client library gets loaded.
474476

475477
The net result is that any static data or code in a client library, or any
476478
data allocated using DynamoRIO's API routines (except dr_raw_mem_alloc() or

core/optionsx.h

+3-1
Original file line numberDiff line numberDiff line change
@@ -1674,7 +1674,9 @@ OPTION_DEFAULT(bool, reachable_heap, false,
16741674
"guarantee that all heap memory is 32-bit-displacement "
16751675
"reachable from the code cache.")
16761676
/* i#3570: For static DR we do not guarantee reachability. */
1677-
OPTION_DEFAULT(bool, reachable_client, IF_STATIC_LIBRARY_ELSE(false, true),
1677+
/* i#7296: On macOS we have no private loader and cannot realiably alloc near client */
1678+
OPTION_DEFAULT(bool, reachable_client,
1679+
IF_STATIC_LIBRARY_ELSE(false, IF_MACOS_ELSE(false, true)),
16781680
"guarantee that clients are reachable from the code cache.")
16791681
#endif
16801682
/* XXX i#3566: Support for W^X has some current limitations:

core/unix/signal.c

+12-2
Original file line numberDiff line numberDiff line change
@@ -7696,8 +7696,6 @@ os_forge_exception(app_pc target_pc, dr_exception_type_t type)
76967696
sigframe_rt_t *frame = (sigframe_rt_t *)frame_no_xstate;
76977697
int sig;
76987698
dr_where_am_i_t cur_whereami = dcontext->whereami;
7699-
kernel_ucontext_t *uc = get_ucontext_from_rt_frame(frame);
7700-
sigcontext_t *sc = SIGCXT_FROM_UCXT(uc);
77017699
switch (type) {
77027700
case ILLEGAL_INSTRUCTION_EXCEPTION: sig = SIGILL; break;
77037701
case UNREADABLE_MEMORY_EXECUTION_EXCEPTION: sig = SIGSEGV; break;
@@ -7715,6 +7713,18 @@ os_forge_exception(app_pc target_pc, dr_exception_type_t type)
77157713
* to a plain frame on delivery.
77167714
*/
77177715
memset(frame, 0, sizeof(*frame));
7716+
7717+
kernel_ucontext_t *uc = get_ucontext_from_rt_frame(frame);
7718+
#if defined(MACOS)
7719+
/* Since SIGCXT_FROM_UCXT just accesses the uc->uc_mcontext ptr field on
7720+
* macOS, sc will be NULL below if we do not initialize uc_mcontext first.
7721+
* On macOS, the uc_mcontext pointer always just points to the mcontext
7722+
* elsewhere in the frame.
7723+
*/
7724+
uc->IF_X64_ELSE(uc_mcontext64, uc_mcontext) = (void *)&frame->mc;
7725+
#endif
7726+
sigcontext_t *sc = SIGCXT_FROM_UCXT(uc);
7727+
77187728
frame->info.si_signo = sig;
77197729
/* Set si_code to match what would happen natively. We also need this to
77207730
* avoid the !is_sys_kill() check in record_pending_signal() to avoid an

suite/tests/CMakeLists.txt

+4-2
Original file line numberDiff line numberDiff line change
@@ -6191,13 +6191,15 @@ if (NOT ARM) # FIXME i#1551: fix bugs on ARM
61916191
endif ()
61926192
endif (NOT ARM)
61936193

6194-
if (X86 OR AARCH64) # FIXME i#1551: port asm to ARM
6194+
# FIXME i#1551: port asm to ARM
6195+
# FIXME i#5383: selfmod tests fail to build on macOS + ARM64
6196+
if (X86 OR AARCH64 AND NOT (APPLE AND AARCH64))
61956197
tobuild(security-common.selfmod2 security-common/selfmod2.c)
61966198
tochcon(security-common.selfmod2 textrel_shlib_t)
61976199

61986200
tobuild(security-common.selfmod security-common/selfmod.c)
61996201
tochcon(security-common.selfmod textrel_shlib_t)
6200-
endif (X86 OR AARCH64)
6202+
endif (X86 OR AARCH64 AND NOT (APPLE AND AARCH64))
62016203

62026204
if (X86) # FIXME i#1551, i#1569: port asm to ARM and AArch64
62036205
if (NOT APPLE) # XXX i#58: port test to MacOS

suite/tests/client-interface/cbr-retarget.c

+6-2
Original file line numberDiff line numberDiff line change
@@ -62,9 +62,13 @@ main(void)
6262
}
6363
;
6464
#elif defined(AARCH64)
65-
__asm("cbnz xzr, skip");
65+
__asm("cbnz xzr, 1f");
66+
# ifdef MACOS
67+
__asm("bl _foo");
68+
# else
6669
__asm("bl foo");
67-
__asm("skip:");
70+
# endif
71+
__asm("1:");
6872
#else
6973
__asm("movl $0x0, %ecx");
7074
__asm("cmp $0x0, %ecx");

suite/tests/client-interface/strace.dll.c

+3-1
Original file line numberDiff line numberDiff line change
@@ -349,7 +349,9 @@ event_post_syscall(void *drcontext, int sysnum)
349349
static int
350350
get_write_sysnum(void)
351351
{
352-
#ifdef UNIX
352+
#if defined(MACOS)
353+
return SYS_write_nocancel;
354+
#elif defined(UNIX)
353355
return SYS_write;
354356
#else
355357
byte *entry;

suite/tests/linux/mangle_pauth.c

+6-3
Original file line numberDiff line numberDiff line change
@@ -109,8 +109,10 @@ handle_signal(int signal, siginfo_t *siginfo, ucontext_t *ucxt)
109109
* We need to strip the PAC from from the fault address to canonicalize it and
110110
* compare it to the expected branch target address.
111111
*/
112-
const uintptr_t fault_pc = strip_pac(ucxt->uc_mcontext.pc);
113-
LOG(" ucxt->uc_mcontext.pc = " PFX "\n", ucxt->uc_mcontext.pc);
112+
const uintptr_t pc =
113+
IF_MACOS_ELSE(ucxt->uc_mcontext->__ss.__pc, ucxt->uc_mcontext.pc);
114+
const uintptr_t fault_pc = strip_pac(pc);
115+
LOG(" ucxt->uc_mcontext.pc = " PFX "\n", pc);
114116
LOG(" fault_pc = " PFX "\n", fault_pc);
115117
LOG(" branch_target_addr = " PFX "\n", branch_target_addr);
116118
if (fault_pc == branch_target_addr)
@@ -123,7 +125,8 @@ handle_signal(int signal, siginfo_t *siginfo, ucontext_t *ucxt)
123125
/* CPU has FEAT_FPACCOMBINE so the branch instruction generated an authentication
124126
* failure exception and the fault PC should match the branch instruction address.
125127
*/
126-
const uintptr_t fault_pc = ucxt->uc_mcontext.pc;
128+
const uintptr_t fault_pc =
129+
IF_MACOS_ELSE(ucxt->uc_mcontext->__ss.__pc, ucxt->uc_mcontext.pc);
127130
LOG(" fault_pc = " PFX "\n", fault_pc);
128131
LOG(" branch_instr_addr = " PFX "\n", branch_instr_addr);
129132
if (fault_pc == branch_instr_addr)

suite/tests/tools.c

+5-1
Original file line numberDiff line numberDiff line change
@@ -526,6 +526,9 @@ set_check_signal_mask(sigset_t *mask, sigset_t *returned_mask)
526526
void
527527
dump_ucontext(ucontext_t *ucxt, bool is_sve, int vl_bytes)
528528
{
529+
# ifdef MACOS
530+
assert(false); /* NYI */
531+
# else
529532
struct _aarch64_ctx *head = (struct _aarch64_ctx *)(ucxt->uc_mcontext.RESERVED);
530533
assert(head->magic == FPSIMD_MAGIC);
531534
assert(head->size == sizeof(struct fpsimd_context));
@@ -541,7 +544,7 @@ dump_ucontext(ucontext_t *ucxt, bool is_sve, int vl_bytes)
541544
}
542545
print("\n");
543546

544-
# ifndef DR_HOST_NOT_TARGET
547+
# ifndef DR_HOST_NOT_TARGET
545548
if (is_sve) {
546549
size_t offset = sizeof(struct fpsimd_context);
547550
struct _aarch64_ctx *next_head =
@@ -621,6 +624,7 @@ dump_ucontext(ucontext_t *ucxt, bool is_sve, int vl_bytes)
621624
next_head = (struct _aarch64_ctx *)(ucxt->uc_mcontext.RESERVED + offset);
622625
}
623626
}
627+
# endif
624628
# endif
625629
}
626630
# endif

0 commit comments

Comments
 (0)