/*
* Licensed to Elasticsearch under one or more contributor
* license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright
* ownership. Elasticsearch licenses this file to you under
* the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.elasticsearch.painless.node;
import org.elasticsearch.painless.Definition.Sort;
import org.elasticsearch.painless.Definition.Type;
import org.elasticsearch.painless.Globals;
import org.elasticsearch.painless.Locals;
import org.elasticsearch.painless.Location;
import org.elasticsearch.painless.MethodWriter;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
/**
* Represents an array load/store and defers to a child subnode.
*/
public final class PBrace extends AStoreable {
private AExpression index;
private AStoreable sub = null;
public PBrace(Location location, AExpression prefix, AExpression index) {
super(location, prefix);
this.index = Objects.requireNonNull(index);
}
@Override
void extractVariables(Set<String> variables) {
prefix.extractVariables(variables);
index.extractVariables(variables);
}
@Override
void analyze(Locals locals) {
prefix.analyze(locals);
prefix.expected = prefix.actual;
prefix = prefix.cast(locals);
Sort sort = prefix.actual.sort;
if (sort == Sort.ARRAY) {
sub = new PSubBrace(location, prefix.actual, index);
} else if (sort == Sort.DEF) {
sub = new PSubDefArray(location, index);
} else if (Map.class.isAssignableFrom(prefix.actual.clazz)) {
sub = new PSubMapShortcut(location, prefix.actual.struct, index);
} else if (List.class.isAssignableFrom(prefix.actual.clazz)) {
sub = new PSubListShortcut(location, prefix.actual.struct, index);
} else {
throw createError(new IllegalArgumentException("Illegal array access on type [" + prefix.actual.name + "]."));
}
sub.write = write;
sub.read = read;
sub.expected = expected;
sub.explicit = explicit;
sub.analyze(locals);
actual = sub.actual;
}
@Override
void write(MethodWriter writer, Globals globals) {
prefix.write(writer, globals);
sub.write(writer, globals);
}
@Override
boolean isDefOptimized() {
return sub.isDefOptimized();
}
@Override
void updateActual(Type actual) {
sub.updateActual(actual);
this.actual = actual;
}
@Override
int accessElementCount() {
return sub.accessElementCount();
}
@Override
void setup(MethodWriter writer, Globals globals) {
prefix.write(writer, globals);
sub.setup(writer, globals);
}
@Override
void load(MethodWriter writer, Globals globals) {
sub.load(writer, globals);
}
@Override
void store(MethodWriter writer, Globals globals) {
sub.store(writer, globals);
}
@Override
public String toString() {
return singleLineToString(prefix, index);
}
}