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