/**
* Copyright 2011-2014 Asakusa Framework Team.
*
* 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 com.asakusafw.compiler.flow.processor;
import java.lang.reflect.Method;
import java.util.List;
import com.asakusafw.compiler.common.EnumUtil;
import com.asakusafw.compiler.common.TargetOperator;
import com.asakusafw.compiler.flow.RendezvousProcessor;
import com.asakusafw.utils.collections.Lists;
import com.asakusafw.utils.collections.Tuple2;
import com.asakusafw.utils.java.model.syntax.Expression;
import com.asakusafw.utils.java.model.syntax.ModelFactory;
import com.asakusafw.utils.java.model.syntax.SimpleName;
import com.asakusafw.utils.java.model.syntax.Statement;
import com.asakusafw.utils.java.model.util.ExpressionBuilder;
import com.asakusafw.utils.java.model.util.Models;
import com.asakusafw.utils.java.model.util.TypeBuilder;
import com.asakusafw.vocabulary.flow.graph.FlowElementPortDescription;
import com.asakusafw.vocabulary.flow.graph.OperatorDescription;
import com.asakusafw.vocabulary.operator.MasterBranch;
/**
* {@link MasterBranch マスタ分岐演算子}を処理する。
*/
@TargetOperator(MasterBranch.class)
public class MasterBranchFlowProcessor extends RendezvousProcessor {
@Override
public void emitRendezvous(Context context) {
ModelFactory f = context.getModelFactory();
MasterKindFlowAnalyzer masterAnalyzer = new MasterKindFlowAnalyzer(context);
FlowElementPortDescription tx = context.getInputPort(MasterBranch.ID_INPUT_TRANSACTION);
OperatorDescription desc = context.getOperatorDescription();
List<Expression> arguments = Lists.create();
arguments.add(masterAnalyzer.getGetCheckedMasterExpression());
arguments.add(context.getProcessInput(tx));
for (OperatorDescription.Parameter param : desc.getParameters()) {
arguments.add(Models.toLiteral(f, param.getValue()));
}
Method method = desc.getDeclaration().toMethod();
assert method != null : desc.getDeclaration();
Class<?> enumType = method.getReturnType();
List<Tuple2<Enum<?>, FlowElementPortDescription>> constants =
EnumUtil.extractConstants(enumType, desc.getOutputPorts());
Expression impl = context.createImplementation();
SimpleName branch = context.createName("branch");
context.addProcess(tx, new ExpressionBuilder(f, impl)
.method(desc.getDeclaration().getName(), arguments)
.toLocalVariableDeclaration(
context.convert(enumType),
branch));
List<Statement> cases = Lists.create();
for (Tuple2<Enum<?>, FlowElementPortDescription> tuple : constants) {
Enum<?> constant = tuple.first;
FlowElementPortDescription port = tuple.second;
ResultMirror next = context.getOutput(port);
cases.add(f.newSwitchCaseLabel(f.newSimpleName(constant.name())));
cases.add(next.createAdd(context.getProcessInput(tx)));
cases.add(f.newBreakStatement());
}
cases.add(f.newSwitchDefaultLabel());
cases.add(new TypeBuilder(f, context.convert(AssertionError.class))
.newObject(branch)
.toThrowStatement());
context.addProcess(tx, f.newSwitchStatement(branch, cases));
}
}