clang 20.0.0 (based on r547379) from build 12806354. Bug: http://b/379133546 Test: N/A Change-Id: I2eb8938af55d809de674be63cb30cf27e801862b Upstream-Commit: ad834e67b1105d15ef907f6255d4c96e8e733f57
1679 lines
54 KiB
C++
1679 lines
54 KiB
C++
//==- MemRegion.h - Abstract memory regions for static analysis -*- C++ -*--==//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file defines MemRegion and its subclasses. MemRegion defines a
|
|
// partially-typed abstraction of memory useful for path-sensitive dataflow
|
|
// analyses.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_CLANG_STATICANALYZER_CORE_PATHSENSITIVE_MEMREGION_H
|
|
#define LLVM_CLANG_STATICANALYZER_CORE_PATHSENSITIVE_MEMREGION_H
|
|
|
|
#include "clang/AST/ASTContext.h"
|
|
#include "clang/AST/CharUnits.h"
|
|
#include "clang/AST/Decl.h"
|
|
#include "clang/AST/DeclObjC.h"
|
|
#include "clang/AST/DeclarationName.h"
|
|
#include "clang/AST/Expr.h"
|
|
#include "clang/AST/ExprObjC.h"
|
|
#include "clang/AST/Type.h"
|
|
#include "clang/Analysis/AnalysisDeclContext.h"
|
|
#include "clang/Basic/LLVM.h"
|
|
#include "clang/Basic/SourceLocation.h"
|
|
#include "clang/StaticAnalyzer/Core/PathSensitive/SVals.h"
|
|
#include "clang/StaticAnalyzer/Core/PathSensitive/SymExpr.h"
|
|
#include "llvm/ADT/DenseMap.h"
|
|
#include "llvm/ADT/FoldingSet.h"
|
|
#include "llvm/ADT/PointerIntPair.h"
|
|
#include "llvm/ADT/iterator_range.h"
|
|
#include "llvm/Support/Allocator.h"
|
|
#include "llvm/Support/Casting.h"
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
#include <cassert>
|
|
#include <cstdint>
|
|
#include <limits>
|
|
#include <optional>
|
|
#include <string>
|
|
#include <utility>
|
|
|
|
namespace clang {
|
|
|
|
class AnalysisDeclContext;
|
|
class CXXRecordDecl;
|
|
class Decl;
|
|
class LocationContext;
|
|
class StackFrameContext;
|
|
|
|
namespace ento {
|
|
|
|
class CodeTextRegion;
|
|
class MemRegion;
|
|
class MemRegionManager;
|
|
class MemSpaceRegion;
|
|
class SValBuilder;
|
|
class SymbolicRegion;
|
|
class VarRegion;
|
|
|
|
/// Represent a region's offset within the top level base region.
|
|
class RegionOffset {
|
|
/// The base region.
|
|
const MemRegion *R = nullptr;
|
|
|
|
/// The bit offset within the base region. Can be negative.
|
|
int64_t Offset;
|
|
|
|
public:
|
|
// We're using a const instead of an enumeration due to the size required;
|
|
// Visual Studio will only create enumerations of size int, not long long.
|
|
static const int64_t Symbolic = std::numeric_limits<int64_t>::max();
|
|
|
|
RegionOffset() = default;
|
|
RegionOffset(const MemRegion *r, int64_t off) : R(r), Offset(off) {}
|
|
|
|
/// It might return null.
|
|
const MemRegion *getRegion() const { return R; }
|
|
|
|
bool hasSymbolicOffset() const { return Offset == Symbolic; }
|
|
|
|
int64_t getOffset() const {
|
|
assert(!hasSymbolicOffset());
|
|
return Offset;
|
|
}
|
|
|
|
bool isValid() const { return R; }
|
|
};
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Base region classes.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
/// MemRegion - The root abstract class for all memory regions.
|
|
class MemRegion : public llvm::FoldingSetNode {
|
|
public:
|
|
enum Kind {
|
|
#define REGION(Id, Parent) Id ## Kind,
|
|
#define REGION_RANGE(Id, First, Last) BEGIN_##Id = First, END_##Id = Last,
|
|
#include "clang/StaticAnalyzer/Core/PathSensitive/Regions.def"
|
|
#undef REGION
|
|
#undef REGION_RANGE
|
|
};
|
|
|
|
private:
|
|
const Kind kind;
|
|
mutable std::optional<RegionOffset> cachedOffset;
|
|
|
|
protected:
|
|
MemRegion(Kind k) : kind(k) {}
|
|
virtual ~MemRegion();
|
|
|
|
public:
|
|
ASTContext &getContext() const;
|
|
|
|
virtual void Profile(llvm::FoldingSetNodeID& ID) const = 0;
|
|
|
|
virtual MemRegionManager &getMemRegionManager() const = 0;
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL const MemSpaceRegion *getMemorySpace() const;
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL const MemRegion *getBaseRegion() const;
|
|
|
|
/// Recursively retrieve the region of the most derived class instance of
|
|
/// regions of C++ base class instances.
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const MemRegion *getMostDerivedObjectRegion() const;
|
|
|
|
/// Check if the region is a subregion of the given region.
|
|
/// Each region is a subregion of itself.
|
|
virtual bool isSubRegionOf(const MemRegion *R) const;
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const MemRegion *StripCasts(bool StripBaseAndDerivedCasts = true) const;
|
|
|
|
/// If this is a symbolic region, returns the region. Otherwise,
|
|
/// goes up the base chain looking for the first symbolic base region.
|
|
/// It might return null.
|
|
const SymbolicRegion *getSymbolicBase() const;
|
|
|
|
bool hasStackStorage() const;
|
|
|
|
bool hasStackNonParametersStorage() const;
|
|
|
|
bool hasStackParametersStorage() const;
|
|
|
|
/// Compute the offset within the top level memory object.
|
|
RegionOffset getAsOffset() const;
|
|
|
|
/// Get a string representation of a region for debug use.
|
|
std::string getString() const;
|
|
|
|
virtual void dumpToStream(raw_ostream &os) const;
|
|
|
|
void dump() const;
|
|
|
|
/// Returns true if this region can be printed in a user-friendly way.
|
|
virtual bool canPrintPretty() const;
|
|
|
|
/// Print the region for use in diagnostics.
|
|
virtual void printPretty(raw_ostream &os) const;
|
|
|
|
/// Returns true if this region's textual representation can be used
|
|
/// as part of a larger expression.
|
|
virtual bool canPrintPrettyAsExpr() const;
|
|
|
|
/// Print the region as expression.
|
|
///
|
|
/// When this region represents a subexpression, the method is for printing
|
|
/// an expression containing it.
|
|
virtual void printPrettyAsExpr(raw_ostream &os) const;
|
|
|
|
Kind getKind() const { return kind; }
|
|
|
|
StringRef getKindStr() const;
|
|
|
|
template<typename RegionTy> const RegionTy* getAs() const;
|
|
template <typename RegionTy>
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL const RegionTy *castAs() const;
|
|
|
|
virtual bool isBoundable() const { return false; }
|
|
|
|
/// Get descriptive name for memory region. The name is obtained from
|
|
/// the variable/field declaration retrieved from the memory region.
|
|
/// Regions that point to an element of an array are returned as: "arr[0]".
|
|
/// Regions that point to a struct are returned as: "st.var".
|
|
//
|
|
/// \param UseQuotes Set if the name should be quoted.
|
|
///
|
|
/// \returns variable name for memory region
|
|
std::string getDescriptiveName(bool UseQuotes = true) const;
|
|
|
|
/// Retrieve source range from memory region. The range retrieval
|
|
/// is based on the decl obtained from the memory region.
|
|
/// For a VarRegion the range of the base region is returned.
|
|
/// For a FieldRegion the range of the field is returned.
|
|
/// If no declaration is found, an empty source range is returned.
|
|
/// The client is responsible for checking if the returned range is valid.
|
|
///
|
|
/// \returns source range for declaration retrieved from memory region
|
|
SourceRange sourceRange() const;
|
|
};
|
|
|
|
/// MemSpaceRegion - A memory region that represents a "memory space";
|
|
/// for example, the set of global variables, the stack frame, etc.
|
|
class MemSpaceRegion : public MemRegion {
|
|
protected:
|
|
MemRegionManager &Mgr;
|
|
|
|
MemSpaceRegion(MemRegionManager &mgr, Kind k) : MemRegion(k), Mgr(mgr) {
|
|
assert(classof(this));
|
|
}
|
|
|
|
MemRegionManager &getMemRegionManager() const override { return Mgr; }
|
|
|
|
public:
|
|
bool isBoundable() const override { return false; }
|
|
|
|
void Profile(llvm::FoldingSetNodeID &ID) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
Kind k = R->getKind();
|
|
return k >= BEGIN_MEMSPACES && k <= END_MEMSPACES;
|
|
}
|
|
};
|
|
|
|
/// CodeSpaceRegion - The memory space that holds the executable code of
|
|
/// functions and blocks.
|
|
class CodeSpaceRegion : public MemSpaceRegion {
|
|
friend class MemRegionManager;
|
|
|
|
CodeSpaceRegion(MemRegionManager &mgr)
|
|
: MemSpaceRegion(mgr, CodeSpaceRegionKind) {}
|
|
|
|
public:
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == CodeSpaceRegionKind;
|
|
}
|
|
};
|
|
|
|
class GlobalsSpaceRegion : public MemSpaceRegion {
|
|
virtual void anchor();
|
|
|
|
protected:
|
|
GlobalsSpaceRegion(MemRegionManager &mgr, Kind k) : MemSpaceRegion(mgr, k) {
|
|
assert(classof(this));
|
|
}
|
|
|
|
public:
|
|
static bool classof(const MemRegion *R) {
|
|
Kind k = R->getKind();
|
|
return k >= BEGIN_GLOBAL_MEMSPACES && k <= END_GLOBAL_MEMSPACES;
|
|
}
|
|
};
|
|
|
|
/// The region of the static variables within the current CodeTextRegion
|
|
/// scope.
|
|
///
|
|
/// Currently, only the static locals are placed there, so we know that these
|
|
/// variables do not get invalidated by calls to other functions.
|
|
class StaticGlobalSpaceRegion : public GlobalsSpaceRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const CodeTextRegion *CR;
|
|
|
|
StaticGlobalSpaceRegion(MemRegionManager &mgr, const CodeTextRegion *cr)
|
|
: GlobalsSpaceRegion(mgr, StaticGlobalSpaceRegionKind), CR(cr) {
|
|
assert(cr);
|
|
}
|
|
|
|
public:
|
|
void Profile(llvm::FoldingSetNodeID &ID) const override;
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const CodeTextRegion *getCodeRegion() const { return CR; }
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == StaticGlobalSpaceRegionKind;
|
|
}
|
|
};
|
|
|
|
/// The region for all the non-static global variables.
|
|
///
|
|
/// This class is further split into subclasses for efficient implementation of
|
|
/// invalidating a set of related global values as is done in
|
|
/// RegionStoreManager::invalidateRegions (instead of finding all the dependent
|
|
/// globals, we invalidate the whole parent region).
|
|
class NonStaticGlobalSpaceRegion : public GlobalsSpaceRegion {
|
|
void anchor() override;
|
|
|
|
protected:
|
|
NonStaticGlobalSpaceRegion(MemRegionManager &mgr, Kind k)
|
|
: GlobalsSpaceRegion(mgr, k) {
|
|
assert(classof(this));
|
|
}
|
|
|
|
public:
|
|
static bool classof(const MemRegion *R) {
|
|
Kind k = R->getKind();
|
|
return k >= BEGIN_NON_STATIC_GLOBAL_MEMSPACES &&
|
|
k <= END_NON_STATIC_GLOBAL_MEMSPACES;
|
|
}
|
|
};
|
|
|
|
/// The region containing globals which are defined in system/external
|
|
/// headers and are considered modifiable by system calls (ex: errno).
|
|
class GlobalSystemSpaceRegion : public NonStaticGlobalSpaceRegion {
|
|
friend class MemRegionManager;
|
|
|
|
GlobalSystemSpaceRegion(MemRegionManager &mgr)
|
|
: NonStaticGlobalSpaceRegion(mgr, GlobalSystemSpaceRegionKind) {}
|
|
|
|
public:
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == GlobalSystemSpaceRegionKind;
|
|
}
|
|
};
|
|
|
|
/// The region containing globals which are considered not to be modified
|
|
/// or point to data which could be modified as a result of a function call
|
|
/// (system or internal). Ex: Const global scalars would be modeled as part of
|
|
/// this region. This region also includes most system globals since they have
|
|
/// low chance of being modified.
|
|
class GlobalImmutableSpaceRegion : public NonStaticGlobalSpaceRegion {
|
|
friend class MemRegionManager;
|
|
|
|
GlobalImmutableSpaceRegion(MemRegionManager &mgr)
|
|
: NonStaticGlobalSpaceRegion(mgr, GlobalImmutableSpaceRegionKind) {}
|
|
|
|
public:
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == GlobalImmutableSpaceRegionKind;
|
|
}
|
|
};
|
|
|
|
/// The region containing globals which can be modified by calls to
|
|
/// "internally" defined functions - (for now just) functions other then system
|
|
/// calls.
|
|
class GlobalInternalSpaceRegion : public NonStaticGlobalSpaceRegion {
|
|
friend class MemRegionManager;
|
|
|
|
GlobalInternalSpaceRegion(MemRegionManager &mgr)
|
|
: NonStaticGlobalSpaceRegion(mgr, GlobalInternalSpaceRegionKind) {}
|
|
|
|
public:
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == GlobalInternalSpaceRegionKind;
|
|
}
|
|
};
|
|
|
|
class HeapSpaceRegion : public MemSpaceRegion {
|
|
friend class MemRegionManager;
|
|
|
|
HeapSpaceRegion(MemRegionManager &mgr)
|
|
: MemSpaceRegion(mgr, HeapSpaceRegionKind) {}
|
|
|
|
public:
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == HeapSpaceRegionKind;
|
|
}
|
|
};
|
|
|
|
class UnknownSpaceRegion : public MemSpaceRegion {
|
|
friend class MemRegionManager;
|
|
|
|
UnknownSpaceRegion(MemRegionManager &mgr)
|
|
: MemSpaceRegion(mgr, UnknownSpaceRegionKind) {}
|
|
|
|
public:
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == UnknownSpaceRegionKind;
|
|
}
|
|
};
|
|
|
|
class StackSpaceRegion : public MemSpaceRegion {
|
|
virtual void anchor();
|
|
|
|
const StackFrameContext *SFC;
|
|
|
|
protected:
|
|
StackSpaceRegion(MemRegionManager &mgr, Kind k, const StackFrameContext *sfc)
|
|
: MemSpaceRegion(mgr, k), SFC(sfc) {
|
|
assert(classof(this));
|
|
assert(sfc);
|
|
}
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const StackFrameContext *getStackFrame() const { return SFC; }
|
|
|
|
void Profile(llvm::FoldingSetNodeID &ID) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
Kind k = R->getKind();
|
|
return k >= BEGIN_STACK_MEMSPACES && k <= END_STACK_MEMSPACES;
|
|
}
|
|
};
|
|
|
|
class StackLocalsSpaceRegion : public StackSpaceRegion {
|
|
friend class MemRegionManager;
|
|
|
|
StackLocalsSpaceRegion(MemRegionManager &mgr, const StackFrameContext *sfc)
|
|
: StackSpaceRegion(mgr, StackLocalsSpaceRegionKind, sfc) {}
|
|
|
|
public:
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == StackLocalsSpaceRegionKind;
|
|
}
|
|
};
|
|
|
|
class StackArgumentsSpaceRegion : public StackSpaceRegion {
|
|
private:
|
|
friend class MemRegionManager;
|
|
|
|
StackArgumentsSpaceRegion(MemRegionManager &mgr, const StackFrameContext *sfc)
|
|
: StackSpaceRegion(mgr, StackArgumentsSpaceRegionKind, sfc) {}
|
|
|
|
public:
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == StackArgumentsSpaceRegionKind;
|
|
}
|
|
};
|
|
|
|
/// SubRegion - A region that subsets another larger region. Most regions
|
|
/// are subclasses of SubRegion.
|
|
class SubRegion : public MemRegion {
|
|
virtual void anchor();
|
|
|
|
protected:
|
|
const MemRegion* superRegion;
|
|
|
|
SubRegion(const MemRegion *sReg, Kind k) : MemRegion(k), superRegion(sReg) {
|
|
assert(classof(this));
|
|
assert(sReg);
|
|
}
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const MemRegion* getSuperRegion() const {
|
|
return superRegion;
|
|
}
|
|
|
|
MemRegionManager &getMemRegionManager() const override;
|
|
|
|
bool isSubRegionOf(const MemRegion* R) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() > END_MEMSPACES;
|
|
}
|
|
};
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// MemRegion subclasses.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
/// AllocaRegion - A region that represents an untyped blob of bytes created
|
|
/// by a call to 'alloca'.
|
|
class AllocaRegion : public SubRegion {
|
|
friend class MemRegionManager;
|
|
|
|
// Block counter. Used to distinguish different pieces of memory allocated by
|
|
// alloca at the same call site.
|
|
unsigned Cnt;
|
|
|
|
const Expr *Ex;
|
|
|
|
AllocaRegion(const Expr *ex, unsigned cnt, const MemSpaceRegion *superRegion)
|
|
: SubRegion(superRegion, AllocaRegionKind), Cnt(cnt), Ex(ex) {
|
|
assert(Ex);
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID& ID, const Expr *Ex,
|
|
unsigned Cnt, const MemRegion *superRegion);
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const Expr *getExpr() const { return Ex; }
|
|
|
|
bool isBoundable() const override { return true; }
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override;
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == AllocaRegionKind;
|
|
}
|
|
};
|
|
|
|
/// TypedRegion - An abstract class representing regions that are typed.
|
|
class TypedRegion : public SubRegion {
|
|
void anchor() override;
|
|
|
|
protected:
|
|
TypedRegion(const MemRegion *sReg, Kind k) : SubRegion(sReg, k) {
|
|
assert(classof(this));
|
|
}
|
|
|
|
public:
|
|
virtual QualType getLocationType() const = 0;
|
|
|
|
QualType getDesugaredLocationType(ASTContext &Context) const {
|
|
return getLocationType().getDesugaredType(Context);
|
|
}
|
|
|
|
bool isBoundable() const override { return true; }
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
unsigned k = R->getKind();
|
|
return k >= BEGIN_TYPED_REGIONS && k <= END_TYPED_REGIONS;
|
|
}
|
|
};
|
|
|
|
/// TypedValueRegion - An abstract class representing regions having a typed value.
|
|
class TypedValueRegion : public TypedRegion {
|
|
void anchor() override;
|
|
|
|
protected:
|
|
TypedValueRegion(const MemRegion* sReg, Kind k) : TypedRegion(sReg, k) {
|
|
assert(classof(this));
|
|
}
|
|
|
|
public:
|
|
virtual QualType getValueType() const = 0;
|
|
|
|
QualType getLocationType() const override {
|
|
// FIXME: We can possibly optimize this later to cache this value.
|
|
QualType T = getValueType();
|
|
ASTContext &ctx = getContext();
|
|
if (T->getAs<ObjCObjectType>())
|
|
return ctx.getObjCObjectPointerType(T);
|
|
return ctx.getPointerType(getValueType());
|
|
}
|
|
|
|
QualType getDesugaredValueType(ASTContext &Context) const {
|
|
QualType T = getValueType();
|
|
return T.getTypePtrOrNull() ? T.getDesugaredType(Context) : T;
|
|
}
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
unsigned k = R->getKind();
|
|
return k >= BEGIN_TYPED_VALUE_REGIONS && k <= END_TYPED_VALUE_REGIONS;
|
|
}
|
|
};
|
|
|
|
class CodeTextRegion : public TypedRegion {
|
|
void anchor() override;
|
|
|
|
protected:
|
|
CodeTextRegion(const MemSpaceRegion *sreg, Kind k) : TypedRegion(sreg, k) {
|
|
assert(classof(this));
|
|
}
|
|
|
|
public:
|
|
bool isBoundable() const override { return false; }
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
Kind k = R->getKind();
|
|
return k >= BEGIN_CODE_TEXT_REGIONS && k <= END_CODE_TEXT_REGIONS;
|
|
}
|
|
};
|
|
|
|
/// FunctionCodeRegion - A region that represents code texts of function.
|
|
class FunctionCodeRegion : public CodeTextRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const NamedDecl *FD;
|
|
|
|
FunctionCodeRegion(const NamedDecl *fd, const CodeSpaceRegion* sreg)
|
|
: CodeTextRegion(sreg, FunctionCodeRegionKind), FD(fd) {
|
|
assert(isa<ObjCMethodDecl>(fd) || isa<FunctionDecl>(fd));
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID& ID, const NamedDecl *FD,
|
|
const MemRegion*);
|
|
|
|
public:
|
|
QualType getLocationType() const override {
|
|
const ASTContext &Ctx = getContext();
|
|
if (const auto *D = dyn_cast<FunctionDecl>(FD)) {
|
|
return Ctx.getPointerType(D->getType());
|
|
}
|
|
|
|
assert(isa<ObjCMethodDecl>(FD));
|
|
assert(false && "Getting the type of ObjCMethod is not supported yet");
|
|
|
|
// TODO: We might want to return a different type here (ex: id (*ty)(...))
|
|
// depending on how it is used.
|
|
return {};
|
|
}
|
|
|
|
const NamedDecl *getDecl() const {
|
|
return FD;
|
|
}
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == FunctionCodeRegionKind;
|
|
}
|
|
};
|
|
|
|
/// BlockCodeRegion - A region that represents code texts of blocks (closures).
|
|
/// Blocks are represented with two kinds of regions. BlockCodeRegions
|
|
/// represent the "code", while BlockDataRegions represent instances of blocks,
|
|
/// which correspond to "code+data". The distinction is important, because
|
|
/// like a closure a block captures the values of externally referenced
|
|
/// variables.
|
|
class BlockCodeRegion : public CodeTextRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const BlockDecl *BD;
|
|
AnalysisDeclContext *AC;
|
|
CanQualType locTy;
|
|
|
|
BlockCodeRegion(const BlockDecl *bd, CanQualType lTy,
|
|
AnalysisDeclContext *ac, const CodeSpaceRegion* sreg)
|
|
: CodeTextRegion(sreg, BlockCodeRegionKind), BD(bd), AC(ac), locTy(lTy) {
|
|
assert(bd);
|
|
assert(ac);
|
|
assert(lTy->getTypePtr()->isBlockPointerType());
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID& ID, const BlockDecl *BD,
|
|
CanQualType, const AnalysisDeclContext*,
|
|
const MemRegion*);
|
|
|
|
public:
|
|
QualType getLocationType() const override {
|
|
return locTy;
|
|
}
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const BlockDecl *getDecl() const {
|
|
return BD;
|
|
}
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
AnalysisDeclContext *getAnalysisDeclContext() const { return AC; }
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == BlockCodeRegionKind;
|
|
}
|
|
};
|
|
|
|
/// BlockDataRegion - A region that represents a block instance.
|
|
/// Blocks are represented with two kinds of regions. BlockCodeRegions
|
|
/// represent the "code", while BlockDataRegions represent instances of blocks,
|
|
/// which correspond to "code+data". The distinction is important, because
|
|
/// like a closure a block captures the values of externally referenced
|
|
/// variables.
|
|
class BlockDataRegion : public TypedRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const BlockCodeRegion *BC;
|
|
const LocationContext *LC; // Can be null
|
|
unsigned BlockCount;
|
|
void *ReferencedVars = nullptr;
|
|
void *OriginalVars = nullptr;
|
|
|
|
BlockDataRegion(const BlockCodeRegion *bc, const LocationContext *lc,
|
|
unsigned count, const MemSpaceRegion *sreg)
|
|
: TypedRegion(sreg, BlockDataRegionKind), BC(bc), LC(lc),
|
|
BlockCount(count) {
|
|
assert(bc);
|
|
assert(bc->getDecl());
|
|
assert(lc);
|
|
assert(isa<GlobalImmutableSpaceRegion>(sreg) ||
|
|
isa<StackLocalsSpaceRegion>(sreg) ||
|
|
isa<UnknownSpaceRegion>(sreg));
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID&, const BlockCodeRegion *,
|
|
const LocationContext *, unsigned,
|
|
const MemRegion *);
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const BlockCodeRegion *getCodeRegion() const { return BC; }
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const BlockDecl *getDecl() const { return BC->getDecl(); }
|
|
|
|
QualType getLocationType() const override { return BC->getLocationType(); }
|
|
|
|
class referenced_vars_iterator {
|
|
const MemRegion * const *R;
|
|
const MemRegion * const *OriginalR;
|
|
|
|
public:
|
|
explicit referenced_vars_iterator(const MemRegion * const *r,
|
|
const MemRegion * const *originalR)
|
|
: R(r), OriginalR(originalR) {}
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const VarRegion *getCapturedRegion() const {
|
|
return cast<VarRegion>(*R);
|
|
}
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const VarRegion *getOriginalRegion() const {
|
|
return cast<VarRegion>(*OriginalR);
|
|
}
|
|
|
|
bool operator==(const referenced_vars_iterator &I) const {
|
|
assert((R == nullptr) == (I.R == nullptr));
|
|
return I.R == R;
|
|
}
|
|
|
|
bool operator!=(const referenced_vars_iterator &I) const {
|
|
assert((R == nullptr) == (I.R == nullptr));
|
|
return I.R != R;
|
|
}
|
|
|
|
referenced_vars_iterator &operator++() {
|
|
++R;
|
|
++OriginalR;
|
|
return *this;
|
|
}
|
|
|
|
// This isn't really a conventional iterator.
|
|
// We just implement the deref as a no-op for now to make range-based for
|
|
// loops work.
|
|
const referenced_vars_iterator &operator*() const { return *this; }
|
|
};
|
|
|
|
/// Return the original region for a captured region, if
|
|
/// one exists. It might return null.
|
|
const VarRegion *getOriginalRegion(const VarRegion *VR) const;
|
|
|
|
referenced_vars_iterator referenced_vars_begin() const;
|
|
referenced_vars_iterator referenced_vars_end() const;
|
|
llvm::iterator_range<referenced_vars_iterator> referenced_vars() const;
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == BlockDataRegionKind;
|
|
}
|
|
|
|
private:
|
|
void LazyInitializeReferencedVars();
|
|
std::pair<const VarRegion *, const VarRegion *>
|
|
getCaptureRegions(const VarDecl *VD);
|
|
};
|
|
|
|
/// SymbolicRegion - A special, "non-concrete" region. Unlike other region
|
|
/// classes, SymbolicRegion represents a region that serves as an alias for
|
|
/// either a real region, a NULL pointer, etc. It essentially is used to
|
|
/// map the concept of symbolic values into the domain of regions. Symbolic
|
|
/// regions do not need to be typed.
|
|
class SymbolicRegion : public SubRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const SymbolRef sym;
|
|
|
|
SymbolicRegion(const SymbolRef s, const MemSpaceRegion *sreg)
|
|
: SubRegion(sreg, SymbolicRegionKind), sym(s) {
|
|
// Because pointer arithmetic is represented by ElementRegion layers,
|
|
// the base symbol here should not contain any arithmetic.
|
|
assert(isa_and_nonnull<SymbolData>(s));
|
|
assert(s->getType()->isAnyPointerType() ||
|
|
s->getType()->isReferenceType() ||
|
|
s->getType()->isBlockPointerType());
|
|
assert(isa<UnknownSpaceRegion>(sreg) || isa<HeapSpaceRegion>(sreg) ||
|
|
isa<GlobalSystemSpaceRegion>(sreg));
|
|
}
|
|
|
|
public:
|
|
/// It might return null.
|
|
SymbolRef getSymbol() const { return sym; }
|
|
|
|
/// Gets the type of the wrapped symbol.
|
|
/// This type might not be accurate at all times - it's just our best guess.
|
|
/// Consider these cases:
|
|
/// void foo(void *data, char *str, base *obj) {...}
|
|
/// The type of the pointee of `data` is of course not `void`, yet that's our
|
|
/// best guess. `str` might point to any object and `obj` might point to some
|
|
/// derived instance. `TypedRegions` other hand are representing the cases
|
|
/// when we actually know their types.
|
|
QualType getPointeeStaticType() const {
|
|
return sym->getType()->getPointeeType();
|
|
}
|
|
|
|
bool isBoundable() const override { return true; }
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override;
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID& ID,
|
|
SymbolRef sym,
|
|
const MemRegion* superRegion);
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == SymbolicRegionKind;
|
|
}
|
|
};
|
|
|
|
/// StringRegion - Region associated with a StringLiteral.
|
|
class StringRegion : public TypedValueRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const StringLiteral *Str;
|
|
|
|
StringRegion(const StringLiteral *str, const GlobalInternalSpaceRegion *sreg)
|
|
: TypedValueRegion(sreg, StringRegionKind), Str(str) {
|
|
assert(str);
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID &ID,
|
|
const StringLiteral *Str,
|
|
const MemRegion *superRegion);
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const StringLiteral *getStringLiteral() const { return Str; }
|
|
|
|
QualType getValueType() const override { return Str->getType(); }
|
|
|
|
bool isBoundable() const override { return false; }
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override {
|
|
ProfileRegion(ID, Str, superRegion);
|
|
}
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == StringRegionKind;
|
|
}
|
|
};
|
|
|
|
/// The region associated with an ObjCStringLiteral.
|
|
class ObjCStringRegion : public TypedValueRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const ObjCStringLiteral *Str;
|
|
|
|
ObjCStringRegion(const ObjCStringLiteral *str,
|
|
const GlobalInternalSpaceRegion *sreg)
|
|
: TypedValueRegion(sreg, ObjCStringRegionKind), Str(str) {
|
|
assert(str);
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID &ID,
|
|
const ObjCStringLiteral *Str,
|
|
const MemRegion *superRegion);
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const ObjCStringLiteral *getObjCStringLiteral() const { return Str; }
|
|
|
|
QualType getValueType() const override { return Str->getType(); }
|
|
|
|
bool isBoundable() const override { return false; }
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override {
|
|
ProfileRegion(ID, Str, superRegion);
|
|
}
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == ObjCStringRegionKind;
|
|
}
|
|
};
|
|
|
|
/// CompoundLiteralRegion - A memory region representing a compound literal.
|
|
/// Compound literals are essentially temporaries that are stack allocated
|
|
/// or in the global constant pool.
|
|
class CompoundLiteralRegion : public TypedValueRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const CompoundLiteralExpr *CL;
|
|
|
|
CompoundLiteralRegion(const CompoundLiteralExpr *cl,
|
|
const MemSpaceRegion *sReg)
|
|
: TypedValueRegion(sReg, CompoundLiteralRegionKind), CL(cl) {
|
|
assert(cl);
|
|
assert(isa<GlobalInternalSpaceRegion>(sReg) ||
|
|
isa<StackLocalsSpaceRegion>(sReg));
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID& ID,
|
|
const CompoundLiteralExpr *CL,
|
|
const MemRegion* superRegion);
|
|
|
|
public:
|
|
QualType getValueType() const override { return CL->getType(); }
|
|
|
|
bool isBoundable() const override { return !CL->isFileScope(); }
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override;
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const CompoundLiteralExpr *getLiteralExpr() const { return CL; }
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == CompoundLiteralRegionKind;
|
|
}
|
|
};
|
|
|
|
class DeclRegion : public TypedValueRegion {
|
|
protected:
|
|
DeclRegion(const MemRegion *sReg, Kind k) : TypedValueRegion(sReg, k) {
|
|
assert(classof(this));
|
|
}
|
|
|
|
public:
|
|
// TODO what does this return?
|
|
virtual const ValueDecl *getDecl() const = 0;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
unsigned k = R->getKind();
|
|
return k >= BEGIN_DECL_REGIONS && k <= END_DECL_REGIONS;
|
|
}
|
|
};
|
|
|
|
class VarRegion : public DeclRegion {
|
|
friend class MemRegionManager;
|
|
|
|
protected:
|
|
// Constructors and protected methods.
|
|
VarRegion(const MemRegion *sReg, Kind k) : DeclRegion(sReg, k) {
|
|
// VarRegion appears in unknown space when it's a block variable as seen
|
|
// from a block using it, when this block is analyzed at top-level.
|
|
// Other block variables appear within block data regions,
|
|
// which, unlike everything else on this list, are not memory spaces.
|
|
assert(isa<GlobalsSpaceRegion>(sReg) || isa<StackSpaceRegion>(sReg) ||
|
|
isa<BlockDataRegion>(sReg) || isa<UnknownSpaceRegion>(sReg));
|
|
}
|
|
|
|
public:
|
|
// TODO what does this return?
|
|
const VarDecl *getDecl() const override = 0;
|
|
|
|
/// It might return null.
|
|
const StackFrameContext *getStackFrame() const;
|
|
|
|
QualType getValueType() const override {
|
|
// FIXME: We can cache this if needed.
|
|
return getDecl()->getType();
|
|
}
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
unsigned k = R->getKind();
|
|
return k >= BEGIN_VAR_REGIONS && k <= END_VAR_REGIONS;
|
|
}
|
|
};
|
|
|
|
class NonParamVarRegion : public VarRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const VarDecl *VD;
|
|
|
|
// Constructors and private methods.
|
|
NonParamVarRegion(const VarDecl *vd, const MemRegion *sReg)
|
|
: VarRegion(sReg, NonParamVarRegionKind), VD(vd) {
|
|
// VarRegion appears in unknown space when it's a block variable as seen
|
|
// from a block using it, when this block is analyzed at top-level.
|
|
// Other block variables appear within block data regions,
|
|
// which, unlike everything else on this list, are not memory spaces.
|
|
assert(isa<GlobalsSpaceRegion>(sReg) || isa<StackSpaceRegion>(sReg) ||
|
|
isa<BlockDataRegion>(sReg) || isa<UnknownSpaceRegion>(sReg));
|
|
assert(vd);
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID &ID, const VarDecl *VD,
|
|
const MemRegion *superRegion);
|
|
|
|
public:
|
|
void Profile(llvm::FoldingSetNodeID &ID) const override;
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const VarDecl *getDecl() const override { return VD; }
|
|
|
|
QualType getValueType() const override {
|
|
// FIXME: We can cache this if needed.
|
|
return getDecl()->getType();
|
|
}
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
bool canPrintPrettyAsExpr() const override;
|
|
|
|
void printPrettyAsExpr(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == NonParamVarRegionKind;
|
|
}
|
|
};
|
|
|
|
/// ParamVarRegion - Represents a region for paremters. Only parameters of the
|
|
/// function in the current stack frame are represented as `ParamVarRegion`s.
|
|
/// Parameters of top-level analyzed functions as well as captured paremeters
|
|
/// by lambdas and blocks are repesented as `VarRegion`s.
|
|
|
|
// FIXME: `ParamVarRegion` only supports parameters of functions, C++
|
|
// constructors, blocks and Objective-C methods with existing `Decl`. Upon
|
|
// implementing stack frame creations for functions without decl (functions
|
|
// passed by unknown function pointer) methods of `ParamVarRegion` must be
|
|
// updated.
|
|
class ParamVarRegion : public VarRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const Expr *OriginExpr;
|
|
unsigned Index;
|
|
|
|
ParamVarRegion(const Expr *OE, unsigned Idx, const MemRegion *SReg)
|
|
: VarRegion(SReg, ParamVarRegionKind), OriginExpr(OE), Index(Idx) {
|
|
assert(!cast<StackSpaceRegion>(SReg)->getStackFrame()->inTopFrame());
|
|
assert(OriginExpr);
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID &ID, const Expr *OE,
|
|
unsigned Idx, const MemRegion *SReg);
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const Expr *getOriginExpr() const { return OriginExpr; }
|
|
unsigned getIndex() const { return Index; }
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override;
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
QualType getValueType() const override;
|
|
|
|
/// TODO: What does this return?
|
|
const ParmVarDecl *getDecl() const override;
|
|
|
|
bool canPrintPrettyAsExpr() const override;
|
|
void printPrettyAsExpr(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == ParamVarRegionKind;
|
|
}
|
|
};
|
|
|
|
/// CXXThisRegion - Represents the region for the implicit 'this' parameter
|
|
/// in a call to a C++ method. This region doesn't represent the object
|
|
/// referred to by 'this', but rather 'this' itself.
|
|
class CXXThisRegion : public TypedValueRegion {
|
|
friend class MemRegionManager;
|
|
|
|
CXXThisRegion(const PointerType *thisPointerTy,
|
|
const StackArgumentsSpaceRegion *sReg)
|
|
: TypedValueRegion(sReg, CXXThisRegionKind),
|
|
ThisPointerTy(thisPointerTy) {
|
|
assert(ThisPointerTy->getPointeeType()->getAsCXXRecordDecl() &&
|
|
"Invalid region type!");
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID &ID,
|
|
const PointerType *PT,
|
|
const MemRegion *sReg);
|
|
|
|
public:
|
|
void Profile(llvm::FoldingSetNodeID &ID) const override;
|
|
|
|
QualType getValueType() const override {
|
|
return QualType(ThisPointerTy, 0);
|
|
}
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == CXXThisRegionKind;
|
|
}
|
|
|
|
private:
|
|
const PointerType *ThisPointerTy;
|
|
};
|
|
|
|
class FieldRegion : public DeclRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const FieldDecl *FD;
|
|
|
|
FieldRegion(const FieldDecl *fd, const SubRegion *sReg)
|
|
: DeclRegion(sReg, FieldRegionKind), FD(fd) {
|
|
assert(FD);
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID &ID, const FieldDecl *FD,
|
|
const MemRegion* superRegion) {
|
|
ID.AddInteger(static_cast<unsigned>(FieldRegionKind));
|
|
ID.AddPointer(FD);
|
|
ID.AddPointer(superRegion);
|
|
}
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const FieldDecl *getDecl() const override { return FD; }
|
|
|
|
void Profile(llvm::FoldingSetNodeID &ID) const override;
|
|
|
|
QualType getValueType() const override {
|
|
// FIXME: We can cache this if needed.
|
|
return getDecl()->getType();
|
|
}
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
bool canPrintPretty() const override;
|
|
void printPretty(raw_ostream &os) const override;
|
|
bool canPrintPrettyAsExpr() const override;
|
|
void printPrettyAsExpr(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == FieldRegionKind;
|
|
}
|
|
};
|
|
|
|
class ObjCIvarRegion : public DeclRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const ObjCIvarDecl *IVD;
|
|
|
|
ObjCIvarRegion(const ObjCIvarDecl *ivd, const SubRegion *sReg);
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID& ID, const ObjCIvarDecl *ivd,
|
|
const MemRegion* superRegion);
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const ObjCIvarDecl *getDecl() const override;
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override;
|
|
|
|
QualType getValueType() const override;
|
|
|
|
bool canPrintPrettyAsExpr() const override;
|
|
void printPrettyAsExpr(raw_ostream &os) const override;
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == ObjCIvarRegionKind;
|
|
}
|
|
};
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Auxiliary data classes for use with MemRegions.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
class RegionRawOffset {
|
|
friend class ElementRegion;
|
|
|
|
const MemRegion *Region;
|
|
CharUnits Offset;
|
|
|
|
RegionRawOffset(const MemRegion* reg, CharUnits offset = CharUnits::Zero())
|
|
: Region(reg), Offset(offset) {}
|
|
|
|
public:
|
|
// FIXME: Eventually support symbolic offsets.
|
|
CharUnits getOffset() const { return Offset; }
|
|
|
|
// It might return null.
|
|
const MemRegion *getRegion() const { return Region; }
|
|
|
|
void dumpToStream(raw_ostream &os) const;
|
|
void dump() const;
|
|
};
|
|
|
|
/// ElementRegion is used to represent both array elements and casts.
|
|
class ElementRegion : public TypedValueRegion {
|
|
friend class MemRegionManager;
|
|
|
|
QualType ElementType;
|
|
NonLoc Index;
|
|
|
|
ElementRegion(QualType elementType, NonLoc Idx, const SubRegion *sReg)
|
|
: TypedValueRegion(sReg, ElementRegionKind), ElementType(elementType),
|
|
Index(Idx) {
|
|
assert((!isa<nonloc::ConcreteInt>(Idx) ||
|
|
Idx.castAs<nonloc::ConcreteInt>().getValue().isSigned()) &&
|
|
"The index must be signed");
|
|
assert(!elementType.isNull() && !elementType->isVoidType() &&
|
|
"Invalid region type!");
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID& ID, QualType elementType,
|
|
SVal Idx, const MemRegion* superRegion);
|
|
|
|
public:
|
|
NonLoc getIndex() const { return Index; }
|
|
|
|
QualType getValueType() const override { return ElementType; }
|
|
|
|
QualType getElementType() const { return ElementType; }
|
|
|
|
/// Compute the offset within the array. The array might also be a subobject.
|
|
RegionRawOffset getAsArrayOffset() const;
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
void Profile(llvm::FoldingSetNodeID& ID) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == ElementRegionKind;
|
|
}
|
|
};
|
|
|
|
// C++ temporary object associated with an expression.
|
|
class CXXTempObjectRegion : public TypedValueRegion {
|
|
friend class MemRegionManager;
|
|
|
|
Expr const *Ex;
|
|
|
|
CXXTempObjectRegion(Expr const *E, MemSpaceRegion const *sReg)
|
|
: TypedValueRegion(sReg, CXXTempObjectRegionKind), Ex(E) {
|
|
assert(E);
|
|
assert(isa<StackLocalsSpaceRegion>(sReg));
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID &ID,
|
|
Expr const *E, const MemRegion *sReg);
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const Expr *getExpr() const { return Ex; }
|
|
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const StackFrameContext *getStackFrame() const;
|
|
|
|
QualType getValueType() const override { return Ex->getType(); }
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
void Profile(llvm::FoldingSetNodeID &ID) const override;
|
|
|
|
static bool classof(const MemRegion* R) {
|
|
return R->getKind() == CXXTempObjectRegionKind;
|
|
}
|
|
};
|
|
|
|
// C++ temporary object that have lifetime extended to lifetime of the
|
|
// variable. Usually they represent temporary bounds to reference variables.
|
|
class CXXLifetimeExtendedObjectRegion : public TypedValueRegion {
|
|
friend class MemRegionManager;
|
|
|
|
Expr const *Ex;
|
|
ValueDecl const *ExD;
|
|
|
|
CXXLifetimeExtendedObjectRegion(Expr const *E, ValueDecl const *D,
|
|
MemSpaceRegion const *sReg)
|
|
: TypedValueRegion(sReg, CXXLifetimeExtendedObjectRegionKind), Ex(E),
|
|
ExD(D) {
|
|
assert(E);
|
|
assert(D);
|
|
assert((isa<StackLocalsSpaceRegion, GlobalInternalSpaceRegion>(sReg)));
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID &ID, Expr const *E,
|
|
ValueDecl const *D, const MemRegion *sReg);
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const Expr *getExpr() const { return Ex; }
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const ValueDecl *getExtendingDecl() const { return ExD; }
|
|
/// It might return null.
|
|
const StackFrameContext *getStackFrame() const;
|
|
|
|
QualType getValueType() const override { return Ex->getType(); }
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
void Profile(llvm::FoldingSetNodeID &ID) const override;
|
|
|
|
static bool classof(const MemRegion *R) {
|
|
return R->getKind() == CXXLifetimeExtendedObjectRegionKind;
|
|
}
|
|
};
|
|
|
|
// CXXBaseObjectRegion represents a base object within a C++ object. It is
|
|
// identified by the base class declaration and the region of its parent object.
|
|
class CXXBaseObjectRegion : public TypedValueRegion {
|
|
friend class MemRegionManager;
|
|
|
|
llvm::PointerIntPair<const CXXRecordDecl *, 1, bool> Data;
|
|
|
|
CXXBaseObjectRegion(const CXXRecordDecl *RD, bool IsVirtual,
|
|
const SubRegion *SReg)
|
|
: TypedValueRegion(SReg, CXXBaseObjectRegionKind), Data(RD, IsVirtual) {
|
|
assert(RD);
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID &ID, const CXXRecordDecl *RD,
|
|
bool IsVirtual, const MemRegion *SReg);
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const CXXRecordDecl *getDecl() const { return Data.getPointer(); }
|
|
bool isVirtual() const { return Data.getInt(); }
|
|
|
|
QualType getValueType() const override;
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
void Profile(llvm::FoldingSetNodeID &ID) const override;
|
|
|
|
bool canPrintPrettyAsExpr() const override;
|
|
|
|
void printPrettyAsExpr(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *region) {
|
|
return region->getKind() == CXXBaseObjectRegionKind;
|
|
}
|
|
};
|
|
|
|
// CXXDerivedObjectRegion represents a derived-class object that surrounds
|
|
// a C++ object. It is identified by the derived class declaration and the
|
|
// region of its parent object. It is a bit counter-intuitive (but not otherwise
|
|
// unseen) that this region represents a larger segment of memory that its
|
|
// super-region.
|
|
class CXXDerivedObjectRegion : public TypedValueRegion {
|
|
friend class MemRegionManager;
|
|
|
|
const CXXRecordDecl *DerivedD;
|
|
|
|
CXXDerivedObjectRegion(const CXXRecordDecl *DerivedD, const SubRegion *SReg)
|
|
: TypedValueRegion(SReg, CXXDerivedObjectRegionKind), DerivedD(DerivedD) {
|
|
assert(DerivedD);
|
|
// In case of a concrete region, it should always be possible to model
|
|
// the base-to-derived cast by undoing a previous derived-to-base cast,
|
|
// otherwise the cast is most likely ill-formed.
|
|
assert(SReg->getSymbolicBase() &&
|
|
"Should have unwrapped a base region instead!");
|
|
}
|
|
|
|
static void ProfileRegion(llvm::FoldingSetNodeID &ID, const CXXRecordDecl *RD,
|
|
const MemRegion *SReg);
|
|
|
|
public:
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL
|
|
const CXXRecordDecl *getDecl() const { return DerivedD; }
|
|
|
|
QualType getValueType() const override;
|
|
|
|
void dumpToStream(raw_ostream &os) const override;
|
|
|
|
void Profile(llvm::FoldingSetNodeID &ID) const override;
|
|
|
|
bool canPrintPrettyAsExpr() const override;
|
|
|
|
void printPrettyAsExpr(raw_ostream &os) const override;
|
|
|
|
static bool classof(const MemRegion *region) {
|
|
return region->getKind() == CXXDerivedObjectRegionKind;
|
|
}
|
|
};
|
|
|
|
template<typename RegionTy>
|
|
const RegionTy* MemRegion::getAs() const {
|
|
if (const auto *RT = dyn_cast<RegionTy>(this))
|
|
return RT;
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
template <typename RegionTy>
|
|
LLVM_ATTRIBUTE_RETURNS_NONNULL const RegionTy *MemRegion::castAs() const {
|
|
return cast<RegionTy>(this);
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// MemRegionManager - Factory object for creating regions.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
class MemRegionManager {
|
|
ASTContext &Ctx;
|
|
llvm::BumpPtrAllocator& A;
|
|
|
|
llvm::FoldingSet<MemRegion> Regions;
|
|
|
|
GlobalInternalSpaceRegion *InternalGlobals = nullptr;
|
|
GlobalSystemSpaceRegion *SystemGlobals = nullptr;
|
|
GlobalImmutableSpaceRegion *ImmutableGlobals = nullptr;
|
|
|
|
llvm::DenseMap<const StackFrameContext *, StackLocalsSpaceRegion *>
|
|
StackLocalsSpaceRegions;
|
|
llvm::DenseMap<const StackFrameContext *, StackArgumentsSpaceRegion *>
|
|
StackArgumentsSpaceRegions;
|
|
llvm::DenseMap<const CodeTextRegion *, StaticGlobalSpaceRegion *>
|
|
StaticsGlobalSpaceRegions;
|
|
|
|
HeapSpaceRegion *heap = nullptr;
|
|
UnknownSpaceRegion *unknown = nullptr;
|
|
CodeSpaceRegion *code = nullptr;
|
|
|
|
public:
|
|
MemRegionManager(ASTContext &c, llvm::BumpPtrAllocator &a) : Ctx(c), A(a) {}
|
|
~MemRegionManager();
|
|
|
|
ASTContext &getContext() { return Ctx; }
|
|
const ASTContext &getContext() const { return Ctx; }
|
|
|
|
llvm::BumpPtrAllocator &getAllocator() { return A; }
|
|
|
|
/// \returns The static size in bytes of the region \p MR.
|
|
/// \note The region \p MR must be a 'SubRegion'.
|
|
DefinedOrUnknownSVal getStaticSize(const MemRegion *MR,
|
|
SValBuilder &SVB) const;
|
|
|
|
/// getStackLocalsRegion - Retrieve the memory region associated with the
|
|
/// specified stack frame.
|
|
const StackLocalsSpaceRegion *
|
|
getStackLocalsRegion(const StackFrameContext *STC);
|
|
|
|
/// getStackArgumentsRegion - Retrieve the memory region associated with
|
|
/// function/method arguments of the specified stack frame.
|
|
const StackArgumentsSpaceRegion *
|
|
getStackArgumentsRegion(const StackFrameContext *STC);
|
|
|
|
/// getGlobalsRegion - Retrieve the memory region associated with
|
|
/// global variables.
|
|
const GlobalsSpaceRegion *getGlobalsRegion(
|
|
MemRegion::Kind K = MemRegion::GlobalInternalSpaceRegionKind,
|
|
const CodeTextRegion *R = nullptr);
|
|
|
|
/// getHeapRegion - Retrieve the memory region associated with the
|
|
/// generic "heap".
|
|
const HeapSpaceRegion *getHeapRegion();
|
|
|
|
/// getUnknownRegion - Retrieve the memory region associated with unknown
|
|
/// memory space.
|
|
const UnknownSpaceRegion *getUnknownRegion();
|
|
|
|
const CodeSpaceRegion *getCodeRegion();
|
|
|
|
/// getAllocaRegion - Retrieve a region associated with a call to alloca().
|
|
const AllocaRegion *getAllocaRegion(const Expr *Ex, unsigned Cnt,
|
|
const LocationContext *LC);
|
|
|
|
/// getCompoundLiteralRegion - Retrieve the region associated with a
|
|
/// given CompoundLiteral.
|
|
const CompoundLiteralRegion*
|
|
getCompoundLiteralRegion(const CompoundLiteralExpr *CL,
|
|
const LocationContext *LC);
|
|
|
|
/// getCXXThisRegion - Retrieve the [artificial] region associated with the
|
|
/// parameter 'this'.
|
|
const CXXThisRegion *getCXXThisRegion(QualType thisPointerTy,
|
|
const LocationContext *LC);
|
|
|
|
/// Retrieve or create a "symbolic" memory region.
|
|
/// If no memory space is specified, `UnknownSpaceRegion` will be used.
|
|
const SymbolicRegion *
|
|
getSymbolicRegion(SymbolRef Sym, const MemSpaceRegion *MemSpace = nullptr);
|
|
|
|
/// Return a unique symbolic region belonging to heap memory space.
|
|
const SymbolicRegion *getSymbolicHeapRegion(SymbolRef sym);
|
|
|
|
const StringRegion *getStringRegion(const StringLiteral *Str);
|
|
|
|
const ObjCStringRegion *getObjCStringRegion(const ObjCStringLiteral *Str);
|
|
|
|
/// getVarRegion - Retrieve or create the memory region associated with
|
|
/// a specified VarDecl and LocationContext.
|
|
const VarRegion *getVarRegion(const VarDecl *VD, const LocationContext *LC);
|
|
|
|
/// getVarRegion - Retrieve or create the memory region associated with
|
|
/// a specified VarDecl and LocationContext.
|
|
const NonParamVarRegion *getNonParamVarRegion(const VarDecl *VD,
|
|
const MemRegion *superR);
|
|
|
|
/// getParamVarRegion - Retrieve or create the memory region
|
|
/// associated with a specified CallExpr, Index and LocationContext.
|
|
const ParamVarRegion *getParamVarRegion(const Expr *OriginExpr,
|
|
unsigned Index,
|
|
const LocationContext *LC);
|
|
|
|
/// getElementRegion - Retrieve the memory region associated with the
|
|
/// associated element type, index, and super region.
|
|
const ElementRegion *getElementRegion(QualType elementType, NonLoc Idx,
|
|
const SubRegion *superRegion,
|
|
const ASTContext &Ctx);
|
|
|
|
const ElementRegion *getElementRegionWithSuper(const ElementRegion *ER,
|
|
const SubRegion *superRegion) {
|
|
return getElementRegion(ER->getElementType(), ER->getIndex(),
|
|
superRegion, ER->getContext());
|
|
}
|
|
|
|
/// getFieldRegion - Retrieve or create the memory region associated with
|
|
/// a specified FieldDecl. 'superRegion' corresponds to the containing
|
|
/// memory region (which typically represents the memory representing
|
|
/// a structure or class).
|
|
const FieldRegion *getFieldRegion(const FieldDecl *fd,
|
|
const SubRegion* superRegion);
|
|
|
|
const FieldRegion *getFieldRegionWithSuper(const FieldRegion *FR,
|
|
const SubRegion *superRegion) {
|
|
return getFieldRegion(FR->getDecl(), superRegion);
|
|
}
|
|
|
|
/// getObjCIvarRegion - Retrieve or create the memory region associated with
|
|
/// a specified Objective-c instance variable. 'superRegion' corresponds
|
|
/// to the containing region (which typically represents the Objective-C
|
|
/// object).
|
|
const ObjCIvarRegion *getObjCIvarRegion(const ObjCIvarDecl *ivd,
|
|
const SubRegion* superRegion);
|
|
|
|
const CXXTempObjectRegion *getCXXTempObjectRegion(Expr const *Ex,
|
|
LocationContext const *LC);
|
|
|
|
/// Create a CXXLifetimeExtendedObjectRegion for temporaries which are
|
|
/// lifetime-extended by local references.
|
|
const CXXLifetimeExtendedObjectRegion *
|
|
getCXXLifetimeExtendedObjectRegion(Expr const *Ex, ValueDecl const *VD,
|
|
LocationContext const *LC);
|
|
|
|
/// Create a CXXLifetimeExtendedObjectRegion for temporaries which are
|
|
/// lifetime-extended by *static* references.
|
|
/// This differs from \ref getCXXLifetimeExtendedObjectRegion(Expr const *,
|
|
/// ValueDecl const *, LocationContext const *) in the super-region used.
|
|
const CXXLifetimeExtendedObjectRegion *
|
|
getCXXStaticLifetimeExtendedObjectRegion(const Expr *Ex, ValueDecl const *VD);
|
|
|
|
/// Create a CXXBaseObjectRegion with the given base class for region
|
|
/// \p Super.
|
|
///
|
|
/// The type of \p Super is assumed be a class deriving from \p BaseClass.
|
|
const CXXBaseObjectRegion *
|
|
getCXXBaseObjectRegion(const CXXRecordDecl *BaseClass, const SubRegion *Super,
|
|
bool IsVirtual);
|
|
|
|
/// Create a CXXBaseObjectRegion with the same CXXRecordDecl but a different
|
|
/// super region.
|
|
const CXXBaseObjectRegion *
|
|
getCXXBaseObjectRegionWithSuper(const CXXBaseObjectRegion *baseReg,
|
|
const SubRegion *superRegion) {
|
|
return getCXXBaseObjectRegion(baseReg->getDecl(), superRegion,
|
|
baseReg->isVirtual());
|
|
}
|
|
|
|
/// Create a CXXDerivedObjectRegion with the given derived class for region
|
|
/// \p Super. This should not be used for casting an existing
|
|
/// CXXBaseObjectRegion back to the derived type; instead, CXXBaseObjectRegion
|
|
/// should be removed.
|
|
const CXXDerivedObjectRegion *
|
|
getCXXDerivedObjectRegion(const CXXRecordDecl *BaseClass,
|
|
const SubRegion *Super);
|
|
|
|
const FunctionCodeRegion *getFunctionCodeRegion(const NamedDecl *FD);
|
|
const BlockCodeRegion *getBlockCodeRegion(const BlockDecl *BD,
|
|
CanQualType locTy,
|
|
AnalysisDeclContext *AC);
|
|
|
|
/// getBlockDataRegion - Get the memory region associated with an instance
|
|
/// of a block. Unlike many other MemRegions, the LocationContext*
|
|
/// argument is allowed to be NULL for cases where we have no known
|
|
/// context.
|
|
const BlockDataRegion *getBlockDataRegion(const BlockCodeRegion *bc,
|
|
const LocationContext *lc,
|
|
unsigned blockCount);
|
|
|
|
private:
|
|
template <typename RegionTy, typename SuperTy,
|
|
typename Arg1Ty>
|
|
RegionTy* getSubRegion(const Arg1Ty arg1,
|
|
const SuperTy* superRegion);
|
|
|
|
template <typename RegionTy, typename SuperTy,
|
|
typename Arg1Ty, typename Arg2Ty>
|
|
RegionTy* getSubRegion(const Arg1Ty arg1, const Arg2Ty arg2,
|
|
const SuperTy* superRegion);
|
|
|
|
template <typename RegionTy, typename SuperTy,
|
|
typename Arg1Ty, typename Arg2Ty, typename Arg3Ty>
|
|
RegionTy* getSubRegion(const Arg1Ty arg1, const Arg2Ty arg2,
|
|
const Arg3Ty arg3,
|
|
const SuperTy* superRegion);
|
|
|
|
template <typename REG>
|
|
const REG* LazyAllocate(REG*& region);
|
|
|
|
template <typename REG, typename ARG>
|
|
const REG* LazyAllocate(REG*& region, ARG a);
|
|
};
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Out-of-line member definitions.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
inline ASTContext &MemRegion::getContext() const {
|
|
return getMemRegionManager().getContext();
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Means for storing region/symbol handling traits.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
/// Information about invalidation for a particular region/symbol.
|
|
class RegionAndSymbolInvalidationTraits {
|
|
using StorageTypeForKinds = unsigned char;
|
|
|
|
llvm::DenseMap<const MemRegion *, StorageTypeForKinds> MRTraitsMap;
|
|
llvm::DenseMap<SymbolRef, StorageTypeForKinds> SymTraitsMap;
|
|
|
|
using const_region_iterator =
|
|
llvm::DenseMap<const MemRegion *, StorageTypeForKinds>::const_iterator;
|
|
using const_symbol_iterator =
|
|
llvm::DenseMap<SymbolRef, StorageTypeForKinds>::const_iterator;
|
|
|
|
public:
|
|
/// Describes different invalidation traits.
|
|
enum InvalidationKinds {
|
|
/// Tells that a region's contents is not changed.
|
|
TK_PreserveContents = 0x1,
|
|
|
|
/// Suppress pointer-escaping of a region.
|
|
TK_SuppressEscape = 0x2,
|
|
|
|
// Do not invalidate super region.
|
|
TK_DoNotInvalidateSuperRegion = 0x4,
|
|
|
|
/// When applied to a MemSpaceRegion, indicates the entire memory space
|
|
/// should be invalidated.
|
|
TK_EntireMemSpace = 0x8
|
|
|
|
// Do not forget to extend StorageTypeForKinds if number of traits exceed
|
|
// the number of bits StorageTypeForKinds can store.
|
|
};
|
|
|
|
void setTrait(SymbolRef Sym, InvalidationKinds IK);
|
|
void setTrait(const MemRegion *MR, InvalidationKinds IK);
|
|
bool hasTrait(SymbolRef Sym, InvalidationKinds IK) const;
|
|
bool hasTrait(const MemRegion *MR, InvalidationKinds IK) const;
|
|
};
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Pretty-printing regions.
|
|
//===----------------------------------------------------------------------===//
|
|
inline raw_ostream &operator<<(raw_ostream &os, const MemRegion *R) {
|
|
R->dumpToStream(os);
|
|
return os;
|
|
}
|
|
|
|
} // namespace ento
|
|
|
|
} // namespace clang
|
|
|
|
#endif // LLVM_CLANG_STATICANALYZER_CORE_PATHSENSITIVE_MEMREGION_H
|