1 #include "generate_java.h"
2 #include "AST.h"
3 #include "Type.h"
4 #include <string.h>
5 #include <stdio.h>
6 #include <stdlib.h>
7 #include <string.h>
8
9 // =================================================
10 class VariableFactory
11 {
12 public:
13 VariableFactory(const string& base); // base must be short
14 Variable* Get(Type* type);
15 Variable* Get(int index);
16 private:
17 vector<Variable*> m_vars;
18 string m_base;
19 int m_index;
20 };
21
VariableFactory(const string & base)22 VariableFactory::VariableFactory(const string& base)
23 :m_base(base),
24 m_index(0)
25 {
26 }
27
28 Variable*
Get(Type * type)29 VariableFactory::Get(Type* type)
30 {
31 char name[100];
32 sprintf(name, "%s%d", m_base.c_str(), m_index);
33 m_index++;
34 Variable* v = new Variable(type, name);
35 m_vars.push_back(v);
36 return v;
37 }
38
39 Variable*
Get(int index)40 VariableFactory::Get(int index)
41 {
42 return m_vars[index];
43 }
44
45 // =================================================
46 class StubClass : public Class
47 {
48 public:
49 StubClass(Type* type, Type* interfaceType);
50 virtual ~StubClass();
51
52 Variable* transact_code;
53 Variable* transact_data;
54 Variable* transact_reply;
55 Variable* transact_flags;
56 SwitchStatement* transact_switch;
57 private:
58 void make_as_interface(Type* interfaceType);
59 };
60
StubClass(Type * type,Type * interfaceType)61 StubClass::StubClass(Type* type, Type* interfaceType)
62 :Class()
63 {
64 this->comment = "/** Local-side IPC implementation stub class. */";
65 this->modifiers = PUBLIC | ABSTRACT | STATIC;
66 this->what = Class::CLASS;
67 this->type = type;
68 this->extends = BINDER_NATIVE_TYPE;
69 this->interfaces.push_back(interfaceType);
70
71 // descriptor
72 Field* descriptor = new Field(STATIC | FINAL | PRIVATE,
73 new Variable(STRING_TYPE, "DESCRIPTOR"));
74 descriptor->value = "\"" + interfaceType->QualifiedName() + "\"";
75 this->elements.push_back(descriptor);
76
77 // ctor
78 Method* ctor = new Method;
79 ctor->modifiers = PUBLIC;
80 ctor->comment = "/** Construct the stub at attach it to the "
81 "interface. */";
82 ctor->name = "Stub";
83 ctor->statements = new StatementBlock;
84 MethodCall* attach = new MethodCall(THIS_VALUE, "attachInterface",
85 2, THIS_VALUE, new LiteralExpression("DESCRIPTOR"));
86 ctor->statements->Add(attach);
87 this->elements.push_back(ctor);
88
89 // asInterface
90 make_as_interface(interfaceType);
91
92 // asBinder
93 Method* asBinder = new Method;
94 asBinder->modifiers = PUBLIC;
95 asBinder->returnType = IBINDER_TYPE;
96 asBinder->name = "asBinder";
97 asBinder->statements = new StatementBlock;
98 asBinder->statements->Add(new ReturnStatement(THIS_VALUE));
99 this->elements.push_back(asBinder);
100
101 // onTransact
102 this->transact_code = new Variable(INT_TYPE, "code");
103 this->transact_data = new Variable(PARCEL_TYPE, "data");
104 this->transact_reply = new Variable(PARCEL_TYPE, "reply");
105 this->transact_flags = new Variable(INT_TYPE, "flags");
106 Method* onTransact = new Method;
107 onTransact->modifiers = PUBLIC | OVERRIDE;
108 onTransact->returnType = BOOLEAN_TYPE;
109 onTransact->name = "onTransact";
110 onTransact->parameters.push_back(this->transact_code);
111 onTransact->parameters.push_back(this->transact_data);
112 onTransact->parameters.push_back(this->transact_reply);
113 onTransact->parameters.push_back(this->transact_flags);
114 onTransact->statements = new StatementBlock;
115 onTransact->exceptions.push_back(REMOTE_EXCEPTION_TYPE);
116 this->elements.push_back(onTransact);
117 this->transact_switch = new SwitchStatement(this->transact_code);
118
119 onTransact->statements->Add(this->transact_switch);
120 MethodCall* superCall = new MethodCall(SUPER_VALUE, "onTransact", 4,
121 this->transact_code, this->transact_data,
122 this->transact_reply, this->transact_flags);
123 onTransact->statements->Add(new ReturnStatement(superCall));
124 }
125
~StubClass()126 StubClass::~StubClass()
127 {
128 }
129
130 void
make_as_interface(Type * interfaceType)131 StubClass::make_as_interface(Type *interfaceType)
132 {
133 Variable* obj = new Variable(IBINDER_TYPE, "obj");
134
135 Method* m = new Method;
136 m->comment = "/**\n * Cast an IBinder object into an ";
137 m->comment += interfaceType->QualifiedName();
138 m->comment += " interface,\n";
139 m->comment += " * generating a proxy if needed.\n */";
140 m->modifiers = PUBLIC | STATIC;
141 m->returnType = interfaceType;
142 m->name = "asInterface";
143 m->parameters.push_back(obj);
144 m->statements = new StatementBlock;
145
146 IfStatement* ifstatement = new IfStatement();
147 ifstatement->expression = new Comparison(obj, "==", NULL_VALUE);
148 ifstatement->statements = new StatementBlock;
149 ifstatement->statements->Add(new ReturnStatement(NULL_VALUE));
150 m->statements->Add(ifstatement);
151
152 // IInterface iin = obj.queryLocalInterface(DESCRIPTOR)
153 MethodCall* queryLocalInterface = new MethodCall(obj, "queryLocalInterface");
154 queryLocalInterface->arguments.push_back(new LiteralExpression("DESCRIPTOR"));
155 IInterfaceType* iinType = new IInterfaceType();
156 Variable *iin = new Variable(iinType, "iin");
157 VariableDeclaration* iinVd = new VariableDeclaration(iin, queryLocalInterface, iinType);
158 m->statements->Add(iinVd);
159
160 // Ensure the instance type of the local object is as expected.
161 // One scenario where this is needed is if another package (with a
162 // different class loader) runs in the same process as the service.
163
164 // if (iin != null && iin instanceof <interfaceType>) return (<interfaceType>) iin;
165 Comparison* iinNotNull = new Comparison(iin, "!=", NULL_VALUE);
166 Comparison* instOfCheck = new Comparison(iin, " instanceof ",
167 new LiteralExpression(interfaceType->QualifiedName()));
168 IfStatement* instOfStatement = new IfStatement();
169 instOfStatement->expression = new Comparison(iinNotNull, "&&", instOfCheck);
170 instOfStatement->statements = new StatementBlock;
171 instOfStatement->statements->Add(new ReturnStatement(new Cast(interfaceType, iin)));
172 m->statements->Add(instOfStatement);
173
174 string proxyType = interfaceType->QualifiedName();
175 proxyType += ".Stub.Proxy";
176 NewExpression* ne = new NewExpression(NAMES.Find(proxyType));
177 ne->arguments.push_back(obj);
178 m->statements->Add(new ReturnStatement(ne));
179
180 this->elements.push_back(m);
181 }
182
183
184
185 // =================================================
186 class ProxyClass : public Class
187 {
188 public:
189 ProxyClass(Type* type, InterfaceType* interfaceType);
190 virtual ~ProxyClass();
191
192 Variable* mRemote;
193 bool mOneWay;
194 };
195
ProxyClass(Type * type,InterfaceType * interfaceType)196 ProxyClass::ProxyClass(Type* type, InterfaceType* interfaceType)
197 :Class()
198 {
199 this->modifiers = PRIVATE | STATIC;
200 this->what = Class::CLASS;
201 this->type = type;
202 this->interfaces.push_back(interfaceType);
203
204 mOneWay = interfaceType->OneWay();
205
206 // IBinder mRemote
207 mRemote = new Variable(IBINDER_TYPE, "mRemote");
208 this->elements.push_back(new Field(PRIVATE, mRemote));
209
210 // Proxy()
211 Variable* remote = new Variable(IBINDER_TYPE, "remote");
212 Method* ctor = new Method;
213 ctor->name = "Proxy";
214 ctor->statements = new StatementBlock;
215 ctor->parameters.push_back(remote);
216 ctor->statements->Add(new Assignment(mRemote, remote));
217 this->elements.push_back(ctor);
218
219 // IBinder asBinder()
220 Method* asBinder = new Method;
221 asBinder->modifiers = PUBLIC;
222 asBinder->returnType = IBINDER_TYPE;
223 asBinder->name = "asBinder";
224 asBinder->statements = new StatementBlock;
225 asBinder->statements->Add(new ReturnStatement(mRemote));
226 this->elements.push_back(asBinder);
227 }
228
~ProxyClass()229 ProxyClass::~ProxyClass()
230 {
231 }
232
233 // =================================================
234 static string
gather_comments(extra_text_type * extra)235 gather_comments(extra_text_type* extra)
236 {
237 string s;
238 while (extra) {
239 if (extra->which == SHORT_COMMENT) {
240 s += extra->data;
241 }
242 else if (extra->which == LONG_COMMENT) {
243 s += "/*";
244 s += extra->data;
245 s += "*/";
246 }
247 extra = extra->next;
248 }
249 return s;
250 }
251
252 static string
append(const char * a,const char * b)253 append(const char* a, const char* b)
254 {
255 string s = a;
256 s += b;
257 return s;
258 }
259
260 static void
generate_new_array(Type * t,StatementBlock * addTo,Variable * v,Variable * parcel)261 generate_new_array(Type* t, StatementBlock* addTo, Variable* v,
262 Variable* parcel)
263 {
264 Variable* len = new Variable(INT_TYPE, v->name + "_length");
265 addTo->Add(new VariableDeclaration(len, new MethodCall(parcel, "readInt")));
266 IfStatement* lencheck = new IfStatement();
267 lencheck->expression = new Comparison(len, "<", new LiteralExpression("0"));
268 lencheck->statements->Add(new Assignment(v, NULL_VALUE));
269 lencheck->elseif = new IfStatement();
270 lencheck->elseif->statements->Add(new Assignment(v,
271 new NewArrayExpression(t, len)));
272 addTo->Add(lencheck);
273 }
274
275 static void
generate_write_to_parcel(Type * t,StatementBlock * addTo,Variable * v,Variable * parcel,int flags)276 generate_write_to_parcel(Type* t, StatementBlock* addTo, Variable* v,
277 Variable* parcel, int flags)
278 {
279 if (v->dimension == 0) {
280 t->WriteToParcel(addTo, v, parcel, flags);
281 }
282 if (v->dimension == 1) {
283 t->WriteArrayToParcel(addTo, v, parcel, flags);
284 }
285 }
286
287 static void
generate_create_from_parcel(Type * t,StatementBlock * addTo,Variable * v,Variable * parcel,Variable ** cl)288 generate_create_from_parcel(Type* t, StatementBlock* addTo, Variable* v,
289 Variable* parcel, Variable** cl)
290 {
291 if (v->dimension == 0) {
292 t->CreateFromParcel(addTo, v, parcel, cl);
293 }
294 if (v->dimension == 1) {
295 t->CreateArrayFromParcel(addTo, v, parcel, cl);
296 }
297 }
298
299 static void
generate_read_from_parcel(Type * t,StatementBlock * addTo,Variable * v,Variable * parcel,Variable ** cl)300 generate_read_from_parcel(Type* t, StatementBlock* addTo, Variable* v,
301 Variable* parcel, Variable** cl)
302 {
303 if (v->dimension == 0) {
304 t->ReadFromParcel(addTo, v, parcel, cl);
305 }
306 if (v->dimension == 1) {
307 t->ReadArrayFromParcel(addTo, v, parcel, cl);
308 }
309 }
310
311
312 static void
generate_method(const method_type * method,Class * interface,StubClass * stubClass,ProxyClass * proxyClass,int index)313 generate_method(const method_type* method, Class* interface,
314 StubClass* stubClass, ProxyClass* proxyClass, int index)
315 {
316 arg_type* arg;
317 int i;
318 bool hasOutParams = false;
319
320 const bool oneway = proxyClass->mOneWay || method->oneway;
321
322 // == the TRANSACT_ constant =============================================
323 string transactCodeName = "TRANSACTION_";
324 transactCodeName += method->name.data;
325
326 char transactCodeValue[50];
327 sprintf(transactCodeValue, "(android.os.IBinder.FIRST_CALL_TRANSACTION + %d)", index);
328
329 Field* transactCode = new Field(STATIC | FINAL,
330 new Variable(INT_TYPE, transactCodeName));
331 transactCode->value = transactCodeValue;
332 stubClass->elements.push_back(transactCode);
333
334 // == the declaration in the interface ===================================
335 Method* decl = new Method;
336 decl->comment = gather_comments(method->comments_token->extra);
337 decl->modifiers = PUBLIC;
338 decl->returnType = NAMES.Search(method->type.type.data);
339 decl->returnTypeDimension = method->type.dimension;
340 decl->name = method->name.data;
341
342 arg = method->args;
343 while (arg != NULL) {
344 decl->parameters.push_back(new Variable(
345 NAMES.Search(arg->type.type.data), arg->name.data,
346 arg->type.dimension));
347 arg = arg->next;
348 }
349
350 decl->exceptions.push_back(REMOTE_EXCEPTION_TYPE);
351
352 interface->elements.push_back(decl);
353
354 // == the stub method ====================================================
355
356 Case* c = new Case(transactCodeName);
357
358 MethodCall* realCall = new MethodCall(THIS_VALUE, method->name.data);
359
360 // interface token validation is the very first thing we do
361 c->statements->Add(new MethodCall(stubClass->transact_data,
362 "enforceInterface", 1, new LiteralExpression("DESCRIPTOR")));
363
364 // args
365 Variable* cl = NULL;
366 VariableFactory stubArgs("_arg");
367 arg = method->args;
368 while (arg != NULL) {
369 Type* t = NAMES.Search(arg->type.type.data);
370 Variable* v = stubArgs.Get(t);
371 v->dimension = arg->type.dimension;
372
373 c->statements->Add(new VariableDeclaration(v));
374
375 if (convert_direction(arg->direction.data) & IN_PARAMETER) {
376 generate_create_from_parcel(t, c->statements, v,
377 stubClass->transact_data, &cl);
378 } else {
379 if (arg->type.dimension == 0) {
380 c->statements->Add(new Assignment(
381 v, new NewExpression(v->type)));
382 }
383 else if (arg->type.dimension == 1) {
384 generate_new_array(v->type, c->statements, v,
385 stubClass->transact_data);
386 }
387 else {
388 fprintf(stderr, "aidl:internal error %s:%d\n", __FILE__,
389 __LINE__);
390 }
391 }
392
393 realCall->arguments.push_back(v);
394
395 arg = arg->next;
396 }
397
398 // the real call
399 Variable* _result = NULL;
400 if (0 == strcmp(method->type.type.data, "void")) {
401 c->statements->Add(realCall);
402
403 if (!oneway) {
404 // report that there were no exceptions
405 MethodCall* ex = new MethodCall(stubClass->transact_reply,
406 "writeNoException", 0);
407 c->statements->Add(ex);
408 }
409 } else {
410 _result = new Variable(decl->returnType, "_result",
411 decl->returnTypeDimension);
412 c->statements->Add(new VariableDeclaration(_result, realCall));
413
414 if (!oneway) {
415 // report that there were no exceptions
416 MethodCall* ex = new MethodCall(stubClass->transact_reply,
417 "writeNoException", 0);
418 c->statements->Add(ex);
419 }
420
421 // marshall the return value
422 generate_write_to_parcel(decl->returnType, c->statements, _result,
423 stubClass->transact_reply,
424 Type::PARCELABLE_WRITE_RETURN_VALUE);
425 }
426
427 // out parameters
428 i = 0;
429 arg = method->args;
430 while (arg != NULL) {
431 Type* t = NAMES.Search(arg->type.type.data);
432 Variable* v = stubArgs.Get(i++);
433
434 if (convert_direction(arg->direction.data) & OUT_PARAMETER) {
435 generate_write_to_parcel(t, c->statements, v,
436 stubClass->transact_reply,
437 Type::PARCELABLE_WRITE_RETURN_VALUE);
438 hasOutParams = true;
439 }
440
441 arg = arg->next;
442 }
443
444 // return true
445 c->statements->Add(new ReturnStatement(TRUE_VALUE));
446 stubClass->transact_switch->cases.push_back(c);
447
448 // == the proxy method ===================================================
449 Method* proxy = new Method;
450 proxy->comment = gather_comments(method->comments_token->extra);
451 proxy->modifiers = PUBLIC;
452 proxy->returnType = NAMES.Search(method->type.type.data);
453 proxy->returnTypeDimension = method->type.dimension;
454 proxy->name = method->name.data;
455 proxy->statements = new StatementBlock;
456 arg = method->args;
457 while (arg != NULL) {
458 proxy->parameters.push_back(new Variable(
459 NAMES.Search(arg->type.type.data), arg->name.data,
460 arg->type.dimension));
461 arg = arg->next;
462 }
463 proxy->exceptions.push_back(REMOTE_EXCEPTION_TYPE);
464 proxyClass->elements.push_back(proxy);
465
466 // the parcels
467 Variable* _data = new Variable(PARCEL_TYPE, "_data");
468 proxy->statements->Add(new VariableDeclaration(_data,
469 new MethodCall(PARCEL_TYPE, "obtain")));
470 Variable* _reply = NULL;
471 if (!oneway) {
472 _reply = new Variable(PARCEL_TYPE, "_reply");
473 proxy->statements->Add(new VariableDeclaration(_reply,
474 new MethodCall(PARCEL_TYPE, "obtain")));
475 }
476
477 // the return value
478 _result = NULL;
479 if (0 != strcmp(method->type.type.data, "void")) {
480 _result = new Variable(proxy->returnType, "_result",
481 method->type.dimension);
482 proxy->statements->Add(new VariableDeclaration(_result));
483 }
484
485 // try and finally
486 TryStatement* tryStatement = new TryStatement();
487 proxy->statements->Add(tryStatement);
488 FinallyStatement* finallyStatement = new FinallyStatement();
489 proxy->statements->Add(finallyStatement);
490
491 // the interface identifier token: the DESCRIPTOR constant, marshalled as a string
492 tryStatement->statements->Add(new MethodCall(_data, "writeInterfaceToken",
493 1, new LiteralExpression("DESCRIPTOR")));
494
495 // the parameters
496 arg = method->args;
497 while (arg != NULL) {
498 Type* t = NAMES.Search(arg->type.type.data);
499 Variable* v = new Variable(t, arg->name.data, arg->type.dimension);
500 int dir = convert_direction(arg->direction.data);
501 if (dir == OUT_PARAMETER && arg->type.dimension != 0) {
502 IfStatement* checklen = new IfStatement();
503 checklen->expression = new Comparison(v, "==", NULL_VALUE);
504 checklen->statements->Add(new MethodCall(_data, "writeInt", 1,
505 new LiteralExpression("-1")));
506 checklen->elseif = new IfStatement();
507 checklen->elseif->statements->Add(new MethodCall(_data, "writeInt",
508 1, new FieldVariable(v, "length")));
509 tryStatement->statements->Add(checklen);
510 }
511 else if (dir & IN_PARAMETER) {
512 generate_write_to_parcel(t, tryStatement->statements, v, _data, 0);
513 }
514 arg = arg->next;
515 }
516
517 // the transact call
518 MethodCall* call = new MethodCall(proxyClass->mRemote, "transact", 4,
519 new LiteralExpression("Stub." + transactCodeName),
520 _data, _reply ? _reply : NULL_VALUE,
521 new LiteralExpression(
522 oneway ? "android.os.IBinder.FLAG_ONEWAY" : "0"));
523 tryStatement->statements->Add(call);
524
525 // throw back exceptions.
526 if (_reply) {
527 MethodCall* ex = new MethodCall(_reply, "readException", 0);
528 tryStatement->statements->Add(ex);
529 }
530
531 // returning and cleanup
532 if (_reply != NULL) {
533 if (_result != NULL) {
534 generate_create_from_parcel(proxy->returnType,
535 tryStatement->statements, _result, _reply, &cl);
536 }
537
538 // the out/inout parameters
539 arg = method->args;
540 while (arg != NULL) {
541 Type* t = NAMES.Search(arg->type.type.data);
542 Variable* v = new Variable(t, arg->name.data, arg->type.dimension);
543 if (convert_direction(arg->direction.data) & OUT_PARAMETER) {
544 generate_read_from_parcel(t, tryStatement->statements,
545 v, _reply, &cl);
546 }
547 arg = arg->next;
548 }
549
550 finallyStatement->statements->Add(new MethodCall(_reply, "recycle"));
551 }
552 finallyStatement->statements->Add(new MethodCall(_data, "recycle"));
553
554 if (_result != NULL) {
555 proxy->statements->Add(new ReturnStatement(_result));
556 }
557 }
558
559 static void
generate_interface_descriptors(StubClass * stub,ProxyClass * proxy)560 generate_interface_descriptors(StubClass* stub, ProxyClass* proxy)
561 {
562 // the interface descriptor transaction handler
563 Case* c = new Case("INTERFACE_TRANSACTION");
564 c->statements->Add(new MethodCall(stub->transact_reply, "writeString",
565 1, new LiteralExpression("DESCRIPTOR")));
566 c->statements->Add(new ReturnStatement(TRUE_VALUE));
567 stub->transact_switch->cases.push_back(c);
568
569 // and the proxy-side method returning the descriptor directly
570 Method* getDesc = new Method;
571 getDesc->modifiers = PUBLIC;
572 getDesc->returnType = STRING_TYPE;
573 getDesc->returnTypeDimension = 0;
574 getDesc->name = "getInterfaceDescriptor";
575 getDesc->statements = new StatementBlock;
576 getDesc->statements->Add(new ReturnStatement(new LiteralExpression("DESCRIPTOR")));
577 proxy->elements.push_back(getDesc);
578 }
579
580 static Class*
generate_interface_class(const interface_type * iface)581 generate_interface_class(const interface_type* iface)
582 {
583 InterfaceType* interfaceType = static_cast<InterfaceType*>(
584 NAMES.Find(iface->package, iface->name.data));
585
586 // the interface class
587 Class* interface = new Class;
588 interface->comment = gather_comments(iface->comments_token->extra);
589 interface->modifiers = PUBLIC;
590 interface->what = Class::INTERFACE;
591 interface->type = interfaceType;
592 interface->interfaces.push_back(IINTERFACE_TYPE);
593
594 // the stub inner class
595 StubClass* stub = new StubClass(
596 NAMES.Find(iface->package, append(iface->name.data, ".Stub").c_str()),
597 interfaceType);
598 interface->elements.push_back(stub);
599
600 // the proxy inner class
601 ProxyClass* proxy = new ProxyClass(
602 NAMES.Find(iface->package,
603 append(iface->name.data, ".Stub.Proxy").c_str()),
604 interfaceType);
605 stub->elements.push_back(proxy);
606
607 // stub and proxy support for getInterfaceDescriptor()
608 generate_interface_descriptors(stub, proxy);
609
610 // all the declared methods of the interface
611 int index = 0;
612 interface_item_type* item = iface->interface_items;
613 while (item != NULL) {
614 if (item->item_type == METHOD_TYPE) {
615 generate_method((method_type*)item, interface, stub, proxy, index);
616 }
617 item = item->next;
618 index++;
619 }
620
621 return interface;
622 }
623
624 int
generate_java(const string & filename,const string & originalSrc,interface_type * iface)625 generate_java(const string& filename, const string& originalSrc,
626 interface_type* iface)
627 {
628 Document* document = new Document;
629 document->comment = "";
630 if (iface->package) document->package = iface->package;
631 document->originalSrc = originalSrc;
632 document->classes.push_back(generate_interface_class(iface));
633
634 // printf("outputting... filename=%s\n", filename.c_str());
635 FILE* to;
636 if (filename == "-") {
637 to = stdout;
638 } else {
639 /* open file in binary mode to ensure that the tool produces the
640 * same output on all platforms !!
641 */
642 to = fopen(filename.c_str(), "wb");
643 if (to == NULL) {
644 fprintf(stderr, "unable to open %s for write\n", filename.c_str());
645 return 1;
646 }
647 }
648
649 document->Write(to);
650
651 fclose(to);
652 return 0;
653 }
654
655