| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500 | // Protocol Buffers - Google's data interchange format// Copyright 2008 Google Inc.  All rights reserved.// https://developers.google.com/protocol-buffers///// Redistribution and use in source and binary forms, with or without// modification, are permitted provided that the following conditions are// met:////     * Redistributions of source code must retain the above copyright// notice, this list of conditions and the following disclaimer.//     * Redistributions in binary form must reproduce the above// copyright notice, this list of conditions and the following disclaimer// in the documentation and/or other materials provided with the// distribution.//     * Neither the name of Google Inc. nor the names of its// contributors may be used to endorse or promote products derived from// this software without specific prior written permission.//// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.#include "protobuf.h"// Forward declare.static void descriptor_init_c_instance(Descriptor* intern TSRMLS_DC);static void descriptor_free_c(Descriptor* object TSRMLS_DC);static void enum_descriptor_init_c_instance(EnumDescriptor* intern TSRMLS_DC);static void enum_descriptor_free_c(EnumDescriptor* object TSRMLS_DC);static void descriptor_pool_free_c(DescriptorPool* object TSRMLS_DC);static void descriptor_pool_init_c_instance(DescriptorPool* pool TSRMLS_DC);// -----------------------------------------------------------------------------// Common Utilities// -----------------------------------------------------------------------------static void check_upb_status(const upb_status* status, const char* msg) {  if (!upb_ok(status)) {    zend_error(E_ERROR, "%s: %s\n", msg, upb_status_errmsg(status));  }}static void upb_filedef_free(void *r) {  upb_filedef *f = *(upb_filedef **)r;  size_t i;  for (i = 0; i < upb_filedef_depcount(f); i++) {    upb_filedef_unref(upb_filedef_dep(f, i), f);  }  upb_inttable_uninit(&f->defs);  upb_inttable_uninit(&f->deps);  upb_gfree((void *)f->name);  upb_gfree((void *)f->package);  upb_gfree(f);}// Camel-case the field name and append "Entry" for generated map entry name.// e.g. map<KeyType, ValueType> foo_map => FooMapEntrystatic void append_map_entry_name(char *result, const char *field_name,                                  int pos) {  bool cap_next = true;  int i;  for (i = 0; i < strlen(field_name); ++i) {    if (field_name[i] == '_') {      cap_next = true;    } else if (cap_next) {      // Note: Do not use ctype.h due to locales.      if ('a' <= field_name[i] && field_name[i] <= 'z') {        result[pos++] = field_name[i] - 'a' + 'A';      } else {        result[pos++] = field_name[i];      }      cap_next = false;    } else {      result[pos++] = field_name[i];    }  }  strcat(result, "Entry");}#define CHECK_UPB(code, msg)             \  do {                                   \    upb_status status = UPB_STATUS_INIT; \    code;                                \    check_upb_status(&status, msg);      \  } while (0)// Define PHP class#define DEFINE_PROTOBUF_INIT_CLASS(CLASSNAME, CAMELNAME, LOWERNAME) \  PHP_PROTO_INIT_CLASS_START(CLASSNAME, CAMELNAME, LOWERNAME)       \  PHP_PROTO_INIT_CLASS_END#define DEFINE_PROTOBUF_CREATE(NAME, LOWERNAME)  \  PHP_PROTO_OBJECT_CREATE_START(NAME, LOWERNAME) \  LOWERNAME##_init_c_instance(intern TSRMLS_CC); \  PHP_PROTO_OBJECT_CREATE_END(NAME, LOWERNAME)#define DEFINE_PROTOBUF_FREE(CAMELNAME, LOWERNAME)  \  PHP_PROTO_OBJECT_FREE_START(CAMELNAME, LOWERNAME) \  LOWERNAME##_free_c(intern TSRMLS_CC);             \  PHP_PROTO_OBJECT_FREE_END#define DEFINE_PROTOBUF_DTOR(CAMELNAME, LOWERNAME)  \  PHP_PROTO_OBJECT_DTOR_START(CAMELNAME, LOWERNAME) \  PHP_PROTO_OBJECT_DTOR_END#define DEFINE_CLASS(NAME, LOWERNAME, string_name) \  zend_class_entry *LOWERNAME##_type;              \  zend_object_handlers *LOWERNAME##_handlers;      \  DEFINE_PROTOBUF_FREE(NAME, LOWERNAME)            \  DEFINE_PROTOBUF_DTOR(NAME, LOWERNAME)            \  DEFINE_PROTOBUF_CREATE(NAME, LOWERNAME)          \  DEFINE_PROTOBUF_INIT_CLASS(string_name, NAME, LOWERNAME)// -----------------------------------------------------------------------------// GPBType// -----------------------------------------------------------------------------zend_class_entry* gpb_type_type;static zend_function_entry gpb_type_methods[] = {  ZEND_FE_END};void gpb_type_init(TSRMLS_D) {  zend_class_entry class_type;  INIT_CLASS_ENTRY(class_type, "Google\\Protobuf\\Internal\\GPBType",                   gpb_type_methods);  gpb_type_type = zend_register_internal_class(&class_type TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("DOUBLE"),  1 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("FLOAT"),   2 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("INT64"),   3 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("UINT64"),  4 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("INT32"),   5 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("FIXED64"), 6 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("FIXED32"), 7 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("BOOL"),    8 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("STRING"),  9 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("GROUP"),   10 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("MESSAGE"), 11 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("BYTES"),   12 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("UINT32"),  13 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("ENUM"),    14 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("SFIXED32"),                                   15 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("SFIXED64"),                                   16 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("SINT32"), 17 TSRMLS_CC);  zend_declare_class_constant_long(gpb_type_type, STR("SINT64"), 18 TSRMLS_CC);}// -----------------------------------------------------------------------------// Descriptor// -----------------------------------------------------------------------------static zend_function_entry descriptor_methods[] = {  ZEND_FE_END};DEFINE_CLASS(Descriptor, descriptor, "Google\\Protobuf\\Internal\\Descriptor");static void descriptor_free_c(Descriptor *self TSRMLS_DC) {  if (self->layout) {    free_layout(self->layout);  }  if (self->fill_handlers) {    upb_handlers_unref(self->fill_handlers, &self->fill_handlers);  }  if (self->fill_method) {    upb_pbdecodermethod_unref(self->fill_method, &self->fill_method);  }  if (self->json_fill_method) {    upb_json_parsermethod_unref(self->json_fill_method,                                &self->json_fill_method);  }  if (self->pb_serialize_handlers) {    upb_handlers_unref(self->pb_serialize_handlers,                       &self->pb_serialize_handlers);  }  if (self->json_serialize_handlers) {    upb_handlers_unref(self->json_serialize_handlers,                       &self->json_serialize_handlers);  }  if (self->json_serialize_handlers_preserve) {    upb_handlers_unref(self->json_serialize_handlers_preserve,                       &self->json_serialize_handlers_preserve);  }}static void descriptor_init_c_instance(Descriptor *desc TSRMLS_DC) {  // zend_object_std_init(&desc->std, descriptor_type TSRMLS_CC);  desc->msgdef = NULL;  desc->layout = NULL;  desc->klass = NULL;  desc->fill_handlers = NULL;  desc->fill_method = NULL;  desc->json_fill_method = NULL;  desc->pb_serialize_handlers = NULL;  desc->json_serialize_handlers = NULL;  desc->json_serialize_handlers_preserve = NULL;}// -----------------------------------------------------------------------------// EnumDescriptor// -----------------------------------------------------------------------------static zend_function_entry enum_descriptor_methods[] = {  ZEND_FE_END};DEFINE_CLASS(EnumDescriptor, enum_descriptor,             "Google\\Protobuf\\Internal\\EnumDescriptor");static void enum_descriptor_free_c(EnumDescriptor *self TSRMLS_DC) {}static void enum_descriptor_init_c_instance(EnumDescriptor *self TSRMLS_DC) {  // zend_object_std_init(&self->std, enum_descriptor_type TSRMLS_CC);  self->enumdef = NULL;  self->klass = NULL;}// -----------------------------------------------------------------------------// FieldDescriptor// -----------------------------------------------------------------------------upb_fieldtype_t to_fieldtype(upb_descriptortype_t type) {  switch (type) {#define CASE(descriptor_type, type)           \  case UPB_DESCRIPTOR_TYPE_##descriptor_type: \    return UPB_TYPE_##type;  CASE(FLOAT,    FLOAT);  CASE(DOUBLE,   DOUBLE);  CASE(BOOL,     BOOL);  CASE(STRING,   STRING);  CASE(BYTES,    BYTES);  CASE(MESSAGE,  MESSAGE);  CASE(GROUP,    MESSAGE);  CASE(ENUM,     ENUM);  CASE(INT32,    INT32);  CASE(INT64,    INT64);  CASE(UINT32,   UINT32);  CASE(UINT64,   UINT64);  CASE(SINT32,   INT32);  CASE(SINT64,   INT64);  CASE(FIXED32,  UINT32);  CASE(FIXED64,  UINT64);  CASE(SFIXED32, INT32);  CASE(SFIXED64, INT64);#undef CONVERT  }  zend_error(E_ERROR, "Unknown field type.");  return 0;}// -----------------------------------------------------------------------------// DescriptorPool// -----------------------------------------------------------------------------static zend_function_entry descriptor_pool_methods[] = {  PHP_ME(DescriptorPool, getGeneratedPool, NULL,         ZEND_ACC_PUBLIC|ZEND_ACC_STATIC)  PHP_ME(DescriptorPool, internalAddGeneratedFile, NULL, ZEND_ACC_PUBLIC)  ZEND_FE_END};DEFINE_CLASS(DescriptorPool, descriptor_pool,             "Google\\Protobuf\\Internal\\DescriptorPool");// wrapper of generated pool#if PHP_MAJOR_VERSION < 7zval* generated_pool_php;#elsezend_object *generated_pool_php;#endifDescriptorPool *generated_pool;  // The actual generated poolstatic void init_generated_pool_once(TSRMLS_D) {  if (generated_pool_php == NULL) {#if PHP_MAJOR_VERSION < 7    MAKE_STD_ZVAL(generated_pool_php);    ZVAL_OBJ(generated_pool_php, descriptor_pool_type->create_object(                                     descriptor_pool_type TSRMLS_CC));    generated_pool = UNBOX(DescriptorPool, generated_pool_php);#else    generated_pool_php =        descriptor_pool_type->create_object(descriptor_pool_type TSRMLS_CC);    generated_pool = (DescriptorPool *)((char *)generated_pool_php -                                        XtOffsetOf(DescriptorPool, std));#endif  }}static void descriptor_pool_init_c_instance(DescriptorPool *pool TSRMLS_DC) {  // zend_object_std_init(&pool->std, descriptor_pool_type TSRMLS_CC);  pool->symtab = upb_symtab_new();  ALLOC_HASHTABLE(pool->pending_list);  zend_hash_init(pool->pending_list, 1, NULL, ZVAL_PTR_DTOR, 0);}static void descriptor_pool_free_c(DescriptorPool *pool TSRMLS_DC) {  upb_symtab_free(pool->symtab);  zend_hash_destroy(pool->pending_list);  FREE_HASHTABLE(pool->pending_list);}static void validate_enumdef(const upb_enumdef *enumdef) {  // Verify that an entry exists with integer value 0. (This is the default  // value.)  const char *lookup = upb_enumdef_iton(enumdef, 0);  if (lookup == NULL) {    zend_error(E_USER_ERROR,               "Enum definition does not contain a value for '0'.");  }}static void validate_msgdef(const upb_msgdef* msgdef) {  // Verify that no required fields exist. proto3 does not support these.  upb_msg_field_iter it;  for (upb_msg_field_begin(&it, msgdef);       !upb_msg_field_done(&it);       upb_msg_field_next(&it)) {    const upb_fielddef* field = upb_msg_iter_field(&it);    if (upb_fielddef_label(field) == UPB_LABEL_REQUIRED) {      zend_error(E_ERROR, "Required fields are unsupported in proto3.");    }  }}PHP_METHOD(DescriptorPool, getGeneratedPool) {  init_generated_pool_once(TSRMLS_C);#if PHP_MAJOR_VERSION < 7  RETURN_ZVAL(generated_pool_php, 1, 0);#else  ++GC_REFCOUNT(generated_pool_php);  RETURN_OBJ(generated_pool_php);#endif}static void convert_to_class_name_inplace(char *class_name,                                          const char* fullname,                                          const char* prefix,                                          const char* package_name) {  size_t i = 0, j;  bool first_char = true;  size_t pkg_name_len = package_name == NULL ? 0 : strlen(package_name);  size_t prefix_len = prefix == NULL ? 0 : strlen(prefix);  size_t message_name_start = package_name == NULL ? 0 : pkg_name_len + 1;  size_t message_len = (strlen(fullname) - message_name_start);  // In php, class name cannot be Empty.  if (strcmp("google.protobuf.Empty", fullname) == 0) {    strcpy(class_name, "\\Google\\Protobuf\\GPBEmpty");    return;  }  if (pkg_name_len != 0) {    class_name[i++] = '\\';    for (j = 0; j < pkg_name_len; j++) {      // php packages are divided by '\'.      if (package_name[j] == '.') {        class_name[i++] = '\\';        first_char = true;      } else if (first_char) {        // PHP package uses camel case.        if (package_name[j] < 'A' || package_name[j] > 'Z') {          class_name[i++] = package_name[j] + 'A' - 'a';        } else {          class_name[i++] = package_name[j];        }        first_char = false;      } else {        class_name[i++] = package_name[j];      }    }    class_name[i++] = '\\';  }  if (prefix_len > 0) {    strcpy(class_name + i, prefix);    i += prefix_len;  }  // Submessage is concatenated with its containing messages by '_'.  for (j = message_name_start; j < message_name_start + message_len; j++) {    if (fullname[j] == '.') {      class_name[i++] = '_';    } else {      class_name[i++] = fullname[j];    }  }}PHP_METHOD(DescriptorPool, internalAddGeneratedFile) {  char *data = NULL;  PHP_PROTO_SIZE data_len;  upb_filedef **files;  size_t i;  if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s", &data, &data_len) ==      FAILURE) {    return;  }  DescriptorPool *pool = UNBOX(DescriptorPool, getThis());  CHECK_UPB(files = upb_loaddescriptor(data, data_len, &pool, &status),            "Parse binary descriptors to internal descriptors failed");  // This method is called only once in each file.  assert(files[0] != NULL);  assert(files[1] == NULL);  CHECK_UPB(upb_symtab_addfile(pool->symtab, files[0], &status),            "Unable to add file to DescriptorPool");  // For each enum/message, we need its PHP class, upb descriptor and its PHP  // wrapper. These information are needed later for encoding, decoding and type  // checking. However, sometimes we just have one of them. In order to find  // them quickly, here, we store the mapping for them.  for (i = 0; i < upb_filedef_defcount(files[0]); i++) {    const upb_def *def = upb_filedef_def(files[0], i);    switch (upb_def_type(def)) {#define CASE_TYPE(def_type, def_type_lower, desc_type, desc_type_lower)        \  case UPB_DEF_##def_type: {                                                   \    CREATE_HASHTABLE_VALUE(desc, desc_php, desc_type, desc_type_lower##_type); \    const upb_##def_type_lower *def_type_lower =                               \        upb_downcast_##def_type_lower(def);                                    \    desc->def_type_lower = def_type_lower;                                     \    add_def_obj(desc->def_type_lower, desc_php);                               \    /* Unlike other messages, MapEntry is shared by all map fields and doesn't \     * have generated PHP class.*/                                             \    if (upb_def_type(def) == UPB_DEF_MSG &&                                    \        upb_msgdef_mapentry(upb_downcast_msgdef(def))) {                       \      break;                                                                   \    }                                                                          \    /* Prepend '.' to package name to make it absolute. In the 5 additional    \     * bytes allocated, one for '.', one for trailing 0, and 3 for 'GPB' if    \     * given message is google.protobuf.Empty.*/                               \    const char *fullname = upb_##def_type_lower##_fullname(def_type_lower);    \    const char *prefix = upb_filedef_phpprefix(files[0]);                      \    size_t klass_name_len = strlen(fullname) + 5;                              \    if (prefix != NULL) {                                                      \      klass_name_len += strlen(prefix);                                        \    }                                                                          \    char *klass_name = ecalloc(sizeof(char), klass_name_len);                  \    convert_to_class_name_inplace(klass_name, fullname, prefix,                \                                  upb_filedef_package(files[0]));              \    PHP_PROTO_CE_DECLARE pce;                                                  \    if (php_proto_zend_lookup_class(klass_name, strlen(klass_name), &pce) ==   \        FAILURE) {                                                             \      zend_error(E_ERROR, "Generated message class %s hasn't been defined",    \                 klass_name);                                                  \      return;                                                                  \    } else {                                                                   \      desc->klass = PHP_PROTO_CE_UNREF(pce);                                   \    }                                                                          \    add_ce_obj(desc->klass, desc_php);                                         \    efree(klass_name);                                                         \    break;                                                                     \  }      CASE_TYPE(MSG, msgdef, Descriptor, descriptor)      CASE_TYPE(ENUM, enumdef, EnumDescriptor, enum_descriptor)#undef CASE_TYPE      default:        break;    }  }  for (i = 0; i < upb_filedef_defcount(files[0]); i++) {    const upb_def *def = upb_filedef_def(files[0], i);    if (upb_def_type(def) == UPB_DEF_MSG) {      const upb_msgdef *msgdef = upb_downcast_msgdef(def);      PHP_PROTO_HASHTABLE_VALUE desc_php = get_def_obj(msgdef);      build_class_from_descriptor(desc_php TSRMLS_CC);    }  }  upb_filedef_unref(files[0], &pool);  upb_gfree(files);}
 |