Skip to content

Commit

Permalink
bpo-31626: Fix _PyObject_DebugReallocApi() (#4310)
Browse files Browse the repository at this point in the history
_PyObject_DebugReallocApi() now calls Py_FatalError() if realloc()
fails to shrink a memory block.

Call Py_FatalError() because _PyObject_DebugReallocApi() erased freed
bytes *before* realloc(), expecting that realloc() *cannot* fail to
shrink a memory block.
  • Loading branch information
vstinner committed Nov 24, 2017
1 parent 35d9983 commit ed4743a
Show file tree
Hide file tree
Showing 2 changed files with 16 additions and 7 deletions.
@@ -0,0 +1,3 @@
When Python is built in debug mode, the memory debug hooks now fail with a
fatal error if realloc() fails to shrink a memory block, because the debug
hook just erased freed bytes without keeping a copy of them.
20 changes: 13 additions & 7 deletions Objects/obmalloc.c
Expand Up @@ -101,7 +101,7 @@ static int running_on_valgrind = -1;
*
* For small requests, the allocator sub-allocates <Big> blocks of memory.
* Requests greater than SMALL_REQUEST_THRESHOLD bytes are routed to the
* system's allocator.
* system's allocator.
*
* Small requests are grouped in size classes spaced 8 bytes apart, due
* to the required valid alignment of the returned address. Requests of
Expand Down Expand Up @@ -134,7 +134,7 @@ static int running_on_valgrind = -1;
* 65-72 72 8
* ... ... ...
* 497-504 504 62
* 505-512 512 63
* 505-512 512 63
*
* 0, SMALL_REQUEST_THRESHOLD + 1 and up: routed to the underlying
* allocator.
Expand Down Expand Up @@ -176,7 +176,7 @@ static int running_on_valgrind = -1;
* Although not required, for better performance and space efficiency,
* it is recommended that SMALL_REQUEST_THRESHOLD is set to a power of 2.
*/
#define SMALL_REQUEST_THRESHOLD 512
#define SMALL_REQUEST_THRESHOLD 512
#define NB_SMALL_SIZE_CLASSES (SMALL_REQUEST_THRESHOLD / ALIGNMENT)

/*
Expand Down Expand Up @@ -209,7 +209,7 @@ static int running_on_valgrind = -1;
* usually an address range reservation for <Big> bytes, unless all pages within
* this space are referenced subsequently. So malloc'ing big blocks and not
* using them does not mean "wasting memory". It's an addressable range
* wastage...
* wastage...
*
* Arenas are allocated with mmap() on systems supporting anonymous memory
* mappings to reduce heap fragmentation.
Expand Down Expand Up @@ -619,7 +619,7 @@ new_arena(void)
#else
address = malloc(ARENA_SIZE);
err = (address == 0);
#endif
#endif
if (err) {
/* The allocation failed: return NULL after putting the
* arenaobj back.
Expand Down Expand Up @@ -1552,7 +1552,7 @@ _PyObject_DebugReallocApi(char api, void *p, size_t nbytes)
/* overflow: can't represent total as a size_t */
return NULL;

if (nbytes < original_nbytes) {
if (nbytes <= original_nbytes) {
/* shrinking: mark old extra memory dead */
memset(q + nbytes, DEADBYTE, original_nbytes - nbytes + 2*SST);
}
Expand All @@ -1562,8 +1562,14 @@ _PyObject_DebugReallocApi(char api, void *p, size_t nbytes)
* but we live with that.
*/
q = (uchar *)PyObject_Realloc(q - 2*SST, total);
if (q == NULL)
if (q == NULL) {
if (nbytes <= original_nbytes) {
/* bpo-31626: the memset() above expects that realloc never fails
on shrinking a memory block. */
Py_FatalError("Shrinking reallocation failed");
}
return NULL;
}

write_size_t(q, nbytes);
assert(q[SST] == (uchar)api);
Expand Down

0 comments on commit ed4743a

Please sign in to comment.