blob: 48072a4fb30683c16c62ff34cb02101dc2d8ef51 [file] [log] [blame]
Austin Schuh40c16522018-10-28 20:27:54 -07001// Protocol Buffers - Google's data interchange format
2// Copyright 2008 Google Inc. All rights reserved.
3// https://developers.google.com/protocol-buffers/
4//
5// Redistribution and use in source and binary forms, with or without
6// modification, are permitted provided that the following conditions are
7// met:
8//
9// * Redistributions of source code must retain the above copyright
10// notice, this list of conditions and the following disclaimer.
11// * Redistributions in binary form must reproduce the above
12// copyright notice, this list of conditions and the following disclaimer
13// in the documentation and/or other materials provided with the
14// distribution.
15// * Neither the name of Google Inc. nor the names of its
16// contributors may be used to endorse or promote products derived from
17// this software without specific prior written permission.
18//
19// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
31#ifndef __GOOGLE_PROTOBUF_PHP_PROTOBUF_H__
32#define __GOOGLE_PROTOBUF_PHP_PROTOBUF_H__
33
34#include <php.h>
35
36// ubp.h has to be placed after php.h. Othwise, php.h will introduce NDEBUG.
37#include "upb.h"
38
39#define PHP_PROTOBUF_EXTNAME "protobuf"
40#define PHP_PROTOBUF_VERSION "3.6.1"
41
42#define MAX_LENGTH_OF_INT64 20
43#define SIZEOF_INT64 8
44
45// -----------------------------------------------------------------------------
46// PHP7 Wrappers
47// ----------------------------------------------------------------------------
48
49#if PHP_VERSION_ID < 70300
50#define GC_ADDREF(h) ++GC_REFCOUNT(h)
51#define GC_DELREF(h) --GC_REFCOUNT(h)
52#endif
53
54#if PHP_MAJOR_VERSION < 7
55
56#define php_proto_zend_literal const zend_literal*
57#define PHP_PROTO_CASE_IS_BOOL IS_BOOL
58#define PHP_PROTO_SIZE int
59#define PHP_PROTO_LONG long
60#define PHP_PROTO_TSRMLS_DC TSRMLS_DC
61#define PHP_PROTO_TSRMLS_CC TSRMLS_CC
62
63// PHP String
64
65#define PHP_PROTO_ZVAL_STRING(zval_ptr, s, copy) \
66 ZVAL_STRING(zval_ptr, s, copy)
67#define PHP_PROTO_ZVAL_STRINGL(zval_ptr, s, len, copy) \
68 ZVAL_STRINGL(zval_ptr, s, len, copy)
69#define PHP_PROTO_RETURN_STRING(s, copy) RETURN_STRING(s, copy)
70#define PHP_PROTO_RETURN_STRINGL(s, len, copy) RETURN_STRINGL(s, len, copy)
71#define PHP_PROTO_RETVAL_STRINGL(s, len, copy) RETVAL_STRINGL(s, len, copy)
72#define php_proto_zend_make_printable_zval(from, to) \
73 { \
74 int use_copy; \
75 zend_make_printable_zval(from, to, &use_copy); \
76 }
77
78// PHP Array
79
80#define PHP_PROTO_HASH_OF(array) Z_ARRVAL_P(array)
81
82#define php_proto_zend_hash_index_update_zval(ht, h, pData) \
83 zend_hash_index_update(ht, h, &(pData), sizeof(void*), NULL)
84
85#define php_proto_zend_hash_update_zval(ht, key, key_len, value) \
86 zend_hash_update(ht, key, key_len, value, sizeof(void*), NULL)
87
88#define php_proto_zend_hash_update(ht, key, key_len) \
89 zend_hash_update(ht, key, key_len, 0, 0, NULL)
90
91#define php_proto_zend_hash_index_update_mem(ht, h, pData, nDataSize, pDest) \
92 zend_hash_index_update(ht, h, pData, nDataSize, pDest)
93
94#define php_proto_zend_hash_update_mem(ht, key, key_len, pData, nDataSize, \
95 pDest) \
96 zend_hash_update(ht, key, key_len, pData, nDataSize, pDest)
97
98#define php_proto_zend_hash_index_find_zval(ht, h, pDest) \
99 zend_hash_index_find(ht, h, pDest)
100
101#define php_proto_zend_hash_find(ht, key, key_len, pDest) \
102 zend_hash_find(ht, key, key_len, pDest)
103
104#define php_proto_zend_hash_index_find_mem(ht, h, pDest) \
105 zend_hash_index_find(ht, h, pDest)
106
107#define php_proto_zend_hash_find_zval(ht, key, key_len, pDest) \
108 zend_hash_find(ht, key, key_len, pDest)
109
110#define php_proto_zend_hash_find_mem(ht, key, key_len, pDest) \
111 zend_hash_find(ht, key, key_len, pDest)
112
113#define php_proto_zend_hash_next_index_insert_zval(ht, pData) \
114 zend_hash_next_index_insert(ht, pData, sizeof(void*), NULL)
115
116#define php_proto_zend_hash_next_index_insert_mem(ht, pData, nDataSize, pDest) \
117 zend_hash_next_index_insert(ht, pData, nDataSize, pDest)
118
119#define php_proto_zend_hash_get_current_data_ex(ht, pDest, pos) \
120 zend_hash_get_current_data_ex(ht, pDest, pos)
121
122// PHP Object
123
124#define PHP_PROTO_WRAP_OBJECT_START(name) \
125 struct name { \
126 zend_object std;
127#define PHP_PROTO_WRAP_OBJECT_END \
128 };
129
130#define PHP_PROTO_INIT_SUBMSGCLASS_START(CLASSNAME, CAMELNAME, LOWWERNAME) \
131 void LOWWERNAME##_init(TSRMLS_D) { \
132 zend_class_entry class_type; \
133 const char* class_name = CLASSNAME; \
134 INIT_CLASS_ENTRY_EX(class_type, CLASSNAME, strlen(CLASSNAME), \
135 LOWWERNAME##_methods); \
136 LOWWERNAME##_type = zend_register_internal_class(&class_type TSRMLS_CC); \
137 LOWWERNAME##_type->create_object = message_create;
138#define PHP_PROTO_INIT_SUBMSGCLASS_END \
139 }
140
141#define PHP_PROTO_INIT_ENUMCLASS_START(CLASSNAME, CAMELNAME, LOWWERNAME) \
142 void LOWWERNAME##_init(TSRMLS_D) { \
143 zend_class_entry class_type; \
144 const char* class_name = CLASSNAME; \
145 INIT_CLASS_ENTRY_EX(class_type, CLASSNAME, strlen(CLASSNAME), \
146 LOWWERNAME##_methods); \
147 LOWWERNAME##_type = zend_register_internal_class(&class_type TSRMLS_CC);
148#define PHP_PROTO_INIT_ENUMCLASS_END \
149 }
150
151#define PHP_PROTO_INIT_CLASS_START(CLASSNAME, CAMELNAME, LOWWERNAME) \
152 void LOWWERNAME##_init(TSRMLS_D) { \
153 zend_class_entry class_type; \
154 const char* class_name = CLASSNAME; \
155 INIT_CLASS_ENTRY_EX(class_type, CLASSNAME, strlen(CLASSNAME), \
156 LOWWERNAME##_methods); \
157 LOWWERNAME##_type = zend_register_internal_class(&class_type TSRMLS_CC); \
158 LOWWERNAME##_type->create_object = LOWWERNAME##_create; \
159 LOWWERNAME##_handlers = PEMALLOC(zend_object_handlers); \
160 memcpy(LOWWERNAME##_handlers, zend_get_std_object_handlers(), \
161 sizeof(zend_object_handlers));
162#define PHP_PROTO_INIT_CLASS_END \
163 }
164
165#define PHP_PROTO_OBJECT_CREATE_START(NAME, LOWWERNAME) \
166 static zend_object_value LOWWERNAME##_create( \
167 zend_class_entry* ce TSRMLS_DC) { \
168 PHP_PROTO_ALLOC_CLASS_OBJECT(NAME, ce); \
169 zend_object_std_init(&intern->std, ce TSRMLS_CC); \
170 object_properties_init(&intern->std, ce);
171#define PHP_PROTO_OBJECT_CREATE_END(NAME, LOWWERNAME) \
172 PHP_PROTO_FREE_CLASS_OBJECT(NAME, LOWWERNAME##_free, LOWWERNAME##_handlers); \
173 }
174
175#define PHP_PROTO_OBJECT_FREE_START(classname, lowername) \
176 void lowername##_free(void* object TSRMLS_DC) { \
177 classname* intern = object;
178#define PHP_PROTO_OBJECT_FREE_END \
179 zend_object_std_dtor(&intern->std TSRMLS_CC); \
180 efree(intern); \
181 }
182
183#define PHP_PROTO_OBJECT_DTOR_START(classname, lowername)
184#define PHP_PROTO_OBJECT_DTOR_END
185
186#define CACHED_VALUE zval*
187#define CACHED_TO_ZVAL_PTR(VALUE) (VALUE)
188#define CACHED_PTR_TO_ZVAL_PTR(VALUE) (*VALUE)
189#define ZVAL_PTR_TO_CACHED_PTR(VALUE) (&VALUE)
190#define ZVAL_PTR_TO_CACHED_VALUE(VALUE) (VALUE)
191#define ZVAL_TO_CACHED_VALUE(VALUE) (&VALUE)
192
193#define CREATE_OBJ_ON_ALLOCATED_ZVAL_PTR(zval_ptr, class_type) \
194 ZVAL_OBJ(zval_ptr, class_type->create_object(class_type TSRMLS_CC));
195
196#define PHP_PROTO_SEPARATE_ZVAL_IF_NOT_REF(value) \
197 SEPARATE_ZVAL_IF_NOT_REF(value)
198
199#define PHP_PROTO_GLOBAL_UNINITIALIZED_ZVAL EG(uninitialized_zval_ptr)
200
201#define OBJ_PROP(OBJECT, OFFSET) &((OBJECT)->properties_table[OFFSET])
202
203#define php_proto_zval_ptr_dtor(zval_ptr) \
204 zval_ptr_dtor(&(zval_ptr))
205
206#define PHP_PROTO_ALLOC_CLASS_OBJECT(class_object, class_type) \
207 class_object* intern; \
208 intern = (class_object*)emalloc(sizeof(class_object)); \
209 memset(intern, 0, sizeof(class_object));
210
211#define PHP_PROTO_FREE_CLASS_OBJECT(class_object, class_object_free, handler) \
212 zend_object_value retval = {0}; \
213 retval.handle = zend_objects_store_put( \
214 intern, (zend_objects_store_dtor_t)zend_objects_destroy_object, \
215 class_object_free, NULL TSRMLS_CC); \
216 retval.handlers = handler; \
217 return retval;
218
219#define PHP_PROTO_ALLOC_ARRAY(zval_ptr) \
220 ALLOC_HASHTABLE(Z_ARRVAL_P(zval_ptr)); \
221 Z_TYPE_P(zval_ptr) = IS_ARRAY;
222
223#define ZVAL_OBJ(zval_ptr, call_create) \
224 Z_TYPE_P(zval_ptr) = IS_OBJECT; \
225 Z_OBJVAL_P(zval_ptr) = call_create;
226
227#define UNBOX(class_name, val) \
228 (class_name*)zend_object_store_get_object(val TSRMLS_CC);
229
230#define UNBOX_HASHTABLE_VALUE(class_name, val) UNBOX(class_name, val)
231
232#define HASHTABLE_VALUE_DTOR ZVAL_PTR_DTOR
233
234#define PHP_PROTO_HASHTABLE_VALUE zval*
235#define HASHTABLE_VALUE_CE(val) Z_OBJCE_P(val)
236
237#define CREATE_HASHTABLE_VALUE(OBJ, WRAPPED_OBJ, OBJ_TYPE, OBJ_CLASS_ENTRY) \
238 OBJ_TYPE* OBJ; \
239 PHP_PROTO_HASHTABLE_VALUE WRAPPED_OBJ; \
240 MAKE_STD_ZVAL(WRAPPED_OBJ); \
241 ZVAL_OBJ(WRAPPED_OBJ, \
242 OBJ_CLASS_ENTRY->create_object(OBJ_CLASS_ENTRY TSRMLS_CC)); \
243 OBJ = UNBOX_HASHTABLE_VALUE(OBJ_TYPE, WRAPPED_OBJ); \
244 Z_DELREF_P(desc_php);
245
246#define PHP_PROTO_CE_DECLARE zend_class_entry**
247#define PHP_PROTO_CE_UNREF(ce) (*ce)
248
249#define php_proto_zend_lookup_class(name, name_length, ce) \
250 zend_lookup_class(name, name_length, ce TSRMLS_CC)
251
252#define PHP_PROTO_RETVAL_ZVAL(value) ZVAL_ZVAL(return_value, value, 1, 0)
253
254#else // PHP_MAJOR_VERSION >= 7
255
256#define php_proto_zend_literal void**
257#define PHP_PROTO_CASE_IS_BOOL IS_TRUE: case IS_FALSE
258#define PHP_PROTO_SIZE size_t
259#define PHP_PROTO_LONG zend_long
260#define PHP_PROTO_TSRMLS_DC
261#define PHP_PROTO_TSRMLS_CC
262
263// PHP String
264
265#define PHP_PROTO_ZVAL_STRING(zval_ptr, s, copy) \
266 ZVAL_STRING(zval_ptr, s)
267#define PHP_PROTO_ZVAL_STRINGL(zval_ptr, s, len, copy) \
268 ZVAL_STRINGL(zval_ptr, s, len)
269#define PHP_PROTO_RETURN_STRING(s, copy) RETURN_STRING(s)
270#define PHP_PROTO_RETURN_STRINGL(s, len, copy) RETURN_STRINGL(s, len)
271#define PHP_PROTO_RETVAL_STRINGL(s, len, copy) RETVAL_STRINGL(s, len)
272#define php_proto_zend_make_printable_zval(from, to) \
273 zend_make_printable_zval(from, to)
274
275// PHP Array
276
277#define PHP_PROTO_HASH_OF(array) Z_ARRVAL_P(&array)
278
279static inline int php_proto_zend_hash_index_update_zval(HashTable* ht, ulong h,
280 zval* pData) {
281 void* result = NULL;
282 result = zend_hash_index_update(ht, h, pData);
283 return result != NULL ? SUCCESS : FAILURE;
284}
285
286static inline int php_proto_zend_hash_update(HashTable* ht, const char* key,
287 size_t key_len) {
288 void* result = NULL;
289 zval temp;
290 ZVAL_LONG(&temp, 0);
291 result = zend_hash_str_update(ht, key, key_len, &temp);
292 return result != NULL ? SUCCESS : FAILURE;
293}
294
295static inline int php_proto_zend_hash_index_update_mem(HashTable* ht, ulong h,
296 void* pData, uint nDataSize,
297 void** pDest) {
298 void* result = NULL;
299 result = zend_hash_index_update_mem(ht, h, pData, nDataSize);
300 if (pDest != NULL) *pDest = result;
301 return result != NULL ? SUCCESS : FAILURE;
302}
303
304static inline int php_proto_zend_hash_update_zval(HashTable* ht,
305 const char* key, uint key_len,
306 zval* pData) {
307 zend_string* internal_key = zend_string_init(key, key_len, 0);
308 zend_hash_update(ht, internal_key, pData);
309}
310
311static inline int php_proto_zend_hash_update_mem(HashTable* ht, const char* key,
312 uint key_len, void* pData,
313 uint nDataSize, void** pDest) {
314 zend_string* internal_key = zend_string_init(key, key_len, 0);
315 void* result = zend_hash_update_mem(ht, internal_key, pData, nDataSize);
316 zend_string_release(internal_key);
317 if (pDest != NULL) *pDest = result;
318 return result != NULL ? SUCCESS : FAILURE;
319}
320
321static inline int php_proto_zend_hash_index_find_zval(const HashTable* ht,
322 ulong h, void** pDest) {
323 zval* result = zend_hash_index_find(ht, h);
324 if (pDest != NULL) *pDest = result;
325 return result != NULL ? SUCCESS : FAILURE;
326}
327
328static inline int php_proto_zend_hash_find(const HashTable* ht, const char* key,
329 size_t key_len, void** pDest) {
330 void* result = NULL;
331 result = zend_hash_str_find(ht, key, key_len);
332 return result != NULL ? SUCCESS : FAILURE;
333}
334
335static inline int php_proto_zend_hash_index_find_mem(const HashTable* ht,
336 ulong h, void** pDest) {
337 void* result = NULL;
338 result = zend_hash_index_find_ptr(ht, h);
339 if (pDest != NULL) *pDest = result;
340 return result != NULL ? SUCCESS : FAILURE;
341}
342
343static inline int php_proto_zend_hash_find_zval(const HashTable* ht,
344 const char* key, uint key_len,
345 void** pDest) {
346 zend_string* internal_key = zend_string_init(key, key_len, 1);
347 zval* result = zend_hash_find(ht, internal_key);
348 if (pDest != NULL) *pDest = result;
349 return result != NULL ? SUCCESS : FAILURE;
350}
351
352static inline int php_proto_zend_hash_find_mem(const HashTable* ht,
353 const char* key, uint key_len,
354 void** pDest) {
355 zend_string* internal_key = zend_string_init(key, key_len, 1);
356 void* result = zend_hash_find_ptr(ht, internal_key);
357 zend_string_release(internal_key);
358 if (pDest != NULL) *pDest = result;
359 return result != NULL ? SUCCESS : FAILURE;
360}
361
362static inline int php_proto_zend_hash_next_index_insert_zval(HashTable* ht,
363 void* pData) {
364 zval tmp;
365 ZVAL_OBJ(&tmp, *(zend_object**)pData);
366 zval* result = zend_hash_next_index_insert(ht, &tmp);
367 return result != NULL ? SUCCESS : FAILURE;
368}
369
370static inline int php_proto_zend_hash_next_index_insert_mem(HashTable* ht,
371 void* pData,
372 uint nDataSize,
373 void** pDest) {
374 void* result = NULL;
375 result = zend_hash_next_index_insert_mem(ht, pData, nDataSize);
376 if (pDest != NULL) *pDest = result;
377 return result != NULL ? SUCCESS : FAILURE;
378}
379
380static inline int php_proto_zend_hash_get_current_data_ex(HashTable* ht,
381 void** pDest,
382 HashPosition* pos) {
383 void* result = NULL;
384 result = zend_hash_get_current_data_ex(ht, pos);
385 if (pDest != NULL) *pDest = result;
386 return result != NULL ? SUCCESS : FAILURE;
387}
388
389// PHP Object
390
391#define PHP_PROTO_WRAP_OBJECT_START(name) struct name {
392#define PHP_PROTO_WRAP_OBJECT_END \
393 zend_object std; \
394 };
395
396#define PHP_PROTO_INIT_SUBMSGCLASS_START(CLASSNAME, CAMELNAME, LOWWERNAME) \
397 void LOWWERNAME##_init(TSRMLS_D) { \
398 zend_class_entry class_type; \
399 const char* class_name = CLASSNAME; \
400 INIT_CLASS_ENTRY_EX(class_type, CLASSNAME, strlen(CLASSNAME), \
401 LOWWERNAME##_methods); \
402 LOWWERNAME##_type = zend_register_internal_class(&class_type TSRMLS_CC); \
403 LOWWERNAME##_type->create_object = message_create;
404#define PHP_PROTO_INIT_SUBMSGCLASS_END \
405 }
406
407#define PHP_PROTO_INIT_ENUMCLASS_START(CLASSNAME, CAMELNAME, LOWWERNAME) \
408 void LOWWERNAME##_init(TSRMLS_D) { \
409 zend_class_entry class_type; \
410 const char* class_name = CLASSNAME; \
411 INIT_CLASS_ENTRY_EX(class_type, CLASSNAME, strlen(CLASSNAME), \
412 LOWWERNAME##_methods); \
413 LOWWERNAME##_type = zend_register_internal_class(&class_type TSRMLS_CC);
414#define PHP_PROTO_INIT_ENUMCLASS_END \
415 }
416
417#define PHP_PROTO_INIT_CLASS_START(CLASSNAME, CAMELNAME, LOWWERNAME) \
418 void LOWWERNAME##_init(TSRMLS_D) { \
419 zend_class_entry class_type; \
420 const char* class_name = CLASSNAME; \
421 INIT_CLASS_ENTRY_EX(class_type, CLASSNAME, strlen(CLASSNAME), \
422 LOWWERNAME##_methods); \
423 LOWWERNAME##_type = zend_register_internal_class(&class_type TSRMLS_CC); \
424 LOWWERNAME##_type->create_object = LOWWERNAME##_create; \
425 LOWWERNAME##_handlers = PEMALLOC(zend_object_handlers); \
426 memcpy(LOWWERNAME##_handlers, zend_get_std_object_handlers(), \
427 sizeof(zend_object_handlers)); \
428 LOWWERNAME##_handlers->free_obj = LOWWERNAME##_free; \
429 LOWWERNAME##_handlers->dtor_obj = LOWWERNAME##_dtor; \
430 LOWWERNAME##_handlers->offset = XtOffsetOf(CAMELNAME, std);
431#define PHP_PROTO_INIT_CLASS_END \
432 }
433
434#define PHP_PROTO_OBJECT_FREE_START(classname, lowername) \
435 void lowername##_free(zend_object* object) { \
436 classname* intern = \
437 (classname*)((char*)object - XtOffsetOf(classname, std));
438#define PHP_PROTO_OBJECT_FREE_END \
439 }
440
441#define PHP_PROTO_OBJECT_DTOR_START(classname, lowername) \
442 void lowername##_dtor(zend_object* object) { \
443 classname* intern = \
444 (classname*)((char*)object - XtOffsetOf(classname, std));
445#define PHP_PROTO_OBJECT_DTOR_END \
446 zend_object_std_dtor(object TSRMLS_CC); \
447 }
448
449#define PHP_PROTO_OBJECT_CREATE_START(NAME, LOWWERNAME) \
450 static zend_object* LOWWERNAME##_create(zend_class_entry* ce TSRMLS_DC) { \
451 PHP_PROTO_ALLOC_CLASS_OBJECT(NAME, ce); \
452 zend_object_std_init(&intern->std, ce TSRMLS_CC); \
453 object_properties_init(&intern->std, ce);
454#define PHP_PROTO_OBJECT_CREATE_END(NAME, LOWWERNAME) \
455 PHP_PROTO_FREE_CLASS_OBJECT(NAME, LOWWERNAME##_free, LOWWERNAME##_handlers); \
456 }
457
458#define CACHED_VALUE zval
459#define CACHED_TO_ZVAL_PTR(VALUE) (&VALUE)
460#define CACHED_PTR_TO_ZVAL_PTR(VALUE) (VALUE)
461#define ZVAL_PTR_TO_CACHED_PTR(VALUE) (VALUE)
462#define ZVAL_PTR_TO_CACHED_VALUE(VALUE) (*VALUE)
463#define ZVAL_TO_CACHED_VALUE(VALUE) (VALUE)
464
465#define CREATE_OBJ_ON_ALLOCATED_ZVAL_PTR(zval_ptr, class_type) \
466 ZVAL_OBJ(zval_ptr, class_type->create_object(class_type));
467
468#define PHP_PROTO_SEPARATE_ZVAL_IF_NOT_REF(value) ;
469
470#define PHP_PROTO_GLOBAL_UNINITIALIZED_ZVAL &EG(uninitialized_zval)
471
472#define php_proto_zval_ptr_dtor(zval_ptr) \
473 zval_ptr_dtor(zval_ptr)
474
475#define PHP_PROTO_ALLOC_CLASS_OBJECT(class_object, class_type) \
476 class_object* intern; \
477 int size = sizeof(class_object) + zend_object_properties_size(class_type); \
478 intern = ecalloc(1, size); \
479 memset(intern, 0, size);
480
481#define PHP_PROTO_FREE_CLASS_OBJECT(class_object, class_object_free, handler) \
482 intern->std.handlers = handler; \
483 return &intern->std;
484
485#define PHP_PROTO_ALLOC_ARRAY(zval_ptr) \
486 ZVAL_NEW_ARR(zval_ptr)
487
488#define UNBOX(class_name, val) \
489 (class_name*)((char*)Z_OBJ_P(val) - XtOffsetOf(class_name, std));
490
491#define UNBOX_HASHTABLE_VALUE(class_name, val) \
492 (class_name*)((char*)val - XtOffsetOf(class_name, std))
493
494#define HASHTABLE_VALUE_DTOR php_proto_hashtable_descriptor_release
495
496#define PHP_PROTO_HASHTABLE_VALUE zend_object*
497#define HASHTABLE_VALUE_CE(val) val->ce
498
499#define CREATE_HASHTABLE_VALUE(OBJ, WRAPPED_OBJ, OBJ_TYPE, OBJ_CLASS_ENTRY) \
500 OBJ_TYPE* OBJ; \
501 PHP_PROTO_HASHTABLE_VALUE WRAPPED_OBJ; \
502 WRAPPED_OBJ = OBJ_CLASS_ENTRY->create_object(OBJ_CLASS_ENTRY); \
503 OBJ = UNBOX_HASHTABLE_VALUE(OBJ_TYPE, WRAPPED_OBJ); \
504 GC_DELREF(WRAPPED_OBJ);
505
506#define PHP_PROTO_CE_DECLARE zend_class_entry*
507#define PHP_PROTO_CE_UNREF(ce) (ce)
508
509static inline int php_proto_zend_lookup_class(
510 const char* name, int name_length, zend_class_entry** ce TSRMLS_DC) {
511 zend_string *zstr_name = zend_string_init(name, name_length, 0);
512 *ce = zend_lookup_class(zstr_name);
513 zend_string_release(zstr_name);
514 return *ce != NULL ? SUCCESS : FAILURE;
515}
516
517#define PHP_PROTO_RETVAL_ZVAL(value) ZVAL_COPY(return_value, value)
518
519#endif // PHP_MAJOR_VERSION >= 7
520
521#if PHP_MAJOR_VERSION < 7 || (PHP_MAJOR_VERSION == 7 && PHP_MINOR_VERSION == 0)
522#define PHP_PROTO_FAKE_SCOPE_BEGIN(klass) \
523 zend_class_entry* old_scope = EG(scope); \
524 EG(scope) = klass;
525#define PHP_PROTO_FAKE_SCOPE_RESTART(klass) \
526 old_scope = EG(scope); \
527 EG(scope) = klass;
528#define PHP_PROTO_FAKE_SCOPE_END EG(scope) = old_scope;
529#else
530#define PHP_PROTO_FAKE_SCOPE_BEGIN(klass) \
531 zend_class_entry* old_scope = EG(fake_scope); \
532 EG(fake_scope) = klass;
533#define PHP_PROTO_FAKE_SCOPE_RESTART(klass) \
534 old_scope = EG(fake_scope); \
535 EG(fake_scope) = klass;
536#define PHP_PROTO_FAKE_SCOPE_END EG(fake_scope) = old_scope;
537#endif
538
539// Define PHP class
540#define DEFINE_PROTOBUF_INIT_CLASS(CLASSNAME, CAMELNAME, LOWERNAME) \
541 PHP_PROTO_INIT_CLASS_START(CLASSNAME, CAMELNAME, LOWERNAME) \
542 PHP_PROTO_INIT_CLASS_END
543
544#define DEFINE_PROTOBUF_CREATE(NAME, LOWERNAME) \
545 PHP_PROTO_OBJECT_CREATE_START(NAME, LOWERNAME) \
546 LOWERNAME##_init_c_instance(intern TSRMLS_CC); \
547 PHP_PROTO_OBJECT_CREATE_END(NAME, LOWERNAME)
548
549#define DEFINE_PROTOBUF_FREE(CAMELNAME, LOWERNAME) \
550 PHP_PROTO_OBJECT_FREE_START(CAMELNAME, LOWERNAME) \
551 LOWERNAME##_free_c(intern TSRMLS_CC); \
552 PHP_PROTO_OBJECT_FREE_END
553
554#define DEFINE_PROTOBUF_DTOR(CAMELNAME, LOWERNAME) \
555 PHP_PROTO_OBJECT_DTOR_START(CAMELNAME, LOWERNAME) \
556 PHP_PROTO_OBJECT_DTOR_END
557
558#define DEFINE_CLASS(NAME, LOWERNAME, string_name) \
559 zend_class_entry *LOWERNAME##_type; \
560 zend_object_handlers *LOWERNAME##_handlers; \
561 DEFINE_PROTOBUF_FREE(NAME, LOWERNAME) \
562 DEFINE_PROTOBUF_DTOR(NAME, LOWERNAME) \
563 DEFINE_PROTOBUF_CREATE(NAME, LOWERNAME) \
564 DEFINE_PROTOBUF_INIT_CLASS(string_name, NAME, LOWERNAME)
565
566// -----------------------------------------------------------------------------
567// Forward Declaration
568// ----------------------------------------------------------------------------
569
570struct Any;
571struct Api;
572struct BoolValue;
573struct BytesValue;
574struct Descriptor;
575struct DescriptorPool;
576struct DoubleValue;
577struct Duration;
578struct Enum;
579struct EnumDescriptor;
580struct EnumValue;
581struct EnumValueDescriptor;
582struct Field;
583struct FieldDescriptor;
584struct FieldMask;
585struct Field_Cardinality;
586struct Field_Kind;
587struct FloatValue;
588struct GPBEmpty;
589struct Int32Value;
590struct Int64Value;
591struct InternalDescriptorPool;
592struct ListValue;
593struct Map;
594struct MapIter;
595struct MessageField;
596struct MessageHeader;
597struct MessageLayout;
598struct Method;
599struct Mixin;
600struct NullValue;
601struct Oneof;
602struct Option;
603struct RepeatedField;
604struct RepeatedFieldIter;
605struct SourceContext;
606struct StringValue;
607struct Struct;
608struct Syntax;
609struct Timestamp;
610struct Type;
611struct UInt32Value;
612struct UInt64Value;
613struct Value;
614
615typedef struct Any Any;
616typedef struct Api Api;
617typedef struct BoolValue BoolValue;
618typedef struct BytesValue BytesValue;
619typedef struct Descriptor Descriptor;
620typedef struct DescriptorPool DescriptorPool;
621typedef struct DoubleValue DoubleValue;
622typedef struct Duration Duration;
623typedef struct EnumDescriptor EnumDescriptor;
624typedef struct Enum Enum;
625typedef struct EnumValueDescriptor EnumValueDescriptor;
626typedef struct EnumValue EnumValue;
627typedef struct Field_Cardinality Field_Cardinality;
628typedef struct FieldDescriptor FieldDescriptor;
629typedef struct Field Field;
630typedef struct Field_Kind Field_Kind;
631typedef struct FieldMask FieldMask;
632typedef struct FloatValue FloatValue;
633typedef struct GPBEmpty GPBEmpty;
634typedef struct Int32Value Int32Value;
635typedef struct Int64Value Int64Value;
636typedef struct InternalDescriptorPool InternalDescriptorPool;
637typedef struct ListValue ListValue;
638typedef struct MapIter MapIter;
639typedef struct Map Map;
640typedef struct MessageField MessageField;
641typedef struct MessageHeader MessageHeader;
642typedef struct MessageLayout MessageLayout;
643typedef struct Method Method;
644typedef struct Mixin Mixin;
645typedef struct NullValue NullValue;
646typedef struct Oneof Oneof;
647typedef struct Option Option;
648typedef struct RepeatedFieldIter RepeatedFieldIter;
649typedef struct RepeatedField RepeatedField;
650typedef struct SourceContext SourceContext;
651typedef struct StringValue StringValue;
652typedef struct Struct Struct;
653typedef struct Syntax Syntax;
654typedef struct Timestamp Timestamp;
655typedef struct Type Type;
656typedef struct UInt32Value UInt32Value;
657typedef struct UInt64Value UInt64Value;
658typedef struct Value Value;
659
660// -----------------------------------------------------------------------------
661// Globals.
662// -----------------------------------------------------------------------------
663
664ZEND_BEGIN_MODULE_GLOBALS(protobuf)
665ZEND_END_MODULE_GLOBALS(protobuf)
666
667// Init module and PHP classes.
668void any_init(TSRMLS_D);
669void api_init(TSRMLS_D);
670void bool_value_init(TSRMLS_D);
671void bytes_value_init(TSRMLS_D);
672void descriptor_init(TSRMLS_D);
673void descriptor_pool_init(TSRMLS_D);
674void double_value_init(TSRMLS_D);
675void duration_init(TSRMLS_D);
676void empty_init(TSRMLS_D);
677void enum_descriptor_init(TSRMLS_D);
678void enum_init(TSRMLS_D);
679void enum_value_init(TSRMLS_D);
680void field_cardinality_init(TSRMLS_D);
681void field_descriptor_init(TSRMLS_D);
682void field_init(TSRMLS_D);
683void field_kind_init(TSRMLS_D);
684void field_mask_init(TSRMLS_D);
685void float_value_init(TSRMLS_D);
686void gpb_type_init(TSRMLS_D);
687void int32_value_init(TSRMLS_D);
688void int64_value_init(TSRMLS_D);
689void internal_descriptor_pool_init(TSRMLS_D);
690void list_value_init(TSRMLS_D);
691void map_field_init(TSRMLS_D);
692void map_field_iter_init(TSRMLS_D);
693void message_init(TSRMLS_D);
694void method_init(TSRMLS_D);
695void mixin_init(TSRMLS_D);
696void null_value_init(TSRMLS_D);
697void oneof_descriptor_init(TSRMLS_D);
698void option_init(TSRMLS_D);
699void repeated_field_init(TSRMLS_D);
700void repeated_field_iter_init(TSRMLS_D);
701void source_context_init(TSRMLS_D);
702void string_value_init(TSRMLS_D);
703void struct_init(TSRMLS_D);
704void syntax_init(TSRMLS_D);
705void timestamp_init(TSRMLS_D);
706void type_init(TSRMLS_D);
707void uint32_value_init(TSRMLS_D);
708void uint64_value_init(TSRMLS_D);
709void util_init(TSRMLS_D);
710void value_init(TSRMLS_D);
711
712void gpb_metadata_any_init(TSRMLS_D);
713void gpb_metadata_api_init(TSRMLS_D);
714void gpb_metadata_duration_init(TSRMLS_D);
715void gpb_metadata_field_mask_init(TSRMLS_D);
716void gpb_metadata_empty_init(TSRMLS_D);
717void gpb_metadata_source_context_init(TSRMLS_D);
718void gpb_metadata_struct_init(TSRMLS_D);
719void gpb_metadata_timestamp_init(TSRMLS_D);
720void gpb_metadata_type_init(TSRMLS_D);
721void gpb_metadata_wrappers_init(TSRMLS_D);
722
723// Global map from upb {msg,enum}defs to wrapper Descriptor/EnumDescriptor
724// instances.
725void add_def_obj(const void* def, PHP_PROTO_HASHTABLE_VALUE value);
726PHP_PROTO_HASHTABLE_VALUE get_def_obj(const void* def);
727
728// Global map from PHP class entries to wrapper Descriptor/EnumDescriptor
729// instances.
730void add_ce_obj(const void* ce, PHP_PROTO_HASHTABLE_VALUE value);
731PHP_PROTO_HASHTABLE_VALUE get_ce_obj(const void* ce);
732bool class_added(const void* ce);
733
734// Global map from message/enum's proto fully-qualified name to corresponding
735// wrapper Descriptor/EnumDescriptor instances.
736void add_proto_obj(const char* proto, PHP_PROTO_HASHTABLE_VALUE value);
737PHP_PROTO_HASHTABLE_VALUE get_proto_obj(const char* proto);
738
739extern zend_class_entry* map_field_type;
740extern zend_class_entry* repeated_field_type;
741
742// -----------------------------------------------------------------------------
743// Descriptor.
744// -----------------------------------------------------------------------------
745
746PHP_PROTO_WRAP_OBJECT_START(DescriptorPool)
747 InternalDescriptorPool* intern;
748PHP_PROTO_WRAP_OBJECT_END
749
750PHP_METHOD(DescriptorPool, getGeneratedPool);
751PHP_METHOD(DescriptorPool, getDescriptorByClassName);
752PHP_METHOD(DescriptorPool, getEnumDescriptorByClassName);
753
754PHP_PROTO_WRAP_OBJECT_START(InternalDescriptorPool)
755 upb_symtab* symtab;
756 HashTable* pending_list;
757PHP_PROTO_WRAP_OBJECT_END
758
759PHP_METHOD(InternalDescriptorPool, getGeneratedPool);
760PHP_METHOD(InternalDescriptorPool, internalAddGeneratedFile);
761
762void internal_add_generated_file(const char* data, PHP_PROTO_SIZE data_len,
763 InternalDescriptorPool* pool TSRMLS_DC);
764void init_generated_pool_once(TSRMLS_D);
765
766// wrapper of generated pool
767#if PHP_MAJOR_VERSION < 7
768extern zval* generated_pool_php;
769extern zval* internal_generated_pool_php;
770void descriptor_pool_free(void* object TSRMLS_DC);
771void internal_descriptor_pool_free(void* object TSRMLS_DC);
772#else
773extern zend_object *generated_pool_php;
774extern zend_object *internal_generated_pool_php;
775void descriptor_pool_free(zend_object* object);
776void internal_descriptor_pool_free(zend_object* object);
777#endif
778extern InternalDescriptorPool* generated_pool; // The actual generated pool
779
780PHP_PROTO_WRAP_OBJECT_START(Descriptor)
781 const upb_msgdef* msgdef;
782 MessageLayout* layout;
783 zend_class_entry* klass; // begins as NULL
784 const upb_handlers* fill_handlers;
785 const upb_pbdecodermethod* fill_method;
786 const upb_json_parsermethod* json_fill_method;
787 const upb_handlers* pb_serialize_handlers;
788 const upb_handlers* json_serialize_handlers;
789 const upb_handlers* json_serialize_handlers_preserve;
790PHP_PROTO_WRAP_OBJECT_END
791
792PHP_METHOD(Descriptor, getClass);
793PHP_METHOD(Descriptor, getFullName);
794PHP_METHOD(Descriptor, getField);
795PHP_METHOD(Descriptor, getFieldCount);
796PHP_METHOD(Descriptor, getOneofDecl);
797PHP_METHOD(Descriptor, getOneofDeclCount);
798
799extern zend_class_entry* descriptor_type;
800
801void descriptor_name_set(Descriptor *desc, const char *name);
802
803PHP_PROTO_WRAP_OBJECT_START(FieldDescriptor)
804 const upb_fielddef* fielddef;
805PHP_PROTO_WRAP_OBJECT_END
806
807PHP_METHOD(FieldDescriptor, getName);
808PHP_METHOD(FieldDescriptor, getNumber);
809PHP_METHOD(FieldDescriptor, getLabel);
810PHP_METHOD(FieldDescriptor, getType);
811PHP_METHOD(FieldDescriptor, isMap);
812PHP_METHOD(FieldDescriptor, getEnumType);
813PHP_METHOD(FieldDescriptor, getMessageType);
814
815extern zend_class_entry* field_descriptor_type;
816
817PHP_PROTO_WRAP_OBJECT_START(EnumDescriptor)
818 const upb_enumdef* enumdef;
819 zend_class_entry* klass; // begins as NULL
820PHP_PROTO_WRAP_OBJECT_END
821
822PHP_METHOD(EnumDescriptor, getValue);
823PHP_METHOD(EnumDescriptor, getValueCount);
824
825extern zend_class_entry* enum_descriptor_type;
826
827PHP_PROTO_WRAP_OBJECT_START(EnumValueDescriptor)
828 const char* name;
829 int32_t number;
830PHP_PROTO_WRAP_OBJECT_END
831
832PHP_METHOD(EnumValueDescriptor, getName);
833PHP_METHOD(EnumValueDescriptor, getNumber);
834
835extern zend_class_entry* enum_value_descriptor_type;
836
837// -----------------------------------------------------------------------------
838// Message class creation.
839// -----------------------------------------------------------------------------
840
841void* message_data(MessageHeader* msg);
842void custom_data_init(const zend_class_entry* ce,
843 MessageHeader* msg PHP_PROTO_TSRMLS_DC);
844
845// Build PHP class for given descriptor. Instead of building from scratch, this
846// function modifies existing class which has been partially defined in PHP
847// code.
848void build_class_from_descriptor(
849 PHP_PROTO_HASHTABLE_VALUE php_descriptor TSRMLS_DC);
850
851extern zend_class_entry* message_type;
852extern zend_object_handlers* message_handlers;
853
854// -----------------------------------------------------------------------------
855// Message layout / storage.
856// -----------------------------------------------------------------------------
857
858/*
859 * In c extension, each protobuf message is a zval instance. The zval instance
860 * is like union, which can be used to store int, string, zend_object_value and
861 * etc. For protobuf message, the zval instance is used to store the
862 * zend_object_value.
863 *
864 * The zend_object_value is composed of handlers and a handle to look up the
865 * actual stored data. The handlers are pointers to functions, e.g., read,
866 * write, and etc, to access properties.
867 *
868 * The actual data of protobuf messages is stored as MessageHeader in zend
869 * engine's central repository. Each MessageHeader instance is composed of a
870 * zend_object, a Descriptor instance and the real message data.
871 *
872 * For the reason that PHP's native types may not be large enough to store
873 * protobuf message's field (e.g., int64), all message's data is stored in
874 * custom memory layout and is indexed by the Descriptor instance.
875 *
876 * The zend_object contains the zend class entry and the properties table. The
877 * zend class entry contains all information about protobuf message's
878 * corresponding PHP class. The most useful information is the offset table of
879 * properties. Because read access to properties requires returning zval
880 * instance, we need to convert data from the custom layout to zval instance.
881 * Instead of creating zval instance for every read access, we use the zval
882 * instances in the properties table in the zend_object as cache. When
883 * accessing properties, the offset is needed to find the zval property in
884 * zend_object's properties table. These properties will be updated using the
885 * data from custom memory layout only when reading these properties.
886 *
887 * zval
888 * |-zend_object_value obj
889 * |-zend_object_handlers* handlers -> |-read_property_handler
890 * | |-write_property_handler
891 * | ++++++++++++++++++++++
892 * |-zend_object_handle handle -> + central repository +
893 * ++++++++++++++++++++++
894 * MessageHeader <-----------------|
895 * |-zend_object std
896 * | |-class_entry* ce -> class_entry
897 * | | |-HashTable properties_table (name->offset)
898 * | |-zval** properties_table <------------------------------|
899 * | |------> zval* property(cache)
900 * |-Descriptor* desc (name->offset)
901 * |-void** data <-----------|
902 * |-----------------------> void* property(data)
903 *
904 */
905
906#define MESSAGE_FIELD_NO_CASE ((size_t)-1)
907
908struct MessageField {
909 size_t offset;
910 int cache_index; // Each field except oneof field has a zval cache to avoid
911 // multiple creation when being accessed.
912 size_t case_offset; // for oneofs, a uint32. Else, MESSAGE_FIELD_NO_CASE.
913};
914
915struct MessageLayout {
916 const upb_msgdef* msgdef;
917 MessageField* fields;
918 size_t size;
919};
920
921PHP_PROTO_WRAP_OBJECT_START(MessageHeader)
922 void* data; // Point to the real message data.
923 // Place needs to be consistent with map_parse_frame_data_t.
924 Descriptor* descriptor; // Kept alive by self.class.descriptor reference.
925PHP_PROTO_WRAP_OBJECT_END
926
927MessageLayout* create_layout(const upb_msgdef* msgdef);
928void layout_init(MessageLayout* layout, void* storage,
929 zend_object* object PHP_PROTO_TSRMLS_DC);
930zval* layout_get(MessageLayout* layout, const void* storage,
931 const upb_fielddef* field, CACHED_VALUE* cache TSRMLS_DC);
932void layout_set(MessageLayout* layout, MessageHeader* header,
933 const upb_fielddef* field, zval* val TSRMLS_DC);
934void layout_merge(MessageLayout* layout, MessageHeader* from,
935 MessageHeader* to TSRMLS_DC);
936const char* layout_get_oneof_case(MessageLayout* layout, const void* storage,
937 const upb_oneofdef* oneof TSRMLS_DC);
938void free_layout(MessageLayout* layout);
939void* slot_memory(MessageLayout* layout, const void* storage,
940 const upb_fielddef* field);
941
942PHP_METHOD(Message, clear);
943PHP_METHOD(Message, mergeFrom);
944PHP_METHOD(Message, readOneof);
945PHP_METHOD(Message, writeOneof);
946PHP_METHOD(Message, whichOneof);
947PHP_METHOD(Message, __construct);
948
949// -----------------------------------------------------------------------------
950// Encode / Decode.
951// -----------------------------------------------------------------------------
952
953// Maximum depth allowed during encoding, to avoid stack overflows due to
954// cycles.
955#define ENCODE_MAX_NESTING 63
956
957// Constructs the upb decoder method for parsing messages of this type.
958// This is called from the message class creation code.
959const upb_pbdecodermethod *new_fillmsg_decodermethod(Descriptor *desc,
960 const void *owner);
961void serialize_to_string(zval* val, zval* return_value TSRMLS_DC);
962void merge_from_string(const char* data, int data_len, const Descriptor* desc,
963 MessageHeader* msg);
964
965PHP_METHOD(Message, serializeToString);
966PHP_METHOD(Message, mergeFromString);
967PHP_METHOD(Message, serializeToJsonString);
968PHP_METHOD(Message, mergeFromJsonString);
969PHP_METHOD(Message, discardUnknownFields);
970
971// -----------------------------------------------------------------------------
972// Type check / conversion.
973// -----------------------------------------------------------------------------
974
975bool protobuf_convert_to_int32(zval* from, int32_t* to);
976bool protobuf_convert_to_uint32(zval* from, uint32_t* to);
977bool protobuf_convert_to_int64(zval* from, int64_t* to);
978bool protobuf_convert_to_uint64(zval* from, uint64_t* to);
979bool protobuf_convert_to_float(zval* from, float* to);
980bool protobuf_convert_to_double(zval* from, double* to);
981bool protobuf_convert_to_bool(zval* from, int8_t* to);
982bool protobuf_convert_to_string(zval* from);
983
984void check_repeated_field(const zend_class_entry* klass, PHP_PROTO_LONG type,
985 zval* val, zval* return_value);
986void check_map_field(const zend_class_entry* klass, PHP_PROTO_LONG key_type,
987 PHP_PROTO_LONG value_type, zval* val, zval* return_value);
988
989PHP_METHOD(Util, checkInt32);
990PHP_METHOD(Util, checkUint32);
991PHP_METHOD(Util, checkInt64);
992PHP_METHOD(Util, checkUint64);
993PHP_METHOD(Util, checkEnum);
994PHP_METHOD(Util, checkFloat);
995PHP_METHOD(Util, checkDouble);
996PHP_METHOD(Util, checkBool);
997PHP_METHOD(Util, checkString);
998PHP_METHOD(Util, checkBytes);
999PHP_METHOD(Util, checkMessage);
1000PHP_METHOD(Util, checkMapField);
1001PHP_METHOD(Util, checkRepeatedField);
1002
1003// -----------------------------------------------------------------------------
1004// Native slot storage abstraction.
1005// -----------------------------------------------------------------------------
1006
1007#define NATIVE_SLOT_MAX_SIZE sizeof(uint64_t)
1008
1009size_t native_slot_size(upb_fieldtype_t type);
1010bool native_slot_set(upb_fieldtype_t type, const zend_class_entry* klass,
1011 void* memory, zval* value TSRMLS_DC);
1012// String/Message is stored differently in array/map from normal message fields.
1013// So we need to make a special method to handle that.
1014bool native_slot_set_by_array(upb_fieldtype_t type,
1015 const zend_class_entry* klass, void* memory,
1016 zval* value TSRMLS_DC);
1017bool native_slot_set_by_map(upb_fieldtype_t type, const zend_class_entry* klass,
1018 void* memory, zval* value TSRMLS_DC);
1019void native_slot_init(upb_fieldtype_t type, void* memory, CACHED_VALUE* cache);
1020// For each property, in order to avoid conversion between the zval object and
1021// the actual data type during parsing/serialization, the containing message
1022// object use the custom memory layout to store the actual data type for each
1023// property inside of it. To access a property from php code, the property
1024// needs to be converted to a zval object. The message object is not responsible
1025// for providing such a zval object. Instead the caller needs to provide one
1026// (cache) and update it with the actual data (memory).
1027void native_slot_get(upb_fieldtype_t type, const void* memory,
1028 CACHED_VALUE* cache TSRMLS_DC);
1029// String/Message is stored differently in array/map from normal message fields.
1030// So we need to make a special method to handle that.
1031void native_slot_get_by_array(upb_fieldtype_t type, const void* memory,
1032 CACHED_VALUE* cache TSRMLS_DC);
1033void native_slot_get_by_map_key(upb_fieldtype_t type, const void* memory,
1034 int length, CACHED_VALUE* cache TSRMLS_DC);
1035void native_slot_get_by_map_value(upb_fieldtype_t type, const void* memory,
1036 CACHED_VALUE* cache TSRMLS_DC);
1037void native_slot_get_default(upb_fieldtype_t type,
1038 CACHED_VALUE* cache TSRMLS_DC);
1039
1040// -----------------------------------------------------------------------------
1041// Map Field.
1042// -----------------------------------------------------------------------------
1043
1044extern zend_object_handlers* map_field_handlers;
1045extern zend_object_handlers* map_field_iter_handlers;
1046
1047PHP_PROTO_WRAP_OBJECT_START(Map)
1048 upb_fieldtype_t key_type;
1049 upb_fieldtype_t value_type;
1050 const zend_class_entry* msg_ce; // class entry for value message
1051 upb_strtable table;
1052PHP_PROTO_WRAP_OBJECT_END
1053
1054PHP_PROTO_WRAP_OBJECT_START(MapIter)
1055 Map* self;
1056 upb_strtable_iter it;
1057PHP_PROTO_WRAP_OBJECT_END
1058
1059void map_begin(zval* self, MapIter* iter TSRMLS_DC);
1060void map_next(MapIter* iter);
1061bool map_done(MapIter* iter);
1062const char* map_iter_key(MapIter* iter, int* len);
1063upb_value map_iter_value(MapIter* iter, int* len);
1064
1065// These operate on a map-entry msgdef.
1066const upb_fielddef* map_entry_key(const upb_msgdef* msgdef);
1067const upb_fielddef* map_entry_value(const upb_msgdef* msgdef);
1068
1069void map_field_create_with_field(const zend_class_entry* ce,
1070 const upb_fielddef* field,
1071 CACHED_VALUE* map_field PHP_PROTO_TSRMLS_DC);
1072void map_field_create_with_type(const zend_class_entry* ce,
1073 upb_fieldtype_t key_type,
1074 upb_fieldtype_t value_type,
1075 const zend_class_entry* msg_ce,
1076 CACHED_VALUE* map_field PHP_PROTO_TSRMLS_DC);
1077void* upb_value_memory(upb_value* v);
1078
1079#define MAP_KEY_FIELD 1
1080#define MAP_VALUE_FIELD 2
1081
1082// These operate on a map field (i.e., a repeated field of submessages whose
1083// submessage type is a map-entry msgdef).
1084bool is_map_field(const upb_fielddef* field);
1085const upb_fielddef* map_field_key(const upb_fielddef* field);
1086const upb_fielddef* map_field_value(const upb_fielddef* field);
1087
1088bool map_index_set(Map *intern, const char* keyval, int length, upb_value v);
1089
1090PHP_METHOD(MapField, __construct);
1091PHP_METHOD(MapField, offsetExists);
1092PHP_METHOD(MapField, offsetGet);
1093PHP_METHOD(MapField, offsetSet);
1094PHP_METHOD(MapField, offsetUnset);
1095PHP_METHOD(MapField, count);
1096PHP_METHOD(MapField, getIterator);
1097
1098PHP_METHOD(MapFieldIter, rewind);
1099PHP_METHOD(MapFieldIter, current);
1100PHP_METHOD(MapFieldIter, key);
1101PHP_METHOD(MapFieldIter, next);
1102PHP_METHOD(MapFieldIter, valid);
1103
1104// -----------------------------------------------------------------------------
1105// Repeated Field.
1106// -----------------------------------------------------------------------------
1107
1108extern zend_object_handlers* repeated_field_handlers;
1109extern zend_object_handlers* repeated_field_iter_handlers;
1110
1111PHP_PROTO_WRAP_OBJECT_START(RepeatedField)
1112#if PHP_MAJOR_VERSION < 7
1113 zval* array;
1114#else
1115 zval array;
1116#endif
1117 upb_fieldtype_t type;
1118 const zend_class_entry* msg_ce; // class entry for containing message
1119 // (for message field only).
1120PHP_PROTO_WRAP_OBJECT_END
1121
1122PHP_PROTO_WRAP_OBJECT_START(RepeatedFieldIter)
1123 RepeatedField* repeated_field;
1124 long position;
1125PHP_PROTO_WRAP_OBJECT_END
1126
1127void repeated_field_create_with_field(
1128 zend_class_entry* ce, const upb_fielddef* field,
1129 CACHED_VALUE* repeated_field PHP_PROTO_TSRMLS_DC);
1130void repeated_field_create_with_type(
1131 zend_class_entry* ce, upb_fieldtype_t type, const zend_class_entry* msg_ce,
1132 CACHED_VALUE* repeated_field PHP_PROTO_TSRMLS_DC);
1133// Return the element at the index position from the repeated field. There is
1134// not restriction on the type of stored elements.
1135void *repeated_field_index_native(RepeatedField *intern, int index TSRMLS_DC);
1136// Add the element to the end of the repeated field. There is not restriction on
1137// the type of stored elements.
1138void repeated_field_push_native(RepeatedField *intern, void *value);
1139
1140PHP_METHOD(RepeatedField, __construct);
1141PHP_METHOD(RepeatedField, append);
1142PHP_METHOD(RepeatedField, offsetExists);
1143PHP_METHOD(RepeatedField, offsetGet);
1144PHP_METHOD(RepeatedField, offsetSet);
1145PHP_METHOD(RepeatedField, offsetUnset);
1146PHP_METHOD(RepeatedField, count);
1147PHP_METHOD(RepeatedField, getIterator);
1148
1149PHP_METHOD(RepeatedFieldIter, rewind);
1150PHP_METHOD(RepeatedFieldIter, current);
1151PHP_METHOD(RepeatedFieldIter, key);
1152PHP_METHOD(RepeatedFieldIter, next);
1153PHP_METHOD(RepeatedFieldIter, valid);
1154
1155// -----------------------------------------------------------------------------
1156// Oneof Field.
1157// -----------------------------------------------------------------------------
1158
1159PHP_PROTO_WRAP_OBJECT_START(Oneof)
1160 upb_oneofdef* oneofdef;
1161 int index; // Index of field in oneof. -1 if not set.
1162 char value[NATIVE_SLOT_MAX_SIZE];
1163PHP_PROTO_WRAP_OBJECT_END
1164
1165PHP_METHOD(Oneof, getName);
1166PHP_METHOD(Oneof, getField);
1167PHP_METHOD(Oneof, getFieldCount);
1168
1169extern zend_class_entry* oneof_descriptor_type;
1170
1171// Oneof case slot value to indicate that no oneof case is set. The value `0` is
1172// safe because field numbers are used as case identifiers, and no field can
1173// have a number of 0.
1174#define ONEOF_CASE_NONE 0
1175
1176// -----------------------------------------------------------------------------
1177// Well Known Type.
1178// -----------------------------------------------------------------------------
1179
1180extern bool is_inited_file_any;
1181extern bool is_inited_file_api;
1182extern bool is_inited_file_duration;
1183extern bool is_inited_file_field_mask;
1184extern bool is_inited_file_empty;
1185extern bool is_inited_file_source_context;
1186extern bool is_inited_file_struct;
1187extern bool is_inited_file_timestamp;
1188extern bool is_inited_file_type;
1189extern bool is_inited_file_wrappers;
1190
1191PHP_METHOD(GPBMetadata_Any, initOnce);
1192PHP_METHOD(GPBMetadata_Api, initOnce);
1193PHP_METHOD(GPBMetadata_Duration, initOnce);
1194PHP_METHOD(GPBMetadata_FieldMask, initOnce);
1195PHP_METHOD(GPBMetadata_Empty, initOnce);
1196PHP_METHOD(GPBMetadata_SourceContext, initOnce);
1197PHP_METHOD(GPBMetadata_Struct, initOnce);
1198PHP_METHOD(GPBMetadata_Timestamp, initOnce);
1199PHP_METHOD(GPBMetadata_Type, initOnce);
1200PHP_METHOD(GPBMetadata_Wrappers, initOnce);
1201
1202PHP_METHOD(Any, __construct);
1203PHP_METHOD(Any, getTypeUrl);
1204PHP_METHOD(Any, setTypeUrl);
1205PHP_METHOD(Any, getValue);
1206PHP_METHOD(Any, setValue);
1207PHP_METHOD(Any, unpack);
1208PHP_METHOD(Any, pack);
1209PHP_METHOD(Any, is);
1210
1211PHP_METHOD(Duration, __construct);
1212PHP_METHOD(Duration, getSeconds);
1213PHP_METHOD(Duration, setSeconds);
1214PHP_METHOD(Duration, getNanos);
1215PHP_METHOD(Duration, setNanos);
1216
1217PHP_METHOD(Timestamp, __construct);
1218PHP_METHOD(Timestamp, fromDateTime);
1219PHP_METHOD(Timestamp, toDateTime);
1220PHP_METHOD(Timestamp, getSeconds);
1221PHP_METHOD(Timestamp, setSeconds);
1222PHP_METHOD(Timestamp, getNanos);
1223PHP_METHOD(Timestamp, setNanos);
1224
1225PHP_METHOD(Api, __construct);
1226PHP_METHOD(Api, getName);
1227PHP_METHOD(Api, setName);
1228PHP_METHOD(Api, getMethods);
1229PHP_METHOD(Api, setMethods);
1230PHP_METHOD(Api, getOptions);
1231PHP_METHOD(Api, setOptions);
1232PHP_METHOD(Api, getVersion);
1233PHP_METHOD(Api, setVersion);
1234PHP_METHOD(Api, getSourceContext);
1235PHP_METHOD(Api, setSourceContext);
1236PHP_METHOD(Api, getMixins);
1237PHP_METHOD(Api, setMixins);
1238PHP_METHOD(Api, getSyntax);
1239PHP_METHOD(Api, setSyntax);
1240
1241PHP_METHOD(BoolValue, __construct);
1242PHP_METHOD(BoolValue, getValue);
1243PHP_METHOD(BoolValue, setValue);
1244
1245PHP_METHOD(BytesValue, __construct);
1246PHP_METHOD(BytesValue, getValue);
1247PHP_METHOD(BytesValue, setValue);
1248
1249PHP_METHOD(DoubleValue, __construct);
1250PHP_METHOD(DoubleValue, getValue);
1251PHP_METHOD(DoubleValue, setValue);
1252
1253PHP_METHOD(Enum, __construct);
1254PHP_METHOD(Enum, getName);
1255PHP_METHOD(Enum, setName);
1256PHP_METHOD(Enum, getEnumvalue);
1257PHP_METHOD(Enum, setEnumvalue);
1258PHP_METHOD(Enum, getOptions);
1259PHP_METHOD(Enum, setOptions);
1260PHP_METHOD(Enum, getSourceContext);
1261PHP_METHOD(Enum, setSourceContext);
1262PHP_METHOD(Enum, getSyntax);
1263PHP_METHOD(Enum, setSyntax);
1264
1265PHP_METHOD(EnumValue, __construct);
1266PHP_METHOD(EnumValue, getName);
1267PHP_METHOD(EnumValue, setName);
1268PHP_METHOD(EnumValue, getNumber);
1269PHP_METHOD(EnumValue, setNumber);
1270PHP_METHOD(EnumValue, getOptions);
1271PHP_METHOD(EnumValue, setOptions);
1272
1273PHP_METHOD(FieldMask, __construct);
1274PHP_METHOD(FieldMask, getPaths);
1275PHP_METHOD(FieldMask, setPaths);
1276
1277PHP_METHOD(Field, __construct);
1278PHP_METHOD(Field, getKind);
1279PHP_METHOD(Field, setKind);
1280PHP_METHOD(Field, getCardinality);
1281PHP_METHOD(Field, setCardinality);
1282PHP_METHOD(Field, getNumber);
1283PHP_METHOD(Field, setNumber);
1284PHP_METHOD(Field, getName);
1285PHP_METHOD(Field, setName);
1286PHP_METHOD(Field, getTypeUrl);
1287PHP_METHOD(Field, setTypeUrl);
1288PHP_METHOD(Field, getOneofIndex);
1289PHP_METHOD(Field, setOneofIndex);
1290PHP_METHOD(Field, getPacked);
1291PHP_METHOD(Field, setPacked);
1292PHP_METHOD(Field, getOptions);
1293PHP_METHOD(Field, setOptions);
1294PHP_METHOD(Field, getJsonName);
1295PHP_METHOD(Field, setJsonName);
1296PHP_METHOD(Field, getDefaultValue);
1297PHP_METHOD(Field, setDefaultValue);
1298
1299PHP_METHOD(FloatValue, __construct);
1300PHP_METHOD(FloatValue, getValue);
1301PHP_METHOD(FloatValue, setValue);
1302
1303PHP_METHOD(GPBEmpty, __construct);
1304
1305PHP_METHOD(Int32Value, __construct);
1306PHP_METHOD(Int32Value, getValue);
1307PHP_METHOD(Int32Value, setValue);
1308
1309PHP_METHOD(Int64Value, __construct);
1310PHP_METHOD(Int64Value, getValue);
1311PHP_METHOD(Int64Value, setValue);
1312
1313PHP_METHOD(ListValue, __construct);
1314PHP_METHOD(ListValue, getValues);
1315PHP_METHOD(ListValue, setValues);
1316
1317PHP_METHOD(Method, __construct);
1318PHP_METHOD(Method, getName);
1319PHP_METHOD(Method, setName);
1320PHP_METHOD(Method, getRequestTypeUrl);
1321PHP_METHOD(Method, setRequestTypeUrl);
1322PHP_METHOD(Method, getRequestStreaming);
1323PHP_METHOD(Method, setRequestStreaming);
1324PHP_METHOD(Method, getResponseTypeUrl);
1325PHP_METHOD(Method, setResponseTypeUrl);
1326PHP_METHOD(Method, getResponseStreaming);
1327PHP_METHOD(Method, setResponseStreaming);
1328PHP_METHOD(Method, getOptions);
1329PHP_METHOD(Method, setOptions);
1330PHP_METHOD(Method, getSyntax);
1331PHP_METHOD(Method, setSyntax);
1332
1333PHP_METHOD(Mixin, __construct);
1334PHP_METHOD(Mixin, getName);
1335PHP_METHOD(Mixin, setName);
1336PHP_METHOD(Mixin, getRoot);
1337PHP_METHOD(Mixin, setRoot);
1338
1339PHP_METHOD(Option, __construct);
1340PHP_METHOD(Option, getName);
1341PHP_METHOD(Option, setName);
1342PHP_METHOD(Option, getValue);
1343PHP_METHOD(Option, setValue);
1344
1345PHP_METHOD(SourceContext, __construct);
1346PHP_METHOD(SourceContext, getFileName);
1347PHP_METHOD(SourceContext, setFileName);
1348
1349PHP_METHOD(StringValue, __construct);
1350PHP_METHOD(StringValue, getValue);
1351PHP_METHOD(StringValue, setValue);
1352
1353PHP_METHOD(Struct, __construct);
1354PHP_METHOD(Struct, getFields);
1355PHP_METHOD(Struct, setFields);
1356
1357PHP_METHOD(Type, __construct);
1358PHP_METHOD(Type, getName);
1359PHP_METHOD(Type, setName);
1360PHP_METHOD(Type, getFields);
1361PHP_METHOD(Type, setFields);
1362PHP_METHOD(Type, getOneofs);
1363PHP_METHOD(Type, setOneofs);
1364PHP_METHOD(Type, getOptions);
1365PHP_METHOD(Type, setOptions);
1366PHP_METHOD(Type, getSourceContext);
1367PHP_METHOD(Type, setSourceContext);
1368PHP_METHOD(Type, getSyntax);
1369PHP_METHOD(Type, setSyntax);
1370
1371PHP_METHOD(UInt32Value, __construct);
1372PHP_METHOD(UInt32Value, getValue);
1373PHP_METHOD(UInt32Value, setValue);
1374
1375PHP_METHOD(UInt64Value, __construct);
1376PHP_METHOD(UInt64Value, getValue);
1377PHP_METHOD(UInt64Value, setValue);
1378
1379PHP_METHOD(Value, __construct);
1380PHP_METHOD(Value, getNullValue);
1381PHP_METHOD(Value, setNullValue);
1382PHP_METHOD(Value, getNumberValue);
1383PHP_METHOD(Value, setNumberValue);
1384PHP_METHOD(Value, getStringValue);
1385PHP_METHOD(Value, setStringValue);
1386PHP_METHOD(Value, getBoolValue);
1387PHP_METHOD(Value, setBoolValue);
1388PHP_METHOD(Value, getStructValue);
1389PHP_METHOD(Value, setStructValue);
1390PHP_METHOD(Value, getListValue);
1391PHP_METHOD(Value, setListValue);
1392PHP_METHOD(Value, getKind);
1393
1394extern zend_class_entry* any_type;
1395extern zend_class_entry* api_type;
1396extern zend_class_entry* bool_value_type;
1397extern zend_class_entry* bytes_value_type;
1398extern zend_class_entry* double_value_type;
1399extern zend_class_entry* duration_type;
1400extern zend_class_entry* empty_type;
1401extern zend_class_entry* enum_type;
1402extern zend_class_entry* enum_value_type;
1403extern zend_class_entry* field_cardinality_type;
1404extern zend_class_entry* field_kind_type;
1405extern zend_class_entry* field_mask_type;
1406extern zend_class_entry* field_type;
1407extern zend_class_entry* float_value_type;
1408extern zend_class_entry* int32_value_type;
1409extern zend_class_entry* int64_value_type;
1410extern zend_class_entry* list_value_type;
1411extern zend_class_entry* method_type;
1412extern zend_class_entry* mixin_type;
1413extern zend_class_entry* null_value_type;
1414extern zend_class_entry* option_type;
1415extern zend_class_entry* source_context_type;
1416extern zend_class_entry* string_value_type;
1417extern zend_class_entry* struct_type;
1418extern zend_class_entry* syntax_type;
1419extern zend_class_entry* timestamp_type;
1420extern zend_class_entry* type_type;
1421extern zend_class_entry* uint32_value_type;
1422extern zend_class_entry* uint64_value_type;
1423extern zend_class_entry* value_type;
1424
1425// -----------------------------------------------------------------------------
1426// Upb.
1427// -----------------------------------------------------------------------------
1428
1429upb_fieldtype_t to_fieldtype(upb_descriptortype_t type);
1430const zend_class_entry* field_type_class(
1431 const upb_fielddef* field PHP_PROTO_TSRMLS_DC);
1432
1433// -----------------------------------------------------------------------------
1434// Utilities.
1435// -----------------------------------------------------------------------------
1436
1437// Memory management
1438#define ALLOC(class_name) (class_name*) emalloc(sizeof(class_name))
1439#define PEMALLOC(class_name) (class_name*) pemalloc(sizeof(class_name), 1)
1440#define ALLOC_N(class_name, n) (class_name*) emalloc(sizeof(class_name) * n)
1441#define FREE(object) efree(object)
1442#define PEFREE(object) pefree(object, 1)
1443
1444// String argument.
1445#define STR(str) (str), strlen(str)
1446
1447// Zend Value
1448#if PHP_MAJOR_VERSION < 7
1449#define Z_OBJ_P(zval_p) \
1450 ((zend_object*)(EG(objects_store) \
1451 .object_buckets[Z_OBJ_HANDLE_P(zval_p)] \
1452 .bucket.obj.object))
1453#endif
1454
1455// Message handler
1456static inline zval* php_proto_message_read_property(
1457 zval* msg, zval* member PHP_PROTO_TSRMLS_DC) {
1458#if PHP_MAJOR_VERSION < 7
1459 return message_handlers->read_property(msg, member, BP_VAR_R,
1460 NULL PHP_PROTO_TSRMLS_CC);
1461#else
1462 return message_handlers->read_property(msg, member, BP_VAR_R, NULL,
1463 NULL PHP_PROTO_TSRMLS_CC);
1464#endif
1465}
1466
1467// Reserved name
1468bool is_reserved_name(const char* name);
1469bool is_valid_constant_name(const char* name);
1470
1471#endif // __GOOGLE_PROTOBUF_PHP_PROTOBUF_H__