/* * Copyright (C) 2015 SoftIndex LLC. * * Licensed 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 io.datakernel.codegen; import org.objectweb.asm.Label; import org.objectweb.asm.Type; import org.objectweb.asm.commons.GeneratorAdapter; import java.util.Iterator; import static io.datakernel.codegen.Expressions.*; import static io.datakernel.codegen.Utils.newLocal; import static org.objectweb.asm.Type.BOOLEAN_TYPE; import static org.objectweb.asm.Type.getType; final class ForEachHppcSet implements Expression { private final Class<?> iteratorType; private final Expression value; private final ForVar forVar; @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; ForEachHppcSet that = (ForEachHppcSet) o; if (iteratorType != null ? !iteratorType.equals(that.iteratorType) : that.iteratorType != null) return false; return !(value != null ? !value.equals(that.value) : that.value != null); } @Override public int hashCode() { int result = iteratorType != null ? iteratorType.hashCode() : 0; result = 31 * result + (value != null ? value.hashCode() : 0); return result; } ForEachHppcSet(Class<?> iteratorType, Expression field, ForVar forVar) { this.iteratorType = iteratorType; this.value = field; this.forVar = forVar; } @Override public Type type(Context ctx) { return Type.VOID_TYPE; } @Override public Type load(Context ctx) { GeneratorAdapter g = ctx.getGeneratorAdapter(); Label labelLoop = new Label(); Label labelExit = new Label(); VarLocal iterator = newLocal(ctx, getType(Iterator.class)); call(value, "iterator").load(ctx); iterator.store(ctx); g.mark(labelLoop); call(iterator, "hasNext").load(ctx); g.push(true); g.ifCmp(BOOLEAN_TYPE, GeneratorAdapter.NE, labelExit); VarLocal item = newLocal(ctx, getType(iteratorType)); cast(call(iterator, "next"), iteratorType).load(ctx); item.store(ctx); forVar.forVar(field(item, "value")).load(ctx); g.goTo(labelLoop); g.mark(labelExit); return Type.VOID_TYPE; } }