| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697 |
- // 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 CARBON_TOOLCHAIN_PARSE_TREE_NODE_LOCATION_TRANSLATOR_H_
- #define CARBON_TOOLCHAIN_PARSE_TREE_NODE_LOCATION_TRANSLATOR_H_
- #include "toolchain/diagnostics/diagnostic_emitter.h"
- #include "toolchain/lex/tokenized_buffer.h"
- #include "toolchain/parse/tree.h"
- namespace Carbon::Parse {
- class NodeLocation {
- public:
- NodeLocation(NodeId node_id) : NodeLocation(node_id, false) {}
- NodeLocation(NodeId node_id, bool token_only)
- : node_id_(node_id), token_only_(token_only) {}
- auto node_id() const -> NodeId { return node_id_; }
- auto token_only() const -> bool { return token_only_; }
- private:
- NodeId node_id_;
- bool token_only_;
- };
- inline auto TokenOnly(NodeId node_id) -> NodeLocation {
- return NodeLocation(node_id, true);
- }
- class NodeLocationTranslator
- : public DiagnosticLocationTranslator<NodeLocation> {
- public:
- explicit NodeLocationTranslator(const Lex::TokenizedBuffer* tokens,
- llvm::StringRef filename,
- const Tree* parse_tree)
- : token_translator_(tokens),
- filename_(filename),
- parse_tree_(parse_tree) {}
- // Map the given token into a diagnostic location.
- auto GetLocation(NodeLocation node_location) -> DiagnosticLocation override {
- // Support the invalid token as a way to emit only the filename, when there
- // is no line association.
- if (!node_location.node_id().is_valid()) {
- return {.file_name = filename_};
- }
- if (node_location.token_only()) {
- return token_translator_.GetLocation(
- parse_tree_->node_token(node_location.node_id()));
- }
- // Construct a location that encompasses all tokens that descend from this
- // node (including the root).
- Lex::TokenIndex start_token =
- parse_tree_->node_token(node_location.node_id());
- Lex::TokenIndex end_token = start_token;
- for (NodeId desc : parse_tree_->postorder(node_location.node_id())) {
- Lex::TokenIndex desc_token = parse_tree_->node_token(desc);
- if (!desc_token.is_valid()) {
- continue;
- }
- if (desc_token < start_token) {
- start_token = desc_token;
- } else if (desc_token > end_token) {
- end_token = desc_token;
- }
- }
- DiagnosticLocation start_loc = token_translator_.GetLocation(start_token);
- if (start_token == end_token) {
- return start_loc;
- }
- DiagnosticLocation end_loc = token_translator_.GetLocation(end_token);
- // For multiline locations we simply return the rest of the line for now
- // since true multiline locations are not yet supported.
- if (start_loc.line_number != end_loc.line_number) {
- start_loc.length = start_loc.line.size() - start_loc.column_number + 1;
- } else {
- if (start_loc.column_number != end_loc.column_number) {
- start_loc.length =
- end_loc.column_number + end_loc.length - start_loc.column_number;
- }
- }
- return start_loc;
- }
- private:
- Lex::TokenLocationTranslator token_translator_;
- llvm::StringRef filename_;
- const Tree* parse_tree_;
- };
- } // namespace Carbon::Parse
- #endif // CARBON_TOOLCHAIN_PARSE_TREE_NODE_LOCATION_TRANSLATOR_H_
|