Skip to content
Snippets Groups Projects
Commit b5579c6b authored by Stepan Sindelar's avatar Stepan Sindelar
Browse files

[GR-3021] Stricter argument validation in some stats externals.

PullRequest: fastr/1281
parents 5d4eb526 6df599b5
No related branches found
No related tags found
No related merge requests found
...@@ -25,29 +25,29 @@ ...@@ -25,29 +25,29 @@
# prints a warning message instructing the user to use grid/lattice/ggplot2 instead # prints a warning message instructing the user to use grid/lattice/ggplot2 instead
eval(expression({ eval(expression({
if (.fastr.option('IgnoreGraphicsCalls')) { graphicsWarning <- function(name) {
# we force the arguments to be evaluated, but otherwise do nothing # lookup original function and fetch signature
graphicsWarning <- function(name) function(...) { list(...); invisible(NULL); } fun <- tryCatch(get(name, environment()), error=function(x) NULL)
} else { if(!is.null(fun)) {
graphicsWarning <- function(name) { sig <- formals(fun)
# lookup original function and fetch signature } else {
fun <- tryCatch(get(name, environment()), error=function(x) NULL) sig <- NULL
if(!is.null(fun)) { }
sig <- formals(fun)
} else {
sig <- NULL
}
if (.fastr.option('IgnoreGraphicsCalls')) {
# we evaluate the arguments to simulate the function effects
replacementFun <- function(...) { if (is.null(sig)) list(...) else get(names(sig)); invisible(NULL); }
} else {
replacementFun <- function(...) { replacementFun <- function(...) {
warning(paste0(name, " not supported.", " Note: FastR does not support graphics package and most of its functions. Please use grid package or grid based packages like lattice instead.")) warning(paste0(name, " not supported.", " Note: FastR does not support graphics package and most of its functions. Please use grid package or grid based packages like lattice instead."))
NULL NULL
} }
}
if(!is.null(sig)) { if(!is.null(sig)) {
formals(replacementFun) <- sig formals(replacementFun) <- sig
}
return(replacementFun)
} }
return(replacementFun)
} }
plot.default <- function (x, y = NULL, type = "p", xlim = NULL, ylim = NULL, plot.default <- function (x, y = NULL, type = "p", xlim = NULL, ylim = NULL,
......
...@@ -26,6 +26,7 @@ import com.oracle.truffle.r.nodes.attributes.SpecialAttributesFunctions.GetDimAt ...@@ -26,6 +26,7 @@ import com.oracle.truffle.r.nodes.attributes.SpecialAttributesFunctions.GetDimAt
import com.oracle.truffle.r.nodes.attributes.SpecialAttributesFunctions.SetClassAttributeNode; import com.oracle.truffle.r.nodes.attributes.SpecialAttributesFunctions.SetClassAttributeNode;
import com.oracle.truffle.r.nodes.builtin.RExternalBuiltinNode; import com.oracle.truffle.r.nodes.builtin.RExternalBuiltinNode;
import com.oracle.truffle.r.runtime.RError; import com.oracle.truffle.r.runtime.RError;
import com.oracle.truffle.r.runtime.RError.Message;
import com.oracle.truffle.r.runtime.RRuntime; import com.oracle.truffle.r.runtime.RRuntime;
import com.oracle.truffle.r.runtime.data.RDataFactory; import com.oracle.truffle.r.runtime.data.RDataFactory;
import com.oracle.truffle.r.runtime.data.RDoubleVector; import com.oracle.truffle.r.runtime.data.RDoubleVector;
...@@ -58,6 +59,10 @@ public abstract class Cdist extends RExternalBuiltinNode.Arg4 { ...@@ -58,6 +59,10 @@ public abstract class Cdist extends RExternalBuiltinNode.Arg4 {
@Cached("create()") SetAttributeNode setAttrNode, @Cached("create()") SetAttributeNode setAttrNode,
@Cached("create()") SetClassAttributeNode setClassAttrNode, @Cached("create()") SetClassAttributeNode setClassAttrNode,
@Cached("create()") GetDimAttributeNode getDimNode) { @Cached("create()") GetDimAttributeNode getDimNode) {
if (!getDimNode.isMatrix(x)) {
// Note: otherwise array index out of bounds
throw error(Message.MUST_BE_SQUARE_MATRIX, "x");
}
int nr = getDimNode.nrows(x); int nr = getDimNode.nrows(x);
int nc = getDimNode.ncols(x); int nc = getDimNode.ncols(x);
int n = nr * (nr - 1) / 2; /* avoid int overflow for N ~ 50,000 */ int n = nr * (nr - 1) / 2; /* avoid int overflow for N ~ 50,000 */
......
...@@ -17,6 +17,7 @@ import com.oracle.truffle.api.dsl.Cached; ...@@ -17,6 +17,7 @@ import com.oracle.truffle.api.dsl.Cached;
import com.oracle.truffle.api.dsl.Specialization; import com.oracle.truffle.api.dsl.Specialization;
import com.oracle.truffle.r.nodes.attributes.SpecialAttributesFunctions.GetDimAttributeNode; import com.oracle.truffle.r.nodes.attributes.SpecialAttributesFunctions.GetDimAttributeNode;
import com.oracle.truffle.r.nodes.builtin.RExternalBuiltinNode; import com.oracle.truffle.r.nodes.builtin.RExternalBuiltinNode;
import com.oracle.truffle.r.runtime.RError.Message;
import com.oracle.truffle.r.runtime.data.RDataFactory; import com.oracle.truffle.r.runtime.data.RDataFactory;
import com.oracle.truffle.r.runtime.data.RIntVector; import com.oracle.truffle.r.runtime.data.RIntVector;
import com.oracle.truffle.r.runtime.data.model.RAbstractIntVector; import com.oracle.truffle.r.runtime.data.model.RAbstractIntVector;
...@@ -50,6 +51,11 @@ public abstract class Cutree extends RExternalBuiltinNode.Arg2 { ...@@ -50,6 +51,11 @@ public abstract class Cutree extends RExternalBuiltinNode.Arg2 {
boolean foundJ; boolean foundJ;
int n = getDimNode.nrows(merge) + 1; int n = getDimNode.nrows(merge) + 1;
if (!getDimNode.isSquareMatrix(merge)) {
// Note: otherwise array index out of bounds
throw error(Message.MUST_BE_SQUARE_MATRIX, "x");
}
/* /*
* The C code uses 1-based indices for the next three arrays and so set the int * value * The C code uses 1-based indices for the next three arrays and so set the int * value
* behind the actual start of the array. To keep the logic equivalent, we call adj(k) on the * behind the actual start of the array. To keep the logic equivalent, we call adj(k) on the
......
...@@ -34,7 +34,7 @@ public abstract class DoubleCentre extends RExternalBuiltinNode.Arg1 { ...@@ -34,7 +34,7 @@ public abstract class DoubleCentre extends RExternalBuiltinNode.Arg1 {
@Cached("createNonShared(a)") VectorReuse reuse, @Cached("createNonShared(a)") VectorReuse reuse,
@Cached("create()") GetDimAttributeNode getDimNode) { @Cached("create()") GetDimAttributeNode getDimNode) {
int n = getDimNode.nrows(a); int n = getDimNode.nrows(a);
if (!getDimNode.isMatrix(a) || n != a.getLength() / n) { if (!getDimNode.isSquareMatrix(a)) {
// Note: otherwise array index out of bounds // Note: otherwise array index out of bounds
throw error(Message.MUST_BE_SQUARE_MATRIX, "x"); throw error(Message.MUST_BE_SQUARE_MATRIX, "x");
} }
......
...@@ -662,6 +662,14 @@ public final class SpecialAttributesFunctions { ...@@ -662,6 +662,14 @@ public final class SpecialAttributesFunctions {
return nullDimsProfile.profile(dims == null) ? false : dims.getLength() == 2; return nullDimsProfile.profile(dims == null) ? false : dims.getLength() == 2;
} }
public final boolean isSquareMatrix(RAbstractVector vector) {
RIntVector dims = (RIntVector) execute(vector);
if (nullDimsProfile.profile(dims == null) || dims.getLength() < 2) {
return false;
}
return dims.getDataAt(0) == dims.getDataAt(1);
}
@Specialization(insertBefore = "getAttrFromAttributable") @Specialization(insertBefore = "getAttrFromAttributable")
protected Object getScalarVectorDims(@SuppressWarnings("unused") RScalarVector x) { protected Object getScalarVectorDims(@SuppressWarnings("unused") RScalarVector x) {
return null; return null;
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment