/******************************************************************************* * Copyright (c) 2009-2013 CWI * 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 * * Contributors: * * Jurgen J. Vinju - Jurgen.Vinju@cwi.nl - CWI * * Mark Hills - Mark.Hills@cwi.nl (CWI) * * Arnold Lankamp - Arnold.Lankamp@cwi.nl *******************************************************************************/ package org.rascalmpl.semantics.dynamic; import java.util.List; import org.rascalmpl.ast.Case; import org.rascalmpl.ast.Expression; import org.rascalmpl.ast.Strategy; import org.rascalmpl.interpreter.IEvaluator; import org.rascalmpl.interpreter.TraversalEvaluator; import org.rascalmpl.interpreter.TraversalEvaluator.CaseBlockList; import org.rascalmpl.interpreter.TraversalEvaluator.DIRECTION; import org.rascalmpl.interpreter.TraversalEvaluator.FIXEDPOINT; import org.rascalmpl.interpreter.TraversalEvaluator.PROGRESS; import org.rascalmpl.interpreter.asserts.ImplementationError; import org.rascalmpl.interpreter.result.Result; import org.rascalmpl.interpreter.staticErrors.UnexpectedType; import org.rascalmpl.interpreter.utils.Cases; import org.rascalmpl.value.IConstructor; import org.rascalmpl.value.ISourceLocation; import org.rascalmpl.value.IValue; import org.rascalmpl.value.type.Type; public abstract class Visit extends org.rascalmpl.ast.Visit { static public class DefaultStrategy extends org.rascalmpl.ast.Visit.DefaultStrategy { private final CaseBlockList blocks; public DefaultStrategy(ISourceLocation __param1, IConstructor tree, Expression __param2, List<Case> __param3) { super(__param1, tree, __param2, __param3); blocks = new CaseBlockList(Cases.precompute(getCases())); } @Override public Result<IValue> interpret(IEvaluator<Result<IValue>> __eval) { Result<IValue> subject = this.getSubject().interpret(__eval); TraversalEvaluator te = new TraversalEvaluator(__eval); try { __eval.__pushTraversalEvaluator(te); IValue val = te.traverse(subject.getValue(), blocks, DIRECTION.BottomUp, PROGRESS.Continuing, FIXEDPOINT.No); if (!val.getType().isSubtypeOf(subject.getType())) { // this is not a static error but an extra run-time sanity check throw new ImplementationError("this should really never happen", new UnexpectedType(subject.getType(), val.getType(), this)); } return org.rascalmpl.interpreter.result.ResultFactory.makeResult(subject.getType(), val, __eval); } catch (UnexpectedType e) { e.setLocation(getLocation()); throw e; } finally { __eval.__popTraversalEvaluator(); } } } static public class GivenStrategy extends org.rascalmpl.ast.Visit.GivenStrategy { private final CaseBlockList blocks; public GivenStrategy(ISourceLocation __param1, IConstructor tree, Strategy __param2, Expression __param3, List<Case> __param4) { super(__param1, tree, __param2, __param3, __param4); blocks = new CaseBlockList(Cases.precompute(getCases())); } @Override public Result<IValue> interpret(IEvaluator<Result<IValue>> __eval) { Result<IValue> subject = this.getSubject().interpret(__eval); // TODO: warning switched to static type here, but not sure if // that's correct... Type subjectType = subject.getType(); if (subjectType.isConstructor()) { subjectType = subjectType.getAbstractDataType(); } Strategy s = this.getStrategy(); DIRECTION direction = DIRECTION.BottomUp; PROGRESS progress = PROGRESS.Continuing; FIXEDPOINT fixedpoint = FIXEDPOINT.No; if (s.isBottomUp()) { direction = DIRECTION.BottomUp; } else if (s.isBottomUpBreak()) { direction = DIRECTION.BottomUp; progress = PROGRESS.Breaking; } else if (s.isInnermost()) { direction = DIRECTION.BottomUp; fixedpoint = FIXEDPOINT.Yes; } else if (s.isTopDown()) { direction = DIRECTION.TopDown; } else if (s.isTopDownBreak()) { direction = DIRECTION.TopDown; progress = PROGRESS.Breaking; } else if (s.isOutermost()) { direction = DIRECTION.TopDown; fixedpoint = FIXEDPOINT.Yes; } else { throw new ImplementationError("Unknown strategy " + s); } TraversalEvaluator te = new TraversalEvaluator(__eval); try { __eval.__pushTraversalEvaluator(te); IValue val = te .traverse(subject.getValue(), blocks,direction, progress, fixedpoint); Type t = val.getType(); return org.rascalmpl.interpreter.result.ResultFactory.makeResult(t, val, __eval); } catch (UnexpectedType e) { e.setLocation(getLocation()); throw e; } finally { __eval.__popTraversalEvaluator(); } } } public Visit(ISourceLocation __param1, IConstructor tree) { super(__param1, tree); } }