• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Protocol Buffers - Google's data interchange format
2 // Copyright 2023 Google LLC.  All rights reserved.
3 //
4 // Use of this source code is governed by a BSD-style
5 // license that can be found in the LICENSE file or at
6 // https://developers.google.com/open-source/licenses/bsd
7 
8 #include "upb/reflection/internal/def_pool.h"
9 
10 #include "upb/base/status.h"
11 #include "upb/hash/int_table.h"
12 #include "upb/hash/str_table.h"
13 #include "upb/mem/alloc.h"
14 #include "upb/mem/arena.h"
15 #include "upb/reflection/def_type.h"
16 #include "upb/reflection/file_def.h"
17 #include "upb/reflection/internal/def_builder.h"
18 #include "upb/reflection/internal/enum_def.h"
19 #include "upb/reflection/internal/enum_value_def.h"
20 #include "upb/reflection/internal/field_def.h"
21 #include "upb/reflection/internal/file_def.h"
22 #include "upb/reflection/internal/message_def.h"
23 #include "upb/reflection/internal/service_def.h"
24 #include "upb/reflection/internal/upb_edition_defaults.h"
25 
26 // Must be last.
27 #include "upb/port/def.inc"
28 
29 struct upb_DefPool {
30   upb_Arena* arena;
31   upb_strtable syms;   // full_name -> packed def ptr
32   upb_strtable files;  // file_name -> (upb_FileDef*)
33   upb_inttable exts;   // (upb_MiniTableExtension*) -> (upb_FieldDef*)
34   upb_ExtensionRegistry* extreg;
35   const UPB_DESC(FeatureSetDefaults) * feature_set_defaults;
36   upb_MiniTablePlatform platform;
37   void* scratch_data;
38   size_t scratch_size;
39   size_t bytes_loaded;
40 };
41 
upb_DefPool_Free(upb_DefPool * s)42 void upb_DefPool_Free(upb_DefPool* s) {
43   upb_Arena_Free(s->arena);
44   upb_gfree(s->scratch_data);
45   upb_gfree(s);
46 }
47 
48 static const char serialized_defaults[] = UPB_INTERNAL_UPB_EDITION_DEFAULTS;
49 
upb_DefPool_New(void)50 upb_DefPool* upb_DefPool_New(void) {
51   upb_DefPool* s = upb_gmalloc(sizeof(*s));
52   if (!s) return NULL;
53 
54   s->arena = upb_Arena_New();
55   s->bytes_loaded = 0;
56 
57   s->scratch_size = 240;
58   s->scratch_data = upb_gmalloc(s->scratch_size);
59   if (!s->scratch_data) goto err;
60 
61   if (!upb_strtable_init(&s->syms, 32, s->arena)) goto err;
62   if (!upb_strtable_init(&s->files, 4, s->arena)) goto err;
63   if (!upb_inttable_init(&s->exts, s->arena)) goto err;
64 
65   s->extreg = upb_ExtensionRegistry_New(s->arena);
66   if (!s->extreg) goto err;
67 
68   s->platform = kUpb_MiniTablePlatform_Native;
69 
70   upb_Status status;
71   if (!upb_DefPool_SetFeatureSetDefaults(
72           s, serialized_defaults, sizeof(serialized_defaults) - 1, &status)) {
73     goto err;
74   }
75 
76   if (!s->feature_set_defaults) goto err;
77 
78   return s;
79 
80 err:
81   upb_DefPool_Free(s);
82   return NULL;
83 }
84 
UPB_DESC(FeatureSetDefaults)85 const UPB_DESC(FeatureSetDefaults) *
86     upb_DefPool_FeatureSetDefaults(const upb_DefPool* s) {
87   return s->feature_set_defaults;
88 }
89 
upb_DefPool_SetFeatureSetDefaults(upb_DefPool * s,const char * serialized_defaults,size_t serialized_len,upb_Status * status)90 bool upb_DefPool_SetFeatureSetDefaults(upb_DefPool* s,
91                                        const char* serialized_defaults,
92                                        size_t serialized_len,
93                                        upb_Status* status) {
94   const UPB_DESC(FeatureSetDefaults)* defaults = UPB_DESC(
95       FeatureSetDefaults_parse)(serialized_defaults, serialized_len, s->arena);
96   if (!defaults) {
97     upb_Status_SetErrorFormat(status, "Failed to parse defaults");
98     return false;
99   }
100   if (upb_strtable_count(&s->files) > 0) {
101     upb_Status_SetErrorFormat(status,
102                               "Feature set defaults can't be changed once the "
103                               "pool has started building");
104     return false;
105   }
106   int min_edition = UPB_DESC(FeatureSetDefaults_minimum_edition(defaults));
107   int max_edition = UPB_DESC(FeatureSetDefaults_maximum_edition(defaults));
108   if (min_edition > max_edition) {
109     upb_Status_SetErrorFormat(status, "Invalid edition range %s to %s",
110                               upb_FileDef_EditionName(min_edition),
111                               upb_FileDef_EditionName(max_edition));
112     return false;
113   }
114   size_t size;
115   const UPB_DESC(
116       FeatureSetDefaults_FeatureSetEditionDefault)* const* default_list =
117       UPB_DESC(FeatureSetDefaults_defaults(defaults, &size));
118   int prev_edition = UPB_DESC(EDITION_UNKNOWN);
119   for (size_t i = 0; i < size; ++i) {
120     int edition = UPB_DESC(
121         FeatureSetDefaults_FeatureSetEditionDefault_edition(default_list[i]));
122     if (edition == UPB_DESC(EDITION_UNKNOWN)) {
123       upb_Status_SetErrorFormat(status, "Invalid edition UNKNOWN specified");
124       return false;
125     }
126     if (edition <= prev_edition) {
127       upb_Status_SetErrorFormat(status,
128                                 "Feature set defaults are not strictly "
129                                 "increasing, %s is greater than or equal to %s",
130                                 upb_FileDef_EditionName(prev_edition),
131                                 upb_FileDef_EditionName(edition));
132       return false;
133     }
134     prev_edition = edition;
135   }
136 
137   // Copy the defaults into the pool.
138   s->feature_set_defaults = defaults;
139   return true;
140 }
141 
_upb_DefPool_InsertExt(upb_DefPool * s,const upb_MiniTableExtension * ext,const upb_FieldDef * f)142 bool _upb_DefPool_InsertExt(upb_DefPool* s, const upb_MiniTableExtension* ext,
143                             const upb_FieldDef* f) {
144   return upb_inttable_insert(&s->exts, (uintptr_t)ext, upb_value_constptr(f),
145                              s->arena);
146 }
147 
_upb_DefPool_InsertSym(upb_DefPool * s,upb_StringView sym,upb_value v,upb_Status * status)148 bool _upb_DefPool_InsertSym(upb_DefPool* s, upb_StringView sym, upb_value v,
149                             upb_Status* status) {
150   // TODO: table should support an operation "tryinsert" to avoid the double
151   // lookup.
152   if (upb_strtable_lookup2(&s->syms, sym.data, sym.size, NULL)) {
153     upb_Status_SetErrorFormat(status, "duplicate symbol '%s'", sym.data);
154     return false;
155   }
156   if (!upb_strtable_insert(&s->syms, sym.data, sym.size, v, s->arena)) {
157     upb_Status_SetErrorMessage(status, "out of memory");
158     return false;
159   }
160   return true;
161 }
162 
_upb_DefPool_Unpack(const upb_DefPool * s,const char * sym,size_t size,upb_deftype_t type)163 static const void* _upb_DefPool_Unpack(const upb_DefPool* s, const char* sym,
164                                        size_t size, upb_deftype_t type) {
165   upb_value v;
166   return upb_strtable_lookup2(&s->syms, sym, size, &v)
167              ? _upb_DefType_Unpack(v, type)
168              : NULL;
169 }
170 
_upb_DefPool_LookupSym(const upb_DefPool * s,const char * sym,size_t size,upb_value * v)171 bool _upb_DefPool_LookupSym(const upb_DefPool* s, const char* sym, size_t size,
172                             upb_value* v) {
173   return upb_strtable_lookup2(&s->syms, sym, size, v);
174 }
175 
_upb_DefPool_ExtReg(const upb_DefPool * s)176 upb_ExtensionRegistry* _upb_DefPool_ExtReg(const upb_DefPool* s) {
177   return s->extreg;
178 }
179 
_upb_DefPool_ScratchData(const upb_DefPool * s)180 void** _upb_DefPool_ScratchData(const upb_DefPool* s) {
181   return (void**)&s->scratch_data;
182 }
183 
_upb_DefPool_ScratchSize(const upb_DefPool * s)184 size_t* _upb_DefPool_ScratchSize(const upb_DefPool* s) {
185   return (size_t*)&s->scratch_size;
186 }
187 
_upb_DefPool_SetPlatform(upb_DefPool * s,upb_MiniTablePlatform platform)188 void _upb_DefPool_SetPlatform(upb_DefPool* s, upb_MiniTablePlatform platform) {
189   assert(upb_strtable_count(&s->files) == 0);
190   s->platform = platform;
191 }
192 
upb_DefPool_FindMessageByName(const upb_DefPool * s,const char * sym)193 const upb_MessageDef* upb_DefPool_FindMessageByName(const upb_DefPool* s,
194                                                     const char* sym) {
195   return _upb_DefPool_Unpack(s, sym, strlen(sym), UPB_DEFTYPE_MSG);
196 }
197 
upb_DefPool_FindMessageByNameWithSize(const upb_DefPool * s,const char * sym,size_t len)198 const upb_MessageDef* upb_DefPool_FindMessageByNameWithSize(
199     const upb_DefPool* s, const char* sym, size_t len) {
200   return _upb_DefPool_Unpack(s, sym, len, UPB_DEFTYPE_MSG);
201 }
202 
upb_DefPool_FindEnumByName(const upb_DefPool * s,const char * sym)203 const upb_EnumDef* upb_DefPool_FindEnumByName(const upb_DefPool* s,
204                                               const char* sym) {
205   return _upb_DefPool_Unpack(s, sym, strlen(sym), UPB_DEFTYPE_ENUM);
206 }
207 
upb_DefPool_FindEnumByNameval(const upb_DefPool * s,const char * sym)208 const upb_EnumValueDef* upb_DefPool_FindEnumByNameval(const upb_DefPool* s,
209                                                       const char* sym) {
210   return _upb_DefPool_Unpack(s, sym, strlen(sym), UPB_DEFTYPE_ENUMVAL);
211 }
212 
upb_DefPool_FindFileByName(const upb_DefPool * s,const char * name)213 const upb_FileDef* upb_DefPool_FindFileByName(const upb_DefPool* s,
214                                               const char* name) {
215   upb_value v;
216   return upb_strtable_lookup(&s->files, name, &v) ? upb_value_getconstptr(v)
217                                                   : NULL;
218 }
219 
upb_DefPool_FindFileByNameWithSize(const upb_DefPool * s,const char * name,size_t len)220 const upb_FileDef* upb_DefPool_FindFileByNameWithSize(const upb_DefPool* s,
221                                                       const char* name,
222                                                       size_t len) {
223   upb_value v;
224   return upb_strtable_lookup2(&s->files, name, len, &v)
225              ? upb_value_getconstptr(v)
226              : NULL;
227 }
228 
upb_DefPool_FindExtensionByNameWithSize(const upb_DefPool * s,const char * name,size_t size)229 const upb_FieldDef* upb_DefPool_FindExtensionByNameWithSize(
230     const upb_DefPool* s, const char* name, size_t size) {
231   upb_value v;
232   if (!upb_strtable_lookup2(&s->syms, name, size, &v)) return NULL;
233 
234   switch (_upb_DefType_Type(v)) {
235     case UPB_DEFTYPE_FIELD:
236       return _upb_DefType_Unpack(v, UPB_DEFTYPE_FIELD);
237     case UPB_DEFTYPE_MSG: {
238       const upb_MessageDef* m = _upb_DefType_Unpack(v, UPB_DEFTYPE_MSG);
239       return _upb_MessageDef_InMessageSet(m)
240                  ? upb_MessageDef_NestedExtension(m, 0)
241                  : NULL;
242     }
243     default:
244       break;
245   }
246 
247   return NULL;
248 }
249 
upb_DefPool_FindExtensionByName(const upb_DefPool * s,const char * sym)250 const upb_FieldDef* upb_DefPool_FindExtensionByName(const upb_DefPool* s,
251                                                     const char* sym) {
252   return upb_DefPool_FindExtensionByNameWithSize(s, sym, strlen(sym));
253 }
254 
upb_DefPool_FindServiceByName(const upb_DefPool * s,const char * name)255 const upb_ServiceDef* upb_DefPool_FindServiceByName(const upb_DefPool* s,
256                                                     const char* name) {
257   return _upb_DefPool_Unpack(s, name, strlen(name), UPB_DEFTYPE_SERVICE);
258 }
259 
upb_DefPool_FindServiceByNameWithSize(const upb_DefPool * s,const char * name,size_t size)260 const upb_ServiceDef* upb_DefPool_FindServiceByNameWithSize(
261     const upb_DefPool* s, const char* name, size_t size) {
262   return _upb_DefPool_Unpack(s, name, size, UPB_DEFTYPE_SERVICE);
263 }
264 
upb_DefPool_FindFileContainingSymbol(const upb_DefPool * s,const char * name)265 const upb_FileDef* upb_DefPool_FindFileContainingSymbol(const upb_DefPool* s,
266                                                         const char* name) {
267   upb_value v;
268   // TODO: non-extension fields and oneofs.
269   if (upb_strtable_lookup(&s->syms, name, &v)) {
270     switch (_upb_DefType_Type(v)) {
271       case UPB_DEFTYPE_EXT: {
272         const upb_FieldDef* f = _upb_DefType_Unpack(v, UPB_DEFTYPE_EXT);
273         return upb_FieldDef_File(f);
274       }
275       case UPB_DEFTYPE_MSG: {
276         const upb_MessageDef* m = _upb_DefType_Unpack(v, UPB_DEFTYPE_MSG);
277         return upb_MessageDef_File(m);
278       }
279       case UPB_DEFTYPE_ENUM: {
280         const upb_EnumDef* e = _upb_DefType_Unpack(v, UPB_DEFTYPE_ENUM);
281         return upb_EnumDef_File(e);
282       }
283       case UPB_DEFTYPE_ENUMVAL: {
284         const upb_EnumValueDef* ev =
285             _upb_DefType_Unpack(v, UPB_DEFTYPE_ENUMVAL);
286         return upb_EnumDef_File(upb_EnumValueDef_Enum(ev));
287       }
288       case UPB_DEFTYPE_SERVICE: {
289         const upb_ServiceDef* service =
290             _upb_DefType_Unpack(v, UPB_DEFTYPE_SERVICE);
291         return upb_ServiceDef_File(service);
292       }
293       default:
294         UPB_UNREACHABLE();
295     }
296   }
297 
298   const char* last_dot = strrchr(name, '.');
299   if (last_dot) {
300     const upb_MessageDef* parent =
301         upb_DefPool_FindMessageByNameWithSize(s, name, last_dot - name);
302     if (parent) {
303       const char* shortname = last_dot + 1;
304       if (upb_MessageDef_FindByNameWithSize(parent, shortname,
305                                             strlen(shortname), NULL, NULL)) {
306         return upb_MessageDef_File(parent);
307       }
308     }
309   }
310 
311   return NULL;
312 }
313 
remove_filedef(upb_DefPool * s,upb_FileDef * file)314 static void remove_filedef(upb_DefPool* s, upb_FileDef* file) {
315   intptr_t iter = UPB_INTTABLE_BEGIN;
316   upb_StringView key;
317   upb_value val;
318   while (upb_strtable_next2(&s->syms, &key, &val, &iter)) {
319     const upb_FileDef* f;
320     switch (_upb_DefType_Type(val)) {
321       case UPB_DEFTYPE_EXT:
322         f = upb_FieldDef_File(_upb_DefType_Unpack(val, UPB_DEFTYPE_EXT));
323         break;
324       case UPB_DEFTYPE_MSG:
325         f = upb_MessageDef_File(_upb_DefType_Unpack(val, UPB_DEFTYPE_MSG));
326         break;
327       case UPB_DEFTYPE_ENUM:
328         f = upb_EnumDef_File(_upb_DefType_Unpack(val, UPB_DEFTYPE_ENUM));
329         break;
330       case UPB_DEFTYPE_ENUMVAL:
331         f = upb_EnumDef_File(upb_EnumValueDef_Enum(
332             _upb_DefType_Unpack(val, UPB_DEFTYPE_ENUMVAL)));
333         break;
334       case UPB_DEFTYPE_SERVICE:
335         f = upb_ServiceDef_File(_upb_DefType_Unpack(val, UPB_DEFTYPE_SERVICE));
336         break;
337       default:
338         UPB_UNREACHABLE();
339     }
340 
341     if (f == file) upb_strtable_removeiter(&s->syms, &iter);
342   }
343 }
344 
upb_DefBuilder_AddFileToPool(upb_DefBuilder * const builder,upb_DefPool * const s,const UPB_DESC (FileDescriptorProto)* const file_proto,const upb_StringView name,upb_Status * const status)345 static const upb_FileDef* upb_DefBuilder_AddFileToPool(
346     upb_DefBuilder* const builder, upb_DefPool* const s,
347     const UPB_DESC(FileDescriptorProto) * const file_proto,
348     const upb_StringView name, upb_Status* const status) {
349   if (UPB_SETJMP(builder->err) != 0) {
350     UPB_ASSERT(!upb_Status_IsOk(status));
351     if (builder->file) {
352       remove_filedef(s, builder->file);
353       builder->file = NULL;
354     }
355   } else if (!builder->arena || !builder->tmp_arena ||
356              !upb_strtable_init(&builder->feature_cache, 16,
357                                 builder->tmp_arena) ||
358              !(builder->legacy_features =
359                    UPB_DESC(FeatureSet_new)(builder->tmp_arena))) {
360     _upb_DefBuilder_OomErr(builder);
361   } else {
362     _upb_FileDef_Create(builder, file_proto);
363     upb_strtable_insert(&s->files, name.data, name.size,
364                         upb_value_constptr(builder->file), builder->arena);
365     UPB_ASSERT(upb_Status_IsOk(status));
366     upb_Arena_Fuse(s->arena, builder->arena);
367   }
368 
369   if (builder->arena) upb_Arena_Free(builder->arena);
370   if (builder->tmp_arena) upb_Arena_Free(builder->tmp_arena);
371   return builder->file;
372 }
373 
_upb_DefPool_AddFile(upb_DefPool * s,const UPB_DESC (FileDescriptorProto)* file_proto,const upb_MiniTableFile * layout,upb_Status * status)374 static const upb_FileDef* _upb_DefPool_AddFile(
375     upb_DefPool* s, const UPB_DESC(FileDescriptorProto) * file_proto,
376     const upb_MiniTableFile* layout, upb_Status* status) {
377   const upb_StringView name = UPB_DESC(FileDescriptorProto_name)(file_proto);
378 
379   // Determine whether we already know about this file.
380   {
381     upb_value v;
382     if (upb_strtable_lookup2(&s->files, name.data, name.size, &v)) {
383       upb_Status_SetErrorFormat(status,
384                                 "duplicate file name " UPB_STRINGVIEW_FORMAT,
385                                 UPB_STRINGVIEW_ARGS(name));
386       return NULL;
387     }
388   }
389 
390   upb_DefBuilder ctx = {
391       .symtab = s,
392       .tmp_buf = NULL,
393       .tmp_buf_size = 0,
394       .layout = layout,
395       .platform = s->platform,
396       .msg_count = 0,
397       .enum_count = 0,
398       .ext_count = 0,
399       .status = status,
400       .file = NULL,
401       .arena = upb_Arena_New(),
402       .tmp_arena = upb_Arena_New(),
403   };
404 
405   return upb_DefBuilder_AddFileToPool(&ctx, s, file_proto, name, status);
406 }
407 
upb_DefPool_AddFile(upb_DefPool * s,const UPB_DESC (FileDescriptorProto)* file_proto,upb_Status * status)408 const upb_FileDef* upb_DefPool_AddFile(upb_DefPool* s,
409                                        const UPB_DESC(FileDescriptorProto) *
410                                            file_proto,
411                                        upb_Status* status) {
412   return _upb_DefPool_AddFile(s, file_proto, NULL, status);
413 }
414 
_upb_DefPool_LoadDefInitEx(upb_DefPool * s,const _upb_DefPool_Init * init,bool rebuild_minitable)415 bool _upb_DefPool_LoadDefInitEx(upb_DefPool* s, const _upb_DefPool_Init* init,
416                                 bool rebuild_minitable) {
417   /* Since this function should never fail (it would indicate a bug in upb) we
418    * print errors to stderr instead of returning error status to the user. */
419   _upb_DefPool_Init** deps = init->deps;
420   UPB_DESC(FileDescriptorProto) * file;
421   upb_Arena* arena;
422   upb_Status status;
423 
424   upb_Status_Clear(&status);
425 
426   if (upb_DefPool_FindFileByName(s, init->filename)) {
427     return true;
428   }
429 
430   arena = upb_Arena_New();
431 
432   for (; *deps; deps++) {
433     if (!_upb_DefPool_LoadDefInitEx(s, *deps, rebuild_minitable)) goto err;
434   }
435 
436   file = UPB_DESC(FileDescriptorProto_parse_ex)(
437       init->descriptor.data, init->descriptor.size, NULL,
438       kUpb_DecodeOption_AliasString, arena);
439   s->bytes_loaded += init->descriptor.size;
440 
441   if (!file) {
442     upb_Status_SetErrorFormat(
443         &status,
444         "Failed to parse compiled-in descriptor for file '%s'. This should "
445         "never happen.",
446         init->filename);
447     goto err;
448   }
449 
450   const upb_MiniTableFile* mt = rebuild_minitable ? NULL : init->layout;
451   if (!_upb_DefPool_AddFile(s, file, mt, &status)) {
452     goto err;
453   }
454 
455   upb_Arena_Free(arena);
456   return true;
457 
458 err:
459   fprintf(stderr,
460           "Error loading compiled-in descriptor for file '%s' (this should "
461           "never happen): %s\n",
462           init->filename, upb_Status_ErrorMessage(&status));
463   upb_Arena_Free(arena);
464   return false;
465 }
466 
_upb_DefPool_BytesLoaded(const upb_DefPool * s)467 size_t _upb_DefPool_BytesLoaded(const upb_DefPool* s) {
468   return s->bytes_loaded;
469 }
470 
_upb_DefPool_Arena(const upb_DefPool * s)471 upb_Arena* _upb_DefPool_Arena(const upb_DefPool* s) { return s->arena; }
472 
upb_DefPool_FindExtensionByMiniTable(const upb_DefPool * s,const upb_MiniTableExtension * ext)473 const upb_FieldDef* upb_DefPool_FindExtensionByMiniTable(
474     const upb_DefPool* s, const upb_MiniTableExtension* ext) {
475   upb_value v;
476   bool ok = upb_inttable_lookup(&s->exts, (uintptr_t)ext, &v);
477   UPB_ASSERT(ok);
478   return upb_value_getconstptr(v);
479 }
480 
upb_DefPool_FindExtensionByNumber(const upb_DefPool * s,const upb_MessageDef * m,int32_t fieldnum)481 const upb_FieldDef* upb_DefPool_FindExtensionByNumber(const upb_DefPool* s,
482                                                       const upb_MessageDef* m,
483                                                       int32_t fieldnum) {
484   const upb_MiniTable* t = upb_MessageDef_MiniTable(m);
485   const upb_MiniTableExtension* ext =
486       upb_ExtensionRegistry_Lookup(s->extreg, t, fieldnum);
487   return ext ? upb_DefPool_FindExtensionByMiniTable(s, ext) : NULL;
488 }
489 
upb_DefPool_ExtensionRegistry(const upb_DefPool * s)490 const upb_ExtensionRegistry* upb_DefPool_ExtensionRegistry(
491     const upb_DefPool* s) {
492   return s->extreg;
493 }
494 
upb_DefPool_GetAllExtensions(const upb_DefPool * s,const upb_MessageDef * m,size_t * count)495 const upb_FieldDef** upb_DefPool_GetAllExtensions(const upb_DefPool* s,
496                                                   const upb_MessageDef* m,
497                                                   size_t* count) {
498   size_t n = 0;
499   intptr_t iter = UPB_INTTABLE_BEGIN;
500   uintptr_t key;
501   upb_value val;
502   // This is O(all exts) instead of O(exts for m).  If we need this to be
503   // efficient we may need to make extreg into a two-level table, or have a
504   // second per-message index.
505   while (upb_inttable_next(&s->exts, &key, &val, &iter)) {
506     const upb_FieldDef* f = upb_value_getconstptr(val);
507     if (upb_FieldDef_ContainingType(f) == m) n++;
508   }
509   const upb_FieldDef** exts = upb_gmalloc(n * sizeof(*exts));
510   iter = UPB_INTTABLE_BEGIN;
511   size_t i = 0;
512   while (upb_inttable_next(&s->exts, &key, &val, &iter)) {
513     const upb_FieldDef* f = upb_value_getconstptr(val);
514     if (upb_FieldDef_ContainingType(f) == m) exts[i++] = f;
515   }
516   *count = n;
517   return exts;
518 }
519 
_upb_DefPool_LoadDefInit(upb_DefPool * s,const _upb_DefPool_Init * init)520 bool _upb_DefPool_LoadDefInit(upb_DefPool* s, const _upb_DefPool_Init* init) {
521   return _upb_DefPool_LoadDefInitEx(s, init, false);
522 }
523