forked from OSchip/llvm-project
[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:
parent
62adfed30a
commit
2e5e42d4ae
|
@ -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,
|
||||||
|
|
|
@ -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.
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue