/*
* 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 "CompositeDeclaration.h"
#include "FunctionDeclaration.h"
#include "VarDeclaration.h"
#include "Declaration.h"
#include <algorithm>
#include <iostream>
#include <string>
namespace android {
CompositeDeclaration::CompositeDeclaration(
const Type::Qualifier::Qualification qualifier,
const std::string &name,
std::vector<android::Declaration *> *fieldDeclarations)
: Declaration(""),
mQualifier(qualifier),
mFieldDeclarations(fieldDeclarations)
{
setName(name);
}
CompositeDeclaration::~CompositeDeclaration() {
if(mFieldDeclarations != NULL) {
for(auto* decl : *mFieldDeclarations) {
delete decl;
}
}
delete mFieldDeclarations;
}
void CompositeDeclaration::setName(const std::string &name) {
Declaration::setName(name);
forcePascalCase();
}
const Type::Qualifier::Qualification &CompositeDeclaration::getQualifier() const {
return mQualifier;
}
const std::vector<android::Declaration *>*
CompositeDeclaration::getFieldDeclarations() const {
return mFieldDeclarations;
}
void CompositeDeclaration::generateInterface(Formatter &out) const {
generateCommentText(out);
out << "interface " << getInterfaceName() << " {\n\n";
generateBody(out);
out << "};\n";
}
void CompositeDeclaration::generateSource(Formatter &out) const {
CHECK(mQualifier == Type::Qualifier::STRUCT ||
mQualifier == Type::Qualifier::UNION ||
mQualifier == Type::Qualifier::ENUM);
out << Type::qualifierText(mQualifier) << " " << getName();
if (mQualifier == Type::Qualifier::ENUM) {
out << " : ";
if (mEnumTypeName.empty()) {
out << "int32_t /* NOTE: type is guessed */";
} else {
out << mEnumTypeName;
}
}
out << " {\n";
generateBody(out);
out << "};\n";
}
void CompositeDeclaration::generateBody(Formatter &out) const {
out.indent();
for (auto *declaration : *mFieldDeclarations) {
declaration->generateCommentText(out);
declaration->generateSource(out);
out << "\n";
}
out.unindent();
}
void CompositeDeclaration::processContents(AST &ast) {
for (auto &declaration : *mFieldDeclarations) {
declaration->processContents(ast);
}
if (isInterface()) {
// move non function fields into a containing struct
auto nonFpDecs = new std::vector<Declaration*>;
auto it = mFieldDeclarations->begin();
while (it != mFieldDeclarations->end()) {
if((*it)->decType() != FunctionDeclaration::type()) {
bool keep = true;
if((*it)->decType() == VarDeclaration::type()) {
VarDeclaration* var = (VarDeclaration *)(*it);
// Conventional HALs were all required to have
// a member of this type.
// This member is no longer needed for HIDL
if(var->getType()->isHwDevice()) {
keep = false;
}
}
if (keep) {
nonFpDecs->push_back(*it);
}
it = mFieldDeclarations->erase(it);
} else {
it++;
}
}
if (!nonFpDecs->empty()) {
auto subStruct = new CompositeDeclaration(Type::Qualifier::STRUCT,
getName(),
nonFpDecs);
mFieldDeclarations->insert(mFieldDeclarations->begin(), subStruct);
}
}
}
std::string CompositeDeclaration::getInterfaceName() const {
return "I" + getName();
}
bool CompositeDeclaration::isInterface() const {
if (mQualifier != Type::Qualifier::STRUCT) {
return false;
}
for (auto &declaration : *mFieldDeclarations) {
if (declaration->decType() == FunctionDeclaration::type()) {
return true;
}
}
return false;
}
void CompositeDeclaration::setEnumTypeName(const std::string &name) {
CHECK(mQualifier == Type::Qualifier::ENUM);
mEnumTypeName = name;
}
} //namespace android