blob: ec91531e892e2d0f29dcf8071681587a2a93645f [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
Iliyan Malcheva72e0d22016-09-09 11:03:08 -070019#include <hidl-util/Formatter.h>
Andreas Huber881227d2016-08-02 14:20:21 -070020#include <android-base/logging.h>
21
Yifan Hongf24fa852016-09-23 11:03:15 -070022#include "ConstantExpression.h"
23
Andreas Huberc9410c72016-07-28 12:18:40 -070024namespace android {
25
Yifan Hongf24fa852016-09-23 11:03:15 -070026ArrayType::ArrayType(ArrayType *srcArray, ConstantExpression *size)
Andreas Huber709b62d2016-09-19 11:21:18 -070027 : mElementType(srcArray->mElementType),
Yifan Hongf24fa852016-09-23 11:03:15 -070028 mSizes(srcArray->mSizes),
29 mSizeComments(srcArray->mSizeComments) {
Andreas Huber709b62d2016-09-19 11:21:18 -070030 addDimension(size);
31}
32
Yifan Hongf24fa852016-09-23 11:03:15 -070033ArrayType::ArrayType(Type *elementType, ConstantExpression *size)
Andreas Huber709b62d2016-09-19 11:21:18 -070034 : mElementType(elementType) {
35 addDimension(size);
36}
37
Yifan Hongf24fa852016-09-23 11:03:15 -070038void ArrayType::addDimension(ConstantExpression *size) {
Yifan Honge77ca132016-09-27 10:49:05 -070039 mSizes.insert(mSizes.begin(), size->castSizeT());
Yifan Hongf24fa852016-09-23 11:03:15 -070040 mSizeComments.insert(mSizeComments.begin(), size->description());
Andreas Huberc9410c72016-07-28 12:18:40 -070041}
42
Andreas Huberf03332a2016-09-22 15:35:43 -070043size_t ArrayType::countDimensions() const {
44 return mSizes.size();
45}
46
Steven Moreland979e0992016-09-07 09:18:08 -070047void ArrayType::addNamedTypesToSet(std::set<const FQName> &set) const {
48 mElementType->addNamedTypesToSet(set);
49}
50
Andreas Huber709b62d2016-09-19 11:21:18 -070051bool ArrayType::isArray() const {
52 return true;
53}
54
Andreas Huberf03332a2016-09-22 15:35:43 -070055Type *ArrayType::getElementType() const {
56 return mElementType;
57}
58
Steven Moreland979e0992016-09-07 09:18:08 -070059std::string ArrayType::getCppType(StorageMode mode,
60 std::string *extra,
61 bool specifyNamespaces) const {
62 const std::string base = mElementType->getCppType(extra, specifyNamespaces);
Andreas Huber709b62d2016-09-19 11:21:18 -070063 CHECK(extra->empty());
Andreas Huber881227d2016-08-02 14:20:21 -070064
Andreas Huberf03332a2016-09-22 15:35:43 -070065 std::string arrayType = "hidl_array<" + base;
66
67 for (size_t i = 0; i < mSizes.size(); ++i) {
68 arrayType += ", ";
69 arrayType += std::to_string(mSizes[i]);
70
71 arrayType += " /* ";
72 arrayType += mSizeComments[i];
73 arrayType += " */";
Andreas Huber709b62d2016-09-19 11:21:18 -070074 }
75
Andreas Huberf03332a2016-09-22 15:35:43 -070076 arrayType += ">";
Andreas Huber881227d2016-08-02 14:20:21 -070077
78 switch (mode) {
79 case StorageMode_Stack:
Andreas Huberf03332a2016-09-22 15:35:43 -070080 return arrayType;
Andreas Huber881227d2016-08-02 14:20:21 -070081
82 case StorageMode_Argument:
Andreas Huberf03332a2016-09-22 15:35:43 -070083 return "const " + arrayType + "&";
84
Andreas Huber881227d2016-08-02 14:20:21 -070085 case StorageMode_Result:
Andreas Huberf03332a2016-09-22 15:35:43 -070086 return "const " + arrayType + "*";
Andreas Huber881227d2016-08-02 14:20:21 -070087 }
Andreas Huberf03332a2016-09-22 15:35:43 -070088
89 CHECK(!"Should not be here");
Andreas Huber881227d2016-08-02 14:20:21 -070090}
91
Andreas Huber4c865b72016-09-14 15:26:27 -070092std::string ArrayType::getJavaType(
93 std::string *extra, bool forInitializer) const {
94 std::string baseExtra;
95 const std::string base =
96 mElementType->getJavaType(&baseExtra, forInitializer);
97
Andreas Huber709b62d2016-09-19 11:21:18 -070098 CHECK(baseExtra.empty());
99
100 extra->clear();
101
Yifan Hongf24fa852016-09-23 11:03:15 -0700102 CHECK(mSizes.size() == mSizeComments.size());
103 for (size_t i = 0; i < mSizes.size(); ++i) {
Andreas Huber709b62d2016-09-19 11:21:18 -0700104 *extra += "[";
105
106 if (forInitializer) {
Yifan Hongf24fa852016-09-23 11:03:15 -0700107 *extra += std::to_string(mSizes[i]);
108 *extra += " ";
Andreas Huber709b62d2016-09-19 11:21:18 -0700109 }
110
Yifan Hongf24fa852016-09-23 11:03:15 -0700111 *extra += "/* " + mSizeComments[i] + " */";
112
Andreas Huber709b62d2016-09-19 11:21:18 -0700113 *extra += "]";
Andreas Huber4c865b72016-09-14 15:26:27 -0700114 }
Andreas Huber4c865b72016-09-14 15:26:27 -0700115
116 return base;
Andreas Huber2831d512016-08-15 09:33:47 -0700117}
118
Andreas Huberf03332a2016-09-22 15:35:43 -0700119std::string ArrayType::getJavaWrapperType() const {
120 return mElementType->getJavaWrapperType();
121}
122
Zhuoyao Zhangc5ea9f52016-10-06 15:05:39 -0700123std::string ArrayType::getVtsType() const {
124 return "TYPE_ARRAY";
125}
126
Andreas Huber881227d2016-08-02 14:20:21 -0700127void ArrayType::emitReaderWriter(
128 Formatter &out,
129 const std::string &name,
130 const std::string &parcelObj,
131 bool parcelObjIsPointer,
132 bool isReader,
133 ErrorMode mode) const {
134 std::string baseExtra;
135 std::string baseType = mElementType->getCppType(&baseExtra);
Andreas Huber709b62d2016-09-19 11:21:18 -0700136 CHECK(baseExtra.empty());
Andreas Huber881227d2016-08-02 14:20:21 -0700137
Iliyan Malchev549e2592016-08-10 08:59:12 -0700138 const std::string parentName = "_hidl_" + name + "_parent";
Andreas Huber881227d2016-08-02 14:20:21 -0700139
140 out << "size_t " << parentName << ";\n\n";
141
142 const std::string parcelObjDeref =
143 parcelObj + (parcelObjIsPointer ? "->" : ".");
144
145 if (isReader) {
Andreas Huberf03332a2016-09-22 15:35:43 -0700146 std::string extra;
147
Andreas Huber881227d2016-08-02 14:20:21 -0700148 out << name
Andreas Huberf03332a2016-09-22 15:35:43 -0700149 << " = ("
150 << getCppResultType(&extra)
151 << ")"
Andreas Huber881227d2016-08-02 14:20:21 -0700152 << parcelObjDeref
153 << "readBuffer(&"
154 << parentName
155 << ");\n\n";
156
157 out << "if (" << name << " == nullptr) {\n";
158
159 out.indent();
160
Iliyan Malchev549e2592016-08-10 08:59:12 -0700161 out << "_hidl_err = ::android::UNKNOWN_ERROR;\n";
Andreas Huber881227d2016-08-02 14:20:21 -0700162 handleError2(out, mode);
163
164 out.unindent();
165 out << "}\n\n";
166 } else {
Andreas Huberf03332a2016-09-22 15:35:43 -0700167 size_t numArrayElements = 1;
168 for (auto size : mSizes) {
169 numArrayElements *= size;
170 }
171
Iliyan Malchev549e2592016-08-10 08:59:12 -0700172 out << "_hidl_err = "
Andreas Huber881227d2016-08-02 14:20:21 -0700173 << parcelObjDeref
174 << "writeBuffer("
175 << name
Andreas Huberf03332a2016-09-22 15:35:43 -0700176 << ".data(), "
177 << numArrayElements
178 << " * sizeof("
Andreas Huber881227d2016-08-02 14:20:21 -0700179 << baseType
180 << "), &"
181 << parentName
182 << ");\n";
183
184 handleError(out, mode);
185 }
186
187 emitReaderWriterEmbedded(
188 out,
Andreas Huberf9d49f12016-09-12 14:58:36 -0700189 0 /* depth */,
Andreas Huber881227d2016-08-02 14:20:21 -0700190 name,
Yifan Hongbe2a3732016-10-05 13:33:41 -0700191 name /* sanitizedName */,
Andreas Huber881227d2016-08-02 14:20:21 -0700192 isReader /* nameIsPointer */,
193 parcelObj,
194 parcelObjIsPointer,
195 isReader,
196 mode,
197 parentName,
198 "0 /* parentOffset */");
199}
200
201void ArrayType::emitReaderWriterEmbedded(
202 Formatter &out,
Andreas Huberf9d49f12016-09-12 14:58:36 -0700203 size_t depth,
Andreas Huber881227d2016-08-02 14:20:21 -0700204 const std::string &name,
Yifan Hongbe2a3732016-10-05 13:33:41 -0700205 const std::string &sanitizedName,
Andreas Huber881227d2016-08-02 14:20:21 -0700206 bool nameIsPointer,
207 const std::string &parcelObj,
208 bool parcelObjIsPointer,
209 bool isReader,
210 ErrorMode mode,
211 const std::string &parentName,
212 const std::string &offsetText) const {
213 if (!mElementType->needsEmbeddedReadWrite()) {
214 return;
215 }
216
217 const std::string nameDeref = name + (nameIsPointer ? "->" : ".");
218
219 std::string baseExtra;
220 std::string baseType = mElementType->getCppType(&baseExtra);
Andreas Huber709b62d2016-09-19 11:21:18 -0700221 CHECK(baseExtra.empty());
Andreas Huber881227d2016-08-02 14:20:21 -0700222
Andreas Huberf9d49f12016-09-12 14:58:36 -0700223 std::string iteratorName = "_hidl_index_" + std::to_string(depth);
224
225 out << "for (size_t "
226 << iteratorName
227 << " = 0; "
228 << iteratorName
229 << " < "
Yifan Hongbf459bc2016-08-23 16:50:37 -0700230 << dimension()
Andreas Huberf9d49f12016-09-12 14:58:36 -0700231 << "; ++"
232 << iteratorName
233 << ") {\n";
Andreas Huber881227d2016-08-02 14:20:21 -0700234
235 out.indent();
236
237 mElementType->emitReaderWriterEmbedded(
238 out,
Andreas Huberf9d49f12016-09-12 14:58:36 -0700239 depth + 1,
Andreas Huberf03332a2016-09-22 15:35:43 -0700240 nameDeref + "data()[" + iteratorName + "]",
Yifan Hongbe2a3732016-10-05 13:33:41 -0700241 sanitizedName + "_indexed",
Andreas Huber881227d2016-08-02 14:20:21 -0700242 false /* nameIsPointer */,
243 parcelObj,
244 parcelObjIsPointer,
245 isReader,
246 mode,
247 parentName,
Andreas Huberf9d49f12016-09-12 14:58:36 -0700248 offsetText
249 + " + " + iteratorName + " * sizeof("
250 + baseType
Andreas Huberf9d49f12016-09-12 14:58:36 -0700251 + ")");
Andreas Huber881227d2016-08-02 14:20:21 -0700252
253 out.unindent();
254
255 out << "}\n\n";
256}
257
Yifan Hongbf459bc2016-08-23 16:50:37 -0700258void ArrayType::emitResolveReferences(
259 Formatter &out,
260 const std::string &name,
261 bool nameIsPointer,
262 const std::string &parcelObj,
263 bool parcelObjIsPointer,
264 bool isReader,
265 ErrorMode mode) const {
266 emitResolveReferencesEmbedded(
267 out,
268 0 /* depth */,
269 name,
270 name /* sanitizedName */,
271 nameIsPointer,
272 parcelObj,
273 parcelObjIsPointer,
274 isReader,
275 mode,
276 "_hidl_" + name + "_parent",
277 "0 /* parentOffset */");
278}
279
280void ArrayType::emitResolveReferencesEmbedded(
281 Formatter &out,
282 size_t depth,
283 const std::string &name,
284 const std::string &sanitizedName,
285 bool nameIsPointer,
286 const std::string &parcelObj,
287 bool parcelObjIsPointer,
288 bool isReader,
289 ErrorMode mode,
290 const std::string &parentName,
291 const std::string &offsetText) const {
292 CHECK(needsResolveReferences() && mElementType->needsResolveReferences());
293
294 const std::string nameDeref = name + (nameIsPointer ? "->" : ".");
295
296 std::string baseExtra;
297 std::string baseType = mElementType->getCppType(&baseExtra);
298
299 std::string iteratorName = "_hidl_index_" + std::to_string(depth);
300
301 out << "for (size_t "
302 << iteratorName
303 << " = 0; "
304 << iteratorName
305 << " < "
306 << dimension()
307 << "; ++"
308 << iteratorName
309 << ") {\n";
310
311 out.indent();
312
313 mElementType->emitResolveReferencesEmbedded(
314 out,
315 depth + 1,
316 nameDeref + "data()[" + iteratorName + "]",
317 sanitizedName + "_indexed",
318 false /* nameIsPointer */,
319 parcelObj,
320 parcelObjIsPointer,
321 isReader,
322 mode,
323 parentName,
324 offsetText + " + " + iteratorName + " * sizeof("
325 + baseType
326 + ")");
327
328 out.unindent();
329
330 out << "}\n\n";
331}
332
333
Andreas Huber881227d2016-08-02 14:20:21 -0700334bool ArrayType::needsEmbeddedReadWrite() const {
335 return mElementType->needsEmbeddedReadWrite();
336}
337
Yifan Hongbf459bc2016-08-23 16:50:37 -0700338bool ArrayType::needsResolveReferences() const {
339 return mElementType->needsResolveReferences();
340}
341
Andreas Huberf03332a2016-09-22 15:35:43 -0700342bool ArrayType::resultNeedsDeref() const {
343 return true;
344}
345
Andreas Huber2831d512016-08-15 09:33:47 -0700346void ArrayType::emitJavaReaderWriter(
347 Formatter &out,
348 const std::string &parcelObj,
349 const std::string &argName,
350 bool isReader) const {
Andreas Huber709b62d2016-09-19 11:21:18 -0700351 if (isReader) {
352 std::string extra;
353 out << "new "
354 << getJavaType(&extra, true /* forInitializer */)
355 << extra
356 << ";\n";
Andreas Huberf630bc82016-09-09 14:52:25 -0700357 }
358
Andreas Huber709b62d2016-09-19 11:21:18 -0700359 out << "{\n";
360 out.indent();
361
362 out << "HwBlob _hidl_blob = ";
363
364 if (isReader) {
365 out << parcelObj
366 << ".readBuffer();\n";
367 } else {
368 size_t align, size;
369 getAlignmentAndSize(&align, &size);
370
371 out << "new HwBlob("
372 << size
373 << " /* size */);\n";
374 }
375
376 emitJavaFieldReaderWriter(
Andreas Huber2831d512016-08-15 09:33:47 -0700377 out,
Andreas Huber709b62d2016-09-19 11:21:18 -0700378 0 /* depth */,
Andreas Huber2831d512016-08-15 09:33:47 -0700379 parcelObj,
Andreas Huber709b62d2016-09-19 11:21:18 -0700380 "_hidl_blob",
Andreas Huber2831d512016-08-15 09:33:47 -0700381 argName,
Andreas Huber709b62d2016-09-19 11:21:18 -0700382 "0 /* offset */",
383 isReader);
384
385 if (!isReader) {
386 out << parcelObj << ".writeBuffer(_hidl_blob);\n";
387 }
388
389 out.unindent();
390 out << "}\n";
Andreas Huber2831d512016-08-15 09:33:47 -0700391}
392
Andreas Huber85eabdb2016-08-25 11:24:49 -0700393void ArrayType::emitJavaFieldInitializer(
394 Formatter &out, const std::string &fieldName) const {
Andreas Huber4c865b72016-09-14 15:26:27 -0700395 std::string extra;
396 std::string typeName = getJavaType(&extra, false /* forInitializer */);
397
398 std::string extraInit;
399 getJavaType(&extraInit, true /* forInitializer */);
400
Andreas Hubercd5e6662016-08-30 15:02:59 -0700401 out << "final "
Andreas Huber4c865b72016-09-14 15:26:27 -0700402 << typeName
403 << extra
404 << " "
Andreas Huber85eabdb2016-08-25 11:24:49 -0700405 << fieldName
406 << " = new "
Andreas Huber4c865b72016-09-14 15:26:27 -0700407 << typeName
408 << extraInit
409 << ";\n";
Andreas Huber85eabdb2016-08-25 11:24:49 -0700410}
411
412void ArrayType::emitJavaFieldReaderWriter(
413 Formatter &out,
Andreas Huber4c865b72016-09-14 15:26:27 -0700414 size_t depth,
Andreas Huber709b62d2016-09-19 11:21:18 -0700415 const std::string &parcelName,
Andreas Huber85eabdb2016-08-25 11:24:49 -0700416 const std::string &blobName,
417 const std::string &fieldName,
418 const std::string &offset,
419 bool isReader) const {
Andreas Huber90e8fc22016-09-21 16:36:15 -0700420 out << "{\n";
421 out.indent();
422
Andreas Huber709b62d2016-09-19 11:21:18 -0700423 std::string offsetName = "_hidl_array_offset_" + std::to_string(depth);
424 out << "long " << offsetName << " = " << offset << ";\n";
Andreas Huber4c865b72016-09-14 15:26:27 -0700425
Andreas Huber709b62d2016-09-19 11:21:18 -0700426 std::string indexString;
427 for (size_t dim = 0; dim < mSizes.size(); ++dim) {
428 std::string iteratorName =
429 "_hidl_index_" + std::to_string(depth) + "_" + std::to_string(dim);
Andreas Huber85eabdb2016-08-25 11:24:49 -0700430
Andreas Huber709b62d2016-09-19 11:21:18 -0700431 out << "for (int "
432 << iteratorName
433 << " = 0; "
434 << iteratorName
435 << " < "
436 << mSizes[dim]
437 << "; ++"
438 << iteratorName
439 << ") {\n";
Andreas Huber85eabdb2016-08-25 11:24:49 -0700440
Andreas Huber709b62d2016-09-19 11:21:18 -0700441 out.indent();
442
443 indexString += "[" + iteratorName + "]";
444 }
445
446 if (isReader && mElementType->isCompoundType()) {
447 std::string extra;
448 std::string typeName =
449 mElementType->getJavaType(&extra, false /* forInitializer */);
450
451 CHECK(extra.empty());
452
453 out << fieldName
454 << indexString
455 << " = new "
456 << typeName
457 << "();\n";
458 }
Andreas Huber85eabdb2016-08-25 11:24:49 -0700459
460 mElementType->emitJavaFieldReaderWriter(
461 out,
Andreas Huber4c865b72016-09-14 15:26:27 -0700462 depth + 1,
Andreas Huber709b62d2016-09-19 11:21:18 -0700463 parcelName,
Andreas Huber85eabdb2016-08-25 11:24:49 -0700464 blobName,
Andreas Huber709b62d2016-09-19 11:21:18 -0700465 fieldName + indexString,
466 offsetName,
Andreas Huber85eabdb2016-08-25 11:24:49 -0700467 isReader);
468
Andreas Huber709b62d2016-09-19 11:21:18 -0700469 size_t elementAlign, elementSize;
470 mElementType->getAlignmentAndSize(&elementAlign, &elementSize);
471
472 out << offsetName << " += " << std::to_string(elementSize) << ";\n";
473
474 for (size_t dim = 0; dim < mSizes.size(); ++dim) {
475 out.unindent();
476 out << "}\n";
477 }
Andreas Huber90e8fc22016-09-21 16:36:15 -0700478
479 out.unindent();
480 out << "}\n";
Andreas Huber85eabdb2016-08-25 11:24:49 -0700481}
482
Zhuoyao Zhang5158db42016-08-10 10:25:20 -0700483status_t ArrayType::emitVtsTypeDeclarations(Formatter &out) const {
Andreas Huber709b62d2016-09-19 11:21:18 -0700484 if (mSizes.size() > 1) {
Zhuoyao Zhangc5ea9f52016-10-06 15:05:39 -0700485 CHECK(!"Multi-dimensional arrays are yet to be supported in VTS.");
Andreas Huber709b62d2016-09-19 11:21:18 -0700486 return UNKNOWN_ERROR;
487 }
488
Zhuoyao Zhangc5ea9f52016-10-06 15:05:39 -0700489 out << "type: " << getVtsType() << "\n";
490 out << "vector_value: {\n";
Zhuoyao Zhang5158db42016-08-10 10:25:20 -0700491 out.indent();
Zhuoyao Zhangc5ea9f52016-10-06 15:05:39 -0700492 out << "vector_size: " << mSizes[0] << "\n";
Zhuoyao Zhang5158db42016-08-10 10:25:20 -0700493 status_t err = mElementType->emitVtsTypeDeclarations(out);
494 if (err != OK) {
495 return err;
496 }
497 out.unindent();
498 out << "}\n";
499 return OK;
500}
501
Andreas Huber70a59e12016-08-16 12:57:01 -0700502bool ArrayType::isJavaCompatible() const {
503 return mElementType->isJavaCompatible();
504}
505
Andreas Huber85eabdb2016-08-25 11:24:49 -0700506void ArrayType::getAlignmentAndSize(size_t *align, size_t *size) const {
507 mElementType->getAlignmentAndSize(align, size);
508
Andreas Huber709b62d2016-09-19 11:21:18 -0700509 for (auto sizeInDimension : mSizes) {
510 (*size) *= sizeInDimension;
511 }
Andreas Huber85eabdb2016-08-25 11:24:49 -0700512}
513
Yifan Hongbf459bc2016-08-23 16:50:37 -0700514size_t ArrayType::dimension() const {
515 size_t numArrayElements = 1;
516 for (auto size : mSizes) {
517 numArrayElements *= size;
518 }
519 return numArrayElements;
520}
521
Andreas Huberc9410c72016-07-28 12:18:40 -0700522} // namespace android
523