blob: 53c15dd120ab1bca85d44a5884e99b84fc95279c [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
19#include "Formatter.h"
20
Andreas Huber881227d2016-08-02 14:20:21 -070021#include <android-base/logging.h>
22
Andreas Huberc9410c72016-07-28 12:18:40 -070023namespace android {
24
25ArrayType::ArrayType(Type *elementType, const char *dimension)
26 : mElementType(elementType),
27 mDimension(dimension) {
28}
29
Andreas Huber881227d2016-08-02 14:20:21 -070030std::string ArrayType::getCppType(StorageMode mode, std::string *extra) const {
31 const std::string base = mElementType->getCppType(extra);
32
33 CHECK(extra->empty());
34
35 *extra = "[" + mDimension + "]";
36
37 switch (mode) {
38 case StorageMode_Stack:
39 return base;
40
41 case StorageMode_Argument:
42 return "const " + base;
43
44 case StorageMode_Result:
45 {
46 extra->clear();
47 return "const " + base + "*";
48 }
49 }
50}
51
Andreas Huber2831d512016-08-15 09:33:47 -070052std::string ArrayType::getJavaType() const {
53 return mElementType->getJavaType() + "[]";
54}
55
Andreas Huber881227d2016-08-02 14:20:21 -070056void ArrayType::emitReaderWriter(
57 Formatter &out,
58 const std::string &name,
59 const std::string &parcelObj,
60 bool parcelObjIsPointer,
61 bool isReader,
62 ErrorMode mode) const {
63 std::string baseExtra;
64 std::string baseType = mElementType->getCppType(&baseExtra);
65
Iliyan Malchev549e2592016-08-10 08:59:12 -070066 const std::string parentName = "_hidl_" + name + "_parent";
Andreas Huber881227d2016-08-02 14:20:21 -070067
68 out << "size_t " << parentName << ";\n\n";
69
70 const std::string parcelObjDeref =
71 parcelObj + (parcelObjIsPointer ? "->" : ".");
72
73 if (isReader) {
74 out << name
75 << " = (const "
76 << baseType
77 << " *)"
78 << parcelObjDeref
79 << "readBuffer(&"
80 << parentName
81 << ");\n\n";
82
83 out << "if (" << name << " == nullptr) {\n";
84
85 out.indent();
86
Iliyan Malchev549e2592016-08-10 08:59:12 -070087 out << "_hidl_err = ::android::UNKNOWN_ERROR;\n";
Andreas Huber881227d2016-08-02 14:20:21 -070088 handleError2(out, mode);
89
90 out.unindent();
91 out << "}\n\n";
92 } else {
Iliyan Malchev549e2592016-08-10 08:59:12 -070093 out << "_hidl_err = "
Andreas Huber881227d2016-08-02 14:20:21 -070094 << parcelObjDeref
95 << "writeBuffer("
96 << name
97 << ", "
98 << mDimension
99 << " * sizeof("
100 << baseType
101 << "), &"
102 << parentName
103 << ");\n";
104
105 handleError(out, mode);
106 }
107
108 emitReaderWriterEmbedded(
109 out,
110 name,
111 isReader /* nameIsPointer */,
112 parcelObj,
113 parcelObjIsPointer,
114 isReader,
115 mode,
116 parentName,
117 "0 /* parentOffset */");
118}
119
120void ArrayType::emitReaderWriterEmbedded(
121 Formatter &out,
122 const std::string &name,
123 bool nameIsPointer,
124 const std::string &parcelObj,
125 bool parcelObjIsPointer,
126 bool isReader,
127 ErrorMode mode,
128 const std::string &parentName,
129 const std::string &offsetText) const {
130 if (!mElementType->needsEmbeddedReadWrite()) {
131 return;
132 }
133
134 const std::string nameDeref = name + (nameIsPointer ? "->" : ".");
135
136 std::string baseExtra;
137 std::string baseType = mElementType->getCppType(&baseExtra);
138
Iliyan Malchev549e2592016-08-10 08:59:12 -0700139 out << "for (size_t _hidl_index = 0; _hidl_index < "
Andreas Huber881227d2016-08-02 14:20:21 -0700140 << mDimension
Iliyan Malchev549e2592016-08-10 08:59:12 -0700141 << "; ++_hidl_index) {\n";
Andreas Huber881227d2016-08-02 14:20:21 -0700142
143 out.indent();
144
145 mElementType->emitReaderWriterEmbedded(
146 out,
Iliyan Malchev549e2592016-08-10 08:59:12 -0700147 name + "[_hidl_index]",
Andreas Huber881227d2016-08-02 14:20:21 -0700148 false /* nameIsPointer */,
149 parcelObj,
150 parcelObjIsPointer,
151 isReader,
152 mode,
153 parentName,
Iliyan Malchev549e2592016-08-10 08:59:12 -0700154 offsetText + " + _hidl_index * sizeof(" + baseType + ")");
Andreas Huber881227d2016-08-02 14:20:21 -0700155
156 out.unindent();
157
158 out << "}\n\n";
159}
160
161bool ArrayType::needsEmbeddedReadWrite() const {
162 return mElementType->needsEmbeddedReadWrite();
163}
164
Andreas Huber2831d512016-08-15 09:33:47 -0700165void ArrayType::emitJavaReaderWriter(
166 Formatter &out,
167 const std::string &parcelObj,
168 const std::string &argName,
169 bool isReader) const {
170 emitJavaReaderWriterWithSuffix(
171 out,
172 parcelObj,
173 argName,
174 isReader,
175 mElementType->getJavaSuffix() + "Array",
176 mDimension);
177}
178
Andreas Huber85eabdb2016-08-25 11:24:49 -0700179void ArrayType::emitJavaFieldInitializer(
180 Formatter &out, const std::string &fieldName) const {
Andreas Hubercd5e6662016-08-30 15:02:59 -0700181 out << "final "
182 << mElementType->getJavaType()
Andreas Huber85eabdb2016-08-25 11:24:49 -0700183 << "[] "
184 << fieldName
185 << " = new "
186 << mElementType->getJavaType()
187 << "["
188 << mDimension
189 << "];\n";
190}
191
192void ArrayType::emitJavaFieldReaderWriter(
193 Formatter &out,
194 const std::string &blobName,
195 const std::string &fieldName,
196 const std::string &offset,
197 bool isReader) const {
198 out << "for (int _hidl_index = 0; _hidl_index < "
199 << mDimension
200 << "; ++_hidl_index) {\n";
201
202 out.indent();
203
204 size_t elementAlign, elementSize;
205 mElementType->getAlignmentAndSize(&elementAlign, &elementSize);
206
207 mElementType->emitJavaFieldReaderWriter(
208 out,
209 blobName,
210 fieldName + "[_hidl_index]",
211 offset + " + _hidl_index * " + std::to_string(elementSize),
212 isReader);
213
214 out.unindent();
215 out << "}\n";
216}
217
Zhuoyao Zhang5158db42016-08-10 10:25:20 -0700218status_t ArrayType::emitVtsTypeDeclarations(Formatter &out) const {
219 out << "type: TYPE_ARRAY\n" << "vector_value: {\n";
220 out.indent();
Zhuoyao Zhang864c7712016-08-16 15:35:28 -0700221 out << "size: " << mDimension << "\n";
Zhuoyao Zhang5158db42016-08-10 10:25:20 -0700222 status_t err = mElementType->emitVtsTypeDeclarations(out);
223 if (err != OK) {
224 return err;
225 }
226 out.unindent();
227 out << "}\n";
228 return OK;
229}
230
Andreas Huber70a59e12016-08-16 12:57:01 -0700231bool ArrayType::isJavaCompatible() const {
232 return mElementType->isJavaCompatible();
233}
234
Andreas Huber85eabdb2016-08-25 11:24:49 -0700235void ArrayType::getAlignmentAndSize(size_t *align, size_t *size) const {
236 mElementType->getAlignmentAndSize(align, size);
237
238 char *end;
239 unsigned long dim = strtoul(mDimension.c_str(), &end, 10);
240 CHECK(end > mDimension.c_str() && *end == '\0');
241
242 (*size) *= dim;
243}
244
Andreas Huberc9410c72016-07-28 12:18:40 -0700245} // namespace android
246