xref: /openbsd-src/gnu/llvm/clang/lib/Basic/ObjCRuntime.cpp (revision e5dd70708596ae51455a0ffa086a00c5b29f8583)
1*e5dd7070Spatrick //===- ObjCRuntime.cpp - Objective-C Runtime Handling ---------------------===//
2*e5dd7070Spatrick //
3*e5dd7070Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*e5dd7070Spatrick // See https://llvm.org/LICENSE.txt for license information.
5*e5dd7070Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*e5dd7070Spatrick //
7*e5dd7070Spatrick //===----------------------------------------------------------------------===//
8*e5dd7070Spatrick //
9*e5dd7070Spatrick // This file implements the ObjCRuntime class, which represents the
10*e5dd7070Spatrick // target Objective-C runtime.
11*e5dd7070Spatrick //
12*e5dd7070Spatrick //===----------------------------------------------------------------------===//
13*e5dd7070Spatrick 
14*e5dd7070Spatrick #include "clang/Basic/ObjCRuntime.h"
15*e5dd7070Spatrick #include "llvm/ADT/StringRef.h"
16*e5dd7070Spatrick #include "llvm/Support/VersionTuple.h"
17*e5dd7070Spatrick #include "llvm/Support/raw_ostream.h"
18*e5dd7070Spatrick #include <cstddef>
19*e5dd7070Spatrick #include <string>
20*e5dd7070Spatrick 
21*e5dd7070Spatrick using namespace clang;
22*e5dd7070Spatrick 
getAsString() const23*e5dd7070Spatrick std::string ObjCRuntime::getAsString() const {
24*e5dd7070Spatrick   std::string Result;
25*e5dd7070Spatrick   {
26*e5dd7070Spatrick     llvm::raw_string_ostream Out(Result);
27*e5dd7070Spatrick     Out << *this;
28*e5dd7070Spatrick   }
29*e5dd7070Spatrick   return Result;
30*e5dd7070Spatrick }
31*e5dd7070Spatrick 
operator <<(raw_ostream & out,const ObjCRuntime & value)32*e5dd7070Spatrick raw_ostream &clang::operator<<(raw_ostream &out, const ObjCRuntime &value) {
33*e5dd7070Spatrick   switch (value.getKind()) {
34*e5dd7070Spatrick   case ObjCRuntime::MacOSX: out << "macosx"; break;
35*e5dd7070Spatrick   case ObjCRuntime::FragileMacOSX: out << "macosx-fragile"; break;
36*e5dd7070Spatrick   case ObjCRuntime::iOS: out << "ios"; break;
37*e5dd7070Spatrick   case ObjCRuntime::WatchOS: out << "watchos"; break;
38*e5dd7070Spatrick   case ObjCRuntime::GNUstep: out << "gnustep"; break;
39*e5dd7070Spatrick   case ObjCRuntime::GCC: out << "gcc"; break;
40*e5dd7070Spatrick   case ObjCRuntime::ObjFW: out << "objfw"; break;
41*e5dd7070Spatrick   }
42*e5dd7070Spatrick   if (value.getVersion() > VersionTuple(0)) {
43*e5dd7070Spatrick     out << '-' << value.getVersion();
44*e5dd7070Spatrick   }
45*e5dd7070Spatrick   return out;
46*e5dd7070Spatrick }
47*e5dd7070Spatrick 
tryParse(StringRef input)48*e5dd7070Spatrick bool ObjCRuntime::tryParse(StringRef input) {
49*e5dd7070Spatrick   // Look for the last dash.
50*e5dd7070Spatrick   std::size_t dash = input.rfind('-');
51*e5dd7070Spatrick 
52*e5dd7070Spatrick   // We permit dashes in the runtime name, and we also permit the
53*e5dd7070Spatrick   // version to be omitted, so if we see a dash not followed by a
54*e5dd7070Spatrick   // digit then we need to ignore it.
55*e5dd7070Spatrick   if (dash != StringRef::npos && dash + 1 != input.size() &&
56*e5dd7070Spatrick       (input[dash+1] < '0' || input[dash+1] > '9')) {
57*e5dd7070Spatrick     dash = StringRef::npos;
58*e5dd7070Spatrick   }
59*e5dd7070Spatrick 
60*e5dd7070Spatrick   // Everything prior to that must be a valid string name.
61*e5dd7070Spatrick   Kind kind;
62*e5dd7070Spatrick   StringRef runtimeName = input.substr(0, dash);
63*e5dd7070Spatrick   Version = VersionTuple(0);
64*e5dd7070Spatrick   if (runtimeName == "macosx") {
65*e5dd7070Spatrick     kind = ObjCRuntime::MacOSX;
66*e5dd7070Spatrick   } else if (runtimeName == "macosx-fragile") {
67*e5dd7070Spatrick     kind = ObjCRuntime::FragileMacOSX;
68*e5dd7070Spatrick   } else if (runtimeName == "ios") {
69*e5dd7070Spatrick     kind = ObjCRuntime::iOS;
70*e5dd7070Spatrick   } else if (runtimeName == "watchos") {
71*e5dd7070Spatrick     kind = ObjCRuntime::WatchOS;
72*e5dd7070Spatrick   } else if (runtimeName == "gnustep") {
73*e5dd7070Spatrick     // If no version is specified then default to the most recent one that we
74*e5dd7070Spatrick     // know about.
75*e5dd7070Spatrick     Version = VersionTuple(1, 6);
76*e5dd7070Spatrick     kind = ObjCRuntime::GNUstep;
77*e5dd7070Spatrick   } else if (runtimeName == "gcc") {
78*e5dd7070Spatrick     kind = ObjCRuntime::GCC;
79*e5dd7070Spatrick   } else if (runtimeName == "objfw") {
80*e5dd7070Spatrick     kind = ObjCRuntime::ObjFW;
81*e5dd7070Spatrick     Version = VersionTuple(0, 8);
82*e5dd7070Spatrick   } else {
83*e5dd7070Spatrick     return true;
84*e5dd7070Spatrick   }
85*e5dd7070Spatrick   TheKind = kind;
86*e5dd7070Spatrick 
87*e5dd7070Spatrick   if (dash != StringRef::npos) {
88*e5dd7070Spatrick     StringRef verString = input.substr(dash + 1);
89*e5dd7070Spatrick     if (Version.tryParse(verString))
90*e5dd7070Spatrick       return true;
91*e5dd7070Spatrick   }
92*e5dd7070Spatrick 
93*e5dd7070Spatrick   if (kind == ObjCRuntime::ObjFW && Version > VersionTuple(0, 8))
94*e5dd7070Spatrick     Version = VersionTuple(0, 8);
95*e5dd7070Spatrick 
96*e5dd7070Spatrick   return false;
97*e5dd7070Spatrick }
98