/* 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 /** * @author Christian * @date 2014 * Solidity exception hierarchy. */ #pragma once #include #include #include #include #include #include #include #include #include #include #include namespace solidity::langutil { class Error; using ErrorList = std::vector>; struct CompilerError: virtual util::Exception {}; struct StackTooDeepError: virtual CompilerError {}; struct InternalCompilerError: virtual util::Exception {}; struct FatalError: virtual util::Exception {}; struct UnimplementedFeatureError: virtual util::Exception {}; struct InvalidAstError: virtual util::Exception {}; /// Assertion that throws an InternalCompilerError containing the given description if it is not met. #if !BOOST_PP_VARIADICS_MSVC #define solAssert(...) BOOST_PP_OVERLOAD(solAssert_,__VA_ARGS__)(__VA_ARGS__) #else #define solAssert(...) BOOST_PP_CAT(BOOST_PP_OVERLOAD(solAssert_,__VA_ARGS__)(__VA_ARGS__),BOOST_PP_EMPTY()) #endif #define solAssert_1(CONDITION) \ solAssert_2(CONDITION, "") #define solAssert_2(CONDITION, DESCRIPTION) \ assertThrowWithDefaultDescription( \ CONDITION, \ ::solidity::langutil::InternalCompilerError, \ DESCRIPTION, \ "Solidity assertion failed" \ ) /// Assertion that throws an UnimplementedFeatureError containing the given description if it is not met. #if !BOOST_PP_VARIADICS_MSVC #define solUnimplementedAssert(...) BOOST_PP_OVERLOAD(solUnimplementedAssert_,__VA_ARGS__)(__VA_ARGS__) #else #define solUnimplementedAssert(...) BOOST_PP_CAT(BOOST_PP_OVERLOAD(solUnimplementedAssert_,__VA_ARGS__)(__VA_ARGS__),BOOST_PP_EMPTY()) #endif #define solUnimplementedAssert_1(CONDITION) \ solUnimplementedAssert_2(CONDITION, "") #define solUnimplementedAssert_2(CONDITION, DESCRIPTION) \ assertThrowWithDefaultDescription( \ CONDITION, \ ::solidity::langutil::UnimplementedFeatureError, \ DESCRIPTION, \ "Unimplemented feature" \ ) /// Helper that unconditionally reports an unimplemented feature. #define solUnimplemented(DESCRIPTION) \ solUnimplementedAssert(false, DESCRIPTION) /// Assertion that throws an InvalidAstError containing the given description if it is not met. #if !BOOST_PP_VARIADICS_MSVC #define astAssert(...) BOOST_PP_OVERLOAD(astAssert_,__VA_ARGS__)(__VA_ARGS__) #else #define astAssert(...) BOOST_PP_CAT(BOOST_PP_OVERLOAD(astAssert_,__VA_ARGS__)(__VA_ARGS__),BOOST_PP_EMPTY()) #endif #define astAssert_1(CONDITION) \ astAssert_2(CONDITION, "") #define astAssert_2(CONDITION, DESCRIPTION) \ assertThrowWithDefaultDescription( \ CONDITION, \ ::solidity::langutil::InvalidAstError, \ DESCRIPTION, \ "AST assertion failed" \ ) using errorSourceLocationInfo = std::pair; class SecondarySourceLocation { public: SecondarySourceLocation& append(std::string const& _errMsg, SourceLocation const& _sourceLocation) { infos.emplace_back(_errMsg, _sourceLocation); return *this; } SecondarySourceLocation& append(SecondarySourceLocation&& _other) { infos += std::move(_other.infos); return *this; } /// Limits the number of secondary source locations to 32 and appends a notice to the /// error message. void limitSize(std::string& _message) { size_t occurrences = infos.size(); if (occurrences > 32) { infos.resize(32); _message += " Truncated from " + std::to_string(occurrences) + " to the first 32 occurrences."; } } std::vector infos; }; using errinfo_sourceLocation = boost::error_info; using errinfo_secondarySourceLocation = boost::error_info; /** * Unique identifiers are used to tag and track individual error cases. * They are passed as the first parameter of error reporting functions. * Suffix _error helps to find them in the sources. * The struct ErrorId prevents incidental calls like typeError(3141) instead of typeError(3141_error). * To create a new ID, one can add 0000_error and then run "python ./scripts/error_codes.py --fix" * from the root of the repo. */ struct ErrorId { unsigned long long error = 0; bool operator==(ErrorId const& _rhs) const { return error == _rhs.error; } bool operator!=(ErrorId const& _rhs) const { return !(*this == _rhs); } bool operator<(ErrorId const& _rhs) const { return error < _rhs.error; } }; constexpr ErrorId operator"" _error(unsigned long long _error) { return ErrorId{ _error }; } class Error: virtual public util::Exception { public: enum class Type { CodeGenerationError, DeclarationError, DocstringParsingError, ParserError, TypeError, SyntaxError, Warning, Info }; enum class Severity { Error, Warning, Info }; Error( ErrorId _errorId, Type _type, std::string const& _description, SourceLocation const& _location = SourceLocation(), SecondarySourceLocation const& _secondaryLocation = SecondarySourceLocation() ); ErrorId errorId() const { return m_errorId; } Type type() const { return m_type; } std::string const& typeName() const { return m_typeName; } /// helper functions static Error const* containsErrorOfType(ErrorList const& _list, Error::Type _type) { for (auto e: _list) if (e->type() == _type) return e.get(); return nullptr; } static Severity errorSeverity(Type _type) { if (_type == Type::Info) return Severity::Info; if (_type == Type::Warning) return Severity::Warning; return Severity::Error; } static bool isError(Severity _severity) { return _severity == Severity::Error; } static bool isError(Type _type) { return isError(errorSeverity(_type)); } static bool containsErrors(ErrorList const& _list) { for (auto e: _list) if (isError(e->type())) return true; return false; } static std::string formatErrorSeverity(Severity _severity) { if (_severity == Severity::Info) return "Info"; if (_severity == Severity::Warning) return "Warning"; solAssert(isError(_severity), ""); return "Error"; } static std::string formatErrorSeverityLowercase(Severity _severity) { switch (_severity) { case Severity::Info: return "info"; case Severity::Warning: return "warning"; case Severity::Error: solAssert(isError(_severity), ""); return "error"; } solAssert(false, ""); } private: ErrorId m_errorId; Type m_type; std::string m_typeName; }; }