mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-02 07:32:52 +00:00
It turns out that clang does use pointer-to-function types to
point to ARC-managed pointers sometimes. This fixes rdar://10551239. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@146577 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
5e46dcbb4b
commit
f9096e450b
@ -179,9 +179,13 @@ static bool IsPotentialUse(const Value *Op) {
|
||||
Arg->hasNestAttr() ||
|
||||
Arg->hasStructRetAttr())
|
||||
return false;
|
||||
// Only consider values with pointer types, and not function pointers.
|
||||
// Only consider values with pointer types.
|
||||
// It seemes intuitive to exclude function pointer types as well, since
|
||||
// functions are never reference-counted, however clang occasionally
|
||||
// bitcasts reference-counted pointers to function-pointer type
|
||||
// temporarily.
|
||||
PointerType *Ty = dyn_cast<PointerType>(Op->getType());
|
||||
if (!Ty || isa<FunctionType>(Ty->getElementType()))
|
||||
if (!Ty)
|
||||
return false;
|
||||
// Conservatively assume anything else is a potential use.
|
||||
return true;
|
||||
|
@ -1528,9 +1528,11 @@ define void @test52(i8** %zz, i8** %pp) {
|
||||
|
||||
; Like test52, but the pointer has function type, so it's assumed to
|
||||
; be not reference counted.
|
||||
; Oops. That's wrong. Clang sometimes uses function types gratuitously.
|
||||
; See rdar://10551239.
|
||||
|
||||
; CHECK: define void @test53(
|
||||
; CHECK-NOT: @objc_
|
||||
; CHECK: @objc_
|
||||
; CHECK: }
|
||||
define void @test53(void ()** %zz, i8** %pp) {
|
||||
%p = load i8** %pp
|
||||
|
31
test/Transforms/ObjCARC/pointer-types.ll
Normal file
31
test/Transforms/ObjCARC/pointer-types.ll
Normal file
@ -0,0 +1,31 @@
|
||||
; RUN: opt -objc-arc -S < %s | FileCheck %s
|
||||
|
||||
; Don't hoist @objc_release past a use of its pointer, even
|
||||
; if the use has function type, because clang uses function types
|
||||
; in dubious ways.
|
||||
; rdar://10551239
|
||||
|
||||
; CHECK: define void @test0(
|
||||
; CHECK: %otherBlock = phi void ()* [ %b1, %if.then ], [ null, %entry ]
|
||||
; CHECK-NEXT: call void @use_fptr(void ()* %otherBlock)
|
||||
; CHECK-NEXT: %tmp11 = bitcast void ()* %otherBlock to i8*
|
||||
; CHECK-NEXT: call void @objc_release(i8* %tmp11)
|
||||
|
||||
define void @test0(i1 %tobool, void ()* %b1) {
|
||||
entry:
|
||||
br i1 %tobool, label %if.end, label %if.then
|
||||
|
||||
if.then: ; preds = %entry
|
||||
br label %if.end
|
||||
|
||||
if.end: ; preds = %if.then, %entry
|
||||
%otherBlock = phi void ()* [ %b1, %if.then ], [ null, %entry ]
|
||||
call void @use_fptr(void ()* %otherBlock)
|
||||
%tmp11 = bitcast void ()* %otherBlock to i8*
|
||||
call void @objc_release(i8* %tmp11) nounwind
|
||||
ret void
|
||||
}
|
||||
|
||||
declare void @use_fptr(void ()*)
|
||||
declare void @objc_release(i8*)
|
||||
|
Loading…
Reference in New Issue
Block a user