mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-16 12:36:37 +00:00

Previously, `LazyCompoundVal` bindings to subregions referred by `LazyCopoundVals`, were not marked as //lazily copied//. This change returns `LazyCompoundVals` from `getInterestingValues()`, so their regions can be marked as //lazily copied// in `RemoveDeadBindingsWorker::VisitBinding()`. Depends on D134947 Authored by: Tomasz Kamiński <tomasz.kamiński@sonarsource.com> Reviewed By: martong Differential Revision: https://reviews.llvm.org/D135136
102 lines
2.9 KiB
C++
102 lines
2.9 KiB
C++
// RUN: %clang_analyze_cc1 -analyzer-checker=core,debug.ExprInspection -verify %s
|
|
|
|
template <typename T> void clang_analyzer_dump(T);
|
|
template <typename T> void clang_analyzer_value(T);
|
|
void clang_analyzer_warnIfReached();
|
|
|
|
struct Node { int* ptr; };
|
|
|
|
void copy_on_stack(Node* n1) {
|
|
Node tmp = *n1;
|
|
Node* n2 = &tmp;
|
|
clang_analyzer_dump(n1); // expected-warning-re {{&SymRegion{reg_${{[0-9]+}}<Node * n1>}}}
|
|
clang_analyzer_dump(n2); // expected-warning {{&tmp}}
|
|
|
|
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>}}}
|
|
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>}}}
|
|
|
|
if (n1->ptr != n2->ptr)
|
|
clang_analyzer_warnIfReached(); // unreachable
|
|
(void)(n1->ptr);
|
|
(void)(n2->ptr);
|
|
}
|
|
|
|
void copy_on_heap(Node* n1) {
|
|
Node* n2 = new Node(*n1);
|
|
|
|
clang_analyzer_dump(n1); // expected-warning-re {{&SymRegion{reg_${{[0-9]+}}<Node * n1>}}}
|
|
clang_analyzer_dump(n2); // expected-warning-re {{&HeapSymRegion{conj_${{[0-9]+}}{Node *, LC{{[0-9]+}}, S{{[0-9]+}}, #{{[0-9]+}}}}}}
|
|
|
|
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>}}}
|
|
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>}}}
|
|
|
|
if (n1->ptr != n2->ptr)
|
|
clang_analyzer_warnIfReached(); // unreachable
|
|
(void)(n1->ptr);
|
|
(void)(n2->ptr);
|
|
}
|
|
|
|
struct List {
|
|
List* next;
|
|
int value;
|
|
int padding;
|
|
};
|
|
|
|
void deadCode(List orig) {
|
|
List c = orig;
|
|
clang_analyzer_dump(c.value);
|
|
// expected-warning-re@-1 {{reg_${{[0-9]+}}<int orig.value>}}
|
|
if (c.value == 42)
|
|
return;
|
|
clang_analyzer_value(c.value);
|
|
// expected-warning@-1 {{32s:{ [-2147483648, 41], [43, 2147483647] }}}
|
|
// Before symbol was garbage collected too early, and we lost the constraints.
|
|
if (c.value != 42)
|
|
return;
|
|
|
|
clang_analyzer_warnIfReached(); // no-warning: Dead code.
|
|
};
|
|
|
|
void ptr1(List* n) {
|
|
List* n2 = new List(*n); // ctor
|
|
if (!n->next) {
|
|
if (n2->next) {
|
|
clang_analyzer_warnIfReached(); // unreachable
|
|
}
|
|
}
|
|
delete n2;
|
|
}
|
|
|
|
void ptr2(List* n) {
|
|
List* n2 = new List(); // ctor
|
|
*n2 = *n; // assignment
|
|
if (!n->next) {
|
|
if (n2->next) {
|
|
clang_analyzer_warnIfReached(); // unreachable
|
|
}
|
|
}
|
|
delete n2;
|
|
}
|
|
|
|
struct Wrapper {
|
|
List head;
|
|
int count;
|
|
};
|
|
|
|
void nestedLazyCompoundVal(List* n) {
|
|
Wrapper* w = 0;
|
|
{
|
|
Wrapper lw;
|
|
lw.head = *n;
|
|
w = new Wrapper(lw);
|
|
}
|
|
if (!n->next) {
|
|
if (w->head.next) {
|
|
// Unreachable, w->head is a copy of *n, therefore
|
|
// w->head.next and n->next are equal
|
|
clang_analyzer_warnIfReached(); // no-warning: unreachable
|
|
}
|
|
}
|
|
delete w;
|
|
}
|