forked from OSchip/llvm-project
[analyzer]Print field region even when the base region is not printable
llvm-svn: 179395
This commit is contained in:
parent
f8bcfbdd5f
commit
6cea7d9e5e
|
@ -169,6 +169,8 @@ public:
|
|||
/// \brief Print the region for use in diagnostics.
|
||||
virtual void printPretty(raw_ostream &os) const;
|
||||
|
||||
virtual void printPrettyNoQuotes(raw_ostream &os) const;
|
||||
|
||||
Kind getKind() const { return kind; }
|
||||
|
||||
template<typename RegionTy> const RegionTy* getAs() const;
|
||||
|
@ -875,7 +877,8 @@ public:
|
|||
}
|
||||
|
||||
bool canPrintPretty() const;
|
||||
void printPretty(raw_ostream &os) const;
|
||||
|
||||
void printPrettyNoQuotes(raw_ostream &os) const;
|
||||
};
|
||||
|
||||
/// CXXThisRegion - Represents the region for the implicit 'this' parameter
|
||||
|
@ -937,6 +940,7 @@ public:
|
|||
|
||||
bool canPrintPretty() const;
|
||||
void printPretty(raw_ostream &os) const;
|
||||
void printPrettyNoQuotes(raw_ostream &os) const;
|
||||
};
|
||||
|
||||
class ObjCIvarRegion : public DeclRegion {
|
||||
|
@ -953,7 +957,7 @@ public:
|
|||
QualType getValueType() const;
|
||||
|
||||
bool canPrintPretty() const;
|
||||
void printPretty(raw_ostream &os) const;
|
||||
void printPrettyNoQuotes(raw_ostream &os) const;
|
||||
|
||||
void dumpToStream(raw_ostream &os) const;
|
||||
|
||||
|
|
|
@ -1606,9 +1606,8 @@ void MallocChecker::reportLeak(SymbolRef Sym, ExplodedNode *N,
|
|||
SmallString<200> buf;
|
||||
llvm::raw_svector_ostream os(buf);
|
||||
if (Region && Region->canPrintPretty()) {
|
||||
os << "Potential leak of memory pointed to by '";
|
||||
os << "Potential leak of memory pointed to by ";
|
||||
Region->printPretty(os);
|
||||
os << '\'';
|
||||
} else {
|
||||
os << "Potential memory leak";
|
||||
}
|
||||
|
|
|
@ -307,9 +307,9 @@ public:
|
|||
if (LValue) {
|
||||
if (const MemRegion *MR = LValue->getAsRegion()) {
|
||||
if (MR->canPrintPretty()) {
|
||||
Out << " (reference to '";
|
||||
Out << " (reference to ";
|
||||
MR->printPretty(Out);
|
||||
Out << "')";
|
||||
Out << ")";
|
||||
}
|
||||
}
|
||||
} else {
|
||||
|
@ -545,13 +545,9 @@ PathDiagnosticPiece *FindLastStoreBRVisitor::VisitNode(const ExplodedNode *Succ,
|
|||
}
|
||||
|
||||
if (action) {
|
||||
if (!R)
|
||||
return 0;
|
||||
|
||||
os << '\'';
|
||||
R->printPretty(os);
|
||||
os << "' ";
|
||||
|
||||
os << " ";
|
||||
|
||||
if (V.getAs<loc::ConcreteInt>()) {
|
||||
bool b = false;
|
||||
if (R->isBoundable()) {
|
||||
|
@ -606,10 +602,8 @@ PathDiagnosticPiece *FindLastStoreBRVisitor::VisitNode(const ExplodedNode *Succ,
|
|||
|
||||
// Printed parameter indexes are 1-based, not 0-based.
|
||||
unsigned Idx = Param->getFunctionScopeIndex() + 1;
|
||||
os << " via " << Idx << llvm::getOrdinalSuffix(Idx) << " parameter '";
|
||||
|
||||
os << " via " << Idx << llvm::getOrdinalSuffix(Idx) << " parameter ";
|
||||
R->printPretty(os);
|
||||
os << '\'';
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -637,9 +631,7 @@ PathDiagnosticPiece *FindLastStoreBRVisitor::VisitNode(const ExplodedNode *Succ,
|
|||
else
|
||||
os << "Value assigned to ";
|
||||
|
||||
os << '\'';
|
||||
R->printPretty(os);
|
||||
os << '\'';
|
||||
}
|
||||
|
||||
// Construct a new PathDiagnosticPiece.
|
||||
|
|
|
@ -559,6 +559,15 @@ bool MemRegion::canPrintPretty() const {
|
|||
}
|
||||
|
||||
void MemRegion::printPretty(raw_ostream &os) const {
|
||||
assert(canPrintPretty() && "This region cannot be printed pretty.");
|
||||
os << "'";
|
||||
printPrettyNoQuotes(os);
|
||||
os << "'";
|
||||
return;
|
||||
}
|
||||
|
||||
void MemRegion::printPrettyNoQuotes(raw_ostream &os) const {
|
||||
assert(canPrintPretty() && "This region cannot be printed pretty.");
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -566,7 +575,7 @@ bool VarRegion::canPrintPretty() const {
|
|||
return true;
|
||||
}
|
||||
|
||||
void VarRegion::printPretty(raw_ostream &os) const {
|
||||
void VarRegion::printPrettyNoQuotes(raw_ostream &os) const {
|
||||
os << getDecl()->getName();
|
||||
}
|
||||
|
||||
|
@ -574,17 +583,32 @@ bool ObjCIvarRegion::canPrintPretty() const {
|
|||
return true;
|
||||
}
|
||||
|
||||
void ObjCIvarRegion::printPretty(raw_ostream &os) const {
|
||||
void ObjCIvarRegion::printPrettyNoQuotes(raw_ostream &os) const {
|
||||
os << getDecl()->getName();
|
||||
}
|
||||
|
||||
bool FieldRegion::canPrintPretty() const {
|
||||
return superRegion->canPrintPretty();
|
||||
return true;
|
||||
}
|
||||
|
||||
void FieldRegion::printPrettyNoQuotes(raw_ostream &os) const {
|
||||
if (superRegion->canPrintPretty()) {
|
||||
superRegion->printPrettyNoQuotes(os);
|
||||
os << "." << getDecl()->getName();
|
||||
} else {
|
||||
os << "field " << "\'" << getDecl()->getName() << "'";
|
||||
}
|
||||
}
|
||||
|
||||
void FieldRegion::printPretty(raw_ostream &os) const {
|
||||
superRegion->printPretty(os);
|
||||
os << "." << getDecl()->getName();
|
||||
if (superRegion->canPrintPretty()) {
|
||||
os << "\'";
|
||||
printPrettyNoQuotes(os);
|
||||
os << "'";
|
||||
} else {
|
||||
printPrettyNoQuotes(os);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
|
|
@ -107,6 +107,20 @@ void testUseOfNullPointer() {
|
|||
// expected-note@-4 {{Calling 'usePointer'}}
|
||||
}
|
||||
|
||||
struct X { char *p; };
|
||||
|
||||
void setFieldToNull(struct X *x) {
|
||||
x->p = 0; // expected-note {{Null pointer value stored to field 'p'}}
|
||||
}
|
||||
|
||||
int testSetFieldToNull(struct X *x) {
|
||||
setFieldToNull(x); // expected-note {{Calling 'setFieldToNull'}}
|
||||
// expected-note@-1{{Returning from 'setFieldToNull'}}
|
||||
return *x->p;
|
||||
// expected-warning@-1 {{Dereference of null pointer (loaded from field 'p')}}
|
||||
// expected-note@-2 {{Dereference of null pointer (loaded from field 'p')}}
|
||||
}
|
||||
|
||||
// CHECK: <key>diagnostics</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
|
@ -2637,4 +2651,253 @@ void testUseOfNullPointer() {
|
|||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>path</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>kind</key><string>event</string>
|
||||
// CHECK-NEXT: <key>location</key>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>117</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>3</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <key>ranges</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>117</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>3</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>117</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>19</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: <key>depth</key><integer>0</integer>
|
||||
// CHECK-NEXT: <key>extended_message</key>
|
||||
// CHECK-NEXT: <string>Calling 'setFieldToNull'</string>
|
||||
// CHECK-NEXT: <key>message</key>
|
||||
// CHECK-NEXT: <string>Calling 'setFieldToNull'</string>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>kind</key><string>event</string>
|
||||
// CHECK-NEXT: <key>location</key>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>112</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>1</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <key>depth</key><integer>1</integer>
|
||||
// CHECK-NEXT: <key>extended_message</key>
|
||||
// CHECK-NEXT: <string>Entered call from 'testSetFieldToNull'</string>
|
||||
// CHECK-NEXT: <key>message</key>
|
||||
// CHECK-NEXT: <string>Entered call from 'testSetFieldToNull'</string>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>kind</key><string>control</string>
|
||||
// CHECK-NEXT: <key>edges</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>start</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>112</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>1</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>112</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>4</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: <key>end</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>113</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>2</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>113</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>2</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>kind</key><string>event</string>
|
||||
// CHECK-NEXT: <key>location</key>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>113</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>2</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <key>ranges</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>113</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>2</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>113</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>9</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: <key>depth</key><integer>1</integer>
|
||||
// CHECK-NEXT: <key>extended_message</key>
|
||||
// CHECK-NEXT: <string>Null pointer value stored to field 'p'</string>
|
||||
// CHECK-NEXT: <key>message</key>
|
||||
// CHECK-NEXT: <string>Null pointer value stored to field 'p'</string>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>kind</key><string>event</string>
|
||||
// CHECK-NEXT: <key>location</key>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>117</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>3</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <key>ranges</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>117</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>3</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>117</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>19</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: <key>depth</key><integer>0</integer>
|
||||
// CHECK-NEXT: <key>extended_message</key>
|
||||
// CHECK-NEXT: <string>Returning from 'setFieldToNull'</string>
|
||||
// CHECK-NEXT: <key>message</key>
|
||||
// CHECK-NEXT: <string>Returning from 'setFieldToNull'</string>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>kind</key><string>control</string>
|
||||
// CHECK-NEXT: <key>edges</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>start</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>117</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>3</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>117</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>16</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: <key>end</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>119</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>3</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>119</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>8</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>kind</key><string>control</string>
|
||||
// CHECK-NEXT: <key>edges</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>start</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>119</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>3</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>119</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>8</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: <key>end</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>119</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>10</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>119</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>10</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>kind</key><string>event</string>
|
||||
// CHECK-NEXT: <key>location</key>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>119</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>10</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <key>ranges</key>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <array>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>119</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>14</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>119</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>14</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: <key>depth</key><integer>0</integer>
|
||||
// CHECK-NEXT: <key>extended_message</key>
|
||||
// CHECK-NEXT: <string>Dereference of null pointer (loaded from field 'p')</string>
|
||||
// CHECK-NEXT: <key>message</key>
|
||||
// CHECK-NEXT: <string>Dereference of null pointer (loaded from field 'p')</string>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
// CHECK-NEXT: <key>description</key><string>Dereference of null pointer (loaded from field 'p')</string>
|
||||
// CHECK-NEXT: <key>category</key><string>Logic error</string>
|
||||
// CHECK-NEXT: <key>type</key><string>Dereference of null pointer</string>
|
||||
// CHECK-NEXT: <key>issue_context_kind</key><string>function</string>
|
||||
// CHECK-NEXT: <key>issue_context</key><string>testSetFieldToNull</string>
|
||||
// CHECK-NEXT: <key>issue_hash</key><string>3</string>
|
||||
// CHECK-NEXT: <key>location</key>
|
||||
// CHECK-NEXT: <dict>
|
||||
// CHECK-NEXT: <key>line</key><integer>119</integer>
|
||||
// CHECK-NEXT: <key>col</key><integer>10</integer>
|
||||
// CHECK-NEXT: <key>file</key><integer>0</integer>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </dict>
|
||||
// CHECK-NEXT: </array>
|
||||
|
|
Loading…
Reference in New Issue