[analyzer][MallocChecker] When modeling realloc-like functions, don't early return if the argument is symbolic

The very essence of MallocChecker lies in 2 overload sets: the FreeMemAux
functions and the MallocMemAux functions. The former houses most of the error
checking as well (aside from leaks), such as incorrect deallocation. There, we
check whether the argument's MemSpaceRegion is the heap or unknown, and if it
isn't, we know we encountered a bug (aside from a corner case patched by
@balazske in D76830), as specified by MEM34-C.

In ReallocMemAux, which really is the combination of  FreeMemAux and
MallocMemAux, we incorrectly early returned if the memory argument of realloc is
non-symbolic. The problem is, one of the cases where this happens when we know
precisely what the region is, like an array, as demonstrated in the test file.
So, lets get rid of this false negative :^)

Side note, I dislike the warning message and the associated checker name, but
I'll address it in a later patch.

Differential Revision: https://reviews.llvm.org/D79415
This commit is contained in:
Kirstóf Umann 2020-05-05 14:55:37 +02:00
parent 62adfed30a
commit 2e5e42d4ae
2 changed files with 23 additions and 6 deletions

View File

@ -2371,13 +2371,7 @@ MallocChecker::ReallocMemAux(CheckerContext &C, const CallExpr *CE,
if (PrtIsNull && SizeIsZero) if (PrtIsNull && SizeIsZero)
return State; return State;
// Get the from and to pointer symbols as in toPtr = realloc(fromPtr, size).
assert(!PrtIsNull); assert(!PrtIsNull);
SymbolRef FromPtr = arg0Val.getAsSymbol();
SVal RetVal = C.getSVal(CE);
SymbolRef ToPtr = RetVal.getAsSymbol();
if (!FromPtr || !ToPtr)
return nullptr;
bool IsKnownToBeAllocated = false; bool IsKnownToBeAllocated = false;
@ -2406,6 +2400,14 @@ MallocChecker::ReallocMemAux(CheckerContext &C, const CallExpr *CE,
else if (!IsKnownToBeAllocated) else if (!IsKnownToBeAllocated)
Kind = OAR_DoNotTrackAfterFailure; Kind = OAR_DoNotTrackAfterFailure;
// Get the from and to pointer symbols as in toPtr = realloc(fromPtr, size).
SymbolRef FromPtr = arg0Val.getAsSymbol();
SVal RetVal = C.getSVal(CE);
SymbolRef ToPtr = RetVal.getAsSymbol();
assert(FromPtr && ToPtr &&
"By this point, FreeMemAux and MallocMemAux should have checked "
"whether the argument or the return value is symbolic!");
// Record the info about the reallocated symbol so that we could properly // Record the info about the reallocated symbol so that we could properly
// process failed reallocation. // process failed reallocation.
stateRealloc = stateRealloc->set<ReallocPairs>(ToPtr, stateRealloc = stateRealloc->set<ReallocPairs>(ToPtr,

View File

@ -1828,6 +1828,21 @@ void testCStyleListItems(struct ListInfo *list) {
list_add(list, &x->li); // will free 'x'. list_add(list, &x->li); // will free 'x'.
} }
// MEM34-C. Only free memory allocated dynamically
// Second non-compliant example.
// https://wiki.sei.cmu.edu/confluence/display/c/MEM34-C.+Only+free+memory+allocated+dynamically
enum { BUFSIZE = 256 };
void MEM34_C(void) {
char buf[BUFSIZE];
char *p = (char *)realloc(buf, 2 * BUFSIZE);
// expected-warning@-1{{Argument to realloc() is the address of the local \
variable 'buf', which is not memory allocated by malloc() [unix.Malloc]}}
if (p == NULL) {
/* Handle error */
}
}
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
// False negatives. // False negatives.