190 lines
4.9 KiB
Plaintext
190 lines
4.9 KiB
Plaintext
|
// RUN: %clang_analyze_cc1 -triple x86_64-apple-darwin10 -analyzer-checker=core,alpha.core.StackAddressAsyncEscape -fblocks -fobjc-arc -verify %s
|
||
|
|
||
|
typedef struct dispatch_queue_s *dispatch_queue_t;
|
||
|
typedef void (^dispatch_block_t)(void);
|
||
|
void dispatch_async(dispatch_queue_t queue, dispatch_block_t block);
|
||
|
typedef long dispatch_once_t;
|
||
|
void dispatch_once(dispatch_once_t *predicate, dispatch_block_t block);
|
||
|
typedef long dispatch_time_t;
|
||
|
void dispatch_after(dispatch_time_t when, dispatch_queue_t queue, dispatch_block_t block);
|
||
|
void dispatch_barrier_sync(dispatch_queue_t queue, dispatch_block_t block);
|
||
|
|
||
|
extern dispatch_queue_t queue;
|
||
|
extern dispatch_once_t *predicate;
|
||
|
extern dispatch_time_t when;
|
||
|
|
||
|
void test_block_expr_async() {
|
||
|
int x = 123;
|
||
|
int *p = &x;
|
||
|
|
||
|
dispatch_async(queue, ^{
|
||
|
*p = 321;
|
||
|
});
|
||
|
// expected-warning@-3 {{Address of stack memory associated with local variable 'x' \
|
||
|
is captured by an asynchronously-executed block}}
|
||
|
}
|
||
|
|
||
|
void test_block_expr_once_no_leak() {
|
||
|
int x = 123;
|
||
|
int *p = &x;
|
||
|
// synchronous, no warning
|
||
|
dispatch_once(predicate, ^{
|
||
|
*p = 321;
|
||
|
});
|
||
|
}
|
||
|
|
||
|
void test_block_expr_after() {
|
||
|
int x = 123;
|
||
|
int *p = &x;
|
||
|
dispatch_after(when, queue, ^{
|
||
|
*p = 321;
|
||
|
});
|
||
|
// expected-warning@-3 {{Address of stack memory associated with local variable 'x' \
|
||
|
is captured by an asynchronously-executed block}}
|
||
|
}
|
||
|
|
||
|
void test_block_expr_async_no_leak() {
|
||
|
int x = 123;
|
||
|
int *p = &x;
|
||
|
// no leak
|
||
|
dispatch_async(queue, ^{
|
||
|
int y = x;
|
||
|
++y;
|
||
|
});
|
||
|
}
|
||
|
|
||
|
void test_block_var_async() {
|
||
|
int x = 123;
|
||
|
int *p = &x;
|
||
|
void (^b)(void) = ^void(void) {
|
||
|
*p = 1;
|
||
|
};
|
||
|
dispatch_async(queue, b);
|
||
|
// expected-warning@-1 {{Address of stack memory associated with local variable 'x' \
|
||
|
is captured by an asynchronously-executed block}}
|
||
|
}
|
||
|
|
||
|
void test_block_with_ref_async() {
|
||
|
int x = 123;
|
||
|
int &r = x;
|
||
|
void (^b)(void) = ^void(void) {
|
||
|
r = 1;
|
||
|
};
|
||
|
dispatch_async(queue, b);
|
||
|
// expected-warning@-1 {{Address of stack memory associated with local variable 'x' \
|
||
|
is captured by an asynchronously-executed block}}
|
||
|
}
|
||
|
|
||
|
dispatch_block_t get_leaking_block() {
|
||
|
int leaked_x = 791;
|
||
|
int *p = &leaked_x;
|
||
|
return ^void(void) {
|
||
|
*p = 1;
|
||
|
};
|
||
|
// expected-warning@-3 {{Address of stack memory associated with local variable 'leaked_x' \
|
||
|
is captured by a returned block}}
|
||
|
}
|
||
|
|
||
|
void test_returned_from_func_block_async() {
|
||
|
dispatch_async(queue, get_leaking_block());
|
||
|
// expected-warning@-1 {{Address of stack memory associated with local variable 'leaked_x' \
|
||
|
is captured by an asynchronously-executed block}}
|
||
|
}
|
||
|
|
||
|
// synchronous, no leak
|
||
|
void test_block_var_once() {
|
||
|
int x = 123;
|
||
|
int *p = &x;
|
||
|
void (^b)(void) = ^void(void) {
|
||
|
*p = 1;
|
||
|
};
|
||
|
dispatch_once(predicate, b); // no-warning
|
||
|
}
|
||
|
|
||
|
void test_block_var_after() {
|
||
|
int x = 123;
|
||
|
int *p = &x;
|
||
|
void (^b)(void) = ^void(void) {
|
||
|
*p = 1;
|
||
|
};
|
||
|
dispatch_after(when, queue, b);
|
||
|
// expected-warning@-1 {{Address of stack memory associated with local variable 'x' \
|
||
|
is captured by an asynchronously-executed block}}
|
||
|
}
|
||
|
|
||
|
void test_block_var_async_no_leak() {
|
||
|
int x = 123;
|
||
|
int *p = &x;
|
||
|
void (^b)(void) = ^void(void) {
|
||
|
int y = x;
|
||
|
++y;
|
||
|
};
|
||
|
dispatch_async(queue, b); // no-warning
|
||
|
}
|
||
|
|
||
|
void test_block_inside_block_async_no_leak() {
|
||
|
int x = 123;
|
||
|
int *p = &x;
|
||
|
void (^inner)(void) = ^void(void) {
|
||
|
int y = x;
|
||
|
++y;
|
||
|
};
|
||
|
void (^outer)(void) = ^void(void) {
|
||
|
int z = x;
|
||
|
++z;
|
||
|
inner();
|
||
|
};
|
||
|
dispatch_async(queue, outer); // no-warning
|
||
|
}
|
||
|
|
||
|
dispatch_block_t accept_and_pass_back_block(dispatch_block_t block) {
|
||
|
block();
|
||
|
return block; // no-warning
|
||
|
}
|
||
|
|
||
|
void test_passing_continuation_no_leak() {
|
||
|
int x = 123;
|
||
|
int *p = &x;
|
||
|
void (^cont)(void) = ^void(void) {
|
||
|
*p = 128;
|
||
|
};
|
||
|
accept_and_pass_back_block(cont); // no-warning
|
||
|
}
|
||
|
|
||
|
@interface NSObject
|
||
|
@end
|
||
|
@protocol OS_dispatch_semaphore
|
||
|
@end
|
||
|
typedef NSObject<OS_dispatch_semaphore> *dispatch_semaphore_t;
|
||
|
dispatch_semaphore_t dispatch_semaphore_create(long value);
|
||
|
long dispatch_semaphore_wait(dispatch_semaphore_t dsema, dispatch_time_t timeout);
|
||
|
long dispatch_semaphore_signal(dispatch_semaphore_t dsema);
|
||
|
|
||
|
void test_no_leaks_on_semaphore_pattern() {
|
||
|
int x = 0;
|
||
|
int *p = &x;
|
||
|
dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
|
||
|
dispatch_async(queue, ^{
|
||
|
*p = 1;
|
||
|
// Some work.
|
||
|
dispatch_semaphore_signal(semaphore);
|
||
|
}); // no-warning
|
||
|
|
||
|
// Do some other work concurrently with the asynchronous work
|
||
|
// Wait for the asynchronous work to finish
|
||
|
dispatch_semaphore_wait(semaphore, 1000);
|
||
|
}
|
||
|
|
||
|
void test_dispatch_barrier_sync() {
|
||
|
int buf[16];
|
||
|
for (int n = 0; n < 16; ++n) {
|
||
|
int *ptr = &buf[n];
|
||
|
// FIXME: Should not warn. The dispatch_barrier_sync() call ensures
|
||
|
// that the block does not outlive 'buf'.
|
||
|
dispatch_async(queue, ^{ // expected-warning{{Address of stack memory associated with local variable 'buf' is captured by an asynchronously-executed block}}
|
||
|
(void)ptr;
|
||
|
});
|
||
|
}
|
||
|
dispatch_barrier_sync(queue, ^{});
|
||
|
}
|