xref: /freebsd/contrib/llvm-project/llvm/lib/TextAPI/Symbol.cpp (revision a90b9d0159070121c221b966469c3e36d912bf82)
1 //===- Symbol.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 // Implements the Symbol.
10 //
11 //===----------------------------------------------------------------------===//
12 
13 #include "llvm/TextAPI/Symbol.h"
14 #include <string>
15 
16 namespace llvm {
17 namespace MachO {
18 
19 #if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
20 LLVM_DUMP_METHOD void Symbol::dump(raw_ostream &OS) const {
21   std::string Result;
22   if (isUndefined())
23     Result += "(undef) ";
24   if (isWeakDefined())
25     Result += "(weak-def) ";
26   if (isWeakReferenced())
27     Result += "(weak-ref) ";
28   if (isThreadLocalValue())
29     Result += "(tlv) ";
30   switch (Kind) {
31   case SymbolKind::GlobalSymbol:
32     Result += Name.str();
33     break;
34   case SymbolKind::ObjectiveCClass:
35     Result += "(ObjC Class) " + Name.str();
36     break;
37   case SymbolKind::ObjectiveCClassEHType:
38     Result += "(ObjC Class EH) " + Name.str();
39     break;
40   case SymbolKind::ObjectiveCInstanceVariable:
41     Result += "(ObjC IVar) " + Name.str();
42     break;
43   }
44   OS << Result;
45 }
46 #endif
47 
48 Symbol::const_filtered_target_range
49 Symbol::targets(ArchitectureSet Architectures) const {
50   std::function<bool(const Target &)> FN =
51       [Architectures](const Target &Target) {
52         return Architectures.has(Target.Arch);
53       };
54   return make_filter_range(Targets, FN);
55 }
56 
57 bool Symbol::operator==(const Symbol &O) const {
58   // Older Tapi files do not express all these symbol flags. In those
59   // cases, ignore those differences.
60   auto RemoveFlag = [](const Symbol &Sym, SymbolFlags &Flag) {
61     if (Sym.isData())
62       Flag &= ~SymbolFlags::Data;
63     if (Sym.isText())
64       Flag &= ~SymbolFlags::Text;
65   };
66   SymbolFlags LHSFlags = Flags;
67   SymbolFlags RHSFlags = O.Flags;
68   // Ignore Text and Data for now.
69   RemoveFlag(*this, LHSFlags);
70   RemoveFlag(O, RHSFlags);
71   return std::tie(Name, Kind, Targets, LHSFlags) ==
72          std::tie(O.Name, O.Kind, O.Targets, RHSFlags);
73 }
74 
75 SimpleSymbol parseSymbol(StringRef SymName, const SymbolFlags Flags) {
76   if (SymName.starts_with(ObjC1ClassNamePrefix))
77     return {SymName.drop_front(ObjC1ClassNamePrefix.size()),
78             SymbolKind::ObjectiveCClass};
79   if (SymName.starts_with(ObjC2ClassNamePrefix))
80     return {SymName.drop_front(ObjC2ClassNamePrefix.size()),
81             SymbolKind::ObjectiveCClass};
82   if (SymName.starts_with(ObjC2MetaClassNamePrefix))
83     return {SymName.drop_front(ObjC2MetaClassNamePrefix.size()),
84             SymbolKind::ObjectiveCClass};
85   if (SymName.starts_with(ObjC2EHTypePrefix)) {
86     // When classes without ehtype are used in try/catch blocks
87     // a weak-defined symbol is exported. In those cases, treat these as a
88     // global instead.
89     if ((Flags & SymbolFlags::WeakDefined) == SymbolFlags::WeakDefined)
90       return {SymName, SymbolKind::GlobalSymbol};
91     return {SymName.drop_front(ObjC2EHTypePrefix.size()),
92             SymbolKind::ObjectiveCClassEHType};
93   }
94 
95   if (SymName.starts_with(ObjC2IVarPrefix))
96     return {SymName.drop_front(ObjC2IVarPrefix.size()),
97             SymbolKind::ObjectiveCInstanceVariable};
98   return {SymName, SymbolKind::GlobalSymbol};
99 }
100 
101 } // end namespace MachO.
102 } // end namespace llvm.
103