21#include "llvm/ADT/StringSet.h"
22#include "llvm/Support/SourceMgr.h"
23#include "llvm/Support/VersionTuple.h"
24#include "llvm/Support/YAMLTraits.h"
45enum class APIAvailability {
53struct BoundsSafetyNotes {
55 std::optional<unsigned> Level;
56 StringRef BoundsExpr =
"";
64 IO.enumCase(AA,
"counted_by",
66 IO.enumCase(AA,
"counted_by_or_null",
69 IO.enumCase(AA,
"sized_by_or_null",
79template <>
struct ScalarEnumerationTraits<APIAvailability> {
81 IO.enumCase(AA,
"none", APIAvailability::None);
82 IO.enumCase(AA,
"nonswift", APIAvailability::NonSwift);
83 IO.enumCase(AA,
"available", APIAvailability::Available);
90enum class MethodKind {
98template <>
struct ScalarEnumerationTraits<MethodKind> {
100 IO.enumCase(MK,
"Class", MethodKind::Class);
101 IO.enumCase(MK,
"Instance", MethodKind::Instance);
110 std::optional<bool> NoEscape =
false;
111 std::optional<bool> Lifetimebound =
false;
112 std::optional<NullabilityKind> Nullability;
113 std::optional<RetainCountConventionKind> RetainCountConvention;
114 std::optional<BoundsSafetyNotes> BoundsSafety;
118typedef std::vector<Param> ParamsSeq;
121LLVM_YAML_IS_SEQUENCE_VECTOR(Param)
147 IO.enumCase(RCCK,
"CFReturnsRetained",
149 IO.enumCase(RCCK,
"CFReturnsNotRetained",
151 IO.enumCase(RCCK,
"NSReturnsRetained",
153 IO.enumCase(RCCK,
"NSReturnsNotRetained",
158template <>
struct MappingTraits<Param> {
160 IO.mapRequired(
"Position", P.Position);
161 IO.mapOptional(
"Nullability", P.Nullability, std::nullopt);
162 IO.mapOptional(
"RetainCountConvention", P.RetainCountConvention);
163 IO.mapOptional(
"NoEscape", P.NoEscape);
164 IO.mapOptional(
"Lifetimebound", P.Lifetimebound);
165 IO.mapOptional(
"Type", P.Type, StringRef(
""));
166 IO.mapOptional(
"BoundsSafety", P.BoundsSafety);
170template <>
struct MappingTraits<BoundsSafetyNotes> {
171 static void mapping(IO &IO, BoundsSafetyNotes &BS) {
172 IO.mapRequired(
"Kind", BS.Kind);
173 IO.mapRequired(
"BoundedBy", BS.BoundsExpr);
174 IO.mapOptional(
"Level", BS.Level);
181typedef std::vector<NullabilityKind> NullabilitySeq;
183struct AvailabilityItem {
184 APIAvailability Mode = APIAvailability::Available;
189enum class FactoryAsInitKind {
202 NullabilitySeq Nullability;
203 std::optional<NullabilityKind> NullabilityOfRet;
204 std::optional<RetainCountConventionKind> RetainCountConvention;
205 AvailabilityItem Availability;
206 std::optional<bool> SwiftPrivate;
208 FactoryAsInitKind FactoryAsInit = FactoryAsInitKind::Infer;
209 bool DesignatedInit =
false;
210 bool Required =
false;
211 StringRef ResultType;
212 StringRef SwiftReturnOwnership;
216typedef std::vector<Method> MethodsSeq;
219LLVM_YAML_IS_SEQUENCE_VECTOR(Method)
223template <>
struct ScalarEnumerationTraits<FactoryAsInitKind> {
225 IO.enumCase(FIK,
"A", FactoryAsInitKind::Infer);
226 IO.enumCase(FIK,
"C", FactoryAsInitKind::AsClassMethod);
227 IO.enumCase(FIK,
"I", FactoryAsInitKind::AsInitializer);
231template <>
struct MappingTraits<Method> {
233 IO.mapRequired(
"Selector", M.Selector);
234 IO.mapRequired(
"MethodKind", M.Kind);
235 IO.mapOptional(
"Parameters", M.Params);
236 IO.mapOptional(
"Nullability", M.Nullability);
237 IO.mapOptional(
"NullabilityOfRet", M.NullabilityOfRet, std::nullopt);
238 IO.mapOptional(
"RetainCountConvention", M.RetainCountConvention);
239 IO.mapOptional(
"Availability", M.Availability.Mode,
240 APIAvailability::Available);
241 IO.mapOptional(
"AvailabilityMsg", M.Availability.Msg, StringRef(
""));
242 IO.mapOptional(
"SwiftPrivate", M.SwiftPrivate);
243 IO.mapOptional(
"SwiftName", M.SwiftName, StringRef(
""));
244 IO.mapOptional(
"FactoryAsInit", M.FactoryAsInit, FactoryAsInitKind::Infer);
245 IO.mapOptional(
"DesignatedInit", M.DesignatedInit,
false);
246 IO.mapOptional(
"Required", M.Required,
false);
247 IO.mapOptional(
"ResultType", M.ResultType, StringRef(
""));
248 IO.mapOptional(
"SwiftReturnOwnership", M.SwiftReturnOwnership,
259 std::optional<MethodKind> Kind;
260 std::optional<NullabilityKind> Nullability;
261 AvailabilityItem Availability;
262 std::optional<bool> SwiftPrivate;
264 std::optional<bool> SwiftImportAsAccessors;
269typedef std::vector<Property> PropertiesSeq;
272LLVM_YAML_IS_SEQUENCE_VECTOR(Property)
276template <>
struct MappingTraits<Property> {
278 IO.mapRequired(
"Name", P.Name);
279 IO.mapOptional(
"PropertyKind", P.Kind);
280 IO.mapOptional(
"Nullability", P.Nullability, std::nullopt);
281 IO.mapOptional(
"Availability", P.Availability.Mode,
282 APIAvailability::Available);
283 IO.mapOptional(
"AvailabilityMsg", P.Availability.Msg, StringRef(
""));
284 IO.mapOptional(
"SwiftPrivate", P.SwiftPrivate);
285 IO.mapOptional(
"SwiftName", P.SwiftName, StringRef(
""));
286 IO.mapOptional(
"SwiftImportAsAccessors", P.SwiftImportAsAccessors);
287 IO.mapOptional(
"Type", P.Type, StringRef(
""));
297 bool AuditedForNullability =
false;
298 AvailabilityItem Availability;
299 std::optional<bool> SwiftPrivate;
301 std::optional<StringRef> SwiftBridge;
302 std::optional<StringRef> NSErrorDomain;
303 std::optional<bool> SwiftImportAsNonGeneric;
304 std::optional<bool> SwiftObjCMembers;
305 std::optional<std::string> SwiftConformance;
307 PropertiesSeq Properties;
311typedef std::vector<Class> ClassesSeq;
314LLVM_YAML_IS_SEQUENCE_VECTOR(Class)
318template <>
struct MappingTraits<Class> {
320 IO.mapRequired(
"Name",
C.Name);
321 IO.mapOptional(
"AuditedForNullability",
C.AuditedForNullability,
false);
322 IO.mapOptional(
"Availability",
C.Availability.Mode,
323 APIAvailability::Available);
324 IO.mapOptional(
"AvailabilityMsg",
C.Availability.Msg, StringRef(
""));
325 IO.mapOptional(
"SwiftPrivate",
C.SwiftPrivate);
326 IO.mapOptional(
"SwiftName",
C.SwiftName, StringRef(
""));
327 IO.mapOptional(
"SwiftBridge",
C.SwiftBridge);
328 IO.mapOptional(
"NSErrorDomain",
C.NSErrorDomain);
329 IO.mapOptional(
"SwiftImportAsNonGeneric",
C.SwiftImportAsNonGeneric);
330 IO.mapOptional(
"SwiftObjCMembers",
C.SwiftObjCMembers);
331 IO.mapOptional(
"SwiftConformsTo",
C.SwiftConformance);
332 IO.mapOptional(
"Methods",
C.Methods);
333 IO.mapOptional(
"Properties",
C.Properties);
341typedef std::vector<StringRef> WhereParamsSeq;
343struct FunctionWhere {
344 std::optional<WhereParamsSeq> Parameters;
349 std::optional<FunctionWhere> Where;
351 NullabilitySeq Nullability;
352 std::optional<NullabilityKind> NullabilityOfRet;
353 std::optional<api_notes::RetainCountConventionKind> RetainCountConvention;
354 AvailabilityItem Availability;
355 std::optional<bool> SwiftPrivate;
358 StringRef ResultType;
359 StringRef SwiftReturnOwnership;
361 bool UnsafeBufferUsage =
false;
364typedef std::vector<Function> FunctionsSeq;
367LLVM_YAML_IS_SEQUENCE_VECTOR(Function)
371template <>
struct MappingTraits<FunctionWhere> {
372 static void mapping(IO &IO, FunctionWhere &W) {
373 IO.mapOptional(
"Parameters", W.Parameters);
377template <>
struct MappingTraits<Function> {
379 IO.mapRequired(
"Name", F.Name);
380 IO.mapOptional(
"Where", F.Where);
381 IO.mapOptional(
"Parameters", F.Params);
382 IO.mapOptional(
"Nullability", F.Nullability);
383 IO.mapOptional(
"NullabilityOfRet", F.NullabilityOfRet, std::nullopt);
384 IO.mapOptional(
"RetainCountConvention", F.RetainCountConvention);
385 IO.mapOptional(
"Availability", F.Availability.Mode,
386 APIAvailability::Available);
387 IO.mapOptional(
"AvailabilityMsg", F.Availability.Msg, StringRef(
""));
388 IO.mapOptional(
"SwiftPrivate", F.SwiftPrivate);
389 IO.mapOptional(
"SwiftName", F.SwiftName, StringRef(
""));
390 IO.mapOptional(
"ResultType", F.ResultType, StringRef(
""));
391 IO.mapOptional(
"SwiftReturnOwnership", F.SwiftReturnOwnership,
394 IO.mapOptional(
"UnsafeBufferUsage", F.UnsafeBufferUsage,
false);
401struct GlobalVariable {
403 std::optional<NullabilityKind> Nullability;
404 AvailabilityItem Availability;
405 std::optional<bool> SwiftPrivate;
411typedef std::vector<GlobalVariable> GlobalVariablesSeq;
414LLVM_YAML_IS_SEQUENCE_VECTOR(GlobalVariable)
418template <>
struct MappingTraits<GlobalVariable> {
419 static void mapping(IO &IO, GlobalVariable &GV) {
420 IO.mapRequired(
"Name", GV.Name);
421 IO.mapOptional(
"Nullability", GV.Nullability, std::nullopt);
422 IO.mapOptional(
"Availability", GV.Availability.Mode,
423 APIAvailability::Available);
424 IO.mapOptional(
"AvailabilityMsg", GV.Availability.Msg, StringRef(
""));
425 IO.mapOptional(
"SwiftPrivate", GV.SwiftPrivate);
426 IO.mapOptional(
"SwiftName", GV.SwiftName, StringRef(
""));
427 IO.mapOptional(
"Type", GV.Type, StringRef(
""));
437 AvailabilityItem Availability;
438 std::optional<bool> SwiftPrivate;
443typedef std::vector<EnumConstant> EnumConstantsSeq;
446LLVM_YAML_IS_SEQUENCE_VECTOR(EnumConstant)
450template <>
struct MappingTraits<EnumConstant> {
451 static void mapping(IO &IO, EnumConstant &EC) {
452 IO.mapRequired(
"Name", EC.Name);
453 IO.mapOptional(
"Availability", EC.Availability.Mode,
454 APIAvailability::Available);
455 IO.mapOptional(
"AvailabilityMsg", EC.Availability.Msg, StringRef(
""));
456 IO.mapOptional(
"SwiftPrivate", EC.SwiftPrivate);
457 IO.mapOptional(
"SwiftName", EC.SwiftName, StringRef(
""));
466enum class EnumConvenienceAliasKind {
480template <>
struct ScalarEnumerationTraits<EnumConvenienceAliasKind> {
482 IO.enumCase(ECAK,
"none", EnumConvenienceAliasKind::None);
483 IO.enumCase(ECAK,
"CFEnum", EnumConvenienceAliasKind::CFEnum);
484 IO.enumCase(ECAK,
"NSEnum", EnumConvenienceAliasKind::CFEnum);
485 IO.enumCase(ECAK,
"CFOptions", EnumConvenienceAliasKind::CFOptions);
486 IO.enumCase(ECAK,
"NSOptions", EnumConvenienceAliasKind::CFOptions);
487 IO.enumCase(ECAK,
"CFClosedEnum", EnumConvenienceAliasKind::CFClosedEnum);
488 IO.enumCase(ECAK,
"NSClosedEnum", EnumConvenienceAliasKind::CFClosedEnum);
497 std::optional<NullabilityKind> Nullability;
498 AvailabilityItem Availability;
499 std::optional<bool> SwiftPrivate;
505typedef std::vector<Field> FieldsSeq;
508LLVM_YAML_IS_SEQUENCE_VECTOR(Field)
512template <>
struct MappingTraits<Field> {
514 IO.mapRequired(
"Name", F.Name);
515 IO.mapOptional(
"Nullability", F.Nullability, std::nullopt);
516 IO.mapOptional(
"Availability", F.Availability.Mode,
517 APIAvailability::Available);
518 IO.mapOptional(
"AvailabilityMsg", F.Availability.Msg, StringRef(
""));
519 IO.mapOptional(
"SwiftPrivate", F.SwiftPrivate);
520 IO.mapOptional(
"SwiftName", F.SwiftName, StringRef(
""));
521 IO.mapOptional(
"Type", F.Type, StringRef(
""));
530typedef std::vector<Tag> TagsSeq;
534 AvailabilityItem Availability;
536 std::optional<bool> SwiftPrivate;
537 std::optional<StringRef> SwiftBridge;
538 std::optional<StringRef> NSErrorDomain;
539 std::optional<std::string> SwiftImportAs;
540 std::optional<std::string> SwiftRetainOp;
541 std::optional<std::string> SwiftReleaseOp;
542 std::optional<std::string> SwiftDestroyOp;
543 std::optional<std::string> SwiftDefaultOwnership;
544 std::optional<std::string> SwiftConformance;
545 std::optional<EnumExtensibilityKind> EnumExtensibility;
546 std::optional<bool> FlagEnum;
547 std::optional<EnumConvenienceAliasKind> EnumConvenienceKind;
548 std::optional<bool> SwiftCopyable;
549 std::optional<bool> SwiftEscapable;
551 FunctionsSeq Methods;
560LLVM_YAML_IS_SEQUENCE_VECTOR(Tag)
572template <>
struct MappingTraits<Tag> {
574 IO.mapRequired(
"Name", T.Name);
575 IO.mapOptional(
"Availability", T.Availability.Mode,
576 APIAvailability::Available);
577 IO.mapOptional(
"AvailabilityMsg", T.Availability.Msg, StringRef(
""));
578 IO.mapOptional(
"SwiftPrivate", T.SwiftPrivate);
579 IO.mapOptional(
"SwiftName", T.SwiftName, StringRef(
""));
580 IO.mapOptional(
"SwiftBridge", T.SwiftBridge);
581 IO.mapOptional(
"NSErrorDomain", T.NSErrorDomain);
582 IO.mapOptional(
"SwiftImportAs", T.SwiftImportAs);
583 IO.mapOptional(
"SwiftReleaseOp", T.SwiftReleaseOp);
584 IO.mapOptional(
"SwiftRetainOp", T.SwiftRetainOp);
585 IO.mapOptional(
"SwiftDestroyOp", T.SwiftDestroyOp);
586 IO.mapOptional(
"SwiftDefaultOwnership", T.SwiftDefaultOwnership);
587 IO.mapOptional(
"SwiftConformsTo", T.SwiftConformance);
588 IO.mapOptional(
"EnumExtensibility", T.EnumExtensibility);
589 IO.mapOptional(
"FlagEnum", T.FlagEnum);
590 IO.mapOptional(
"EnumKind", T.EnumConvenienceKind);
591 IO.mapOptional(
"SwiftCopyable", T.SwiftCopyable);
592 IO.mapOptional(
"SwiftEscapable", T.SwiftEscapable);
593 IO.mapOptional(
"Methods", T.Methods);
594 IO.mapOptional(
"Fields", T.Fields);
595 IO.mapOptional(
"Tags", T.Tags);
605 AvailabilityItem Availability;
607 std::optional<bool> SwiftPrivate;
608 std::optional<StringRef> SwiftBridge;
609 std::optional<StringRef> NSErrorDomain;
610 std::optional<SwiftNewTypeKind> SwiftType;
611 std::optional<std::string> SwiftConformance;
615typedef std::vector<Typedef> TypedefsSeq;
618LLVM_YAML_IS_SEQUENCE_VECTOR(Typedef)
630template <>
struct MappingTraits<Typedef> {
632 IO.mapRequired(
"Name", T.Name);
633 IO.mapOptional(
"Availability", T.Availability.Mode,
634 APIAvailability::Available);
635 IO.mapOptional(
"AvailabilityMsg", T.Availability.Msg, StringRef(
""));
636 IO.mapOptional(
"SwiftPrivate", T.SwiftPrivate);
637 IO.mapOptional(
"SwiftName", T.SwiftName, StringRef(
""));
638 IO.mapOptional(
"SwiftBridge", T.SwiftBridge);
639 IO.mapOptional(
"NSErrorDomain", T.NSErrorDomain);
640 IO.mapOptional(
"SwiftWrapper", T.SwiftType);
641 IO.mapOptional(
"SwiftConformsTo", T.SwiftConformance);
649typedef std::vector<Namespace> NamespacesSeq;
651struct TopLevelItems {
653 ClassesSeq Protocols;
654 FunctionsSeq Functions;
655 GlobalVariablesSeq Globals;
656 EnumConstantsSeq EnumConstants;
658 TypedefsSeq Typedefs;
659 NamespacesSeq Namespaces;
666 IO.mapOptional(
"Classes", TLI.Classes);
667 IO.mapOptional(
"Protocols", TLI.Protocols);
668 IO.mapOptional(
"Functions", TLI.Functions);
669 IO.mapOptional(
"Globals", TLI.Globals);
670 IO.mapOptional(
"Enumerators", TLI.EnumConstants);
671 IO.mapOptional(
"Tags", TLI.Tags);
672 IO.mapOptional(
"Typedefs", TLI.Typedefs);
673 IO.mapOptional(
"Namespaces", TLI.Namespaces);
681 AvailabilityItem Availability;
683 std::optional<bool> SwiftPrivate;
689LLVM_YAML_IS_SEQUENCE_VECTOR(Namespace)
693template <>
struct MappingTraits<Namespace> {
695 IO.mapRequired(
"Name", T.Name);
696 IO.mapOptional(
"Availability", T.Availability.Mode,
697 APIAvailability::Available);
698 IO.mapOptional(
"AvailabilityMsg", T.Availability.Msg, StringRef(
""));
699 IO.mapOptional(
"SwiftPrivate", T.SwiftPrivate);
700 IO.mapOptional(
"SwiftName", T.SwiftName, StringRef(
""));
709 VersionTuple Version;
713typedef std::vector<Versioned> VersionedSeq;
716LLVM_YAML_IS_SEQUENCE_VECTOR(Versioned)
720template <>
struct MappingTraits<Versioned> {
722 IO.mapRequired(
"Version",
V.Version);
732 AvailabilityItem Availability;
733 TopLevelItems TopLevel;
734 VersionedSeq SwiftVersions;
736 std::optional<bool> SwiftInferImportAsMember;
738#if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
739 LLVM_DUMP_METHOD
void dump() ;
746template <>
struct MappingTraits<
Module> {
748 IO.mapRequired(
"Name", M.
Name);
749 IO.mapOptional(
"Availability", M.Availability.Mode,
750 APIAvailability::Available);
751 IO.mapOptional(
"AvailabilityMsg", M.Availability.Msg, StringRef(
""));
752 IO.mapOptional(
"SwiftInferImportAsMember", M.SwiftInferImportAsMember);
754 IO.mapOptional(
"SwiftVersions", M.SwiftVersions);
760#if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
762 llvm::yaml::Output OS(llvm::errs());
768bool parseAPINotes(StringRef YI,
Module &M, llvm::SourceMgr::DiagHandlerTy
Diag,
770 llvm::yaml::Input IS(YI,
nullptr,
Diag, DiagContext);
772 return static_cast<bool>(IS.error());
777 llvm::raw_ostream &OS) {
779 if (parseAPINotes(YI, M,
nullptr,
nullptr))
782 llvm::yaml::Output YOS(OS);
793 APINotesWriter Writer;
794 llvm::raw_ostream &
OS;
795 llvm::SourceMgr::DiagHandlerTy DiagHandler;
796 void *DiagHandlerCtxt;
800 bool emitError(llvm::Twine Message) {
802 llvm::SMDiagnostic(
"", llvm::SourceMgr::DK_Error,
Message.str()),
804 ErrorOccurred =
true;
809 YAMLConverter(
const Module &TheModule,
const FileEntry *SourceFile,
810 llvm::raw_ostream &OS,
811 llvm::SourceMgr::DiagHandlerTy DiagHandler,
812 void *DiagHandlerCtxt)
813 : M(TheModule), Writer(TheModule.Name, SourceFile),
OS(
OS),
814 DiagHandler(DiagHandler), DiagHandlerCtxt(DiagHandlerCtxt),
815 ErrorOccurred(
false) {}
817 void convertAvailability(
const AvailabilityItem &Availability,
818 CommonEntityInfo &CEI, llvm::StringRef APIName) {
820 CEI.
Unavailable = (Availability.Mode == APIAvailability::None);
825 if (!Availability.Msg.empty())
826 emitError(llvm::Twine(
"availability message for available API '") +
827 APIName +
"' will not be used");
831 void convertParams(
const ParamsSeq &Params, FunctionInfo &OutInfo,
832 std::optional<ParamInfo> &thisOrSelf) {
833 for (
const auto &P : Params) {
839 PI.
setType(std::string(P.Type));
841 if (P.BoundsSafety) {
842 BoundsSafetyInfo BSI;
844 if (P.BoundsSafety->Level)
849 if (
static_cast<int>(OutInfo.
Params.size()) <= P.Position)
850 OutInfo.
Params.resize(P.Position + 1);
851 if (P.Position == -1)
853 else if (P.Position >= 0)
854 OutInfo.
Params[P.Position] |= PI;
856 emitError(
"invalid parameter position " + llvm::itostr(P.Position));
860 void convertNullability(
const NullabilitySeq &Nullability,
861 std::optional<NullabilityKind> ReturnNullability,
862 FunctionInfo &OutInfo, llvm::StringRef APIName) {
864 emitError(llvm::Twine(
"nullability info for '") + APIName +
869 bool audited =
false;
870 unsigned int idx = 1;
871 for (
const auto &N : Nullability)
873 audited =
Nullability.size() > 0 || ReturnNullability;
876 ReturnNullability.value_or(NullabilityKind::NonNull));
884 template <
typename T>
885 void convertCommonEntity(
const T &Common, CommonEntityInfo &Info,
887 convertAvailability(Common.Availability, Info, APIName);
889 if (Common.SafetyKind != SwiftSafetyKind::None)
891 Info.
SwiftName = std::string(Common.SwiftName);
895 template <
typename T>
896 void convertCommonType(
const T &Common, CommonTypeInfo &Info,
898 convertCommonEntity(Common, Info, APIName);
899 if (Common.SwiftBridge)
902 if (
auto conformance = Common.SwiftConformance)
907 void convertMethod(
const Method &M, ContextID ClassID, StringRef ClassName,
908 VersionTuple SwiftVersion) {
910 convertCommonEntity(M, MI, M.Selector);
913 bool takesArguments = M.Selector.ends_with(
":");
916 llvm::SmallVector<StringRef, 4> Args;
917 M.Selector.split(Args,
":", -1,
false);
918 if (!takesArguments && Args.size() > 1) {
919 emitError(
"selector '" + M.Selector +
"' is missing a ':' at the end");
924 api_notes::ObjCSelectorRef Selector;
925 Selector.
NumArgs = !takesArguments ? 0 : Args.size();
931 if (M.FactoryAsInit != FactoryAsInitKind::Infer)
932 emitError(
"'FactoryAsInit' is no longer valid; use 'SwiftName' instead");
938 convertParams(M.Params, MI, MI.
Self);
941 convertNullability(M.Nullability, M.NullabilityOfRet, MI, M.Selector);
946 Writer.
addObjCMethod(ClassID, Selector, M.Kind == MethodKind::Instance, MI,
950 template <
typename T>
951 void convertVariable(
const T &Entity, VariableInfo &VI) {
952 convertAvailability(Entity.Availability, VI, Entity.Name);
954 VI.
SwiftName = std::string(Entity.SwiftName);
955 if (Entity.Nullability)
957 VI.
setType(std::string(Entity.Type));
960 void convertContext(std::optional<ContextID> ParentContextID,
const Class &
C,
964 convertCommonType(
C, CI,
C.Name);
966 if (
C.AuditedForNullability)
968 if (
C.SwiftImportAsNonGeneric)
970 if (
C.SwiftObjCMembers)
974 Writer.
addContext(ParentContextID,
C.Name, Kind, CI, SwiftVersion);
977 llvm::StringMap<std::pair<bool, bool>> KnownMethods;
978 for (
const auto &method :
C.Methods) {
982 : KnownMethods[method.Selector].second;
984 emitError(llvm::Twine(
"duplicate definition of method '") +
985 (IsInstanceMethod ?
"-" :
"+") +
"[" +
C.Name +
" " +
986 method.Selector +
"]'");
991 convertMethod(method, CtxID,
C.Name, SwiftVersion);
995 llvm::StringSet<> KnownInstanceProperties;
996 llvm::StringSet<> KnownClassProperties;
997 for (
const auto &
Property :
C.Properties) {
1000 !KnownInstanceProperties.insert(
Property.Name).second) {
1001 emitError(llvm::Twine(
"duplicate definition of instance property '") +
1007 !KnownClassProperties.insert(
Property.Name).second) {
1008 emitError(llvm::Twine(
"duplicate definition of class property '") +
1014 ObjCPropertyInfo PI;
1016 if (
Property.SwiftImportAsAccessors)
1022 *
Property.Kind == MethodKind::Instance, PI,
1031 void convertNamespaceContext(std::optional<ContextID> ParentContextID,
1033 VersionTuple SwiftVersion) {
1036 convertCommonEntity(TheNamespace, CI, TheNamespace.Name);
1039 Writer.
addContext(ParentContextID, TheNamespace.Name,
1040 ContextKind::Namespace, CI, SwiftVersion);
1042 convertTopLevelItems(Context(CtxID, ContextKind::Namespace),
1043 TheNamespace.Items, SwiftVersion);
1046 template <
typename FuncOrMethodInfo>
1049 FI.setSwiftPrivate(
Function.SwiftPrivate);
1050 if (
Function.SafetyKind != SwiftSafetyKind::None)
1051 FI.setSwiftSafety(
Function.SafetyKind);
1052 FI.SwiftName = std::string(
Function.SwiftName);
1053 std::optional<ParamInfo>
This;
1054 convertParams(
Function.Params, FI, This);
1055 if constexpr (std::is_same_v<FuncOrMethodInfo, CXXMethodInfo>)
1058 emitError(
"implicit instance parameter is only permitted on C++ and "
1059 "Objective-C methods");
1062 FI.ResultType = std::string(
Function.ResultType);
1063 FI.SwiftReturnOwnership = std::string(
Function.SwiftReturnOwnership);
1064 FI.setRetainCountConvention(
Function.RetainCountConvention);
1065 FI.UnsafeBufferUsage =
Function.UnsafeBufferUsage;
1068 void convertTagContext(std::optional<Context> ParentContext,
const Tag &T,
1069 VersionTuple SwiftVersion) {
1071 std::optional<ContextID> ParentContextID =
1072 ParentContext ? std::optional<ContextID>(ParentContext->id)
1074 convertCommonType(T, TI, T.Name);
1076 if ((T.SwiftRetainOp || T.SwiftReleaseOp) && !T.SwiftImportAs) {
1077 emitError(llvm::Twine(
"should declare SwiftImportAs to use "
1078 "SwiftRetainOp and SwiftReleaseOp (for ") +
1082 if (T.SwiftReleaseOp.has_value() != T.SwiftRetainOp.has_value()) {
1083 emitError(llvm::Twine(
"should declare both SwiftReleaseOp and "
1084 "SwiftRetainOp (for ") +
1089 if (T.SwiftImportAs)
1091 if (T.SwiftRetainOp)
1093 if (T.SwiftReleaseOp)
1095 if (T.SwiftDestroyOp)
1097 if (T.SwiftDefaultOwnership)
1100 if (T.SwiftCopyable)
1102 if (T.SwiftEscapable)
1105 if (T.EnumConvenienceKind) {
1106 if (T.EnumExtensibility) {
1108 llvm::Twine(
"cannot mix EnumKind and EnumExtensibility (for ") +
1113 emitError(llvm::Twine(
"cannot mix EnumKind and FlagEnum (for ") +
1117 switch (*T.EnumConvenienceKind) {
1118 case EnumConvenienceAliasKind::None:
1122 case EnumConvenienceAliasKind::CFEnum:
1126 case EnumConvenienceAliasKind::CFOptions:
1130 case EnumConvenienceAliasKind::CFClosedEnum:
1140 Writer.
addTag(ParentContext, T.Name, TI, SwiftVersion);
1143 auto TagCtxID = Writer.
addContext(ParentContextID, T.Name, ContextKind::Tag,
1145 Context TagCtx(TagCtxID, ContextKind::Tag);
1147 for (
const auto &Field : T.Fields) {
1149 convertVariable(Field, FI);
1153 for (
const auto &CXXMethod : T.Methods) {
1154 if (CXXMethod.Where) {
1155 emitError(
"'Where' is not supported by binary API notes yet");
1160 convertFunction(CXXMethod, MI);
1161 Writer.
addCXXMethod(TagCtxID, CXXMethod.Name, MI, SwiftVersion);
1165 for (
const auto &
Tag : T.Tags)
1166 convertTagContext(TagCtx,
Tag, SwiftVersion);
1169 void convertTopLevelItems(std::optional<Context> Ctx,
1170 const TopLevelItems &TLItems,
1171 VersionTuple SwiftVersion) {
1172 std::optional<ContextID> CtxID =
1173 Ctx ? std::optional(Ctx->id) : std::nullopt;
1176 llvm::StringSet<> KnownClasses;
1177 for (
const auto &
Class : TLItems.Classes) {
1179 if (!KnownClasses.insert(
Class.Name).second) {
1180 emitError(llvm::Twine(
"multiple definitions of class '") +
Class.Name +
1185 convertContext(CtxID,
Class, ContextKind::ObjCClass, SwiftVersion);
1189 llvm::StringSet<> KnownProtocols;
1190 for (
const auto &Protocol : TLItems.Protocols) {
1192 if (!KnownProtocols.insert(
Protocol.Name).second) {
1193 emitError(llvm::Twine(
"multiple definitions of protocol '") +
1198 convertContext(CtxID, Protocol, ContextKind::ObjCProtocol, SwiftVersion);
1202 llvm::StringSet<> KnownNamespaces;
1203 for (
const auto &
Namespace : TLItems.Namespaces) {
1205 if (!KnownNamespaces.insert(
Namespace.Name).second) {
1206 emitError(llvm::Twine(
"multiple definitions of namespace '") +
1211 convertNamespaceContext(CtxID,
Namespace, SwiftVersion);
1215 llvm::StringSet<> KnownGlobals;
1216 for (
const auto &
Global : TLItems.Globals) {
1218 if (!KnownGlobals.insert(
Global.Name).second) {
1219 emitError(llvm::Twine(
"multiple definitions of global variable '") +
1224 GlobalVariableInfo GVI;
1225 convertVariable(
Global, GVI);
1230 llvm::StringSet<> KnownFunctions;
1231 for (
const auto &
Function : TLItems.Functions) {
1233 emitError(
"'Where' is not supported by binary API notes yet");
1238 if (!KnownFunctions.insert(
Function.Name).second) {
1239 emitError(llvm::Twine(
"multiple definitions of global function '") +
1244 GlobalFunctionInfo GFI;
1250 llvm::StringSet<> KnownEnumConstants;
1251 for (
const auto &EnumConstant : TLItems.EnumConstants) {
1253 if (!KnownEnumConstants.insert(
EnumConstant.Name).second) {
1254 emitError(llvm::Twine(
"multiple definitions of enumerator '") +
1259 EnumConstantInfo ECI;
1267 llvm::StringSet<> KnownTags;
1268 for (
const auto &
Tag : TLItems.Tags) {
1270 if (!KnownTags.insert(
Tag.Name).second) {
1271 emitError(llvm::Twine(
"multiple definitions of tag '") +
Tag.Name +
1276 convertTagContext(Ctx,
Tag, SwiftVersion);
1280 llvm::StringSet<> KnownTypedefs;
1281 for (
const auto &
Typedef : TLItems.Typedefs) {
1283 if (!KnownTypedefs.insert(
Typedef.Name).second) {
1284 emitError(llvm::Twine(
"multiple definitions of typedef '") +
1297 bool convertModule() {
1299 convertTopLevelItems( std::nullopt, M.TopLevel,
1303 for (
const auto &Versioned : M.SwiftVersions)
1304 convertTopLevelItems( std::nullopt, Versioned.Items,
1310 return ErrorOccurred;
1316 llvm::raw_ostream &OS,
1317 llvm::SourceMgr::DiagHandlerTy DiagHandler,
1318 void *DiagHandlerCtxt) {
1319 YAMLConverter
C(M, SourceFile, OS, DiagHandler, DiagHandlerCtxt);
1320 return C.convertModule();
1325 Diag.print(
nullptr, llvm::errs());
1330 llvm::raw_ostream &OS,
1331 llvm::SourceMgr::DiagHandlerTy DiagHandler,
1332 void *DiagHandlerCtxt) {
1338 if (parseAPINotes(YAMLInput, TheModule, DiagHandler, DiagHandlerCtxt))
1341 return compile(TheModule, SourceFile, OS, DiagHandler, DiagHandlerCtxt);
static void printDiagnostic(const llvm::SMDiagnostic &Diag, void *Context)
Simple diagnostic handler that prints diagnostics to standard error.
static bool compile(const Module &M, const FileEntry *SourceFile, llvm::raw_ostream &OS, llvm::SourceMgr::DiagHandlerTy DiagHandler, void *DiagHandlerCtxt)
Forward-declares and imports various common LLVM datatypes that clang wants to use unqualified.
static DiagnosticBuilder Diag(DiagnosticsEngine *Diags, const LangOptions &Features, FullSourceLoc TokLoc, const char *TokBegin, const char *TokRangeBegin, const char *TokRangeEnd, unsigned DiagID)
Produce a diagnostic highlighting some portion of a literal.
Defines various enumerations that describe declaration and type specifiers.
Cached information about one file (either on disk or in the virtual file system).
Describes a module or submodule.
std::string Name
The name of this module.
void dump() const
Dump the contents of this module to the given output stream.
void addObjCMethod(ContextID CtxID, ObjCSelectorRef Selector, bool IsInstanceMethod, const ObjCMethodInfo &Info, llvm::VersionTuple SwiftVersion)
Add information about a specific Objective-C method.
void addEnumConstant(llvm::StringRef Name, const EnumConstantInfo &Info, llvm::VersionTuple SwiftVersion)
Add information about an enumerator.
ContextID addContext(std::optional< ContextID > ParentCtxID, llvm::StringRef Name, ContextKind Kind, const ContextInfo &Info, llvm::VersionTuple SwiftVersion)
Add information about a specific Objective-C class or protocol or a C++ namespace.
void addGlobalFunction(std::optional< Context > Ctx, llvm::StringRef Name, const GlobalFunctionInfo &Info, llvm::VersionTuple SwiftVersion)
Add information about a global function.
void addObjCProperty(ContextID CtxID, llvm::StringRef Name, bool IsInstanceProperty, const ObjCPropertyInfo &Info, llvm::VersionTuple SwiftVersion)
Add information about a specific Objective-C property.
void addField(ContextID CtxID, llvm::StringRef Name, const FieldInfo &Info, llvm::VersionTuple SwiftVersion)
Add information about a specific C record field.
void addGlobalVariable(std::optional< Context > Ctx, llvm::StringRef Name, const GlobalVariableInfo &Info, llvm::VersionTuple SwiftVersion)
Add information about a global variable.
void addTypedef(std::optional< Context > Ctx, llvm::StringRef Name, const TypedefInfo &Info, llvm::VersionTuple SwiftVersion)
Add information about a typedef.
void writeToStream(llvm::raw_ostream &OS)
void addCXXMethod(ContextID CtxID, llvm::StringRef Name, const CXXMethodInfo &Info, llvm::VersionTuple SwiftVersion)
Add information about a specific C++ method.
void addTag(std::optional< Context > Ctx, llvm::StringRef Name, const TagInfo &Info, llvm::VersionTuple SwiftVersion)
Add information about a tag (struct/union/enum/C++ class).
std::string ExternalBounds
void setKindAudited(BoundsSafetyKind kind)
void setLevelAudited(unsigned level)
unsigned UnavailableInSwift
Whether this entity is marked unavailable in Swift.
unsigned Unavailable
Whether this entity is marked unavailable.
std::string SwiftName
Swift name of this entity.
void setSwiftSafety(SwiftSafetyKind Safety)
void setSwiftPrivate(std::optional< bool > Private)
std::string UnavailableMsg
Message to use when this entity is unavailable.
void setSwiftConformance(std::optional< std::string > conformance)
void setNSErrorDomain(const std::optional< std::string > &Domain)
void setSwiftBridge(std::optional< std::string > SwiftType)
void setDefaultNullability(NullabilityKind Kind)
Set the default nullability for properties and methods of this class.
void setSwiftObjCMembers(std::optional< bool > Value)
void setSwiftImportAsNonGeneric(std::optional< bool > Value)
std::string SwiftReturnOwnership
Ownership convention for return value.
void addTypeInfo(unsigned index, NullabilityKind kind)
void setRetainCountConvention(std::optional< RetainCountConventionKind > Value)
std::vector< ParamInfo > Params
The function parameters.
unsigned NumAdjustedNullable
Number of types whose nullability is encoded with the NullabilityPayload.
std::string ResultType
The result type of this function, as a C type.
static unsigned getMaxNullabilityIndex()
unsigned NullabilityAudited
Whether the signature has been audited with respect to nullability.
unsigned DesignatedInit
Whether this is a designated initializer of its class.
std::optional< ParamInfo > Self
unsigned RequiredInit
Whether this is a required initializer.
void setSwiftImportAsAccessors(std::optional< bool > Value)
std::optional< BoundsSafetyInfo > BoundsSafety
void setNoEscape(std::optional< bool > Value)
void setLifetimebound(std::optional< bool > Value)
void setRetainCountConvention(std::optional< RetainCountConventionKind > Value)
std::optional< std::string > SwiftReleaseOp
std::optional< std::string > SwiftRetainOp
std::optional< std::string > SwiftImportAs
std::optional< std::string > SwiftDefaultOwnership
std::optional< EnumExtensibilityKind > EnumExtensibility
void setSwiftCopyable(std::optional< bool > Value)
std::optional< std::string > SwiftDestroyOp
void setSwiftEscapable(std::optional< bool > Value)
void setFlagEnum(std::optional< bool > Value)
std::optional< SwiftNewTypeKind > SwiftWrapper
void setNullabilityAudited(NullabilityKind kind)
void setType(const std::string &type)
RetainCountConventionKind
bool compileAPINotes(llvm::StringRef YAMLInput, const FileEntry *SourceFile, llvm::raw_ostream &OS, llvm::SourceMgr::DiagHandlerTy DiagHandler=nullptr, void *DiagHandlerCtxt=nullptr)
Converts API notes from YAML format to binary format.
SwiftNewTypeKind
The kind of a swift_wrapper/swift_newtype.
EnumExtensibilityKind
The payload for an enum_extensibility attribute.
bool parseAndDumpAPINotes(llvm::StringRef YI, llvm::raw_ostream &OS)
Parses the APINotes YAML content and writes the representation back to the specified stream.
@ OS
Indicates that the tracking object is a descendant of a referenced-counted OSObject,...
bool This(InterpState &S, CodePtr OpPC)
The JSON file list parser is used to communicate input to InstallAPI.
NullabilityKind
Describes the nullability of a particular type.
@ Nullable
Values of this type can be null.
@ Unspecified
Whether values of this type can be null is (explicitly) unspecified.
@ NonNull
Values of this type can never be null.
@ Module
Module linkage, which indicates that the entity can be referred to from other translation units withi...
@ Property
The type of a property.
@ Class
The "class" keyword introduces the elaborated-type-specifier.
static void mapTopLevelItems(IO &IO, TopLevelItems &TLI)
Diagnostic wrappers for TextAPI types for error reporting.
llvm::ArrayRef< llvm::StringRef > Identifiers
static void mapping(IO &IO, BoundsSafetyNotes &BS)
static void mapping(IO &IO, Class &C)
static void mapping(IO &IO, EnumConstant &EC)
static void mapping(IO &IO, Field &F)
static void mapping(IO &IO, FunctionWhere &W)
static void mapping(IO &IO, Function &F)
static void mapping(IO &IO, GlobalVariable &GV)
static void mapping(IO &IO, Method &M)
static void mapping(IO &IO, Module &M)
static void mapping(IO &IO, Namespace &T)
static void mapping(IO &IO, Param &P)
static void mapping(IO &IO, Property &P)
static void mapping(IO &IO, Tag &T)
static void mapping(IO &IO, Typedef &T)
static void mapping(IO &IO, Versioned &V)
static void enumeration(IO &IO, APIAvailability &AA)
static void enumeration(IO &IO, BoundsSafetyInfo::BoundsSafetyKind &AA)
static void enumeration(IO &IO, EnumConvenienceAliasKind &ECAK)
static void enumeration(IO &IO, EnumExtensibilityKind &EEK)
static void enumeration(IO &IO, FactoryAsInitKind &FIK)
static void enumeration(IO &IO, MethodKind &MK)
static void enumeration(IO &IO, NullabilityKind &NK)
static void enumeration(IO &IO, RetainCountConventionKind &RCCK)
static void enumeration(IO &IO, SwiftNewTypeKind &SWK)
static void enumeration(IO &IO, SwiftSafetyKind &SK)