blocks.m revision 1fc9a647f9fbcb2374ebc569964c687576437920
1// RUN: %clang_cc1 -triple x86_64-apple-darwin10 -analyze -analyzer-checker=core -analyzer-store=region -fblocks -analyzer-opt-analyze-nested-blocks -verify %s 2 3//===----------------------------------------------------------------------===// 4// The following code is reduced using delta-debugging from Mac OS X headers: 5//===----------------------------------------------------------------------===// 6 7typedef __builtin_va_list va_list; 8typedef unsigned int uint32_t; 9typedef struct dispatch_queue_s *dispatch_queue_t; 10typedef struct dispatch_queue_attr_s *dispatch_queue_attr_t; 11typedef void (^dispatch_block_t)(void); 12void dispatch_async(dispatch_queue_t queue, dispatch_block_t block); 13__attribute__((visibility("default"))) __attribute__((__malloc__)) __attribute__((__warn_unused_result__)) __attribute__((__nothrow__)) dispatch_queue_t dispatch_queue_create(const char *label, dispatch_queue_attr_t attr); 14typedef long dispatch_once_t; 15void dispatch_once(dispatch_once_t *predicate, dispatch_block_t block); 16typedef signed char BOOL; 17typedef unsigned long NSUInteger; 18typedef struct _NSZone NSZone; 19@class NSInvocation, NSMethodSignature, NSCoder, NSString, NSEnumerator; 20@protocol NSObject 21- (BOOL)isEqual:(id)object; 22- (oneway void)release; 23@end 24@protocol NSCopying - (id)copyWithZone:(NSZone *)zone; @end 25@protocol NSMutableCopying - (id)mutableCopyWithZone:(NSZone *)zone; @end 26@protocol NSCoding - (void)encodeWithCoder:(NSCoder *)aCoder; @end 27@interface NSObject <NSObject> {} 28+ (id)alloc; 29- (id)init; 30- (id)copy; 31@end 32extern id NSAllocateObject(Class aClass, NSUInteger extraBytes, NSZone *zone); 33@interface NSString : NSObject <NSCopying, NSMutableCopying, NSCoding> 34- (NSUInteger)length; 35- (const char *)UTF8String; 36- (id)initWithFormat:(NSString *)format arguments:(va_list)argList __attribute__((format(__NSString__, 1, 0))); 37@end 38@class NSString, NSData; 39typedef struct cssm_sample {} CSSM_SAMPLEGROUP, *CSSM_SAMPLEGROUP_PTR; 40typedef struct __aslclient *aslclient; 41typedef struct __aslmsg *aslmsg; 42aslclient asl_open(const char *ident, const char *facility, uint32_t opts); 43int asl_log(aslclient asl, aslmsg msg, int level, const char *format, ...) __attribute__((__format__ (__printf__, 4, 5))); 44 45//===----------------------------------------------------------------------===// 46// Begin actual test cases. 47//===----------------------------------------------------------------------===// 48 49// test1 - This test case exposed logic that caused the analyzer to crash because of a memory bug 50// in BlockDataRegion. It represents real code that contains two block literals. Eventually 51// via IPA 'logQueue' and 'client' should be updated after the call to 'dispatch_once'. 52void test1(NSString *format, ...) { 53 static dispatch_queue_t logQueue; 54 static aslclient client; 55 static dispatch_once_t pred; 56 do { 57 if (__builtin_expect(*(&pred), ~0l) != ~0l) 58 dispatch_once(&pred, ^{ 59 logQueue = dispatch_queue_create("com.mycompany.myproduct.asl", ((void*)0)); 60 client = asl_open(((void*)0), "com.mycompany.myproduct", 0); 61 }); 62 } while (0); 63 64 va_list args; 65 __builtin_va_start(args, format); 66 67 NSString *str = [[NSString alloc] initWithFormat:format arguments:args]; 68 dispatch_async(logQueue, ^{ asl_log(client, ((void*)0), 4, "%s", [str UTF8String]); }); 69 [str release]; 70 71 __builtin_va_end(args); 72} 73 74// test2 - Test that captured variables that are uninitialized are flagged 75// as such. 76void test2() { 77 static int y = 0; 78 int x; 79 ^{ y = x + 1; }(); // expected-warning{{Variable 'x' is uninitialized when captured by block}} 80} 81 82void test2_b() { 83 static int y = 0; 84 __block int x; 85 ^{ y = x + 1; }(); // expected-warning {{left operand of '+' is a garbage value}} 86} 87 88void test2_c() { 89 typedef void (^myblock)(void); 90 myblock f = ^() { f(); }; // expected-warning{{Variable 'f' is uninitialized when captured by block}} 91} 92 93 94void testMessaging() { 95 // <rdar://problem/12119814> 96 [[^(){} copy] release]; 97} 98 99 100// FALSE POSITIVE <rdar://problem/12415065> 101@interface rdar12415065 : NSObject 102@end 103 104@implementation rdar12415065 105- (void)test { 106 // At one point this crashed because we created a path note at a 107 // PreStmtPurgeDeadSymbols point but only knew how to deal with PostStmt 108 // points. <rdar://problem/12687586> 109 110 extern dispatch_queue_t queue; 111 112 if (!queue) 113 return; 114 115 // FALSE POSITIVE 116 // expected-warning@+1 {{Variable 'x' is uninitialized when captured by block}} 117 dispatch_async(queue, ^{ 118 double x = 0.0; 119 if (24.0f < x) { 120 dispatch_async(queue, ^{ (void)x; }); 121 [self test]; 122 } 123 }); 124} 125@end 126