xref: /freebsd/contrib/llvm-project/llvm/lib/Object/TapiFile.cpp (revision 8a4dda33d67586ca2624f2a38417baa03a533a7f)
1  //===- TapiFile.cpp -------------------------------------------------------===//
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 defines the Text-based Dynamcic Library Stub format.
10  //
11  //===----------------------------------------------------------------------===//
12  
13  #include "llvm/Object/TapiFile.h"
14  #include "llvm/ADT/StringRef.h"
15  #include "llvm/BinaryFormat/MachO.h"
16  #include "llvm/Object/Error.h"
17  #include "llvm/Support/MemoryBufferRef.h"
18  #include "llvm/TextAPI/ArchitectureSet.h"
19  #include "llvm/TextAPI/InterfaceFile.h"
20  #include "llvm/TextAPI/Platform.h"
21  #include "llvm/TextAPI/Symbol.h"
22  
23  using namespace llvm;
24  using namespace MachO;
25  using namespace object;
26  
27  static uint32_t getFlags(const Symbol *Sym) {
28    uint32_t Flags = BasicSymbolRef::SF_Global;
29    if (Sym->isUndefined())
30      Flags |= BasicSymbolRef::SF_Undefined;
31    else
32      Flags |= BasicSymbolRef::SF_Exported;
33  
34    if (Sym->isWeakDefined() || Sym->isWeakReferenced())
35      Flags |= BasicSymbolRef::SF_Weak;
36  
37    return Flags;
38  }
39  
40  static SymbolRef::Type getType(const Symbol *Sym) {
41    SymbolRef::Type Type = SymbolRef::ST_Unknown;
42    if (Sym->isData())
43      Type = SymbolRef::ST_Data;
44    else if (Sym->isText())
45      Type = SymbolRef::ST_Function;
46  
47    return Type;
48  }
49  
50  TapiFile::TapiFile(MemoryBufferRef Source, const InterfaceFile &Interface,
51                     Architecture Arch)
52      : SymbolicFile(ID_TapiFile, Source), Arch(Arch) {
53    for (const auto *Symbol : Interface.symbols()) {
54      if (!Symbol->getArchitectures().has(Arch))
55        continue;
56  
57      switch (Symbol->getKind()) {
58      case SymbolKind::GlobalSymbol:
59        Symbols.emplace_back(StringRef(), Symbol->getName(), getFlags(Symbol),
60                             ::getType(Symbol));
61        break;
62      case SymbolKind::ObjectiveCClass:
63        if (Interface.getPlatforms().count(PLATFORM_MACOS) && Arch == AK_i386) {
64          Symbols.emplace_back(ObjC1ClassNamePrefix, Symbol->getName(),
65                               getFlags(Symbol), ::getType(Symbol));
66        } else {
67          Symbols.emplace_back(ObjC2ClassNamePrefix, Symbol->getName(),
68                               getFlags(Symbol), ::getType(Symbol));
69          Symbols.emplace_back(ObjC2MetaClassNamePrefix, Symbol->getName(),
70                               getFlags(Symbol), ::getType(Symbol));
71        }
72        break;
73      case SymbolKind::ObjectiveCClassEHType:
74        Symbols.emplace_back(ObjC2EHTypePrefix, Symbol->getName(),
75                             getFlags(Symbol), ::getType(Symbol));
76        break;
77      case SymbolKind::ObjectiveCInstanceVariable:
78        Symbols.emplace_back(ObjC2IVarPrefix, Symbol->getName(), getFlags(Symbol),
79                             ::getType(Symbol));
80        break;
81      }
82    }
83  }
84  
85  TapiFile::~TapiFile() = default;
86  
87  void TapiFile::moveSymbolNext(DataRefImpl &DRI) const { DRI.d.a++; }
88  
89  Error TapiFile::printSymbolName(raw_ostream &OS, DataRefImpl DRI) const {
90    assert(DRI.d.a < Symbols.size() && "Attempt to access symbol out of bounds");
91    const Symbol &Sym = Symbols[DRI.d.a];
92    OS << Sym.Prefix << Sym.Name;
93    return Error::success();
94  }
95  
96  Expected<SymbolRef::Type> TapiFile::getSymbolType(DataRefImpl DRI) const {
97    assert(DRI.d.a < Symbols.size() && "Attempt to access symbol out of bounds");
98    return Symbols[DRI.d.a].Type;
99  }
100  
101  Expected<uint32_t> TapiFile::getSymbolFlags(DataRefImpl DRI) const {
102    assert(DRI.d.a < Symbols.size() && "Attempt to access symbol out of bounds");
103    return Symbols[DRI.d.a].Flags;
104  }
105  
106  basic_symbol_iterator TapiFile::symbol_begin() const {
107    DataRefImpl DRI;
108    DRI.d.a = 0;
109    return BasicSymbolRef{DRI, this};
110  }
111  
112  basic_symbol_iterator TapiFile::symbol_end() const {
113    DataRefImpl DRI;
114    DRI.d.a = Symbols.size();
115    return BasicSymbolRef{DRI, this};
116  }
117