mysqlnd_block_alloc.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 2006-2018 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Authors: Andrey Hristov <andrey@php.net> |
  16. | Ulf Wendel <uw@php.net> |
  17. | Dmitry Stogov <dmitry@php.net> |
  18. +----------------------------------------------------------------------+
  19. */
  20. #include "php.h"
  21. #include "mysqlnd.h"
  22. #include "mysqlnd_block_alloc.h"
  23. #include "mysqlnd_debug.h"
  24. #include "mysqlnd_priv.h"
  25. /* {{{ mysqlnd_arena_create */
  26. static zend_always_inline zend_arena* mysqlnd_arena_create(size_t size)
  27. {
  28. zend_arena *arena = (zend_arena*)mnd_emalloc(size);
  29. arena->ptr = (char*) arena + ZEND_MM_ALIGNED_SIZE(sizeof(zend_arena));
  30. arena->end = (char*) arena + size;
  31. arena->prev = NULL;
  32. return arena;
  33. }
  34. /* }}} */
  35. /* {{{ mysqlnd_arena_destroy */
  36. static zend_always_inline void mysqlnd_arena_destroy(zend_arena *arena)
  37. {
  38. do {
  39. zend_arena *prev = arena->prev;
  40. mnd_efree(arena);
  41. arena = prev;
  42. } while (arena);
  43. }
  44. /* }}} */
  45. /* {{{ mysqlnd_arena_alloc */
  46. static zend_always_inline void* mysqlnd_arena_alloc(zend_arena **arena_ptr, size_t size)
  47. {
  48. zend_arena *arena = *arena_ptr;
  49. char *ptr = arena->ptr;
  50. size = ZEND_MM_ALIGNED_SIZE(size);
  51. if (EXPECTED(size <= (size_t)(arena->end - ptr))) {
  52. arena->ptr = ptr + size;
  53. } else {
  54. size_t arena_size =
  55. UNEXPECTED((size + ZEND_MM_ALIGNED_SIZE(sizeof(zend_arena))) > (size_t)(arena->end - (char*) arena)) ?
  56. (size + ZEND_MM_ALIGNED_SIZE(sizeof(zend_arena))) :
  57. (size_t)(arena->end - (char*) arena);
  58. zend_arena *new_arena = (zend_arena*)mnd_emalloc(arena_size);
  59. ptr = (char*) new_arena + ZEND_MM_ALIGNED_SIZE(sizeof(zend_arena));
  60. new_arena->ptr = (char*) new_arena + ZEND_MM_ALIGNED_SIZE(sizeof(zend_arena)) + size;
  61. new_arena->end = (char*) new_arena + arena_size;
  62. new_arena->prev = arena;
  63. *arena_ptr = new_arena;
  64. }
  65. return (void*) ptr;
  66. }
  67. /* }}} */
  68. static zend_always_inline void* mysqlnd_arena_checkpoint(zend_arena *arena)
  69. {
  70. return arena->ptr;
  71. }
  72. static zend_always_inline void mysqlnd_arena_release(zend_arena **arena_ptr, void *checkpoint)
  73. {
  74. zend_arena *arena = *arena_ptr;
  75. while (UNEXPECTED((char*)checkpoint > arena->end) ||
  76. UNEXPECTED((char*)checkpoint <= (char*)arena)) {
  77. zend_arena *prev = arena->prev;
  78. mnd_efree(arena);
  79. *arena_ptr = arena = prev;
  80. }
  81. ZEND_ASSERT((char*)checkpoint > (char*)arena && (char*)checkpoint <= arena->end);
  82. arena->ptr = (char*)checkpoint;
  83. }
  84. /* {{{ mysqlnd_mempool_free_chunk */
  85. static void
  86. mysqlnd_mempool_free_chunk(MYSQLND_MEMORY_POOL * pool, void * ptr)
  87. {
  88. DBG_ENTER("mysqlnd_mempool_free_chunk");
  89. /* Try to back-off and guess if this is the last block allocated */
  90. if (ptr == pool->last) {
  91. /*
  92. This was the last allocation. Lucky us, we can free
  93. a bit of memory from the pool. Next time we will return from the same ptr.
  94. */
  95. pool->arena->ptr = (char*)ptr;
  96. pool->last = NULL;
  97. }
  98. DBG_VOID_RETURN;
  99. }
  100. /* }}} */
  101. /* {{{ mysqlnd_mempool_resize_chunk */
  102. static void *
  103. mysqlnd_mempool_resize_chunk(MYSQLND_MEMORY_POOL * pool, void * ptr, size_t old_size, size_t size)
  104. {
  105. DBG_ENTER("mysqlnd_mempool_resize_chunk");
  106. /* Try to back-off and guess if this is the last block allocated */
  107. if (ptr == pool->last
  108. && (ZEND_MM_ALIGNED_SIZE(size) <= ((char*)pool->arena->end - (char*)ptr))) {
  109. /*
  110. This was the last allocation. Lucky us, we can free
  111. a bit of memory from the pool. Next time we will return from the same ptr.
  112. */
  113. pool->arena->ptr = (char*)ptr + ZEND_MM_ALIGNED_SIZE(size);
  114. } else {
  115. void *new_ptr = mysqlnd_arena_alloc(&pool->arena, size);
  116. memcpy(new_ptr, ptr, MIN(old_size, size));
  117. pool->last = ptr = new_ptr;
  118. }
  119. DBG_RETURN(ptr);
  120. }
  121. /* }}} */
  122. /* {{{ mysqlnd_mempool_get_chunk */
  123. static void *
  124. mysqlnd_mempool_get_chunk(MYSQLND_MEMORY_POOL * pool, size_t size)
  125. {
  126. void *ptr = NULL;
  127. DBG_ENTER("mysqlnd_mempool_get_chunk");
  128. ptr = mysqlnd_arena_alloc(&pool->arena, size);
  129. pool->last = ptr;
  130. DBG_RETURN(ptr);
  131. }
  132. /* }}} */
  133. /* {{{ mysqlnd_mempool_create */
  134. PHPAPI MYSQLND_MEMORY_POOL *
  135. mysqlnd_mempool_create(size_t arena_size)
  136. {
  137. zend_arena * arena;
  138. MYSQLND_MEMORY_POOL * ret;
  139. DBG_ENTER("mysqlnd_mempool_create");
  140. arena = mysqlnd_arena_create(MAX(arena_size, ZEND_MM_ALIGNED_SIZE(sizeof(zend_arena))));
  141. ret = mysqlnd_arena_alloc(&arena, sizeof(MYSQLND_MEMORY_POOL));
  142. ret->arena = arena;
  143. ret->last = NULL;
  144. ret->checkpoint = NULL;
  145. ret->get_chunk = mysqlnd_mempool_get_chunk;
  146. ret->free_chunk = mysqlnd_mempool_free_chunk;
  147. ret->resize_chunk = mysqlnd_mempool_resize_chunk;
  148. DBG_RETURN(ret);
  149. }
  150. /* }}} */
  151. /* {{{ mysqlnd_mempool_destroy */
  152. PHPAPI void
  153. mysqlnd_mempool_destroy(MYSQLND_MEMORY_POOL * pool)
  154. {
  155. DBG_ENTER("mysqlnd_mempool_destroy");
  156. /* mnd_free will reference LOCK_access and might crash, depending on the caller...*/
  157. mysqlnd_arena_destroy(pool->arena);
  158. DBG_VOID_RETURN;
  159. }
  160. /* }}} */
  161. /* {{{ mysqlnd_mempool_save_state */
  162. PHPAPI void
  163. mysqlnd_mempool_save_state(MYSQLND_MEMORY_POOL * pool)
  164. {
  165. DBG_ENTER("mysqlnd_mempool_save_state");
  166. pool->checkpoint = mysqlnd_arena_checkpoint(pool->arena);
  167. DBG_VOID_RETURN;
  168. }
  169. /* }}} */
  170. /* {{{ mysqlnd_mempool_restore_state */
  171. PHPAPI void
  172. mysqlnd_mempool_restore_state(MYSQLND_MEMORY_POOL * pool)
  173. {
  174. DBG_ENTER("mysqlnd_mempool_restore_state");
  175. #if ZEND_DEBUG
  176. ZEND_ASSERT(pool->checkpoint);
  177. #endif
  178. if (pool->checkpoint) {
  179. mysqlnd_arena_release(&pool->arena, pool->checkpoint);
  180. pool->last = NULL;
  181. pool->checkpoint = NULL;
  182. }
  183. DBG_VOID_RETURN;
  184. }
  185. /* }}} */
  186. /*
  187. * Local variables:
  188. * tab-width: 4
  189. * c-basic-offset: 4
  190. * End:
  191. * vim600: noet sw=4 ts=4 fdm=marker
  192. * vim<600: noet sw=4 ts=4
  193. */