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