1 /*
2  * Copyright 2019 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "fields/array_field.h"
18 
19 #include "fields/custom_field.h"
20 #include "fields/scalar_field.h"
21 #include "util.h"
22 
23 const std::string ArrayField::kFieldType = "ArrayField";
24 
ArrayField(std::string name,int element_size,int array_size,ParseLocation loc)25 ArrayField::ArrayField(std::string name, int element_size, int array_size, ParseLocation loc)
26     : PacketField(name, loc), element_field_(new ScalarField("val", element_size, loc)), element_size_(element_size),
27       array_size_(array_size) {
28   if (element_size > 64 || element_size < 0)
29     ERROR(this) << __func__ << ": Not implemented for element size = " << element_size;
30   if (element_size % 8 != 0) {
31     ERROR(this) << "Can only have arrays with elements that are byte aligned (" << element_size << ")";
32   }
33 }
34 
ArrayField(std::string name,TypeDef * type_def,int array_size,ParseLocation loc)35 ArrayField::ArrayField(std::string name, TypeDef* type_def, int array_size, ParseLocation loc)
36     : PacketField(name, loc), element_field_(type_def->GetNewField("val", loc)),
37       element_size_(element_field_->GetSize()), array_size_(array_size) {
38   if (!element_size_.empty() && element_size_.bits() % 8 != 0) {
39     ERROR(this) << "Can only have arrays with elements that are byte aligned (" << element_size_ << ")";
40   }
41 }
42 
GetFieldType() const43 const std::string& ArrayField::GetFieldType() const {
44   return ArrayField::kFieldType;
45 }
46 
GetSize() const47 Size ArrayField::GetSize() const {
48   if (!element_size_.empty() && !element_size_.has_dynamic()) {
49     return Size(array_size_ * element_size_.bits());
50   }
51   return Size();
52 }
53 
GetBuilderSize() const54 Size ArrayField::GetBuilderSize() const {
55   if (!element_size_.empty() && !element_size_.has_dynamic()) {
56     return GetSize();
57   } else if (element_field_->BuilderParameterMustBeMoved()) {
58     std::string ret = "[this](){ size_t length = 0; for (const auto& elem : " + GetName() +
59                       "_) { length += elem->size() * 8; } return length; }()";
60     return ret;
61   } else {
62     std::string ret = "[this](){ size_t length = 0; for (const auto& elem : " + GetName() +
63                       "_) { length += elem.size() * 8; } return length; }()";
64     return ret;
65   }
66 }
67 
GetStructSize() const68 Size ArrayField::GetStructSize() const {
69   if (!element_size_.empty() && !element_size_.has_dynamic()) {
70     return GetSize();
71   } else if (element_field_->BuilderParameterMustBeMoved()) {
72     std::string ret = "[this](){ size_t length = 0; for (const auto& elem : to_fill->" + GetName() +
73                       "_) { length += elem->size() * 8; } return length; }()";
74     return ret;
75   } else {
76     std::string ret = "[this](){ size_t length = 0; for (const auto& elem : to_fill->" + GetName() +
77                       "_) { length += elem.size() * 8; } return length; }()";
78     return ret;
79   }
80 }
81 
GetDataType() const82 std::string ArrayField::GetDataType() const {
83   return "std::array<" + element_field_->GetDataType() + "," + std::to_string(array_size_) + ">";
84 }
85 
GenExtractor(std::ostream & s,int num_leading_bits,bool for_struct) const86 void ArrayField::GenExtractor(std::ostream& s, int num_leading_bits, bool for_struct) const {
87   s << GetDataType() << "::iterator ret_it = " << GetName() << "_ptr->begin();";
88   s << "auto " << element_field_->GetName() << "_it = " << GetName() << "_it;";
89   if (!element_size_.empty()) {
90     s << "while (" << element_field_->GetName() << "_it.NumBytesRemaining() >= " << element_size_.bytes();
91     s << " && ret_it < " << GetName() << "_ptr->end()) {";
92   } else {
93     s << "while (" << element_field_->GetName() << "_it.NumBytesRemaining() > 0 ";
94     s << " && ret_it < " << GetName() << "_ptr->end()) {";
95   }
96   if (element_field_->BuilderParameterMustBeMoved()) {
97     s << element_field_->GetDataType() << " " << element_field_->GetName() << "_ptr;";
98   } else {
99     s << "auto " << element_field_->GetName() << "_ptr = ret_it;";
100   }
101   element_field_->GenExtractor(s, num_leading_bits, for_struct);
102   if (element_field_->BuilderParameterMustBeMoved()) {
103     s << "*ret_it = std::move(" << element_field_->GetName() << "_ptr);";
104   }
105   s << "ret_it++;";
106   s << "}";
107 }
108 
GetGetterFunctionName() const109 std::string ArrayField::GetGetterFunctionName() const {
110   std::stringstream ss;
111   ss << "Get" << util::UnderscoreToCamelCase(GetName());
112   return ss.str();
113 }
114 
GenGetter(std::ostream & s,Size start_offset,Size end_offset) const115 void ArrayField::GenGetter(std::ostream& s, Size start_offset, Size end_offset) const {
116   s << GetDataType() << " " << GetGetterFunctionName() << "() {";
117   s << "ASSERT(was_validated_);";
118   s << "size_t end_index = size();";
119   s << "auto to_bound = begin();";
120 
121   int num_leading_bits = GenBounds(s, start_offset, end_offset, GetSize());
122   s << GetDataType() << " " << GetName() << "_value{};";
123   s << GetDataType() << "* " << GetName() << "_ptr = &" << GetName() << "_value;";
124   GenExtractor(s, num_leading_bits, false);
125 
126   s << "return " << GetName() << "_value;";
127   s << "}\n";
128 }
129 
GetBuilderParameterType() const130 std::string ArrayField::GetBuilderParameterType() const {
131   std::stringstream ss;
132   if (element_field_->BuilderParameterMustBeMoved()) {
133     ss << "std::array<" << element_field_->GetDataType() << "," << array_size_ << ">";
134   } else {
135     ss << "const std::array<" << element_field_->GetDataType() << "," << array_size_ << ">&";
136   }
137   return ss.str();
138 }
139 
BuilderParameterMustBeMoved() const140 bool ArrayField::BuilderParameterMustBeMoved() const {
141   return element_field_->BuilderParameterMustBeMoved();
142 }
143 
GenBuilderMember(std::ostream & s) const144 bool ArrayField::GenBuilderMember(std::ostream& s) const {
145   s << "std::array<" << element_field_->GetDataType() << "," << array_size_ << "> " << GetName();
146   return true;
147 }
148 
HasParameterValidator() const149 bool ArrayField::HasParameterValidator() const {
150   return false;
151 }
152 
GenParameterValidator(std::ostream &) const153 void ArrayField::GenParameterValidator(std::ostream&) const {
154   // Array length is validated by the compiler
155 }
156 
GenInserter(std::ostream & s) const157 void ArrayField::GenInserter(std::ostream& s) const {
158   s << "for (const auto& val_ : " << GetName() << "_) {";
159   element_field_->GenInserter(s);
160   s << "}\n";
161 }
162 
GenValidator(std::ostream &) const163 void ArrayField::GenValidator(std::ostream&) const {
164   // NOTE: We could check if the element size divides cleanly into the array size, but we decided to forgo that
165   // in favor of just returning as many elements as possible in a best effort style.
166   //
167   // Other than that there is nothing that arrays need to be validated on other than length so nothing needs to
168   // be done here.
169 }
170 
IsContainerField() const171 bool ArrayField::IsContainerField() const {
172   return true;
173 }
174 
GetElementField() const175 const PacketField* ArrayField::GetElementField() const {
176   return element_field_;
177 }
178 
GenStringRepresentation(std::ostream & s,std::string accessor) const179 void ArrayField::GenStringRepresentation(std::ostream& s, std::string accessor) const {
180   s << "\"ARRAY[\";";
181   s << "/* " << element_field_->GetDataType() << "   " << element_field_->GetFieldType() << " */";
182 
183   std::string arr_idx = "arridx_" + accessor;
184   std::string arr_size = std::to_string(array_size_);
185   s << "for (size_t index = 0; index < " << arr_size << "; index++) {";
186   std::string element_accessor = "(" + accessor + "[index])";
187   s << "ss << ((index == 0) ? \"\" : \", \") << ";
188 
189   if (element_field_->GetFieldType() == CustomField::kFieldType) {
190     s << element_accessor << ".ToString()";
191   } else {
192     element_field_->GenStringRepresentation(s, element_accessor);
193   }
194 
195   s << ";}";
196   s << "ss << \"]\"";
197 }
198 
GetRustDataType() const199 std::string ArrayField::GetRustDataType() const {
200   return "[" + element_field_->GetRustDataType() + "; " + std::to_string(array_size_) + "]";
201 }
202 
GenRustGetter(std::ostream & s,Size start_offset,Size) const203 void ArrayField::GenRustGetter(std::ostream& s, Size start_offset, Size) const {
204   s << "let " << GetName() << " = ";
205   s << "bytes[" << start_offset.bytes() << "..";
206   s << start_offset.bytes() + GetSize().bytes() << "].try_into().unwrap();";
207 }
208 
GenRustWriter(std::ostream & s,Size start_offset,Size) const209 void ArrayField::GenRustWriter(std::ostream& s, Size start_offset, Size) const {
210   s << "&buffer[" << start_offset.bytes() << ".." << start_offset.bytes() + GetSize().bytes()
211     << "].copy_from_slice(&self." << GetName() << ");";
212 }
213