/* * #%~ * The VDM Type Checker * %% * Copyright (C) 2008 - 2014 Overture * %% * This program 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. * * This program 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 this program. If not, see * <http://www.gnu.org/licenses/gpl-3.0.html>. * #~% */ package org.overture.typechecker.utilities; import org.overture.ast.analysis.AnalysisException; import org.overture.ast.analysis.QuestionAnswerAdaptor; import org.overture.ast.definitions.AImportedDefinition; import org.overture.ast.definitions.AInheritedDefinition; import org.overture.ast.definitions.ARenamedDefinition; import org.overture.ast.definitions.AStateDefinition; import org.overture.ast.definitions.ATypeDefinition; import org.overture.ast.definitions.PDefinition; import org.overture.ast.definitions.SClassDefinition; import org.overture.ast.intf.lex.ILexNameToken; import org.overture.ast.node.INode; import org.overture.ast.typechecker.NameScope; import org.overture.typechecker.assistant.ITypeCheckerAssistantFactory; /** * This class implements a way to find type from a node in the AST * * @author kel */ public class DefinitionFinder extends QuestionAnswerAdaptor<DefinitionFinder.Newquestion, PDefinition> { public static class Newquestion { final ILexNameToken sought; final String fromModule; public Newquestion(ILexNameToken sought, String fromModule) { this.fromModule = fromModule; this.sought = sought; } } protected ITypeCheckerAssistantFactory af; public DefinitionFinder(ITypeCheckerAssistantFactory af) { this.af = af; } @Override public PDefinition defaultSClassDefinition(SClassDefinition node, Newquestion question) throws AnalysisException { if (!question.sought.getExplicit() && question.sought.getName().equals(node.getName().getName()) || question.sought.equals(node.getName().getClassName())) { return node; // Class referred to as "A" or "CLASS`A" } PDefinition def = af.createPDefinitionAssistant().findType(node.getDefinitions(), question.sought, null); if (def == null) { for (PDefinition d : node.getAllInheritedDefinitions()) { PDefinition indef = af.createPDefinitionAssistant().findType(d, question.sought, null); if (indef != null) { def = indef; break; } } } return def; } @Override public PDefinition caseAImportedDefinition(AImportedDefinition node, Newquestion question) throws AnalysisException { // We can only find an import if it is being sought from the module that // imports it. if (question.fromModule != null && !node.getLocation().getModule().equals(question.fromModule)) { return null; // Someone else's import } PDefinition def = af.createPDefinitionAssistant().findType(node.getDef(), question.sought, question.fromModule); if (def != null) { af.createPDefinitionAssistant().markUsed(node); } return def; } @Override public PDefinition caseAInheritedDefinition(AInheritedDefinition node, Newquestion question) throws AnalysisException { if (node.getSuperdef() instanceof ATypeDefinition && question.sought.equals(node.getName())) { return node; } return null; } @Override public PDefinition caseARenamedDefinition(ARenamedDefinition node, Newquestion question) throws AnalysisException { // We can only find an import if it is being sought from the module that // imports it. if (question.fromModule != null && !node.getLocation().getModule().equals(question.fromModule)) { return null; // Someone else's import } PDefinition renamed = af.createPDefinitionAssistant().findName(node, question.sought, NameScope.TYPENAME); if (renamed != null && node.getDef() instanceof ATypeDefinition) { af.createPDefinitionAssistant().markUsed(node.getDef()); return renamed; } else { return af.createPDefinitionAssistant().findType(node.getDef(), question.sought, question.fromModule); } } @Override public PDefinition caseAStateDefinition(AStateDefinition node, Newquestion question) throws AnalysisException { if (af.createPDefinitionAssistant().findName(node, question.sought, NameScope.STATE) != null) { return node; } return null; } @Override public PDefinition caseATypeDefinition(ATypeDefinition node, Newquestion question) throws AnalysisException { if (!node.getComposeDefinitions().isEmpty()) { for (PDefinition compose : node.getComposeDefinitions()) { PDefinition found = af.createPDefinitionAssistant().findNameBaseCase(compose, question.sought, NameScope.TYPENAME); if (found != null) { return found; } } } return af.createPDefinitionAssistant().findNameBaseCase(node, question.sought, NameScope.TYPENAME); } @Override public PDefinition defaultPDefinition(PDefinition node, Newquestion question) throws AnalysisException { return null; } @Override public PDefinition createNewReturnValue(INode node, Newquestion question) { assert false : "default case should never happen in TypeFinder"; return null; } @Override public PDefinition createNewReturnValue(Object node, Newquestion question) { assert false : "default case should never happen in TypeFinder"; return null; } }