8189429: SA: MacOSX: Replace the deprecated PT_ATTACH with PT_ATTACHEXC

Avoid the waitpid() and receive, handle and reply to the incoming Mach exception message obtained with PT_ATTACHEXC.

Reviewed-by: sballal, erikj, gziemski, rwestberg, dholmes, dcubed, poonam, dsamersoff
This commit is contained in:
Jini George 2018-06-27 20:04:31 +05:30
parent 4098f2560a
commit 93314be3c2
6 changed files with 330 additions and 111 deletions

View File

@ -1,5 +1,5 @@
# #
# Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. # Copyright (c) 2017, 2018, Oracle and/or its affiliates. All rights reserved.
# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
# #
# This code is free software; you can redistribute it and/or modify it # This code is free software; you can redistribute it and/or modify it
@ -147,6 +147,7 @@ GZIP := gzip
HEAD := head HEAD := head
LS := ls LS := ls
LN := ln LN := ln
MIG := mig
MKDIR := mkdir MKDIR := mkdir
MV := mv MV := mv
NAWK := nawk NAWK := nawk

View File

@ -1208,6 +1208,7 @@ AC_DEFUN_ONCE([BASIC_SETUP_COMPLEX_TOOLS],
if test "x$OPENJDK_TARGET_OS" = "xmacosx"; then if test "x$OPENJDK_TARGET_OS" = "xmacosx"; then
BASIC_REQUIRE_PROGS(DSYMUTIL, dsymutil) BASIC_REQUIRE_PROGS(DSYMUTIL, dsymutil)
BASIC_REQUIRE_PROGS(MIG, mig)
BASIC_REQUIRE_PROGS(XATTR, xattr) BASIC_REQUIRE_PROGS(XATTR, xattr)
BASIC_PATH_PROGS(CODESIGN, codesign) BASIC_PATH_PROGS(CODESIGN, codesign)
if test "x$CODESIGN" != "x"; then if test "x$CODESIGN" != "x"; then

View File

@ -675,6 +675,7 @@ GZIP:=@GZIP@
HEAD:=@HEAD@ HEAD:=@HEAD@
LS:=@LS@ LS:=@LS@
LN:=@LN@ LN:=@LN@
MIG:=@MIG@
MKDIR:=@MKDIR@ MKDIR:=@MKDIR@
MV:=@MV@ MV:=@MV@
NAWK:=@NAWK@ NAWK:=@NAWK@

View File

@ -1,5 +1,5 @@
# #
# Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. # Copyright (c) 2015, 2018, Oracle and/or its affiliates. All rights reserved.
# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
# #
# This code is free software; you can redistribute it and/or modify it # This code is free software; you can redistribute it and/or modify it
@ -42,6 +42,21 @@ TARGETS += $(SA_PROPERTIES)
################################################################################ ################################################################################
ifeq ($(OPENJDK_TARGET_OS), macosx)
MIG_OUTPUT_DIR := $(SUPPORT_OUTPUTDIR)/gensrc/jdk.hotspot.agent
MACH_EXC_HEADER := $(MIG_OUTPUT_DIR)/mach_exc.h
MACH_EXC_USER := $(MIG_OUTPUT_DIR)/mach_excUser.c
MACH_EXC_SERVER := $(MIG_OUTPUT_DIR)/mach_excServer.c
$(MACH_EXC_SERVER): $(SDKROOT)/usr/include/mach/mach_exc.defs
$(MIG) -isysroot $(SDKROOT) -server $@ -user $(MACH_EXC_USER) \
-header $(MACH_EXC_HEADER) $(SDKROOT)/usr/include/mach/mach_exc.defs
TARGETS += $(MACH_EXC_SERVER)
endif
################################################################################
all: $(TARGETS) all: $(TARGETS)
.PHONY: all default .PHONY: all default

View File

@ -38,6 +38,7 @@ else ifeq ($(OPENJDK_TARGET_OS), solaris)
else ifeq ($(OPENJDK_TARGET_OS), macosx) else ifeq ($(OPENJDK_TARGET_OS), macosx)
SA_CFLAGS := -Damd64 -D_GNU_SOURCE -mno-omit-leaf-frame-pointer \ SA_CFLAGS := -Damd64 -D_GNU_SOURCE -mno-omit-leaf-frame-pointer \
-mstack-alignment=16 -fPIC -mstack-alignment=16 -fPIC
LIBSA_EXTRA_SRC := $(SUPPORT_OUTPUTDIR)/gensrc/jdk.hotspot.agent
else ifeq ($(OPENJDK_TARGET_OS), windows) else ifeq ($(OPENJDK_TARGET_OS), windows)
SA_CFLAGS := -D_WINDOWS -D_DEBUG -D_CONSOLE -D_MBCS -EHsc SA_CFLAGS := -D_WINDOWS -D_DEBUG -D_CONSOLE -D_MBCS -EHsc
ifeq ($(OPENJDK_TARGET_CPU), x86_64) ifeq ($(OPENJDK_TARGET_CPU), x86_64)
@ -57,6 +58,7 @@ $(eval $(call SetupJdkLibrary, BUILD_LIBSA, \
DISABLED_WARNINGS_CXX_solstudio := truncwarn unknownpragma, \ DISABLED_WARNINGS_CXX_solstudio := truncwarn unknownpragma, \
CFLAGS := $(CFLAGS_JDKLIB) $(SA_CFLAGS), \ CFLAGS := $(CFLAGS_JDKLIB) $(SA_CFLAGS), \
CXXFLAGS := $(CXXFLAGS_JDKLIB) $(SA_CFLAGS) $(SA_CXXFLAGS), \ CXXFLAGS := $(CXXFLAGS_JDKLIB) $(SA_CFLAGS) $(SA_CXXFLAGS), \
EXTRA_SRC := $(LIBSA_EXTRA_SRC), \
LDFLAGS := $(LDFLAGS_JDKLIB) $(SA_LDFLAGS), \ LDFLAGS := $(LDFLAGS_JDKLIB) $(SA_LDFLAGS), \
LIBS_linux := -lthread_db $(LIBDL), \ LIBS_linux := -lthread_db $(LIBDL), \
LIBS_solaris := -ldl -ldemangle -lthread -lproc, \ LIBS_solaris := -ldl -ldemangle -lthread -lproc, \

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2002, 2017, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2002, 2018, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -33,6 +33,7 @@
#import <mach/mach_types.h> #import <mach/mach_types.h>
#import <sys/sysctl.h> #import <sys/sysctl.h>
#import <stdio.h> #import <stdio.h>
#import <string.h>
#import <stdarg.h> #import <stdarg.h>
#import <stdlib.h> #import <stdlib.h>
#import <strings.h> #import <strings.h>
@ -69,6 +70,63 @@ static jmethodID getJavaThreadsInfo_ID = 0;
// indicator if thread id (lwpid_t) was set // indicator if thread id (lwpid_t) was set
static bool _threads_filled = false; static bool _threads_filled = false;
// mach_exc_server defined in the generated mach_excServer.c
extern boolean_t mach_exc_server(mach_msg_header_t *input_msg_hdr,
mach_msg_header_t *output_msg_hdr);
kern_return_t catch_mach_exception_raise(
mach_port_t exception_port, mach_port_t thread,
mach_port_t task, exception_type_t exception,
mach_exception_data_t code,
mach_msg_type_number_t code_cnt);
kern_return_t catch_mach_exception_raise_state(
mach_port_t exception_port, exception_type_t exception,
const mach_exception_data_t code, mach_msg_type_number_t code_cnt,
int *flavor, const thread_state_t old_state,
mach_msg_type_number_t old_state_cnt, thread_state_t new_state,
mach_msg_type_number_t *new_state_cnt);
kern_return_t catch_mach_exception_raise_state_identity(
mach_port_t exception_port, mach_port_t thread, mach_port_t task,
exception_type_t exception, mach_exception_data_t code,
mach_msg_type_number_t code_cnt, int *flavor, thread_state_t old_state,
mach_msg_type_number_t old_state_cnt, thread_state_t new_state,
mach_msg_type_number_t *new_state_cnt);
static struct exception_saved_state {
exception_mask_t saved_masks[EXC_TYPES_COUNT];
mach_port_t saved_ports[EXC_TYPES_COUNT];
exception_behavior_t saved_behaviors[EXC_TYPES_COUNT];
thread_state_flavor_t saved_flavors[EXC_TYPES_COUNT];
mach_msg_type_number_t saved_exception_types_count;
} exception_saved_state;
static mach_port_t tgt_exception_port;
// Mirrors __Reply__mach_exception_raise_t generated in mach_excServer.c
static struct rep_msg {
mach_msg_header_t header;
NDR_record_t ndr;
kern_return_t ret_code;
} rep_msg;
// Mirrors __Request__mach_exception_raise_t generated in mach_excServer.c
// with a large trailing pad to avoid MACH_MSG_RCV_TOO_LARGE
static struct exc_msg {
mach_msg_header_t header;
// start of the kernel processed data
mach_msg_body_t msgh_body;
mach_msg_port_descriptor_t thread;
mach_msg_port_descriptor_t task;
// end of the kernel processed data
NDR_record_t ndr;
exception_type_t exception;
mach_msg_type_number_t code_cnt;
mach_exception_data_t code; // an array of int64_t
char pad[512];
} exc_msg;
static void putSymbolicator(JNIEnv *env, jobject this_obj, id symbolicator) { static void putSymbolicator(JNIEnv *env, jobject this_obj, id symbolicator) {
(*env)->SetLongField(env, this_obj, symbolicatorID, (jlong)(intptr_t)symbolicator); (*env)->SetLongField(env, this_obj, symbolicatorID, (jlong)(intptr_t)symbolicator);
} }
@ -630,78 +688,96 @@ Java_sun_jvm_hotspot_debugger_macosx_MacOSXDebuggerLocal_translateTID0(
return (jint) usable_tid; return (jint) usable_tid;
} }
// attach to a process/thread specified by "pid"
static bool ptrace_attach(pid_t pid) {
errno = 0;
ptrace(PT_ATTACHEXC, pid, 0, 0);
static bool ptrace_continue(pid_t pid, int signal) { if (errno != 0) {
// pass the signal to the process so we don't swallow it print_error("ptrace_attach: ptrace(PT_ATTACHEXC,...) failed: %s", strerror(errno));
int res;
if ((res = ptrace(PT_CONTINUE, pid, (caddr_t)1, signal)) < 0) {
print_error("attach: ptrace(PT_CONTINUE, %d) failed with %d\n", pid, res);
return false; return false;
} }
return true; return true;
} }
// waits until the ATTACH has stopped the process kern_return_t catch_mach_exception_raise(
// by signal SIGSTOP mach_port_t exception_port, mach_port_t thread_port, mach_port_t task_port,
static bool ptrace_waitpid(pid_t pid) { exception_type_t exception_type, mach_exception_data_t codes,
int ret; mach_msg_type_number_t num_codes) {
int status;
while (true) { print_debug("catch_mach_exception_raise: Exception port = %d thread_port = %d "
// Wait for debuggee to stop. "task port %d exc type = %d num_codes %d\n",
ret = waitpid(pid, &status, 0); exception_port, thread_port, task_port, exception_type, num_codes);
if (ret >= 0) {
if (WIFSTOPPED(status)) { // This message should denote a Unix soft signal, with
// Any signal will stop the thread, make sure it is SIGSTOP. Otherwise SIGSTOP // 1. the exception type = EXC_SOFTWARE
// will still be pending and delivered when the process is DETACHED and the process // 2. codes[0] (which is the code) = EXC_SOFT_SIGNAL
// will go to sleep. // 3. codes[1] (which is the sub-code) = SIGSTOP
if (WSTOPSIG(status) == SIGSTOP) { if (!(exception_type == EXC_SOFTWARE &&
// Debuggee stopped by SIGSTOP. codes[0] == EXC_SOFT_SIGNAL &&
return true; codes[num_codes -1] == SIGSTOP)) {
} print_error("catch_mach_exception_raise: Message doesn't denote a Unix "
if (!ptrace_continue(pid, WSTOPSIG(status))) { "soft signal. exception_type = %d, codes[0] = %d, "
print_error("attach: Failed to correctly attach to VM. VM might HANG! [PTRACE_CONT failed, stopped by %d]\n", WSTOPSIG(status)); "codes[num_codes -1] = %d, num_codes = %d\n",
return false; exception_type, codes[0], codes[num_codes - 1], num_codes);
} return MACH_RCV_INVALID_TYPE;
} else {
print_error("attach: waitpid(): Child process exited/terminated (status = 0x%x)\n", status);
return false;
}
} else {
switch (errno) {
case EINTR:
continue;
break;
case ECHILD:
print_error("attach: waitpid() failed. Child process pid (%d) does not exist \n", pid);
break;
case EINVAL:
print_error("attach: waitpid() failed. Invalid options argument.\n");
break;
default:
print_error("attach: waitpid() failed. Unexpected error %d\n",errno);
break;
}
return false;
}
} }
return KERN_SUCCESS;
} }
// attach to a process/thread specified by "pid" kern_return_t catch_mach_exception_raise_state(
static bool ptrace_attach(pid_t pid) { mach_port_t exception_port, exception_type_t exception, const mach_exception_data_t code,
int res; mach_msg_type_number_t code_cnt, int *flavor, const thread_state_t old_state,
#ifdef __clang__ mach_msg_type_number_t old_state_cnt, thread_state_t new_state,
#pragma clang diagnostic push mach_msg_type_number_t *new_state_cnt) {
#pragma clang diagnostic ignored "-Wdeprecated-declarations" return MACH_RCV_INVALID_TYPE;
#endif }
if ((res = ptrace(PT_ATTACH, pid, 0, 0)) < 0) {
print_error("ptrace(PT_ATTACH, %d) failed with %d\n", pid, res);
#ifdef __clang__ kern_return_t catch_mach_exception_raise_state_identity(
#pragma clang diagnostic pop mach_port_t exception_port, mach_port_t thread, mach_port_t task,
#endif exception_type_t exception, mach_exception_data_t code,
mach_msg_type_number_t code_cnt, int *flavor,
thread_state_t old_state, mach_msg_type_number_t old_state_cnt,
thread_state_t new_state, mach_msg_type_number_t *new_state_cnt) {
return MACH_RCV_INVALID_TYPE;
}
// wait to receive an exception message
static bool wait_for_exception() {
kern_return_t result;
result = mach_msg(&exc_msg.header,
MACH_RCV_MSG,
0,
sizeof(exc_msg),
tgt_exception_port,
MACH_MSG_TIMEOUT_NONE,
MACH_PORT_NULL);
if (result != MACH_MSG_SUCCESS) {
print_error("attach: wait_for_exception: mach_msg() failed: '%s' (%d)\n",
mach_error_string(result), result);
return false; return false;
} else {
return ptrace_waitpid(pid);
} }
if (mach_exc_server(&exc_msg.header, &rep_msg.header) == false ||
rep_msg.ret_code != KERN_SUCCESS) {
print_error("attach: wait_for_exception: mach_exc_server failure\n");
if (rep_msg.ret_code != KERN_SUCCESS) {
print_error("catch_mach_exception_raise() failed '%s' (%d)\n",
mach_error_string(rep_msg.ret_code), rep_msg.ret_code);
}
return false;
}
print_debug("reply msg from mach_exc_server: (msg->{bits = %#x, size = %u, "
"remote_port = %#x, local_port = %#x, reserved = 0x%x, id = 0x%x},)",
rep_msg.header.msgh_bits, rep_msg.header.msgh_size,
rep_msg.header.msgh_remote_port, rep_msg.header.msgh_local_port,
rep_msg.header.msgh_reserved, rep_msg.header.msgh_id);
return true;
} }
/* /*
@ -719,18 +795,97 @@ JNF_COCOA_ENTER(env);
kern_return_t result; kern_return_t result;
task_t gTask = 0; task_t gTask = 0;
result = task_for_pid(mach_task_self(), jpid, &gTask); result = task_for_pid(mach_task_self(), jpid, &gTask);
if (result != KERN_SUCCESS) { if (result != KERN_SUCCESS) {
print_error("attach: task_for_pid(%d) failed: '%s' (%d)\n", (int)jpid, mach_error_string(result), result); print_error("attach: task_for_pid(%d) failed: '%s' (%d)\n", (int)jpid, mach_error_string(result), result);
THROW_NEW_DEBUGGER_EXCEPTION("Can't attach to the process. Could be caused by an incorrect pid or lack of privileges."); THROW_NEW_DEBUGGER_EXCEPTION(
"Can't attach to the process. Could be caused by an incorrect pid or lack of privileges.");
} }
putTask(env, this_obj, gTask); putTask(env, this_obj, gTask);
// Allocate an exception port.
result = mach_port_allocate(mach_task_self(),
MACH_PORT_RIGHT_RECEIVE,
&tgt_exception_port);
if (result != KERN_SUCCESS) {
print_error("attach: mach_port_allocate() for tgt_exception_port failed: '%s' (%d)\n",
mach_error_string(result), result);
THROW_NEW_DEBUGGER_EXCEPTION(
"Can't attach to the process. Couldn't allocate an exception port.");
}
// Enable the new exception port to send messages.
result = mach_port_insert_right (mach_task_self(),
tgt_exception_port,
tgt_exception_port,
MACH_MSG_TYPE_MAKE_SEND);
if (result != KERN_SUCCESS) {
print_error("attach: mach_port_insert_right() failed for port 0x%x: '%s' (%d)\n",
tgt_exception_port, mach_error_string(result), result);
THROW_NEW_DEBUGGER_EXCEPTION(
"Can't attach to the process. Couldn't add send privileges to the exception port.");
}
// Save the existing original exception ports registered with the target
// process (for later restoration while detaching from the process).
result = task_get_exception_ports(gTask,
EXC_MASK_ALL,
exception_saved_state.saved_masks,
&exception_saved_state.saved_exception_types_count,
exception_saved_state.saved_ports,
exception_saved_state.saved_behaviors,
exception_saved_state.saved_flavors);
if (result != KERN_SUCCESS) {
print_error("attach: task_get_exception_ports() failed: '%s' (%d)\n",
mach_error_string(result), result);
THROW_NEW_DEBUGGER_EXCEPTION(
"Can't attach to the process. Could not get the target exception ports.");
}
// register the exception port to be used for all future exceptions with the
// target process.
result = task_set_exception_ports(gTask,
EXC_MASK_ALL,
tgt_exception_port,
EXCEPTION_DEFAULT | MACH_EXCEPTION_CODES,
THREAD_STATE_NONE);
if (result != KERN_SUCCESS) {
print_error("attach: task_set_exception_ports() failed -- port 0x%x: '%s' (%d)\n",
tgt_exception_port, mach_error_string(result), result);
mach_port_deallocate(mach_task_self(), gTask);
mach_port_deallocate(mach_task_self(), tgt_exception_port);
THROW_NEW_DEBUGGER_EXCEPTION(
"Can't attach to the process. Could not register the exception port "
"with the target process.");
}
// use ptrace to stop the process // use ptrace to stop the process
// on os x, ptrace only needs to be called on the process, not the individual threads // on os x, ptrace only needs to be called on the process, not the individual threads
if (ptrace_attach(jpid) != true) { if (ptrace_attach(jpid) != true) {
print_error("attach: ptrace failure in attaching to %d\n", (int)jpid);
mach_port_deallocate(mach_task_self(), gTask); mach_port_deallocate(mach_task_self(), gTask);
THROW_NEW_DEBUGGER_EXCEPTION("Can't attach to the process"); mach_port_deallocate(mach_task_self(), tgt_exception_port);
THROW_NEW_DEBUGGER_EXCEPTION("Can't ptrace attach to the process");
}
if (wait_for_exception() != true) {
mach_port_deallocate(mach_task_self(), gTask);
mach_port_deallocate(mach_task_self(), tgt_exception_port);
THROW_NEW_DEBUGGER_EXCEPTION(
"Can't attach to the process. Issues with reception of the exception message.");
}
// suspend all the threads in the task
result = task_suspend(gTask);
if (result != KERN_SUCCESS) {
print_error("attach: task_suspend() failed: '%s' (%d)\n",
mach_error_string(result), result);
mach_port_deallocate(mach_task_self(), gTask);
mach_port_deallocate(mach_task_self(), tgt_exception_port);
THROW_NEW_DEBUGGER_EXCEPTION("Can't attach. Unable to suspend all the threads in the task.");
} }
id symbolicator = nil; id symbolicator = nil;
@ -745,6 +900,8 @@ JNF_COCOA_ENTER(env);
putSymbolicator(env, this_obj, symbolicator); putSymbolicator(env, this_obj, symbolicator);
if (symbolicator == nil) { if (symbolicator == nil) {
mach_port_deallocate(mach_task_self(), gTask);
mach_port_deallocate(mach_task_self(), tgt_exception_port);
THROW_NEW_DEBUGGER_EXCEPTION("Can't attach symbolicator to the process"); THROW_NEW_DEBUGGER_EXCEPTION("Can't attach symbolicator to the process");
} }
@ -811,6 +968,19 @@ Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_attach0__Ljava_lang_String_2L
fillLoadObjects(env, this_obj, ph); fillLoadObjects(env, this_obj, ph);
} }
static void detach_cleanup(task_t gTask, JNIEnv *env, jobject this_obj, bool throw_exception) {
mach_port_deallocate(mach_task_self(), tgt_exception_port);
mach_port_deallocate(mach_task_self(), gTask);
id symbolicator = getSymbolicator(env, this_obj);
if (symbolicator != nil) {
CFRelease(symbolicator);
}
if (throw_exception) {
THROW_NEW_DEBUGGER_EXCEPTION("Cannot detach.");
}
}
/* /*
* Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal * Class: sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal
* Method: detach0 * Method: detach0
@ -827,26 +997,55 @@ Java_sun_jvm_hotspot_debugger_bsd_BsdDebuggerLocal_detach0(
return; return;
} }
JNF_COCOA_ENTER(env); JNF_COCOA_ENTER(env);
task_t gTask = getTask(env, this_obj);
// detach from the ptraced process causing it to resume execution task_t gTask = getTask(env, this_obj);
int pid; kern_return_t k_res = 0;
kern_return_t k_res;
k_res = pid_for_task(gTask, &pid); // Restore the pre-saved original exception ports registered with the target process
if (k_res != KERN_SUCCESS) { for (uint32_t i = 0; i < exception_saved_state.saved_exception_types_count; ++i) {
print_error("detach: pid_for_task(%d) failed (%d)\n", pid, k_res); k_res = task_set_exception_ports(gTask,
} exception_saved_state.saved_masks[i],
else { exception_saved_state.saved_ports[i],
int res = ptrace(PT_DETACH, pid, 0, 0); exception_saved_state.saved_behaviors[i],
if (res < 0) { exception_saved_state.saved_flavors[i]);
print_error("detach: ptrace(PT_DETACH, %d) failed (%d)\n", pid, res); if (k_res != KERN_SUCCESS) {
print_error("detach: task_set_exception_ports failed with %d while "
"restoring the target exception ports.\n", k_res);
detach_cleanup(gTask, env, this_obj, true);
} }
} }
mach_port_deallocate(mach_task_self(), gTask); // detach from the ptraced process causing it to resume execution
id symbolicator = getSymbolicator(env, this_obj); int pid;
if (symbolicator != nil) { k_res = pid_for_task(gTask, &pid);
CFRelease(symbolicator); if (k_res != KERN_SUCCESS) {
print_error("detach: pid_for_task(%d) failed (%d)\n", pid, k_res);
detach_cleanup(gTask, env, this_obj, true);
} }
else {
errno = 0;
ptrace(PT_DETACH, pid, (caddr_t)1, 0);
if (errno != 0) {
print_error("detach: ptrace(PT_DETACH,...) failed: %s", strerror(errno));
detach_cleanup(gTask, env, this_obj, true);
}
}
// reply to the previous exception message
k_res = mach_msg(&rep_msg.header,
MACH_SEND_MSG| MACH_SEND_INTERRUPT,
rep_msg.header.msgh_size,
0,
MACH_PORT_NULL,
MACH_MSG_TIMEOUT_NONE,
MACH_PORT_NULL);
if (k_res != MACH_MSG_SUCCESS) {
print_error("detach: mach_msg() for replying to pending exceptions failed: '%s' (%d)\n",
mach_error_string(k_res), k_res);
detach_cleanup(gTask, env, this_obj, true);
}
detach_cleanup(gTask, env, this_obj, false);
JNF_COCOA_EXIT(env); JNF_COCOA_EXIT(env);
} }