// Part of the Carbon Language project, under the Apache License v2.0 with LLVM // Exceptions. See /LICENSE for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception #ifndef PARSER_PARSE_NODE_KIND_H_ #define PARSER_PARSE_NODE_KIND_H_ #include #include #include "llvm/ADT/StringRef.h" namespace Carbon { // A class wrapping an enumeration of the different kinds of nodes in the parse // tree. // // Rather than using a raw enumerator for each distinct kind of node produced by // the parser, we wrap the enumerator in a class to expose a more rich API // including bidirectional mappings to string spellings of the different kinds // and any relevant classification. // // Instances of this type should always be created using the `constexpr` static // member functions. These instances are designed specifically to be usable in // `case` labels of `switch` statements just like an enumerator would. class ParseNodeKind { public: // The formatting for this macro is weird due to a `clang-format` bug. See // https://bugs.llvm.org/show_bug.cgi?id=48320 for details. #define CARBON_PARSE_NODE_KIND(Name) \ static constexpr auto Name()->ParseNodeKind { \ return ParseNodeKind(KindEnum::Name); \ } #include "parser/parse_node_kind.def" // The default constructor is deleted as objects of this type should always be // constructed using the above factory functions for each unique kind. ParseNodeKind() = delete; auto operator==(const ParseNodeKind& rhs) const -> bool { return kind == rhs.kind; } auto operator!=(const ParseNodeKind& rhs) const -> bool { return kind != rhs.kind; } // Gets a friendly name for the token for logging or debugging. [[nodiscard]] auto GetName() const -> llvm::StringRef; private: enum class KindEnum : uint8_t { #define CARBON_PARSE_NODE_KIND(Name) Name, #include "parser/parse_node_kind.def" }; constexpr explicit ParseNodeKind(KindEnum k) : kind(k) {} // Enable conversion to our private enum, including in a `constexpr` context, // to enable usage in `switch` and `case`. The enum remains private and // nothing else should be using this. // NOLINTNEXTLINE(google-explicit-constructor) constexpr operator KindEnum() const { return kind; } KindEnum kind; }; // We expect the parse node kind to fit compactly into 8 bits. static_assert(sizeof(ParseNodeKind) == 1, "Kind objects include padding!"); } // namespace Carbon #endif // PARSER_PARSE_NODE_KIND_H_