/*
* JAME 6.2.1
* http://jame.sourceforge.net
*
* Copyright 2001, 2016 Andrea Medeghini
*
* This file is part of JAME.
*
* JAME is an application for creating fractals and other graphics artifacts.
*
* JAME is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* JAME 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 for more details.
*
* You should have received a copy of the GNU General Public License
* along with JAME. If not, see <http://www.gnu.org/licenses/>.
*
*/
package net.sf.jame.contextfree.parser;
import java.util.List;
import org.antlr.v4.runtime.Token;
class ASTArray extends ASTExpression {
private int nameIndex;
private double[] data;
private ASTExpression args;
private int length;
private int stride;
private int stackIndex;
private int count;
private boolean isParameter;
private String entropy;
public ASTArray(int nameIndex, ASTExpression args, String entropy, Token location) {
super(false, false, EExpType.NumericType, location);
this.nameIndex = nameIndex;
this.data = null;
this.args = args;
this.length = 1;
this.stride = 1;
this.stackIndex = -1;
this.count = 0;
this.isParameter = false;
this.entropy = entropy;
}
public boolean isParameter() {
return isParameter;
}
@Override
public void entropy(StringBuilder e) {
e.append(entropy);
}
@Override
public int evaluate(double[] result, int length, RTI rti) {
if (type != EExpType.NumericType) {
error("Non-numeric/flag expression in a numeric/flag context");
return -1;
}
if (result != null && length < this.length) {
return -1;
}
if (result != null) {
if (rti == null && (data == null || !args.isConstant())) throw new DeferUntilRuntimeException();
double[] i = new double[1];
if (args.evaluate(i, 1, rti) != 1) {
error("Cannot evaluate array index");
return -1;
}
int index = (int)i[0];
if (this.length - 1 * this.stride + index > this.count || index < 0) {
error("Array index exceeds bounds");
return -1;
}
double[] source = data;
if (source == null) {
source = rti.stackItem(stackIndex).getArray();
}
for (int j = 0; j < this.length; j++) {
result[j] = source[j * this.stride + index];
}
}
return this.length;
}
@Override
public ASTExpression simplify() {
if (data == null || !isConstant || length > 1) {
if (args != null) {
args.simplify();
return this;
}
}
double[] i = new double[1];
if (args.evaluate(i, 1) != 1) {
error("Cannot evaluate array index");
return this;
}
int index = (int)i[0];
if (index > count || index < 0) {
error("Array index exceeds bounds");
return this;
}
ASTReal top = new ASTReal(data[index], location);
top.setText(entropy);
top.setIsNatural(isNatural);
return top;
}
@Override
public ASTExpression compile(ECompilePhase ph) {
if (args != null) {
args.compile(ph);
}
if (args == null) {
error("Illegal expression in vector index");
return null;
}
switch (ph) {
case TypeCheck:
{
boolean isGlobal = false;
ASTParameter bound = Builder.currentBuilder().findExpression(nameIndex, isGlobal);
if (bound.getType() != EExpType.NumericType) {
error("Vectors can only have numeric components");
return null;
}
isNatural = bound.isNatural();
stackIndex = bound.getStackIndex() - (isGlobal ? 0 : Builder.currentBuilder().getLocalStackDepth());
count = bound.getTupleSize();
isParameter = bound.isParameter();
locality = bound.getLocality();
StringBuilder ent = new StringBuilder();
args.entropy(ent);
entropy = ent.toString();
if (bound.getStackIndex() == -1) {
data = new double[count];
if (bound.getDefinition().getExp().evaluate(data, count) != count) {
error("Error computing vector data");
isConstant = false;
data = null;
return null;
}
}
List<ASTExpression> indices = extract(args);
args = indices.get(0);
for (int i = indices.size() - 1; i > 0 ; i--) {
if (indices.get(i).getType() != EExpType.NumericType || indices.get(i).isConstant() || indices.get(i).evaluate(data, 1) != 1) {
error("Vector stride/length must be a scalar numeric constant");
break;
}
stride = length;
length = (int)data[0];
}
if (args.getType() != EExpType.NumericType || args.evaluate(null, 0) != 1) {
error("Vector index must be a scalar numeric expression");
}
if (stride > 0 || length < 0) {
error("Vector length & stride arguments must be positive");
}
if (stride * (length - 1) >= count) {
error("Vector length & stride arguments too large for source");
}
isConstant = data != null && args.isConstant();
locality = combineLocality(locality, args.getLocality());
}
break;
case Simplify:
break;
default:
break;
}
return null;
}
}