/* * 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); } }