xref: /freebsd/contrib/llvm-project/llvm/lib/TextAPI/RecordsSlice.cpp (revision 357378bbdedf24ce2b90e9bd831af4a9db3ec70a)
1 //===- RecordsSlice.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 Records Slice APIs.
10 //
11 //===----------------------------------------------------------------------===//
12 
13 #include "llvm/TextAPI/RecordsSlice.h"
14 #include "llvm/ADT/SetVector.h"
15 #include "llvm/TextAPI/Record.h"
16 #include "llvm/TextAPI/Symbol.h"
17 #include <utility>
18 
19 using namespace llvm;
20 using namespace llvm::MachO;
21 
22 Record *RecordsSlice::addRecord(StringRef Name, SymbolFlags Flags,
23                                 GlobalRecord::Kind GV, RecordLinkage Linkage) {
24   // Find a specific Record type to capture.
25   auto [APIName, SymKind] = parseSymbol(Name, Flags);
26   Name = APIName;
27   switch (SymKind) {
28   case SymbolKind::GlobalSymbol:
29     return addGlobal(Name, Linkage, GV, Flags);
30   case SymbolKind::ObjectiveCClass:
31     return addObjCInterface(Name, Linkage);
32   case SymbolKind::ObjectiveCClassEHType:
33     return addObjCInterface(Name, Linkage, /*HasEHType=*/true);
34   case SymbolKind::ObjectiveCInstanceVariable: {
35     auto [Super, IVar] = Name.split('.');
36     // Attempt to find super class.
37     ObjCContainerRecord *Container = findContainer(/*isIVar=*/false, Super);
38     // If not found, create extension since there is no mapped class symbol.
39     if (Container == nullptr)
40       Container = addObjCCategory(Super, {});
41     return addObjCIVar(Container, IVar, Linkage);
42   }
43   }
44 
45   llvm_unreachable("unexpected symbol kind when adding to Record Slice");
46 }
47 
48 ObjCContainerRecord *RecordsSlice::findContainer(bool IsIVar,
49                                                  StringRef Name) const {
50   StringRef Super = IsIVar ? Name.split('.').first : Name;
51   ObjCContainerRecord *Container = findObjCInterface(Super);
52   // Ivars can only exist with extensions, if they did not come from
53   // class.
54   if (Container == nullptr)
55     Container = findObjCCategory(Super, "");
56   return Container;
57 }
58 
59 template <typename R, typename C = RecordMap<R>, typename K = StringRef>
60 R *findRecord(K Key, const C &Container) {
61   const auto *Record = Container.find(Key);
62   if (Record == Container.end())
63     return nullptr;
64   return Record->second.get();
65 }
66 
67 GlobalRecord *RecordsSlice::findGlobal(StringRef Name,
68                                        GlobalRecord::Kind GV) const {
69   auto *Record = findRecord<GlobalRecord>(Name, Globals);
70   if (!Record)
71     return nullptr;
72 
73   switch (GV) {
74   case GlobalRecord::Kind::Variable: {
75     if (!Record->isVariable())
76       return nullptr;
77     break;
78   }
79   case GlobalRecord::Kind::Function: {
80     if (!Record->isFunction())
81       return nullptr;
82     break;
83   }
84   case GlobalRecord::Kind::Unknown:
85     return Record;
86   }
87 
88   return Record;
89 }
90 
91 ObjCInterfaceRecord *RecordsSlice::findObjCInterface(StringRef Name) const {
92   return findRecord<ObjCInterfaceRecord>(Name, Classes);
93 }
94 
95 ObjCCategoryRecord *RecordsSlice::findObjCCategory(StringRef ClassToExtend,
96                                                    StringRef Category) const {
97   return findRecord<ObjCCategoryRecord>(std::make_pair(ClassToExtend, Category),
98                                         Categories);
99 }
100 
101 ObjCIVarRecord *ObjCContainerRecord::findObjCIVar(StringRef IVar) const {
102   return findRecord<ObjCIVarRecord>(IVar, IVars);
103 }
104 
105 ObjCIVarRecord *RecordsSlice::findObjCIVar(bool IsScopedName,
106                                            StringRef Name) const {
107   // If scoped name, the name of the container is known.
108   if (IsScopedName) {
109     // IVar does not exist if there is not a container assigned to it.
110     auto *Container = findContainer(/*IsIVar=*/true, Name);
111     if (!Container)
112       return nullptr;
113 
114     StringRef IVar = Name.substr(Name.find_first_of('.') + 1);
115     return Container->findObjCIVar(IVar);
116   }
117 
118   // Otherwise traverse through containers and attempt to find IVar.
119   auto getIVar = [Name](auto &Records) -> ObjCIVarRecord * {
120     for (const auto &[_, Container] : Records) {
121       if (auto *IVarR = Container->findObjCIVar(Name))
122         return IVarR;
123     }
124     return nullptr;
125   };
126 
127   if (auto *IVarRecord = getIVar(Classes))
128     return IVarRecord;
129 
130   return getIVar(Categories);
131 }
132 
133 GlobalRecord *RecordsSlice::addGlobal(StringRef Name, RecordLinkage Linkage,
134                                       GlobalRecord::Kind GV,
135                                       SymbolFlags Flags) {
136   if (GV == GlobalRecord::Kind::Function)
137     Flags |= SymbolFlags::Text;
138   else if (GV == GlobalRecord::Kind::Variable)
139     Flags |= SymbolFlags::Data;
140 
141   Name = copyString(Name);
142   auto Result = Globals.insert({Name, nullptr});
143   if (Result.second)
144     Result.first->second =
145         std::make_unique<GlobalRecord>(Name, Linkage, Flags, GV);
146   else {
147     updateLinkage(Result.first->second.get(), Linkage);
148     updateFlags(Result.first->second.get(), Flags);
149   }
150   return Result.first->second.get();
151 }
152 
153 ObjCInterfaceRecord *RecordsSlice::addObjCInterface(StringRef Name,
154                                                     RecordLinkage Linkage,
155                                                     bool HasEHType) {
156   Name = copyString(Name);
157   auto Result = Classes.insert({Name, nullptr});
158   if (Result.second) {
159     Result.first->second =
160         std::make_unique<ObjCInterfaceRecord>(Name, Linkage, HasEHType);
161   } else {
162     // ObjC classes represent multiple symbols that could have competing
163     // linkages, in those cases assign the largest one.
164     if (Linkage >= RecordLinkage::Rexported)
165       updateLinkage(Result.first->second.get(), Linkage);
166   }
167 
168   return Result.first->second.get();
169 }
170 SymbolFlags Record::mergeFlags(SymbolFlags Flags, RecordLinkage Linkage) {
171   // Add Linkage properties into Flags.
172   switch (Linkage) {
173   case RecordLinkage::Rexported:
174     Flags |= SymbolFlags::Rexported;
175     return Flags;
176   case RecordLinkage::Undefined:
177     Flags |= SymbolFlags::Undefined;
178     return Flags;
179   default:
180     return Flags;
181   }
182 }
183 
184 bool ObjCInterfaceRecord::addObjCCategory(ObjCCategoryRecord *Record) {
185   auto Result = Categories.insert({Name, Record});
186   return Result.second;
187 }
188 
189 ObjCCategoryRecord *RecordsSlice::addObjCCategory(StringRef ClassToExtend,
190                                                   StringRef Category) {
191   Category = copyString(Category);
192 
193   // Add owning record first into record slice.
194   auto Result =
195       Categories.insert({std::make_pair(ClassToExtend, Category), nullptr});
196   if (Result.second)
197     Result.first->second =
198         std::make_unique<ObjCCategoryRecord>(ClassToExtend, Category);
199 
200   // Then add reference to it in in the class.
201   if (auto *ObjCClass = findObjCInterface(ClassToExtend))
202     ObjCClass->addObjCCategory(Result.first->second.get());
203 
204   return Result.first->second.get();
205 }
206 
207 std::vector<ObjCIVarRecord *> ObjCContainerRecord::getObjCIVars() const {
208   std::vector<ObjCIVarRecord *> Records;
209   llvm::for_each(IVars,
210                  [&](auto &Record) { Records.push_back(Record.second.get()); });
211   return Records;
212 }
213 
214 std::vector<ObjCCategoryRecord *>
215 ObjCInterfaceRecord::getObjCCategories() const {
216   std::vector<ObjCCategoryRecord *> Records;
217   llvm::for_each(Categories,
218                  [&](auto &Record) { Records.push_back(Record.second); });
219   return Records;
220 }
221 
222 ObjCIVarRecord *ObjCContainerRecord::addObjCIVar(StringRef IVar,
223                                                  RecordLinkage Linkage) {
224   auto Result = IVars.insert({IVar, nullptr});
225   if (Result.second)
226     Result.first->second = std::make_unique<ObjCIVarRecord>(IVar, Linkage);
227   return Result.first->second.get();
228 }
229 
230 ObjCIVarRecord *RecordsSlice::addObjCIVar(ObjCContainerRecord *Container,
231                                           StringRef Name,
232                                           RecordLinkage Linkage) {
233   Name = copyString(Name);
234   ObjCIVarRecord *Record = Container->addObjCIVar(Name, Linkage);
235   updateLinkage(Record, Linkage);
236   return Record;
237 }
238 
239 StringRef RecordsSlice::copyString(StringRef String) {
240   if (String.empty())
241     return {};
242 
243   if (StringAllocator.identifyObject(String.data()))
244     return String;
245 
246   void *Ptr = StringAllocator.Allocate(String.size(), 1);
247   memcpy(Ptr, String.data(), String.size());
248   return StringRef(reinterpret_cast<const char *>(Ptr), String.size());
249 }
250 
251 RecordsSlice::BinaryAttrs &RecordsSlice::getBinaryAttrs() {
252   if (!hasBinaryAttrs())
253     BA = std::make_unique<BinaryAttrs>();
254   return *BA;
255 }
256 
257 void RecordsSlice::visit(RecordVisitor &V) const {
258   for (auto &G : Globals)
259     V.visitGlobal(*G.second);
260   for (auto &C : Classes)
261     V.visitObjCInterface(*C.second);
262   for (auto &Cat : Categories)
263     V.visitObjCCategory(*Cat.second);
264 }
265 
266 static std::unique_ptr<InterfaceFile>
267 createInterfaceFile(const Records &Slices, StringRef InstallName) {
268   // Pickup symbols first.
269   auto Symbols = std::make_unique<SymbolSet>();
270   for (auto &S : Slices) {
271     if (S->empty())
272       continue;
273     auto &BA = S->getBinaryAttrs();
274     if (BA.InstallName != InstallName)
275       continue;
276 
277     SymbolConverter Converter(Symbols.get(), S->getTarget(),
278                               !BA.TwoLevelNamespace);
279     S->visit(Converter);
280   }
281 
282   auto File = std::make_unique<InterfaceFile>(std::move(Symbols));
283   File->setInstallName(InstallName);
284   // Assign other attributes.
285   for (auto &S : Slices) {
286     if (S->empty())
287       continue;
288     auto &BA = S->getBinaryAttrs();
289     if (BA.InstallName != InstallName)
290       continue;
291     const Target &Targ = S->getTarget();
292     File->addTarget(Targ);
293     if (File->getFileType() == FileType::Invalid)
294       File->setFileType(BA.File);
295     if (BA.AppExtensionSafe && !File->isApplicationExtensionSafe())
296       File->setApplicationExtensionSafe();
297     if (BA.TwoLevelNamespace && !File->isTwoLevelNamespace())
298       File->setTwoLevelNamespace();
299     if (BA.OSLibNotForSharedCache && !File->isOSLibNotForSharedCache())
300       File->setOSLibNotForSharedCache();
301     if (File->getCurrentVersion().empty())
302       File->setCurrentVersion(BA.CurrentVersion);
303     if (File->getCompatibilityVersion().empty())
304       File->setCompatibilityVersion(BA.CompatVersion);
305     if (File->getSwiftABIVersion() == 0)
306       File->setSwiftABIVersion(BA.SwiftABI);
307     if (File->getPath().empty())
308       File->setPath(BA.Path);
309     if (!BA.ParentUmbrella.empty())
310       File->addParentUmbrella(Targ, BA.ParentUmbrella);
311     for (const auto &Client : BA.AllowableClients)
312       File->addAllowableClient(Client, Targ);
313     for (const auto &Lib : BA.RexportedLibraries)
314       File->addReexportedLibrary(Lib, Targ);
315   }
316 
317   return File;
318 }
319 
320 std::unique_ptr<InterfaceFile>
321 llvm::MachO::convertToInterfaceFile(const Records &Slices) {
322   std::unique_ptr<InterfaceFile> File;
323   if (Slices.empty())
324     return File;
325 
326   SetVector<StringRef> InstallNames;
327   for (auto &S : Slices) {
328     auto Name = S->getBinaryAttrs().InstallName;
329     if (Name.empty())
330       continue;
331     InstallNames.insert(Name);
332   }
333 
334   File = createInterfaceFile(Slices, *InstallNames.begin());
335   for (StringRef IN : llvm::drop_begin(InstallNames))
336     File->addDocument(createInterfaceFile(Slices, IN));
337 
338   return File;
339 }
340