/******************************************************************************* * * Copyright (C) 2008 Fujitsu Services Ltd. * * Author: Nick Battle * * This file is part of VDMJ. * * VDMJ is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * VDMJ is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with VDMJ. If not, see <http://www.gnu.org/licenses/>. * ******************************************************************************/ package org.overture.interpreter.values; import java.io.Serializable; import org.overture.ast.analysis.AnalysisException; import org.overture.ast.intf.lex.ILexLocation; import org.overture.config.Settings; import org.overture.interpreter.runtime.Context; import org.overture.interpreter.runtime.ContextException; import org.overture.interpreter.runtime.ValueException; public class ClassInvariantListener implements ValueListener, Serializable { private static final long serialVersionUID = 1L; public final OperationValue invopvalue; public boolean doInvariantChecks = true; public ClassInvariantListener(OperationValue invopvalue) { this.invopvalue = invopvalue; } public void changedValue(ILexLocation location, Value value, Context ctxt) throws AnalysisException { if (doInvariantChecks && Settings.invchecks) { try { // In VDM++ and VDM-RT, we do not want to do thread swaps half way // through an invariant check, so we set the atomic flag around the // conversion. This also stops VDM-RT from performing "time step" // calculations. try { ctxt.threadState.setAtomic(true); ctxt.threadState.setPure(true); boolean inv = invopvalue.eval(location, new ValueList(), ctxt).boolValue(ctxt); if (!inv) { throw new ContextException(4130, "Instance invariant violated: " + invopvalue.name, location, ctxt); } } finally { ctxt.threadState.setAtomic(false); ctxt.threadState.setPure(false); } } catch (ValueException e) { throw new ContextException(e, location); } } } }