You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1012 lines
36 KiB
1012 lines
36 KiB
/*
|
|
* Copyright (C) 2016, The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "generate_java.h"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <algorithm>
|
|
#include <map>
|
|
#include <memory>
|
|
#include <optional>
|
|
#include <sstream>
|
|
|
|
#include <android-base/format.h>
|
|
#include <android-base/stringprintf.h>
|
|
|
|
#include "aidl_to_java.h"
|
|
#include "code_writer.h"
|
|
#include "logging.h"
|
|
|
|
using ::android::base::EndsWith;
|
|
using ::android::base::Join;
|
|
using ::android::base::StartsWith;
|
|
using std::string;
|
|
using std::unique_ptr;
|
|
using std::vector;
|
|
|
|
namespace {
|
|
using android::aidl::java::CodeGeneratorContext;
|
|
using android::aidl::java::ConstantValueDecorator;
|
|
|
|
inline string getter_name(const AidlVariableDeclaration& variable) {
|
|
return "get" + variable.GetCapitalizedName();
|
|
}
|
|
inline string setter_name(const AidlVariableDeclaration& variable) {
|
|
return "set" + variable.GetCapitalizedName();
|
|
}
|
|
|
|
// clang-format off
|
|
const map<string, string> contents_describers {
|
|
{"FileDescriptor", R"(if (_v instanceof java.io.FileDescriptor) {
|
|
return android.os.Parcelable.CONTENTS_FILE_DESCRIPTOR;
|
|
})"},
|
|
{"Parcelable", R"(if (_v instanceof android.os.Parcelable) {
|
|
return ((android.os.Parcelable) _v).describeContents();
|
|
})"},
|
|
{"Map", R"(if (_v instanceof java.util.Map) {
|
|
return describeContents(((java.util.Map) _v).values());
|
|
})"},
|
|
{"List", R"(if (_v instanceof java.util.Collection) {
|
|
int _mask = 0;
|
|
for (Object o : (java.util.Collection) _v) {
|
|
_mask |= describeContents(o);
|
|
}
|
|
return _mask;
|
|
})"},
|
|
{"Array", R"(Class<?> _clazz = _v.getClass();
|
|
if (_clazz.isArray() && _clazz.getComponentType() == Object.class) {
|
|
int _mask = 0;
|
|
for (Object o : (Object[]) _v) {
|
|
_mask |= describeContents(o);
|
|
}
|
|
return _mask;
|
|
})"},
|
|
};
|
|
// clang-format on
|
|
|
|
void GenerateDescribeContentsHelper(CodeWriter& out, const set<string>& describers) {
|
|
out << "private int describeContents(Object _v) {\n";
|
|
out.Indent();
|
|
out << "if (_v == null) return 0;\n";
|
|
for (const auto& d : describers) {
|
|
out << contents_describers.at(d) << "\n";
|
|
}
|
|
out << "return 0;\n";
|
|
out.Dedent();
|
|
out << "}\n";
|
|
}
|
|
|
|
// Some types contribute to Parcelable.describeContents().
|
|
// e.g. FileDescriptor, Parcelables, List<Parcelables> ...
|
|
bool CanDescribeContents(const AidlTypeSpecifier& type, const AidlTypenames& types,
|
|
set<string>* describers) {
|
|
if (type.IsArray()) {
|
|
if (CanDescribeContents(type.ArrayBase(), types, describers)) {
|
|
describers->insert("Array");
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
if (type.GetName() == "List") {
|
|
if (CanDescribeContents(*type.GetTypeParameters()[0], types, describers)) {
|
|
describers->insert("List");
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
if (type.GetName() == "Map") {
|
|
if (CanDescribeContents(*type.GetTypeParameters()[1], types, describers)) {
|
|
describers->insert("Map"); // Map describer uses List describer
|
|
describers->insert("List");
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
if (type.GetName() == "FileDescriptor") {
|
|
describers->insert("FileDescriptor");
|
|
return true;
|
|
}
|
|
|
|
if (type.GetName() == "ParcelFileDescriptor" || type.GetName() == "ParcelableHolder" ||
|
|
types.GetParcelable(type) != nullptr) {
|
|
describers->insert("Parcelable");
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
void GenerateParcelableDescribeContents(CodeWriter& out, const AidlStructuredParcelable& decl,
|
|
const AidlTypenames& types) {
|
|
set<string> describers;
|
|
|
|
out << "@Override\n";
|
|
out << "public int describeContents() {\n";
|
|
out.Indent();
|
|
out << "int _mask = 0;\n";
|
|
for (const auto& f : decl.GetFields()) {
|
|
if (CanDescribeContents(f->GetType(), types, &describers)) {
|
|
out << "_mask |= describeContents(" << f->GetName() << ");\n";
|
|
}
|
|
}
|
|
out << "return _mask;\n";
|
|
out.Dedent();
|
|
out << "}\n";
|
|
if (!describers.empty()) {
|
|
GenerateDescribeContentsHelper(out, describers);
|
|
}
|
|
}
|
|
|
|
void GenerateParcelableDescribeContents(CodeWriter& out, const AidlUnionDecl& decl,
|
|
const AidlTypenames& types) {
|
|
set<string> describers;
|
|
|
|
out << "@Override\n";
|
|
out << "public int describeContents() {\n";
|
|
out.Indent();
|
|
out << "int _mask = 0;\n";
|
|
out << "switch (getTag()) {\n";
|
|
for (const auto& f : decl.GetFields()) {
|
|
if (CanDescribeContents(f->GetType(), types, &describers)) {
|
|
out << "case " << f->GetName() << ":\n";
|
|
out.Indent();
|
|
out << "_mask |= describeContents(" << getter_name(*f) << "());\n";
|
|
out << "break;\n";
|
|
out.Dedent();
|
|
}
|
|
}
|
|
out << "}\n";
|
|
out << "return _mask;\n";
|
|
out.Dedent();
|
|
out << "}\n";
|
|
if (!describers.empty()) {
|
|
GenerateDescribeContentsHelper(out, describers);
|
|
}
|
|
}
|
|
|
|
void GenerateToString(CodeWriter& out, const AidlStructuredParcelable& parcel,
|
|
const AidlTypenames& typenames) {
|
|
out << "@Override\n";
|
|
out << "public String toString() {\n";
|
|
out.Indent();
|
|
out << "java.util.StringJoiner _aidl_sj = new java.util.StringJoiner(";
|
|
out << "\", \", \"{\", \"}\");\n";
|
|
for (const auto& field : parcel.GetFields()) {
|
|
CodeGeneratorContext ctx{
|
|
.writer = out,
|
|
.typenames = typenames,
|
|
.type = field->GetType(),
|
|
.var = field->GetName(),
|
|
};
|
|
out << "_aidl_sj.add(\"" << field->GetName() << ": \" + (";
|
|
ToStringFor(ctx);
|
|
out << "));\n";
|
|
}
|
|
out << "return \"" << parcel.GetCanonicalName() << "\" + _aidl_sj.toString() ;\n";
|
|
out.Dedent();
|
|
out << "}\n";
|
|
}
|
|
|
|
void GenerateToString(CodeWriter& out, const AidlUnionDecl& parcel,
|
|
const AidlTypenames& typenames) {
|
|
out << "@Override\n";
|
|
out << "public String toString() {\n";
|
|
out.Indent();
|
|
out << "switch (_tag) {\n";
|
|
for (const auto& field : parcel.GetFields()) {
|
|
CodeGeneratorContext ctx{
|
|
.writer = out,
|
|
.typenames = typenames,
|
|
.type = field->GetType(),
|
|
.var = getter_name(*field) + "()",
|
|
};
|
|
out << "case " << field->GetName() << ": return \"" << parcel.GetCanonicalName() << "."
|
|
<< field->GetName() << "(\" + (";
|
|
ToStringFor(ctx);
|
|
out << ") + \")\";\n";
|
|
}
|
|
out << "}\n";
|
|
out << "throw new IllegalStateException(\"unknown field: \" + _tag);\n";
|
|
out.Dedent();
|
|
out << "}\n";
|
|
}
|
|
|
|
void GenerateEqualsAndHashCode(CodeWriter& out, const AidlStructuredParcelable& parcel,
|
|
const AidlTypenames&) {
|
|
out << "@Override\n";
|
|
out << "public boolean equals(Object other) {\n";
|
|
out.Indent();
|
|
out << "if (this == other) return true;\n";
|
|
out << "if (other == null) return false;\n";
|
|
out << "if (!(other instanceof " << parcel.GetName() << ")) return false;\n";
|
|
out << parcel.GetName() << " that = (" << parcel.GetName() << ")other;\n";
|
|
for (const auto& field : parcel.GetFields()) {
|
|
out << "if (!java.util.Objects.deepEquals(" << field->GetName() << ", that." << field->GetName()
|
|
<< ")) return false;\n";
|
|
}
|
|
out << "return true;\n";
|
|
out.Dedent();
|
|
out << "}\n";
|
|
out << "\n";
|
|
out << "@Override\n";
|
|
out << "public int hashCode() {\n";
|
|
out.Indent();
|
|
out << "return java.util.Arrays.deepHashCode(java.util.Arrays.asList(";
|
|
std::vector<std::string> names;
|
|
for (const auto& field : parcel.GetFields()) {
|
|
names.push_back(field->GetName());
|
|
}
|
|
out << android::base::Join(names, ", ") << ").toArray());\n";
|
|
out.Dedent();
|
|
out << "}\n";
|
|
}
|
|
|
|
void GenerateEqualsAndHashCode(CodeWriter& out, const AidlUnionDecl& decl,
|
|
const AidlTypenames&) {
|
|
out << "@Override\n";
|
|
out << "public boolean equals(Object other) {\n";
|
|
out.Indent();
|
|
out << "if (this == other) return true;\n";
|
|
out << "if (other == null) return false;\n";
|
|
out << "if (!(other instanceof " << decl.GetName() << ")) return false;\n";
|
|
out << decl.GetName() << " that = (" << decl.GetName() << ")other;\n";
|
|
out << "if (_tag != that._tag) return false;\n";
|
|
out << "if (!java.util.Objects.deepEquals(_value, that._value)) return false;\n";
|
|
out << "return true;\n";
|
|
out.Dedent();
|
|
out << "}\n";
|
|
out << "\n";
|
|
out << "@Override\n";
|
|
out << "public int hashCode() {\n";
|
|
out.Indent();
|
|
out << "return java.util.Arrays.deepHashCode(java.util.Arrays.asList(_tag, _value).toArray());\n";
|
|
out.Dedent();
|
|
out << "}\n";
|
|
out << "\n";
|
|
}
|
|
|
|
} // namespace
|
|
|
|
namespace android {
|
|
namespace aidl {
|
|
namespace java {
|
|
|
|
std::string GenerateComments(const AidlCommentable& node) {
|
|
std::string comments = FormatCommentsForJava(node.GetComments());
|
|
if (!comments.empty() && comments.back() != '\n') {
|
|
comments += '\n';
|
|
}
|
|
return comments;
|
|
}
|
|
|
|
std::string GenerateAnnotations(const AidlNode& node) {
|
|
std::string result;
|
|
for (const auto& a : JavaAnnotationsFor(node)) {
|
|
result += a + "\n";
|
|
}
|
|
return result;
|
|
}
|
|
|
|
bool generate_java_interface(const string& filename, const AidlInterface* iface,
|
|
const AidlTypenames& typenames, const IoDelegate& io_delegate,
|
|
const Options& options) {
|
|
auto cl = generate_binder_interface_class(iface, typenames, options);
|
|
|
|
std::unique_ptr<Document> document =
|
|
std::make_unique<Document>("" /* no comment */, iface->GetPackage(), std::move(cl));
|
|
|
|
CodeWriterPtr code_writer = io_delegate.GetCodeWriter(filename);
|
|
document->Write(code_writer.get());
|
|
|
|
return true;
|
|
}
|
|
|
|
bool generate_java_parcel(const std::string& filename, const AidlStructuredParcelable* parcel,
|
|
const AidlTypenames& typenames, const IoDelegate& io_delegate) {
|
|
auto cl = generate_parcel_class(parcel, typenames);
|
|
|
|
std::unique_ptr<Document> document =
|
|
std::make_unique<Document>("" /* no comment */, parcel->GetPackage(), std::move(cl));
|
|
|
|
CodeWriterPtr code_writer = io_delegate.GetCodeWriter(filename);
|
|
document->Write(code_writer.get());
|
|
|
|
return true;
|
|
}
|
|
|
|
bool generate_java_enum_declaration(const std::string& filename,
|
|
const AidlEnumDeclaration* enum_decl,
|
|
const AidlTypenames& typenames, const IoDelegate& io_delegate) {
|
|
CodeWriterPtr code_writer = io_delegate.GetCodeWriter(filename);
|
|
generate_enum(code_writer, enum_decl, typenames);
|
|
return true;
|
|
}
|
|
|
|
bool generate_java_union_declaration(const std::string& filename, const AidlUnionDecl* decl,
|
|
const AidlTypenames& typenames,
|
|
const IoDelegate& io_delegate) {
|
|
CodeWriterPtr code_writer = io_delegate.GetCodeWriter(filename);
|
|
generate_union(*code_writer, decl, typenames);
|
|
return true;
|
|
}
|
|
|
|
bool generate_java(const std::string& filename, const AidlDefinedType* defined_type,
|
|
const AidlTypenames& typenames, const IoDelegate& io_delegate,
|
|
const Options& options) {
|
|
if (const AidlStructuredParcelable* parcelable = defined_type->AsStructuredParcelable();
|
|
parcelable != nullptr) {
|
|
return generate_java_parcel(filename, parcelable, typenames, io_delegate);
|
|
}
|
|
|
|
if (const AidlEnumDeclaration* enum_decl = defined_type->AsEnumDeclaration();
|
|
enum_decl != nullptr) {
|
|
return generate_java_enum_declaration(filename, enum_decl, typenames, io_delegate);
|
|
}
|
|
|
|
if (const AidlInterface* interface = defined_type->AsInterface(); interface != nullptr) {
|
|
return generate_java_interface(filename, interface, typenames, io_delegate, options);
|
|
}
|
|
|
|
if (const AidlUnionDecl* union_decl = defined_type->AsUnionDeclaration(); union_decl != nullptr) {
|
|
return generate_java_union_declaration(filename, union_decl, typenames, io_delegate);
|
|
}
|
|
|
|
AIDL_FATAL(defined_type) << "Unrecognized type sent for Java generation.";
|
|
return false;
|
|
}
|
|
|
|
std::unique_ptr<android::aidl::java::Class> generate_parcel_class(
|
|
const AidlStructuredParcelable* parcel, const AidlTypenames& typenames) {
|
|
auto parcel_class = std::make_unique<Class>();
|
|
parcel_class->comment = GenerateComments(*parcel);
|
|
parcel_class->modifiers = PUBLIC;
|
|
parcel_class->what = Class::CLASS;
|
|
parcel_class->type = parcel->GetCanonicalName();
|
|
parcel_class->interfaces.push_back("android.os.Parcelable");
|
|
parcel_class->annotations = JavaAnnotationsFor(*parcel);
|
|
|
|
if (parcel->IsGeneric()) {
|
|
parcel_class->type += "<" + base::Join(parcel->GetTypeParameters(), ",") + ">";
|
|
}
|
|
|
|
for (const auto& variable : parcel->GetFields()) {
|
|
std::ostringstream out;
|
|
out << GenerateComments(*variable);
|
|
out << GenerateAnnotations(*variable);
|
|
out << "public ";
|
|
|
|
if (variable->GetType().GetName() == "ParcelableHolder" || parcel->IsJavaOnlyImmutable()) {
|
|
out << "final ";
|
|
}
|
|
out << JavaSignatureOf(variable->GetType(), typenames) << " " << variable->GetName();
|
|
if (!parcel->IsJavaOnlyImmutable() && variable->GetDefaultValue()) {
|
|
out << " = " << variable->ValueString(ConstantValueDecorator);
|
|
} else if (variable->GetType().GetName() == "ParcelableHolder") {
|
|
out << std::boolalpha;
|
|
out << " = new " << JavaSignatureOf(variable->GetType(), typenames) << "(";
|
|
if (parcel->IsVintfStability()) {
|
|
out << "android.os.Parcelable.PARCELABLE_STABILITY_VINTF";
|
|
} else {
|
|
out << "android.os.Parcelable.PARCELABLE_STABILITY_LOCAL";
|
|
}
|
|
out << ")";
|
|
out << std::noboolalpha;
|
|
}
|
|
out << ";\n";
|
|
parcel_class->elements.push_back(std::make_shared<LiteralClassElement>(out.str()));
|
|
}
|
|
|
|
std::ostringstream out;
|
|
if (parcel->IsJavaOnlyImmutable()) {
|
|
auto builder_class = std::make_shared<Class>();
|
|
builder_class->modifiers = PUBLIC | FINAL | STATIC;
|
|
builder_class->what = Class::CLASS;
|
|
builder_class->type = "Builder";
|
|
|
|
out.str("");
|
|
for (const auto& variable : parcel->GetFields()) {
|
|
out << "private " << JavaSignatureOf(variable->GetType(), typenames) << " "
|
|
<< variable->GetName();
|
|
if (variable->GetDefaultValue()) {
|
|
out << " = " << variable->ValueString(ConstantValueDecorator);
|
|
}
|
|
out << ";\n";
|
|
out << "public Builder " << setter_name(*variable) << "("
|
|
<< JavaSignatureOf(variable->GetType(), typenames) << " " << variable->GetName()
|
|
<< ") {\n"
|
|
<< " "
|
|
<< "this." << variable->GetName() << " = " << variable->GetName() << ";\n"
|
|
<< " return this;\n"
|
|
<< "}\n";
|
|
}
|
|
out << "public " << parcel->GetCanonicalName() << " build() {\n"
|
|
<< " return new " << parcel->GetCanonicalName() << "(";
|
|
std::vector<std::string> variables;
|
|
std::transform(parcel->GetFields().begin(), parcel->GetFields().end(),
|
|
std::back_inserter(variables), [](const auto& f) { return f->GetName(); });
|
|
out << base::Join(variables, ", ") << ");\n"
|
|
<< "}\n";
|
|
builder_class->elements.push_back(std::make_shared<LiteralClassElement>(out.str()));
|
|
parcel_class->elements.push_back(builder_class);
|
|
}
|
|
if (parcel->IsVintfStability()) {
|
|
parcel_class->elements.push_back(std::make_shared<LiteralClassElement>(
|
|
"@Override\n public final int getStability() { return "
|
|
"android.os.Parcelable.PARCELABLE_STABILITY_VINTF; }\n"));
|
|
}
|
|
|
|
out.str("");
|
|
out << "public static final android.os.Parcelable.Creator<" << parcel->GetName() << "> CREATOR = "
|
|
<< "new android.os.Parcelable.Creator<" << parcel->GetName() << ">() {\n";
|
|
out << " @Override\n";
|
|
out << " public " << parcel->GetName()
|
|
<< " createFromParcel(android.os.Parcel _aidl_source) {\n";
|
|
if (parcel->IsJavaOnlyImmutable()) {
|
|
out << " return internalCreateFromParcel(_aidl_source);\n";
|
|
} else {
|
|
out << " " << parcel->GetName() << " _aidl_out = new " << parcel->GetName() << "();\n";
|
|
out << " _aidl_out.readFromParcel(_aidl_source);\n";
|
|
out << " return _aidl_out;\n";
|
|
}
|
|
out << " }\n";
|
|
out << " @Override\n";
|
|
out << " public " << parcel->GetName() << "[] newArray(int _aidl_size) {\n";
|
|
out << " return new " << parcel->GetName() << "[_aidl_size];\n";
|
|
out << " }\n";
|
|
out << "};\n";
|
|
parcel_class->elements.push_back(std::make_shared<LiteralClassElement>(out.str()));
|
|
|
|
auto flag_variable = std::make_shared<Variable>("int", "_aidl_flag");
|
|
auto parcel_variable = std::make_shared<Variable>("android.os.Parcel", "_aidl_parcel");
|
|
|
|
auto write_method = std::make_shared<Method>();
|
|
write_method->modifiers = PUBLIC | OVERRIDE | FINAL;
|
|
write_method->returnType = "void";
|
|
write_method->name = "writeToParcel";
|
|
write_method->parameters.push_back(parcel_variable);
|
|
write_method->parameters.push_back(flag_variable);
|
|
write_method->statements = std::make_shared<StatementBlock>();
|
|
|
|
out.str("");
|
|
out << "int _aidl_start_pos = _aidl_parcel.dataPosition();\n"
|
|
<< "_aidl_parcel.writeInt(0);\n";
|
|
write_method->statements->Add(std::make_shared<LiteralStatement>(out.str()));
|
|
|
|
for (const auto& field : parcel->GetFields()) {
|
|
string code;
|
|
CodeWriterPtr writer = CodeWriter::ForString(&code);
|
|
CodeGeneratorContext context{
|
|
.writer = *(writer.get()),
|
|
.typenames = typenames,
|
|
.type = field->GetType(),
|
|
.parcel = parcel_variable->name,
|
|
.var = field->GetName(),
|
|
.is_return_value = false,
|
|
};
|
|
WriteToParcelFor(context);
|
|
writer->Close();
|
|
write_method->statements->Add(std::make_shared<LiteralStatement>(code));
|
|
}
|
|
|
|
out.str("");
|
|
out << "int _aidl_end_pos = _aidl_parcel.dataPosition();\n"
|
|
<< "_aidl_parcel.setDataPosition(_aidl_start_pos);\n"
|
|
<< "_aidl_parcel.writeInt(_aidl_end_pos - _aidl_start_pos);\n"
|
|
<< "_aidl_parcel.setDataPosition(_aidl_end_pos);\n";
|
|
|
|
write_method->statements->Add(std::make_shared<LiteralStatement>(out.str()));
|
|
|
|
parcel_class->elements.push_back(write_method);
|
|
|
|
if (parcel->IsJavaOnlyImmutable()) {
|
|
auto constructor = std::make_shared<Method>();
|
|
constructor->modifiers = PUBLIC;
|
|
constructor->name = parcel->GetName();
|
|
constructor->statements = std::make_shared<StatementBlock>();
|
|
for (const auto& field : parcel->GetFields()) {
|
|
constructor->parameters.push_back(std::make_shared<Variable>(
|
|
JavaSignatureOf(field->GetType(), typenames), field->GetName()));
|
|
out.str("");
|
|
|
|
out << "this." << field->GetName() << " = ";
|
|
if (field->GetType().GetName() == "List") {
|
|
out << field->GetName() << " == null ? null : java.util.Collections.unmodifiableList("
|
|
<< field->GetName() << ");\n";
|
|
} else if (field->GetType().GetName() == "Map") {
|
|
out << field->GetName() << " == null ? null : java.util.Collections.unmodifiableMap("
|
|
<< field->GetName() << ");\n";
|
|
} else {
|
|
out << field->GetName() << ";\n";
|
|
}
|
|
constructor->statements->Add(std::make_shared<LiteralStatement>(out.str()));
|
|
}
|
|
parcel_class->elements.push_back(constructor);
|
|
}
|
|
|
|
// For an immutable parcelable, generate internalCreateFromParcel method.
|
|
// Otherwise, generate readFromParcel method.
|
|
auto read_or_create_method = std::make_shared<Method>();
|
|
if (parcel->IsJavaOnlyImmutable()) {
|
|
auto constructor = std::make_shared<Method>();
|
|
read_or_create_method->modifiers = PRIVATE | STATIC;
|
|
read_or_create_method->returnType = parcel->GetName();
|
|
read_or_create_method->name = "internalCreateFromParcel";
|
|
read_or_create_method->parameters.push_back(parcel_variable);
|
|
read_or_create_method->statements = std::make_shared<StatementBlock>();
|
|
} else {
|
|
read_or_create_method->modifiers = PUBLIC | FINAL;
|
|
read_or_create_method->returnType = "void";
|
|
read_or_create_method->name = "readFromParcel";
|
|
read_or_create_method->parameters.push_back(parcel_variable);
|
|
read_or_create_method->statements = std::make_shared<StatementBlock>();
|
|
}
|
|
out.str("");
|
|
const string builder_variable = "_aidl_parcelable_builder";
|
|
if (parcel->IsJavaOnlyImmutable()) {
|
|
out << "Builder " << builder_variable << " = new Builder();\n";
|
|
}
|
|
out << "int _aidl_start_pos = _aidl_parcel.dataPosition();\n"
|
|
<< "int _aidl_parcelable_size = _aidl_parcel.readInt();\n"
|
|
<< "try {\n"
|
|
<< " if (_aidl_parcelable_size < 0) return";
|
|
if (parcel->IsJavaOnlyImmutable()) {
|
|
out << " " << builder_variable << ".build()";
|
|
}
|
|
out << ";\n";
|
|
|
|
read_or_create_method->statements->Add(std::make_shared<LiteralStatement>(out.str()));
|
|
|
|
out.str("");
|
|
out << " if (_aidl_parcel.dataPosition() - _aidl_start_pos >= _aidl_parcelable_size) return";
|
|
if (parcel->IsJavaOnlyImmutable()) {
|
|
out << " " << builder_variable << ".build()";
|
|
}
|
|
out << ";\n";
|
|
|
|
std::shared_ptr<LiteralStatement> sizeCheck = std::make_shared<LiteralStatement>(out.str());
|
|
// keep this across different fields in order to create the classloader
|
|
// at most once.
|
|
bool is_classloader_created = false;
|
|
for (const auto& field : parcel->GetFields()) {
|
|
read_or_create_method->statements->Add(sizeCheck);
|
|
const auto field_variable_name =
|
|
(parcel->IsJavaOnlyImmutable() ? "_aidl_temp_" : "") + field->GetName();
|
|
string code;
|
|
CodeWriterPtr writer = CodeWriter::ForString(&code);
|
|
CodeGeneratorContext context{
|
|
.writer = *(writer.get()),
|
|
.typenames = typenames,
|
|
.type = field->GetType(),
|
|
.parcel = parcel_variable->name,
|
|
.var = field_variable_name,
|
|
.is_classloader_created = &is_classloader_created,
|
|
};
|
|
context.writer.Indent();
|
|
if (parcel->IsJavaOnlyImmutable()) {
|
|
context.writer.Write("%s %s;\n", JavaSignatureOf(field->GetType(), typenames).c_str(),
|
|
field_variable_name.c_str());
|
|
}
|
|
CreateFromParcelFor(context);
|
|
if (parcel->IsJavaOnlyImmutable()) {
|
|
context.writer.Write("%s.%s(%s);\n", builder_variable.c_str(), setter_name(*field).c_str(),
|
|
field_variable_name.c_str());
|
|
}
|
|
writer->Close();
|
|
read_or_create_method->statements->Add(std::make_shared<LiteralStatement>(code));
|
|
}
|
|
|
|
out.str("");
|
|
out << "} finally {\n"
|
|
<< " if (_aidl_start_pos > (Integer.MAX_VALUE - _aidl_parcelable_size)) {\n"
|
|
<< " throw new android.os.BadParcelableException(\"Overflow in the size of "
|
|
"parcelable\");\n"
|
|
<< " }\n"
|
|
<< " _aidl_parcel.setDataPosition(_aidl_start_pos + _aidl_parcelable_size);\n";
|
|
if (parcel->IsJavaOnlyImmutable()) {
|
|
out << " return " << builder_variable << ".build();\n";
|
|
}
|
|
out << "}\n";
|
|
|
|
read_or_create_method->statements->Add(std::make_shared<LiteralStatement>(out.str()));
|
|
|
|
parcel_class->elements.push_back(read_or_create_method);
|
|
|
|
string constants;
|
|
generate_constant_declarations(*CodeWriter::ForString(&constants), *parcel);
|
|
parcel_class->elements.push_back(std::make_shared<LiteralClassElement>(constants));
|
|
|
|
if (parcel->JavaDerive("toString")) {
|
|
string to_string;
|
|
GenerateToString(*CodeWriter::ForString(&to_string), *parcel, typenames);
|
|
parcel_class->elements.push_back(std::make_shared<LiteralClassElement>(to_string));
|
|
}
|
|
|
|
if (parcel->JavaDerive("equals")) {
|
|
string to_string;
|
|
GenerateEqualsAndHashCode(*CodeWriter::ForString(&to_string), *parcel, typenames);
|
|
parcel_class->elements.push_back(std::make_shared<LiteralClassElement>(to_string));
|
|
}
|
|
|
|
string describe_contents;
|
|
GenerateParcelableDescribeContents(*CodeWriter::ForString(&describe_contents), *parcel,
|
|
typenames);
|
|
parcel_class->elements.push_back(std::make_shared<LiteralClassElement>(describe_contents));
|
|
|
|
return parcel_class;
|
|
}
|
|
|
|
void generate_enum(const CodeWriterPtr& code_writer, const AidlEnumDeclaration* enum_decl,
|
|
const AidlTypenames& typenames) {
|
|
code_writer->Write(
|
|
"/*\n"
|
|
" * This file is auto-generated. DO NOT MODIFY.\n"
|
|
" */\n");
|
|
|
|
code_writer->Write("package %s;\n", enum_decl->GetPackage().c_str());
|
|
(*code_writer) << GenerateComments(*enum_decl);
|
|
(*code_writer) << GenerateAnnotations(*enum_decl);
|
|
code_writer->Write("public @interface %s {\n", enum_decl->GetName().c_str());
|
|
code_writer->Indent();
|
|
for (const auto& enumerator : enum_decl->GetEnumerators()) {
|
|
(*code_writer) << GenerateComments(*enumerator);
|
|
(*code_writer) << GenerateAnnotations(*enumerator);
|
|
code_writer->Write(
|
|
"public static final %s %s = %s;\n",
|
|
JavaSignatureOf(enum_decl->GetBackingType(), typenames).c_str(),
|
|
enumerator->GetName().c_str(),
|
|
enumerator->ValueString(enum_decl->GetBackingType(), ConstantValueDecorator).c_str());
|
|
}
|
|
code_writer->Dedent();
|
|
code_writer->Write("}\n");
|
|
}
|
|
|
|
void generate_union(CodeWriter& out, const AidlUnionDecl* decl, const AidlTypenames& typenames) {
|
|
const string tag_type = "int";
|
|
const AidlTypeSpecifier tag_type_specifier(AIDL_LOCATION_HERE, tag_type, false /* isArray */,
|
|
nullptr /* type_params */, Comments{});
|
|
const string clazz = decl->GetName();
|
|
|
|
out << "/*\n";
|
|
out << " * This file is auto-generated. DO NOT MODIFY.\n";
|
|
out << " */\n";
|
|
|
|
out << "package " + decl->GetPackage() + ";\n";
|
|
out << "\n";
|
|
out << GenerateComments(*decl);
|
|
out << GenerateAnnotations(*decl);
|
|
|
|
out << "public final class " + clazz + " implements android.os.Parcelable {\n";
|
|
out.Indent();
|
|
|
|
size_t tag_index = 0;
|
|
out << "// tags for union fields\n";
|
|
for (const auto& variable : decl->GetFields()) {
|
|
auto signature = variable->Signature() + ";";
|
|
out << "public final static " + tag_type + " " + variable->GetName() + " = " +
|
|
std::to_string(tag_index++) + "; // " + signature + "\n";
|
|
}
|
|
out << "\n";
|
|
|
|
const auto final_opt = decl->IsJavaOnlyImmutable() ? "final " : "";
|
|
out << "private " << final_opt << tag_type + " _tag;\n";
|
|
out << "private " << final_opt << "Object _value;\n";
|
|
out << "\n";
|
|
|
|
AIDL_FATAL_IF(decl->GetFields().empty(), *decl) << "Union '" << clazz << "' is empty.";
|
|
const auto& first_field = decl->GetFields()[0];
|
|
const auto& first_type = JavaSignatureOf(first_field->GetType(), typenames);
|
|
const auto& first_value = first_field->ValueString(ConstantValueDecorator);
|
|
|
|
// default ctor() inits with first member's default value
|
|
out << "public " + clazz + "() {\n";
|
|
out.Indent();
|
|
out << first_type + " _value = " << (first_value.empty() ? "null" : first_value) << ";\n";
|
|
out << "this._tag = " << first_field->GetName() << ";\n";
|
|
out << "this._value = _value;\n";
|
|
out.Dedent();
|
|
out << "}\n\n";
|
|
|
|
if (!decl->IsJavaOnlyImmutable()) {
|
|
// private ctor(Parcel)
|
|
out << "private " + clazz + "(android.os.Parcel _aidl_parcel) {\n";
|
|
out << " readFromParcel(_aidl_parcel);\n";
|
|
out << "}\n\n";
|
|
}
|
|
|
|
// private ctor(tag, value)
|
|
out << "private " + clazz + "(" + tag_type + " _tag, Object _value) {\n";
|
|
out.Indent();
|
|
out << "this._tag = _tag;\n";
|
|
out << "this._value = _value;\n";
|
|
out.Dedent();
|
|
out << "}\n\n";
|
|
|
|
// getTag()
|
|
out << "public " + tag_type + " " + "getTag() {\n";
|
|
out.Indent();
|
|
out << "return _tag;\n";
|
|
out.Dedent();
|
|
out << "}\n\n";
|
|
|
|
// value ctor, getter, setter(for mutable) for each field
|
|
for (const auto& variable : decl->GetFields()) {
|
|
out << "// " + variable->Signature() + ";\n\n";
|
|
|
|
auto var_name = variable->GetName();
|
|
auto var_type = JavaSignatureOf(variable->GetType(), typenames);
|
|
|
|
// value ctor
|
|
out << GenerateComments(*variable);
|
|
out << GenerateAnnotations(*variable);
|
|
out << "public static " + clazz + " " + var_name + "(" + var_type + " _value) {\n";
|
|
out.Indent();
|
|
out << "return new " + clazz + "(" + var_name + ", _value);\n";
|
|
out.Dedent();
|
|
out << "}\n\n";
|
|
|
|
// getter
|
|
if (variable->GetType().IsGeneric()) {
|
|
out << "@SuppressWarnings(\"unchecked\")\n";
|
|
}
|
|
out << "public " + var_type + " " + getter_name(*variable) + "() {\n";
|
|
out.Indent();
|
|
out << "_assertTag(" + var_name + ");\n";
|
|
out << "return (" + var_type + ") _value;\n";
|
|
out.Dedent();
|
|
out << "}\n\n";
|
|
|
|
// setter
|
|
if (!decl->IsJavaOnlyImmutable()) {
|
|
out << "public void " + setter_name(*variable) + "(" + var_type + " _value) {\n";
|
|
out.Indent();
|
|
out << "_set(" + var_name + ", _value);\n";
|
|
out.Dedent();
|
|
out << "}\n\n";
|
|
}
|
|
}
|
|
|
|
if (decl->IsVintfStability()) {
|
|
out << "@Override\n";
|
|
out << "public final int getStability() {\n";
|
|
out << " return android.os.Parcelable.PARCELABLE_STABILITY_VINTF;\n";
|
|
out << "}\n\n";
|
|
}
|
|
|
|
out << "public static final android.os.Parcelable.Creator<" << clazz << "> CREATOR = "
|
|
<< "new android.os.Parcelable.Creator<" << clazz << ">() {\n";
|
|
out << " @Override\n";
|
|
out << " public " << clazz << " createFromParcel(android.os.Parcel _aidl_source) {\n";
|
|
if (decl->IsJavaOnlyImmutable()) {
|
|
out << " return internalCreateFromParcel(_aidl_source);\n";
|
|
} else {
|
|
out << " return new " + clazz + "(_aidl_source);\n";
|
|
}
|
|
out << " }\n";
|
|
out << " @Override\n";
|
|
out << " public " << clazz << "[] newArray(int _aidl_size) {\n";
|
|
out << " return new " << clazz << "[_aidl_size];\n";
|
|
out << " }\n";
|
|
out << "};\n\n";
|
|
|
|
auto write_to_parcel = [&](const AidlTypeSpecifier& type, std::string name, std::string parcel) {
|
|
string code;
|
|
CodeWriterPtr writer = CodeWriter::ForString(&code);
|
|
CodeGeneratorContext context{
|
|
.writer = *(writer.get()),
|
|
.typenames = typenames,
|
|
.type = type,
|
|
.parcel = parcel,
|
|
.var = name,
|
|
.is_return_value = false,
|
|
};
|
|
WriteToParcelFor(context);
|
|
writer->Close();
|
|
return code;
|
|
};
|
|
|
|
out << "@Override\n";
|
|
out << "public final void writeToParcel(android.os.Parcel _aidl_parcel, int _aidl_flag) {\n";
|
|
out.Indent();
|
|
out << write_to_parcel(tag_type_specifier, "_tag", "_aidl_parcel");
|
|
out << "switch (_tag) {\n";
|
|
for (const auto& variable : decl->GetFields()) {
|
|
out << "case " + variable->GetName() + ":\n";
|
|
out.Indent();
|
|
out << write_to_parcel(variable->GetType(), getter_name(*variable) + "()", "_aidl_parcel");
|
|
out << "break;\n";
|
|
out.Dedent();
|
|
}
|
|
out << "}\n";
|
|
out.Dedent();
|
|
out << "}\n\n";
|
|
|
|
// keep this across different fields in order to create the classloader
|
|
// at most once.
|
|
bool is_classloader_created = false;
|
|
auto read_from_parcel = [&](const AidlTypeSpecifier& type, std::string name, std::string parcel) {
|
|
string code;
|
|
CodeWriterPtr writer = CodeWriter::ForString(&code);
|
|
CodeGeneratorContext context{
|
|
.writer = *(writer.get()),
|
|
.typenames = typenames,
|
|
.type = type,
|
|
.parcel = parcel,
|
|
.var = name,
|
|
.is_classloader_created = &is_classloader_created,
|
|
};
|
|
CreateFromParcelFor(context);
|
|
writer->Close();
|
|
return code;
|
|
};
|
|
|
|
if (decl->IsJavaOnlyImmutable()) {
|
|
// When it's immutable we don't need readFromParcel, but we can use it from createFromParcel
|
|
out << "private static " + clazz +
|
|
" internalCreateFromParcel(android.os.Parcel _aidl_parcel) {\n";
|
|
} else {
|
|
// Not override, but as a user-defined parcelable, this method should be public
|
|
out << "public void readFromParcel(android.os.Parcel _aidl_parcel) {\n";
|
|
}
|
|
out.Indent();
|
|
out << tag_type + " _aidl_tag;\n";
|
|
out << read_from_parcel(tag_type_specifier, "_aidl_tag", "_aidl_parcel");
|
|
out << "switch (_aidl_tag) {\n";
|
|
for (const auto& variable : decl->GetFields()) {
|
|
auto var_name = variable->GetName();
|
|
auto var_type = JavaSignatureOf(variable->GetType(), typenames);
|
|
out << "case " + var_name + ": {\n";
|
|
out.Indent();
|
|
out << var_type + " _aidl_value;\n";
|
|
out << read_from_parcel(variable->GetType(), "_aidl_value", "_aidl_parcel");
|
|
if (decl->IsJavaOnlyImmutable()) {
|
|
out << "return new " << clazz << "(_aidl_tag, _aidl_value); }\n";
|
|
} else {
|
|
out << "_set(_aidl_tag, _aidl_value);\n";
|
|
out << "return; }\n";
|
|
}
|
|
out.Dedent();
|
|
}
|
|
out << "}\n";
|
|
out << "throw new IllegalArgumentException(\"union: unknown tag: \" + _aidl_tag);\n";
|
|
out.Dedent();
|
|
out << "}\n\n";
|
|
|
|
generate_constant_declarations(out, *decl);
|
|
|
|
GenerateParcelableDescribeContents(out, *decl, typenames);
|
|
out << "\n";
|
|
if (decl->JavaDerive("toString")) {
|
|
GenerateToString(out, *decl, typenames);
|
|
}
|
|
|
|
if (decl->JavaDerive("equals")) {
|
|
GenerateEqualsAndHashCode(out, *decl, typenames);
|
|
}
|
|
|
|
// helper: _assertTag
|
|
out << "private void _assertTag(" + tag_type + " tag) {\n";
|
|
out << " if (getTag() != tag) {\n";
|
|
out << " throw new IllegalStateException(\"bad access: \" + _tagString(tag) + \", \" + "
|
|
"_tagString(getTag()) + \" is available.\");\n";
|
|
out << " }\n";
|
|
out << "}\n\n";
|
|
|
|
// helper: _tagString
|
|
out << "private String _tagString(" + tag_type + " _tag) {\n";
|
|
out << " switch (_tag) {\n";
|
|
for (const auto& variable : decl->GetFields()) {
|
|
auto var_name = variable->GetName();
|
|
out << " case " + var_name + ": return \"" + var_name + "\";\n";
|
|
}
|
|
out << " }\n";
|
|
out << " throw new IllegalStateException(\"unknown field: \" + _tag);\n";
|
|
out << "}\n";
|
|
|
|
if (!decl->IsJavaOnlyImmutable()) {
|
|
out << "\n";
|
|
out << "private void _set(int _tag, Object _value) {\n";
|
|
out.Indent();
|
|
out << "this._tag = _tag;\n";
|
|
out << "this._value = _value;\n";
|
|
out.Dedent();
|
|
out << "}\n";
|
|
}
|
|
|
|
out.Dedent();
|
|
out << "}\n";
|
|
}
|
|
|
|
std::string dump_location(const AidlNode& method) {
|
|
return method.PrintLocation();
|
|
}
|
|
|
|
std::string generate_java_unsupportedappusage_parameters(const AidlAnnotation& a) {
|
|
const std::map<std::string, std::string> params = a.AnnotationParams(ConstantValueDecorator);
|
|
std::vector<string> parameters_decl;
|
|
for (const auto& name_and_param : params) {
|
|
const std::string& param_name = name_and_param.first;
|
|
const std::string& param_value = name_and_param.second;
|
|
parameters_decl.push_back(param_name + " = " + param_value);
|
|
}
|
|
parameters_decl.push_back("overrideSourcePosition=\"" + dump_location(a) + "\"");
|
|
return "(" + Join(parameters_decl, ", ") + ")";
|
|
}
|
|
|
|
std::vector<std::string> generate_java_annotations(const AidlAnnotatable& a) {
|
|
std::vector<std::string> result;
|
|
if (a.IsHide()) {
|
|
result.emplace_back("@android.annotation.Hide");
|
|
}
|
|
|
|
const AidlAnnotation* unsupported_app_usage = a.UnsupportedAppUsage();
|
|
if (unsupported_app_usage != nullptr) {
|
|
result.emplace_back("@android.compat.annotation.UnsupportedAppUsage" +
|
|
generate_java_unsupportedappusage_parameters(*unsupported_app_usage));
|
|
}
|
|
|
|
for (const auto& annotation : a.GetAnnotations()) {
|
|
if (annotation.GetType() == AidlAnnotation::Type::JAVA_PASSTHROUGH) {
|
|
result.emplace_back(annotation.ParamValue<std::string>("annotation").value());
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
struct JavaAnnotationsVisitor : AidlVisitor {
|
|
JavaAnnotationsVisitor(std::vector<std::string>& result) : result(result) {}
|
|
void Visit(const AidlTypeSpecifier& t) override { result = generate_java_annotations(t); }
|
|
void Visit(const AidlInterface& t) override { ForDefinedType(t); }
|
|
void Visit(const AidlParcelable& t) override { ForDefinedType(t); }
|
|
void Visit(const AidlStructuredParcelable& t) override { ForDefinedType(t); }
|
|
void Visit(const AidlUnionDecl& t) override { ForDefinedType(t); }
|
|
void Visit(const AidlEnumDeclaration& t) override { ForDefinedType(t); }
|
|
void Visit(const AidlMethod& m) override { ForMember(m); }
|
|
void Visit(const AidlConstantDeclaration& c) override { ForMember(c); }
|
|
void Visit(const AidlVariableDeclaration& v) override { ForMember(v); }
|
|
std::vector<std::string>& result;
|
|
|
|
void ForDefinedType(const AidlDefinedType& t) {
|
|
result = generate_java_annotations(t);
|
|
if (t.IsDeprecated()) {
|
|
result.push_back("@Deprecated");
|
|
}
|
|
}
|
|
template <typename Member>
|
|
void ForMember(const Member& t) {
|
|
result = generate_java_annotations(t.GetType());
|
|
if (t.IsDeprecated()) {
|
|
result.push_back("@Deprecated");
|
|
}
|
|
}
|
|
};
|
|
|
|
std::vector<std::string> JavaAnnotationsFor(const AidlNode& a) {
|
|
std::vector<std::string> result;
|
|
JavaAnnotationsVisitor visitor{result};
|
|
a.DispatchVisit(visitor);
|
|
return result;
|
|
}
|
|
|
|
} // namespace java
|
|
} // namespace aidl
|
|
} // namespace android
|