blob: 1fc47184b8127fb2dca211cbffc4ceacce15cdf1 [file] [log] [blame]
Steven Morelande8a3a192018-09-20 14:14:28 -07001/*
2 * Copyright (C) 2018, 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 * limitations under the License.
13 */
14
15#include "aidl_to_ndk.h"
16#include "aidl_language.h"
Steven Morelandaada3422018-09-20 15:55:33 -070017#include "aidl_to_cpp_common.h"
Steven Morelande8a3a192018-09-20 14:14:28 -070018#include "logging.h"
Steven Moreland7c933372018-10-11 15:20:04 -070019#include "os.h"
Steven Morelande8a3a192018-09-20 14:14:28 -070020
Devin Moore53fc99c2020-08-12 08:07:52 -070021#include <android-base/stringprintf.h>
Steven Morelande8a3a192018-09-20 14:14:28 -070022#include <android-base/strings.h>
23
24#include <functional>
25
26using ::android::base::Join;
Jooyung Han4b832522021-10-06 16:08:30 +090027using ::android::base::Split;
Steven Morelande8a3a192018-09-20 14:14:28 -070028
29namespace android {
30namespace aidl {
31namespace ndk {
32
Steven Moreland7c933372018-10-11 15:20:04 -070033std::string NdkHeaderFile(const AidlDefinedType& defined_type, cpp::ClassNames name,
34 bool use_os_sep) {
35 char seperator = (use_os_sep) ? OS_PATH_SEPARATOR : '/';
36 return std::string("aidl") + seperator + cpp::HeaderFile(defined_type, name, use_os_sep);
37}
38
Steven Moreland055d8792018-11-14 12:48:42 -080039// This represents a type in AIDL (e.g. 'String' which can be referenced in multiple ways)
Steven Morelande8a3a192018-09-20 14:14:28 -070040struct TypeInfo {
Steven Moreland055d8792018-11-14 12:48:42 -080041 struct Aspect {
42 // name of the type in C++ output
43 std::string cpp_name;
44 // whether to prefer 'value type' over 'const&'
45 bool value_is_cheap;
Steven Morelande8a3a192018-09-20 14:14:28 -070046
Steven Moreland055d8792018-11-14 12:48:42 -080047 std::function<void(const CodeGeneratorContext& c)> read_func;
48 std::function<void(const CodeGeneratorContext& c)> write_func;
49 };
Steven Morelandeb38ee72018-10-15 14:20:04 -070050
Steven Moreland055d8792018-11-14 12:48:42 -080051 // e.g. 'String'
52 Aspect raw;
53
54 // e.g. 'String[]'
55 std::shared_ptr<Aspect> array;
56
57 // note: Nullable types do not exist in Java. For most Java types, the type is split into a
58 // nullable and non-nullable variant. This is because C++ types are more usually non-nullable, but
59 // everything in Java is non-nullable. This does mean that some Java interfaces may have to have
60 // '@nullable' added to them in order to function as expected w/ the NDK. It also means that some
61 // transactions will be allowed in Java which are not allowed in C++. However, in Java, if a null
62 // is ignored, it will just result in a NullPointerException and be delivered to the other side.
63 // C++ does not have this same capacity (in Android), and so instead, we distinguish nullability
64 // in the type system.
65
66 // e.g. '@nullable String'
67 std::shared_ptr<Aspect> nullable;
68
69 // e.g. '@nullable String[]'
70 std::shared_ptr<Aspect> nullable_array;
Steven Morelande8a3a192018-09-20 14:14:28 -070071};
72
Daniel Norman37d43dd2019-09-09 17:22:34 -070073std::string ConstantValueDecorator(const AidlTypeSpecifier& type, const std::string& raw_value) {
Jooyung Han690f5842020-12-04 13:02:04 +090074 if (type.IsArray()) {
75 return raw_value;
76 }
77
Will McVickerd7d18df2019-09-12 13:40:50 -070078 if (type.GetName() == "long" && !type.IsArray()) {
Daniel Norman37d43dd2019-09-09 17:22:34 -070079 return raw_value + "L";
80 }
81
Jooyung Han690f5842020-12-04 13:02:04 +090082 if (auto defined_type = type.GetDefinedType(); defined_type) {
83 auto enum_type = defined_type->AsEnumDeclaration();
84 AIDL_FATAL_IF(!enum_type, type) << "Invalid type for \"" << raw_value << "\"";
85 return NdkFullClassName(*enum_type, cpp::ClassNames::RAW) +
86 "::" + raw_value.substr(raw_value.find_last_of('.') + 1);
87 }
88
Daniel Norman37d43dd2019-09-09 17:22:34 -070089 return raw_value;
90};
91
Steven Moreland67caf422018-10-15 12:39:12 -070092static std::function<void(const CodeGeneratorContext& c)> StandardRead(const std::string& name) {
Steven Morelande8a3a192018-09-20 14:14:28 -070093 return [name](const CodeGeneratorContext& c) {
Steven Moreland67caf422018-10-15 12:39:12 -070094 c.writer << name << "(" << c.parcel << ", " << c.var << ")";
Steven Morelande8a3a192018-09-20 14:14:28 -070095 };
96}
Steven Moreland67caf422018-10-15 12:39:12 -070097static std::function<void(const CodeGeneratorContext& c)> StandardWrite(const std::string& name) {
Steven Morelande8a3a192018-09-20 14:14:28 -070098 return [name](const CodeGeneratorContext& c) {
Steven Moreland67caf422018-10-15 12:39:12 -070099 c.writer << name << "(" << c.parcel << ", " << c.var << ")";
100 };
101}
102
Jooyung Han241fdda2020-02-21 21:17:44 +0900103TypeInfo PrimitiveType(const std::string& cpp_name, const std::string& pretty_name,
104 const std::optional<std::string>& cpp_name_for_array_opt = std::nullopt) {
105 std::string cpp_name_for_array = cpp_name_for_array_opt.value_or(cpp_name);
Steven Moreland67caf422018-10-15 12:39:12 -0700106 return TypeInfo{
Steven Moreland055d8792018-11-14 12:48:42 -0800107 .raw =
108 TypeInfo::Aspect{
109 .cpp_name = cpp_name,
110 .value_is_cheap = true,
111 .read_func = StandardRead("AParcel_read" + pretty_name),
112 .write_func = StandardWrite("AParcel_write" + pretty_name),
113 },
114 .array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
Jooyung Han241fdda2020-02-21 21:17:44 +0900115 .cpp_name = "std::vector<" + cpp_name_for_array + ">",
Steven Moreland055d8792018-11-14 12:48:42 -0800116 .value_is_cheap = false,
117 .read_func = StandardRead("::ndk::AParcel_readVector"),
118 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
119 }),
120 .nullable = nullptr,
Steven Morelandc10ed462018-11-15 14:50:28 -0800121 .nullable_array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
Jooyung Han241fdda2020-02-21 21:17:44 +0900122 .cpp_name = "std::optional<std::vector<" + cpp_name_for_array + ">>",
Steven Morelandc10ed462018-11-15 14:50:28 -0800123 .value_is_cheap = false,
124 .read_func = StandardRead("::ndk::AParcel_readVector"),
125 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
126 }),
Steven Moreland055d8792018-11-14 12:48:42 -0800127 };
128}
129
130TypeInfo InterfaceTypeInfo(const AidlInterface& type) {
131 const std::string clazz = NdkFullClassName(type, cpp::ClassNames::INTERFACE);
132
133 return TypeInfo{
134 .raw =
135 TypeInfo::Aspect{
136 .cpp_name = "std::shared_ptr<" + clazz + ">",
137 .value_is_cheap = false,
Steven Moreland055d8792018-11-14 12:48:42 -0800138 .read_func = StandardRead(clazz + "::readFromParcel"),
139 .write_func = StandardWrite(clazz + "::writeToParcel"),
140 },
Jooyung Han1d3ec712021-11-02 04:41:36 +0900141 .array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
142 .cpp_name = "std::vector<std::shared_ptr<" + clazz + ">>",
143 .value_is_cheap = false,
144 .read_func = StandardRead("::ndk::AParcel_readVector"),
145 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
146 }),
Steven Moreland055d8792018-11-14 12:48:42 -0800147 .nullable = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
148 .cpp_name = "std::shared_ptr<" + clazz + ">",
149 .value_is_cheap = false,
150 .read_func = StandardRead(clazz + "::readFromParcel"),
151 .write_func = StandardWrite(clazz + "::writeToParcel"),
152 }),
Jooyung Han1d3ec712021-11-02 04:41:36 +0900153 .nullable_array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
154 .cpp_name = "std::optional<std::vector<std::shared_ptr<" + clazz + ">>>",
155 .value_is_cheap = false,
156 .read_func = StandardRead("::ndk::AParcel_readVector"),
157 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
158 }),
Steven Moreland055d8792018-11-14 12:48:42 -0800159 };
160}
161
Devin Moore53fc99c2020-08-12 08:07:52 -0700162TypeInfo ParcelableTypeInfo(const AidlParcelable& type, const AidlTypeSpecifier& typeSpec,
163 const AidlTypenames& types) {
164 std::string clazz = NdkFullClassName(type, cpp::ClassNames::RAW);
165 std::string template_params = "";
166 if (typeSpec.IsGeneric()) {
167 std::vector<std::string> type_params;
168 for (const auto& parameter : typeSpec.GetTypeParameters()) {
169 type_params.push_back(NdkNameOf(types, *parameter, StorageMode::STACK));
170 }
171 clazz += base::StringPrintf("<%s>", base::Join(type_params, ", ").c_str());
172 }
Jooyung Han01720ed2021-08-13 07:46:07 +0900173 const std::string nullable = typeSpec.IsHeapNullable() ? "std::unique_ptr" : "std::optional";
Steven Moreland055d8792018-11-14 12:48:42 -0800174 return TypeInfo{
175 .raw =
176 TypeInfo::Aspect{
177 .cpp_name = clazz,
178 .value_is_cheap = false,
Steven Moreland4348f9a2019-12-16 16:33:01 -0800179 .read_func = StandardRead("::ndk::AParcel_readParcelable"),
180 .write_func = StandardWrite("::ndk::AParcel_writeParcelable"),
Steven Moreland055d8792018-11-14 12:48:42 -0800181 },
Steven Moreland5ecec6b2018-12-11 18:56:41 -0800182 .array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
183 .cpp_name = "std::vector<" + clazz + ">",
184 .value_is_cheap = false,
185 .read_func = StandardRead("::ndk::AParcel_readVector"),
186 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
187 }),
Steven Moreland18f75262019-12-19 16:58:55 -0800188 .nullable = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
Jooyung Han01720ed2021-08-13 07:46:07 +0900189 .cpp_name = nullable + "<" + clazz + ">",
Steven Moreland18f75262019-12-19 16:58:55 -0800190 .value_is_cheap = false,
191 .read_func = StandardRead("::ndk::AParcel_readNullableParcelable"),
192 .write_func = StandardWrite("::ndk::AParcel_writeNullableParcelable"),
193 }),
Jooyung Hanb4997aa2021-10-16 03:26:12 +0900194 .nullable_array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
195 .cpp_name = "std::optional<std::vector<std::optional<" + clazz + ">>>",
196 .value_is_cheap = false,
197 .read_func = StandardRead("::ndk::AParcel_readVector"),
198 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
199 }),
Steven Morelande8a3a192018-09-20 14:14:28 -0700200 };
201}
202
Daniel Norman37d43dd2019-09-09 17:22:34 -0700203TypeInfo EnumDeclarationTypeInfo(const AidlEnumDeclaration& enum_decl) {
Steven Morelandb8df37d2019-11-21 12:33:24 -0800204 const std::string clazz = NdkFullClassName(enum_decl, cpp::ClassNames::RAW);
Daniel Norman37d43dd2019-09-09 17:22:34 -0700205
206 static map<std::string, std::string> kAParcelTypeNameMap = {
207 {"byte", "Byte"},
208 {"int", "Int32"},
209 {"long", "Int64"},
210 };
211 auto aparcel_name_it = kAParcelTypeNameMap.find(enum_decl.GetBackingType().GetName());
Steven Moreland21780812020-09-11 01:29:45 +0000212 AIDL_FATAL_IF(aparcel_name_it == kAParcelTypeNameMap.end(), enum_decl);
Daniel Norman37d43dd2019-09-09 17:22:34 -0700213 const std::string aparcel_name = aparcel_name_it->second;
214
215 const std::string backing_type_name =
216 NdkNameOf(AidlTypenames(), enum_decl.GetBackingType(), StorageMode::STACK);
217
218 return TypeInfo{
219 .raw = TypeInfo::Aspect{
220 .cpp_name = clazz,
221 .value_is_cheap = true,
222 .read_func =
223 [aparcel_name, backing_type_name](const CodeGeneratorContext& c) {
224 c.writer << "AParcel_read" << aparcel_name << "(" << c.parcel
225 << ", reinterpret_cast<" << backing_type_name << "*>(" << c.var << "))";
226 },
227 .write_func =
228 [aparcel_name, backing_type_name](const CodeGeneratorContext& c) {
229 c.writer << "AParcel_write" << aparcel_name << "(" << c.parcel << ", static_cast<"
230 << backing_type_name << ">(" << c.var << "))";
231 },
232 },
Daniel Normanee8674f2019-09-20 16:07:00 -0700233 .array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
234 .cpp_name = "std::vector<" + clazz + ">",
235 .value_is_cheap = false,
236 .read_func =
237 [aparcel_name, backing_type_name](const CodeGeneratorContext& c) {
238 c.writer << "AParcel_read" << aparcel_name << "Array(" << c.parcel
239 << ", static_cast<void*>(" << c.var
240 << "), ndk::AParcel_stdVectorAllocator<" << backing_type_name << ">)";
241 },
242 .write_func =
243 [aparcel_name, backing_type_name](const CodeGeneratorContext& c) {
244 c.writer << "AParcel_write" << aparcel_name << "Array(" << c.parcel
245 << ", reinterpret_cast<const " << backing_type_name << "*>(" << c.var
246 << ".data()), " << c.var << ".size())";
247 },
248 }),
Daniel Norman37d43dd2019-09-09 17:22:34 -0700249 .nullable = nullptr,
Daniel Normanee8674f2019-09-20 16:07:00 -0700250 .nullable_array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
251 .cpp_name = "std::optional<std::vector<" + clazz + ">>",
252 .value_is_cheap = false,
253 .read_func =
254 [aparcel_name, backing_type_name](const CodeGeneratorContext& c) {
255 c.writer << "AParcel_read" << aparcel_name << "Array(" << c.parcel
256 << ", static_cast<void*>(" << c.var
257 << "), ndk::AParcel_nullableStdVectorAllocator<" << backing_type_name
258 << ">)";
259 },
260 .write_func =
261 [aparcel_name, backing_type_name](const CodeGeneratorContext& c) {
262 // If the var exists, use writeArray with data() and size().
263 // Otherwise, use nullptr and -1.
264 c.writer << "AParcel_write" << aparcel_name << "Array(" << c.parcel << ", ("
265 << c.var << " ? reinterpret_cast<const " << backing_type_name << "*>("
266 << c.var << "->data()) : nullptr)"
267 << ", (" << c.var << " ? " << c.var << "->size() : -1))";
268 },
269 }),
Daniel Norman37d43dd2019-09-09 17:22:34 -0700270 };
271}
272
273// map from AIDL built-in type name to the corresponding Ndk type info
Steven Morelande8a3a192018-09-20 14:14:28 -0700274static map<std::string, TypeInfo> kNdkTypeInfoMap = {
Steven Moreland055d8792018-11-14 12:48:42 -0800275 {"void", TypeInfo{{"void", true, nullptr, nullptr}, nullptr, nullptr, nullptr}},
Steven Moreland67caf422018-10-15 12:39:12 -0700276 {"boolean", PrimitiveType("bool", "Bool")},
Jooyung Han241fdda2020-02-21 21:17:44 +0900277 {"byte", PrimitiveType("int8_t", "Byte", "uint8_t")},
Steven Moreland67caf422018-10-15 12:39:12 -0700278 {"char", PrimitiveType("char16_t", "Char")},
279 {"int", PrimitiveType("int32_t", "Int32")},
280 {"long", PrimitiveType("int64_t", "Int64")},
281 {"float", PrimitiveType("float", "Float")},
282 {"double", PrimitiveType("double", "Double")},
Steven Moreland63404532018-10-08 14:31:00 -0700283 {"String",
Steven Moreland055d8792018-11-14 12:48:42 -0800284 TypeInfo{
285 .raw =
286 TypeInfo::Aspect{
287 .cpp_name = "std::string",
288 .value_is_cheap = false,
289 .read_func = StandardRead("::ndk::AParcel_readString"),
290 .write_func = StandardWrite("::ndk::AParcel_writeString"),
291 },
292 .array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
293 .cpp_name = "std::vector<std::string>",
294 .value_is_cheap = false,
295 .read_func = StandardRead("::ndk::AParcel_readVector"),
296 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
297 }),
Steven Moreland18ed9782018-11-14 17:08:09 -0800298 .nullable = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
299 .cpp_name = "std::optional<std::string>",
300 .value_is_cheap = false,
301 .read_func = StandardRead("::ndk::AParcel_readString"),
302 .write_func = StandardWrite("::ndk::AParcel_writeString"),
303 }),
304 .nullable_array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
305 .cpp_name = "std::optional<std::vector<std::optional<std::string>>>",
306 .value_is_cheap = false,
307 .read_func = StandardRead("::ndk::AParcel_readVector"),
308 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
309 }),
Steven Moreland055d8792018-11-14 12:48:42 -0800310 }},
Jeongik Cha1a0f22d2019-11-18 23:22:23 +0900311 // TODO(b/136048684) {"Map", ""},
Steven Moreland055d8792018-11-14 12:48:42 -0800312 {"IBinder",
313 TypeInfo{
314 .raw =
315 TypeInfo::Aspect{
316 .cpp_name = "::ndk::SpAIBinder",
317 .value_is_cheap = false,
Steven Morelande1048a32018-11-16 12:52:17 -0800318 .read_func = StandardRead("::ndk::AParcel_readRequiredStrongBinder"),
319 .write_func = StandardRead("::ndk::AParcel_writeRequiredStrongBinder"),
Steven Moreland055d8792018-11-14 12:48:42 -0800320 },
Jooyung Hanbc7ab902021-10-14 07:09:49 +0900321 .array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
322 .cpp_name = "std::vector<::ndk::SpAIBinder>",
323 .value_is_cheap = false,
324 .read_func = StandardRead("::ndk::AParcel_readVector"),
325 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
326 }),
Steven Moreland055d8792018-11-14 12:48:42 -0800327 .nullable = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
328 .cpp_name = "::ndk::SpAIBinder",
329 .value_is_cheap = false,
Steven Morelande1048a32018-11-16 12:52:17 -0800330 .read_func = StandardRead("::ndk::AParcel_readNullableStrongBinder"),
331 .write_func = StandardRead("::ndk::AParcel_writeNullableStrongBinder"),
Steven Moreland055d8792018-11-14 12:48:42 -0800332 }),
Jooyung Hanbc7ab902021-10-14 07:09:49 +0900333 .nullable_array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
334 .cpp_name = "std::optional<std::vector<::ndk::SpAIBinder>>",
335 .value_is_cheap = false,
336 .read_func = StandardRead("::ndk::AParcel_readVector"),
337 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
338 }),
Steven Moreland055d8792018-11-14 12:48:42 -0800339 }},
Steven Moreland055d8792018-11-14 12:48:42 -0800340 {"ParcelFileDescriptor",
341 TypeInfo{
342 .raw =
343 TypeInfo::Aspect{
344 .cpp_name = "::ndk::ScopedFileDescriptor",
345 .value_is_cheap = false,
Steven Moreland962c60d2018-11-16 15:19:27 -0800346 .read_func = StandardRead("::ndk::AParcel_readRequiredParcelFileDescriptor"),
347 .write_func = StandardRead("::ndk::AParcel_writeRequiredParcelFileDescriptor"),
Steven Moreland055d8792018-11-14 12:48:42 -0800348 },
Jeongik Chacc8dd1d2019-11-14 10:22:57 +0900349 .array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
350 .cpp_name = "std::vector<::ndk::ScopedFileDescriptor>",
351 .value_is_cheap = false,
352 .read_func = StandardRead("::ndk::AParcel_readVector"),
353 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
354 }),
Steven Moreland962c60d2018-11-16 15:19:27 -0800355 .nullable = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
356 .cpp_name = "::ndk::ScopedFileDescriptor",
357 .value_is_cheap = false,
358 .read_func = StandardRead("::ndk::AParcel_readNullableParcelFileDescriptor"),
359 .write_func = StandardRead("::ndk::AParcel_writeNullableParcelFileDescriptor"),
360 }),
Jooyung Hanb4997aa2021-10-16 03:26:12 +0900361 .nullable_array = std::shared_ptr<TypeInfo::Aspect>(new TypeInfo::Aspect{
362 .cpp_name = "std::optional<std::vector<::ndk::ScopedFileDescriptor>>",
363 .value_is_cheap = false,
364 .read_func = StandardRead("::ndk::AParcel_readVector"),
365 .write_func = StandardWrite("::ndk::AParcel_writeVector"),
366 }),
Steven Moreland055d8792018-11-14 12:48:42 -0800367 }},
Jeongik Cha8f02a532020-10-14 00:16:28 +0900368 {"ParcelableHolder",
369 TypeInfo{
370 .raw =
371 TypeInfo::Aspect{
372 .cpp_name = "::ndk::AParcelableHolder",
373 .value_is_cheap = false,
374 .read_func = StandardRead("::ndk::AParcel_readParcelable"),
375 .write_func = StandardWrite("::ndk::AParcel_writeParcelable"),
376 },
377 .array = nullptr,
378 .nullable = nullptr,
379 .nullable_array = nullptr,
380 }},
Steven Morelande8a3a192018-09-20 14:14:28 -0700381};
382
Jooyung Han5014b3e2021-10-15 09:58:20 +0900383static TypeInfo GetTypeInfo(const AidlTypenames& types, const AidlTypeSpecifier& aidl) {
384 auto& aidl_name = aidl.GetName();
385
386 if (AidlTypenames::IsBuiltinTypename(aidl_name)) {
387 auto it = kNdkTypeInfoMap.find(aidl_name);
388 AIDL_FATAL_IF(it == kNdkTypeInfoMap.end(), aidl_name);
389 return it->second;
390 }
391 const AidlDefinedType* type = types.TryGetDefinedType(aidl_name);
392 AIDL_FATAL_IF(type == nullptr, aidl_name) << "Unrecognized type.";
393
394 if (const AidlInterface* intf = type->AsInterface(); intf != nullptr) {
395 return InterfaceTypeInfo(*intf);
396 } else if (const AidlParcelable* parcelable = type->AsParcelable(); parcelable != nullptr) {
397 return ParcelableTypeInfo(*parcelable, aidl, types);
398 } else if (const AidlEnumDeclaration* enum_decl = type->AsEnumDeclaration();
399 enum_decl != nullptr) {
400 return EnumDeclarationTypeInfo(*enum_decl);
401 } else {
402 AIDL_FATAL(aidl_name) << "Unrecognized type";
403 }
404}
405
Steven Moreland055d8792018-11-14 12:48:42 -0800406static TypeInfo::Aspect GetTypeAspect(const AidlTypenames& types, const AidlTypeSpecifier& aidl) {
Steven Moreland21780812020-09-11 01:29:45 +0000407 AIDL_FATAL_IF(!aidl.IsResolved(), aidl) << aidl.ToString();
Jeongik Cha1a0f22d2019-11-18 23:22:23 +0900408 auto& aidl_name = aidl.GetName();
Steven Moreland1cb099e2018-10-17 16:31:08 -0700409
Steven Moreland67caf422018-10-15 12:39:12 -0700410 TypeInfo info;
Jeongik Cha1a0f22d2019-11-18 23:22:23 +0900411
412 // TODO(b/136048684): For now, List<T> is converted to T[].(Both are using vector<T>)
413 if (aidl_name == "List") {
Steven Morelandfe52c9f2019-11-27 19:04:48 -0800414 AIDL_FATAL_IF(!aidl.IsGeneric(), aidl) << "List must be generic type.";
415 AIDL_FATAL_IF(aidl.GetTypeParameters().size() != 1, aidl)
Jeongik Cha1a0f22d2019-11-18 23:22:23 +0900416 << "List can accept only one type parameter.";
Jooyung Han5014b3e2021-10-15 09:58:20 +0900417 const auto& type_param = *aidl.GetTypeParameters()[0];
Jeongik Cha1a0f22d2019-11-18 23:22:23 +0900418 // TODO(b/136048684) AIDL doesn't support nested type parameter yet.
Jooyung Han5014b3e2021-10-15 09:58:20 +0900419 AIDL_FATAL_IF(type_param.IsGeneric(), aidl) << "AIDL doesn't support nested type parameter";
Jeongik Cha1a0f22d2019-11-18 23:22:23 +0900420
Jooyung Han5014b3e2021-10-15 09:58:20 +0900421 info = GetTypeInfo(types, type_param);
Steven Moreland67caf422018-10-15 12:39:12 -0700422 } else {
Jooyung Han5014b3e2021-10-15 09:58:20 +0900423 info = GetTypeInfo(types, aidl);
Steven Moreland67caf422018-10-15 12:39:12 -0700424 }
425
Jooyung Han5014b3e2021-10-15 09:58:20 +0900426 if (aidl.IsArray() || aidl_name == "List") {
Steven Moreland055d8792018-11-14 12:48:42 -0800427 if (aidl.IsNullable()) {
Steven Morelandd59e3172020-05-11 16:42:09 -0700428 AIDL_FATAL_IF(info.nullable_array == nullptr, aidl)
429 << "Unsupported type in NDK Backend: " << aidl.ToString();
Steven Moreland055d8792018-11-14 12:48:42 -0800430 return *info.nullable_array;
431 }
Steven Morelandd59e3172020-05-11 16:42:09 -0700432 AIDL_FATAL_IF(info.array == nullptr, aidl)
433 << "Unsupported type in NDK Backend: " << aidl.ToString();
Steven Moreland055d8792018-11-14 12:48:42 -0800434 return *info.array;
435 }
Steven Moreland1cb099e2018-10-17 16:31:08 -0700436
Steven Moreland055d8792018-11-14 12:48:42 -0800437 if (aidl.IsNullable()) {
Steven Morelandd59e3172020-05-11 16:42:09 -0700438 AIDL_FATAL_IF(info.nullable == nullptr, aidl)
439 << "Unsupported type in NDK Backend: " << aidl.ToString();
Steven Moreland055d8792018-11-14 12:48:42 -0800440 return *info.nullable;
441 }
442
443 return info.raw;
Steven Moreland67caf422018-10-15 12:39:12 -0700444}
445
Steven Moreland2bea13b2018-10-03 15:12:33 -0700446std::string NdkFullClassName(const AidlDefinedType& type, cpp::ClassNames name) {
Steven Moreland63404532018-10-08 14:31:00 -0700447 std::vector<std::string> pieces = {"::aidl"};
Jooyung Han4b832522021-10-06 16:08:30 +0900448 std::vector<std::string> split_name = Split(type.GetCanonicalName(), ".");
449 pieces.insert(pieces.end(), split_name.begin(), split_name.end());
450 // Override name part with cpp::ClassName(type, name)
451 pieces.back() = cpp::ClassName(type, name);
Steven Moreland2bea13b2018-10-03 15:12:33 -0700452 return Join(pieces, "::");
453}
454
455std::string NdkNameOf(const AidlTypenames& types, const AidlTypeSpecifier& aidl, StorageMode mode) {
Steven Moreland055d8792018-11-14 12:48:42 -0800456 TypeInfo::Aspect aspect = GetTypeAspect(types, aidl);
Steven Morelandeb38ee72018-10-15 14:20:04 -0700457
Steven Morelande8a3a192018-09-20 14:14:28 -0700458 switch (mode) {
459 case StorageMode::STACK:
Steven Moreland055d8792018-11-14 12:48:42 -0800460 return aspect.cpp_name;
Steven Morelande8a3a192018-09-20 14:14:28 -0700461 case StorageMode::ARGUMENT:
Steven Moreland055d8792018-11-14 12:48:42 -0800462 if (aspect.value_is_cheap) {
463 return aspect.cpp_name;
Steven Morelande8a3a192018-09-20 14:14:28 -0700464 } else {
Steven Moreland055d8792018-11-14 12:48:42 -0800465 return "const " + aspect.cpp_name + "&";
Steven Morelande8a3a192018-09-20 14:14:28 -0700466 }
467 case StorageMode::OUT_ARGUMENT:
Steven Moreland055d8792018-11-14 12:48:42 -0800468 return aspect.cpp_name + "*";
Steven Morelande8a3a192018-09-20 14:14:28 -0700469 default:
470 AIDL_FATAL(aidl.GetName()) << "Unrecognized mode type: " << static_cast<int>(mode);
471 }
472}
473
Devin Moore83e76982020-09-30 09:32:35 -0700474size_t NdkAlignmentOf(const AidlTypenames& types, const AidlTypeSpecifier& aidl) {
475 // map from NDK type name to the corresponding alignment size
476 static map<string, int> alignment = {
477 {"bool", 1}, {"int8_t", 1}, {"char16_t", 2}, {"double", 8},
478 {"float", 4}, {"int32_t", 4}, {"int64_t", 8},
479 };
480
481 const string& name = NdkNameOf(types, aidl, StorageMode::STACK);
482 if (alignment.find(name) != alignment.end()) {
483 return alignment[name];
484 } else {
485 const auto& definedType = types.TryGetDefinedType(aidl.GetName());
486 AIDL_FATAL_IF(definedType == nullptr, aidl) << "Failed to resolve type.";
487 if (const auto& enumType = definedType->AsEnumDeclaration(); enumType != nullptr) {
488 return NdkAlignmentOf(types, enumType->GetBackingType());
489 }
490 }
491 return 0;
492}
493
Steven Morelande8a3a192018-09-20 14:14:28 -0700494void WriteToParcelFor(const CodeGeneratorContext& c) {
Steven Moreland055d8792018-11-14 12:48:42 -0800495 TypeInfo::Aspect aspect = GetTypeAspect(c.types, c.type);
496 aspect.write_func(c);
Steven Morelande8a3a192018-09-20 14:14:28 -0700497}
498
499void ReadFromParcelFor(const CodeGeneratorContext& c) {
Steven Moreland055d8792018-11-14 12:48:42 -0800500 TypeInfo::Aspect aspect = GetTypeAspect(c.types, c.type);
501 aspect.read_func(c);
Steven Morelande8a3a192018-09-20 14:14:28 -0700502}
503
Jiyong Park965c5b92018-11-21 13:37:15 +0900504std::string NdkArgList(
505 const AidlTypenames& types, const AidlMethod& method,
506 std::function<std::string(const std::string& type, const std::string& name, bool isOut)>
507 formatter) {
Steven Morelandaada3422018-09-20 15:55:33 -0700508 std::vector<std::string> method_arguments;
509 for (const auto& a : method.GetArguments()) {
510 StorageMode mode = a->IsOut() ? StorageMode::OUT_ARGUMENT : StorageMode::ARGUMENT;
Steven Moreland2bea13b2018-10-03 15:12:33 -0700511 std::string type = NdkNameOf(types, a->GetType(), mode);
Steven Morelandaada3422018-09-20 15:55:33 -0700512 std::string name = cpp::BuildVarName(*a);
Jiyong Park965c5b92018-11-21 13:37:15 +0900513 method_arguments.emplace_back(formatter(type, name, a->IsOut()));
Steven Morelandaada3422018-09-20 15:55:33 -0700514 }
515
516 if (method.GetType().GetName() != "void") {
Jiyong Park965c5b92018-11-21 13:37:15 +0900517 std::string type = NdkNameOf(types, method.GetType(), StorageMode::OUT_ARGUMENT);
518 std::string name = "_aidl_return";
519 method_arguments.emplace_back(formatter(type, name, true));
Steven Morelandaada3422018-09-20 15:55:33 -0700520 }
521
522 return Join(method_arguments, ", ");
523}
524
Steven Moreland2bea13b2018-10-03 15:12:33 -0700525std::string NdkMethodDecl(const AidlTypenames& types, const AidlMethod& method,
526 const std::string& clazz) {
Steven Morelandaada3422018-09-20 15:55:33 -0700527 std::string class_prefix = clazz.empty() ? "" : (clazz + "::");
Steven Moreland63404532018-10-08 14:31:00 -0700528 return "::ndk::ScopedAStatus " + class_prefix + method.GetName() + "(" +
Jiyong Park965c5b92018-11-21 13:37:15 +0900529 NdkArgList(types, method, FormatArgForDecl) + ")";
Steven Morelandaada3422018-09-20 15:55:33 -0700530}
531
Steven Morelande8a3a192018-09-20 14:14:28 -0700532} // namespace ndk
533} // namespace aidl
534} // namespace android