/*******************************************************************************
* Copyright (c) 2016 Institute for Software, HSR Hochschule fuer Technik
* Rapperswil, University of applied sciences and others
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*******************************************************************************/
package org.eclipse.cdt.internal.core.dom.parser.cpp.semantics;
import org.eclipse.cdt.internal.core.dom.parser.ITypeMarshalBuffer;
import org.eclipse.cdt.internal.core.dom.parser.cpp.ICPPEvaluation;
import org.eclipse.cdt.internal.core.dom.parser.cpp.ICPPEvaluation.ConstexprEvaluationContext;
import org.eclipse.cdt.internal.core.dom.parser.cpp.ICPPExecution;
import org.eclipse.cdt.internal.core.dom.parser.cpp.InstantiationContext;
import org.eclipse.core.runtime.CoreException;
public class ExecFor implements ICPPExecution {
private final ICPPExecution initializerExec;
private final ICPPEvaluation conditionExprEval;
private final ExecSimpleDeclaration conditionDeclExec;
private final ICPPEvaluation iterationEval;
private final ICPPExecution bodyExec;
public ExecFor(ICPPExecution initializerExec, ICPPEvaluation conditionExprEval, ExecSimpleDeclaration conditionDeclExec, ICPPEvaluation iterationEval, ICPPExecution bodyExec) {
this.initializerExec = initializerExec;
this.conditionExprEval = conditionExprEval;
this.conditionDeclExec = conditionDeclExec;
this.iterationEval = iterationEval;
this.bodyExec = bodyExec;
}
@Override
public ICPPExecution executeForFunctionCall(ActivationRecord record, ConstexprEvaluationContext context) {
for (evaluateInitializationStatement(record, context); conditionSatisfied(record, context); evaluateIterationExpression(record, context)) {
if (context.getStepsPerformed() >= ConstexprEvaluationContext.MAX_CONSTEXPR_EVALUATION_STEPS) {
return ExecIncomplete.INSTANCE;
}
ICPPExecution result = EvalUtil.executeStatement(bodyExec, record, context);
if (result instanceof ExecReturn) {
return result;
} else if (result instanceof ExecBreak) {
break;
} else if (result instanceof ExecContinue) {
continue;
}
}
return null;
}
private void evaluateInitializationStatement(ActivationRecord record, ConstexprEvaluationContext context) {
if (initializerExec != null) {
EvalUtil.executeStatement(initializerExec, record, context);
}
}
private boolean conditionSatisfied(ActivationRecord record, ConstexprEvaluationContext context) {
if (conditionExprEval != null) {
return EvalUtil.conditionExprSatisfied(conditionExprEval, record, context);
} else if (conditionDeclExec != null) {
return EvalUtil.conditionDeclSatisfied(conditionDeclExec, record, context);
}
return true;
}
private void evaluateIterationExpression(ActivationRecord record, ConstexprEvaluationContext context) {
if (iterationEval != null) {
iterationEval.computeForFunctionCall(record, context);
}
}
@Override
public ICPPExecution instantiate(InstantiationContext context, int maxDepth) {
ICPPExecution newInitializerExec = initializerExec != null ? initializerExec.instantiate(context, maxDepth) : null;
ICPPEvaluation newConditionExprEval = conditionExprEval != null ? conditionExprEval.instantiate(context, maxDepth) : null;
ExecSimpleDeclaration newConditionDeclExec = conditionDeclExec != null ? (ExecSimpleDeclaration) conditionDeclExec.instantiate(context, maxDepth) : null;
ICPPEvaluation newIterationEval = iterationEval != null ? iterationEval.instantiate(context, maxDepth) : null;
ICPPExecution newBodyExec = bodyExec.instantiate(context, maxDepth);
if (newInitializerExec == initializerExec &&
newConditionExprEval == conditionExprEval &&
newConditionDeclExec == conditionDeclExec &&
newIterationEval == iterationEval &&
newBodyExec == bodyExec) {
return this;
}
return new ExecFor(newInitializerExec, newConditionExprEval, newConditionDeclExec, newIterationEval, newBodyExec);
}
@Override
public void marshal(ITypeMarshalBuffer buffer, boolean includeValue) throws CoreException {
buffer.putShort(ITypeMarshalBuffer.EXEC_FOR);
buffer.marshalExecution(initializerExec, includeValue);
buffer.marshalEvaluation(conditionExprEval, includeValue);
buffer.marshalExecution(conditionDeclExec, includeValue);
buffer.marshalEvaluation(iterationEval, includeValue);
buffer.marshalExecution(bodyExec, includeValue);
}
public static ICPPExecution unmarshal(short firstBytes, ITypeMarshalBuffer buffer) throws CoreException {
ICPPExecution initializerExec = buffer.unmarshalExecution();
ICPPEvaluation conditionExprEval = buffer.unmarshalEvaluation();
ExecSimpleDeclaration conditionDeclExec = (ExecSimpleDeclaration) buffer.unmarshalExecution();
ICPPEvaluation iterationEval = buffer.unmarshalEvaluation();
ICPPExecution bodyExec = buffer.unmarshalExecution();
return new ExecFor(initializerExec, conditionExprEval, conditionDeclExec, iterationEval, bodyExec);
}
}