2008-11-13 03:21:30 +08:00
|
|
|
//== Environment.cpp - Map from Stmt* to Locations/Values -------*- C++ -*--==//
|
2008-07-09 05:46:56 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file defined the Environment and EnvironmentManager classes.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
2010-03-28 05:19:47 +08:00
|
|
|
|
2012-01-28 20:06:22 +08:00
|
|
|
#include "clang/AST/ExprCXX.h"
|
2011-08-20 14:23:25 +08:00
|
|
|
#include "clang/AST/ExprObjC.h"
|
2017-09-07 05:45:03 +08:00
|
|
|
#include "clang/Analysis/AnalysisDeclContext.h"
|
2010-03-28 05:19:47 +08:00
|
|
|
#include "clang/Analysis/CFG.h"
|
2011-08-16 06:09:50 +08:00
|
|
|
#include "clang/StaticAnalyzer/Core/PathSensitive/ProgramState.h"
|
2012-12-02 01:12:56 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2008-07-09 05:46:56 +08:00
|
|
|
|
|
|
|
using namespace clang;
|
2010-12-23 15:20:52 +08:00
|
|
|
using namespace ento;
|
2008-07-09 05:46:56 +08:00
|
|
|
|
2012-10-18 03:35:44 +08:00
|
|
|
static const Expr *ignoreTransparentExprs(const Expr *E) {
|
|
|
|
E = E->IgnoreParens();
|
|
|
|
|
|
|
|
switch (E->getStmtClass()) {
|
|
|
|
case Stmt::OpaqueValueExprClass:
|
|
|
|
E = cast<OpaqueValueExpr>(E)->getSourceExpr();
|
|
|
|
break;
|
|
|
|
case Stmt::ExprWithCleanupsClass:
|
|
|
|
E = cast<ExprWithCleanups>(E)->getSubExpr();
|
|
|
|
break;
|
|
|
|
case Stmt::CXXBindTemporaryExprClass:
|
|
|
|
E = cast<CXXBindTemporaryExpr>(E)->getSubExpr();
|
|
|
|
break;
|
|
|
|
case Stmt::SubstNonTypeTemplateParmExprClass:
|
|
|
|
E = cast<SubstNonTypeTemplateParmExpr>(E)->getReplacement();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// This is the base case: we can't look through more than we already have.
|
|
|
|
return E;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ignoreTransparentExprs(E);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const Stmt *ignoreTransparentExprs(const Stmt *S) {
|
|
|
|
if (const Expr *E = dyn_cast<Expr>(S))
|
|
|
|
return ignoreTransparentExprs(E);
|
|
|
|
return S;
|
|
|
|
}
|
|
|
|
|
|
|
|
EnvironmentEntry::EnvironmentEntry(const Stmt *S, const LocationContext *L)
|
|
|
|
: std::pair<const Stmt *,
|
|
|
|
const StackFrameContext *>(ignoreTransparentExprs(S),
|
2014-05-27 10:45:47 +08:00
|
|
|
L ? L->getCurrentStackFrame()
|
|
|
|
: nullptr) {}
|
2012-10-18 03:35:44 +08:00
|
|
|
|
2012-01-07 06:09:28 +08:00
|
|
|
SVal Environment::lookupExpr(const EnvironmentEntry &E) const {
|
2010-12-06 07:36:15 +08:00
|
|
|
const SVal* X = ExprBindings.lookup(E);
|
|
|
|
if (X) {
|
|
|
|
SVal V = *X;
|
|
|
|
return V;
|
|
|
|
}
|
|
|
|
return UnknownVal();
|
|
|
|
}
|
|
|
|
|
2012-01-07 06:09:28 +08:00
|
|
|
SVal Environment::getSVal(const EnvironmentEntry &Entry,
|
2012-10-13 13:05:20 +08:00
|
|
|
SValBuilder& svalBuilder) const {
|
2012-10-18 03:35:44 +08:00
|
|
|
const Stmt *S = Entry.getStmt();
|
2012-01-07 06:09:28 +08:00
|
|
|
const LocationContext *LCtx = Entry.getLocationContext();
|
2012-10-18 03:35:44 +08:00
|
|
|
|
|
|
|
switch (S->getStmtClass()) {
|
|
|
|
case Stmt::CXXBindTemporaryExprClass:
|
|
|
|
case Stmt::ExprWithCleanupsClass:
|
|
|
|
case Stmt::GenericSelectionExprClass:
|
|
|
|
case Stmt::OpaqueValueExprClass:
|
|
|
|
case Stmt::ParenExprClass:
|
|
|
|
case Stmt::SubstNonTypeTemplateParmExprClass:
|
|
|
|
llvm_unreachable("Should have been handled by ignoreTransparentExprs");
|
|
|
|
|
|
|
|
case Stmt::AddrLabelExprClass:
|
[analyzer] Consolidate constant evaluation logic in SValBuilder.
Previously, this was scattered across Environment (literal expressions),
ExprEngine (default arguments), and RegionStore (global constants). The
former special-cased several kinds of simple constant expressions, while
the latter two deferred to the AST's constant evaluator.
Now, these are all unified as SValBuilder::getConstantVal(). To keep
Environment fast, the special cases for simple constant expressions have
been left in, but the main benefits are that (a) unusual constants like
ObjCStringLiterals now work as default arguments and global constant
initializers, and (b) we're not duplicating code between ExprEngine and
RegionStore.
This actually caught a bug in our test suite, which is awesome: we stop
tracking allocated memory if it's passed as an argument along with some
kind of callback, but not if the callback is 0. We were testing this in
a case where the callback parameter had a default value, but that value
was 0. After this change, the analyzer now (correctly) flags that as a
leak!
<rdar://problem/13773117>
llvm-svn: 180894
2013-05-02 07:10:44 +08:00
|
|
|
case Stmt::CharacterLiteralClass:
|
2012-10-18 03:35:44 +08:00
|
|
|
case Stmt::CXXBoolLiteralExprClass:
|
|
|
|
case Stmt::CXXScalarValueInitExprClass:
|
[analyzer] Consolidate constant evaluation logic in SValBuilder.
Previously, this was scattered across Environment (literal expressions),
ExprEngine (default arguments), and RegionStore (global constants). The
former special-cased several kinds of simple constant expressions, while
the latter two deferred to the AST's constant evaluator.
Now, these are all unified as SValBuilder::getConstantVal(). To keep
Environment fast, the special cases for simple constant expressions have
been left in, but the main benefits are that (a) unusual constants like
ObjCStringLiterals now work as default arguments and global constant
initializers, and (b) we're not duplicating code between ExprEngine and
RegionStore.
This actually caught a bug in our test suite, which is awesome: we stop
tracking allocated memory if it's passed as an argument along with some
kind of callback, but not if the callback is 0. We were testing this in
a case where the callback parameter had a default value, but that value
was 0. After this change, the analyzer now (correctly) flags that as a
leak!
<rdar://problem/13773117>
llvm-svn: 180894
2013-05-02 07:10:44 +08:00
|
|
|
case Stmt::ImplicitValueInitExprClass:
|
2012-10-18 03:35:44 +08:00
|
|
|
case Stmt::IntegerLiteralClass:
|
|
|
|
case Stmt::ObjCBoolLiteralExprClass:
|
|
|
|
case Stmt::CXXNullPtrLiteralExprClass:
|
[analyzer] Consolidate constant evaluation logic in SValBuilder.
Previously, this was scattered across Environment (literal expressions),
ExprEngine (default arguments), and RegionStore (global constants). The
former special-cased several kinds of simple constant expressions, while
the latter two deferred to the AST's constant evaluator.
Now, these are all unified as SValBuilder::getConstantVal(). To keep
Environment fast, the special cases for simple constant expressions have
been left in, but the main benefits are that (a) unusual constants like
ObjCStringLiterals now work as default arguments and global constant
initializers, and (b) we're not duplicating code between ExprEngine and
RegionStore.
This actually caught a bug in our test suite, which is awesome: we stop
tracking allocated memory if it's passed as an argument along with some
kind of callback, but not if the callback is 0. We were testing this in
a case where the callback parameter had a default value, but that value
was 0. After this change, the analyzer now (correctly) flags that as a
leak!
<rdar://problem/13773117>
llvm-svn: 180894
2013-05-02 07:10:44 +08:00
|
|
|
case Stmt::ObjCStringLiteralClass:
|
|
|
|
case Stmt::StringLiteralClass:
|
2015-09-23 03:33:15 +08:00
|
|
|
case Stmt::TypeTraitExprClass:
|
[analyzer] Consolidate constant evaluation logic in SValBuilder.
Previously, this was scattered across Environment (literal expressions),
ExprEngine (default arguments), and RegionStore (global constants). The
former special-cased several kinds of simple constant expressions, while
the latter two deferred to the AST's constant evaluator.
Now, these are all unified as SValBuilder::getConstantVal(). To keep
Environment fast, the special cases for simple constant expressions have
been left in, but the main benefits are that (a) unusual constants like
ObjCStringLiterals now work as default arguments and global constant
initializers, and (b) we're not duplicating code between ExprEngine and
RegionStore.
This actually caught a bug in our test suite, which is awesome: we stop
tracking allocated memory if it's passed as an argument along with some
kind of callback, but not if the callback is 0. We were testing this in
a case where the callback parameter had a default value, but that value
was 0. After this change, the analyzer now (correctly) flags that as a
leak!
<rdar://problem/13773117>
llvm-svn: 180894
2013-05-02 07:10:44 +08:00
|
|
|
// Known constants; defer to SValBuilder.
|
|
|
|
return svalBuilder.getConstantVal(cast<Expr>(S)).getValue();
|
2012-10-18 03:35:44 +08:00
|
|
|
|
|
|
|
case Stmt::ReturnStmtClass: {
|
|
|
|
const ReturnStmt *RS = cast<ReturnStmt>(S);
|
|
|
|
if (const Expr *RE = RS->getRetValue())
|
|
|
|
return getSVal(EnvironmentEntry(RE, LCtx), svalBuilder);
|
2015-09-08 11:50:52 +08:00
|
|
|
return UndefinedVal();
|
2012-10-18 03:35:44 +08:00
|
|
|
}
|
2015-09-08 11:50:52 +08:00
|
|
|
|
2012-10-18 03:35:44 +08:00
|
|
|
// Handle all other Stmt* using a lookup.
|
|
|
|
default:
|
[analyzer] Consolidate constant evaluation logic in SValBuilder.
Previously, this was scattered across Environment (literal expressions),
ExprEngine (default arguments), and RegionStore (global constants). The
former special-cased several kinds of simple constant expressions, while
the latter two deferred to the AST's constant evaluator.
Now, these are all unified as SValBuilder::getConstantVal(). To keep
Environment fast, the special cases for simple constant expressions have
been left in, but the main benefits are that (a) unusual constants like
ObjCStringLiterals now work as default arguments and global constant
initializers, and (b) we're not duplicating code between ExprEngine and
RegionStore.
This actually caught a bug in our test suite, which is awesome: we stop
tracking allocated memory if it's passed as an argument along with some
kind of callback, but not if the callback is 0. We were testing this in
a case where the callback parameter had a default value, but that value
was 0. After this change, the analyzer now (correctly) flags that as a
leak!
<rdar://problem/13773117>
llvm-svn: 180894
2013-05-02 07:10:44 +08:00
|
|
|
return lookupExpr(EnvironmentEntry(S, LCtx));
|
2008-07-11 01:19:18 +08:00
|
|
|
}
|
|
|
|
}
|
2008-07-09 05:46:56 +08:00
|
|
|
|
2012-01-07 06:09:28 +08:00
|
|
|
Environment EnvironmentManager::bindExpr(Environment Env,
|
|
|
|
const EnvironmentEntry &E,
|
|
|
|
SVal V,
|
|
|
|
bool Invalidate) {
|
2009-09-09 23:08:12 +08:00
|
|
|
if (V.isUnknown()) {
|
2008-07-11 01:19:18 +08:00
|
|
|
if (Invalidate)
|
2012-01-07 06:09:28 +08:00
|
|
|
return Environment(F.remove(Env.ExprBindings, E));
|
2008-07-11 01:19:18 +08:00
|
|
|
else
|
|
|
|
return Env;
|
|
|
|
}
|
2012-01-07 06:09:28 +08:00
|
|
|
return Environment(F.add(Env.ExprBindings, E, V));
|
2008-07-11 01:19:18 +08:00
|
|
|
}
|
2008-08-21 01:08:29 +08:00
|
|
|
|
2009-02-14 11:16:10 +08:00
|
|
|
namespace {
|
2015-08-14 06:50:09 +08:00
|
|
|
class MarkLiveCallback final : public SymbolVisitor {
|
2009-02-14 11:16:10 +08:00
|
|
|
SymbolReaper &SymReaper;
|
|
|
|
public:
|
2009-09-09 23:08:12 +08:00
|
|
|
MarkLiveCallback(SymbolReaper &symreaper) : SymReaper(symreaper) {}
|
2014-03-15 12:29:04 +08:00
|
|
|
bool VisitSymbol(SymbolRef sym) override {
|
2012-02-21 08:46:29 +08:00
|
|
|
SymReaper.markLive(sym);
|
|
|
|
return true;
|
|
|
|
}
|
2014-03-15 12:29:04 +08:00
|
|
|
bool VisitMemRegion(const MemRegion *R) override {
|
2012-02-21 08:46:29 +08:00
|
|
|
SymReaper.markLive(R);
|
|
|
|
return true;
|
|
|
|
}
|
2009-02-14 11:16:10 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
2011-01-15 04:34:15 +08:00
|
|
|
// removeDeadBindings:
|
2009-03-12 15:54:17 +08:00
|
|
|
// - Remove subexpression bindings.
|
|
|
|
// - Remove dead block expression bindings.
|
|
|
|
// - Keep live block expression bindings:
|
2009-09-09 23:08:12 +08:00
|
|
|
// - Mark their reachable symbols live in SymbolReaper,
|
2009-03-12 15:54:17 +08:00
|
|
|
// see ScanReachableSymbols.
|
|
|
|
// - Mark the region in DRoots if the binding is a loc::MemRegionVal.
|
2009-09-09 23:08:12 +08:00
|
|
|
Environment
|
2011-01-15 04:34:15 +08:00
|
|
|
EnvironmentManager::removeDeadBindings(Environment Env,
|
2009-08-27 09:39:13 +08:00
|
|
|
SymbolReaper &SymReaper,
|
2012-01-27 05:29:00 +08:00
|
|
|
ProgramStateRef ST) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-08-27 09:39:13 +08:00
|
|
|
// We construct a new Environment object entirely, as this is cheaper than
|
|
|
|
// individually removing all the subexpression bindings (which will greatly
|
|
|
|
// outnumber block-level expression bindings).
|
2010-03-05 12:45:36 +08:00
|
|
|
Environment NewEnv = getInitialEnvironment();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-09-23 02:10:41 +08:00
|
|
|
MarkLiveCallback CB(SymReaper);
|
|
|
|
ScanReachableSymbols RSScaner(ST, CB);
|
|
|
|
|
2012-01-07 06:09:28 +08:00
|
|
|
llvm::ImmutableMapRef<EnvironmentEntry,SVal>
|
2011-09-24 03:14:09 +08:00
|
|
|
EBMapRef(NewEnv.ExprBindings.getRootWithoutRetain(),
|
|
|
|
F.getTreeFactory());
|
|
|
|
|
2008-08-21 01:08:29 +08:00
|
|
|
// Iterate over the block-expr bindings.
|
2009-09-09 23:08:12 +08:00
|
|
|
for (Environment::iterator I = Env.begin(), E = Env.end();
|
2008-08-21 01:08:29 +08:00
|
|
|
I != E; ++I) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2012-01-07 06:09:28 +08:00
|
|
|
const EnvironmentEntry &BlkExpr = I.getKey();
|
2010-04-05 21:16:29 +08:00
|
|
|
const SVal &X = I.getData();
|
|
|
|
|
2012-01-07 06:09:28 +08:00
|
|
|
if (SymReaper.isLive(BlkExpr.getStmt(), BlkExpr.getLocationContext())) {
|
2009-08-27 09:39:13 +08:00
|
|
|
// Copy the binding to the new map.
|
2011-09-24 03:14:09 +08:00
|
|
|
EBMapRef = EBMapRef.add(BlkExpr, X);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-14 11:16:10 +08:00
|
|
|
// Mark all symbols in the block expr's value live.
|
2011-09-23 02:10:41 +08:00
|
|
|
RSScaner.scan(X);
|
2009-08-27 09:39:13 +08:00
|
|
|
continue;
|
2012-11-16 03:11:27 +08:00
|
|
|
} else {
|
|
|
|
SymExpr::symbol_iterator SI = X.symbol_begin(), SE = X.symbol_end();
|
|
|
|
for (; SI != SE; ++SI)
|
|
|
|
SymReaper.maybeDead(*SI);
|
2008-08-21 01:08:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-24 03:14:09 +08:00
|
|
|
NewEnv.ExprBindings = EBMapRef.asImmutableMap();
|
2009-08-27 09:39:13 +08:00
|
|
|
return NewEnv;
|
2008-08-21 01:08:29 +08:00
|
|
|
}
|
2012-01-07 06:09:28 +08:00
|
|
|
|
|
|
|
void Environment::print(raw_ostream &Out, const char *NL,
|
2018-02-09 06:24:38 +08:00
|
|
|
const char *Sep, const LocationContext *WithLC) const {
|
|
|
|
if (ExprBindings.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!WithLC) {
|
|
|
|
// Find the freshest location context.
|
|
|
|
llvm::SmallPtrSet<const LocationContext *, 16> FoundContexts;
|
|
|
|
for (auto I : *this) {
|
|
|
|
const LocationContext *LC = I.first.getLocationContext();
|
|
|
|
if (FoundContexts.count(LC) == 0) {
|
|
|
|
// This context is fresher than all other contexts so far.
|
|
|
|
WithLC = LC;
|
|
|
|
for (const LocationContext *LCI = LC; LCI; LCI = LCI->getParent())
|
|
|
|
FoundContexts.insert(LCI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-01-07 06:09:28 +08:00
|
|
|
|
2018-02-09 06:24:38 +08:00
|
|
|
assert(WithLC);
|
2015-09-08 11:50:52 +08:00
|
|
|
|
2018-02-09 06:24:38 +08:00
|
|
|
LangOptions LO; // FIXME.
|
|
|
|
PrintingPolicy PP(LO);
|
2015-09-08 11:50:52 +08:00
|
|
|
|
2018-02-09 06:24:38 +08:00
|
|
|
Out << NL << NL << "Expressions by stack frame:" << NL;
|
|
|
|
WithLC->dumpStack(Out, "", NL, Sep, [&](const LocationContext *LC) {
|
|
|
|
for (auto I : ExprBindings) {
|
|
|
|
if (I.first.getLocationContext() != LC)
|
|
|
|
continue;
|
2014-06-10 06:53:25 +08:00
|
|
|
|
2018-02-09 06:24:38 +08:00
|
|
|
const Stmt *S = I.first.getStmt();
|
|
|
|
assert(S != nullptr && "Expected non-null Stmt");
|
|
|
|
|
|
|
|
Out << "(" << (const void *)LC << ',' << (const void *)S << ") ";
|
|
|
|
S->printPretty(Out, nullptr, PP);
|
|
|
|
Out << " : " << I.second << NL;
|
|
|
|
}
|
|
|
|
});
|
2012-01-07 06:09:28 +08:00
|
|
|
}
|