1f8643a9bSBalazs Benics // RUN: %clang_analyze_cc1 -analyzer-checker=core,debug.ExprInspection -verify %s
2f8643a9bSBalazs Benics
3f8643a9bSBalazs Benics template <typename T> void clang_analyzer_dump(T);
473716baaSTomasz Kamiński template <typename T> void clang_analyzer_value(T);
5f8643a9bSBalazs Benics void clang_analyzer_warnIfReached();
6f8643a9bSBalazs Benics
7f8643a9bSBalazs Benics struct Node { int* ptr; };
8f8643a9bSBalazs Benics
copy_on_stack(Node * n1)9f8643a9bSBalazs Benics void copy_on_stack(Node* n1) {
10f8643a9bSBalazs Benics Node tmp = *n1;
11f8643a9bSBalazs Benics Node* n2 = &tmp;
12f8643a9bSBalazs Benics clang_analyzer_dump(n1); // expected-warning-re {{&SymRegion{reg_${{[0-9]+}}<Node * n1>}}}
13f8643a9bSBalazs Benics clang_analyzer_dump(n2); // expected-warning {{&tmp}}
14f8643a9bSBalazs Benics
15f8643a9bSBalazs Benics clang_analyzer_dump(n1->ptr); // expected-warning-re {{&SymRegion{reg_${{[0-9]+}}<int * Element{SymRegion{reg_${{[0-9]+}}<Node * n1>},0 S64b,struct Node}.ptr>}}}
16f8643a9bSBalazs Benics clang_analyzer_dump(n2->ptr); // expected-warning-re {{&SymRegion{reg_${{[0-9]+}}<int * Element{SymRegion{reg_${{[0-9]+}}<Node * n1>},0 S64b,struct Node}.ptr>}}}
17f8643a9bSBalazs Benics
18f8643a9bSBalazs Benics if (n1->ptr != n2->ptr)
19f8643a9bSBalazs Benics clang_analyzer_warnIfReached(); // unreachable
20f8643a9bSBalazs Benics (void)(n1->ptr);
21f8643a9bSBalazs Benics (void)(n2->ptr);
22f8643a9bSBalazs Benics }
23f8643a9bSBalazs Benics
copy_on_heap(Node * n1)24f8643a9bSBalazs Benics void copy_on_heap(Node* n1) {
25f8643a9bSBalazs Benics Node* n2 = new Node(*n1);
26f8643a9bSBalazs Benics
27f8643a9bSBalazs Benics clang_analyzer_dump(n1); // expected-warning-re {{&SymRegion{reg_${{[0-9]+}}<Node * n1>}}}
28f8643a9bSBalazs Benics clang_analyzer_dump(n2); // expected-warning-re {{&HeapSymRegion{conj_${{[0-9]+}}{Node *, LC{{[0-9]+}}, S{{[0-9]+}}, #{{[0-9]+}}}}}}
29f8643a9bSBalazs Benics
30f8643a9bSBalazs Benics clang_analyzer_dump(n1->ptr); // expected-warning-re {{&SymRegion{reg_${{[0-9]+}}<int * Element{SymRegion{reg_${{[0-9]+}}<Node * n1>},0 S64b,struct Node}.ptr>}}}
31afcd862bSBalazs Benics clang_analyzer_dump(n2->ptr); // expected-warning-re {{&SymRegion{reg_${{[0-9]+}}<int * Element{SymRegion{reg_${{[0-9]+}}<Node * n1>},0 S64b,struct Node}.ptr>}}}
32f8643a9bSBalazs Benics
33f8643a9bSBalazs Benics if (n1->ptr != n2->ptr)
34afcd862bSBalazs Benics clang_analyzer_warnIfReached(); // unreachable
35f8643a9bSBalazs Benics (void)(n1->ptr);
36f8643a9bSBalazs Benics (void)(n2->ptr);
37f8643a9bSBalazs Benics }
3873716baaSTomasz Kamiński
3973716baaSTomasz Kamiński struct List {
4073716baaSTomasz Kamiński List* next;
4173716baaSTomasz Kamiński int value;
4273716baaSTomasz Kamiński int padding;
4373716baaSTomasz Kamiński };
4473716baaSTomasz Kamiński
deadCode(List orig)4573716baaSTomasz Kamiński void deadCode(List orig) {
4673716baaSTomasz Kamiński List c = orig;
4773716baaSTomasz Kamiński clang_analyzer_dump(c.value);
4873716baaSTomasz Kamiński // expected-warning-re@-1 {{reg_${{[0-9]+}}<int orig.value>}}
4973716baaSTomasz Kamiński if (c.value == 42)
5073716baaSTomasz Kamiński return;
5173716baaSTomasz Kamiński clang_analyzer_value(c.value);
52a6b42040STomasz Kamiński // expected-warning@-1 {{32s:{ [-2147483648, 41], [43, 2147483647] }}}
53a6b42040STomasz Kamiński // Before symbol was garbage collected too early, and we lost the constraints.
5473716baaSTomasz Kamiński if (c.value != 42)
5573716baaSTomasz Kamiński return;
5673716baaSTomasz Kamiński
57a6b42040STomasz Kamiński clang_analyzer_warnIfReached(); // no-warning: Dead code.
58a6b42040STomasz Kamiński };
59a6b42040STomasz Kamiński
ptr1(List * n)60a6b42040STomasz Kamiński void ptr1(List* n) {
61a6b42040STomasz Kamiński List* n2 = new List(*n); // ctor
62a6b42040STomasz Kamiński if (!n->next) {
63a6b42040STomasz Kamiński if (n2->next) {
64a6b42040STomasz Kamiński clang_analyzer_warnIfReached(); // unreachable
65a6b42040STomasz Kamiński }
66a6b42040STomasz Kamiński }
67a6b42040STomasz Kamiński delete n2;
68a6b42040STomasz Kamiński }
69a6b42040STomasz Kamiński
ptr2(List * n)70a6b42040STomasz Kamiński void ptr2(List* n) {
71a6b42040STomasz Kamiński List* n2 = new List(); // ctor
72a6b42040STomasz Kamiński *n2 = *n; // assignment
73a6b42040STomasz Kamiński if (!n->next) {
74a6b42040STomasz Kamiński if (n2->next) {
75a6b42040STomasz Kamiński clang_analyzer_warnIfReached(); // unreachable
76a6b42040STomasz Kamiński }
77a6b42040STomasz Kamiński }
78a6b42040STomasz Kamiński delete n2;
79a6b42040STomasz Kamiński }
80a6b42040STomasz Kamiński
81a6b42040STomasz Kamiński struct Wrapper {
82a6b42040STomasz Kamiński List head;
83a6b42040STomasz Kamiński int count;
84a6b42040STomasz Kamiński };
85a6b42040STomasz Kamiński
nestedLazyCompoundVal(List * n)86a6b42040STomasz Kamiński void nestedLazyCompoundVal(List* n) {
87a6b42040STomasz Kamiński Wrapper* w = 0;
88a6b42040STomasz Kamiński {
89a6b42040STomasz Kamiński Wrapper lw;
90a6b42040STomasz Kamiński lw.head = *n;
91a6b42040STomasz Kamiński w = new Wrapper(lw);
92a6b42040STomasz Kamiński }
93a6b42040STomasz Kamiński if (!n->next) {
94a6b42040STomasz Kamiński if (w->head.next) {
95*6194229cSTomasz Kamiński // Unreachable, w->head is a copy of *n, therefore
96a6b42040STomasz Kamiński // w->head.next and n->next are equal
97*6194229cSTomasz Kamiński clang_analyzer_warnIfReached(); // no-warning: unreachable
9873716baaSTomasz Kamiński }
99a6b42040STomasz Kamiński }
100a6b42040STomasz Kamiński delete w;
101a6b42040STomasz Kamiński }
102