Ted Kremenek | 8382cf5 | 2009-11-13 18:46:29 +0000 | [diff] [blame] | 1 | // RUN: clang-cc -analyze -analyzer-experimental-internal-checks -checker-cfref -analyzer-store=basic -verify -fobjc-gc -analyzer-constraints=basic %s |
| 2 | // RUN: clang-cc -analyze -analyzer-experimental-internal-checks -checker-cfref -analyzer-store=basic -verify -fobjc-gc -analyzer-constraints=range %s |
| 3 | // RUN: clang-cc -analyze -analyzer-experimental-internal-checks -checker-cfref -analyzer-store=basic -verify -fobjc-gc -disable-free %s |
| 4 | // RUN: clang-cc -analyze -analyzer-experimental-internal-checks -checker-cfref -analyzer-store=region -analyzer-constraints=basic -verify -fobjc-gc %s |
| 5 | // RUN: clang-cc -analyze -analyzer-experimental-internal-checks -checker-cfref -analyzer-store=region -analyzer-constraints=range -verify -fobjc-gc %s |
Ted Kremenek | 88739bf | 2008-06-16 18:46:17 +0000 | [diff] [blame] | 6 | |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | // The following code is reduced using delta-debugging from |
| 9 | // Foundation.h and CoreFoundation.h (Mac OS X). |
| 10 | // |
| 11 | // It includes the basic definitions for the test cases below. |
| 12 | // Not directly including [Core]Foundation.h directly makes this test case |
Ted Kremenek | dfc996c | 2008-06-16 19:51:41 +0000 | [diff] [blame] | 13 | // both svelte and portable to non-Mac platforms. |
Ted Kremenek | 88739bf | 2008-06-16 18:46:17 +0000 | [diff] [blame] | 14 | //===----------------------------------------------------------------------===// |
| 15 | |
| 16 | typedef const void * CFTypeRef; |
Ted Kremenek | 1261938 | 2009-01-12 21:45:02 +0000 | [diff] [blame] | 17 | void CFRelease(CFTypeRef cf); |
| 18 | CFTypeRef CFRetain(CFTypeRef cf); |
| 19 | CFTypeRef CFMakeCollectable(CFTypeRef cf); |
Ted Kremenek | 88739bf | 2008-06-16 18:46:17 +0000 | [diff] [blame] | 20 | typedef const struct __CFAllocator * CFAllocatorRef; |
| 21 | typedef double CFTimeInterval; |
| 22 | typedef CFTimeInterval CFAbsoluteTime; |
| 23 | typedef const struct __CFDate * CFDateRef; |
| 24 | extern CFDateRef CFDateCreate(CFAllocatorRef allocator, CFAbsoluteTime at); |
Ted Kremenek | 1261938 | 2009-01-12 21:45:02 +0000 | [diff] [blame] | 25 | extern CFAbsoluteTime CFDateGetAbsoluteTime(CFDateRef theDate); |
| 26 | typedef struct objc_object {} *id; |
Ted Kremenek | 88739bf | 2008-06-16 18:46:17 +0000 | [diff] [blame] | 27 | typedef signed char BOOL; |
Mike Stump | 655a63d | 2009-07-21 19:01:48 +0000 | [diff] [blame] | 28 | static __inline__ __attribute__((always_inline)) id NSMakeCollectable(CFTypeRef cf) { return 0; } |
Ted Kremenek | 1261938 | 2009-01-12 21:45:02 +0000 | [diff] [blame] | 29 | @protocol NSObject - (BOOL)isEqual:(id)object; |
| 30 | - (oneway void)release; |
Ted Kremenek | 23b8eaa | 2009-01-28 06:25:48 +0000 | [diff] [blame] | 31 | - (id)retain; |
Ted Kremenek | 1261938 | 2009-01-12 21:45:02 +0000 | [diff] [blame] | 32 | @end |
| 33 | @class NSArray; |
Ted Kremenek | 88739bf | 2008-06-16 18:46:17 +0000 | [diff] [blame] | 34 | |
| 35 | //===----------------------------------------------------------------------===// |
| 36 | // Test cases. |
| 37 | //===----------------------------------------------------------------------===// |
| 38 | |
Ted Kremenek | 1261938 | 2009-01-12 21:45:02 +0000 | [diff] [blame] | 39 | CFAbsoluteTime f1_use_after_release() { |
Ted Kremenek | 88739bf | 2008-06-16 18:46:17 +0000 | [diff] [blame] | 40 | CFAbsoluteTime t = CFAbsoluteTimeGetCurrent(); |
| 41 | CFDateRef date = CFDateCreate(0, t); |
| 42 | CFRetain(date); |
| 43 | [NSMakeCollectable(date) release]; |
| 44 | CFDateGetAbsoluteTime(date); // no-warning |
| 45 | CFRelease(date); |
| 46 | t = CFDateGetAbsoluteTime(date); // expected-warning{{Reference-counted object is used after it is released.}} |
| 47 | return t; |
| 48 | } |
| 49 | |
Ted Kremenek | 1261938 | 2009-01-12 21:45:02 +0000 | [diff] [blame] | 50 | // The following two test cases verifies that CFMakeCollectable is a no-op |
| 51 | // in non-GC mode and a "release" in GC mode. |
Ted Kremenek | 23b8eaa | 2009-01-28 06:25:48 +0000 | [diff] [blame] | 52 | CFAbsoluteTime f2_use_after_release() { |
Ted Kremenek | 1261938 | 2009-01-12 21:45:02 +0000 | [diff] [blame] | 53 | CFAbsoluteTime t = CFAbsoluteTimeGetCurrent(); |
| 54 | CFDateRef date = CFDateCreate(0, t); |
| 55 | CFRetain(date); |
| 56 | [(id) CFMakeCollectable(date) release]; |
| 57 | CFDateGetAbsoluteTime(date); // no-warning |
| 58 | CFRelease(date); |
| 59 | t = CFDateGetAbsoluteTime(date); // expected-warning{{Reference-counted object is used after it is released.}} |
| 60 | return t; |
| 61 | } |
| 62 | |
| 63 | CFAbsoluteTime f2_noleak() { |
Ted Kremenek | 2eff0f9 | 2008-11-05 22:17:39 +0000 | [diff] [blame] | 64 | CFAbsoluteTime t = CFAbsoluteTimeGetCurrent(); |
| 65 | CFDateRef date = CFDateCreate(0, t); |
| 66 | CFRetain(date); |
| 67 | [(id) CFMakeCollectable(date) release]; |
| 68 | CFDateGetAbsoluteTime(date); // no-warning |
| 69 | t = CFDateGetAbsoluteTime(date); // no-warning |
| 70 | CFRelease(date); // no-warning |
| 71 | return t; |
| 72 | } |
| 73 | |
Ted Kremenek | 23b8eaa | 2009-01-28 06:25:48 +0000 | [diff] [blame] | 74 | void f3_leak_with_gc() { |
Ted Kremenek | 7a10181 | 2009-02-13 00:39:34 +0000 | [diff] [blame] | 75 | CFDateRef date = CFDateCreate(0, CFAbsoluteTimeGetCurrent()); // expected-warning 2 {{leak}} |
Ted Kremenek | cf118d4 | 2009-02-04 23:49:09 +0000 | [diff] [blame] | 76 | [[(id) date retain] release]; |
Ted Kremenek | 23b8eaa | 2009-01-28 06:25:48 +0000 | [diff] [blame] | 77 | } |
| 78 | |
Ted Kremenek | 1261938 | 2009-01-12 21:45:02 +0000 | [diff] [blame] | 79 | // The following test case verifies that we "stop tracking" a retained object |
| 80 | // when it is passed as an argument to an implicitly defined function. |
| 81 | CFAbsoluteTime f4() { |
| 82 | CFAbsoluteTime t = CFAbsoluteTimeGetCurrent(); |
| 83 | CFDateRef date = CFDateCreate(0, t); |
| 84 | CFRetain(date); |
| 85 | some_implicitly_defined_function_stop_tracking(date); // no-warning |
| 86 | return t; |
| 87 | } |