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