• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 #include "AST.h"
2 #include "Type.h"
3 
4 void
WriteModifiers(FILE * to,int mod,int mask)5 WriteModifiers(FILE* to, int mod, int mask)
6 {
7     int m = mod & mask;
8 
9     if (m & OVERRIDE) {
10         fprintf(to, "@Override ");
11     }
12 
13     if ((m & SCOPE_MASK) == PUBLIC) {
14         fprintf(to, "public ");
15     }
16     else if ((m & SCOPE_MASK) == PRIVATE) {
17         fprintf(to, "private ");
18     }
19     else if ((m & SCOPE_MASK) == PROTECTED) {
20         fprintf(to, "protected ");
21     }
22 
23     if (m & STATIC) {
24         fprintf(to, "static ");
25     }
26 
27     if (m & FINAL) {
28         fprintf(to, "final ");
29     }
30 
31     if (m & ABSTRACT) {
32         fprintf(to, "abstract ");
33     }
34 }
35 
36 void
WriteArgumentList(FILE * to,const vector<Expression * > & arguments)37 WriteArgumentList(FILE* to, const vector<Expression*>& arguments)
38 {
39     size_t N = arguments.size();
40     for (size_t i=0; i<N; i++) {
41         arguments[i]->Write(to);
42         if (i != N-1) {
43             fprintf(to, ", ");
44         }
45     }
46 }
47 
ClassElement()48 ClassElement::ClassElement()
49 {
50 }
51 
~ClassElement()52 ClassElement::~ClassElement()
53 {
54 }
55 
Field()56 Field::Field()
57     :ClassElement(),
58      modifiers(0),
59      variable(NULL)
60 {
61 }
62 
Field(int m,Variable * v)63 Field::Field(int m, Variable* v)
64     :ClassElement(),
65      modifiers(m),
66      variable(v)
67 {
68 }
69 
~Field()70 Field::~Field()
71 {
72 }
73 
74 void
GatherTypes(set<Type * > * types) const75 Field::GatherTypes(set<Type*>* types) const
76 {
77     types->insert(this->variable->type);
78 }
79 
80 void
Write(FILE * to)81 Field::Write(FILE* to)
82 {
83     if (this->comment.length() != 0) {
84         fprintf(to, "%s\n", this->comment.c_str());
85     }
86     WriteModifiers(to, this->modifiers, SCOPE_MASK | STATIC | FINAL | OVERRIDE);
87     fprintf(to, "%s %s", this->variable->type->QualifiedName().c_str(),
88             this->variable->name.c_str());
89     if (this->value.length() != 0) {
90         fprintf(to, " = %s", this->value.c_str());
91     }
92     fprintf(to, ";\n");
93 }
94 
~Expression()95 Expression::~Expression()
96 {
97 }
98 
LiteralExpression(const string & v)99 LiteralExpression::LiteralExpression(const string& v)
100     :value(v)
101 {
102 }
103 
~LiteralExpression()104 LiteralExpression::~LiteralExpression()
105 {
106 }
107 
108 void
Write(FILE * to)109 LiteralExpression::Write(FILE* to)
110 {
111     fprintf(to, "%s", this->value.c_str());
112 }
113 
Variable()114 Variable::Variable()
115     :type(NULL),
116      name(),
117      dimension(0)
118 {
119 }
120 
Variable(Type * t,const string & n)121 Variable::Variable(Type* t, const string& n)
122     :type(t),
123      name(n),
124      dimension(0)
125 {
126 }
127 
Variable(Type * t,const string & n,int d)128 Variable::Variable(Type* t, const string& n, int d)
129     :type(t),
130      name(n),
131      dimension(d)
132 {
133 }
134 
~Variable()135 Variable::~Variable()
136 {
137 }
138 
139 void
GatherTypes(set<Type * > * types) const140 Variable::GatherTypes(set<Type*>* types) const
141 {
142     types->insert(this->type);
143 }
144 
145 void
WriteDeclaration(FILE * to)146 Variable::WriteDeclaration(FILE* to)
147 {
148     string dim;
149     for (int i=0; i<this->dimension; i++) {
150         dim += "[]";
151     }
152     fprintf(to, "%s%s %s", this->type->QualifiedName().c_str(), dim.c_str(),
153             this->name.c_str());
154 }
155 
156 void
Write(FILE * to)157 Variable::Write(FILE* to)
158 {
159     fprintf(to, "%s", name.c_str());
160 }
161 
FieldVariable(Expression * o,const string & n)162 FieldVariable::FieldVariable(Expression* o, const string& n)
163     :object(o),
164      clazz(NULL),
165      name(n)
166 {
167 }
168 
FieldVariable(Type * c,const string & n)169 FieldVariable::FieldVariable(Type* c, const string& n)
170     :object(NULL),
171      clazz(c),
172      name(n)
173 {
174 }
175 
~FieldVariable()176 FieldVariable::~FieldVariable()
177 {
178 }
179 
180 void
Write(FILE * to)181 FieldVariable::Write(FILE* to)
182 {
183     if (this->object != NULL) {
184         this->object->Write(to);
185     }
186     else if (this->clazz != NULL) {
187         fprintf(to, "%s", this->clazz->QualifiedName().c_str());
188     }
189     fprintf(to, ".%s", name.c_str());
190 }
191 
192 
~Statement()193 Statement::~Statement()
194 {
195 }
196 
StatementBlock()197 StatementBlock::StatementBlock()
198 {
199 }
200 
~StatementBlock()201 StatementBlock::~StatementBlock()
202 {
203 }
204 
205 void
Write(FILE * to)206 StatementBlock::Write(FILE* to)
207 {
208     fprintf(to, "{\n");
209     int N = this->statements.size();
210     for (int i=0; i<N; i++) {
211         this->statements[i]->Write(to);
212     }
213     fprintf(to, "}\n");
214 }
215 
216 void
Add(Statement * statement)217 StatementBlock::Add(Statement* statement)
218 {
219     this->statements.push_back(statement);
220 }
221 
222 void
Add(Expression * expression)223 StatementBlock::Add(Expression* expression)
224 {
225     this->statements.push_back(new ExpressionStatement(expression));
226 }
227 
ExpressionStatement(Expression * e)228 ExpressionStatement::ExpressionStatement(Expression* e)
229     :expression(e)
230 {
231 }
232 
~ExpressionStatement()233 ExpressionStatement::~ExpressionStatement()
234 {
235 }
236 
237 void
Write(FILE * to)238 ExpressionStatement::Write(FILE* to)
239 {
240     this->expression->Write(to);
241     fprintf(to, ";\n");
242 }
243 
Assignment(Variable * l,Expression * r)244 Assignment::Assignment(Variable* l, Expression* r)
245     :lvalue(l),
246      rvalue(r),
247      cast(NULL)
248 {
249 }
250 
Assignment(Variable * l,Expression * r,Type * c)251 Assignment::Assignment(Variable* l, Expression* r, Type* c)
252     :lvalue(l),
253      rvalue(r),
254      cast(c)
255 {
256 }
257 
~Assignment()258 Assignment::~Assignment()
259 {
260 }
261 
262 void
Write(FILE * to)263 Assignment::Write(FILE* to)
264 {
265     this->lvalue->Write(to);
266     fprintf(to, " = ");
267     if (this->cast != NULL) {
268         fprintf(to, "(%s)", this->cast->QualifiedName().c_str());
269     }
270     this->rvalue->Write(to);
271 }
272 
MethodCall(const string & n)273 MethodCall::MethodCall(const string& n)
274     :obj(NULL),
275      clazz(NULL),
276      name(n)
277 {
278 }
279 
MethodCall(Expression * o,const string & n)280 MethodCall::MethodCall(Expression* o, const string& n)
281     :obj(o),
282      clazz(NULL),
283      name(n)
284 {
285 }
286 
MethodCall(Type * t,const string & n)287 MethodCall::MethodCall(Type* t, const string& n)
288     :obj(NULL),
289      clazz(t),
290      name(n)
291 {
292 }
293 
MethodCall(Expression * o,const string & n,int argc=0,...)294 MethodCall::MethodCall(Expression* o, const string& n, int argc = 0, ...)
295     :obj(o),
296      clazz(NULL),
297      name(n)
298 {
299   va_list args;
300   va_start(args, argc);
301   init(argc, args);
302   va_end(args);
303 }
304 
MethodCall(Type * t,const string & n,int argc=0,...)305 MethodCall::MethodCall(Type* t, const string& n, int argc = 0, ...)
306     :obj(NULL),
307      clazz(t),
308      name(n)
309 {
310   va_list args;
311   va_start(args, argc);
312   init(argc, args);
313   va_end(args);
314 }
315 
~MethodCall()316 MethodCall::~MethodCall()
317 {
318 }
319 
320 void
init(int n,va_list args)321 MethodCall::init(int n, va_list args)
322 {
323     for (int i=0; i<n; i++) {
324         Expression* expression = (Expression*)va_arg(args, void*);
325         this->arguments.push_back(expression);
326     }
327 }
328 
329 void
Write(FILE * to)330 MethodCall::Write(FILE* to)
331 {
332     if (this->obj != NULL) {
333         this->obj->Write(to);
334         fprintf(to, ".");
335     }
336     else if (this->clazz != NULL) {
337         fprintf(to, "%s.", this->clazz->QualifiedName().c_str());
338     }
339     fprintf(to, "%s(", this->name.c_str());
340     WriteArgumentList(to, this->arguments);
341     fprintf(to, ")");
342 }
343 
Comparison(Expression * l,const string & o,Expression * r)344 Comparison::Comparison(Expression* l, const string& o, Expression* r)
345     :lvalue(l),
346      op(o),
347      rvalue(r)
348 {
349 }
350 
~Comparison()351 Comparison::~Comparison()
352 {
353 }
354 
355 void
Write(FILE * to)356 Comparison::Write(FILE* to)
357 {
358     fprintf(to, "(");
359     this->lvalue->Write(to);
360     fprintf(to, "%s", this->op.c_str());
361     this->rvalue->Write(to);
362     fprintf(to, ")");
363 }
364 
NewExpression(Type * t)365 NewExpression::NewExpression(Type* t)
366     :type(t)
367 {
368 }
369 
~NewExpression()370 NewExpression::~NewExpression()
371 {
372 }
373 
374 void
Write(FILE * to)375 NewExpression::Write(FILE* to)
376 {
377     fprintf(to, "new %s(", this->type->InstantiableName().c_str());
378     WriteArgumentList(to, this->arguments);
379     fprintf(to, ")");
380 }
381 
NewArrayExpression(Type * t,Expression * s)382 NewArrayExpression::NewArrayExpression(Type* t, Expression* s)
383     :type(t),
384      size(s)
385 {
386 }
387 
~NewArrayExpression()388 NewArrayExpression::~NewArrayExpression()
389 {
390 }
391 
392 void
Write(FILE * to)393 NewArrayExpression::Write(FILE* to)
394 {
395     fprintf(to, "new %s[", this->type->QualifiedName().c_str());
396     size->Write(to);
397     fprintf(to, "]");
398 }
399 
Ternary()400 Ternary::Ternary()
401     :condition(NULL),
402      ifpart(NULL),
403      elsepart(NULL)
404 {
405 }
406 
Ternary(Expression * a,Expression * b,Expression * c)407 Ternary::Ternary(Expression* a, Expression* b, Expression* c)
408     :condition(a),
409      ifpart(b),
410      elsepart(c)
411 {
412 }
413 
~Ternary()414 Ternary::~Ternary()
415 {
416 }
417 
418 void
Write(FILE * to)419 Ternary::Write(FILE* to)
420 {
421     fprintf(to, "((");
422     this->condition->Write(to);
423     fprintf(to, ")?(");
424     this->ifpart->Write(to);
425     fprintf(to, "):(");
426     this->elsepart->Write(to);
427     fprintf(to, "))");
428 }
429 
Cast()430 Cast::Cast()
431     :type(NULL),
432      expression(NULL)
433 {
434 }
435 
Cast(Type * t,Expression * e)436 Cast::Cast(Type* t, Expression* e)
437     :type(t),
438      expression(e)
439 {
440 }
441 
~Cast()442 Cast::~Cast()
443 {
444 }
445 
446 void
Write(FILE * to)447 Cast::Write(FILE* to)
448 {
449     fprintf(to, "((%s)", this->type->QualifiedName().c_str());
450     expression->Write(to);
451     fprintf(to, ")");
452 }
453 
VariableDeclaration(Variable * l,Expression * r,Type * c)454 VariableDeclaration::VariableDeclaration(Variable* l, Expression* r, Type* c)
455     :lvalue(l),
456      cast(c),
457      rvalue(r)
458 {
459 }
460 
VariableDeclaration(Variable * l)461 VariableDeclaration::VariableDeclaration(Variable* l)
462     :lvalue(l),
463      cast(NULL),
464      rvalue(NULL)
465 {
466 }
467 
~VariableDeclaration()468 VariableDeclaration::~VariableDeclaration()
469 {
470 }
471 
472 void
Write(FILE * to)473 VariableDeclaration::Write(FILE* to)
474 {
475     this->lvalue->WriteDeclaration(to);
476     if (this->rvalue != NULL) {
477         fprintf(to, " = ");
478         if (this->cast != NULL) {
479             fprintf(to, "(%s)", this->cast->QualifiedName().c_str());
480         }
481         this->rvalue->Write(to);
482     }
483     fprintf(to, ";\n");
484 }
485 
IfStatement()486 IfStatement::IfStatement()
487     :expression(NULL),
488      statements(new StatementBlock),
489      elseif(NULL)
490 {
491 }
492 
~IfStatement()493 IfStatement::~IfStatement()
494 {
495 }
496 
497 void
Write(FILE * to)498 IfStatement::Write(FILE* to)
499 {
500     if (this->expression != NULL) {
501         fprintf(to, "if (");
502         this->expression->Write(to);
503         fprintf(to, ") ");
504     }
505     this->statements->Write(to);
506     if (this->elseif != NULL) {
507         fprintf(to, "else ");
508         this->elseif->Write(to);
509     }
510 }
511 
ReturnStatement(Expression * e)512 ReturnStatement::ReturnStatement(Expression* e)
513     :expression(e)
514 {
515 }
516 
~ReturnStatement()517 ReturnStatement::~ReturnStatement()
518 {
519 }
520 
521 void
Write(FILE * to)522 ReturnStatement::Write(FILE* to)
523 {
524     fprintf(to, "return ");
525     this->expression->Write(to);
526     fprintf(to, ";\n");
527 }
528 
TryStatement()529 TryStatement::TryStatement()
530     :statements(new StatementBlock)
531 {
532 }
533 
~TryStatement()534 TryStatement::~TryStatement()
535 {
536 }
537 
538 void
Write(FILE * to)539 TryStatement::Write(FILE* to)
540 {
541     fprintf(to, "try ");
542     this->statements->Write(to);
543 }
544 
CatchStatement(Variable * e)545 CatchStatement::CatchStatement(Variable* e)
546     :statements(new StatementBlock),
547      exception(e)
548 {
549 }
550 
~CatchStatement()551 CatchStatement::~CatchStatement()
552 {
553 }
554 
555 void
Write(FILE * to)556 CatchStatement::Write(FILE* to)
557 {
558     fprintf(to, "catch ");
559     if (this->exception != NULL) {
560         fprintf(to, "(");
561         this->exception->WriteDeclaration(to);
562         fprintf(to, ") ");
563     }
564     this->statements->Write(to);
565 }
566 
FinallyStatement()567 FinallyStatement::FinallyStatement()
568     :statements(new StatementBlock)
569 {
570 }
571 
~FinallyStatement()572 FinallyStatement::~FinallyStatement()
573 {
574 }
575 
576 void
Write(FILE * to)577 FinallyStatement::Write(FILE* to)
578 {
579     fprintf(to, "finally ");
580     this->statements->Write(to);
581 }
582 
Case()583 Case::Case()
584     :statements(new StatementBlock)
585 {
586 }
587 
Case(const string & c)588 Case::Case(const string& c)
589     :statements(new StatementBlock)
590 {
591     cases.push_back(c);
592 }
593 
~Case()594 Case::~Case()
595 {
596 }
597 
598 void
Write(FILE * to)599 Case::Write(FILE* to)
600 {
601     int N = this->cases.size();
602     if (N > 0) {
603         for (int i=0; i<N; i++) {
604             string s = this->cases[i];
605             if (s.length() != 0) {
606                 fprintf(to, "case %s:\n", s.c_str());
607             } else {
608                 fprintf(to, "default:\n");
609             }
610         }
611     } else {
612         fprintf(to, "default:\n");
613     }
614     statements->Write(to);
615 }
616 
SwitchStatement(Expression * e)617 SwitchStatement::SwitchStatement(Expression* e)
618     :expression(e)
619 {
620 }
621 
~SwitchStatement()622 SwitchStatement::~SwitchStatement()
623 {
624 }
625 
626 void
Write(FILE * to)627 SwitchStatement::Write(FILE* to)
628 {
629     fprintf(to, "switch (");
630     this->expression->Write(to);
631     fprintf(to, ")\n{\n");
632     int N = this->cases.size();
633     for (int i=0; i<N; i++) {
634         this->cases[i]->Write(to);
635     }
636     fprintf(to, "}\n");
637 }
638 
Method()639 Method::Method()
640     :ClassElement(),
641      modifiers(0),
642      returnType(NULL), // (NULL means constructor)
643      returnTypeDimension(0),
644      statements(NULL)
645 {
646 }
647 
~Method()648 Method::~Method()
649 {
650 }
651 
652 void
GatherTypes(set<Type * > * types) const653 Method::GatherTypes(set<Type*>* types) const
654 {
655     size_t N, i;
656 
657     if (this->returnType) {
658         types->insert(this->returnType);
659     }
660 
661     N = this->parameters.size();
662     for (i=0; i<N; i++) {
663         this->parameters[i]->GatherTypes(types);
664     }
665 
666     N = this->exceptions.size();
667     for (i=0; i<N; i++) {
668         types->insert(this->exceptions[i]);
669     }
670 }
671 
672 void
Write(FILE * to)673 Method::Write(FILE* to)
674 {
675     size_t N, i;
676 
677     if (this->comment.length() != 0) {
678         fprintf(to, "%s\n", this->comment.c_str());
679     }
680 
681     WriteModifiers(to, this->modifiers, SCOPE_MASK | STATIC | FINAL | OVERRIDE);
682 
683     if (this->returnType != NULL) {
684         string dim;
685         for (i=0; i<this->returnTypeDimension; i++) {
686             dim += "[]";
687         }
688         fprintf(to, "%s%s ", this->returnType->QualifiedName().c_str(),
689                 dim.c_str());
690     }
691 
692     fprintf(to, "%s(", this->name.c_str());
693 
694     N = this->parameters.size();
695     for (i=0; i<N; i++) {
696         this->parameters[i]->WriteDeclaration(to);
697         if (i != N-1) {
698             fprintf(to, ", ");
699         }
700     }
701 
702     fprintf(to, ")");
703 
704     N = this->exceptions.size();
705     for (i=0; i<N; i++) {
706         if (i == 0) {
707             fprintf(to, " throws ");
708         } else {
709             fprintf(to, ", ");
710         }
711         fprintf(to, "%s", this->exceptions[i]->QualifiedName().c_str());
712     }
713 
714     if (this->statements == NULL) {
715         fprintf(to, ";\n");
716     } else {
717         fprintf(to, "\n");
718         this->statements->Write(to);
719     }
720 }
721 
Class()722 Class::Class()
723     :modifiers(0),
724      what(CLASS),
725      type(NULL),
726      extends(NULL)
727 {
728 }
729 
~Class()730 Class::~Class()
731 {
732 }
733 
734 void
GatherTypes(set<Type * > * types) const735 Class::GatherTypes(set<Type*>* types) const
736 {
737     int N, i;
738 
739     types->insert(this->type);
740     if (this->extends != NULL) {
741         types->insert(this->extends);
742     }
743 
744     N = this->interfaces.size();
745     for (i=0; i<N; i++) {
746         types->insert(this->interfaces[i]);
747     }
748 
749     N = this->elements.size();
750     for (i=0; i<N; i++) {
751         this->elements[i]->GatherTypes(types);
752     }
753 }
754 
755 void
Write(FILE * to)756 Class::Write(FILE* to)
757 {
758     size_t N, i;
759 
760     if (this->comment.length() != 0) {
761         fprintf(to, "%s\n", this->comment.c_str());
762     }
763 
764     WriteModifiers(to, this->modifiers, ALL_MODIFIERS);
765 
766     if (this->what == Class::CLASS) {
767         fprintf(to, "class ");
768     } else {
769         fprintf(to, "interface ");
770     }
771 
772     string name = this->type->Name();
773     size_t pos = name.rfind('.');
774     if (pos != string::npos) {
775         name = name.c_str() + pos + 1;
776     }
777 
778     fprintf(to, "%s", name.c_str());
779 
780     if (this->extends != NULL) {
781         fprintf(to, " extends %s", this->extends->QualifiedName().c_str());
782     }
783 
784     N = this->interfaces.size();
785     if (N != 0) {
786         if (this->what == Class::CLASS) {
787             fprintf(to, " implements");
788         } else {
789             fprintf(to, " extends");
790         }
791         for (i=0; i<N; i++) {
792             fprintf(to, " %s", this->interfaces[i]->QualifiedName().c_str());
793         }
794     }
795 
796     fprintf(to, "\n");
797     fprintf(to, "{\n");
798 
799     N = this->elements.size();
800     for (i=0; i<N; i++) {
801         this->elements[i]->Write(to);
802     }
803 
804     fprintf(to, "}\n");
805 
806 }
807 
Document()808 Document::Document()
809 {
810 }
811 
~Document()812 Document::~Document()
813 {
814 }
815 
816 static string
escape_backslashes(const string & str)817 escape_backslashes(const string& str)
818 {
819     string result;
820     const size_t I=str.length();
821     for (size_t i=0; i<I; i++) {
822         char c = str[i];
823         if (c == '\\') {
824             result += "\\\\";
825         } else {
826             result += c;
827         }
828     }
829     return result;
830 }
831 
832 void
Write(FILE * to)833 Document::Write(FILE* to)
834 {
835     size_t N, i;
836 
837     if (this->comment.length() != 0) {
838         fprintf(to, "%s\n", this->comment.c_str());
839     }
840     fprintf(to, "/*\n"
841                 " * This file is auto-generated.  DO NOT MODIFY.\n"
842                 " * Original file: %s\n"
843                 " */\n", escape_backslashes(this->originalSrc).c_str());
844     if (this->package.length() != 0) {
845         fprintf(to, "package %s;\n", this->package.c_str());
846     }
847 
848     N = this->classes.size();
849     for (i=0; i<N; i++) {
850         Class* c = this->classes[i];
851         c->Write(to);
852     }
853 }
854 
855