1 | //===- SymbolRemappingReader.cpp - Read symbol remapping file -------------===// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | // |
9 | // This file contains definitions needed for reading and applying symbol |
10 | // remapping files. |
11 | // |
12 | //===----------------------------------------------------------------------===// |
13 | |
14 | #include "llvm/ProfileData/SymbolRemappingReader.h" |
15 | #include "llvm/ADT/StringSwitch.h" |
16 | #include "llvm/ADT/Twine.h" |
17 | #include "llvm/Support/LineIterator.h" |
18 | #include "llvm/Support/MemoryBuffer.h" |
19 | |
20 | using namespace llvm; |
21 | |
22 | char SymbolRemappingParseError::ID; |
23 | |
24 | /// Load a set of name remappings from a text file. |
25 | /// |
26 | /// See the documentation at the top of the file for an explanation of |
27 | /// the expected format. |
28 | Error SymbolRemappingReader::read(MemoryBuffer &B) { |
29 | line_iterator LineIt(B, /*SkipBlanks=*/true, '#'); |
30 | |
31 | auto ReportError = [&](Twine Msg) { |
32 | return llvm::make_error<SymbolRemappingParseError>( |
33 | Args: B.getBufferIdentifier(), Args: LineIt.line_number(), Args&: Msg); |
34 | }; |
35 | |
36 | for (; !LineIt.is_at_eof(); ++LineIt) { |
37 | StringRef Line = *LineIt; |
38 | Line = Line.ltrim(Char: ' '); |
39 | // line_iterator only detects comments starting in column 1. |
40 | if (Line.starts_with(Prefix: "#" ) || Line.empty()) |
41 | continue; |
42 | |
43 | SmallVector<StringRef, 4> Parts; |
44 | Line.split(A&: Parts, Separator: ' ', /*MaxSplits*/MaxSplit: -1, /*KeepEmpty*/false); |
45 | |
46 | if (Parts.size() != 3) |
47 | return ReportError("Expected 'kind mangled_name mangled_name', " |
48 | "found '" + Line + "'" ); |
49 | |
50 | using FK = ItaniumManglingCanonicalizer::FragmentKind; |
51 | std::optional<FK> FragmentKind = StringSwitch<std::optional<FK>>(Parts[0]) |
52 | .Case(S: "name" , Value: FK::Name) |
53 | .Case(S: "type" , Value: FK::Type) |
54 | .Case(S: "encoding" , Value: FK::Encoding) |
55 | .Default(Value: std::nullopt); |
56 | if (!FragmentKind) |
57 | return ReportError("Invalid kind, expected 'name', 'type', or 'encoding'," |
58 | " found '" + Parts[0] + "'" ); |
59 | |
60 | using EE = ItaniumManglingCanonicalizer::EquivalenceError; |
61 | switch (Canonicalizer.addEquivalence(Kind: *FragmentKind, First: Parts[1], Second: Parts[2])) { |
62 | case EE::Success: |
63 | break; |
64 | |
65 | case EE::ManglingAlreadyUsed: |
66 | return ReportError("Manglings '" + Parts[1] + "' and '" + Parts[2] + "' " |
67 | "have both been used in prior remappings. Move this " |
68 | "remapping earlier in the file." ); |
69 | |
70 | case EE::InvalidFirstMangling: |
71 | return ReportError("Could not demangle '" + Parts[1] + "' " |
72 | "as a <" + Parts[0] + ">; invalid mangling?" ); |
73 | |
74 | case EE::InvalidSecondMangling: |
75 | return ReportError("Could not demangle '" + Parts[2] + "' " |
76 | "as a <" + Parts[0] + ">; invalid mangling?" ); |
77 | } |
78 | } |
79 | |
80 | return Error::success(); |
81 | } |
82 | |