blob: 5e491cf00a47a2a53cd04283e8c3459a009cc843 [file] [log] [blame]
Andreas Huber1aec3972016-08-26 09:26:32 -07001/*
2 * Copyright (C) 2016 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
Andreas Huberc9410c72016-07-28 12:18:40 -070017#include "ArrayType.h"
18
Andreas Huber881227d2016-08-02 14:20:21 -070019#include <android-base/logging.h>
Timur Iskhakove8ee6a02017-09-06 11:42:10 -070020#include <hidl-util/Formatter.h>
21#include <iostream>
Andreas Huber881227d2016-08-02 14:20:21 -070022
Yifan Hongf24fa852016-09-23 11:03:15 -070023#include "ConstantExpression.h"
24
Andreas Huberc9410c72016-07-28 12:18:40 -070025namespace android {
26
Timur Iskhakov63f39902017-08-29 15:47:29 -070027ArrayType::ArrayType(const Reference<Type>& elementType, ConstantExpression* size, Scope* parent)
Neel Mehta3b414a82019-07-02 15:47:48 -070028 : Type(parent, elementType.localName()), mElementType(elementType) {
Timur Iskhakov505316c2017-08-05 03:38:59 +000029 CHECK(!elementType.isEmptyReference());
Neel Mehta3b414a82019-07-02 15:47:48 -070030
31 appendDimension(size);
Andreas Huberc9410c72016-07-28 12:18:40 -070032}
33
Yifan Hongbd33e382016-11-02 13:30:17 -070034void ArrayType::appendDimension(ConstantExpression *size) {
Yifan Hong5706a432016-11-02 09:44:18 -070035 mSizes.push_back(size);
Neel Mehta3b414a82019-07-02 15:47:48 -070036 mDefinedName = mDefinedName + "[" + size->expression() + "]";
Yifan Hongbd33e382016-11-02 13:30:17 -070037}
38
Andreas Huberf03332a2016-09-22 15:35:43 -070039size_t ArrayType::countDimensions() const {
40 return mSizes.size();
41}
42
Andreas Huber709b62d2016-09-19 11:21:18 -070043bool ArrayType::isArray() const {
44 return true;
45}
46
Timur Iskhakov5dc72fe2017-09-07 23:13:44 -070047bool ArrayType::deepCanCheckEquality(std::unordered_set<const Type*>* visited) const {
48 return mElementType->canCheckEquality(visited);
Yifan Hongc6752dc2016-12-20 14:00:14 -080049}
50
Timur Iskhakov24e605b2017-08-30 14:02:55 -070051const Type* ArrayType::getElementType() const {
Timur Iskhakovb3f8bcb2017-08-30 15:33:29 -070052 return mElementType.get();
Andreas Huberf03332a2016-09-22 15:35:43 -070053}
54
Steven Moreland0ecc7b82017-07-19 12:59:23 -070055std::string ArrayType::typeName() const {
56 if (dimension() == 1) {
57 return "array of " + mElementType->typeName();
58 }
59
60 return std::to_string(dimension()) + "d array of " + mElementType->typeName();
61}
62
Timur Iskhakovb58f4182017-08-29 15:19:24 -070063std::vector<const Reference<Type>*> ArrayType::getReferences() const {
64 return {&mElementType};
Timur Iskhakov33431e62017-08-21 17:31:23 -070065}
66
Timur Iskhakovb58f4182017-08-29 15:19:24 -070067std::vector<const ConstantExpression*> ArrayType::getConstantExpressions() const {
68 std::vector<const ConstantExpression*> ret;
69 ret.insert(ret.end(), mSizes.begin(), mSizes.end());
70 return ret;
Timur Iskhakovcec46c42017-08-09 00:22:02 -070071}
72
Timur Iskhakov0737c2a2017-08-31 10:57:28 -070073status_t ArrayType::resolveInheritance() {
74 // Resolve for typedefs
75 while (mElementType->isArray()) {
76 ArrayType* innerArray = static_cast<ArrayType*>(mElementType.get());
77 mSizes.insert(mSizes.end(), innerArray->mSizes.begin(), innerArray->mSizes.end());
78 mElementType = innerArray->mElementType;
79 }
80 return Type::resolveInheritance();
81}
82
Timur Iskhakovcec46c42017-08-09 00:22:02 -070083status_t ArrayType::validate() const {
Timur Iskhakov0737c2a2017-08-31 10:57:28 -070084 CHECK(!mElementType->isArray());
85
Steven Moreland7df0f392018-10-31 13:23:22 -070086 if (mElementType->isInterface()) {
Timur Iskhakovcec46c42017-08-09 00:22:02 -070087 std::cerr << "ERROR: Arrays of interface types are not supported"
88 << " at " << mElementType.location() << "\n";
89
90 return UNKNOWN_ERROR;
91 }
92 return Type::validate();
93}
94
Steven Moreland979e0992016-09-07 09:18:08 -070095std::string ArrayType::getCppType(StorageMode mode,
Steven Moreland979e0992016-09-07 09:18:08 -070096 bool specifyNamespaces) const {
Yifan Hong3b320f82016-11-01 15:15:54 -070097 const std::string base = mElementType->getCppStackType(specifyNamespaces);
Andreas Huber881227d2016-08-02 14:20:21 -070098
Steven Morelandc46e9842016-11-02 13:21:26 -070099 std::string space = specifyNamespaces ? "::android::hardware::" : "";
100 std::string arrayType = space + "hidl_array<" + base;
Andreas Huberf03332a2016-09-22 15:35:43 -0700101
102 for (size_t i = 0; i < mSizes.size(); ++i) {
Steven Morelandf21962d2018-08-09 12:44:40 -0700103 arrayType += ", " + mSizes[i]->cppValue();
Andreas Huber709b62d2016-09-19 11:21:18 -0700104 }
105
Andreas Huberf03332a2016-09-22 15:35:43 -0700106 arrayType += ">";
Andreas Huber881227d2016-08-02 14:20:21 -0700107
108 switch (mode) {
109 case StorageMode_Stack:
Andreas Huberf03332a2016-09-22 15:35:43 -0700110 return arrayType;
Andreas Huber881227d2016-08-02 14:20:21 -0700111
112 case StorageMode_Argument:
Andreas Huberf03332a2016-09-22 15:35:43 -0700113 return "const " + arrayType + "&";
114
Andreas Huber881227d2016-08-02 14:20:21 -0700115 case StorageMode_Result:
Andreas Huberf03332a2016-09-22 15:35:43 -0700116 return "const " + arrayType + "*";
Andreas Huber881227d2016-08-02 14:20:21 -0700117 }
Andreas Huberf03332a2016-09-22 15:35:43 -0700118
119 CHECK(!"Should not be here");
Andreas Huber881227d2016-08-02 14:20:21 -0700120}
121
Yifan Hong30b5d1f2017-04-03 12:19:25 -0700122std::string ArrayType::getInternalDataCppType() const {
123 std::string result = mElementType->getCppStackType();
124 for (size_t i = 0; i < mSizes.size(); ++i) {
125 result += "[";
126 result += mSizes[i]->cppValue();
127 result += "]";
128 }
129 return result;
130}
131
Yifan Hong4ed13472016-11-02 10:44:11 -0700132std::string ArrayType::getJavaType(bool forInitializer) const {
133 std::string base =
134 mElementType->getJavaType(forInitializer);
Andreas Huber709b62d2016-09-19 11:21:18 -0700135
Yifan Hongf24fa852016-09-23 11:03:15 -0700136 for (size_t i = 0; i < mSizes.size(); ++i) {
Yifan Hong4ed13472016-11-02 10:44:11 -0700137 base += "[";
Andreas Huber709b62d2016-09-19 11:21:18 -0700138
139 if (forInitializer) {
Yifan Hong4ed13472016-11-02 10:44:11 -0700140 base += mSizes[i]->javaValue();
Steven Morelandf21962d2018-08-09 12:44:40 -0700141 } else {
142 base += "/* " + mSizes[i]->expression() + " */";
Yifan Hong5706a432016-11-02 09:44:18 -0700143 }
Yifan Hongf24fa852016-09-23 11:03:15 -0700144
Yifan Hong4ed13472016-11-02 10:44:11 -0700145 base += "]";
Andreas Huber4c865b72016-09-14 15:26:27 -0700146 }
Andreas Huber4c865b72016-09-14 15:26:27 -0700147
148 return base;
Andreas Huber2831d512016-08-15 09:33:47 -0700149}
150
Zhuoyao Zhangc5ea9f52016-10-06 15:05:39 -0700151std::string ArrayType::getVtsType() const {
152 return "TYPE_ARRAY";
153}
154
Andreas Huber881227d2016-08-02 14:20:21 -0700155void ArrayType::emitReaderWriter(
156 Formatter &out,
157 const std::string &name,
158 const std::string &parcelObj,
159 bool parcelObjIsPointer,
160 bool isReader,
161 ErrorMode mode) const {
Yifan Hong3b320f82016-11-01 15:15:54 -0700162 std::string baseType = mElementType->getCppStackType();
Andreas Huber881227d2016-08-02 14:20:21 -0700163
Iliyan Malchev549e2592016-08-10 08:59:12 -0700164 const std::string parentName = "_hidl_" + name + "_parent";
Andreas Huber881227d2016-08-02 14:20:21 -0700165
166 out << "size_t " << parentName << ";\n\n";
167
168 const std::string parcelObjDeref =
169 parcelObj + (parcelObjIsPointer ? "->" : ".");
170
Martijn Coenenb2a861c2017-04-18 15:54:25 -0700171 size_t numArrayElements = 1;
172 for (auto size : mSizes) {
173 numArrayElements *= size->castSizeT();
174 }
Andreas Huber881227d2016-08-02 14:20:21 -0700175 if (isReader) {
Martijn Coenen6a082c62017-01-11 12:47:02 +0100176 out << "_hidl_err = "
Andreas Huber881227d2016-08-02 14:20:21 -0700177 << parcelObjDeref
Martijn Coenenb2a861c2017-04-18 15:54:25 -0700178 << "readBuffer("
179 << numArrayElements
180 << " * sizeof("
181 << baseType
182 << "), &"
Andreas Huber881227d2016-08-02 14:20:21 -0700183 << parentName
Martijn Coenen6a082c62017-01-11 12:47:02 +0100184 << ", "
185 << " reinterpret_cast<const void **>("
186 << "&" << name
187 << "));\n\n";
Andreas Huber881227d2016-08-02 14:20:21 -0700188
Martijn Coenen6a082c62017-01-11 12:47:02 +0100189 handleError(out, mode);
Andreas Huber881227d2016-08-02 14:20:21 -0700190 } else {
Andreas Huberf03332a2016-09-22 15:35:43 -0700191
Iliyan Malchev549e2592016-08-10 08:59:12 -0700192 out << "_hidl_err = "
Andreas Huber881227d2016-08-02 14:20:21 -0700193 << parcelObjDeref
194 << "writeBuffer("
195 << name
Andreas Huberf03332a2016-09-22 15:35:43 -0700196 << ".data(), "
197 << numArrayElements
198 << " * sizeof("
Andreas Huber881227d2016-08-02 14:20:21 -0700199 << baseType
200 << "), &"
201 << parentName
202 << ");\n";
203
204 handleError(out, mode);
205 }
206
207 emitReaderWriterEmbedded(
208 out,
Andreas Huberf9d49f12016-09-12 14:58:36 -0700209 0 /* depth */,
Andreas Huber881227d2016-08-02 14:20:21 -0700210 name,
Yifan Hongbe2a3732016-10-05 13:33:41 -0700211 name /* sanitizedName */,
Andreas Huber881227d2016-08-02 14:20:21 -0700212 isReader /* nameIsPointer */,
213 parcelObj,
214 parcelObjIsPointer,
215 isReader,
216 mode,
217 parentName,
218 "0 /* parentOffset */");
219}
220
221void ArrayType::emitReaderWriterEmbedded(
222 Formatter &out,
Andreas Huberf9d49f12016-09-12 14:58:36 -0700223 size_t depth,
Andreas Huber881227d2016-08-02 14:20:21 -0700224 const std::string &name,
Yifan Hongbe2a3732016-10-05 13:33:41 -0700225 const std::string &sanitizedName,
Andreas Huber881227d2016-08-02 14:20:21 -0700226 bool nameIsPointer,
227 const std::string &parcelObj,
228 bool parcelObjIsPointer,
229 bool isReader,
230 ErrorMode mode,
231 const std::string &parentName,
232 const std::string &offsetText) const {
233 if (!mElementType->needsEmbeddedReadWrite()) {
234 return;
235 }
236
237 const std::string nameDeref = name + (nameIsPointer ? "->" : ".");
238
Yifan Hong3b320f82016-11-01 15:15:54 -0700239 std::string baseType = mElementType->getCppStackType();
Andreas Huber881227d2016-08-02 14:20:21 -0700240
Andreas Huberf9d49f12016-09-12 14:58:36 -0700241 std::string iteratorName = "_hidl_index_" + std::to_string(depth);
242
243 out << "for (size_t "
244 << iteratorName
245 << " = 0; "
246 << iteratorName
247 << " < "
Yifan Hongbf459bc2016-08-23 16:50:37 -0700248 << dimension()
Andreas Huberf9d49f12016-09-12 14:58:36 -0700249 << "; ++"
250 << iteratorName
251 << ") {\n";
Andreas Huber881227d2016-08-02 14:20:21 -0700252
253 out.indent();
254
255 mElementType->emitReaderWriterEmbedded(
256 out,
Andreas Huberf9d49f12016-09-12 14:58:36 -0700257 depth + 1,
Andreas Huberf03332a2016-09-22 15:35:43 -0700258 nameDeref + "data()[" + iteratorName + "]",
Yifan Hongbe2a3732016-10-05 13:33:41 -0700259 sanitizedName + "_indexed",
Andreas Huber881227d2016-08-02 14:20:21 -0700260 false /* nameIsPointer */,
261 parcelObj,
262 parcelObjIsPointer,
263 isReader,
264 mode,
265 parentName,
Andreas Huberf9d49f12016-09-12 14:58:36 -0700266 offsetText
267 + " + " + iteratorName + " * sizeof("
268 + baseType
Andreas Huberf9d49f12016-09-12 14:58:36 -0700269 + ")");
Andreas Huber881227d2016-08-02 14:20:21 -0700270
271 out.unindent();
272
273 out << "}\n\n";
274}
275
Yifan Honge45b5302017-02-22 10:49:07 -0800276void ArrayType::emitJavaDump(
277 Formatter &out,
278 const std::string &streamName,
279 const std::string &name) const {
280 out << streamName << ".append(java.util.Arrays."
281 << (countDimensions() > 1 ? "deepToString" : "toString")
282 << "("
Nirav Atre66842a92018-06-28 18:14:13 -0700283 << name
284 << "));\n";
Yifan Honge45b5302017-02-22 10:49:07 -0800285}
286
287
Andreas Huber881227d2016-08-02 14:20:21 -0700288bool ArrayType::needsEmbeddedReadWrite() const {
289 return mElementType->needsEmbeddedReadWrite();
290}
291
Andreas Huberf03332a2016-09-22 15:35:43 -0700292bool ArrayType::resultNeedsDeref() const {
293 return true;
294}
295
Andreas Huber2831d512016-08-15 09:33:47 -0700296void ArrayType::emitJavaReaderWriter(
297 Formatter &out,
298 const std::string &parcelObj,
299 const std::string &argName,
300 bool isReader) const {
Martijn Coenenb2a861c2017-04-18 15:54:25 -0700301 size_t align, size;
302 getAlignmentAndSize(&align, &size);
303
Andreas Huber709b62d2016-09-19 11:21:18 -0700304 if (isReader) {
Andreas Huber709b62d2016-09-19 11:21:18 -0700305 out << "new "
Yifan Hong4ed13472016-11-02 10:44:11 -0700306 << getJavaType(true /* forInitializer */)
Andreas Huber709b62d2016-09-19 11:21:18 -0700307 << ";\n";
Andreas Huberf630bc82016-09-09 14:52:25 -0700308 }
309
Andreas Huber709b62d2016-09-19 11:21:18 -0700310 out << "{\n";
311 out.indent();
312
Yifan Hong1af73532016-11-09 14:32:58 -0800313 out << "android.os.HwBlob _hidl_blob = ";
Andreas Huber709b62d2016-09-19 11:21:18 -0700314
315 if (isReader) {
316 out << parcelObj
Martijn Coenenb2a861c2017-04-18 15:54:25 -0700317 << ".readBuffer("
318 << size
319 << " /* size */);\n";
Andreas Huber709b62d2016-09-19 11:21:18 -0700320 } else {
Yifan Hong1af73532016-11-09 14:32:58 -0800321 out << "new android.os.HwBlob("
Andreas Huber709b62d2016-09-19 11:21:18 -0700322 << size
323 << " /* size */);\n";
324 }
325
326 emitJavaFieldReaderWriter(
Andreas Huber2831d512016-08-15 09:33:47 -0700327 out,
Andreas Huber709b62d2016-09-19 11:21:18 -0700328 0 /* depth */,
Andreas Huber2831d512016-08-15 09:33:47 -0700329 parcelObj,
Andreas Huber709b62d2016-09-19 11:21:18 -0700330 "_hidl_blob",
Andreas Huber2831d512016-08-15 09:33:47 -0700331 argName,
Andreas Huber709b62d2016-09-19 11:21:18 -0700332 "0 /* offset */",
333 isReader);
334
335 if (!isReader) {
336 out << parcelObj << ".writeBuffer(_hidl_blob);\n";
337 }
338
339 out.unindent();
340 out << "}\n";
Andreas Huber2831d512016-08-15 09:33:47 -0700341}
342
Andreas Huber85eabdb2016-08-25 11:24:49 -0700343void ArrayType::emitJavaFieldInitializer(
344 Formatter &out, const std::string &fieldName) const {
Nirav Atre66842a92018-06-28 18:14:13 -0700345 const std::string typeName = getJavaType(false /* forInitializer */);
Nirav Atre1d565622018-07-13 15:41:21 -0700346 const std::string fieldDeclaration = typeName + " " + fieldName;
Andreas Huber4c865b72016-09-14 15:26:27 -0700347
Nirav Atre66842a92018-06-28 18:14:13 -0700348 emitJavaFieldDefaultInitialValue(out, fieldDeclaration);
349}
350
351void ArrayType::emitJavaFieldDefaultInitialValue(
352 Formatter &out, const std::string &declaredFieldName) const {
353 out << declaredFieldName
Andreas Huber85eabdb2016-08-25 11:24:49 -0700354 << " = new "
Nirav Atre66842a92018-06-28 18:14:13 -0700355 << getJavaType(true /* forInitializer */)
Andreas Huber4c865b72016-09-14 15:26:27 -0700356 << ";\n";
Andreas Huber85eabdb2016-08-25 11:24:49 -0700357}
358
359void ArrayType::emitJavaFieldReaderWriter(
360 Formatter &out,
Andreas Huber4c865b72016-09-14 15:26:27 -0700361 size_t depth,
Andreas Huber709b62d2016-09-19 11:21:18 -0700362 const std::string &parcelName,
Andreas Huber85eabdb2016-08-25 11:24:49 -0700363 const std::string &blobName,
364 const std::string &fieldName,
365 const std::string &offset,
366 bool isReader) const {
Andreas Huber90e8fc22016-09-21 16:36:15 -0700367 out << "{\n";
368 out.indent();
369
Andreas Huber709b62d2016-09-19 11:21:18 -0700370 std::string offsetName = "_hidl_array_offset_" + std::to_string(depth);
371 out << "long " << offsetName << " = " << offset << ";\n";
Andreas Huber4c865b72016-09-14 15:26:27 -0700372
Andreas Hubera23f1ae2017-10-31 11:50:18 -0700373 const bool isPrimitiveArray = mElementType->isScalar();
374
375 /* If the element type corresponds to a Java primitive type we can optimize
376 the innermost loop by copying a linear range of memory instead of doing
377 a per-element copy. As a result the outer nested loop does not include
378 the final dimension. */
379 const size_t loopDimensions = mSizes.size() - (isPrimitiveArray ? 1 : 0);
380
Andreas Huber709b62d2016-09-19 11:21:18 -0700381 std::string indexString;
Andreas Hubera23f1ae2017-10-31 11:50:18 -0700382 for (size_t dim = 0; dim < loopDimensions; ++dim) {
Andreas Huber709b62d2016-09-19 11:21:18 -0700383 std::string iteratorName =
384 "_hidl_index_" + std::to_string(depth) + "_" + std::to_string(dim);
Andreas Huber85eabdb2016-08-25 11:24:49 -0700385
Andreas Huber709b62d2016-09-19 11:21:18 -0700386 out << "for (int "
387 << iteratorName
388 << " = 0; "
389 << iteratorName
390 << " < "
Yifan Hong5706a432016-11-02 09:44:18 -0700391 << mSizes[dim]->javaValue()
Andreas Huber709b62d2016-09-19 11:21:18 -0700392 << "; ++"
393 << iteratorName
394 << ") {\n";
Andreas Huber85eabdb2016-08-25 11:24:49 -0700395
Andreas Huber709b62d2016-09-19 11:21:18 -0700396 out.indent();
397
398 indexString += "[" + iteratorName + "]";
399 }
400
Nirav Atre66842a92018-06-28 18:14:13 -0700401 const bool isIndexed = (loopDimensions > 0);
402 const std::string fieldNameWithCast = isIndexed
403 ? "(" + getJavaTypeCast(fieldName) + ")" + indexString
404 : getJavaTypeCast(fieldName);
Andreas Huber709b62d2016-09-19 11:21:18 -0700405
Nirav Atre66842a92018-06-28 18:14:13 -0700406 if (isReader && mElementType->isCompoundType()) {
407 mElementType->emitJavaFieldDefaultInitialValue(out, fieldNameWithCast);
Andreas Huber709b62d2016-09-19 11:21:18 -0700408 }
Andreas Huber85eabdb2016-08-25 11:24:49 -0700409
Andreas Hubera23f1ae2017-10-31 11:50:18 -0700410 if (!isPrimitiveArray) {
411 mElementType->emitJavaFieldReaderWriter(
412 out,
413 depth + 1,
414 parcelName,
415 blobName,
Nirav Atre66842a92018-06-28 18:14:13 -0700416 fieldNameWithCast,
Andreas Hubera23f1ae2017-10-31 11:50:18 -0700417 offsetName,
418 isReader);
Andreas Huber85eabdb2016-08-25 11:24:49 -0700419
Andreas Hubera23f1ae2017-10-31 11:50:18 -0700420 size_t elementAlign, elementSize;
421 mElementType->getAlignmentAndSize(&elementAlign, &elementSize);
Andreas Huber709b62d2016-09-19 11:21:18 -0700422
Andreas Hubera23f1ae2017-10-31 11:50:18 -0700423 out << offsetName << " += " << std::to_string(elementSize) << ";\n";
424 } else {
425 if (isReader) {
426 out << blobName
427 << ".copyTo"
428 << mElementType->getJavaSuffix()
429 << "Array("
430 << offsetName
431 << ", "
Nirav Atre66842a92018-06-28 18:14:13 -0700432 << fieldNameWithCast
Andreas Hubera23f1ae2017-10-31 11:50:18 -0700433 << ", "
434 << mSizes.back()->javaValue()
435 << " /* size */);\n";
436 } else {
Andreas Huber2dafb922018-08-03 13:54:41 -0700437 std::string elemName = "_hidl_array_item_" + std::to_string(depth);
438
439 out << mElementType->getJavaType(false /* forInitializer */)
440 << "[] "
441 << elemName
442 << " = "
443 << fieldNameWithCast
444 << ";\n\n";
445
446 out << "if ("
447 << elemName
448 << " == null || "
449 << elemName
450 << ".length != "
451 << mSizes.back()->javaValue()
452 << ") {\n";
453
454 out.indent();
455
456 out << "throw new IllegalArgumentException("
457 << "\"Array element is not of the expected length\");\n";
458
459 out.unindent();
460 out << "}\n\n";
461
Andreas Hubera23f1ae2017-10-31 11:50:18 -0700462 out << blobName
463 << ".put"
464 << mElementType->getJavaSuffix()
465 << "Array("
466 << offsetName
467 << ", "
Andreas Huber2dafb922018-08-03 13:54:41 -0700468 << elemName
Andreas Hubera23f1ae2017-10-31 11:50:18 -0700469 << ");\n";
470 }
Andreas Huber709b62d2016-09-19 11:21:18 -0700471
Andreas Hubera23f1ae2017-10-31 11:50:18 -0700472 size_t elementAlign, elementSize;
473 mElementType->getAlignmentAndSize(&elementAlign, &elementSize);
474
475 out << offsetName
476 << " += "
477 << mSizes.back()->javaValue()
478 << " * "
479 << elementSize
480 << ";\n";
481 }
482
483 for (size_t dim = 0; dim < loopDimensions; ++dim) {
Andreas Huber709b62d2016-09-19 11:21:18 -0700484 out.unindent();
485 out << "}\n";
486 }
Andreas Huber90e8fc22016-09-21 16:36:15 -0700487
488 out.unindent();
489 out << "}\n";
Andreas Huber85eabdb2016-08-25 11:24:49 -0700490}
491
Steven Moreland6ec9eb92018-02-16 14:21:49 -0800492void ArrayType::emitVtsTypeDeclarations(Formatter& out) const {
Zhuoyao Zhangc5ea9f52016-10-06 15:05:39 -0700493 out << "type: " << getVtsType() << "\n";
Steven Morelandf21962d2018-08-09 12:44:40 -0700494 out << "vector_size: " << mSizes[0]->rawValue() << "\n";
Zhuoyao Zhangc5ea9f52016-10-06 15:05:39 -0700495 out << "vector_value: {\n";
Zhuoyao Zhang5158db42016-08-10 10:25:20 -0700496 out.indent();
Zhuoyao Zhangeb355ee2016-10-20 16:00:40 -0700497 // Simple array case.
498 if (mSizes.size() == 1) {
Steven Moreland6ec9eb92018-02-16 14:21:49 -0800499 mElementType->emitVtsTypeDeclarations(out);
Zhuoyao Zhangeb355ee2016-10-20 16:00:40 -0700500 } else { // Multi-dimension array case.
501 for (size_t index = 1; index < mSizes.size(); index++) {
502 out << "type: " << getVtsType() << "\n";
Steven Morelandf21962d2018-08-09 12:44:40 -0700503 out << "vector_size: " << mSizes[index]->rawValue() << "\n";
Zhuoyao Zhangeb355ee2016-10-20 16:00:40 -0700504 out << "vector_value: {\n";
505 out.indent();
Zhuoyao Zhangeb355ee2016-10-20 16:00:40 -0700506 if (index == mSizes.size() - 1) {
Steven Moreland6ec9eb92018-02-16 14:21:49 -0800507 mElementType->emitVtsTypeDeclarations(out);
Zhuoyao Zhangeb355ee2016-10-20 16:00:40 -0700508 }
509 }
Zhuoyao Zhang5158db42016-08-10 10:25:20 -0700510 }
Zhuoyao Zhangeb355ee2016-10-20 16:00:40 -0700511 for (size_t index = 0; index < mSizes.size(); index++) {
512 out.unindent();
513 out << "}\n";
514 }
Zhuoyao Zhang5158db42016-08-10 10:25:20 -0700515}
516
Timur Iskhakov5dc72fe2017-09-07 23:13:44 -0700517bool ArrayType::deepIsJavaCompatible(std::unordered_set<const Type*>* visited) const {
518 if (!mElementType->isJavaCompatible(visited)) {
519 return false;
520 }
521 return Type::deepIsJavaCompatible(visited);
Andreas Huber70a59e12016-08-16 12:57:01 -0700522}
523
Timur Iskhakov5dc72fe2017-09-07 23:13:44 -0700524bool ArrayType::deepContainsPointer(std::unordered_set<const Type*>* visited) const {
525 if (mElementType->containsPointer(visited)) {
526 return true;
527 }
528 return Type::deepContainsPointer(visited);
Andreas Huber60d3b222017-03-30 09:10:56 -0700529}
530
Andreas Huber85eabdb2016-08-25 11:24:49 -0700531void ArrayType::getAlignmentAndSize(size_t *align, size_t *size) const {
532 mElementType->getAlignmentAndSize(align, size);
533
Andreas Huber709b62d2016-09-19 11:21:18 -0700534 for (auto sizeInDimension : mSizes) {
Yifan Hong5706a432016-11-02 09:44:18 -0700535 (*size) *= sizeInDimension->castSizeT();
Andreas Huber709b62d2016-09-19 11:21:18 -0700536 }
Andreas Huber85eabdb2016-08-25 11:24:49 -0700537}
538
Yifan Hongbf459bc2016-08-23 16:50:37 -0700539size_t ArrayType::dimension() const {
540 size_t numArrayElements = 1;
541 for (auto size : mSizes) {
Yifan Hong5706a432016-11-02 09:44:18 -0700542 numArrayElements *= size->castSizeT();
Yifan Hongbf459bc2016-08-23 16:50:37 -0700543 }
544 return numArrayElements;
545}
546
Andreas Huberc9410c72016-07-28 12:18:40 -0700547} // namespace android
548