Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add runtime-enabled heap debugging capabilities #18172

Merged
merged 4 commits into from
Mar 31, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
228 changes: 226 additions & 2 deletions Zend/zend_alloc.c
Original file line number Diff line number Diff line change
Expand Up @@ -305,7 +305,17 @@ struct _zend_mm_heap {
size_t (*_gc)(void);
void (*_shutdown)(bool full, bool silent);
} custom_heap;
HashTable *tracked_allocs;
union {
HashTable *tracked_allocs;
struct {
bool poison_alloc;
uint8_t poison_alloc_value;
bool poison_free;
uint8_t poison_free_value;
uint8_t padding;
bool check_freelists_on_shutdown;
} debug;
};
#endif
pid_t pid;
zend_random_bytes_insecure_state rand_state;
Expand Down Expand Up @@ -2389,8 +2399,19 @@ static void zend_mm_check_leaks(zend_mm_heap *heap)
#if ZEND_MM_CUSTOM
static void *tracked_malloc(size_t size ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC);
static void tracked_free_all(zend_mm_heap *heap);
static void *poison_malloc(size_t size ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC);
#endif

static void zend_mm_check_freelists(zend_mm_heap *heap)
{
for (uint32_t bin_num = 0; bin_num < ZEND_MM_BINS; bin_num++) {
zend_mm_free_slot *slot = heap->free_slot[bin_num];
while (slot) {
slot = zend_mm_get_next_free_slot(heap, bin_num, slot);
}
}
}

ZEND_API void zend_mm_shutdown(zend_mm_heap *heap, bool full, bool silent)
{
zend_mm_chunk *p;
Expand Down Expand Up @@ -2555,8 +2576,9 @@ ZEND_API size_t ZEND_FASTCALL _zend_mm_block_size(zend_mm_heap *heap, void *ptr
if (size_zv) {
return Z_LVAL_P(size_zv);
}
} else if (heap->custom_heap._malloc != poison_malloc) {
return 0;
}
return 0;
}
#endif
return zend_mm_size(heap, ptr ZEND_FILE_LINE_RELAY_CC ZEND_FILE_LINE_ORIG_RELAY_CC);
Expand Down Expand Up @@ -3021,6 +3043,200 @@ static void tracked_free_all(zend_mm_heap *heap) {
}
#endif

static void* poison_malloc(size_t size ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC)
{
zend_mm_heap *heap = AG(mm_heap);

if (SIZE_MAX - heap->debug.padding * 2 < size) {
zend_mm_panic("Integer overflow in memory allocation");
}
size += heap->debug.padding * 2;

void *ptr = zend_mm_alloc_heap(heap, size ZEND_FILE_LINE_RELAY_CC ZEND_FILE_LINE_ORIG_RELAY_CC);

if (EXPECTED(ptr)) {
if (heap->debug.poison_alloc) {
memset(ptr, heap->debug.poison_alloc_value, size);
}

ptr = (char*)ptr + heap->debug.padding;
}

return ptr;
}

static void poison_free(void *ptr ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC)
{
zend_mm_heap *heap = AG(mm_heap);

if (EXPECTED(ptr)) {
/* zend_mm_shutdown() will try to free the heap when custom handlers
* are installed */
if (UNEXPECTED(ptr == heap)) {
return;
}

ptr = (char*)ptr - heap->debug.padding;

size_t size = zend_mm_size(heap, ptr ZEND_FILE_LINE_RELAY_CC ZEND_FILE_LINE_ORIG_RELAY_CC);

if (heap->debug.poison_free) {
memset(ptr, heap->debug.poison_free_value, size);
}
}

zend_mm_free_heap(heap, ptr ZEND_FILE_LINE_RELAY_CC ZEND_FILE_LINE_ORIG_RELAY_CC);
}

static void* poison_realloc(void *ptr, size_t size ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC)
{
zend_mm_heap *heap = AG(mm_heap);

void *new = poison_malloc(size ZEND_FILE_LINE_RELAY_CC ZEND_FILE_LINE_ORIG_RELAY_CC);

if (ptr) {
/* Determine the size of the old allocation from the unpadded pointer. */
size_t oldsize = zend_mm_size(heap, (char*)ptr - heap->debug.padding ZEND_FILE_LINE_RELAY_CC ZEND_FILE_LINE_ORIG_RELAY_CC);

/* Remove the padding size to determine the size that is available to the user. */
oldsize -= (2 * heap->debug.padding);

#if ZEND_DEBUG
oldsize -= sizeof(zend_mm_debug_info);
#endif

memcpy(new, ptr, MIN(oldsize, size));
poison_free(ptr ZEND_FILE_LINE_RELAY_CC ZEND_FILE_LINE_ORIG_RELAY_CC);
}

return new;
}

static size_t poison_gc(void)
{
zend_mm_heap *heap = AG(mm_heap);

void* (*_malloc)(size_t ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC);
void (*_free)(void* ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC);
void* (*_realloc)(void*, size_t ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC);
size_t (*_gc)(void);
void (*_shutdown)(bool, bool);

zend_mm_get_custom_handlers_ex(heap, &_malloc, &_free, &_realloc, &_gc, &_shutdown);
zend_mm_set_custom_handlers_ex(heap, NULL, NULL, NULL, NULL, NULL);

size_t collected = zend_mm_gc(heap);

zend_mm_set_custom_handlers_ex(heap, _malloc, _free, _realloc, _gc, _shutdown);

return collected;
}

static void poison_shutdown(bool full, bool silent)
{
zend_mm_heap *heap = AG(mm_heap);

void* (*_malloc)(size_t ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC);
void (*_free)(void* ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC);
void* (*_realloc)(void*, size_t ZEND_FILE_LINE_DC ZEND_FILE_LINE_ORIG_DC);
size_t (*_gc)(void);
void (*_shutdown)(bool, bool);

zend_mm_get_custom_handlers_ex(heap, &_malloc, &_free, &_realloc, &_gc, &_shutdown);
zend_mm_set_custom_handlers_ex(heap, NULL, NULL, NULL, NULL, NULL);

if (heap->debug.check_freelists_on_shutdown) {
zend_mm_check_freelists(heap);
}

zend_mm_shutdown(heap, full, silent);

if (!full) {
zend_mm_set_custom_handlers_ex(heap, _malloc, _free, _realloc, _gc, _shutdown);
}
}

static void poison_enable(zend_mm_heap *heap, char *parameters)
{
char *tmp = parameters;
char *end = tmp + strlen(tmp);

/* Trim heading/trailing whitespaces */
while (*tmp == ' ' || *tmp == '\t' || *tmp == '\n') {
tmp++;
}
while (end != tmp && (*(end-1) == ' ' || *(end-1) == '\t' || *(end-1) == '\n')) {
end--;
}

if (tmp == end) {
return;
}

while (1) {
char *key = tmp;

tmp = memchr(tmp, '=', end - tmp);
if (!tmp) {
size_t key_len = end - key;
fprintf(stderr, "Unexpected EOF after ZEND_MM_DEBUG parameter '%.*s', expected '='\n",
(int)key_len, key);
return;
}

size_t key_len = tmp - key;
char *value = tmp + 1;

if (key_len == strlen("poison_alloc")
&& !memcmp(key, "poison_alloc", key_len)) {

heap->debug.poison_alloc = true;
heap->debug.poison_alloc_value = (uint8_t) ZEND_STRTOUL(value, &tmp, 0);

} else if (key_len == strlen("poison_free")
&& !memcmp(key, "poison_free", key_len)) {

heap->debug.poison_free = true;
heap->debug.poison_free_value = (uint8_t) ZEND_STRTOUL(value, &tmp, 0);

} else if (key_len == strlen("padding")
&& !memcmp(key, "padding", key_len)) {

uint8_t padding = ZEND_STRTOUL(value, &tmp, 0);
if (ZEND_MM_ALIGNED_SIZE(padding) != padding) {
fprintf(stderr, "ZEND_MM_DEBUG padding must be a multiple of %u, %u given\n",
(unsigned int)ZEND_MM_ALIGNMENT,
(unsigned int)padding);
return;
}
heap->debug.padding = padding;

} else if (key_len == strlen("check_freelists_on_shutdown")
&& !memcmp(key, "check_freelists_on_shutdown", key_len)) {

heap->debug.check_freelists_on_shutdown = (bool) ZEND_STRTOUL(value, &tmp, 0);

} else {
fprintf(stderr, "Unknown ZEND_MM_DEBUG parameter: '%.*s'\n",
(int)key_len, key);
return;
}

if (tmp == end) {
break;
}
if (*tmp != ',') {
fprintf(stderr, "Unexpected '%c' after value of ZEND_MM_DEBUG parameter '%.*s', expected ','\n",
*tmp, (int)key_len, key);
return;
}
tmp++;
}

zend_mm_set_custom_handlers_ex(heap, poison_malloc, poison_free,
poison_realloc, poison_gc, poison_shutdown);
}

static void alloc_globals_ctor(zend_alloc_globals *alloc_globals)
{
char *tmp;
Expand Down Expand Up @@ -3057,6 +3273,14 @@ static void alloc_globals_ctor(zend_alloc_globals *alloc_globals)
zend_mm_use_huge_pages = true;
}
alloc_globals->mm_heap = zend_mm_init();

#if ZEND_MM_CUSTOM
ZEND_ASSERT(!alloc_globals->mm_heap->tracked_allocs);
tmp = getenv("ZEND_MM_DEBUG");
if (tmp) {
poison_enable(alloc_globals->mm_heap, tmp);
}
#endif
}

#ifdef ZTS
Expand Down
14 changes: 14 additions & 0 deletions ext/zend_test/test.c
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
*/

#include "zend_modules.h"
#include "zend_types.h"
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
Expand Down Expand Up @@ -182,6 +183,19 @@ static ZEND_FUNCTION(zend_leak_variable)
Z_ADDREF_P(zv);
}

static ZEND_FUNCTION(zend_delref)
{
zval *zv;

if (zend_parse_parameters(ZEND_NUM_ARGS(), "z", &zv) == FAILURE) {
RETURN_THROWS();
}

Z_TRY_DELREF_P(zv);

RETURN_NULL();
}

/* Tests Z_PARAM_OBJ_OR_STR */
static ZEND_FUNCTION(zend_string_or_object)
{
Expand Down
2 changes: 2 additions & 0 deletions ext/zend_test/test.stub.php
Original file line number Diff line number Diff line change
Expand Up @@ -235,6 +235,8 @@ function zend_leak_variable(mixed $variable): void {}

function zend_leak_bytes(int $bytes = 3): void {}

function zend_delref(mixed $variable): void {}

function zend_string_or_object(object|string $param): object|string {}

function zend_string_or_object_or_null(object|string|null $param): object|string|null {}
Expand Down
6 changes: 5 additions & 1 deletion ext/zend_test/test_arginfo.h

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 6 additions & 0 deletions ext/zend_test/tests/opline_dangling.phpt
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,12 @@ https://github.com/php/php-src/pull/12758
zend_test
--ENV--
USE_ZEND_ALLOC=1
--SKIPIF--
<?php
if (getenv("ZEND_MM_DEBUG")) {
die("skip zend_test.observe_opline_in_zendmm not compatible with ZEND_MM_DEBUG");
}
?>
--INI--
zend_test.observe_opline_in_zendmm=1
--FILE--
Expand Down
6 changes: 6 additions & 0 deletions ext/zend_test/tests/opline_dangling_02.phpt
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,12 @@ possible segfault in `ZEND_FUNC_GET_ARGS`
zend_test
--ENV--
USE_ZEND_ALLOC=1
--SKIPIF--
<?php
if (getenv("ZEND_MM_DEBUG")) {
die("skip zend_test.observe_opline_in_zendmm not compatible with ZEND_MM_DEBUG");
}
?>
--INI--
zend_test.observe_opline_in_zendmm=1
--FILE--
Expand Down
Loading