forked from OSchip/llvm-project
[ObjC][CodeGen] CodeGen support for @available.
CodeGens uses of @available into calls to the compiler-rt function __isOSVersionAtLeast. This commit is part of a feature that I proposed here: http://lists.llvm.org/pipermail/cfe-dev/2016-July/049851.html Differential revision: https://reviews.llvm.org/D27827 llvm-svn: 296015
This commit is contained in:
parent
ee2d77f6d6
commit
9c42a8d43e
|
@ -300,6 +300,24 @@ public:
|
|||
return V;
|
||||
}
|
||||
|
||||
Value *VisitObjCAvailabilityCheckExpr(ObjCAvailabilityCheckExpr *E) {
|
||||
VersionTuple Version = E->getVersion();
|
||||
|
||||
// If we're checking for a platform older than our minimum deployment
|
||||
// target, we can fold the check away.
|
||||
if (Version <= CGF.CGM.getTarget().getPlatformMinVersion())
|
||||
return llvm::ConstantInt::get(Builder.getInt1Ty(), 1);
|
||||
|
||||
Optional<unsigned> Min = Version.getMinor(), SMin = Version.getSubminor();
|
||||
llvm::Value *Args[] = {
|
||||
llvm::ConstantInt::get(CGF.CGM.Int32Ty, Version.getMajor()),
|
||||
llvm::ConstantInt::get(CGF.CGM.Int32Ty, Min ? *Min : 0),
|
||||
llvm::ConstantInt::get(CGF.CGM.Int32Ty, SMin ? *SMin : 0),
|
||||
};
|
||||
|
||||
return CGF.EmitBuiltinAvailable(Args);
|
||||
}
|
||||
|
||||
Value *VisitArraySubscriptExpr(ArraySubscriptExpr *E);
|
||||
Value *VisitShuffleVectorExpr(ShuffleVectorExpr *E);
|
||||
Value *VisitConvertVectorExpr(ConvertVectorExpr *E);
|
||||
|
|
|
@ -3399,5 +3399,21 @@ CodeGenFunction::EmitBlockCopyAndAutorelease(llvm::Value *Block, QualType Ty) {
|
|||
return Val;
|
||||
}
|
||||
|
||||
llvm::Value *
|
||||
CodeGenFunction::EmitBuiltinAvailable(ArrayRef<llvm::Value *> Args) {
|
||||
assert(Args.size() == 3 && "Expected 3 argument here!");
|
||||
|
||||
if (!CGM.IsOSVersionAtLeastFn) {
|
||||
llvm::FunctionType *FTy =
|
||||
llvm::FunctionType::get(Int32Ty, {Int32Ty, Int32Ty, Int32Ty}, false);
|
||||
CGM.IsOSVersionAtLeastFn =
|
||||
CGM.CreateRuntimeFunction(FTy, "__isOSVersionAtLeast");
|
||||
}
|
||||
|
||||
llvm::Value *CallRes =
|
||||
EmitNounwindRuntimeCall(CGM.IsOSVersionAtLeastFn, Args);
|
||||
|
||||
return Builder.CreateICmpNE(CallRes, llvm::Constant::getNullValue(Int32Ty));
|
||||
}
|
||||
|
||||
CGObjCRuntime::~CGObjCRuntime() {}
|
||||
|
|
|
@ -3169,6 +3169,8 @@ private:
|
|||
public:
|
||||
llvm::Value *EmitMSVCBuiltinExpr(MSVCIntrin BuiltinID, const CallExpr *E);
|
||||
|
||||
llvm::Value *EmitBuiltinAvailable(ArrayRef<llvm::Value *> Args);
|
||||
|
||||
llvm::Value *EmitObjCProtocolExpr(const ObjCProtocolExpr *E);
|
||||
llvm::Value *EmitObjCStringLiteral(const ObjCStringLiteral *E);
|
||||
llvm::Value *EmitObjCBoxedExpr(const ObjCBoxedExpr *E);
|
||||
|
|
|
@ -546,6 +546,10 @@ public:
|
|||
return *ObjCData;
|
||||
}
|
||||
|
||||
// Version checking function, used to implement ObjC's @available:
|
||||
// i32 @__isOSVersionAtLeast(i32, i32, i32)
|
||||
llvm::Constant *IsOSVersionAtLeastFn = nullptr;
|
||||
|
||||
InstrProfStats &getPGOStats() { return PGOStats; }
|
||||
llvm::IndexedInstrProfReader *getPGOReader() const { return PGOReader.get(); }
|
||||
|
||||
|
|
|
@ -0,0 +1,31 @@
|
|||
// RUN: %clang_cc1 -triple x86_64-apple-macosx10.11 -emit-llvm -o - %s | FileCheck %s
|
||||
|
||||
void use_at_available() {
|
||||
// CHECK: call i32 @__isOSVersionAtLeast(i32 10, i32 12, i32 0)
|
||||
// CHECK-NEXT: icmp ne
|
||||
if (__builtin_available(macos 10.12, *))
|
||||
;
|
||||
|
||||
// CHECK: call i32 @__isOSVersionAtLeast(i32 10, i32 12, i32 0)
|
||||
// CHECK-NEXT: icmp ne
|
||||
if (@available(macos 10.12, *))
|
||||
;
|
||||
|
||||
// CHECK: call i32 @__isOSVersionAtLeast(i32 10, i32 12, i32 42)
|
||||
// CHECK-NEXT: icmp ne
|
||||
if (__builtin_available(ios 10, macos 10.12.42, *))
|
||||
;
|
||||
|
||||
// CHECK-NOT: call i32 @__isOSVersionAtLeast
|
||||
// CHECK: br i1 true
|
||||
if (__builtin_available(ios 10, *))
|
||||
;
|
||||
|
||||
// This check should be folded: our deployment target is 10.11.
|
||||
// CHECK-NOT: call i32 @__isOSVersionAtLeast
|
||||
// CHECK: br i1 true
|
||||
if (__builtin_available(macos 10.11, *))
|
||||
;
|
||||
}
|
||||
|
||||
// CHECK: declare i32 @__isOSVersionAtLeast(i32, i32, i32)
|
Loading…
Reference in New Issue