From 7200ada98e6d5be895479be80b6b08eb9175143b Mon Sep 17 00:00:00 2001 From: Lukas Stadler <lukas.stadler@oracle.com> Date: Wed, 15 Nov 2017 16:46:08 +0100 Subject: [PATCH] add VectorAccess pattern for efficient access to vectors and implement it for all data types --- .../oracle/truffle/r/runtime/RRuntime.java | 4 + .../com/oracle/truffle/r/runtime/RType.java | 14 + .../r/runtime/data/NativeDataAccess.java | 44 +- .../truffle/r/runtime/data/RComplex.java | 59 ++ .../r/runtime/data/RComplexVector.java | 84 ++- .../truffle/r/runtime/data/RDouble.java | 35 + .../r/runtime/data/RDoubleSequence.java | 37 + .../truffle/r/runtime/data/RDoubleVector.java | 58 +- .../truffle/r/runtime/data/RExpression.java | 53 ++ .../truffle/r/runtime/data/RExternalPtr.java | 2 +- .../truffle/r/runtime/data/RFactor.java | 5 +- .../truffle/r/runtime/data/RIntSequence.java | 37 + .../truffle/r/runtime/data/RIntVector.java | 58 +- .../truffle/r/runtime/data/RInteger.java | 35 + .../truffle/r/runtime/data/RLanguage.java | 49 ++ .../oracle/truffle/r/runtime/data/RList.java | 54 ++ .../truffle/r/runtime/data/RListBase.java | 6 - .../truffle/r/runtime/data/RLogical.java | 35 + .../r/runtime/data/RLogicalVector.java | 58 +- .../truffle/r/runtime/data/RPairList.java | 49 ++ .../oracle/truffle/r/runtime/data/RRaw.java | 35 + .../truffle/r/runtime/data/RRawVector.java | 65 +- .../truffle/r/runtime/data/RScalarList.java | 45 ++ .../truffle/r/runtime/data/RScalarVector.java | 5 + .../truffle/r/runtime/data/RSequence.java | 5 + .../truffle/r/runtime/data/RString.java | 35 + .../r/runtime/data/RStringSequence.java | 37 + .../truffle/r/runtime/data/RStringVector.java | 54 +- .../truffle/r/runtime/data/RVector.java | 31 +- .../data/closures/RToVectorClosure.java | 16 +- .../data/model/RAbstractComplexVector.java | 4 - .../data/model/RAbstractContainer.java | 12 +- .../data/model/RAbstractRawVector.java | 4 - .../data/nodes/FastPathVectorAccess.java | 635 ++++++++++++++++++ .../data/nodes/SlowPathVectorAccess.java | 606 +++++++++++++++++ .../r/runtime/data/nodes/VectorAccess.java | 501 ++++++++++++++ .../interop/TruffleObjectConverter.java | 17 +- .../truffle/r/runtime/ops/na/NACheck.java | 11 + 38 files changed, 2806 insertions(+), 88 deletions(-) create mode 100644 com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/FastPathVectorAccess.java create mode 100644 com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/SlowPathVectorAccess.java create mode 100644 com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/VectorAccess.java diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RRuntime.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RRuntime.java index d3965fab24..ddba79ddb2 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RRuntime.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RRuntime.java @@ -762,6 +762,10 @@ public class RRuntime { return isNA(value.getRealPart()) || isNA(value.getImaginaryPart()); } + public static boolean isNA(double real, double imag) { + return isNA(real) || isNA(imag); + } + @TruffleBoundary public static String escapeString(String value, boolean encodeNonASCII, boolean quote) { if (isNA(value)) { diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RType.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RType.java index 58096f6025..ccd1f53dc1 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RType.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RType.java @@ -96,6 +96,20 @@ public enum RType { } } + public boolean isAtomic() { + switch (this) { + case Logical: + case Double: + case Integer: + case Complex: + case Character: + case Raw: + return true; + default: + return false; + } + } + public boolean isVector() { switch (this) { case Logical: diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/NativeDataAccess.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/NativeDataAccess.java index efe8ee17d0..ab9c0f0439 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/NativeDataAccess.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/NativeDataAccess.java @@ -314,21 +314,35 @@ public final class NativeDataAccess { UnsafeAdapter.UNSAFE.getDouble(address + (index * 2 + 1) * Unsafe.ARRAY_DOUBLE_INDEX_SCALE)); } - public static void setNativeMirrorData(Object nativeMirror, int index, double value) { + public static double getComplexNativeMirrorDataR(Object nativeMirror, int index) { + long address = ((NativeMirror) nativeMirror).dataAddress; + assert address != 0; + assert index < ((NativeMirror) nativeMirror).length; + return UnsafeAdapter.UNSAFE.getDouble(address + index * 2 * Unsafe.ARRAY_DOUBLE_INDEX_SCALE); + } + + public static double getComplexNativeMirrorDataI(Object nativeMirror, int index) { + long address = ((NativeMirror) nativeMirror).dataAddress; + assert address != 0; + assert index < ((NativeMirror) nativeMirror).length; + return UnsafeAdapter.UNSAFE.getDouble(address + (index * 2 + 1) * Unsafe.ARRAY_DOUBLE_INDEX_SCALE); + } + + public static void setNativeMirrorDoubleData(Object nativeMirror, int index, double value) { long address = ((NativeMirror) nativeMirror).dataAddress; assert address != 0; assert index < ((NativeMirror) nativeMirror).length; UnsafeAdapter.UNSAFE.putDouble(address + index * Unsafe.ARRAY_DOUBLE_INDEX_SCALE, value); } - public static void setNativeMirrorData(Object nativeMirror, int index, byte value) { + public static void setNativeMirrorRawData(Object nativeMirror, int index, byte value) { long address = ((NativeMirror) nativeMirror).dataAddress; assert address != 0; assert index < ((NativeMirror) nativeMirror).length; UnsafeAdapter.UNSAFE.putByte(address + index * Unsafe.ARRAY_BYTE_INDEX_SCALE, value); } - public static void setNativeMirrorData(Object nativeMirror, int index, int value) { + public static void setNativeMirrorIntData(Object nativeMirror, int index, int value) { long address = ((NativeMirror) nativeMirror).dataAddress; assert address != 0; assert index < ((NativeMirror) nativeMirror).length; @@ -390,10 +404,14 @@ public final class NativeDataAccess { if (noIntNative.isValid() || data != null) { return data.length; } else { - return (int) ((NativeMirror) vector.getNativeMirror()).length; + return getDataLengthFromMirror(vector.getNativeMirror()); } } + static int getDataLengthFromMirror(Object mirror) { + return (int) ((NativeMirror) mirror).length; + } + static void setData(RIntVector vector, int[] data, int index, int value) { if (noIntNative.isValid() || data != null) { data[index] = value; @@ -478,7 +496,7 @@ public final class NativeDataAccess { if (noDoubleNative.isValid() || data != null) { data[index] = value; } else { - setNativeMirrorData(vector.getNativeMirror(), index, value); + setNativeMirrorDoubleData(vector.getNativeMirror(), index, value); } } @@ -490,6 +508,22 @@ public final class NativeDataAccess { } } + static double getDataR(RComplexVector vector, double[] data, int index) { + if (noComplexNative.isValid() || data != null) { + return data[index * 2]; + } else { + return getComplexNativeMirrorDataR(vector.getNativeMirror(), index); + } + } + + static double getDataI(RComplexVector vector, double[] data, int index) { + if (noComplexNative.isValid() || data != null) { + return data[index * 2 + 1]; + } else { + return getComplexNativeMirrorDataI(vector.getNativeMirror(), index); + } + } + static int getDataLength(RComplexVector vector, double[] data) { if (noComplexNative.isValid() || data != null) { return data.length >> 1; diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RComplex.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RComplex.java index af6a3028bc..367c91a771 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RComplex.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RComplex.java @@ -30,7 +30,11 @@ import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.context.RContext; import com.oracle.truffle.r.runtime.data.model.RAbstractComplexVector; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromComplexAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromComplexAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; @ValueType public final class RComplex extends RScalarVector implements RAbstractComplexVector { @@ -143,4 +147,59 @@ public final class RComplex extends RScalarVector implements RAbstractComplexVec return Math.sqrt(re * re + im * im); } } + + private static final class FastPathAccess extends FastPathFromComplexAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected RComplex getComplex(Object store, int index) { + assert index == 0; + return (RComplex) store; + } + + @Override + protected double getComplexR(Object store, int index) { + assert index == 0; + return ((RComplex) store).realPart; + } + + @Override + protected double getComplexI(Object store, int index) { + assert index == 0; + return ((RComplex) store).imaginaryPart; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromComplexAccess SLOW_PATH_ACCESS = new SlowPathFromComplexAccess() { + @Override + protected RComplex getComplex(Object store, int index) { + assert index == 0; + return (RComplex) store; + } + + @Override + protected double getComplexR(Object store, int index) { + assert index == 0; + return ((RComplex) store).realPart; + } + + @Override + protected double getComplexI(Object store, int index) { + assert index == 0; + return ((RComplex) store).imaginaryPart; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RComplexVector.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RComplexVector.java index a3cedbc005..b6ebdd2696 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RComplexVector.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RComplexVector.java @@ -30,7 +30,11 @@ import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.Utils; import com.oracle.truffle.r.runtime.data.closures.RClosures; import com.oracle.truffle.r.runtime.data.model.RAbstractComplexVector; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromComplexAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromComplexAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; import com.oracle.truffle.r.runtime.ops.na.NACheck; public final class RComplexVector extends RVector<double[]> implements RAbstractComplexVector { @@ -104,10 +108,8 @@ public final class RComplexVector extends RVector<double[]> implements RAbstract NativeDataAccess.setData(this, (double[]) store, index, value.getRealPart(), value.getImaginaryPart()); } - @Override - public RComplex getDataAt(Object store, int index) { - assert data == store; - return NativeDataAccess.getData(this, (double[]) store, index); + public void setDataAt(int index, RComplex value) { + NativeDataAccess.setData(this, data, index, value.getRealPart(), value.getImaginaryPart()); } @Override @@ -115,11 +117,6 @@ public final class RComplexVector extends RVector<double[]> implements RAbstract return NativeDataAccess.getData(this, data, index); } - @Override - public String toString() { - return toString(i -> getDataAt(i).toString()); - } - @Override public boolean verify() { if (isComplete()) { @@ -219,4 +216,73 @@ public final class RComplexVector extends RVector<double[]> implements RAbstract complete = false; } } + + private static final class FastPathAccess extends FastPathFromComplexAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected RComplex getComplex(Object store, int index) { + return RComplex.valueOf(getComplexR(store, index), getComplexI(store, index)); + } + + @Override + protected double getComplexR(Object store, int index) { + return hasStore ? ((double[]) store)[index * 2] : NativeDataAccess.getDoubleNativeMirrorData(store, index * 2); + } + + @Override + protected double getComplexI(Object store, int index) { + return hasStore ? ((double[]) store)[index * 2 + 1] : NativeDataAccess.getDoubleNativeMirrorData(store, index * 2 + 1); + } + + @Override + protected void setComplex(Object store, int index, double real, double imaginary) { + if (hasStore) { + ((double[]) store)[index * 2] = real; + ((double[]) store)[index * 2 + 1] = imaginary; + } else { + NativeDataAccess.setNativeMirrorDoubleData(store, index * 2, real); + NativeDataAccess.setNativeMirrorDoubleData(store, index * 2 + 1, imaginary); + } + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromComplexAccess SLOW_PATH_ACCESS = new SlowPathFromComplexAccess() { + @Override + protected RComplex getComplex(Object store, int index) { + RComplexVector vector = (RComplexVector) store; + return NativeDataAccess.getData(vector, vector.data, index); + } + + @Override + protected double getComplexR(Object store, int index) { + RComplexVector vector = (RComplexVector) store; + return NativeDataAccess.getDataR(vector, vector.data, index); + } + + @Override + protected double getComplexI(Object store, int index) { + RComplexVector vector = (RComplexVector) store; + return NativeDataAccess.getDataI(vector, vector.data, index); + } + + @Override + protected void setComplex(Object store, int index, double real, double imaginary) { + RComplexVector vector = (RComplexVector) store; + NativeDataAccess.setData(vector, vector.data, index, real, imaginary); + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDouble.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDouble.java index 6a66889e9c..c0eff99975 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDouble.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDouble.java @@ -28,8 +28,12 @@ import com.oracle.truffle.api.profiles.ConditionProfile; import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.context.RContext; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractDoubleVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromDoubleAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromDoubleAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; @ValueType public final class RDouble extends RScalarVector implements RAbstractDoubleVector { @@ -99,4 +103,35 @@ public final class RDouble extends RScalarVector implements RAbstractDoubleVecto public boolean isNA() { return RRuntime.isNA(getValue()); } + + private static final class FastPathAccess extends FastPathFromDoubleAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected double getDouble(Object store, int index) { + assert index == 0; + return ((RDouble) store).value; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromDoubleAccess SLOW_PATH_ACCESS = new SlowPathFromDoubleAccess() { + @Override + protected double getDouble(Object store, int index) { + assert index == 0; + return ((RDouble) store).value; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDoubleSequence.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDoubleSequence.java index a94cce37d0..5d0ca6da84 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDoubleSequence.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDoubleSequence.java @@ -26,8 +26,12 @@ import com.oracle.truffle.api.CompilerAsserts; import com.oracle.truffle.api.profiles.ConditionProfile; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.data.closures.RClosures; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractDoubleVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromDoubleAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromDoubleAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; public final class RDoubleSequence extends RSequence implements RAbstractDoubleVector { @@ -134,4 +138,37 @@ public final class RDoubleSequence extends RSequence implements RAbstractDoubleV CompilerAsserts.neverPartOfCompilation(); return "[" + start + " - " + getEnd() + "]"; } + + private static final class FastPathAccess extends FastPathFromDoubleAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected double getDouble(Object store, int index) { + RDoubleSequence vector = (RDoubleSequence) store; + assert index >= 0 && index < vector.getLength(); + return vector.start + vector.stride * index; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromDoubleAccess SLOW_PATH_ACCESS = new SlowPathFromDoubleAccess() { + @Override + protected double getDouble(Object store, int index) { + RDoubleSequence vector = (RDoubleSequence) store; + assert index >= 0 && index < vector.getLength(); + return vector.start + vector.stride * index; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDoubleVector.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDoubleVector.java index cc84373f42..197d34d214 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDoubleVector.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RDoubleVector.java @@ -29,8 +29,12 @@ import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.Utils; import com.oracle.truffle.r.runtime.data.closures.RClosures; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractDoubleVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromDoubleAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromDoubleAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; import com.oracle.truffle.r.runtime.ops.na.NACheck; public final class RDoubleVector extends RVector<double[]> implements RAbstractDoubleVector { @@ -99,6 +103,10 @@ public final class RDoubleVector extends RVector<double[]> implements RAbstractD NativeDataAccess.setData(this, (double[]) store, index, value); } + public void setDataAt(int index, double value) { + NativeDataAccess.setData(this, data, index, value); + } + @Override public double getDataAt(Object store, int index) { assert data == store; @@ -123,11 +131,6 @@ public final class RDoubleVector extends RVector<double[]> implements RAbstractD return NativeDataAccess.getDataLength(this, data); } - @Override - public String toString() { - return toString(i -> Double.toString(getDataAt(i))); - } - @Override public boolean verify() { if (isComplete()) { @@ -238,4 +241,49 @@ public final class RDoubleVector extends RVector<double[]> implements RAbstractD complete = false; } } + + private static final class FastPathAccess extends FastPathFromDoubleAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected double getDouble(Object store, int index) { + return hasStore ? ((double[]) store)[index] : NativeDataAccess.getDoubleNativeMirrorData(store, index); + } + + @Override + protected void setDouble(Object store, int index, double value) { + if (hasStore) { + ((double[]) store)[index] = value; + } else { + NativeDataAccess.setNativeMirrorDoubleData(store, index, value); + } + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromDoubleAccess SLOW_PATH_ACCESS = new SlowPathFromDoubleAccess() { + @Override + protected double getDouble(Object store, int index) { + RDoubleVector vector = (RDoubleVector) store; + return NativeDataAccess.getData(vector, vector.data, index); + } + + @Override + protected void setDouble(Object store, int index, double value) { + RDoubleVector vector = (RDoubleVector) store; + NativeDataAccess.setData(vector, vector.data, index, value); + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RExpression.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RExpression.java index f30f71ef14..1c9e240c9a 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RExpression.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RExpression.java @@ -26,7 +26,11 @@ import java.util.Arrays; import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary; import com.oracle.truffle.r.runtime.RType; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromListAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromListAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; public final class RExpression extends RListBase implements RAbstractVector { @@ -84,4 +88,53 @@ public final class RExpression extends RListBase implements RAbstractVector { protected RExpression internalCopyResized(int size, boolean fillNA, int[] dimensions) { return RDataFactory.createExpression(copyResizedData(size, fillNA), dimensions); } + + private static final class FastPathAccess extends FastPathFromListAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + public RType getType() { + return RType.Expression; + } + + @Override + protected Object getListElement(Object store, int index) { + return ((Object[]) store)[index]; + } + + @Override + protected void setListElement(Object store, int index, Object value) { + ((Object[]) store)[index] = value; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromListAccess SLOW_PATH_ACCESS = new SlowPathFromListAccess() { + @Override + public RType getType() { + return RType.Expression; + } + + @Override + protected Object getListElement(Object store, int index) { + return ((RExpression) store).data[index]; + } + + @Override + protected void setListElement(Object store, int index, Object value) { + ((RExpression) store).data[index] = value; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RExternalPtr.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RExternalPtr.java index 57aa7933b9..fa03686af9 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RExternalPtr.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RExternalPtr.java @@ -28,7 +28,7 @@ import com.oracle.truffle.r.runtime.ffi.DLL.SymbolHandle; /** * The rarely seen {@code externalptr} type used in native code. */ -public class RExternalPtr extends RAttributeStorage implements RTypedValue { +public final class RExternalPtr extends RAttributeStorage implements RTypedValue { /** * In GNU R, typically the address of some C structure, so a {@code void*}. Represented here as * our abstraction of a "native symbol" (even though there may not actually be a symbol diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RFactor.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RFactor.java index 3d0746749a..cbda3db37f 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RFactor.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RFactor.java @@ -36,10 +36,7 @@ public final class RFactor { * be replaced with FactorNodes.GetLevel in the future. */ public static RVector<?> getLevels(RAbstractIntVector factor) { - return getLevelsImpl(factor.getAttr(RRuntime.LEVELS_ATTR_KEY)); - } - - private static RVector<?> getLevelsImpl(Object attr) { + Object attr = factor.getAttr(RRuntime.LEVELS_ATTR_KEY); // convert scalar to RVector<?>if necessary return attr instanceof RVector ? (RVector<?>) attr : (RVector<?>) RRuntime.asAbstractVector(attr); } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RIntSequence.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RIntSequence.java index c0e25112c7..5182f8cba1 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RIntSequence.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RIntSequence.java @@ -26,8 +26,12 @@ import com.oracle.truffle.api.CompilerAsserts; import com.oracle.truffle.api.profiles.ConditionProfile; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.data.closures.RClosures; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractIntVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromIntAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromIntAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; public final class RIntSequence extends RSequence implements RAbstractIntVector { @@ -143,4 +147,37 @@ public final class RIntSequence extends RSequence implements RAbstractIntVector public RIntVector createEmptySameType(int newLength, boolean newIsComplete) { return RDataFactory.createIntVector(new int[newLength], newIsComplete); } + + private static final class FastPathAccess extends FastPathFromIntAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected int getInt(Object store, int index) { + RIntSequence vector = (RIntSequence) store; + assert index >= 0 && index < vector.getLength(); + return vector.start + vector.stride * index; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromIntAccess SLOW_PATH_ACCESS = new SlowPathFromIntAccess() { + @Override + protected int getInt(Object store, int index) { + RIntSequence vector = (RIntSequence) store; + assert index >= 0 && index < vector.getLength(); + return vector.start + vector.stride * index; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RIntVector.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RIntVector.java index f52c7cb614..450619ae32 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RIntVector.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RIntVector.java @@ -29,8 +29,12 @@ import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.Utils; import com.oracle.truffle.r.runtime.data.closures.RClosures; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractIntVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromIntAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromIntAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; import com.oracle.truffle.r.runtime.ops.na.NACheck; public final class RIntVector extends RVector<int[]> implements RAbstractIntVector { @@ -99,6 +103,10 @@ public final class RIntVector extends RVector<int[]> implements RAbstractIntVect NativeDataAccess.setData(this, (int[]) store, index, value); } + public void setDataAt(int index, int value) { + NativeDataAccess.setData(this, data, index, value); + } + @Override protected RIntVector internalCopy() { if (data != null) { @@ -126,11 +134,6 @@ public final class RIntVector extends RVector<int[]> implements RAbstractIntVect return NativeDataAccess.getDataLength(this, data); } - @Override - public String toString() { - return toString(i -> Double.toString(getDataAt(i))); - } - @Override public boolean verify() { if (isComplete()) { @@ -246,4 +249,49 @@ public final class RIntVector extends RVector<int[]> implements RAbstractIntVect complete = false; } } + + private static final class FastPathAccess extends FastPathFromIntAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected int getInt(Object store, int index) { + return hasStore ? ((int[]) store)[index] : NativeDataAccess.getIntNativeMirrorData(store, index); + } + + @Override + protected void setInt(Object store, int index, int value) { + if (hasStore) { + ((int[]) store)[index] = value; + } else { + NativeDataAccess.setNativeMirrorIntData(store, index, value); + } + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromIntAccess SLOW_PATH_ACCESS = new SlowPathFromIntAccess() { + @Override + protected int getInt(Object store, int index) { + RIntVector vector = (RIntVector) store; + return NativeDataAccess.getData(vector, vector.data, index); + } + + @Override + protected void setInt(Object store, int index, int value) { + RIntVector vector = (RIntVector) store; + NativeDataAccess.setData(vector, vector.data, index, value); + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RInteger.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RInteger.java index eb0ea1f399..0eed431c3a 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RInteger.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RInteger.java @@ -26,8 +26,12 @@ import com.oracle.truffle.api.CompilerDirectives.ValueType; import com.oracle.truffle.api.profiles.ConditionProfile; import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractIntVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromIntAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromIntAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; @ValueType public final class RInteger extends RScalarVector implements RAbstractIntVector { @@ -95,4 +99,35 @@ public final class RInteger extends RScalarVector implements RAbstractIntVector public boolean isNA() { return RRuntime.isNA(value); } + + private static final class FastPathAccess extends FastPathFromIntAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected int getInt(Object store, int index) { + assert index == 0; + return ((RInteger) store).value; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromIntAccess SLOW_PATH_ACCESS = new SlowPathFromIntAccess() { + @Override + protected int getInt(Object store, int index) { + assert index == 0; + return ((RInteger) store).value; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLanguage.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLanguage.java index 7fa8714d7a..43c77e0ca5 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLanguage.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLanguage.java @@ -30,6 +30,9 @@ import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.context.RContext; import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromListAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromListAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; import com.oracle.truffle.r.runtime.nodes.RBaseNode; /** @@ -76,6 +79,11 @@ public final class RLanguage extends RSharingAttributeStorage implements RAbstra this.length = length; } + @Override + public Object getInternalStore() { + return this; + } + @TruffleBoundary public static Object fromList(Object o, RLanguage.RepType type) { RList l; @@ -276,4 +284,45 @@ public final class RLanguage extends RSharingAttributeStorage implements RAbstra private void setNamesOnPairList(RStringVector names) { list.setNames(names); } + + private static final class FastPathAccess extends FastPathFromListAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + public RType getType() { + return RType.Language; + } + + @TruffleBoundary + @Override + protected Object getListElement(Object store, int index) { + return ((RLanguage) store).getDataAtAsObject(index); + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromListAccess SLOW_PATH_ACCESS = new SlowPathFromListAccess() { + @Override + public RType getType() { + return RType.Language; + } + + @TruffleBoundary + @Override + protected Object getListElement(Object store, int index) { + return ((RLanguage) store).getDataAtAsObject(index); + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RList.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RList.java index 650c2f1173..ee3847e78b 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RList.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RList.java @@ -25,7 +25,12 @@ package com.oracle.truffle.r.runtime.data; import java.util.Arrays; import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary; +import com.oracle.truffle.r.runtime.RType; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractListVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromListAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromListAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; public final class RList extends RListBase implements RAbstractListVector { @@ -83,4 +88,53 @@ public final class RList extends RListBase implements RAbstractListVector { protected RList internalCopyResized(int size, boolean fillNA, int[] dimensions) { return RDataFactory.createList(copyResizedData(size, fillNA), dimensions); } + + private static final class FastPathAccess extends FastPathFromListAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + public RType getType() { + return RType.List; + } + + @Override + protected Object getListElement(Object store, int index) { + return ((Object[]) store)[index]; + } + + @Override + protected void setListElement(Object store, int index, Object value) { + ((Object[]) store)[index] = value; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromListAccess SLOW_PATH_ACCESS = new SlowPathFromListAccess() { + @Override + public RType getType() { + return RType.List; + } + + @Override + protected Object getListElement(Object store, int index) { + return ((RList) store).data[index]; + } + + @Override + protected void setListElement(Object store, int index, Object value) { + ((RList) store).data[index] = value; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RListBase.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RListBase.java index 4538d1a4ee..504c6d4173 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RListBase.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RListBase.java @@ -24,7 +24,6 @@ package com.oracle.truffle.r.runtime.data; import java.util.Arrays; -import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.Utils; import com.oracle.truffle.r.runtime.data.model.RAbstractListBaseVector; import com.oracle.truffle.r.runtime.data.model.RAbstractListVector; @@ -94,11 +93,6 @@ public abstract class RListBase extends RVector<Object[]> implements RAbstractLi ((Object[]) store)[index] = value; } - @Override - public String toString() { - return toString(i -> RRuntime.toString(getDataAt(i))); - } - @Override public final boolean verify() { for (Object item : data) { diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLogical.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLogical.java index 7fade92f1b..c65961c55e 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLogical.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLogical.java @@ -26,8 +26,12 @@ import com.oracle.truffle.api.CompilerDirectives.ValueType; import com.oracle.truffle.api.profiles.ConditionProfile; import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractLogicalVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromLogicalAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromLogicalAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; @ValueType public final class RLogical extends RScalarVector implements RAbstractLogicalVector { @@ -105,4 +109,35 @@ public final class RLogical extends RScalarVector implements RAbstractLogicalVec public static boolean isValid(byte left) { return left == RRuntime.LOGICAL_NA || left == RRuntime.LOGICAL_FALSE || left == RRuntime.LOGICAL_TRUE; } + + private static final class FastPathAccess extends FastPathFromLogicalAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected byte getLogical(Object store, int index) { + assert index == 0; + return ((RLogical) store).value; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromLogicalAccess SLOW_PATH_ACCESS = new SlowPathFromLogicalAccess() { + @Override + protected byte getLogical(Object store, int index) { + assert index == 0; + return ((RLogical) store).value; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLogicalVector.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLogicalVector.java index 1d5bcca090..048c4df3c3 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLogicalVector.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RLogicalVector.java @@ -29,8 +29,12 @@ import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.Utils; import com.oracle.truffle.r.runtime.data.closures.RClosures; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractLogicalVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromLogicalAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromLogicalAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; import com.oracle.truffle.r.runtime.ops.na.NACheck; public final class RLogicalVector extends RVector<byte[]> implements RAbstractLogicalVector { @@ -95,6 +99,10 @@ public final class RLogicalVector extends RVector<byte[]> implements RAbstractLo NativeDataAccess.setData(this, (byte[]) store, index, value); } + public void setDataAt(int index, byte value) { + NativeDataAccess.setData(this, data, index, value); + } + @Override public byte getDataAt(Object store, int index) { assert data == store; @@ -128,11 +136,6 @@ public final class RLogicalVector extends RVector<byte[]> implements RAbstractLo return NativeDataAccess.getDataLength(this, data); } - @Override - public String toString() { - return toString(i -> RRuntime.logicalToString(getDataAt(i))); - } - @Override public boolean verify() { if (isComplete()) { @@ -249,4 +252,49 @@ public final class RLogicalVector extends RVector<byte[]> implements RAbstractLo } return result; } + + private static final class FastPathAccess extends FastPathFromLogicalAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected byte getLogical(Object store, int index) { + return hasStore ? ((byte[]) store)[index] : NativeDataAccess.getLogicalNativeMirrorData(store, index); + } + + @Override + protected void setLogical(Object store, int index, byte value) { + if (hasStore) { + ((byte[]) store)[index] = value; + } else { + NativeDataAccess.setNativeMirrorLogicalData(store, index, value); + } + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromLogicalAccess SLOW_PATH_ACCESS = new SlowPathFromLogicalAccess() { + @Override + protected byte getLogical(Object store, int index) { + RLogicalVector vector = (RLogicalVector) store; + return NativeDataAccess.getData(vector, vector.data, index); + } + + @Override + protected void setLogical(Object store, int index, byte value) { + RLogicalVector vector = (RLogicalVector) store; + NativeDataAccess.setData(vector, vector.data, index, value); + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RPairList.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RPairList.java index 18c853f56a..74765b994d 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RPairList.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RPairList.java @@ -33,6 +33,9 @@ import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.Utils; import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromListAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromListAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; import com.oracle.truffle.r.runtime.gnur.SEXPTYPE; /** @@ -73,6 +76,11 @@ public final class RPairList extends RSharingAttributeStorage implements RAbstra this.type = type; } + @Override + public Object getInternalStore() { + return this; + } + /** * Creates a new pair list of given size > 0. Note: pair list of size 0 is NULL. */ @@ -375,4 +383,45 @@ public final class RPairList extends RSharingAttributeStorage implements RAbstra } }; } + + private static final class FastPathAccess extends FastPathFromListAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + public RType getType() { + return RType.PairList; + } + + @TruffleBoundary + @Override + protected Object getListElement(Object store, int index) { + return ((RPairList) store).getDataAtAsObject(index); + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromListAccess SLOW_PATH_ACCESS = new SlowPathFromListAccess() { + @Override + public RType getType() { + return RType.PairList; + } + + @TruffleBoundary + @Override + protected Object getListElement(Object store, int index) { + return ((RPairList) store).getDataAtAsObject(index); + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RRaw.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RRaw.java index ff3e3945be..a99b64ca06 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RRaw.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RRaw.java @@ -26,8 +26,12 @@ import com.oracle.truffle.api.CompilerDirectives.ValueType; import com.oracle.truffle.api.profiles.ConditionProfile; import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractRawVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromRawAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromRawAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; @ValueType public final class RRaw extends RScalarVector implements RAbstractRawVector { @@ -99,4 +103,35 @@ public final class RRaw extends RScalarVector implements RAbstractRawVector { public static RRaw valueOf(byte value) { return new RRaw(value); } + + private static final class FastPathAccess extends FastPathFromRawAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected byte getRaw(Object store, int index) { + assert index == 0; + return ((RRaw) store).value; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromRawAccess SLOW_PATH_ACCESS = new SlowPathFromRawAccess() { + @Override + protected byte getRaw(Object store, int index) { + assert index == 0; + return ((RRaw) store).value; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RRawVector.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RRawVector.java index 97c9435c9d..d953368972 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RRawVector.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RRawVector.java @@ -25,12 +25,15 @@ package com.oracle.truffle.r.runtime.data; import java.util.Arrays; import com.oracle.truffle.api.profiles.ConditionProfile; -import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.Utils; import com.oracle.truffle.r.runtime.data.closures.RClosures; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractRawVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromRawAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromRawAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; import com.oracle.truffle.r.runtime.ops.na.NACheck; public final class RRawVector extends RVector<byte[]> implements RAbstractRawVector { @@ -92,18 +95,16 @@ public final class RRawVector extends RVector<byte[]> implements RAbstractRawVec return NativeDataAccess.getData(this, data, index); } - @Override - public byte getRawDataAt(Object store, int index) { - assert data == store; - return NativeDataAccess.getData(this, (byte[]) store, index); - } - @Override public void setRawDataAt(Object store, int index, byte value) { assert data == store; NativeDataAccess.setData(this, (byte[]) store, index, value); } + public void setRawDataAt(int index, byte value) { + NativeDataAccess.setData(this, data, index, value); + } + @Override protected RRawVector internalCopy() { if (data != null) { @@ -118,11 +119,6 @@ public final class RRawVector extends RVector<byte[]> implements RAbstractRawVec return NativeDataAccess.getDataLength(this, data); } - @Override - public String toString() { - return toString(i -> RRuntime.rawToString(getRawDataAt(i))); - } - @Override public boolean verify() { return true; @@ -202,4 +198,49 @@ public final class RRawVector extends RVector<byte[]> implements RAbstractRawVec complete = false; } } + + private static final class FastPathAccess extends FastPathFromRawAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected byte getRaw(Object store, int index) { + return hasStore ? ((byte[]) store)[index] : NativeDataAccess.getRawNativeMirrorData(store, index); + } + + @Override + protected void setRaw(Object store, int index, byte value) { + if (hasStore) { + ((byte[]) store)[index] = value; + } else { + NativeDataAccess.setNativeMirrorRawData(store, index, value); + } + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromRawAccess SLOW_PATH_ACCESS = new SlowPathFromRawAccess() { + @Override + protected byte getRaw(Object store, int index) { + RRawVector vector = (RRawVector) store; + return NativeDataAccess.getData(vector, vector.data, index); + } + + @Override + protected void setRaw(Object store, int index, byte value) { + RRawVector vector = (RRawVector) store; + NativeDataAccess.setData(vector, vector.data, index, value); + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RScalarList.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RScalarList.java index 1ea23adc66..044cdffb58 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RScalarList.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RScalarList.java @@ -25,8 +25,12 @@ package com.oracle.truffle.r.runtime.data; import com.oracle.truffle.api.CompilerDirectives.ValueType; import com.oracle.truffle.api.profiles.ConditionProfile; import com.oracle.truffle.r.runtime.RType; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractListVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromListAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromListAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; @ValueType public final class RScalarList extends RScalarVector implements RAbstractListVector { @@ -87,4 +91,45 @@ public final class RScalarList extends RScalarVector implements RAbstractListVec public boolean isNA() { return false; } + + private static final class FastPathAccess extends FastPathFromListAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + public RType getType() { + return RType.List; + } + + @Override + protected Object getListElement(Object store, int index) { + assert index == 0; + return ((RScalarList) store).value; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromListAccess SLOW_PATH_ACCESS = new SlowPathFromListAccess() { + @Override + public RType getType() { + return RType.List; + } + + @Override + protected Object getListElement(Object store, int index) { + assert index == 0; + return ((RScalarList) store).value; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RScalarVector.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RScalarVector.java index 80f1da3147..76ce7c6cfb 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RScalarVector.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RScalarVector.java @@ -36,6 +36,11 @@ public abstract class RScalarVector extends RScalar implements RAbstractVector { return this; } + @Override + public Object getInternalStore() { + return this; + } + @Override public void setComplete(boolean complete) { // scalar vectors don't need this information. diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RSequence.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RSequence.java index d9aa895489..8781775fad 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RSequence.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RSequence.java @@ -41,6 +41,11 @@ public abstract class RSequence implements RAbstractVector { this.length = length; } + @Override + public Object getInternalStore() { + return this; + } + @Override public final int getLength() { return length; diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RString.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RString.java index abf51a343f..2929ffb79f 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RString.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RString.java @@ -27,8 +27,12 @@ import com.oracle.truffle.api.CompilerDirectives.ValueType; import com.oracle.truffle.api.profiles.ConditionProfile; import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractStringVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromStringAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromStringAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; @ValueType public final class RString extends RScalarVector implements RAbstractStringVector { @@ -95,4 +99,35 @@ public final class RString extends RScalarVector implements RAbstractStringVecto throw new AssertionError(); } } + + private static final class FastPathAccess extends FastPathFromStringAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected String getString(Object store, int index) { + assert index == 0; + return ((RString) store).value; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromStringAccess SLOW_PATH_ACCESS = new SlowPathFromStringAccess() { + @Override + protected String getString(Object store, int index) { + assert index == 0; + return ((RString) store).value; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RStringSequence.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RStringSequence.java index 1b90cafe41..220df12974 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RStringSequence.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RStringSequence.java @@ -29,8 +29,12 @@ import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.Utils; import com.oracle.truffle.r.runtime.data.closures.RClosures; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractStringVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromStringAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromStringAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; public final class RStringSequence extends RSequence implements RAbstractStringVector { @@ -168,4 +172,37 @@ public final class RStringSequence extends RSequence implements RAbstractStringV CompilerAsserts.neverPartOfCompilation(); return "[\"" + getStartObject() + "\" - \"" + prefix + getEnd() + suffix + "\"]"; } + + private static final class FastPathAccess extends FastPathFromStringAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected String getString(Object store, int index) { + RStringSequence vector = (RStringSequence) store; + assert index >= 0 && index < vector.getLength(); + return vector.prefix + (vector.start + vector.stride * index) + vector.suffix; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromStringAccess SLOW_PATH_ACCESS = new SlowPathFromStringAccess() { + @Override + protected String getString(Object store, int index) { + RStringSequence vector = (RStringSequence) store; + assert index >= 0 && index < vector.getLength(); + return vector.prefix + (vector.start + vector.stride * index) + vector.suffix; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RStringVector.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RStringVector.java index 4a0374063a..682431c382 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RStringVector.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RStringVector.java @@ -30,8 +30,12 @@ import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.Utils; import com.oracle.truffle.r.runtime.data.closures.RClosures; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractStringVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.FastPathVectorAccess.FastPathFromStringAccess; +import com.oracle.truffle.r.runtime.data.nodes.SlowPathVectorAccess.SlowPathFromStringAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; import com.oracle.truffle.r.runtime.ops.na.NACheck; public final class RStringVector extends RVector<String[]> implements RAbstractStringVector { @@ -78,6 +82,10 @@ public final class RStringVector extends RVector<String[]> implements RAbstractS ((String[]) store)[index] = value; } + public void setDataAt(int index, String value) { + data[index] = value; + } + @Override public String getDataAt(Object store, int index) { assert data == store; @@ -119,11 +127,6 @@ public final class RStringVector extends RVector<String[]> implements RAbstractS return data; } - @Override - public String toString() { - return toString(i -> getDataAt(i)); - } - @Override public boolean verify() { if (isComplete()) { @@ -219,4 +222,45 @@ public final class RStringVector extends RVector<String[]> implements RAbstractS public void setElement(int i, Object value) { data[i] = (String) value; } + + private static final class FastPathAccess extends FastPathFromStringAccess { + + FastPathAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected String getString(Object store, int index) { + assert hasStore; + return ((String[]) store)[index]; + } + + @Override + protected void setString(Object store, int index, String value) { + assert hasStore; + ((String[]) store)[index] = value; + } + } + + @Override + public VectorAccess access() { + return new FastPathAccess(this); + } + + private static final SlowPathFromStringAccess SLOW_PATH_ACCESS = new SlowPathFromStringAccess() { + @Override + protected String getString(Object store, int index) { + return ((RStringVector) store).data[index]; + } + + @Override + protected void setString(Object store, int index, String value) { + ((RStringVector) store).data[index] = value; + } + }; + + @Override + public VectorAccess slowPathAccess() { + return SLOW_PATH_ACCESS; + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RVector.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RVector.java index 8956b2df5d..47b40aec93 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RVector.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/RVector.java @@ -24,8 +24,6 @@ package com.oracle.truffle.r.runtime.data; import static com.oracle.truffle.r.runtime.RError.NO_CALLER; -import java.util.function.Function; - import com.oracle.truffle.api.CompilerAsserts; import com.oracle.truffle.api.CompilerDirectives; import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary; @@ -40,6 +38,8 @@ 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; import com.oracle.truffle.r.runtime.data.nodes.GetReadonlyData; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess.SequentialIterator; import com.oracle.truffle.r.runtime.nodes.RBaseNode; import com.oracle.truffle.r.runtime.ops.na.NACheck; @@ -780,18 +780,25 @@ public abstract class RVector<ArrayT> extends RSharingAttributeStorage implement private static final int MAX_TOSTRING_LENGTH = 100; - protected final String toString(Function<Integer, String> element) { + @Override + public final String toString() { CompilerAsserts.neverPartOfCompilation(); StringBuilder str = new StringBuilder("["); - for (int i = 0; i < getLength(); i++) { - if (i > 0) { - str.append(", "); - } - str.append(element.apply(i)); - if (str.length() > MAX_TOSTRING_LENGTH - 1) { - str.setLength(MAX_TOSTRING_LENGTH - 4); - str.append("..."); - break; + VectorAccess access = slowPathAccess(); + try (SequentialIterator iter = access.access(this)) { + if (access.next(iter)) { + while (true) { + str.append(access.getType().isAtomic() ? access.getString(iter) : access.getListElement(iter).toString()); + if (!access.next(iter)) { + break; + } + str.append(", "); + if (str.length() > MAX_TOSTRING_LENGTH - 1) { + str.setLength(MAX_TOSTRING_LENGTH - 4); + str.append("..."); + break; + } + } } } return str.append(']').toString(); diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/closures/RToVectorClosure.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/closures/RToVectorClosure.java index e8c5104485..7495f52ae8 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/closures/RToVectorClosure.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/closures/RToVectorClosure.java @@ -24,6 +24,7 @@ package com.oracle.truffle.r.runtime.data.closures; import com.oracle.truffle.api.object.DynamicObject; import com.oracle.truffle.api.profiles.ConditionProfile; +import com.oracle.truffle.r.runtime.RInternalError; import com.oracle.truffle.r.runtime.RType; import com.oracle.truffle.r.runtime.data.RList; import com.oracle.truffle.r.runtime.data.RStringVector; @@ -31,6 +32,7 @@ import com.oracle.truffle.r.runtime.data.RTypedValue; import com.oracle.truffle.r.runtime.data.RVector; import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; abstract class RToVectorClosure implements RAbstractVector { @@ -49,8 +51,8 @@ abstract class RToVectorClosure implements RAbstractVector { } @Override - public EmptyInternalStore getInternalStore() { - return EmptyInternalStore.INSTANCE; + public Object getInternalStore() { + return this; } @Override @@ -184,4 +186,14 @@ abstract class RToVectorClosure implements RAbstractVector { // first materialize and then cast and do not create a closure over a closure. return materialize().castSafe(type, isNAProfile, keepAttrs); } + + @Override + public final VectorAccess access() { + throw RInternalError.shouldNotReachHere("access() for " + getClass().getSimpleName()); + } + + @Override + public final VectorAccess slowPathAccess() { + throw RInternalError.shouldNotReachHere("slowPathAccess() for " + getClass().getSimpleName()); + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractComplexVector.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractComplexVector.java index b2d9e6cd06..4e1a9ef486 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractComplexVector.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractComplexVector.java @@ -36,10 +36,6 @@ public interface RAbstractComplexVector extends RAbstractAtomicVector { RComplex getDataAt(int index); - default RComplex getDataAt(@SuppressWarnings("unused") Object store, int index) { - return getDataAt(index); - } - @Override RComplexVector materialize(); diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractContainer.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractContainer.java index 8723bed3da..93742f57b5 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractContainer.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractContainer.java @@ -28,6 +28,7 @@ import com.oracle.truffle.r.runtime.data.RAttributable; import com.oracle.truffle.r.runtime.data.RList; import com.oracle.truffle.r.runtime.data.RStringVector; import com.oracle.truffle.r.runtime.data.RTypedValue; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; public interface RAbstractContainer extends RAttributable, RTypedValue { @@ -67,9 +68,7 @@ public interface RAbstractContainer extends RAttributable, RTypedValue { * vector's fields, but instead read the necessary data from a local variable, which could be * beneficial when in loop. */ - default Object getInternalStore() { - return EmptyInternalStore.INSTANCE; - } + Object getInternalStore(); default RStringVector getNames() { CompilerAsserts.neverPartOfCompilation(); @@ -101,10 +100,7 @@ public interface RAbstractContainer extends RAttributable, RTypedValue { setAttr(RRuntime.ROWNAMES_ATTR_KEY, rowNames); } - final class EmptyInternalStore { - private EmptyInternalStore() { - } + VectorAccess access(); - public static final EmptyInternalStore INSTANCE = new EmptyInternalStore(); - } + VectorAccess slowPathAccess(); } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractRawVector.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractRawVector.java index 5f8697e39d..53f2c49bd1 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractRawVector.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/model/RAbstractRawVector.java @@ -33,10 +33,6 @@ public interface RAbstractRawVector extends RAbstractAtomicVector { return RRaw.valueOf(getRawDataAt(index)); } - default byte getRawDataAt(@SuppressWarnings("unused") Object store, int index) { - return getRawDataAt(index); - } - @SuppressWarnings("unused") default void setRawDataAt(Object store, int index, byte value) { throw new UnsupportedOperationException(); diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/FastPathVectorAccess.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/FastPathVectorAccess.java new file mode 100644 index 0000000000..e17d63ac7c --- /dev/null +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/FastPathVectorAccess.java @@ -0,0 +1,635 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package com.oracle.truffle.r.runtime.data.nodes; + +import com.oracle.truffle.api.CompilerAsserts; +import com.oracle.truffle.r.runtime.RError; +import com.oracle.truffle.r.runtime.RError.Message; +import com.oracle.truffle.r.runtime.RInternalError; +import com.oracle.truffle.r.runtime.RRuntime; +import com.oracle.truffle.r.runtime.RType; +import com.oracle.truffle.r.runtime.context.RContext; +import com.oracle.truffle.r.runtime.data.RComplex; +import com.oracle.truffle.r.runtime.data.RNull; +import com.oracle.truffle.r.runtime.data.RRaw; +import com.oracle.truffle.r.runtime.data.RVector; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; + +/** + * Base classes for {@link VectorAccess} implementations that are used on the fast path. For + * implementation reasons, most of this code is mirrored in {@link SlowPathVectorAccess}, so that + * any changes need to be mirrored there. + */ +public abstract class FastPathVectorAccess extends VectorAccess { + + protected boolean naReported; // TODO: move this into the iterator + + protected FastPathVectorAccess(RAbstractContainer value) { + super(value.getClass(), value.getInternalStore() != null); + } + + @Override + protected final Object getStore(RAbstractContainer vector) { + return hasStore ? vector.getInternalStore() : ((RVector<?>) vector).getNativeMirror(); + } + + protected final void warning(RError.Message message) { + CompilerAsserts.neverPartOfCompilation(); + if (!naReported) { + RError.warning(RError.SHOW_CALLER, message); + naReported = true; + } + } + + public abstract static class FastPathFromIntAccess extends FastPathVectorAccess { + + public FastPathFromIntAccess(RAbstractContainer value) { + super(value); + } + + @Override + public final RType getType() { + return RType.Integer; + } + + @Override + protected final double getDouble(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RRuntime.DOUBLE_NA : RRuntime.int2doubleNoCheck(value); + } + + @Override + protected final byte getRaw(Object store, int index) { + int value = getInt(store, index); + byte result = (byte) value; + if ((result & 0xff) != value) { + warning(Message.OUT_OF_RANGE); + return 0; + } + return result; + } + + @Override + protected final byte getLogical(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RRuntime.LOGICAL_NA : RRuntime.int2logicalNoCheck(value); + } + + @Override + protected final RComplex getComplex(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RComplex.createNA() : RRuntime.int2complexNoCheck(value); + } + + @Override + protected final double getComplexR(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_REAL_PART : value; + } + + @Override + protected final double getComplexI(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_IMAGINARY_PART : 0; + } + + @Override + protected final String getString(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RRuntime.STRING_NA : RRuntime.intToStringNoCheck(value); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getInt(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setInt(store, index, sourceAccess.getInt(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setInt(store, index, sourceAccess.getInt(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + setInt(store, index, RRuntime.INT_NA); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.check(getInt(store, index)); + } + } + + public abstract static class FastPathFromDoubleAccess extends FastPathVectorAccess { + + public FastPathFromDoubleAccess(RAbstractContainer value) { + super(value); + } + + @Override + public final RType getType() { + return RType.Double; + } + + @Override + protected final int getInt(Object store, int index) { + double value = getDouble(store, index); + if (Double.isNaN(value)) { + na.enable(true); + return RRuntime.INT_NA; + } + if (value > Integer.MAX_VALUE || value <= Integer.MIN_VALUE) { + na.enable(true); + warning(Message.NA_INTRODUCED_COERCION_INT); + return RRuntime.INT_NA; + } + return (int) value; + } + + @Override + protected final byte getRaw(Object store, int index) { + int value = (int) getDouble(store, index); + byte result = (byte) value; + if ((result & 0xff) != value) { + warning(Message.OUT_OF_RANGE); + return 0; + } + return result; + } + + @Override + protected final byte getLogical(Object store, int index) { + double value = getDouble(store, index); + return na.check(value) ? RRuntime.LOGICAL_NA : RRuntime.double2logicalNoCheck(value); + } + + @Override + protected final RComplex getComplex(Object store, int index) { + double value = getDouble(store, index); + return na.check(value) ? RComplex.createNA() : RRuntime.double2complexNoCheck(value); + } + + @Override + protected final double getComplexR(Object store, int index) { + double value = getDouble(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_REAL_PART : value; + } + + @Override + protected final double getComplexI(Object store, int index) { + double value = getDouble(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_IMAGINARY_PART : 0; + } + + @Override + protected final String getString(Object store, int index) { + double value = getDouble(store, index); + return na.check(value) ? RRuntime.STRING_NA : RContext.getRRuntimeASTAccess().encodeDouble(value); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getDouble(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setDouble(store, index, sourceAccess.getDouble(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setDouble(store, index, sourceAccess.getDouble(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + setDouble(store, index, RRuntime.DOUBLE_NA); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.check(getDouble(store, index)); + } + } + + public abstract static class FastPathFromLogicalAccess extends FastPathVectorAccess { + + public FastPathFromLogicalAccess(RAbstractContainer value) { + super(value); + } + + @Override + public final RType getType() { + return RType.Logical; + } + + @Override + protected final int getInt(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RRuntime.INT_NA : RRuntime.logical2intNoCheck(value); + } + + @Override + protected final double getDouble(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RRuntime.DOUBLE_NA : RRuntime.logical2doubleNoCheck(value); + } + + @Override + protected final byte getRaw(Object store, int index) { + byte value = getLogical(store, index); + if (na.check(value)) { + warning(Message.OUT_OF_RANGE); + return 0; + } + return value; + } + + @Override + protected final RComplex getComplex(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RComplex.createNA() : RRuntime.logical2complexNoCheck(value); + } + + @Override + protected final double getComplexR(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_REAL_PART : value; + } + + @Override + protected final double getComplexI(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_IMAGINARY_PART : 0; + } + + @Override + protected final String getString(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RRuntime.STRING_NA : RRuntime.logicalToStringNoCheck(value); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getLogical(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setLogical(store, index, sourceAccess.getLogical(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setLogical(store, index, sourceAccess.getLogical(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + setLogical(store, index, RRuntime.LOGICAL_NA); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.check(getLogical(store, index)); + } + } + + public abstract static class FastPathFromRawAccess extends FastPathVectorAccess { + + public FastPathFromRawAccess(RAbstractContainer value) { + super(value); + } + + @Override + public final RType getType() { + return RType.Raw; + } + + @Override + protected final int getInt(Object store, int index) { + return getRaw(store, index) & 0xff; + } + + @Override + protected final double getDouble(Object store, int index) { + return getRaw(store, index) & 0xff; + } + + @Override + protected final byte getLogical(Object store, int index) { + return getRaw(store, index) == 0 ? RRuntime.LOGICAL_FALSE : RRuntime.LOGICAL_TRUE; + } + + @Override + protected final RComplex getComplex(Object store, int index) { + return RComplex.valueOf(getRaw(store, index) & 0xff, 0); + } + + @Override + protected final double getComplexR(Object store, int index) { + return getRaw(store, index) & 0xff; + } + + @Override + protected final double getComplexI(Object store, int index) { + return 0; + } + + @Override + protected final String getString(Object store, int index) { + return RRuntime.rawToHexString(getRaw(store, index)); + } + + @Override + protected final Object getListElement(Object store, int index) { + return RRaw.valueOf(getRaw(store, index)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setRaw(store, index, sourceAccess.getRaw(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setRaw(store, index, sourceAccess.getRaw(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + /* + * There is no raw NA, but places that write NA for other types usually write 0 for raw. + */ + setRaw(store, index, (byte) 0); + } + + @Override + protected boolean isNA(Object store, int index) { + return false; + } + } + + public abstract static class FastPathFromComplexAccess extends FastPathVectorAccess { + + public FastPathFromComplexAccess(RAbstractContainer value) { + super(value); + } + + @Override + public final RType getType() { + return RType.Complex; + } + + @Override + protected final int getInt(Object store, int index) { + double value = getComplexR(store, index); + if (Double.isNaN(value)) { + na.enable(true); + return RRuntime.INT_NA; + } + if (value > Integer.MAX_VALUE || value <= Integer.MIN_VALUE) { + na.enable(true); + warning(Message.NA_INTRODUCED_COERCION_INT); + return RRuntime.INT_NA; + } + if (getComplexI(store, index) != 0) { + warning(Message.IMAGINARY_PARTS_DISCARDED_IN_COERCION); + } + return (int) value; + } + + @Override + protected final double getDouble(Object store, int index) { + double value = getComplexR(store, index); + if (Double.isNaN(value)) { + na.enable(true); + return RRuntime.DOUBLE_NA; + } + if (getComplexI(store, index) != 0) { + warning(Message.IMAGINARY_PARTS_DISCARDED_IN_COERCION); + } + return value; + } + + @Override + protected final byte getRaw(Object store, int index) { + double value = getComplexR(store, index); + if (Double.isNaN(value) || value < 0 || value >= 256) { + warning(Message.OUT_OF_RANGE); + return 0; + } + if (getComplexI(store, index) != 0) { + warning(Message.IMAGINARY_PARTS_DISCARDED_IN_COERCION); + } + return (byte) value; + } + + @Override + protected final byte getLogical(Object store, int index) { + RComplex value = getComplex(store, index); + return na.check(value) ? RRuntime.LOGICAL_NA : RRuntime.complex2logicalNoCheck(value); + } + + @Override + protected final String getString(Object store, int index) { + RComplex value = getComplex(store, index); + return na.check(value) ? RRuntime.STRING_NA : RContext.getRRuntimeASTAccess().encodeComplex(value); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getComplex(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setComplex(store, index, sourceAccess.getComplexR(sourceIter), sourceAccess.getComplexI(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setComplex(store, index, sourceAccess.getComplexR(sourceIter, sourceIndex), sourceAccess.getComplexI(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + setComplex(store, index, RRuntime.COMPLEX_NA_REAL_PART, RRuntime.COMPLEX_NA_IMAGINARY_PART); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.check(getComplexR(store, index), getComplexI(store, index)); + } + } + + public abstract static class FastPathFromStringAccess extends FastPathVectorAccess { + + public FastPathFromStringAccess(RAbstractContainer value) { + super(value); + } + + @Override + public final RType getType() { + return RType.Character; + } + + @Override + protected final int getInt(Object store, int index) { + return na.convertStringToInt(getString(store, index)); + } + + @Override + protected final double getDouble(Object store, int index) { + return na.convertStringToDouble(getString(store, index)); + } + + @Override + protected final byte getRaw(Object store, int index) { + int value = na.convertStringToInt(getString(store, index)); + return value >= 0 && value <= 255 ? (byte) value : 0; + } + + @Override + protected final byte getLogical(Object store, int index) { + return na.convertStringToLogical(getString(store, index)); + } + + @Override + protected final RComplex getComplex(Object store, int index) { + return na.convertStringToComplex(getString(store, index)); + } + + @Override + protected final double getComplexR(Object store, int index) { + return na.convertStringToComplex(getString(store, index)).getRealPart(); + } + + @Override + protected final double getComplexI(Object store, int index) { + return na.convertStringToComplex(getString(store, index)).getImaginaryPart(); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getString(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setString(store, index, sourceAccess.getString(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setString(store, index, sourceAccess.getString(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + setString(store, index, RRuntime.STRING_NA); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.check(getString(store, index)); + } + } + + public abstract static class FastPathFromListAccess extends FastPathVectorAccess { + + public FastPathFromListAccess(RAbstractContainer value) { + super(value); + } + + @Override + protected final int getInt(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final double getDouble(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final byte getRaw(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final byte getLogical(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final RComplex getComplex(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final double getComplexR(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final double getComplexI(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final String getString(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setListElement(store, index, sourceAccess.getListElement(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setListElement(store, index, sourceAccess.getListElement(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + /* + * There is no list NA, but places that write NA for other types usually write NULL for + * lists. + */ + setListElement(store, index, RNull.instance); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.checkListElement(getListElement(store, index)); + } + } +} diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/SlowPathVectorAccess.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/SlowPathVectorAccess.java new file mode 100644 index 0000000000..18fb51f466 --- /dev/null +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/SlowPathVectorAccess.java @@ -0,0 +1,606 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package com.oracle.truffle.r.runtime.data.nodes; + +import com.oracle.truffle.api.CompilerAsserts; +import com.oracle.truffle.r.runtime.RError; +import com.oracle.truffle.r.runtime.RError.Message; +import com.oracle.truffle.r.runtime.RInternalError; +import com.oracle.truffle.r.runtime.RRuntime; +import com.oracle.truffle.r.runtime.RType; +import com.oracle.truffle.r.runtime.context.RContext; +import com.oracle.truffle.r.runtime.data.RComplex; +import com.oracle.truffle.r.runtime.data.RNull; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; + +/** + * Base classes for {@link VectorAccess} implementations that are used on the slow path, i.e., that + * are created as singletons. For implementation reasons, most of this code is mirrored in + * {@link FastPathVectorAccess}, so that any changes need to be mirrored there. + */ +public abstract class SlowPathVectorAccess extends VectorAccess { + + protected boolean naReported; // TODO: move this into the iterator + + protected SlowPathVectorAccess() { + // VectorAccess.supports has an assertion that relies on this being RAbstractContainer.class + super(RAbstractContainer.class, true); + } + + @Override + protected final Object getStore(RAbstractContainer vector) { + return vector; + } + + protected final void warning(RError.Message message) { + CompilerAsserts.neverPartOfCompilation(); + if (!naReported) { + RError.warning(RError.SHOW_CALLER, message); + naReported = true; + } + } + + public abstract static class SlowPathFromIntAccess extends SlowPathVectorAccess { + + @Override + public final RType getType() { + return RType.Integer; + } + + @Override + protected final double getDouble(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RRuntime.DOUBLE_NA : RRuntime.int2doubleNoCheck(value); + } + + @Override + protected final byte getRaw(Object store, int index) { + int value = getInt(store, index); + byte result = (byte) value; + if ((result & 0xff) != value) { + warning(Message.OUT_OF_RANGE); + return 0; + } + return result; + } + + @Override + protected final byte getLogical(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RRuntime.LOGICAL_NA : RRuntime.int2logicalNoCheck(value); + } + + @Override + protected final RComplex getComplex(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RComplex.createNA() : RRuntime.int2complexNoCheck(value); + } + + @Override + protected final double getComplexR(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_REAL_PART : value; + } + + @Override + protected final double getComplexI(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_IMAGINARY_PART : 0; + } + + @Override + protected final String getString(Object store, int index) { + int value = getInt(store, index); + return na.check(value) ? RRuntime.STRING_NA : RRuntime.intToStringNoCheck(value); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getInt(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setInt(store, index, sourceAccess.getInt(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setInt(store, index, sourceAccess.getInt(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + setInt(store, index, RRuntime.INT_NA); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.check(getInt(store, index)); + } + } + + public abstract static class SlowPathFromDoubleAccess extends SlowPathVectorAccess { + + @Override + public final RType getType() { + return RType.Double; + } + + @Override + protected final int getInt(Object store, int index) { + double value = getDouble(store, index); + if (Double.isNaN(value)) { + na.enable(true); + return RRuntime.INT_NA; + } + if (value > Integer.MAX_VALUE || value <= Integer.MIN_VALUE) { + na.enable(true); + warning(Message.NA_INTRODUCED_COERCION_INT); + return RRuntime.INT_NA; + } + return (int) value; + } + + @Override + protected final byte getRaw(Object store, int index) { + int value = (int) getDouble(store, index); + byte result = (byte) value; + if ((result & 0xff) != value) { + warning(Message.OUT_OF_RANGE); + return 0; + } + return result; + } + + @Override + protected final byte getLogical(Object store, int index) { + double value = getDouble(store, index); + return na.check(value) ? RRuntime.LOGICAL_NA : RRuntime.double2logicalNoCheck(value); + } + + @Override + protected final RComplex getComplex(Object store, int index) { + double value = getDouble(store, index); + return na.check(value) ? RComplex.createNA() : RRuntime.double2complexNoCheck(value); + } + + @Override + protected final double getComplexR(Object store, int index) { + double value = getDouble(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_REAL_PART : value; + } + + @Override + protected final double getComplexI(Object store, int index) { + double value = getDouble(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_IMAGINARY_PART : 0; + } + + @Override + protected final String getString(Object store, int index) { + double value = getDouble(store, index); + return na.check(value) ? RRuntime.STRING_NA : RContext.getRRuntimeASTAccess().encodeDouble(value); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getDouble(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setDouble(store, index, sourceAccess.getDouble(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setDouble(store, index, sourceAccess.getDouble(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + setDouble(store, index, RRuntime.DOUBLE_NA); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.check(getDouble(store, index)); + } + } + + public abstract static class SlowPathFromLogicalAccess extends SlowPathVectorAccess { + + @Override + public final RType getType() { + return RType.Logical; + } + + @Override + protected final int getInt(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RRuntime.INT_NA : RRuntime.logical2intNoCheck(value); + } + + @Override + protected final double getDouble(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RRuntime.DOUBLE_NA : RRuntime.logical2doubleNoCheck(value); + } + + @Override + protected final byte getRaw(Object store, int index) { + byte value = getLogical(store, index); + if (na.check(value)) { + warning(Message.OUT_OF_RANGE); + return 0; + } + return value; + } + + @Override + protected final RComplex getComplex(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RComplex.createNA() : RRuntime.logical2complexNoCheck(value); + } + + @Override + protected final double getComplexR(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_REAL_PART : value; + } + + @Override + protected final double getComplexI(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RRuntime.COMPLEX_NA_IMAGINARY_PART : 0; + } + + @Override + protected final String getString(Object store, int index) { + byte value = getLogical(store, index); + return na.check(value) ? RRuntime.STRING_NA : RRuntime.logicalToStringNoCheck(value); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getLogical(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setLogical(store, index, sourceAccess.getLogical(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setLogical(store, index, sourceAccess.getLogical(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + setLogical(store, index, RRuntime.LOGICAL_NA); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.check(getLogical(store, index)); + } + } + + public abstract static class SlowPathFromRawAccess extends SlowPathVectorAccess { + + @Override + public final RType getType() { + return RType.Raw; + } + + @Override + protected final int getInt(Object store, int index) { + return getRaw(store, index) & 0xff; + } + + @Override + protected final double getDouble(Object store, int index) { + return getRaw(store, index) & 0xff; + } + + @Override + protected final byte getLogical(Object store, int index) { + return getRaw(store, index) == 0 ? RRuntime.LOGICAL_FALSE : RRuntime.LOGICAL_TRUE; + } + + @Override + protected final RComplex getComplex(Object store, int index) { + return RComplex.valueOf(getRaw(store, index) & 0xff, 0); + } + + @Override + protected final double getComplexR(Object store, int index) { + return getRaw(store, index) & 0xff; + } + + @Override + protected final double getComplexI(Object store, int index) { + return 0; + } + + @Override + protected final String getString(Object store, int index) { + return RRuntime.rawToHexString(getRaw(store, index)); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getRaw(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setRaw(store, index, sourceAccess.getRaw(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setRaw(store, index, sourceAccess.getRaw(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + /* + * There is no raw NA, but places that write NA for other types usually write 0 for raw. + */ + setRaw(store, index, (byte) 0); + } + + @Override + protected boolean isNA(Object store, int index) { + return false; + } + } + + public abstract static class SlowPathFromComplexAccess extends SlowPathVectorAccess { + + @Override + public final RType getType() { + return RType.Complex; + } + + @Override + protected final int getInt(Object store, int index) { + double value = getComplexR(store, index); + if (Double.isNaN(value)) { + na.enable(true); + return RRuntime.INT_NA; + } + if (value > Integer.MAX_VALUE || value <= Integer.MIN_VALUE) { + na.enable(true); + warning(Message.NA_INTRODUCED_COERCION_INT); + return RRuntime.INT_NA; + } + if (getComplexI(store, index) != 0) { + warning(Message.IMAGINARY_PARTS_DISCARDED_IN_COERCION); + } + return (int) value; + } + + @Override + protected final double getDouble(Object store, int index) { + double value = getComplexR(store, index); + if (Double.isNaN(value)) { + na.enable(true); + return RRuntime.DOUBLE_NA; + } + if (getComplexI(store, index) != 0) { + warning(Message.IMAGINARY_PARTS_DISCARDED_IN_COERCION); + } + return value; + } + + @Override + protected final byte getRaw(Object store, int index) { + double value = getComplexR(store, index); + if (Double.isNaN(value) || value < 0 || value >= 256) { + warning(Message.OUT_OF_RANGE); + return 0; + } + if (getComplexI(store, index) != 0) { + warning(Message.IMAGINARY_PARTS_DISCARDED_IN_COERCION); + } + return (byte) value; + } + + @Override + protected final byte getLogical(Object store, int index) { + RComplex value = getComplex(store, index); + return na.check(value) ? RRuntime.LOGICAL_NA : RRuntime.complex2logicalNoCheck(value); + } + + @Override + protected final String getString(Object store, int index) { + RComplex value = getComplex(store, index); + return na.check(value) ? RRuntime.STRING_NA : RContext.getRRuntimeASTAccess().encodeComplex(value); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getComplex(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setComplex(store, index, sourceAccess.getComplexR(sourceIter), sourceAccess.getComplexI(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setComplex(store, index, sourceAccess.getComplexR(sourceIter, sourceIndex), sourceAccess.getComplexI(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + setComplex(store, index, RRuntime.COMPLEX_NA_REAL_PART, RRuntime.COMPLEX_NA_IMAGINARY_PART); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.check(getComplexR(store, index), getComplexI(store, index)); + } + } + + public abstract static class SlowPathFromStringAccess extends SlowPathVectorAccess { + + @Override + public final RType getType() { + return RType.Character; + } + + @Override + protected final int getInt(Object store, int index) { + return na.convertStringToInt(getString(store, index)); + } + + @Override + protected final double getDouble(Object store, int index) { + return na.convertStringToDouble(getString(store, index)); + } + + @Override + protected final byte getRaw(Object store, int index) { + int value = na.convertStringToInt(getString(store, index)); + return value >= 0 && value <= 255 ? (byte) value : 0; + } + + @Override + protected final byte getLogical(Object store, int index) { + return na.convertStringToLogical(getString(store, index)); + } + + @Override + protected final RComplex getComplex(Object store, int index) { + return na.convertStringToComplex(getString(store, index)); + } + + @Override + protected final double getComplexR(Object store, int index) { + return na.convertStringToComplex(getString(store, index)).getRealPart(); + } + + @Override + protected final double getComplexI(Object store, int index) { + return na.convertStringToComplex(getString(store, index)).getImaginaryPart(); + } + + @Override + protected final Object getListElement(Object store, int index) { + return getString(store, index); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setString(store, index, sourceAccess.getString(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setString(store, index, sourceAccess.getString(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + setString(store, index, RRuntime.STRING_NA); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.check(getString(store, index)); + } + } + + public abstract static class SlowPathFromListAccess extends SlowPathVectorAccess { + + @Override + protected final int getInt(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final double getDouble(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final byte getRaw(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final byte getLogical(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final RComplex getComplex(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final double getComplexR(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final double getComplexI(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final String getString(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setListElement(store, index, sourceAccess.getListElement(sourceIter)); + } + + @Override + protected final void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setListElement(store, index, sourceAccess.getListElement(sourceIter, sourceIndex)); + } + + @Override + protected void setNA(Object store, int index) { + /* + * There is no list NA, but places that write NA for other types usually write NULL for + * lists. + */ + setListElement(store, index, RNull.instance); + } + + @Override + protected boolean isNA(Object store, int index) { + return na.checkListElement(getListElement(store, index)); + } + } +} diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/VectorAccess.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/VectorAccess.java new file mode 100644 index 0000000000..27bba89617 --- /dev/null +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/data/nodes/VectorAccess.java @@ -0,0 +1,501 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package com.oracle.truffle.r.runtime.data.nodes; + +import com.oracle.truffle.api.CompilerAsserts; +import com.oracle.truffle.api.dsl.Cached; +import com.oracle.truffle.api.dsl.Specialization; +import com.oracle.truffle.api.nodes.Node; +import com.oracle.truffle.r.runtime.RInternalError; +import com.oracle.truffle.r.runtime.RType; +import com.oracle.truffle.r.runtime.data.RComplex; +import com.oracle.truffle.r.runtime.data.RComplexVector; +import com.oracle.truffle.r.runtime.data.RDataFactory; +import com.oracle.truffle.r.runtime.data.RDoubleVector; +import com.oracle.truffle.r.runtime.data.RExpression; +import com.oracle.truffle.r.runtime.data.RIntVector; +import com.oracle.truffle.r.runtime.data.RList; +import com.oracle.truffle.r.runtime.data.RLogicalVector; +import com.oracle.truffle.r.runtime.data.RRawVector; +import com.oracle.truffle.r.runtime.data.RStringVector; +import com.oracle.truffle.r.runtime.data.model.RAbstractContainer; +import com.oracle.truffle.r.runtime.nodes.RBaseNode; +import com.oracle.truffle.r.runtime.ops.na.NACheck; + +/** + * This class is the main access point for reading and writing vectors. Every implementation of + * {@link RAbstractContainer} needs to be able to supply an instance of {@link VectorAccess} via the + * {@link RAbstractContainer#access()} function. These instances can be asked, via the + * {@link #supports(Object)} function, whether they support a specific object.<br/> + * + * The usual interaction with vectors looks like this: + * <ul> + * <li>{@link Specialization} {@code foo} has a vector as a parameter, and adds a {@link Cached} + * parameter of type {@link VectorAccess} that is initialized by {@link RAbstractContainer#access()} + * .</li> + * <li>The specialization guards with {@code access.supports(vector)} so that it only handles cases + * that it actually supports.</li> + * <li>try-with-resources is used to open an access (either sequential or random) on the vector: + * <br/> + * {@code try (SequentialIterator iter = access.access(vector))...}</li> + * <li>Inside the try-with-resources, individual elements can be accessed using the + * {@link VectorAccess#getInt(SequentialIterator)}, etc. functions</li> + * <li>A fallback specialization, which {@link Specialization#replaces()} the original one and + * creates the {@link VectorAccess} via {@link RAbstractContainer#slowPathAccess()}, calls the first + * specialization with the slow path vector access.</li> + * </ul> + */ +public abstract class VectorAccess extends Node { + + public final NACheck na = NACheck.create(); + + protected final Class<? extends RAbstractContainer> clazz; + protected final boolean hasStore; + + public VectorAccess(Class<? extends RAbstractContainer> clazz, boolean hasStore) { + CompilerAsserts.neverPartOfCompilation(); + this.clazz = clazz; + this.hasStore = hasStore; + } + + public static final class SequentialIterator implements AutoCloseable { + + protected final Object store; // internal store, native mirror or vector + protected final int length; + protected int index; + + private SequentialIterator(Object store, int length) { + this.store = store; + this.length = length; + this.index = -1; + } + + @Override + public void close() { + // nothing to do + } + + public int getIndex() { + return index; + } + + Object getStore() { + return store; + } + + @Override + public String toString() { + return String.format("<iterator %d of %d, %s>", index, length, store == null ? "null" : store.getClass().getSimpleName()); + } + } + + public static final class RandomIterator implements AutoCloseable { + + protected final Object store; // internal store, native mirror or vector + protected final int length; + + private RandomIterator(Object store, int length) { + this.store = store; + this.length = length; + } + + @Override + public void close() { + // nothing to do + } + + Object getStore() { + return store; + } + + @Override + public String toString() { + return String.format("<random access %s>", store == null ? "null" : store.getClass().getSimpleName()); + } + } + + protected abstract int getInt(Object store, int index); + + protected abstract double getDouble(Object store, int index); + + protected abstract RComplex getComplex(Object store, int index); + + protected abstract double getComplexR(Object store, int index); + + protected abstract double getComplexI(Object store, int index); + + protected abstract byte getRaw(Object store, int index); + + protected abstract byte getLogical(Object store, int index); + + protected abstract String getString(Object store, int index); + + protected abstract Object getListElement(Object store, int index); + + @SuppressWarnings("unused") + protected void setInt(Object store, int index, int value) { + throw RInternalError.shouldNotReachHere(); + } + + @SuppressWarnings("unused") + protected void setDouble(Object store, int index, double value) { + throw RInternalError.shouldNotReachHere(); + } + + @SuppressWarnings("unused") + protected void setComplex(Object store, int index, double real, double imaginary) { + throw RInternalError.shouldNotReachHere(); + } + + @SuppressWarnings("unused") + protected void setRaw(Object store, int index, byte value) { + throw RInternalError.shouldNotReachHere(); + } + + @SuppressWarnings("unused") + protected void setLogical(Object store, int index, byte value) { + throw RInternalError.shouldNotReachHere(); + } + + @SuppressWarnings("unused") + protected void setString(Object store, int index, String value) { + throw RInternalError.shouldNotReachHere(); + } + + @SuppressWarnings("unused") + protected void setListElement(Object store, int index, Object value) { + throw RInternalError.shouldNotReachHere(); + } + + @SuppressWarnings("unused") + protected void setFromSameType(Object store, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + throw RInternalError.shouldNotReachHere(); + } + + @SuppressWarnings("unused") + protected void setFromSameType(Object store, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + throw RInternalError.shouldNotReachHere(); + } + + @SuppressWarnings("unused") + protected void setNA(Object store, int index) { + throw RInternalError.shouldNotReachHere(); + } + + protected abstract boolean isNA(Object store, int index); + + public final RAbstractContainer cast(Object value) { + return clazz.cast(value); + } + + public final boolean supports(Object value) { + assert clazz != RAbstractContainer.class : "cannot call 'supports' on slow path vector access"; + return value.getClass() == clazz && (cast(value).getInternalStore() != null) == hasStore; + } + + protected abstract Object getStore(RAbstractContainer vector); + + protected int getLength(RAbstractContainer vector) { + return vector.getLength(); + } + + /** + * Creates a new iterator that will point to before the beginning of the vector, so that + * {@link #next(SequentialIterator)} will move it to the first element. + */ + public final SequentialIterator access(RAbstractContainer vector) { + RAbstractContainer castVector = cast(vector); + int length = getLength(castVector); + RBaseNode.reportWork(this, length); + na.enable(castVector); + return new SequentialIterator(getStore(castVector), length); + } + + @SuppressWarnings("static-method") + public final boolean next(SequentialIterator iter) { + return ++iter.index < iter.length; + } + + @SuppressWarnings("static-method") + public final void nextWithWrap(SequentialIterator iter) { + assert iter.length > 0; + if (++iter.index >= iter.length) { + iter.index = 0; + } + } + + @SuppressWarnings("static-method") + public final int getLength(SequentialIterator iter) { + return iter.length; + } + + /** + * Resets the iterator to point to before the first element, calling + * {@link #next(SequentialIterator)} will move it to the first element. + */ + @SuppressWarnings("static-method") + public final void reset(SequentialIterator iter) { + iter.index = -1; + } + + public abstract RType getType(); + + public final int getInt(SequentialIterator iter) { + return getInt(iter.store, iter.index); + } + + public final double getDouble(SequentialIterator iter) { + return getDouble(iter.store, iter.index); + } + + public final RComplex getComplex(SequentialIterator iter) { + return getComplex(iter.store, iter.index); + } + + public final double getComplexR(SequentialIterator iter) { + return getComplexR(iter.store, iter.index); + } + + public final double getComplexI(SequentialIterator iter) { + return getComplexI(iter.store, iter.index); + } + + public final byte getRaw(SequentialIterator iter) { + return getRaw(iter.store, iter.index); + } + + public final byte getLogical(SequentialIterator iter) { + return getLogical(iter.store, iter.index); + } + + public final String getString(SequentialIterator iter) { + return getString(iter.store, iter.index); + } + + public final Object getListElement(SequentialIterator iter) { + return getListElement(iter.store, iter.index); + } + + public final void setInt(SequentialIterator iter, int value) { + setInt(iter.store, iter.index, value); + } + + public final void setDouble(SequentialIterator iter, double value) { + setDouble(iter.store, iter.index, value); + } + + public final void setComplex(SequentialIterator iter, double real, double imaginary) { + setComplex(iter.store, iter.index, real, imaginary); + } + + public final void setRaw(SequentialIterator iter, byte value) { + setRaw(iter.store, iter.index, value); + } + + public final void setLogical(SequentialIterator iter, byte value) { + setLogical(iter.store, iter.index, value); + } + + public final void setString(SequentialIterator iter, String value) { + setString(iter.store, iter.index, value); + } + + public final void setListElement(SequentialIterator iter, Object value) { + setListElement(iter.store, iter.index, value); + } + + public final void setFromSameType(SequentialIterator iter, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setFromSameType(iter.store, iter.index, sourceAccess, sourceIter); + } + + public final void setFromSameType(SequentialIterator iter, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setFromSameType(iter.store, iter.index, sourceAccess, sourceIter, sourceIndex); + } + + public final void setNA(SequentialIterator iter) { + setNA(iter.store, iter.index); + } + + public final boolean isNA(SequentialIterator iter) { + return isNA(iter.store, iter.index); + } + + /** + * Creates a new random access on the given vector. + */ + public final RandomIterator randomAccess(RAbstractContainer vector) { + RAbstractContainer castVector = cast(vector); + int length = getLength(castVector); + RBaseNode.reportWork(this, length); + na.enable(castVector); + return new RandomIterator(getStore(castVector), length); + } + + @SuppressWarnings("static-method") + public final int getLength(RandomIterator iter) { + return iter.length; + } + + public final int getInt(RandomIterator iter, int index) { + return getInt(iter.store, index); + } + + public final double getDouble(RandomIterator iter, int index) { + return getDouble(iter.store, index); + } + + public final RComplex getComplex(RandomIterator iter, int index) { + return getComplex(iter.store, index); + } + + public final double getComplexR(RandomIterator iter, int index) { + return getComplexR(iter.store, index); + } + + public final double getComplexI(RandomIterator iter, int index) { + return getComplexI(iter.store, index); + } + + public final byte getRaw(RandomIterator iter, int index) { + return getRaw(iter.store, index); + } + + public final byte getLogical(RandomIterator iter, int index) { + return getLogical(iter.store, index); + } + + public final String getString(RandomIterator iter, int index) { + return getString(iter.store, index); + } + + public final Object getListElement(RandomIterator iter, int index) { + return getListElement(iter.store, index); + } + + public final void setInt(RandomIterator iter, int index, int value) { + setInt(iter.store, index, value); + } + + public final void setDouble(RandomIterator iter, int index, double value) { + setDouble(iter.store, index, value); + } + + public final void setComplex(RandomIterator iter, int index, double real, double imaginary) { + setComplex(iter.store, index, real, imaginary); + } + + public final void setRaw(RandomIterator iter, int index, byte value) { + setRaw(iter.store, index, value); + } + + public final void setLogical(RandomIterator iter, int index, byte value) { + setLogical(iter.store, index, value); + } + + public final void setString(RandomIterator iter, int index, String value) { + setString(iter.store, index, value); + } + + public final void setListElement(RandomIterator iter, int index, Object value) { + setListElement(iter.store, index, value); + } + + public final void setFromSameType(RandomIterator iter, int index, VectorAccess sourceAccess, SequentialIterator sourceIter) { + setFromSameType(iter.store, index, sourceAccess, sourceIter); + } + + public final void setFromSameType(RandomIterator iter, int index, VectorAccess sourceAccess, RandomIterator sourceIter, int sourceIndex) { + setFromSameType(iter.store, index, sourceAccess, sourceIter, sourceIndex); + } + + public final void setNA(RandomIterator iter, int index) { + setNA(iter.store, index); + } + + public final boolean isNA(RandomIterator iter, int index) { + return isNA(iter.store, index); + } + + private static final RStringVector TEMPLATE_CHARACTER = RDataFactory.getPermanent().createStringVector(4); + private static final RComplexVector TEMPLATE_COMPLEX = RDataFactory.getPermanent().createComplexVector(4); + private static final RDoubleVector TEMPLATE_DOUBLE = RDataFactory.getPermanent().createDoubleVector(4); + private static final RIntVector TEMPLATE_INTEGER = RDataFactory.getPermanent().createIntVector(4); + private static final RList TEMPLATE_LIST = RDataFactory.getPermanent().createList(4); + private static final RExpression TEMPLATE_EXPRESSION = RDataFactory.createExpression(4); + private static final RLogicalVector TEMPLATE_LOGICAL = RDataFactory.getPermanent().createLogicalVector(4); + private static final RRawVector TEMPLATE_RAW = RDataFactory.getPermanent().createRawVector(4); + + public static VectorAccess createNew(RType type) { + switch (type) { + case Character: + return TEMPLATE_CHARACTER.access(); + case Complex: + return TEMPLATE_COMPLEX.access(); + case Double: + return TEMPLATE_DOUBLE.access(); + case Integer: + return TEMPLATE_INTEGER.access(); + case List: + return TEMPLATE_LIST.access(); + case Expression: + return TEMPLATE_EXPRESSION.access(); + case Logical: + return TEMPLATE_LOGICAL.access(); + case Raw: + return TEMPLATE_RAW.access(); + case RInteropChar: + case RInteropFloat: + case RInteropLong: + case RInteropShort: + default: + throw RInternalError.shouldNotReachHere(); + } + } + + public static VectorAccess createSlowPathNew(RType type) { + switch (type) { + case Character: + return TEMPLATE_CHARACTER.slowPathAccess(); + case Complex: + return TEMPLATE_COMPLEX.slowPathAccess(); + case Double: + return TEMPLATE_DOUBLE.slowPathAccess(); + case Integer: + return TEMPLATE_INTEGER.slowPathAccess(); + case List: + return TEMPLATE_LIST.slowPathAccess(); + case Expression: + return TEMPLATE_EXPRESSION.slowPathAccess(); + case Logical: + return TEMPLATE_LOGICAL.slowPathAccess(); + case Raw: + return TEMPLATE_RAW.slowPathAccess(); + case RInteropChar: + case RInteropFloat: + case RInteropLong: + case RInteropShort: + default: + throw RInternalError.shouldNotReachHere(); + } + } +} diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/interop/TruffleObjectConverter.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/interop/TruffleObjectConverter.java index 2bb74514a0..96de1c9491 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/interop/TruffleObjectConverter.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/interop/TruffleObjectConverter.java @@ -49,6 +49,7 @@ import com.oracle.truffle.r.runtime.data.model.RAbstractListVector; import com.oracle.truffle.r.runtime.data.model.RAbstractLogicalVector; import com.oracle.truffle.r.runtime.data.model.RAbstractStringVector; import com.oracle.truffle.r.runtime.data.model.RAbstractVector; +import com.oracle.truffle.r.runtime.data.nodes.VectorAccess; public final class TruffleObjectConverter { @@ -288,6 +289,11 @@ public final class TruffleObjectConverter { this.length = length; } + @Override + public Object getInternalStore() { + return this; + } + @Override public RAbstractVector copy() { throw RInternalError.shouldNotReachHere(); @@ -407,6 +413,15 @@ public final class TruffleObjectConverter { public ForeignAccess getForeignAccess() { throw RInternalError.shouldNotReachHere(); } - } + @Override + public VectorAccess access() { + throw RInternalError.unimplemented("access() for " + getClass().getSimpleName()); + } + + @Override + public VectorAccess slowPathAccess() { + throw RInternalError.unimplemented("slowPathAccess() for " + getClass().getSimpleName()); + } + } } diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/ops/na/NACheck.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/ops/na/NACheck.java index cbf0f6f606..cdb561977c 100644 --- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/ops/na/NACheck.java +++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/ops/na/NACheck.java @@ -120,6 +120,17 @@ public final class NACheck { return false; } + public boolean check(double real, double imag) { + if (state != NO_CHECK && RRuntime.isNA(real, imag)) { + if (state == CHECK_DEOPT) { + CompilerDirectives.transferToInterpreterAndInvalidate(); + state = CHECK; + } + return true; + } + return false; + } + public void seenNA() { if (state != CHECK) { CompilerDirectives.transferToInterpreterAndInvalidate(); -- GitLab