299 lines
11 KiB
C++
299 lines
11 KiB
C++
/*
|
|
* 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 <map>
|
|
#include <memory>
|
|
#include <sstream>
|
|
|
|
#include <android-base/stringprintf.h>
|
|
|
|
#include "aidl_to_java.h"
|
|
#include "code_writer.h"
|
|
#include "logging.h"
|
|
|
|
using std::unique_ptr;
|
|
using ::android::aidl::java::Variable;
|
|
using std::string;
|
|
|
|
namespace android {
|
|
namespace aidl {
|
|
namespace java {
|
|
|
|
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(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);
|
|
}
|
|
|
|
CHECK(false) << "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 = parcel->GetComments();
|
|
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 = generate_java_annotations(*parcel);
|
|
|
|
for (const auto& variable : parcel->GetFields()) {
|
|
std::ostringstream out;
|
|
out << variable->GetType().GetComments() << "\n";
|
|
for (const auto& a : generate_java_annotations(variable->GetType())) {
|
|
out << a << "\n";
|
|
}
|
|
out << "public " << JavaSignatureOf(variable->GetType(), typenames) << " "
|
|
<< variable->GetName();
|
|
if (variable->GetDefaultValue()) {
|
|
out << " = " << variable->ValueString(ConstantValueDecorator);
|
|
}
|
|
out << ";\n";
|
|
parcel_class->elements.push_back(std::make_shared<LiteralClassElement>(out.str()));
|
|
}
|
|
|
|
std::ostringstream out;
|
|
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";
|
|
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);
|
|
|
|
auto read_method = std::make_shared<Method>();
|
|
read_method->modifiers = PUBLIC | FINAL;
|
|
read_method->returnType = "void";
|
|
read_method->name = "readFromParcel";
|
|
read_method->parameters.push_back(parcel_variable);
|
|
read_method->statements = std::make_shared<StatementBlock>();
|
|
|
|
out.str("");
|
|
out << "int _aidl_start_pos = _aidl_parcel.dataPosition();\n"
|
|
<< "int _aidl_parcelable_size = _aidl_parcel.readInt();\n"
|
|
<< "if (_aidl_parcelable_size < 0) return;\n"
|
|
<< "try {\n";
|
|
|
|
read_method->statements->Add(std::make_shared<LiteralStatement>(out.str()));
|
|
|
|
out.str("");
|
|
out << " if (_aidl_parcel.dataPosition() - _aidl_start_pos >= _aidl_parcelable_size) return;\n";
|
|
|
|
std::shared_ptr<LiteralStatement> sizeCheck = nullptr;
|
|
// 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()) {
|
|
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_classloader_created = &is_classloader_created,
|
|
};
|
|
context.writer.Indent();
|
|
CreateFromParcelFor(context);
|
|
writer->Close();
|
|
read_method->statements->Add(std::make_shared<LiteralStatement>(code));
|
|
if (!sizeCheck) sizeCheck = std::make_shared<LiteralStatement>(out.str());
|
|
read_method->statements->Add(sizeCheck);
|
|
}
|
|
|
|
out.str("");
|
|
out << "} finally {\n"
|
|
<< " _aidl_parcel.setDataPosition(_aidl_start_pos + _aidl_parcelable_size);\n"
|
|
<< "}\n";
|
|
|
|
read_method->statements->Add(std::make_shared<LiteralStatement>(out.str()));
|
|
|
|
parcel_class->elements.push_back(read_method);
|
|
|
|
auto describe_contents_method = std::make_shared<Method>();
|
|
describe_contents_method->modifiers = PUBLIC | OVERRIDE;
|
|
describe_contents_method->returnType = "int";
|
|
describe_contents_method->name = "describeContents";
|
|
describe_contents_method->statements = std::make_shared<StatementBlock>();
|
|
describe_contents_method->statements->Add(std::make_shared<LiteralStatement>("return 0;\n"));
|
|
parcel_class->elements.push_back(describe_contents_method);
|
|
|
|
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->Write("%s\n", enum_decl->GetComments().c_str());
|
|
for (const std::string& annotation : generate_java_annotations(*enum_decl)) {
|
|
code_writer->Write("%s", annotation.c_str());
|
|
}
|
|
code_writer->Write("public @interface %s {\n", enum_decl->GetName().c_str());
|
|
code_writer->Indent();
|
|
for (const auto& enumerator : enum_decl->GetEnumerators()) {
|
|
code_writer->Write("%s", enumerator->GetComments().c_str());
|
|
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");
|
|
}
|
|
|
|
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 "(" + base::Join(parameters_decl, ", ") + ")";
|
|
}
|
|
|
|
std::vector<std::string> generate_java_annotations(const AidlAnnotatable& a) {
|
|
std::vector<std::string> result;
|
|
const AidlAnnotation* unsupported_app_usage = a.UnsupportedAppUsage();
|
|
if (a.IsHide()) {
|
|
result.emplace_back("@android.annotation.Hide");
|
|
}
|
|
if (unsupported_app_usage != nullptr) {
|
|
result.emplace_back("@android.compat.annotation.UnsupportedAppUsage" +
|
|
generate_java_unsupportedappusage_parameters(*unsupported_app_usage));
|
|
}
|
|
return result;
|
|
}
|
|
|
|
} // namespace java
|
|
} // namespace aidl
|
|
} // namespace android
|