Skip to content
Snippets Groups Projects
Commit 54464f10 authored by Florian Angerer's avatar Florian Angerer
Browse files

Fix: Disable R_ReleaseObject in shutdown phase to prevent SIGSEGV.

parent 82d812a9
No related branches found
No related tags found
No related merge requests found
Showing
with 125 additions and 4 deletions
...@@ -223,6 +223,19 @@ public class TruffleLLVM_Base implements BaseRFFI { ...@@ -223,6 +223,19 @@ public class TruffleLLVM_Base implements BaseRFFI {
} }
} }
private static class TruffleLLVM_SetShutdownFlagNode extends TruffleLLVM_DownCallNode implements SetShutdownFlagNode {
@Override
protected NativeFunction getFunction() {
return NativeFunction.set_shutdown_phase;
}
@Override
public void execute(boolean value) {
call(value ? 1 : 0);
}
}
@Override @Override
public GetpidNode createGetpidNode() { public GetpidNode createGetpidNode() {
return new TruffleLLVM_GetpidNode(); return new TruffleLLVM_GetpidNode();
...@@ -272,4 +285,9 @@ public class TruffleLLVM_Base implements BaseRFFI { ...@@ -272,4 +285,9 @@ public class TruffleLLVM_Base implements BaseRFFI {
public GlobNode createGlobNode() { public GlobNode createGlobNode() {
return new TruffleLLVM_GlobNode(); return new TruffleLLVM_GlobNode();
} }
@Override
public SetShutdownFlagNode createSetShutdownFlagNode() {
return new TruffleLLVM_SetShutdownFlagNode();
}
} }
...@@ -200,4 +200,17 @@ public class Managed_Base implements BaseRFFI { ...@@ -200,4 +200,17 @@ public class Managed_Base implements BaseRFFI {
public GlobNode createGlobNode() { public GlobNode createGlobNode() {
return null; return null;
} }
private static final class ManagedSetShutdownFlagNode extends Node implements SetShutdownFlagNode {
@Override
public void execute(boolean value) {
// do nothing
}
}
@Override
public SetShutdownFlagNode createSetShutdownFlagNode() {
return new ManagedSetShutdownFlagNode();
}
} }
...@@ -27,6 +27,7 @@ import java.util.ArrayList; ...@@ -27,6 +27,7 @@ import java.util.ArrayList;
import com.oracle.truffle.r.ffi.impl.common.LibPaths; import com.oracle.truffle.r.ffi.impl.common.LibPaths;
import com.oracle.truffle.r.runtime.context.RContext; import com.oracle.truffle.r.runtime.context.RContext;
import com.oracle.truffle.r.runtime.context.RContext.ContextState; import com.oracle.truffle.r.runtime.context.RContext.ContextState;
import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
import com.oracle.truffle.r.runtime.ffi.DLL; import com.oracle.truffle.r.runtime.ffi.DLL;
import com.oracle.truffle.r.runtime.ffi.DLLRFFI; import com.oracle.truffle.r.runtime.ffi.DLLRFFI;
import com.oracle.truffle.r.runtime.ffi.RFFIContext; import com.oracle.truffle.r.runtime.ffi.RFFIContext;
...@@ -49,9 +50,20 @@ class NFIContext extends RFFIContext { ...@@ -49,9 +50,20 @@ class NFIContext extends RFFIContext {
// force initialization of NFI // force initialization of NFI
DLLRFFI.DLOpenRootNode.create(context).call(librffiPath, false, false); DLLRFFI.DLOpenRootNode.create(context).call(librffiPath, false, false);
} }
if (RContext.getContextCnt() == 0) {
BaseRFFI.SetShutdownFlagRootNode.create().getCallTarget().call(false);
}
return this; return this;
} }
@Override
public void beforeDispose(RContext context) {
if (RContext.getContextCnt() == 1) {
BaseRFFI.SetShutdownFlagRootNode.create().getCallTarget().call(true);
}
super.beforeDispose(context);
}
@Override @Override
public void afterDowncall() { public void afterDowncall() {
for (Long ptr : transientAllocations) { for (Long ptr : transientAllocations) {
......
...@@ -78,7 +78,8 @@ public enum NativeFunction { ...@@ -78,7 +78,8 @@ public enum NativeFunction {
dqrls("([double], sint32, sint32, [double], sint32, double, [double], [double], [double], [sint32], [sint32], [double], [double]): void", "call_misc_"), dqrls("([double], sint32, sint32, [double], sint32, double, [double], [double], [double], [sint32], [sint32], [double], [double]): void", "call_misc_"),
// stats // stats
fft_factor("(sint32, [sint32], [sint32]): void", TruffleNFI_Utils::lookupAndBindStats), fft_factor("(sint32, [sint32], [sint32]): void", TruffleNFI_Utils::lookupAndBindStats),
fft_work("([double], sint32, sint32, sint32, sint32, [double], [sint32]): sint32", TruffleNFI_Utils::lookupAndBindStats); fft_work("([double], sint32, sint32, sint32, sint32, [double], [sint32]): sint32", TruffleNFI_Utils::lookupAndBindStats),
set_shutdown_phase("(uint8): void", TruffleNFI_Utils::lookupAndBind);
private final int argumentCount; private final int argumentCount;
private final String signature; private final String signature;
......
...@@ -213,6 +213,19 @@ public class TruffleNFI_Base implements BaseRFFI { ...@@ -213,6 +213,19 @@ public class TruffleNFI_Base implements BaseRFFI {
} }
} }
private static class TruffleNFI_SetShutdownFlagNode extends TruffleNFI_DownCallNode implements SetShutdownFlagNode {
@Override
protected NativeFunction getFunction() {
return NativeFunction.set_shutdown_phase;
}
@Override
public void execute(boolean value) {
call(value ? 1 : 0);
}
}
@Override @Override
public GetpidNode createGetpidNode() { public GetpidNode createGetpidNode() {
return new TruffleNFI_GetpidNode(); return new TruffleNFI_GetpidNode();
...@@ -262,4 +275,9 @@ public class TruffleNFI_Base implements BaseRFFI { ...@@ -262,4 +275,9 @@ public class TruffleNFI_Base implements BaseRFFI {
public GlobNode createGlobNode() { public GlobNode createGlobNode() {
return new TruffleNFI_GlobNode(); return new TruffleNFI_GlobNode();
} }
@Override
public SetShutdownFlagNode createSetShutdownFlagNode() {
return new TruffleNFI_SetShutdownFlagNode();
}
} }
...@@ -828,8 +828,8 @@ void SET_PRCODE(SEXP x, SEXP v) { ...@@ -828,8 +828,8 @@ void SET_PRCODE(SEXP x, SEXP v) {
} }
int TRUELENGTH(SEXP x) { int TRUELENGTH(SEXP x) {
TRACE0(); TRACE(TARGp, x);
unimplemented("unimplemented"); // TODO do not throw an error for now
return 0; return 0;
} }
...@@ -1342,7 +1342,9 @@ void R_PreserveObject(SEXP x) { ...@@ -1342,7 +1342,9 @@ void R_PreserveObject(SEXP x) {
void R_ReleaseObject(SEXP x) { void R_ReleaseObject(SEXP x) {
TRACE0(); TRACE0();
((call_R_ReleaseObject) callbacks[R_ReleaseObject_x])(x); if(!is_shutdown_phase()) {
((call_R_ReleaseObject) callbacks[R_ReleaseObject_x])(x);
}
} }
void R_dot_Last(void) { void R_dot_Last(void) {
......
...@@ -41,6 +41,8 @@ void init_utils(TruffleEnv *env) { ...@@ -41,6 +41,8 @@ void init_utils(TruffleEnv *env) {
} }
} }
static unsigned char shutdown_phase = 0;
#define ERROR_JMP_BUF_STACK_SIZE 32 #define ERROR_JMP_BUF_STACK_SIZE 32
static jmp_buf *callErrorJmpBufStack[ERROR_JMP_BUF_STACK_SIZE]; static jmp_buf *callErrorJmpBufStack[ERROR_JMP_BUF_STACK_SIZE];
static int callErrorJmpBufStackIndex = 0; static int callErrorJmpBufStackIndex = 0;
...@@ -79,6 +81,14 @@ static void popJmpBuf() { ...@@ -79,6 +81,14 @@ static void popJmpBuf() {
popJmpBuf(); \ popJmpBuf(); \
return result; return result;
void set_shutdown_phase(unsigned char value) {
shutdown_phase = value;
}
int is_shutdown_phase() {
return shutdown_phase;
}
void dot_call_void0(callvoid0func fun) { void dot_call_void0(callvoid0func fun) {
DO_CALL_VOID(fun()); DO_CALL_VOID(fun());
} }
......
...@@ -53,6 +53,10 @@ typedef void (*callvoid0func)(void); ...@@ -53,6 +53,10 @@ typedef void (*callvoid0func)(void);
typedef void (*callvoid1func)(SEXP arg1); typedef void (*callvoid1func)(SEXP arg1);
void set_shutdown_phase(unsigned char value);
int is_shutdown_phase();
void dot_call_void0(callvoid0func); void dot_call_void0(callvoid0func);
void dot_call_void1(callvoid1func, SEXP); void dot_call_void1(callvoid1func, SEXP);
......
...@@ -48,6 +48,7 @@ import java.util.TimeZone; ...@@ -48,6 +48,7 @@ import java.util.TimeZone;
import java.util.WeakHashMap; import java.util.WeakHashMap;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Executor; import java.util.concurrent.Executor;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Supplier; import java.util.function.Supplier;
import com.oracle.truffle.api.Assumption; import com.oracle.truffle.api.Assumption;
...@@ -292,6 +293,8 @@ public final class RContext implements RTruffleObject { ...@@ -292,6 +293,8 @@ public final class RContext implements RTruffleObject {
*/ */
private static boolean embedded; private static boolean embedded;
private static final AtomicInteger CONTEXT_CNT = new AtomicInteger(0);
/* /*
* Workarounds to finesse project circularities between runtime/nodes. * Workarounds to finesse project circularities between runtime/nodes.
*/ */
...@@ -372,6 +375,10 @@ public final class RContext implements RTruffleObject { ...@@ -372,6 +375,10 @@ public final class RContext implements RTruffleObject {
return embedded; return embedded;
} }
public static int getContextCnt() {
return CONTEXT_CNT.get();
}
/** /**
* Sets the fields that do not depend on complex initialization. * Sets the fields that do not depend on complex initialization.
* *
...@@ -523,6 +530,7 @@ public final class RContext implements RTruffleObject { ...@@ -523,6 +530,7 @@ public final class RContext implements RTruffleObject {
if (initial && !embedded) { if (initial && !embedded) {
initialContextInitialized = true; initialContextInitialized = true;
} }
CONTEXT_CNT.incrementAndGet();
return this; return this;
} }
...@@ -574,6 +582,7 @@ public final class RContext implements RTruffleObject { ...@@ -574,6 +582,7 @@ public final class RContext implements RTruffleObject {
if (contextKind == ContextKind.SHARE_PARENT_RW) { if (contextKind == ContextKind.SHARE_PARENT_RW) {
parentContext.sharedChild = null; parentContext.sharedChild = null;
} }
CONTEXT_CNT.decrementAndGet();
state = EnumSet.of(State.DISPOSED); state = EnumSet.of(State.DISPOSED);
this.allocationReporter.removePropertyChangeListener(ALLOCATION_ACTIVATION_LISTENER); this.allocationReporter.removePropertyChangeListener(ALLOCATION_ACTIVATION_LISTENER);
......
...@@ -27,6 +27,7 @@ import java.util.ArrayList; ...@@ -27,6 +27,7 @@ import java.util.ArrayList;
import com.oracle.truffle.api.frame.VirtualFrame; import com.oracle.truffle.api.frame.VirtualFrame;
import com.oracle.truffle.api.nodes.NodeInterface; import com.oracle.truffle.api.nodes.NodeInterface;
import com.oracle.truffle.r.runtime.data.RNull;
/** /**
* A statically typed interface to exactly those native functions required by the R {@code base} * A statically typed interface to exactly those native functions required by the R {@code base}
...@@ -163,6 +164,14 @@ public interface BaseRFFI { ...@@ -163,6 +164,14 @@ public interface BaseRFFI {
} }
} }
interface SetShutdownFlagNode extends NodeInterface {
void execute(boolean value);
static SetShutdownFlagNode create() {
return RFFIFactory.getBaseRFFI().createSetShutdownFlagNode();
}
}
/* /*
* The RFFI implementation influences exactly what subclass of the above nodes is created. Each * The RFFI implementation influences exactly what subclass of the above nodes is created. Each
* implementation must therefore, implement these methods that are called by the associated * implementation must therefore, implement these methods that are called by the associated
...@@ -189,6 +198,8 @@ public interface BaseRFFI { ...@@ -189,6 +198,8 @@ public interface BaseRFFI {
GlobNode createGlobNode(); GlobNode createGlobNode();
SetShutdownFlagNode createSetShutdownFlagNode();
/* /*
* Some functions are called from non-Truffle contexts, which requires a RootNode * Some functions are called from non-Truffle contexts, which requires a RootNode
*/ */
...@@ -273,4 +284,27 @@ public interface BaseRFFI { ...@@ -273,4 +284,27 @@ public interface BaseRFFI {
return unameRootNode; return unameRootNode;
} }
} }
final class SetShutdownFlagRootNode extends RFFIRootNode<SetShutdownFlagNode> {
protected SetShutdownFlagRootNode() {
super(RFFIFactory.getBaseRFFI().createSetShutdownFlagNode());
}
private static SetShutdownFlagRootNode setShutdownFlagRootNode;
@Override
public Object execute(VirtualFrame frame) {
Object[] args = frame.getArguments();
rffiNode.execute((boolean) args[0]);
return RNull.instance;
}
public static SetShutdownFlagRootNode create() {
if (setShutdownFlagRootNode == null) {
setShutdownFlagRootNode = new SetShutdownFlagRootNode();
}
return setShutdownFlagRootNode;
}
}
} }
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment