Skip to content
Snippets Groups Projects
Commit 24f4f16b authored by stepan's avatar stepan
Browse files

FastR Grid: color palette support.

parent 6d614304
No related branches found
No related tags found
No related merge requests found
...@@ -25,6 +25,8 @@ package com.oracle.truffle.r.library.fastrGrid; ...@@ -25,6 +25,8 @@ package com.oracle.truffle.r.library.fastrGrid;
import com.oracle.truffle.r.library.fastrGrid.DisplayList.LGetDisplayListElement; import com.oracle.truffle.r.library.fastrGrid.DisplayList.LGetDisplayListElement;
import com.oracle.truffle.r.library.fastrGrid.DisplayList.LInitDisplayList; import com.oracle.truffle.r.library.fastrGrid.DisplayList.LInitDisplayList;
import com.oracle.truffle.r.library.fastrGrid.DisplayList.LSetDisplayListOn; import com.oracle.truffle.r.library.fastrGrid.DisplayList.LSetDisplayListOn;
import com.oracle.truffle.r.library.fastrGrid.PaletteExternals.CPalette;
import com.oracle.truffle.r.library.fastrGrid.PaletteExternals.CPalette2;
import com.oracle.truffle.r.library.fastrGrid.grDevices.DevCairo; import com.oracle.truffle.r.library.fastrGrid.grDevices.DevCairo;
import com.oracle.truffle.r.library.fastrGrid.grDevices.DevCurr; import com.oracle.truffle.r.library.fastrGrid.grDevices.DevCurr;
import com.oracle.truffle.r.library.fastrGrid.grDevices.DevHoldFlush; import com.oracle.truffle.r.library.fastrGrid.grDevices.DevHoldFlush;
...@@ -75,6 +77,10 @@ public final class FastRGridExternalLookup { ...@@ -75,6 +77,10 @@ public final class FastRGridExternalLookup {
return SavePlot.create(); return SavePlot.create();
case "X11": case "X11":
return new InitWindowedDevice(); return new InitWindowedDevice();
case "palette":
return CPalette.create();
case "palette2":
return CPalette2.create();
default: default:
return null; return null;
} }
......
...@@ -18,6 +18,7 @@ import static com.oracle.truffle.r.library.fastrGrid.GridUtils.getDataAtMod; ...@@ -18,6 +18,7 @@ import static com.oracle.truffle.r.library.fastrGrid.GridUtils.getDataAtMod;
import java.util.Arrays; import java.util.Arrays;
import java.util.function.Function; import java.util.function.Function;
import com.oracle.truffle.r.library.fastrGrid.GridState.GridPalette;
import com.oracle.truffle.r.library.fastrGrid.device.DrawingContext; import com.oracle.truffle.r.library.fastrGrid.device.DrawingContext;
import com.oracle.truffle.r.library.fastrGrid.device.DrawingContextDefaults; import com.oracle.truffle.r.library.fastrGrid.device.DrawingContextDefaults;
import com.oracle.truffle.r.library.fastrGrid.device.GridColor; import com.oracle.truffle.r.library.fastrGrid.device.GridColor;
...@@ -25,6 +26,7 @@ import com.oracle.truffle.r.library.fastrGrid.device.GridDevice; ...@@ -25,6 +26,7 @@ import com.oracle.truffle.r.library.fastrGrid.device.GridDevice;
import com.oracle.truffle.r.runtime.RError; import com.oracle.truffle.r.runtime.RError;
import com.oracle.truffle.r.runtime.RError.Message; import com.oracle.truffle.r.runtime.RError.Message;
import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RRuntime;
import com.oracle.truffle.r.runtime.Utils;
import com.oracle.truffle.r.runtime.data.RDataFactory; import com.oracle.truffle.r.runtime.data.RDataFactory;
import com.oracle.truffle.r.runtime.data.RList; import com.oracle.truffle.r.runtime.data.RList;
import com.oracle.truffle.r.runtime.data.RNull; import com.oracle.truffle.r.runtime.data.RNull;
...@@ -265,15 +267,21 @@ public final class GPar { ...@@ -265,15 +267,21 @@ public final class GPar {
private GridColor getGridColor(int listIndex) { private GridColor getGridColor(int listIndex) {
Object value = data[listIndex]; Object value = data[listIndex];
GridColor color = getPaletteColor(value);
if (color != null) {
return color;
}
String strValue = null; String strValue = null;
if (value instanceof String) { if (value instanceof String) {
strValue = (String) value; strValue = (String) value;
} else if (value instanceof RAbstractStringVector && ((RAbstractStringVector) value).getLength() > 0) { } else if (value instanceof RAbstractStringVector && ((RAbstractStringVector) value).getLength() > 0) {
strValue = ((RAbstractStringVector) value).getDataAt(listIndex % ((RAbstractStringVector) value).getLength()); strValue = ((RAbstractStringVector) value).getDataAt(index % ((RAbstractStringVector) value).getLength());
} else { } else {
return GridColor.TRANSPARENT; return GridColor.TRANSPARENT;
} }
GridColor color = GridColorUtils.gridColorFromString(strValue);
color = GridColorUtils.gridColorFromString(strValue);
double alpha = asDouble(data[GP_ALPHA], index); double alpha = asDouble(data[GP_ALPHA], index);
if (alpha != 1.) { if (alpha != 1.) {
int newAlpha = Math.min(255, (int) (alpha * ((color.getAlpha() / 255.0) * 255))); int newAlpha = Math.min(255, (int) (alpha * ((color.getAlpha() / 255.0) * 255)));
...@@ -283,6 +291,44 @@ public final class GPar { ...@@ -283,6 +291,44 @@ public final class GPar {
} }
} }
private GridColor getPaletteColor(Object colorIdIn) {
Object colorId = colorIdIn;
if (colorId instanceof RAbstractVector) {
RAbstractVector vec = (RAbstractVector) colorId;
colorId = vec.getDataAtAsObject(index % vec.getLength());
}
int paletteIdx = RRuntime.INT_NA;
if (colorId instanceof Integer) {
paletteIdx = (int) colorId;
} else if (colorId instanceof Double && !RRuntime.isNA((Double) colorId)) {
paletteIdx = (int) (double) colorId;
} else if (colorId instanceof String && !RRuntime.isNA((String) colorId)) {
paletteIdx = paletteIdxFromString((String) colorId);
} else if (colorId instanceof Byte && !RRuntime.isNA((byte) colorId)) {
paletteIdx = (int) (byte) colorId;
}
if (RRuntime.isNA(paletteIdx)) {
return null;
}
if (paletteIdx < 0) {
throw RError.error(RError.NO_CALLER, Message.GENERIC, Utils.stringFormat("numerical color values must be >= 0, found %d", paletteIdx));
}
if (paletteIdx == 0) {
return GridColor.TRANSPARENT;
}
GridPalette palette = GridContext.getContext().getGridState().getPalette();
GridColor result = palette.colors[(paletteIdx - 1) % palette.colors.length];
return result; // one based index
}
private int paletteIdxFromString(String colorId) {
try {
return Integer.parseInt(colorId, 10);
} catch (NumberFormatException ex) {
return RRuntime.INT_NA;
}
}
private static final byte[] DASHED_LINE = new byte[]{4, 4}; private static final byte[] DASHED_LINE = new byte[]{4, 4};
private static final byte[] DOTTED_LINE = new byte[]{1, 3}; private static final byte[] DOTTED_LINE = new byte[]{1, 3};
private static final byte[] DOTDASH_LINE = new byte[]{1, 3, 4, 3}; private static final byte[] DOTDASH_LINE = new byte[]{1, 3, 4, 3};
......
...@@ -14,15 +14,36 @@ package com.oracle.truffle.r.library.fastrGrid; ...@@ -14,15 +14,36 @@ package com.oracle.truffle.r.library.fastrGrid;
import java.util.HashMap; import java.util.HashMap;
import java.util.Locale; import java.util.Locale;
import com.oracle.truffle.r.library.fastrGrid.GridState.GridPalette;
import com.oracle.truffle.r.library.fastrGrid.device.GridColor; import com.oracle.truffle.r.library.fastrGrid.device.GridColor;
import com.oracle.truffle.r.runtime.RError; import com.oracle.truffle.r.runtime.RError;
import com.oracle.truffle.r.runtime.RError.Message; import com.oracle.truffle.r.runtime.RError.Message;
public final class GridColorUtils { public final class GridColorUtils {
private static GridPalette defaultPalette;
private GridColorUtils() { private GridColorUtils() {
// only static members // only static members
} }
public static GridPalette getDefaultPalette() {
if (defaultPalette == null) {
// Note: default palette copied from GNU R
defaultPalette = new GridPalette(new String[]{
"black",
"red",
"green3",
"blue",
"cyan",
"magenta",
"yellow",
"grey"
});
}
return defaultPalette;
}
/** /**
* Converts the representation of color used within R, e.g. as value for * Converts the representation of color used within R, e.g. as value for
* {@code gpar(col='value')}, to our internal representation that grid device should understand. * {@code gpar(col='value')}, to our internal representation that grid device should understand.
......
...@@ -13,6 +13,7 @@ package com.oracle.truffle.r.library.fastrGrid; ...@@ -13,6 +13,7 @@ package com.oracle.truffle.r.library.fastrGrid;
import java.util.function.Supplier; import java.util.function.Supplier;
import com.oracle.truffle.r.library.fastrGrid.device.GridColor;
import com.oracle.truffle.r.library.fastrGrid.device.GridDevice; import com.oracle.truffle.r.library.fastrGrid.device.GridDevice;
import com.oracle.truffle.r.runtime.data.RList; import com.oracle.truffle.r.runtime.data.RList;
import com.oracle.truffle.r.runtime.data.RNull; import com.oracle.truffle.r.runtime.data.RNull;
...@@ -21,6 +22,7 @@ import com.oracle.truffle.r.runtime.env.REnvironment; ...@@ -21,6 +22,7 @@ import com.oracle.truffle.r.runtime.env.REnvironment;
public final class GridState { public final class GridState {
private REnvironment gridEnv; private REnvironment gridEnv;
private GridDeviceState devState; private GridDeviceState devState;
private GridPalette palette;
/** /**
* Current grob being drawn (for determining the list of grobs to search when evaluating a * Current grob being drawn (for determining the list of grobs to search when evaluating a
...@@ -87,6 +89,14 @@ public final class GridState { ...@@ -87,6 +89,14 @@ public final class GridState {
devState.displayListIndex = newValue; devState.displayListIndex = newValue;
} }
public GridPalette getPalette() {
return palette == null ? GridColorUtils.getDefaultPalette() : palette;
}
public void setPalette(GridPalette palette) {
this.palette = palette;
}
public void init(REnvironment gridEnv) { public void init(REnvironment gridEnv) {
this.gridEnv = gridEnv; this.gridEnv = gridEnv;
this.currentGrob = RNull.instance; this.currentGrob = RNull.instance;
...@@ -153,6 +163,28 @@ public final class GridState { ...@@ -153,6 +163,28 @@ public final class GridState {
return devState.scale; return devState.scale;
} }
public static final class GridPalette {
public final GridColor[] colors;
public final String[] colorNames;
public GridPalette(String[] colorNames) {
this.colorNames = colorNames;
colors = new GridColor[colorNames.length];
for (int i = 0; i < colorNames.length; i++) {
colors[i] = GridColorUtils.gridColorFromString(colorNames[i]);
}
}
public GridPalette(int[] colors) {
this.colors = new GridColor[colors.length];
colorNames = new String[colors.length];
for (int i = 0; i < colors.length; i++) {
this.colors[i] = GridColor.fromRawValue(colors[i]);
colorNames[i] = GridColorUtils.gridColorToRString(this.colors[i]);
}
}
}
static final class GridDeviceState { static final class GridDeviceState {
private boolean isDeviceInitialized = false; private boolean isDeviceInitialized = false;
private RList gpar; private RList gpar;
......
/*
* This material is distributed under the GNU General Public License
* Version 2. You may review the terms of this license at
* http://www.gnu.org/licenses/gpl-2.0.html
*
* Copyright (c) 1997-2014, The R Core Team
* Copyright (c) 2003, The R Foundation
* Copyright (c) 2017, Oracle and/or its affiliates
*
* All rights reserved.
*/
package com.oracle.truffle.r.library.fastrGrid;
import static com.oracle.truffle.r.nodes.builtin.CastBuilder.Predef.abstractVectorValue;
import static com.oracle.truffle.r.nodes.builtin.CastBuilder.Predef.stringValue;
import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
import com.oracle.truffle.api.dsl.Specialization;
import com.oracle.truffle.r.library.fastrGrid.GridState.GridPalette;
import com.oracle.truffle.r.library.fastrGrid.PaletteExternalsFactory.CPalette2NodeGen;
import com.oracle.truffle.r.library.fastrGrid.PaletteExternalsFactory.CPaletteNodeGen;
import com.oracle.truffle.r.nodes.builtin.RExternalBuiltinNode;
import com.oracle.truffle.r.runtime.RError.Message;
import com.oracle.truffle.r.runtime.RRuntime;
import com.oracle.truffle.r.runtime.data.RDataFactory;
import com.oracle.truffle.r.runtime.data.RIntVector;
import com.oracle.truffle.r.runtime.data.RStringVector;
import com.oracle.truffle.r.runtime.data.model.RAbstractIntVector;
import com.oracle.truffle.r.runtime.data.model.RAbstractStringVector;
import com.oracle.truffle.r.runtime.data.model.RAbstractVector;
public final class PaletteExternals {
private PaletteExternals() {
// only static members
}
/**
* Implements external {@code C_palette} used in palette R function.
*/
public abstract static class CPalette extends RExternalBuiltinNode.Arg1 {
static {
Casts casts = new Casts(CPalette.class);
casts.arg(0).mustBe(stringValue());
}
public static CPalette create() {
return CPaletteNodeGen.create();
}
@Specialization
@TruffleBoundary
public RStringVector updatePalette(RAbstractStringVector palette) {
GridState state = GridContext.getContext().getGridState();
GridPalette newPalette = null;
if (palette.getLength() == 1) {
if (palette.getDataAt(0).toLowerCase().equals("default")) {
newPalette = GridColorUtils.getDefaultPalette();
} else {
throw error(Message.GENERIC, "unknown palette (need >= 2 colors)");
}
} else if (palette.getLength() > 1) {
newPalette = new GridPalette(palette.materialize().getDataCopy());
}
String[] original = state.getPalette().colorNames;
if (newPalette != null) {
// the contract is that if the argument's length is zero, we only return the palette
// and
// do not set anything.
state.setPalette(newPalette);
}
// don't know the completeness, assume the worst rather than finding out
RStringVector result = RDataFactory.createStringVector(original, RDataFactory.INCOMPLETE_VECTOR);
result.makeSharedPermanent();
return result;
}
}
/**
* Implements external {@code C_palette2} used only internally in grDevices, the parameter are
* colors encoded in integer already.
*/
public abstract static class CPalette2 extends RExternalBuiltinNode.Arg1 {
static {
Casts casts = new Casts(CPalette2.class);
casts.arg(0).mustBe(abstractVectorValue());
}
public static CPalette2 create() {
return CPalette2NodeGen.create();
}
@Specialization
@TruffleBoundary
public RIntVector updatePalette(RAbstractVector palette) {
GridState state = GridContext.getContext().getGridState();
int[] newPalette = null;
if (palette.getLength() > 0) {
RAbstractIntVector data = GridUtils.asIntVector(palette);
newPalette = data.materialize().getDataCopy();
}
RIntVector result = getResult(state);
if (newPalette != null) {
state.setPalette(new GridPalette(newPalette));
}
return result;
}
private static RIntVector getResult(GridState state) {
GridPalette palette = state.getPalette();
int[] result = new int[palette.colors.length];
boolean complete = true;
for (int i = 0; i < result.length; i++) {
result[i] = palette.colors[i].getRawValue();
complete &= !RRuntime.isNA(result[i]);
}
return RDataFactory.createIntVector(result, complete);
}
}
}
...@@ -33,6 +33,10 @@ public final class GridColor { ...@@ -33,6 +33,10 @@ public final class GridColor {
private final int value; private final int value;
private GridColor(int value) {
this.value = value;
}
public GridColor(int red, int green, int blue, int alpha) { public GridColor(int red, int green, int blue, int alpha) {
value = ((alpha & 0xFF) << 24) | value = ((alpha & 0xFF) << 24) |
((red & 0xFF) << 16) | ((red & 0xFF) << 16) |
...@@ -40,6 +44,10 @@ public final class GridColor { ...@@ -40,6 +44,10 @@ public final class GridColor {
(blue & 0xFF); (blue & 0xFF);
} }
public static GridColor fromRawValue(int value) {
return new GridColor(value);
}
public int getRed() { public int getRed() {
return (value >> 16) & 0xFF; return (value >> 16) & 0xFF;
} }
...@@ -56,6 +64,10 @@ public final class GridColor { ...@@ -56,6 +64,10 @@ public final class GridColor {
return (value >> 24) & 0xff; return (value >> 24) & 0xff;
} }
public int getRawValue() {
return value;
}
@Override @Override
public boolean equals(Object obj) { public boolean equals(Object obj) {
return obj instanceof GridColor && value == ((GridColor) obj).value; return obj instanceof GridColor && value == ((GridColor) obj).value;
......
...@@ -770,6 +770,7 @@ com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/GridLine ...@@ -770,6 +770,7 @@ com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/GridLine
com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/ViewPortTransform.java,gnu_r_murrel_core.copyright com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/ViewPortTransform.java,gnu_r_murrel_core.copyright
com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/LGridDirty.java,gnu_r_murrel_core.copyright com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/LGridDirty.java,gnu_r_murrel_core.copyright
com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/GridColorUtils.java,gnu_r.copyright com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/GridColorUtils.java,gnu_r.copyright
com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/PaletteExternals.java,gnu_r.copyright
com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/LCircle.java,gnu_r_murrel_core.copyright com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/LCircle.java,gnu_r_murrel_core.copyright
com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/EdgeDetection.java,gnu_r_murrel_core.copyright com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/EdgeDetection.java,gnu_r_murrel_core.copyright
com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/GridTextNode.java,gnu_r_murrel_core.copyright com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/fastrGrid/GridTextNode.java,gnu_r_murrel_core.copyright
......
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