LLDB mainline
XcodeSDK.cpp
Go to the documentation of this file.
1//===-- XcodeSDK.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
11
12#include "lldb/lldb-types.h"
13
14#include "llvm/TargetParser/Triple.h"
15
16#include <string>
17
18using namespace lldb;
19using namespace lldb_private;
20
21static llvm::StringRef GetName(XcodeSDK::Type type) {
22 switch (type) {
24 return "MacOSX";
26 return "iPhoneSimulator";
28 return "iPhoneOS";
30 return "AppleTVSimulator";
32 return "AppleTVOS";
34 return "WatchSimulator";
36 return "WatchOS";
38 return "XRSimulator";
39 case XcodeSDK::XROS:
40 return "XROS";
42 return "bridgeOS";
43 case XcodeSDK::Linux:
44 return "Linux";
46 return {};
47 }
48 llvm_unreachable("Unhandled sdk type!");
49}
50
51XcodeSDK::XcodeSDK(XcodeSDK::Info info) : m_name(GetName(info.type).str()) {
52 if (!m_name.empty()) {
53 if (!info.version.empty())
54 m_name += info.version.getAsString();
55 if (info.internal)
56 m_name += ".Internal";
57 m_name += ".sdk";
58 }
59}
60
61XcodeSDK &XcodeSDK::operator=(const XcodeSDK &other) = default;
62
63bool XcodeSDK::operator==(const XcodeSDK &other) const {
64 return m_name == other.m_name;
65}
66
67static XcodeSDK::Type ParseSDKName(llvm::StringRef &name) {
68 if (name.consume_front("MacOSX"))
69 return XcodeSDK::MacOSX;
70 if (name.consume_front("iPhoneSimulator"))
72 if (name.consume_front("iPhoneOS"))
73 return XcodeSDK::iPhoneOS;
74 if (name.consume_front("AppleTVSimulator"))
76 if (name.consume_front("AppleTVOS"))
78 if (name.consume_front("WatchSimulator"))
80 if (name.consume_front("WatchOS"))
81 return XcodeSDK::watchOS;
82 if (name.consume_front("XRSimulator"))
84 if (name.consume_front("XROS"))
85 return XcodeSDK::XROS;
86 if (name.consume_front("bridgeOS"))
87 return XcodeSDK::bridgeOS;
88 if (name.consume_front("Linux"))
89 return XcodeSDK::Linux;
90 static_assert(XcodeSDK::Linux == XcodeSDK::numSDKTypes - 1,
91 "New SDK type was added, update this list!");
92 return XcodeSDK::unknown;
93}
94
95static llvm::VersionTuple ParseSDKVersion(llvm::StringRef &name) {
96 unsigned i = 0;
97 while (i < name.size() && name[i] >= '0' && name[i] <= '9')
98 ++i;
99 if (i == name.size() || name[i++] != '.')
100 return {};
101 while (i < name.size() && name[i] >= '0' && name[i] <= '9')
102 ++i;
103 if (i == name.size() || name[i++] != '.')
104 return {};
105
106 llvm::VersionTuple version;
107 version.tryParse(name.slice(0, i - 1));
108 name = name.drop_front(i);
109 return version;
110}
111
112static bool ParseAppleInternalSDK(llvm::StringRef &name) {
113 return name.consume_front("Internal.") || name.consume_front(".Internal.");
114}
115
117 XcodeSDK::Info info;
118 llvm::StringRef input(m_name);
119 info.type = ParseSDKName(input);
120 info.version = ParseSDKVersion(input);
121 info.internal = ParseAppleInternalSDK(input);
122 return info;
123}
124
126 llvm::StringRef input(m_name);
127 ParseSDKName(input);
128 ParseSDKVersion(input);
129 return ParseAppleInternalSDK(input);
130}
131
132llvm::VersionTuple XcodeSDK::GetVersion() const {
133 llvm::StringRef input(m_name);
134 ParseSDKName(input);
135 return ParseSDKVersion(input);
136}
137
139 llvm::StringRef input(m_name);
140 return ParseSDKName(input);
141}
142
143llvm::StringRef XcodeSDK::GetString() const { return m_name; }
144
145bool XcodeSDK::Info::operator<(const Info &other) const {
146 return std::tie(type, version, internal) <
147 std::tie(other.type, other.version, other.internal);
148}
149
150bool XcodeSDK::Info::operator==(const Info &other) const {
151 return std::tie(type, version, internal) ==
152 std::tie(other.type, other.version, other.internal);
153}
154
155void XcodeSDK::Merge(const XcodeSDK &other) {
156 // The "bigger" SDK always wins.
157 auto l = Parse();
158 auto r = other.Parse();
159 if (l < r)
160 *this = other;
161 else {
162 // The Internal flag always wins.
163 if (llvm::StringRef(m_name).ends_with(".sdk"))
164 if (!l.internal && r.internal)
165 m_name =
166 m_name.substr(0, m_name.size() - 3) + std::string("Internal.sdk");
167 }
168}
169
171 std::string name;
172 switch (info.type) {
173 case MacOSX:
174 name = "macosx";
175 break;
176 case iPhoneSimulator:
177 name = "iphonesimulator";
178 break;
179 case iPhoneOS:
180 name = "iphoneos";
181 break;
182 case AppleTVSimulator:
183 name = "appletvsimulator";
184 break;
185 case AppleTVOS:
186 name = "appletvos";
187 break;
188 case WatchSimulator:
189 name = "watchsimulator";
190 break;
191 case watchOS:
192 name = "watchos";
193 break;
194 case XRSimulator:
195 name = "xrsimulator";
196 break;
197 case XROS:
198 name = "xros";
199 break;
200 case bridgeOS:
201 name = "bridgeos";
202 break;
203 case Linux:
204 name = "linux";
205 break;
206 case unknown:
207 return {};
208 }
209 if (!info.version.empty())
210 name += info.version.getAsString();
211 if (info.internal)
212 name += ".internal";
213 return name;
214}
215
217 llvm::VersionTuple version) {
218 switch (sdk_type) {
219 case Type::MacOSX:
220 return version >= llvm::VersionTuple(10, 10);
221 case Type::iPhoneOS:
223 case Type::AppleTVOS:
225 return version >= llvm::VersionTuple(8);
226 case Type::watchOS:
228 return version >= llvm::VersionTuple(6);
229 case Type::XROS:
231 return true;
232 default:
233 return false;
234 }
235
236 return false;
237}
238
240 XcodeSDK::Info info = Parse();
241 switch (info.type) {
242 case Type::MacOSX:
243 return info.version.empty() || info.version >= llvm::VersionTuple(10, 10);
244 case Type::iPhoneOS:
246 return info.version.empty() || info.version >= llvm::VersionTuple(8);
248 case Type::AppleTVOS:
249 return info.version.empty() || info.version >= llvm::VersionTuple(9);
251 case Type::watchOS:
252 return info.version.empty() || info.version >= llvm::VersionTuple(2);
253 case Type::XROS:
255 case Type::Linux:
256 return true;
257 default:
258 return false;
259 }
260}
261
262bool XcodeSDK::SDKSupportsBuiltinModules(const llvm::Triple &target_triple,
263 llvm::VersionTuple sdk_version) {
264 using namespace llvm;
265
266 switch (target_triple.getOS()) {
267 case Triple::OSType::MacOSX:
268 return sdk_version >= VersionTuple(15U);
269 case Triple::OSType::IOS:
270 return sdk_version >= VersionTuple(18U);
271 case Triple::OSType::TvOS:
272 return sdk_version >= VersionTuple(18U);
273 case Triple::OSType::WatchOS:
274 return sdk_version >= VersionTuple(11U);
275 case Triple::OSType::XROS:
276 return sdk_version >= VersionTuple(2U);
277 default:
278 // New SDKs support builtin modules from the start.
279 return true;
280 }
281}
282
284 const FileSpec &sdk_path) {
285 ConstString last_path_component = sdk_path.GetFilename();
286
287 if (!last_path_component)
288 return false;
289
290 XcodeSDK sdk(last_path_component.GetStringRef().str());
291 if (sdk.GetType() != desired_type)
292 return false;
293 return SDKSupportsModules(sdk.GetType(), sdk.GetVersion());
294}
295
297 using namespace llvm;
298 switch (triple.getOS()) {
299 case Triple::MacOSX:
300 case Triple::Darwin:
301 return XcodeSDK::MacOSX;
302 case Triple::IOS:
303 switch (triple.getEnvironment()) {
304 case Triple::MacABI:
305 return XcodeSDK::MacOSX;
306 case Triple::Simulator:
308 default:
309 return XcodeSDK::iPhoneOS;
310 }
311 case Triple::TvOS:
312 if (triple.getEnvironment() == Triple::Simulator)
314 return XcodeSDK::AppleTVOS;
315 case Triple::WatchOS:
316 if (triple.getEnvironment() == Triple::Simulator)
318 return XcodeSDK::watchOS;
319 case Triple::XROS:
320 if (triple.getEnvironment() == Triple::Simulator)
322 return XcodeSDK::XROS;
323 case Triple::Linux:
324 return XcodeSDK::Linux;
325 default:
326 return XcodeSDK::unknown;
327 }
328}
329
330std::string XcodeSDK::FindXcodeContentsDirectoryInPath(llvm::StringRef path) {
331 auto begin = llvm::sys::path::begin(path);
332 auto end = llvm::sys::path::end(path);
333
334 // Iterate over the path components until we find something that ends with
335 // .app. If the next component is Contents then we've found the Contents
336 // directory.
337 for (auto it = begin; it != end; ++it) {
338 if (it->ends_with(".app")) {
339 auto next = it;
340 if (++next != end && *next == "Contents") {
341 llvm::SmallString<128> buffer;
342 llvm::sys::path::append(buffer, begin, ++next,
343 llvm::sys::path::Style::posix);
344 return buffer.str().str();
345 }
346 }
347 }
348
349 return {};
350}
static llvm::VersionTuple ParseSDKVersion(llvm::StringRef &name)
Definition: XcodeSDK.cpp:95
static llvm::StringRef GetName(XcodeSDK::Type type)
Definition: XcodeSDK.cpp:21
static bool ParseAppleInternalSDK(llvm::StringRef &name)
Definition: XcodeSDK.cpp:112
static XcodeSDK::Type ParseSDKName(llvm::StringRef &name)
Definition: XcodeSDK.cpp:67
A uniqued constant string class.
Definition: ConstString.h:40
llvm::StringRef GetStringRef() const
Get the string value as a llvm::StringRef.
Definition: ConstString.h:197
A file utility class.
Definition: FileSpec.h:56
const ConstString & GetFilename() const
Filename string const get accessor.
Definition: FileSpec.h:240
An abstraction for Xcode-style SDKs that works like ArchSpec.
Definition: XcodeSDK.h:24
static constexpr int numSDKTypes
Definition: XcodeSDK.h:43
bool SupportsSwift() const
Whether this Xcode SDK supports Swift.
Definition: XcodeSDK.cpp:239
Type
Different types of Xcode SDKs.
Definition: XcodeSDK.h:29
static std::string FindXcodeContentsDirectoryInPath(llvm::StringRef path)
Definition: XcodeSDK.cpp:330
Type GetType() const
Definition: XcodeSDK.cpp:138
void Merge(const XcodeSDK &other)
The merge function follows a strict order to maintain monotonicity:
Definition: XcodeSDK.cpp:155
XcodeSDK & operator=(const XcodeSDK &other)
XcodeSDK()=default
Default constructor, constructs an empty string.
static bool SDKSupportsBuiltinModules(const llvm::Triple &target_triple, llvm::VersionTuple sdk_version)
Returns true if the SDK for the specified triple supports builtin modules in system headers.
Definition: XcodeSDK.cpp:262
std::string m_name
Definition: XcodeSDK.h:25
llvm::VersionTuple GetVersion() const
Definition: XcodeSDK.cpp:132
llvm::StringRef GetString() const
Definition: XcodeSDK.cpp:143
static std::string GetCanonicalName(Info info)
Return the canonical SDK name, such as "macosx" for the macOS SDK.
Definition: XcodeSDK.cpp:170
static XcodeSDK::Type GetSDKTypeForTriple(const llvm::Triple &triple)
Return the best-matching SDK type for a specific triple.
Definition: XcodeSDK.cpp:296
Info Parse() const
Return parsed SDK type and version number.
Definition: XcodeSDK.cpp:116
bool operator==(const XcodeSDK &other) const
Definition: XcodeSDK.cpp:63
bool IsAppleInternalSDK() const
Definition: XcodeSDK.cpp:125
static bool SDKSupportsModules(Type type, llvm::VersionTuple version)
Whether LLDB feels confident importing Clang modules from this SDK.
Definition: XcodeSDK.cpp:216
A class that represents a running process on the host machine.
Definition: SBAddress.h:15
Definition: Debugger.h:54
A parsed SDK directory name.
Definition: XcodeSDK.h:46
llvm::VersionTuple version
Definition: XcodeSDK.h:48
bool operator<(const Info &other) const
Definition: XcodeSDK.cpp:145
bool operator==(const Info &other) const
Definition: XcodeSDK.cpp:150