llvm-project/clang/test/Analysis/cxx-for-range.cpp
Roman Lebedev f4877c78c0
[clang] Improve -Wnull-dereference diag to be more in-line with reality
* Drop any mention of `volatile`.
  Please refer to https://reviews.llvm.org/D105338
* Drop address space check - it really doesn't affect the behavior,
  the store will still be dropped: https://godbolt.org/z/dP8fevxG4
2021-07-09 12:51:12 +03:00

105 lines
3.2 KiB
C++

// RUN: %clang_analyze_cc1 -std=c++11 -analyzer-checker=core -analyzer-output=plist-multi-file -o %t.plist -verify -analyzer-config eagerly-assume=false %s
// RUN: %normalize_plist <%t.plist | diff -ub %S/Inputs/expected-plists/cxx-for-range.cpp.plist -
extern void work();
void testLoop() {
int z[] = {1,2};
for (int y : z) {
work();
work();
if (y == 2)
*(volatile int *)0 = 1; // expected-warning{{indirection of null pointer will be deleted, not trap}} expected-note{{consider using __builtin_trap()}} expected-warning {{Dereference of null pointer}}
work();
work();
(void)y;
}
*(volatile int *)0 = 1; // expected-warning{{indirection of null pointer will be deleted, not trap}} expected-note{{consider using __builtin_trap()}}
}
class MagicVector {
public:
MagicVector();
using iterator = int *;
iterator begin() const;
iterator end() const;
};
MagicVector get(bool fail = false) {
if (fail)
*(volatile int *)0 = 1; // expected-warning{{indirection of null pointer will be deleted, not trap}} expected-note{{consider using __builtin_trap()}} expected-warning {{Dereference of null pointer}}
return MagicVector{};
}
void testLoopOpaqueCollection() {
for (int y : get()) {
work();
work();
if (y == 2)
*(volatile int *)0 = 1; // expected-warning{{indirection of null pointer will be deleted, not trap}} expected-note{{consider using __builtin_trap()}} expected-warning {{Dereference of null pointer}}
work();
work();
(void)y;
}
*(volatile int *)0 = 1; // expected-warning{{indirection of null pointer will be deleted, not trap}} expected-note{{consider using __builtin_trap()}} expected-warning {{Dereference of null pointer}}
}
class MagicVector2 {
public:
MagicVector2();
class iterator {
public:
int operator*() const;
iterator &operator++();
bool operator==(const iterator &);
bool operator!=(const iterator &);
};
iterator begin() const;
iterator end() const;
};
MagicVector2 get2() {
return MagicVector2{};
}
void testLoopOpaqueIterator() {
for (int y : get2()) {
work();
work();
if (y == 2)
*(volatile int *)0 = 1; // expected-warning{{indirection of null pointer will be deleted, not trap}} expected-note{{consider using __builtin_trap()}} expected-warning {{Dereference of null pointer}}
work();
work();
(void)y;
}
*(volatile int *)0 = 1; // expected-warning{{indirection of null pointer will be deleted, not trap}} expected-note{{consider using __builtin_trap()}} expected-warning {{Dereference of null pointer}}
}
void testLoopErrorInRange() {
for (int y : get(true)) { // error inside get()
work();
work();
if (y == 2)
*(volatile int *)0 = 1; // expected-warning{{indirection of null pointer will be deleted, not trap}} expected-note{{consider using __builtin_trap()}}
work();
work();
(void)y;
}
*(volatile int *)0 = 1; // expected-warning{{indirection of null pointer will be deleted, not trap}} expected-note{{consider using __builtin_trap()}}
}
void testForRangeInit() {
for (int *arr[3] = {nullptr, nullptr, nullptr}; int *p : arr) // expected-warning {{extension}}
*p = 1; // expected-warning {{Dereference of null pointer}}
}