mirror of
https://github.com/Gericom/teak-llvm.git
synced 2025-06-26 06:48:51 -04:00

body of global block invoke functions. This commit fixes an infinite loop in IRGen that occurs when compiling the following code: void FUNC2() { static void (^const block1)(int) = ^(int a){ if (a--) block1(a); }; } This is how IRGen gets stuck in the infinite loop: 1. GenerateBlockFunction is called to emit the body of "block1". 2. GetAddrOfGlobalBlock is called to get the address of "block1". The function calls getAddrOfGlobalBlockIfEmitted to check whether the global block has been emitted. If it hasn't been emitted, it then tries to emit the body of the block function by calling GenerateBlockFunction, which goes back to step 1. This commit prevents the inifinite loop by building the global block in GenerateBlockFunction before emitting the body of the block function. rdar://problem/34541684 Differential Revision: https://reviews.llvm.org/D38118 llvm-svn: 314029
77 lines
2.2 KiB
Objective-C
77 lines
2.2 KiB
Objective-C
// RUN: %clang_cc1 -fblocks -triple x86_64-apple-darwin -fobjc-runtime=macosx-fragile-10.5 -emit-llvm %s -o %t-64.ll
|
|
// RUN: FileCheck -check-prefix CHECK-LP64 --input-file=%t-64.ll %s
|
|
// rdar: // 8390455
|
|
|
|
@class NSArray;
|
|
|
|
static NSArray *(^ArrayRecurs)(NSArray *addresses, unsigned long level) = ^(NSArray *addresses, unsigned long level) {
|
|
|
|
for(id rawAddress in addresses)
|
|
{
|
|
NSArray *separatedAddresses = ((NSArray*)0);
|
|
separatedAddresses = ArrayRecurs((NSArray *)rawAddress, level+1);
|
|
}
|
|
return (NSArray *)0;
|
|
};
|
|
|
|
extern NSArray *address;
|
|
extern unsigned long level;
|
|
|
|
void FUNC()
|
|
{
|
|
ArrayRecurs(address, level);
|
|
|
|
static NSArray *(^ArrayRecurs)(NSArray *addresses, unsigned long level) = ^(NSArray *addresses, unsigned long level) {
|
|
|
|
for(id rawAddress in addresses)
|
|
{
|
|
NSArray *separatedAddresses = ((NSArray*)0);
|
|
separatedAddresses = ArrayRecurs((NSArray *)rawAddress, level+1);
|
|
}
|
|
return (NSArray *)0;
|
|
};
|
|
ArrayRecurs(address, level);
|
|
|
|
if (ArrayRecurs) {
|
|
static NSArray *(^ArrayRecurs)(NSArray *addresses, unsigned long level) = ^(NSArray *addresses, unsigned long level) {
|
|
|
|
for(id rawAddress in addresses)
|
|
{
|
|
NSArray *separatedAddresses = ((NSArray*)0);
|
|
separatedAddresses = ArrayRecurs((NSArray *)rawAddress, level+1);
|
|
}
|
|
return (NSArray *)0;
|
|
};
|
|
ArrayRecurs(address, level);
|
|
}
|
|
}
|
|
|
|
void FUNC2() {
|
|
static void (^const block1)(int) = ^(int a){
|
|
if (a--)
|
|
block1(a);
|
|
};
|
|
}
|
|
|
|
// CHECK-LABEL-LP64: define void @FUNC2(
|
|
// CHECK: define internal void @_block_invoke{{.*}}(
|
|
// CHECK: call void %{{.*}}(i8* bitcast ({ i8**, i32, i32, i8*, %struct.__block_descriptor* }* @__block_literal_global{{.*}} to i8*), i32 %{{.*}})
|
|
|
|
void FUNC1()
|
|
{
|
|
static NSArray *(^ArrayRecurs)(NSArray *addresses, unsigned long level) = ^(NSArray *addresses, unsigned long level) {
|
|
|
|
for(id rawAddress in addresses)
|
|
{
|
|
NSArray *separatedAddresses = ((NSArray*)0);
|
|
separatedAddresses = ArrayRecurs((NSArray *)rawAddress, level+1);
|
|
}
|
|
return (NSArray *)0;
|
|
};
|
|
ArrayRecurs(address, level);
|
|
}
|
|
// CHECK-LP64: @ArrayRecurs = internal global
|
|
// CHECK-LP64: @FUNC.ArrayRecurs = internal global
|
|
// CHECK-LP64: @FUNC.ArrayRecurs.1 = internal global
|
|
// CHECK-LP64: @FUNC1.ArrayRecurs = internal global
|