From 7eb2ae08d1829798578c6aeb744e61e0b8cc2124 Mon Sep 17 00:00:00 2001 From: Miloslav Metelka <miloslav.metelka@oracle.com> Date: Mon, 7 Aug 2017 16:57:13 +0200 Subject: [PATCH] Fixed max() on empty int vector. --- .../truffle/r/nodes/builtin/base/Max.java | 2 +- .../truffle/r/nodes/builtin/base/Min.java | 2 +- .../truffle/r/nodes/builtin/base/PMinMax.java | 4 +- .../truffle/r/nodes/builtin/base/Range.java | 4 +- .../truffle/r/nodes/builtin/base/Sum.java | 2 +- .../unary/UnaryArithmeticReduceNode.java | 38 ++++++++++++++++--- .../truffle/r/test/ExpectedTestOutput.test | 16 +++++++- .../r/test/builtins/TestBuiltin_max.java | 8 ++-- 8 files changed, 58 insertions(+), 18 deletions(-) diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Max.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Max.java index 34cc91da91..ce41444d9e 100644 --- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Max.java +++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Max.java @@ -43,7 +43,7 @@ import com.oracle.truffle.r.runtime.ops.BinaryArithmetic; public abstract class Max extends RBuiltinNode.Arg2 { private static final ReduceSemantics semantics = new ReduceSemantics(RRuntime.INT_MIN_VALUE, Double.NEGATIVE_INFINITY, false, RError.Message.NO_NONMISSING_MAX, - RError.Message.NO_NONMISSING_MAX_NA, false, true); + RError.Message.NO_NONMISSING_MAX_NA, null, false, true, true); @Child private UnaryArithmeticReduceNode reduce = UnaryArithmeticReduceNodeGen.create(semantics, BinaryArithmetic.MAX); diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Min.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Min.java index ca072450b4..7be389e2d1 100644 --- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Min.java +++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Min.java @@ -43,7 +43,7 @@ import com.oracle.truffle.r.runtime.ops.BinaryArithmetic; public abstract class Min extends RBuiltinNode.Arg2 { private static final ReduceSemantics semantics = new ReduceSemantics(RRuntime.INT_MAX_VALUE, Double.POSITIVE_INFINITY, false, RError.Message.NO_NONMISSING_MIN, - RError.Message.NO_NONMISSING_MIN_NA, false, true); + RError.Message.NO_NONMISSING_MIN_NA, null, false, true, true); @Child private UnaryArithmeticReduceNode reduce = UnaryArithmeticReduceNodeGen.create(semantics, BinaryArithmetic.MIN); diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/PMinMax.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/PMinMax.java index 6350ec5ad7..4dd253311b 100644 --- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/PMinMax.java +++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/PMinMax.java @@ -341,7 +341,7 @@ public abstract class PMinMax extends RBuiltinNode.Arg2 { public abstract static class PMax extends PMinMax { public PMax() { - super(new ReduceSemantics(RRuntime.INT_MIN_VALUE, Double.NEGATIVE_INFINITY, false, RError.Message.NO_NONMISSING_MAX, RError.Message.NO_NONMISSING_MAX_NA, false, true), + super(new ReduceSemantics(RRuntime.INT_MIN_VALUE, Double.NEGATIVE_INFINITY, false, RError.Message.NO_NONMISSING_MAX, RError.Message.NO_NONMISSING_MAX_NA, null, false, true, true), BinaryArithmetic.MAX); } @@ -354,7 +354,7 @@ public abstract class PMinMax extends RBuiltinNode.Arg2 { public abstract static class PMin extends PMinMax { public PMin() { - super(new ReduceSemantics(RRuntime.INT_MAX_VALUE, Double.POSITIVE_INFINITY, false, RError.Message.NO_NONMISSING_MIN, RError.Message.NO_NONMISSING_MIN_NA, false, true), + super(new ReduceSemantics(RRuntime.INT_MAX_VALUE, Double.POSITIVE_INFINITY, false, RError.Message.NO_NONMISSING_MIN, RError.Message.NO_NONMISSING_MIN_NA, null, false, true, true), BinaryArithmetic.MIN); } diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Range.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Range.java index 828b494e5f..9395c6682c 100644 --- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Range.java +++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Range.java @@ -45,9 +45,9 @@ import com.oracle.truffle.r.runtime.ops.BinaryArithmetic; public abstract class Range extends RBuiltinNode.Arg3 { private static final ReduceSemantics minSemantics = new ReduceSemantics(RRuntime.INT_MAX_VALUE, Double.POSITIVE_INFINITY, false, RError.Message.NO_NONMISSING_MIN, - RError.Message.NO_NONMISSING_MIN_NA, false, true); + RError.Message.NO_NONMISSING_MIN_NA, null, false, true, true); private static final ReduceSemantics maxSemantics = new ReduceSemantics(RRuntime.INT_MIN_VALUE, Double.NEGATIVE_INFINITY, false, RError.Message.NO_NONMISSING_MAX, - RError.Message.NO_NONMISSING_MAX_NA, false, true); + RError.Message.NO_NONMISSING_MAX_NA, null, false, true, true); @Child private UnaryArithmeticReduceNode minReduce = UnaryArithmeticReduceNodeGen.create(minSemantics, BinaryArithmetic.MIN); @Child private UnaryArithmeticReduceNode maxReduce = UnaryArithmeticReduceNodeGen.create(maxSemantics, BinaryArithmetic.MAX); diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Sum.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Sum.java index a370cef06f..00fbadeb93 100644 --- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Sum.java +++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Sum.java @@ -54,7 +54,7 @@ public abstract class Sum extends RBuiltinNode.Arg2 { protected static final boolean FULL_PRECISION = FastROptions.FullPrecisionSum.getBooleanValue(); - private static final ReduceSemantics semantics = new ReduceSemantics(0, 0.0, true, null, null, Message.INTEGER_OVERFLOW_USE_SUM_NUMERIC, true, false); + private static final ReduceSemantics semantics = new ReduceSemantics(0, 0.0, true, null, null, Message.INTEGER_OVERFLOW_USE_SUM_NUMERIC, true, false, false); @Child private UnaryArithmeticReduceNode reduce = UnaryArithmeticReduceNodeGen.create(semantics, BinaryArithmetic.ADD); diff --git a/com.oracle.truffle.r.nodes/src/com/oracle/truffle/r/nodes/unary/UnaryArithmeticReduceNode.java b/com.oracle.truffle.r.nodes/src/com/oracle/truffle/r/nodes/unary/UnaryArithmeticReduceNode.java index f54f392a94..0a6de6adf1 100644 --- a/com.oracle.truffle.r.nodes/src/com/oracle/truffle/r/nodes/unary/UnaryArithmeticReduceNode.java +++ b/com.oracle.truffle.r.nodes/src/com/oracle/truffle/r/nodes/unary/UnaryArithmeticReduceNode.java @@ -72,6 +72,7 @@ public abstract class UnaryArithmeticReduceNode extends RBaseNode { protected final ReduceSemantics semantics; protected final boolean supportString; protected final boolean supportComplex; + protected final boolean useDoubleStartForEmptyVector; private final NACheck na = NACheck.create(); private final ConditionProfile naRmProfile = ConditionProfile.createBinaryProfile(); @@ -82,6 +83,7 @@ public abstract class UnaryArithmeticReduceNode extends RBaseNode { this.arithmetic = factory.createOperation(); this.supportString = semantics.supportString; this.supportComplex = semantics.supportComplex; + this.useDoubleStartForEmptyVector = semantics.useDoubleStartForEmptyVector; } private String handleString(RStringVector operand, boolean naRm, boolean finite, int offset) { @@ -200,6 +202,14 @@ public abstract class UnaryArithmeticReduceNode extends RBaseNode { } } + @Specialization(guards = {"useDoubleStartForEmptyVector", "operand.getLength() == 0"}) + protected double doIntVectorEmpty(@SuppressWarnings("unused") RIntVector operand, + @SuppressWarnings("unused") boolean naRm, + @SuppressWarnings("unused") boolean finite) { + emptyWarning(); + return semantics.getDoubleStart(); + } + @Specialization protected int doIntVector(RIntVector operand, boolean naRm, boolean finite) { RBaseNode.reportWork(this, operand.getLength()); @@ -267,6 +277,14 @@ public abstract class UnaryArithmeticReduceNode extends RBaseNode { return result; } + @Specialization(guards = {"useDoubleStartForEmptyVector", "operand.getLength() == 0"}) + protected double doLogicalVectorEmpty(@SuppressWarnings("unused") RLogicalVector operand, + @SuppressWarnings("unused") boolean naRm, + @SuppressWarnings("unused") boolean finite) { + emptyWarning(); + return semantics.getDoubleStart(); + } + @Specialization protected int doLogicalVector(RLogicalVector operand, boolean naRm, @SuppressWarnings("unused") boolean finite) { RBaseNode.reportWork(this, operand.getLength()); @@ -298,6 +316,14 @@ public abstract class UnaryArithmeticReduceNode extends RBaseNode { return result; } + @Specialization(guards = {"useDoubleStartForEmptyVector", "operand.getLength() == 0"}) + protected double doIntSequenceEmpty(@SuppressWarnings("unused") RIntSequence operand, + @SuppressWarnings("unused") boolean naRm, + @SuppressWarnings("unused") boolean finite) { + emptyWarning(); + return semantics.getDoubleStart(); + } + @Specialization protected int doIntSequence(RIntSequence operand, @SuppressWarnings("unused") boolean naRm, @SuppressWarnings("unused") boolean finite) { RBaseNode.reportWork(this, operand.getLength()); @@ -421,9 +447,10 @@ public abstract class UnaryArithmeticReduceNode extends RBaseNode { private final RError.Message naResultWarning; private final boolean supportComplex; private final boolean supportString; + private final boolean useDoubleStartForEmptyVector; public ReduceSemantics(int intStart, double doubleStart, boolean nullInt, RError.Message emptyWarning, RError.Message emptyWarningCharacter, RError.Message naResultWarning, - boolean supportComplex, boolean supportString) { + boolean supportComplex, boolean supportString, boolean useDoubleStartForEmptyVector) { this.intStart = intStart; this.doubleStart = doubleStart; this.nullInt = nullInt; @@ -432,10 +459,7 @@ public abstract class UnaryArithmeticReduceNode extends RBaseNode { this.naResultWarning = naResultWarning; this.supportComplex = supportComplex; this.supportString = supportString; - } - - public ReduceSemantics(int intStart, double doubleStart, boolean nullInt, RError.Message emptyWarning, RError.Message emptyWarningCharacter, boolean supportComplex, boolean supportString) { - this(intStart, doubleStart, nullInt, emptyWarning, emptyWarningCharacter, null, supportComplex, supportString); + this.useDoubleStartForEmptyVector = useDoubleStartForEmptyVector; } public int getIntStart() { @@ -465,6 +489,10 @@ public abstract class UnaryArithmeticReduceNode extends RBaseNode { public RError.Message getNAResultWarning() { return naResultWarning; } + + public boolean isUseDoubleStartForEmptyVector() { + return useDoubleStartForEmptyVector; + } } private static final class MultiElemStringHandlerNode extends RBaseNode { diff --git a/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/ExpectedTestOutput.test b/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/ExpectedTestOutput.test index 2ef82b8378..d3334b70c5 100644 --- a/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/ExpectedTestOutput.test +++ b/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/ExpectedTestOutput.test @@ -36185,18 +36185,30 @@ In max(double()) : no non-missing arguments to max; returning -Inf Warning message: In max(double(0)) : no non-missing arguments to max; returning -Inf -##com.oracle.truffle.r.test.builtins.TestBuiltin_max.testMaximum#Ignored.ImplementationError# +##com.oracle.truffle.r.test.builtins.TestBuiltin_max.testMaximum# #{ max(integer()) } [1] -Inf Warning message: In max(integer()) : no non-missing arguments to max; returning -Inf -##com.oracle.truffle.r.test.builtins.TestBuiltin_max.testMaximum#Ignored.ImplementationError# +##com.oracle.truffle.r.test.builtins.TestBuiltin_max.testMaximum# #{ max(integer(0)) } [1] -Inf Warning message: In max(integer(0)) : no non-missing arguments to max; returning -Inf +##com.oracle.truffle.r.test.builtins.TestBuiltin_max.testMaximum# +#{ max(logical(0)) } +[1] -Inf +Warning message: +In max(logical(0)) : no non-missing arguments to max; returning -Inf + +##com.oracle.truffle.r.test.builtins.TestBuiltin_max.testMaximum# +#{ max(seq_len(0)) } +[1] -Inf +Warning message: +In max(seq_len(0)) : no non-missing arguments to max; returning -Inf + ##com.oracle.truffle.r.test.builtins.TestBuiltin_max.testmax1# #argv <- list(10L, 1L);max(argv[[1]],argv[[2]]); [1] 10 diff --git a/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/builtins/TestBuiltin_max.java b/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/builtins/TestBuiltin_max.java index f8cf0fa929..a06988258f 100644 --- a/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/builtins/TestBuiltin_max.java +++ b/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/builtins/TestBuiltin_max.java @@ -214,15 +214,15 @@ public class TestBuiltin_max extends TestBase { assertEval("max(v<-42)"); - // FIXME -2147483647 returned instead of -Inf - assertEval(Ignored.ImplementationError, "{ max(integer(0)) }"); - // FIXME -2147483647 returned instead of -Inf - assertEval(Ignored.ImplementationError, "{ max(integer()) }"); + assertEval("{ max(integer(0)) }"); + assertEval("{ max(integer()) }"); assertEval("{ max(as.double(NA), na.rm=TRUE) }"); // FIXME -2147483647 returned instead of -Inf assertEval(Ignored.ImplementationError, "{ max(as.integer(NA), na.rm=TRUE) }"); // FIXME -2147483647 returned instead of -Inf assertEval(Ignored.ImplementationError, "{ max(as.integer(NA), as.integer(NA), na.rm=TRUE) }"); + assertEval("{ max(logical(0)) }"); + assertEval("{ max(seq_len(0)) }"); assertEval("max(c(1,NA,2), na.rm=NA)"); } -- GitLab