diff --git a/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/WindowDevice.java b/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/WindowDevice.java index 99140fc5bd74d5e31ae6076df1b53a58ada73878..63e453ec48a1f771882cf9bdfcac1588138a64f3 100644 --- a/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/WindowDevice.java +++ b/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/WindowDevice.java @@ -24,9 +24,14 @@ package com.oracle.truffle.r.library.fastrGrid; import com.oracle.truffle.r.library.fastrGrid.device.GridDevice; import com.oracle.truffle.r.library.fastrGrid.device.awt.JFrameDevice; +import com.oracle.truffle.r.nodes.function.PromiseHelperNode; +import com.oracle.truffle.r.runtime.RCaller; import com.oracle.truffle.r.runtime.RError; import com.oracle.truffle.r.runtime.RError.Message; import com.oracle.truffle.r.runtime.context.RContext; +import com.oracle.truffle.r.runtime.data.RFunction; +import com.oracle.truffle.r.runtime.data.RPromise; +import com.oracle.truffle.r.runtime.env.REnvironment; /** * Contains code specific to FastR device that shows the graphical output interactively in a window. @@ -45,6 +50,7 @@ public final class WindowDevice { RContext ctx = RContext.getInstance(); if (ctx.hasExecutor()) { frameDevice.setResizeListener(() -> redrawAll(ctx)); + frameDevice.setCloseListener(() -> devOff(ctx)); } else { noSchedulingSupportWarning(); } @@ -56,14 +62,44 @@ public final class WindowDevice { } private static void redrawAll(RContext ctx) { - if (ctx.hasExecutor()) { + if (!ctx.hasExecutor()) { + // to be robust we re-check the executor availability + return; + } + ctx.schedule(() -> { + Object prev = ctx.getEnv().getContext().enter(); + GridContext.getContext(ctx).evalInternalRFunction("redrawAll"); + ctx.getEnv().getContext().leave(prev); + }); + } + + private static void devOff(RContext ctx) { + if (!ctx.hasExecutor()) { // to be robust we re-check the executor availability - ctx.schedule(() -> { - Object prev = ctx.getEnv().getContext().enter(); - GridContext.getContext(ctx).evalInternalRFunction("redrawAll"); - ctx.getEnv().getContext().leave(prev); - }); + return; + } + ctx.schedule(() -> { + Object prev = ctx.getEnv().getContext().enter(); + RFunction devOffFun = getDevOffFunction(ctx); + if (devOffFun != null) { + RContext.getEngine().evalFunction(devOffFun, REnvironment.baseEnv(ctx).getFrame(), RCaller.topLevel, true, null); + } else { + RError.warning(RError.NO_CALLER, Message.GENERIC, "Could not locate grDevices::dev.off to close the window device."); + } + ctx.getEnv().getContext().leave(prev); + }); + } + + private static RFunction getDevOffFunction(RContext ctx) { + Object grDevices = ctx.stateREnvironment.getNamespaceRegistry().get("grDevices"); + if (!(grDevices instanceof REnvironment)) { + return null; + } + Object devOff = ((REnvironment) grDevices).get("dev.off"); + if (devOff instanceof RPromise) { + devOff = PromiseHelperNode.evaluateSlowPath((RPromise) devOff); } + return devOff instanceof RFunction ? (RFunction) devOff : null; } private static void noSchedulingSupportWarning() { diff --git a/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/device/awt/JFrameDevice.java b/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/device/awt/JFrameDevice.java index c904c6d641d5c773b6b6831de64d32ebab4f9378..163c7ee664651df3d4662f058e92a7f8ea22bde5 100644 --- a/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/device/awt/JFrameDevice.java +++ b/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/device/awt/JFrameDevice.java @@ -42,6 +42,7 @@ import java.util.TimerTask; import javax.imageio.ImageIO; import javax.swing.JFrame; import javax.swing.JPanel; +import javax.swing.SwingUtilities; import com.oracle.truffle.r.library.fastrGrid.device.DrawingContext; import com.oracle.truffle.r.library.fastrGrid.device.GridDevice; @@ -49,7 +50,7 @@ import com.oracle.truffle.r.library.fastrGrid.device.ImageSaver; /** * This device paints everything into an image, which is painted into a AWT component in its - * {@code paint} method. + * {@code paint} method. Note that this class is not thread safe. */ public final class JFrameDevice implements GridDevice, ImageSaver { @@ -64,14 +65,26 @@ public final class JFrameDevice implements GridDevice, ImageSaver { private Graphics2DDevice inner; private boolean isOnHold = false; - private FastRFrame currentFrame; + /** + * Value of this field is set from the AWT thread, any code using it in the main thread should + * first check if it is not {@code null}. + */ + private volatile FastRFrame currentFrame; + /** + * The closing operation invokes {@code dev.off()} to close the device on the R side (remove it + * from {@code .Devices} etc.), but that eventually calls into {@link #close()} where we need to + * know that the AWT window is being closed by the user and so we do not need to close it. + */ + private volatile boolean isClosing = false; private Runnable onResize; private Runnable onClose; public JFrameDevice(int width, int height) { - currentFrame = new FastRFrame(new FastRPanel(width, height)); - openGraphics2DDevice(currentFrame.fastRComponent.getWidth(), currentFrame.fastRComponent.getHeight()); + openGraphics2DDevice(width, height); componentImage = image; + SwingUtilities.invokeLater(() -> { + currentFrame = new FastRFrame(new FastRPanel(width, height)); + }); } @Override @@ -110,7 +123,9 @@ public final class JFrameDevice implements GridDevice, ImageSaver { @Override public void close() throws DeviceCloseException { disposeGraphics2DDevice(); - currentFrame.dispose(); + if (!isClosing && currentFrame != null) { + currentFrame.dispose(); + } componentImage = null; } @@ -231,17 +246,21 @@ public final class JFrameDevice implements GridDevice, ImageSaver { } private void ensureOpen() { - if (!currentFrame.isVisible()) { + if (currentFrame != null && !currentFrame.isVisible()) { int width = inner.getWidthAwt(); int height = inner.getHeightAwt(); - currentFrame = new FastRFrame(new FastRPanel(width, height)); + // Note: the assumption is that this class is single threaded disposeGraphics2DDevice(); openGraphics2DDevice(width, height); + currentFrame = null; + SwingUtilities.invokeLater(() -> { + currentFrame = new FastRFrame(new FastRPanel(width, height)); + }); } } private void repaint() { - if (!isOnHold) { + if (!isOnHold && currentFrame != null) { currentFrame.repaint(); } } @@ -312,6 +331,7 @@ public final class JFrameDevice implements GridDevice, ImageSaver { FastRFrame(FastRPanel fastRComponent) throws HeadlessException { super("FastR"); this.fastRComponent = fastRComponent; + setDefaultCloseOperation(DISPOSE_ON_CLOSE); addListeners(); getContentPane().add(fastRComponent); pack(); @@ -320,10 +340,12 @@ public final class JFrameDevice implements GridDevice, ImageSaver { } private void addListeners() { - addWindowFocusListener(new WindowAdapter() { + addWindowListener(new WindowAdapter() { @Override public void windowClosing(WindowEvent e) { - if (onClose != null) { + super.windowClosing(e); + if (!isClosing && onClose != null) { + isClosing = true; onClose.run(); } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/env/REnvironment.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/env/REnvironment.java index 0706937205d7518fa53d7fc8c54f7991998a93ee..1d310d540876f092b09263668573d9b22d68bddb 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/env/REnvironment.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/env/REnvironment.java @@ -1062,7 +1062,7 @@ public abstract class REnvironment extends RAttributeStorage { /** * When a function is invoked a {@link Function} environment may be created in response to the R * {@code environment()} base package function, and it will have an associated frame. We hide - * the creation of {@link Function} environments to ensure the <i>at most one>/i> invariant and + * the creation of {@link Function} environments to ensure the <i>at most one</i> invariant and * store the value in the frame immediately. */ public static final class Function extends REnvironment {