/*
* Copyright (C) 2017 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 "AST.h"
#include "Coordinator.h"
#include "EnumType.h"
#include "HidlTypeAssertion.h"
#include "Interface.h"
#include "Method.h"
#include "Reference.h"
#include "ScalarType.h"
#include "Scope.h"
#include <android-base/logging.h>
#include <hidl-util/Formatter.h>
#include <hidl-util/StringHelper.h>
#include <algorithm>
#include <string>
#include <vector>
namespace android {
void AST::generateCppAdapterHeader(Formatter& out) const {
const std::string klassName = AST::isInterface() ? getInterface()->getAdapterName() : "Atypes";
const std::string guard = makeHeaderGuard(klassName, true /* indicateGenerated */);
out << "#ifndef " << guard << "\n";
out << "#define " << guard << "\n\n";
if (AST::isInterface()) {
generateCppPackageInclude(out, mPackage, getInterface()->localName());
enterLeaveNamespace(out, true /* enter */);
out.endl();
const std::string mockName = getInterface()->fqName().cppName();
out << "class " << klassName << " : public " << mockName << " ";
out.block([&] {
out << "public:\n";
out << "typedef " << mockName << " Pure;\n";
out << klassName << "(::android::sp<" << mockName << "> impl);\n";
generateMethods(out, [&](const Method* method, const Interface* /* interface */) {
if (method->isHidlReserved()) {
return;
}
out << "virtual ";
method->generateCppSignature(out);
out << " override;\n";
});
out << "private:\n";
out << "::android::sp<" << mockName << "> mImpl;\n";
}) << ";\n\n";
enterLeaveNamespace(out, false /* enter */);
} else {
out << "// no adapters for types.hal\n";
}
out << "#endif // " << guard << "\n";
}
void AST::generateCppAdapterSource(Formatter& out) const {
const std::string klassName = AST::isInterface() ? getInterface()->getAdapterName() : "Atypes";
generateCppPackageInclude(out, mPackage, klassName);
if (AST::isInterface()) {
out << "#include <hidladapter/HidlBinderAdapter.h>\n";
generateCppPackageInclude(out, mPackage, getInterface()->localName());
std::set<FQName> allImportedNames;
getAllImportedNames(&allImportedNames);
for (const auto& item : allImportedNames) {
if (item.name() == "types") {
continue;
}
generateCppPackageInclude(out, item, item.getInterfaceAdapterName());
}
out.endl();
enterLeaveNamespace(out, true /* enter */);
out.endl();
const std::string mockName = getInterface()->fqName().cppName();
out << klassName << "::" << klassName << "(::android::sp<" << mockName
<< "> impl) : mImpl(impl) {}";
generateMethods(out, [&](const Method* method, const Interface* /* interface */) {
generateAdapterMethod(out, method);
});
enterLeaveNamespace(out, false /* enter */);
out.endl();
} else {
out << "// no adapters for types.hal\n";
}
}
void AST::generateAdapterMethod(Formatter& out, const Method* method) const {
if (method->isHidlReserved()) {
return;
}
const auto adapt = [](Formatter& out, const std::string& var, const Type* type) {
if (!type->isInterface()) {
out << var;
return;
}
// TODO(b/66900959): if we are creating the adapter for a 1.1 IFoo
// and we are using a method that takes/returns a 1.0 Callback, but
// there exists a 1.1 Callback (or other subclass that is depended
// on by this module), then wrap with the adapter subclass adapter
// IFF that callback is a subclass. However, if the callback
// is 1.0 ICallback, then wrap with a 1.0 adapter.
const Interface* interface = static_cast<const Interface*>(type);
out << "static_cast<::android::sp<" << interface->fqName().cppName() << ">>("
<< interface->fqName().cppName() << "::castFrom("
<< "::android::hardware::details::adaptWithDefault("
<< "static_cast<::android::sp<" << interface->fqName().cppName() << ">>(" << var
<< "), [&] { return new " << interface->fqName().getInterfaceAdapterFqName().cppName()
<< "(" << var << "); })))";
};
const std::string klassName = getInterface()->getAdapterName();
method->generateCppSignature(out, klassName);
out.block([&] {
bool hasCallback = !method->canElideCallback() && !method->results().empty();
if (hasCallback) {
out << method->name() << "_cb _hidl_cb_wrapped = [&](";
method->emitCppResultSignature(out);
out << ") ";
out.block([&] {
out << "return _hidl_cb(\n";
out.indent([&]() {
out.join(method->results().begin(), method->results().end(), ",\n",
[&](auto arg) { adapt(out, arg->name(), arg->get()); });
});
out << ");\n";
});
out << ";\n";
}
out << "auto _hidl_out = mImpl->" << method->name() << "(\n";
out.indent([&]() {
out.join(method->args().begin(), method->args().end(), ",\n",
[&](auto arg) { adapt(out, arg->name(), arg->get()); });
if (hasCallback) {
if (!method->args().empty()) {
out << ",\n";
}
out << "_hidl_cb_wrapped";
}
});
out << ");\n";
const auto elidedCallback = method->canElideCallback();
if (elidedCallback) {
out.sIf("!_hidl_out.isOkUnchecked()", [&] { out << "return _hidl_out;\n"; });
out << "return ";
adapt(out, "_hidl_out", elidedCallback->get());
out << ";\n";
} else {
out << "return _hidl_out;\n";
}
}).endl();
}
} // namespace android