blob: 25310f53a033cfd749e6a3e8ce9a767a8b0e2694 [file] [log] [blame]
Steven Moreland9c387612016-09-07 09:54:26 -07001/*
2 * Copyright (C) 2016 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 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "AST.h"
18
19#include "Coordinator.h"
20#include "EnumType.h"
Steven Moreland9c387612016-09-07 09:54:26 -070021#include "Interface.h"
22#include "Method.h"
23#include "ScalarType.h"
24#include "Scope.h"
25
26#include <algorithm>
Iliyan Malcheva72e0d22016-09-09 11:03:08 -070027#include <hidl-util/Formatter.h>
Steven Moreland9c387612016-09-07 09:54:26 -070028#include <android-base/logging.h>
29#include <string>
30#include <vector>
31#include <set>
32
Steven Moreland9c387612016-09-07 09:54:26 -070033namespace android {
34
Steven Moreland9c387612016-09-07 09:54:26 -070035status_t AST::generateCppImpl(const std::string &outputPath) const {
36 status_t err = generateStubImplHeader(outputPath);
37
38 if (err == OK) {
39 err = generateStubImplSource(outputPath);
40 }
41
42 return err;
43}
44
45void AST::generateFetchSymbol(Formatter &out, const std::string& ifaceName) const {
46 out << "HIDL_FETCH_" << ifaceName;
47}
48
49status_t AST::generateStubImplMethod(Formatter &out,
50 const std::string &className,
Yifan Hong068c5522016-10-31 14:07:25 -070051 const Method *method) const {
Steven Moreland9c387612016-09-07 09:54:26 -070052
Yifan Hong1254b552016-10-27 15:03:29 -070053 // ignore HIDL reserved methods -- implemented in IFoo already.
54 if (method->isHidlReserved()) {
55 return OK;
56 }
57
Yifan Hong068c5522016-10-31 14:07:25 -070058 method->generateCppSignature(out, className, false /* specifyNamespaces */);
Steven Moreland9c387612016-09-07 09:54:26 -070059
60 out << " {\n";
61
62 out.indent();
63 out << "// TODO implement\n";
64
65 const TypedVar *elidedReturn = method->canElideCallback();
66
67 if (elidedReturn == nullptr) {
68 out << "return Void();\n";
69 } else {
70 std::string extra;
71 out << "return "
72 << elidedReturn->type().getCppResultType(&extra)
73 << " {};\n";
74 }
75
76 out.unindent();
77
78 out << "}\n\n";
79
80 return OK;
81}
82
Steven Moreland9c387612016-09-07 09:54:26 -070083status_t AST::generateStubImplHeader(const std::string &outputPath) const {
84 std::string ifaceName;
85 if (!AST::isInterface(&ifaceName)) {
86 // types.hal does not get a stub header.
87 return OK;
88 }
89
90 const Interface *iface = mRootScope->getInterface();
91
Jayant Chowdhary3f32c1f2016-09-15 16:53:56 -070092 const std::string baseName = iface->getBaseName();
Steven Moreland9c387612016-09-07 09:54:26 -070093
94 std::string path = outputPath;
95 path.append(baseName);
96 path.append(".h");
97
98 CHECK(Coordinator::MakeParentHierarchy(path));
99 FILE *file = fopen(path.c_str(), "w");
100
101 if (file == NULL) {
102 return -errno;
103 }
104
105 Formatter out(file);
106
Martijn Coenen119f5212016-11-04 15:24:30 +0000107 const std::string guard = makeHeaderGuard(baseName);
Steven Moreland9c387612016-09-07 09:54:26 -0700108
109 out << "#ifndef " << guard << "\n";
110 out << "#define " << guard << "\n\n";
111
Steven Morelandee88eed2016-10-31 17:49:00 -0700112 generateCppPackageInclude(out, mPackage, "I" + baseName);
Steven Moreland9c387612016-09-07 09:54:26 -0700113
114 out << "#include <hidl/Status.h>\n\n";
Andreas Huber4bcf97d2016-08-30 11:27:49 -0700115 out << "#include <hidl/MQDescriptor.h>\n";
Steven Moreland9c387612016-09-07 09:54:26 -0700116
117 enterLeaveNamespace(out, true /* enter */);
118 out << "namespace implementation {\n\n";
119
120 // this is namespace aware code and doesn't require post-processing
121 out.setNamespace("");
122
Yifan Hong10fe0b52016-10-19 14:20:17 -0700123 std::vector<const Interface *> chain = iface->typeChain();
Steven Moreland9c387612016-09-07 09:54:26 -0700124
125 std::set<const FQName> usedTypes{};
126
127 for (auto it = chain.rbegin(); it != chain.rend(); ++it) {
128 const Interface *superInterface = *it;
Steven Moreland9c387612016-09-07 09:54:26 -0700129 superInterface->addNamedTypesToSet(usedTypes);
Yifan Hong10fe0b52016-10-19 14:20:17 -0700130 }
Steven Moreland9c387612016-09-07 09:54:26 -0700131
Yifan Hong10fe0b52016-10-19 14:20:17 -0700132 for (const auto &tuple : iface->allMethodsFromRoot()) {
133 const Method *method = tuple.method();
134 for(const auto & arg : method->args()) {
135 arg->type().addNamedTypesToSet(usedTypes);
136 }
137 for(const auto & results : method->results()) {
138 results->type().addNamedTypesToSet(usedTypes);
Steven Moreland9c387612016-09-07 09:54:26 -0700139 }
140 }
141
142 std::set<const FQName> topLevelTypes{};
143
144 for (const auto &name : usedTypes) {
145 topLevelTypes.insert(name.getTopLevelType());
146 }
147
148 for (const FQName &name : topLevelTypes) {
149 out << "using " << name.cppName() << ";\n";
150 }
151
152 out << "using ::android::hardware::Return;\n";
153 out << "using ::android::hardware::Void;\n";
154 out << "using ::android::hardware::hidl_vec;\n";
155 out << "using ::android::hardware::hidl_string;\n";
156 out << "using ::android::sp;\n";
157
158 out << "\n";
159
160 out << "struct "
161 << baseName
162 << " : public "
163 << ifaceName
164 << " {\n";
165
166 out.indent();
167
Yifan Hong068c5522016-10-31 14:07:25 -0700168 status_t err = generateMethods(out, [&](const Method *method, const Interface *) {
169 // ignore HIDL reserved methods -- implemented in IFoo already.
170 if (method->isHidlReserved()) {
171 return OK;
172 }
173 method->generateCppSignature(out, "" /* className */,
174 false /* specifyNamespaces */);
175 out << " override;\n";
176 return OK;
177 });
Steven Moreland9c387612016-09-07 09:54:26 -0700178
179 if (err != OK) {
180 return err;
181 }
182
183 out.unindent();
184
185 out << "};\n\n";
186
187 out << "extern \"C\" "
188 << ifaceName
189 << "* ";
190 generateFetchSymbol(out, ifaceName);
191 out << "(const char* name);\n\n";
192
193 out << "} // namespace implementation\n";
194 enterLeaveNamespace(out, false /* leave */);
195
196 out << "\n#endif // " << guard << "\n";
197
198 return OK;
199}
200
201status_t AST::generateStubImplSource(const std::string &outputPath) const {
202 std::string ifaceName;
203 if (!AST::isInterface(&ifaceName)) {
204 // types.hal does not get a stub header.
205 return OK;
206 }
207
Jayant Chowdhary3f32c1f2016-09-15 16:53:56 -0700208 const Interface *iface = mRootScope->getInterface();
209 const std::string baseName = iface->getBaseName();
Steven Moreland9c387612016-09-07 09:54:26 -0700210
211 std::string path = outputPath;
212 path.append(baseName);
213 path.append(".cpp");
214
215 CHECK(Coordinator::MakeParentHierarchy(path));
216 FILE *file = fopen(path.c_str(), "w");
217
218 if (file == NULL) {
219 return -errno;
220 }
221
222 Formatter out(file);
223
224 out << "#include \"" << baseName << ".h\"\n\n";
225
226 enterLeaveNamespace(out, true /* enter */);
227 out << "namespace implementation {\n\n";
228
229 // this is namespace aware code and doesn't require post-processing
230 out.setNamespace("");
231
Yifan Hong068c5522016-10-31 14:07:25 -0700232 status_t err = generateMethods(out, [&](const Method *method, const Interface *) {
233 return generateStubImplMethod(out, baseName, method);
234 });
Steven Moreland9c387612016-09-07 09:54:26 -0700235
236 out << ifaceName
237 << "* ";
238 generateFetchSymbol(out, ifaceName);
239 out << "(const char* /* name */) {\n";
240 out.indent();
241 out << "return new " << baseName << "();\n";
242 out.unindent();
243 out << "}\n\n";
244
245 out << "} // namespace implementation\n";
246 enterLeaveNamespace(out, false /* leave */);
247
248 return OK;
249}
250
251} // namespace android