php-internal-docs 8.4.8
Unofficial docs for php/php-src
Loading...
Searching...
No Matches
zend_arena.h
Go to the documentation of this file.
1/*
2 +----------------------------------------------------------------------+
3 | Zend Engine |
4 +----------------------------------------------------------------------+
5 | Copyright (c) Zend Technologies Ltd. (http://www.zend.com) |
6 +----------------------------------------------------------------------+
7 | This source file is subject to version 2.00 of the Zend 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.zend.com/license/2_00.txt. |
11 | If you did not receive a copy of the Zend license and are unable to |
12 | obtain it through the world-wide-web, please send a note to |
13 | license@zend.com so we can mail you a copy immediately. |
14 +----------------------------------------------------------------------+
15 | Authors: Dmitry Stogov <dmitry@php.net> |
16 +----------------------------------------------------------------------+
17*/
18
19#ifndef _ZEND_ARENA_H_
20#define _ZEND_ARENA_H_
21
22#include "zend.h"
23
24#ifndef ZEND_TRACK_ARENA_ALLOC
25
26typedef struct _zend_arena zend_arena;
27
29 char *ptr;
30 char *end;
32};
33
34static zend_always_inline zend_arena* zend_arena_create(size_t size)
35{
37
38 arena->ptr = (char*) arena + ZEND_MM_ALIGNED_SIZE(sizeof(zend_arena));
39 arena->end = (char*) arena + size;
40 arena->prev = NULL;
41 return arena;
42}
43
44static zend_always_inline void zend_arena_destroy(zend_arena *arena)
45{
46 do {
47 zend_arena *prev = arena->prev;
48 efree(arena);
49 arena = prev;
50 } while (arena);
51}
52
53static zend_always_inline void* zend_arena_alloc(zend_arena **arena_ptr, size_t size)
54{
55 zend_arena *arena = *arena_ptr;
56 char *ptr = arena->ptr;
57
59
60 if (EXPECTED(size <= (size_t)(arena->end - ptr))) {
61 arena->ptr = ptr + size;
62 } else {
63 size_t arena_size =
64 UNEXPECTED((size + ZEND_MM_ALIGNED_SIZE(sizeof(zend_arena))) > (size_t)(arena->end - (char*) arena)) ?
66 (size_t)(arena->end - (char*) arena);
67 zend_arena *new_arena = (zend_arena*)emalloc(arena_size);
68
69 ptr = (char*) new_arena + ZEND_MM_ALIGNED_SIZE(sizeof(zend_arena));
70 new_arena->ptr = (char*) new_arena + ZEND_MM_ALIGNED_SIZE(sizeof(zend_arena)) + size;
71 new_arena->end = (char*) new_arena + arena_size;
72 new_arena->prev = arena;
73 *arena_ptr = new_arena;
74 }
75
76 return (void*) ptr;
77}
78
79static zend_always_inline void* zend_arena_calloc(zend_arena **arena_ptr, size_t count, size_t unit_size)
80{
81 bool overflow;
82 size_t size;
83 void *ret;
84
85 size = zend_safe_address(unit_size, count, 0, &overflow);
86 if (UNEXPECTED(overflow)) {
87 zend_error_noreturn(E_ERROR, "Possible integer overflow in zend_arena_calloc() (%zu * %zu)", unit_size, count);
88 }
89 ret = zend_arena_alloc(arena_ptr, size);
90 memset(ret, 0, size);
91 return ret;
92}
93
94static zend_always_inline void* zend_arena_checkpoint(zend_arena *arena)
95{
96 return arena->ptr;
97}
98
99static zend_always_inline void zend_arena_release(zend_arena **arena_ptr, void *checkpoint)
100{
101 zend_arena *arena = *arena_ptr;
102
103 while (UNEXPECTED((char*)checkpoint > arena->end) ||
104 UNEXPECTED((char*)checkpoint <= (char*)arena)) {
105 zend_arena *prev = arena->prev;
106 efree(arena);
107 *arena_ptr = arena = prev;
108 }
109 ZEND_ASSERT((char*)checkpoint > (char*)arena && (char*)checkpoint <= arena->end);
110 arena->ptr = (char*)checkpoint;
111}
112
113static zend_always_inline bool zend_arena_contains(zend_arena *arena, void *ptr)
114{
115 while (arena) {
116 if ((char*)ptr > (char*)arena && (char*)ptr <= arena->ptr) {
117 return 1;
118 }
119 arena = arena->prev;
120 }
121 return 0;
122}
123
124#else
125
126/* Use normal allocations and keep track of them for mass-freeing.
127 * This is intended for use with asan/valgrind. */
128
129typedef struct _zend_arena zend_arena;
130
131struct _zend_arena {
132 void **ptr;
133 void **end;
134 struct _zend_arena *prev;
135 void *ptrs[0];
136};
137
138#define ZEND_TRACKED_ARENA_SIZE 1000
139
140static zend_always_inline zend_arena *zend_arena_create(size_t _size)
141{
143 sizeof(zend_arena) + sizeof(void *) * ZEND_TRACKED_ARENA_SIZE);
144 arena->ptr = &arena->ptrs[0];
145 arena->end = &arena->ptrs[ZEND_TRACKED_ARENA_SIZE];
146 arena->prev = NULL;
147 return arena;
148}
149
150static zend_always_inline void zend_arena_destroy(zend_arena *arena)
151{
152 do {
153 zend_arena *prev = arena->prev;
154 void **ptr;
155 for (ptr = arena->ptrs; ptr < arena->ptr; ptr++) {
156 efree(*ptr);
157 }
158 efree(arena);
159 arena = prev;
160 } while (arena);
161}
162
163static zend_always_inline void *zend_arena_alloc(zend_arena **arena_ptr, size_t size)
164{
165 zend_arena *arena = *arena_ptr;
166 if (arena->ptr == arena->end) {
167 *arena_ptr = zend_arena_create(0);
168 (*arena_ptr)->prev = arena;
169 arena = *arena_ptr;
170 }
171
172 return *arena->ptr++ = emalloc(size);
173}
174
175static zend_always_inline void* zend_arena_calloc(zend_arena **arena_ptr, size_t count, size_t unit_size)
176{
177 bool overflow;
178 size_t size;
179 void *ret;
180
181 size = zend_safe_address(unit_size, count, 0, &overflow);
182 if (UNEXPECTED(overflow)) {
183 zend_error_noreturn(E_ERROR, "Possible integer overflow in zend_arena_calloc() (%zu * %zu)", unit_size, count);
184 }
185 ret = zend_arena_alloc(arena_ptr, size);
186 memset(ret, 0, size);
187 return ret;
188}
189
190static zend_always_inline void* zend_arena_checkpoint(zend_arena *arena)
191{
192 return arena->ptr;
193}
194
195static zend_always_inline void zend_arena_release(zend_arena **arena_ptr, void *checkpoint)
196{
197 while (1) {
198 zend_arena *arena = *arena_ptr;
199 zend_arena *prev = arena->prev;
200 while (1) {
201 if (arena->ptr == (void **) checkpoint) {
202 return;
203 }
204 if (arena->ptr == arena->ptrs) {
205 break;
206 }
207 arena->ptr--;
208 efree(*arena->ptr);
209 }
210 efree(arena);
211 *arena_ptr = prev;
212 ZEND_ASSERT(*arena_ptr);
213 }
214}
215
216static zend_always_inline bool zend_arena_contains(zend_arena *arena, void *ptr)
217{
218 /* TODO: Dummy */
219 return 1;
220}
221
222#endif
223
224#endif /* _ZEND_ARENA_H_ */
prev(array|object &$array)
count(Countable|array $value, int $mode=COUNT_NORMAL)
new_type size
Definition ffi.c:4365
void * ptr
Definition ffi.c:3814
memset(ptr, 0, type->size)
#define NULL
Definition gdcache.h:45
char * arena
Definition php_bcmath.h:37
unsigned const char * end
Definition php_ffi.h:51
zend_arena * prev
Definition zend_arena.h:31
char * end
Definition zend_arena.h:30
char * ptr
Definition zend_arena.h:29
ZEND_API ZEND_COLD ZEND_NORETURN void zend_error_noreturn(int type, const char *format,...)
Definition zend.c:1703
#define ZEND_MM_ALIGNED_SIZE(size)
Definition zend_alloc.h:35
#define efree(ptr)
Definition zend_alloc.h:155
#define emalloc(size)
Definition zend_alloc.h:151
struct _zend_arena zend_arena
Definition zend_arena.h:26
#define E_ERROR
Definition zend_errors.h:23
#define EXPECTED(condition)
#define zend_always_inline
#define ZEND_ASSERT(c)
#define UNEXPECTED(condition)
zval * ret