mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-25 05:26:06 +00:00
334 lines
12 KiB
C++
334 lines
12 KiB
C++
// RUN: %clang_analyze_cc1 -std=c++14 -triple amdgcn-unknown-unknown \
|
|
// RUN: -analyzer-checker=core,apiModeling.llvm.CastValue,debug.ExprInspection\
|
|
// RUN: -analyzer-output=text -verify -DX86 -DSUPPRESSED %s 2>&1 | FileCheck %s -check-prefix=X86-CHECK
|
|
//
|
|
// RUN: %clang_analyze_cc1 -std=c++14 -triple amdgcn-unknown-unknown \
|
|
// RUN: -analyzer-checker=core,apiModeling.llvm.CastValue,debug.ExprInspection\
|
|
// RUN: -analyzer-config suppress-dereferences-from-any-address-space=false\
|
|
// RUN: -analyzer-output=text -verify -DX86 -DNOT_SUPPRESSED %s 2>&1 | FileCheck %s -check-prefix=X86-CHECK
|
|
//
|
|
// RUN: %clang_analyze_cc1 -std=c++14 -triple amdgcn-unknown-unknown \
|
|
// RUN: -analyzer-checker=core,apiModeling.llvm.CastValue,debug.ExprInspection\
|
|
// RUN: -analyzer-config suppress-dereferences-from-any-address-space=true\
|
|
// RUN: -analyzer-output=text -verify -DX86 -DSUPPRESSED %s 2>&1 | FileCheck %s -check-prefix=X86-CHECK
|
|
//
|
|
// RUN: %clang_analyze_cc1 -std=c++14 -triple x86_64-unknown-unknown \
|
|
// RUN: -analyzer-checker=core,apiModeling.llvm.CastValue,debug.ExprInspection\
|
|
// RUN: -analyzer-output=text -verify -DX86 -DSUPPRESSED %s 2>&1 | FileCheck %s --check-prefix=X86-CHECK
|
|
//
|
|
// RUN: %clang_analyze_cc1 -std=c++14 -triple x86_64-unknown-unknown \
|
|
// RUN: -analyzer-checker=core,apiModeling.llvm.CastValue,debug.ExprInspection\
|
|
// RUN: -analyzer-config suppress-dereferences-from-any-address-space=true\
|
|
// RUN: -analyzer-output=text -verify -DX86 -DSUPPRESSED %s 2>&1 | FileCheck %s --check-prefix=X86-CHECK-SUPPRESSED
|
|
//
|
|
// RUN: %clang_analyze_cc1 -std=c++14 -triple x86_64-unknown-unknown \
|
|
// RUN: -analyzer-checker=core,apiModeling.llvm.CastValue,debug.ExprInspection\
|
|
// RUN: -analyzer-config suppress-dereferences-from-any-address-space=false\
|
|
// RUN: -analyzer-output=text -verify -DX86 -DNOT_SUPPRESSED %s 2>&1 | FileCheck %s --check-prefix=X86-CHECK
|
|
//
|
|
// RUN: %clang_analyze_cc1 -std=c++14 -triple mips-unknown-unknown \
|
|
// RUN: -analyzer-checker=core,apiModeling.llvm.CastValue,debug.ExprInspection\
|
|
// RUN: -analyzer-output=text -verify -DMIPS %s 2>&1
|
|
//
|
|
// RUN: %clang_analyze_cc1 -std=c++14 -triple mips-unknown-unknown \
|
|
// RUN: -analyzer-checker=core,apiModeling.llvm.CastValue,debug.ExprInspection\
|
|
// RUN: -analyzer-config suppress-dereferences-from-any-address-space=false\
|
|
// RUN: -analyzer-output=text -verify -DMIPS %s 2>&1
|
|
//
|
|
// RUN: %clang_analyze_cc1 -std=c++14 -triple mips-unknown-unknown \
|
|
// RUN: -analyzer-checker=core,apiModeling.llvm.CastValue,debug.ExprInspection\
|
|
// RUN: -analyzer-config suppress-dereferences-from-any-address-space=true\
|
|
// RUN: -analyzer-output=text -verify -DMIPS_SUPPRESSED %s
|
|
|
|
#include "Inputs/llvm.h"
|
|
|
|
// The amggcn triple case uses an intentionally different address space.
|
|
// The core.NullDereference checker intentionally ignores checks
|
|
// that use address spaces, so the case is differentiated here.
|
|
//
|
|
// From https://llvm.org/docs/AMDGPUUsage.html#address-spaces,
|
|
// select address space 3 (local), since the pointer size is
|
|
// different than Generic.
|
|
#define DEVICE __attribute__((address_space(3)))
|
|
|
|
namespace clang {
|
|
struct Shape {
|
|
template <typename T>
|
|
const T *castAs() const;
|
|
|
|
template <typename T>
|
|
const T *getAs() const;
|
|
};
|
|
class Triangle : public Shape {};
|
|
class Rectangle : public Shape {};
|
|
class Hexagon : public Shape {};
|
|
class Circle : public Shape {};
|
|
} // namespace clang
|
|
|
|
using namespace llvm;
|
|
using namespace clang;
|
|
|
|
void clang_analyzer_printState();
|
|
|
|
#if defined(X86)
|
|
void evalReferences(const Shape &S) {
|
|
const auto &C = dyn_cast<Circle>(S);
|
|
// expected-note@-1 {{Assuming 'S' is not a 'const class clang::Circle &'}}
|
|
// expected-note@-2 {{Dereference of null pointer}}
|
|
// expected-warning@-3 {{Dereference of null pointer}}
|
|
clang_analyzer_printState();
|
|
// XX86-CHECK: "dynamic_types": [
|
|
// XX86-CHECK-NEXT: { "region": "SymRegion{reg_$0<const struct clang::Shape & S>}", "dyn_type": "const class clang::Circle &", "sub_classable": true }
|
|
(void)C;
|
|
}
|
|
#if defined(SUPPRESSED)
|
|
void evalReferences_addrspace(const Shape &S) {
|
|
const auto &C = dyn_cast<DEVICE Circle>(S);
|
|
clang_analyzer_printState();
|
|
// X86-CHECK-SUPPRESSED: "dynamic_types": [
|
|
// X86-CHECK-SUPPRESSED-NEXT: { "region": "SymRegion{reg_$0<const Shape & S>}", "dyn_type": "const __attribute__((address_space(3))) class clang::Circle &", "sub_classable": true }
|
|
(void)C;
|
|
}
|
|
#endif
|
|
#if defined(NOT_SUPPRESSED)
|
|
void evalReferences_addrspace(const Shape &S) {
|
|
const auto &C = dyn_cast<DEVICE Circle>(S);
|
|
// expected-note@-1 {{Assuming 'S' is not a 'const __attribute__((address_space(3))) class clang::Circle &'}}
|
|
// expected-note@-2 {{Dereference of null pointer}}
|
|
// expected-warning@-3 {{Dereference of null pointer}}
|
|
clang_analyzer_printState();
|
|
// X86-CHECK: "dynamic_types": [
|
|
// X86-CHECK-NEXT: { "region": "SymRegion{reg_$0<const Shape & S>}", "dyn_type": "const __attribute__((address_space(3))) class clang::Circle &", "sub_classable": true }
|
|
(void)C;
|
|
}
|
|
#endif
|
|
#elif defined(MIPS)
|
|
void evalReferences(const Shape &S) {
|
|
const auto &C = dyn_cast<Circle>(S);
|
|
// expected-note@-1 {{Assuming 'S' is not a 'const class clang::Circle &'}}
|
|
// expected-note@-2 {{Dereference of null pointer}}
|
|
// expected-warning@-3 {{Dereference of null pointer}}
|
|
}
|
|
#if defined(MIPS_SUPPRESSED)
|
|
void evalReferences_addrspace(const Shape &S) {
|
|
const auto &C = dyn_cast<DEVICE Circle>(S);
|
|
(void)C;
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
void evalNonNullParamNonNullReturnReference(const Shape &S) {
|
|
const auto *C = dyn_cast_or_null<Circle>(S);
|
|
// expected-note@-1 {{'C' initialized here}}
|
|
|
|
if (!dyn_cast_or_null<Circle>(C)) {
|
|
// expected-note@-1 {{Assuming 'C' is a 'const class clang::Circle *'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (dyn_cast_or_null<Triangle>(C)) {
|
|
// expected-note@-1 {{Assuming 'C' is not a 'const class clang::Triangle *'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (dyn_cast_or_null<Rectangle>(C)) {
|
|
// expected-note@-1 {{Assuming 'C' is not a 'const class clang::Rectangle *'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (dyn_cast_or_null<Hexagon>(C)) {
|
|
// expected-note@-1 {{Assuming 'C' is not a 'const class clang::Hexagon *'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (isa<Triangle>(C)) {
|
|
// expected-note@-1 {{'C' is not a 'Triangle'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (isa<Triangle, Rectangle>(C)) {
|
|
// expected-note@-1 {{'C' is neither a 'Triangle' nor a 'Rectangle'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (isa<Triangle, Rectangle, Hexagon>(C)) {
|
|
// expected-note@-1 {{'C' is neither a 'Triangle' nor a 'Rectangle' nor a 'Hexagon'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (isa<Circle, Rectangle, Hexagon>(C)) {
|
|
// expected-note@-1 {{'C' is a 'Circle'}}
|
|
// expected-note@-2 {{Taking true branch}}
|
|
|
|
(void)(1 / !C);
|
|
// expected-note@-1 {{'C' is non-null}}
|
|
// expected-note@-2 {{Division by zero}}
|
|
// expected-warning@-3 {{Division by zero}}
|
|
}
|
|
}
|
|
|
|
void evalNonNullParamNonNullReturn(const Shape *S) {
|
|
const auto *C = cast<Circle>(S);
|
|
// expected-note@-1 {{'S' is a 'const class clang::Circle *'}}
|
|
// expected-note@-2 {{'C' initialized here}}
|
|
|
|
if (!dyn_cast_or_null<Circle>(C)) {
|
|
// expected-note@-1 {{Assuming 'C' is a 'const class clang::Circle *'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (dyn_cast_or_null<Triangle>(C)) {
|
|
// expected-note@-1 {{Assuming 'C' is not a 'const class clang::Triangle *'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (dyn_cast_or_null<Rectangle>(C)) {
|
|
// expected-note@-1 {{Assuming 'C' is not a 'const class clang::Rectangle *'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (dyn_cast_or_null<Hexagon>(C)) {
|
|
// expected-note@-1 {{Assuming 'C' is not a 'const class clang::Hexagon *'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (isa<Triangle>(C)) {
|
|
// expected-note@-1 {{'C' is not a 'Triangle'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (isa<Triangle, Rectangle>(C)) {
|
|
// expected-note@-1 {{'C' is neither a 'Triangle' nor a 'Rectangle'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (isa<Triangle, Rectangle, Hexagon>(C)) {
|
|
// expected-note@-1 {{'C' is neither a 'Triangle' nor a 'Rectangle' nor a 'Hexagon'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (isa<Circle, Rectangle, Hexagon>(C)) {
|
|
// expected-note@-1 {{'C' is a 'Circle'}}
|
|
// expected-note@-2 {{Taking true branch}}
|
|
|
|
(void)(1 / !C);
|
|
// expected-note@-1 {{'C' is non-null}}
|
|
// expected-note@-2 {{Division by zero}}
|
|
// expected-warning@-3 {{Division by zero}}
|
|
}
|
|
}
|
|
|
|
void evalNonNullParamNullReturn(const Shape *S) {
|
|
const auto *C = dyn_cast_or_null<Circle>(S);
|
|
// expected-note@-1 {{Assuming 'S' is not a 'const class clang::Circle *'}}
|
|
|
|
if (const auto *T = dyn_cast_or_null<Triangle>(S)) {
|
|
// expected-note@-1 {{Assuming 'S' is a 'const class clang::Triangle *'}}
|
|
// expected-note@-2 {{'T' initialized here}}
|
|
// expected-note@-3 {{'T' is non-null}}
|
|
// expected-note@-4 {{Taking true branch}}
|
|
|
|
(void)(1 / !T);
|
|
// expected-note@-1 {{'T' is non-null}}
|
|
// expected-note@-2 {{Division by zero}}
|
|
// expected-warning@-3 {{Division by zero}}
|
|
}
|
|
}
|
|
|
|
void evalNullParamNullReturn(const Shape *S) {
|
|
const auto *C = dyn_cast_or_null<Circle>(S);
|
|
// expected-note@-1 {{Assuming null pointer is passed into cast}}
|
|
// expected-note@-2 {{'C' initialized to a null pointer value}}
|
|
|
|
(void)(1 / (bool)C);
|
|
// expected-note@-1 {{Division by zero}}
|
|
// expected-warning@-2 {{Division by zero}}
|
|
}
|
|
|
|
void evalZeroParamNonNullReturnPointer(const Shape *S) {
|
|
const auto *C = S->castAs<Circle>();
|
|
// expected-note@-1 {{'S' is a 'const class clang::Circle *'}}
|
|
// expected-note@-2 {{'C' initialized here}}
|
|
|
|
(void)(1 / !C);
|
|
// expected-note@-1 {{'C' is non-null}}
|
|
// expected-note@-2 {{Division by zero}}
|
|
// expected-warning@-3 {{Division by zero}}
|
|
}
|
|
|
|
void evalZeroParamNonNullReturn(const Shape &S) {
|
|
const auto *C = S.castAs<Circle>();
|
|
// expected-note@-1 {{'C' initialized here}}
|
|
|
|
(void)(1 / !C);
|
|
// expected-note@-1 {{'C' is non-null}}
|
|
// expected-note@-2 {{Division by zero}}
|
|
// expected-warning@-3 {{Division by zero}}
|
|
}
|
|
|
|
void evalZeroParamNullReturn(const Shape *S) {
|
|
const auto &C = S->getAs<Circle>();
|
|
// expected-note@-1 {{Assuming 'S' is not a 'const class clang::Circle *'}}
|
|
// expected-note@-2 {{Storing null pointer value}}
|
|
// expected-note@-3 {{'C' initialized here}}
|
|
|
|
if (!dyn_cast_or_null<Triangle>(S)) {
|
|
// expected-note@-1 {{Assuming 'S' is a 'const class clang::Triangle *'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
if (!dyn_cast_or_null<Triangle>(S)) {
|
|
// expected-note@-1 {{'S' is a 'Triangle'}}
|
|
// expected-note@-2 {{Taking false branch}}
|
|
return;
|
|
}
|
|
|
|
(void)(1 / (bool)C);
|
|
// expected-note@-1 {{Division by zero}}
|
|
// expected-warning@-2 {{Division by zero}}
|
|
}
|
|
|
|
// don't crash
|
|
// CastValueChecker was using QualType()->getPointeeCXXRecordDecl(), in
|
|
// getNoteTag which evaluated to nullptr, then crashed when attempting to
|
|
// deref an invocation to getNameAsString(). The fix is to use
|
|
// QualType().getAsString().
|
|
//
|
|
// Example:
|
|
// std::string CastToName =
|
|
// CastInfo ? CastInfo->to()->getAsCXXRecordDecl()->getNameAsString()
|
|
// : CastToTy->getPointeeCXXRecordDecl()->getNameAsString();
|
|
// Changed to:
|
|
// std::string CastToName =
|
|
// CastInfo ? CastInfo->to()->getAsCXXRecordDecl()->getNameAsString()
|
|
// : CastToTy.getAsString();
|
|
namespace llvm {
|
|
template <typename, typename a> void isa(a &);
|
|
template <typename> class PointerUnion {
|
|
public:
|
|
template <typename T> T *getAs() {
|
|
(void)isa<int>(*this);
|
|
return nullptr;
|
|
}
|
|
};
|
|
class LLVMContext {
|
|
PointerUnion<LLVMContext> c;
|
|
void d() { c.getAs<int>(); }
|
|
};
|
|
} // namespace llvm
|