/* This file is part of solidity. solidity 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. solidity 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 solidity. If not, see . */ // SPDX-License-Identifier: GPL-3.0 #include #include #include #include #include #include #include using namespace solidity::langutil; using namespace solidity::frontend::experimental; // TODO: creating all of them for all nodes up front may be wasteful, we should improve the mechanism. struct Analysis::AnnotationContainer { TypeClassMemberRegistration::Annotation typeClassMemberRegistrationAnnotation; TypeClassRegistration::Annotation typeClassRegistrationAnnotation; TypeRegistration::Annotation typeRegistrationAnnotation; TypeInference::Annotation typeInferenceAnnotation; }; struct Analysis::GlobalAnnotationContainer { TypeClassMemberRegistration::GlobalAnnotation typeClassMemberRegistrationAnnotation; TypeClassRegistration::GlobalAnnotation typeClassRegistrationAnnotation; TypeRegistration::GlobalAnnotation typeRegistrationAnnotation; TypeInference::GlobalAnnotation typeInferenceAnnotation; }; template<> TypeClassMemberRegistration::Annotation& solidity::frontend::experimental::detail::AnnotationFetcher::get(ASTNode const& _node) { return analysis.annotationContainer(_node).typeClassMemberRegistrationAnnotation; } template<> TypeClassMemberRegistration::GlobalAnnotation const& solidity::frontend::experimental::detail::ConstAnnotationFetcher::get() const { return analysis.annotationContainer().typeClassMemberRegistrationAnnotation; } template<> TypeClassMemberRegistration::GlobalAnnotation& solidity::frontend::experimental::detail::AnnotationFetcher::get() { return analysis.annotationContainer().typeClassMemberRegistrationAnnotation; } template<> TypeClassMemberRegistration::Annotation const& solidity::frontend::experimental::detail::ConstAnnotationFetcher::get(ASTNode const& _node) const { return analysis.annotationContainer(_node).typeClassMemberRegistrationAnnotation; } template<> TypeClassRegistration::Annotation& solidity::frontend::experimental::detail::AnnotationFetcher::get(ASTNode const& _node) { return analysis.annotationContainer(_node).typeClassRegistrationAnnotation; } template<> TypeClassRegistration::GlobalAnnotation const& solidity::frontend::experimental::detail::ConstAnnotationFetcher::get() const { return analysis.annotationContainer().typeClassRegistrationAnnotation; } template<> TypeClassRegistration::GlobalAnnotation& solidity::frontend::experimental::detail::AnnotationFetcher::get() { return analysis.annotationContainer().typeClassRegistrationAnnotation; } template<> TypeClassRegistration::Annotation const& solidity::frontend::experimental::detail::ConstAnnotationFetcher::get(ASTNode const& _node) const { return analysis.annotationContainer(_node).typeClassRegistrationAnnotation; } template<> TypeRegistration::Annotation& solidity::frontend::experimental::detail::AnnotationFetcher::get(ASTNode const& _node) { return analysis.annotationContainer(_node).typeRegistrationAnnotation; } template<> TypeRegistration::GlobalAnnotation const& solidity::frontend::experimental::detail::ConstAnnotationFetcher::get() const { return analysis.annotationContainer().typeRegistrationAnnotation; } template<> TypeRegistration::GlobalAnnotation& solidity::frontend::experimental::detail::AnnotationFetcher::get() { return analysis.annotationContainer().typeRegistrationAnnotation; } template<> TypeRegistration::Annotation const& solidity::frontend::experimental::detail::ConstAnnotationFetcher::get(ASTNode const& _node) const { return analysis.annotationContainer(_node).typeRegistrationAnnotation; } template<> TypeInference::Annotation& solidity::frontend::experimental::detail::AnnotationFetcher::get(ASTNode const& _node) { return analysis.annotationContainer(_node).typeInferenceAnnotation; } template<> TypeInference::Annotation const& solidity::frontend::experimental::detail::ConstAnnotationFetcher::get(ASTNode const& _node) const { return analysis.annotationContainer(_node).typeInferenceAnnotation; } template<> TypeInference::GlobalAnnotation const& solidity::frontend::experimental::detail::ConstAnnotationFetcher::get() const { return analysis.annotationContainer().typeInferenceAnnotation; } template<> TypeInference::GlobalAnnotation& solidity::frontend::experimental::detail::AnnotationFetcher::get() { return analysis.annotationContainer().typeInferenceAnnotation; } Analysis::AnnotationContainer& Analysis::annotationContainer(ASTNode const& _node) { solAssert(_node.id() > 0); size_t id = static_cast(_node.id()); solAssert(id <= m_maxAstId); return m_annotations[id]; } Analysis::AnnotationContainer const& Analysis::annotationContainer(ASTNode const& _node) const { solAssert(_node.id() > 0); size_t id = static_cast(_node.id()); solAssert(id <= m_maxAstId); return m_annotations[id]; } Analysis::Analysis(langutil::ErrorReporter& _errorReporter, uint64_t _maxAstId): m_errorReporter(_errorReporter), m_maxAstId(_maxAstId), m_annotations(std::make_unique(static_cast(_maxAstId + 1))), m_globalAnnotation(std::make_unique()) { } Analysis::~Analysis() {} template std::tuple...> makeIndexTuple(std::index_sequence) { return std::make_tuple( std::integral_constant{}...); } bool Analysis::check(std::vector> const& _sourceUnits) { using AnalysisSteps = std::tuple< SyntaxRestrictor, TypeClassRegistration, TypeRegistration, TypeClassMemberRegistration, TypeInference, DebugWarner >; return std::apply([&](auto... _indexTuple) { return ([&](auto&& _step) { for (auto source: _sourceUnits) if (!_step.analyze(*source)) return false; return true; }(std::tuple_element_t{*this}) && ...); }, makeIndexTuple(std::make_index_sequence>{})); /* { SyntaxRestrictor syntaxRestrictor{*this}; for (auto source: _sourceUnits) if (!syntaxRestrictor.analyze(*source)) return false; } { TypeRegistration typeRegistration{*this}; for (auto source: _sourceUnits) if (!typeRegistration.analyze(*source)) return false; } { TypeInference typeInference{*this}; for (auto source: _sourceUnits) if (!typeInference.analyze(*source)) return false; } return true; */ }