summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorNikita Popov <nikita.ppv@gmail.com>2019-05-23 12:29:08 +0200
committerNikita Popov <nikita.ppv@gmail.com>2019-05-23 13:40:52 +0200
commit6f9dfd947302f9a0d2fa6a78bf385b1ca7dafdf3 (patch)
tree23c3e16eb4ba716b4a082228382d5a91aef436cd
parentb394654c1eb35f99aa356dacd73e2249be4278cc (diff)
downloadphp-git-6f9dfd947302f9a0d2fa6a78bf385b1ca7dafdf3.tar.gz
Fix bug #77955
Free metadata before freeing the arena. I don't have a repro script, but the added assertion fails for many existing tests prior to this change.
-rw-r--r--NEWS4
-rw-r--r--Zend/zend_arena.h11
-rw-r--r--ext/mysqlnd/mysqlnd_result.c5
3 files changed, 18 insertions, 2 deletions
diff --git a/NEWS b/NEWS
index a62c9dd881..7ddff4c431 100644
--- a/NEWS
+++ b/NEWS
@@ -14,6 +14,10 @@ PHP NEWS
. Fixed bug #77956 (When mysqli.allow_local_infile = Off, use a meaningful
error message). (Sjon Hortensius)
+- MySQLnd:
+ . Fixed bug #77955 (Random segmentation fault in mysqlnd from php-fpm).
+ (Nikita)
+
- Opcache:
. Fixed bug #78015 (Incorrect evaluation of expressions involving partials
arrays in SCCP). (Nikita)
diff --git a/Zend/zend_arena.h b/Zend/zend_arena.h
index f83a5103ce..604285fc97 100644
--- a/Zend/zend_arena.h
+++ b/Zend/zend_arena.h
@@ -110,6 +110,17 @@ static zend_always_inline void zend_arena_release(zend_arena **arena_ptr, void *
arena->ptr = (char*)checkpoint;
}
+static zend_always_inline zend_bool zend_arena_contains(zend_arena *arena, void *ptr)
+{
+ while (arena) {
+ if ((char*)ptr > (char*)arena && (char*)ptr <= arena->ptr) {
+ return 1;
+ }
+ arena = arena->prev;
+ }
+ return 0;
+}
+
#endif /* _ZEND_ARENA_H_ */
/*
diff --git a/ext/mysqlnd/mysqlnd_result.c b/ext/mysqlnd/mysqlnd_result.c
index f0852af360..06a7793c9a 100644
--- a/ext/mysqlnd/mysqlnd_result.c
+++ b/ext/mysqlnd/mysqlnd_result.c
@@ -294,13 +294,14 @@ void MYSQLND_METHOD(mysqlnd_res, free_result_contents_internal)(MYSQLND_RES * re
{
DBG_ENTER("mysqlnd_res::free_result_contents_internal");
- result->m.free_result_buffers(result);
-
if (result->meta) {
+ ZEND_ASSERT(zend_arena_contains(result->memory_pool->arena, result->meta));
result->meta->m->free_metadata(result->meta);
result->meta = NULL;
}
+ result->m.free_result_buffers(result);
+
DBG_VOID_RETURN;
}
/* }}} */