1 // RUN: %clang_analyze_cc1 -analyzer-checker=core,debug.ExprInspection -verify %s -analyzer-config eagerly-assume=false
2 
3 // Tests for c11 atomics. Many of these tests currently yield unknown
4 // because we don't fully model the atomics and instead imprecisely
5 // treat their arguments as escaping.
6 
7 typedef unsigned int uint32_t;
8 typedef enum memory_order {
9   memory_order_relaxed = __ATOMIC_RELAXED,
10   memory_order_consume = __ATOMIC_CONSUME,
11   memory_order_acquire = __ATOMIC_ACQUIRE,
12   memory_order_release = __ATOMIC_RELEASE,
13   memory_order_acq_rel = __ATOMIC_ACQ_REL,
14   memory_order_seq_cst = __ATOMIC_SEQ_CST
15 } memory_order;
16 
17 void clang_analyzer_eval(int);
18 
19 struct RefCountedStruct {
20   uint32_t refCount;
21   void *ptr;
22 };
23 
test_atomic_fetch_add(struct RefCountedStruct * s)24 void test_atomic_fetch_add(struct RefCountedStruct *s) {
25   s->refCount = 1;
26 
27   uint32_t result = __c11_atomic_fetch_add((volatile _Atomic(uint32_t) *)&s->refCount,- 1, memory_order_relaxed);
28 
29   // When we model atomics fully this should (probably) be FALSE. It should never
30   // be TRUE (because the operation mutates the passed in storage).
31   clang_analyzer_eval(s->refCount == 1); // expected-warning {{UNKNOWN}}
32 
33   // When fully modeled this should be TRUE
34   clang_analyzer_eval(result == 1); // expected-warning {{UNKNOWN}}
35 }
36 
test_atomic_load(struct RefCountedStruct * s)37 void test_atomic_load(struct RefCountedStruct *s) {
38   s->refCount = 1;
39 
40   uint32_t result = __c11_atomic_load((volatile _Atomic(uint32_t) *)&s->refCount, memory_order_relaxed);
41 
42   // When we model atomics fully this should (probably) be TRUE.
43   clang_analyzer_eval(s->refCount == 1); // expected-warning {{UNKNOWN}}
44 
45   // When fully modeled this should be TRUE
46   clang_analyzer_eval(result == 1); // expected-warning {{UNKNOWN}}
47 }
48 
test_atomic_store(struct RefCountedStruct * s)49 void test_atomic_store(struct RefCountedStruct *s) {
50   s->refCount = 1;
51 
52   __c11_atomic_store((volatile _Atomic(uint32_t) *)&s->refCount, 2, memory_order_relaxed);
53 
54   // When we model atomics fully this should (probably) be FALSE. It should never
55   // be TRUE (because the operation mutates the passed in storage).
56   clang_analyzer_eval(s->refCount == 1); // expected-warning {{UNKNOWN}}
57 }
58 
test_atomic_exchange(struct RefCountedStruct * s)59 void test_atomic_exchange(struct RefCountedStruct *s) {
60   s->refCount = 1;
61 
62   uint32_t result = __c11_atomic_exchange((volatile _Atomic(uint32_t) *)&s->refCount, 2, memory_order_relaxed);
63 
64   // When we model atomics fully this should (probably) be FALSE. It should never
65   // be TRUE (because the operation mutates the passed in storage).
66   clang_analyzer_eval(s->refCount == 1); // expected-warning {{UNKNOWN}}
67 
68   // When fully modeled this should be TRUE
69   clang_analyzer_eval(result == 1); // expected-warning {{UNKNOWN}}
70 }
71 
72 
test_atomic_compare_exchange_strong(struct RefCountedStruct * s)73 void test_atomic_compare_exchange_strong(struct RefCountedStruct *s) {
74   s->refCount = 1;
75   uint32_t expected = 2;
76   uint32_t desired = 3;
77   _Bool result = __c11_atomic_compare_exchange_strong((volatile _Atomic(uint32_t) *)&s->refCount, &expected, desired, memory_order_relaxed, memory_order_relaxed);
78 
79   // For now we expect both expected and refCount to be invalidated by the
80   // call. In the future we should model more precisely.
81   clang_analyzer_eval(s->refCount == 3); // expected-warning {{UNKNOWN}}
82   clang_analyzer_eval(expected == 2); // expected-warning {{UNKNOWN}}
83 }
84 
test_atomic_compare_exchange_weak(struct RefCountedStruct * s)85 void test_atomic_compare_exchange_weak(struct RefCountedStruct *s) {
86   s->refCount = 1;
87   uint32_t expected = 2;
88   uint32_t desired = 3;
89   _Bool result = __c11_atomic_compare_exchange_weak((volatile _Atomic(uint32_t) *)&s->refCount, &expected, desired, memory_order_relaxed, memory_order_relaxed);
90 
91   // For now we expect both expected and refCount to be invalidated by the
92   // call. In the future we should model more precisely.
93   clang_analyzer_eval(s->refCount == 3); // expected-warning {{UNKNOWN}}
94   clang_analyzer_eval(expected == 2); // expected-warning {{UNKNOWN}}
95 }
96