1 // Copyright (c) 2009-2021, Google LLC
2 // All rights reserved.
3 //
4 // Redistribution and use in source and binary forms, with or without
5 // modification, are permitted provided that the following conditions are met:
6 // * Redistributions of source code must retain the above copyright
7 // notice, this list of conditions and the following disclaimer.
8 // * Redistributions in binary form must reproduce the above copyright
9 // notice, this list of conditions and the following disclaimer in the
10 // documentation and/or other materials provided with the distribution.
11 // * Neither the name of Google LLC nor the
12 // names of its contributors may be used to endorse or promote products
13 // derived from this software without specific prior written permission.
14 //
15 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
16 // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17 // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18 // ARE DISCLAIMED. IN NO EVENT SHALL Google LLC BE LIABLE FOR ANY DIRECT,
19 // INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
20 // (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
21 // LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
22 // ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
24 // SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25
26 #include "upbc/file_layout.h"
27
28 #include <string>
29 #include <unordered_set>
30
31 #include "upb/mini_table/extension_internal.h"
32 #include "upbc/common.h"
33
34 namespace upbc {
35
36 const char* kEnumsInit = "enums_layout";
37 const char* kExtensionsInit = "extensions_layout";
38 const char* kMessagesInit = "messages_layout";
39
AddEnums(upb::MessageDefPtr message,std::vector<upb::EnumDefPtr> * enums)40 void AddEnums(upb::MessageDefPtr message, std::vector<upb::EnumDefPtr>* enums) {
41 enums->reserve(enums->size() + message.enum_type_count());
42 for (int i = 0; i < message.enum_type_count(); i++) {
43 enums->push_back(message.enum_type(i));
44 }
45 for (int i = 0; i < message.nested_message_count(); i++) {
46 AddEnums(message.nested_message(i), enums);
47 }
48 }
49
SortedEnums(upb::FileDefPtr file)50 std::vector<upb::EnumDefPtr> SortedEnums(upb::FileDefPtr file) {
51 std::vector<upb::EnumDefPtr> enums;
52 enums.reserve(file.toplevel_enum_count());
53 for (int i = 0; i < file.toplevel_enum_count(); i++) {
54 enums.push_back(file.toplevel_enum(i));
55 }
56 for (int i = 0; i < file.toplevel_message_count(); i++) {
57 AddEnums(file.toplevel_message(i), &enums);
58 }
59 std::sort(enums.begin(), enums.end(),
60 [](upb::EnumDefPtr a, upb::EnumDefPtr b) {
61 return strcmp(a.full_name(), b.full_name()) < 0;
62 });
63 return enums;
64 }
65
SortedUniqueEnumNumbers(upb::EnumDefPtr e)66 std::vector<uint32_t> SortedUniqueEnumNumbers(upb::EnumDefPtr e) {
67 std::vector<uint32_t> values;
68 values.reserve(e.value_count());
69 for (int i = 0; i < e.value_count(); i++) {
70 values.push_back(static_cast<uint32_t>(e.value(i).number()));
71 }
72 std::sort(values.begin(), values.end());
73 auto last = std::unique(values.begin(), values.end());
74 values.erase(last, values.end());
75 return values;
76 }
77
AddMessages(upb::MessageDefPtr message,std::vector<upb::MessageDefPtr> * messages)78 void AddMessages(upb::MessageDefPtr message,
79 std::vector<upb::MessageDefPtr>* messages) {
80 messages->push_back(message);
81 for (int i = 0; i < message.nested_message_count(); i++) {
82 AddMessages(message.nested_message(i), messages);
83 }
84 }
85
86 // Ordering must match upb/def.c!
87 //
88 // The ordering is significant because each upb_MessageDef* will point at the
89 // corresponding upb_MiniTable and we just iterate through the list without
90 // any search or lookup.
SortedMessages(upb::FileDefPtr file)91 std::vector<upb::MessageDefPtr> SortedMessages(upb::FileDefPtr file) {
92 std::vector<upb::MessageDefPtr> messages;
93 for (int i = 0; i < file.toplevel_message_count(); i++) {
94 AddMessages(file.toplevel_message(i), &messages);
95 }
96 return messages;
97 }
98
AddExtensionsFromMessage(upb::MessageDefPtr message,std::vector<upb::FieldDefPtr> * exts)99 void AddExtensionsFromMessage(upb::MessageDefPtr message,
100 std::vector<upb::FieldDefPtr>* exts) {
101 for (int i = 0; i < message.nested_extension_count(); i++) {
102 exts->push_back(message.nested_extension(i));
103 }
104 for (int i = 0; i < message.nested_message_count(); i++) {
105 AddExtensionsFromMessage(message.nested_message(i), exts);
106 }
107 }
108
109 // Ordering must match upb/def.c!
110 //
111 // The ordering is significant because each upb_FieldDef* will point at the
112 // corresponding upb_MiniTableExtension and we just iterate through the list
113 // without any search or lookup.
SortedExtensions(upb::FileDefPtr file)114 std::vector<upb::FieldDefPtr> SortedExtensions(upb::FileDefPtr file) {
115 std::vector<upb::FieldDefPtr> ret;
116 ret.reserve(file.toplevel_extension_count());
117 for (int i = 0; i < file.toplevel_extension_count(); i++) {
118 ret.push_back(file.toplevel_extension(i));
119 }
120 for (int i = 0; i < file.toplevel_message_count(); i++) {
121 AddExtensionsFromMessage(file.toplevel_message(i), &ret);
122 }
123 return ret;
124 }
125
FieldNumberOrder(upb::MessageDefPtr message)126 std::vector<upb::FieldDefPtr> FieldNumberOrder(upb::MessageDefPtr message) {
127 std::vector<upb::FieldDefPtr> fields;
128 fields.reserve(message.field_count());
129 for (int i = 0; i < message.field_count(); i++) {
130 fields.push_back(message.field(i));
131 }
132 std::sort(fields.begin(), fields.end(),
133 [](upb::FieldDefPtr a, upb::FieldDefPtr b) {
134 return a.number() < b.number();
135 });
136 return fields;
137 }
138
139 } // namespace upbc
140