]>
git.saurik.com Git - cycript.git/blob - Output.cpp
1e4d437045d77f3804fff807cfce8b1bd22be462
6 #include <objc/runtime.h>
9 _finline CYFlags
operator ~(CYFlags rhs
) {
10 return static_cast<CYFlags
>(~static_cast<unsigned>(rhs
));
13 _finline CYFlags
operator &(CYFlags lhs
, CYFlags rhs
) {
14 return static_cast<CYFlags
>(static_cast<unsigned>(lhs
) & static_cast<unsigned>(rhs
));
17 _finline CYFlags
operator |(CYFlags lhs
, CYFlags rhs
) {
18 return static_cast<CYFlags
>(static_cast<unsigned>(lhs
) | static_cast<unsigned>(rhs
));
21 _finline CYFlags
&operator |=(CYFlags
&lhs
, CYFlags rhs
) {
22 return lhs
= lhs
| rhs
;
25 _finline CYFlags
CYLeft(CYFlags flags
) {
26 return flags
& ~CYNoDangle
;
29 _finline CYFlags
CYRight(CYFlags flags
) {
30 return flags
& ~CYNoBF
;
33 _finline CYFlags
CYCenter(CYFlags flags
) {
34 return CYLeft(CYRight(flags
));
39 void CYOutput::Terminate() {
44 CYOutput
&CYOutput::operator <<(char rhs
) {
45 if (rhs
== ' ' || rhs
== '\n')
51 for (unsigned i(0); i
!= indent_
; ++i
)
60 if (mode_
== Terminated
&& rhs
!= '}')
70 } else if (rhs
== '+') {
74 } else if (rhs
== '-') {
75 if (mode_
== NoHyphen
)
78 } else if (WordEndRange_
[rhs
]) {
79 if (mode_
== NoLetter
)
90 CYOutput
&CYOutput::operator <<(const char *rhs
) {
91 size_t size(strlen(rhs
));
96 if (mode_
== Terminated
)
99 mode_
== NoPlus
&& *rhs
== '+' ||
100 mode_
== NoHyphen
&& *rhs
== '-' ||
101 mode_
== NoLetter
&& WordEndRange_
[*rhs
]
105 if (WordEndRange_
[rhs
[size
- 1]])
114 void OutputBody(CYOutput
&out
, CYStatement
*body
) {
115 out
<< ' ' << '{' << '\n';
123 void CYAddressOf::Output(CYOutput
&out
, CYFlags flags
) const {
124 rhs_
->Output(out
, 1, CYLeft(flags
));
128 void CYArgument::Output(CYOutput
&out
) const {
135 value_
->Output(out
, CYPA
, CYNoFlags
);
137 if (next_
->name_
== NULL
)
139 out
<< ' ' << *next_
;
143 void CYArray::Output(CYOutput
&out
, CYFlags flags
) const {
144 out
<< '[' << elements_
<< ']';
147 void CYArrayComprehension::Output(CYOutput
&out
, CYFlags flags
) const {
148 // XXX: I don't necc. need the ()s
149 out
<< "(function($cyv";
150 for (CYComprehension
*comprehension(comprehensions_
); comprehension
!= NULL
; comprehension
= comprehension
->next_
)
151 if (const char *name
= comprehension
->Name())
155 comprehensions_
->Output(out
);
157 expression_
->Output(out
, CYPA
, CYNoFlags
);
159 for (CYComprehension
*comprehension(comprehensions_
); comprehension
!= NULL
; comprehension
= comprehension
->next_
)
160 comprehension
->End_(out
);
161 out
<< "return $cyv;";
165 void CYAssignment::Output(CYOutput
&out
, CYFlags flags
) const {
166 lhs_
->Output(out
, Precedence() - 1, CYLeft(flags
) | CYNoRightHand
);
167 out
<< ' ' << Operator() << ' ';
168 rhs_
->Output(out
, Precedence(), CYRight(flags
));
171 void CYBlock::Output(CYOutput
&out
, CYFlags flags
) const {
172 statements_
->Single(out
, flags
);
175 void CYBoolean::Output(CYOutput
&out
, CYFlags flags
) const {
176 out
<< (Value() ? "true" : "false");
179 void CYBreak::Output(CYOutput
&out
, CYFlags flags
) const {
182 out
<< ' ' << *label_
;
186 void CYCall::Output(CYOutput
&out
, CYFlags flags
) const {
187 bool protect((flags
& CYNoCall
) != 0);
190 function_
->Output(out
, Precedence(), protect
? CYNoFlags
: flags
);
191 out
<< '(' << arguments_
<< ')';
196 void CYCatch::Output(CYOutput
&out
) const {
197 out
<< "catch" << ' ' << '(' << *name_
<< ')' << ' ' << '{';
199 code_
->Multiple(out
);
203 void CYCategory::Output(CYOutput
&out
, CYFlags flags
) const {
204 out
<< "(function($cys,$cyp,$cyc,$cyn,$cyt){";
205 out
<< "$cyp=object_getClass($cys);";
207 if (messages_
!= NULL
)
208 messages_
->Output(out
, true);
210 name_
->ClassName(out
, true);
215 void CYClass::Output(CYOutput
&out
, CYFlags flags
) const {
216 // XXX: I don't necc. need the ()s
217 out
<< "(function($cys,$cyp,$cyc,$cyn,$cyt,$cym){";
218 out
<< "$cyp=object_getClass($cys);";
219 out
<< "$cyc=objc_allocateClassPair($cys,";
221 name_
->ClassName(out
, false);
223 out
<< "$cyq(\"CY$\")";
225 out
<< "$cym=object_getClass($cyc);";
227 fields_
->Output(out
);
228 if (messages_
!= NULL
)
229 messages_
->Output(out
, false);
230 out
<< "objc_registerClassPair($cyc);";
231 out
<< "return $cyc;";
234 super_
->Output(out
, CYPA
, CYNoFlags
);
240 void CYClassExpression::Output(CYOutput
&out
, CYFlags flags
) const {
241 CYClass::Output(out
, flags
);
244 void CYClassStatement::Output(CYOutput
&out
, CYFlags flags
) const {
245 CYClass::Output(out
, flags
);
248 void CYCompound::Output(CYOutput
&out
, CYFlags flags
) const {
249 if (CYExpression
*expression
= expressions_
)
250 if (CYExpression
*next
= expression
->next_
) {
251 expression
->Output(out
, CYLeft(flags
));
252 CYFlags
center(CYCenter(flags
));
253 while (next
!= NULL
) {
256 next
= expression
->next_
;
257 CYFlags
right(next
!= NULL
? center
: CYRight(flags
));
258 expression
->Output(out
, right
);
261 expression
->Output(out
, flags
);
264 void CYComprehension::Output(CYOutput
&out
) const {
269 void CYCondition::Output(CYOutput
&out
, CYFlags flags
) const {
270 test_
->Output(out
, Precedence() - 1, CYLeft(flags
));
271 out
<< ' ' << '?' << ' ';
273 true_
->Output(out
, CYPA
, CYNoFlags
);
274 out
<< ' ' << ':' << ' ';
275 false_
->Output(out
, CYPA
, CYRight(flags
));
278 void CYContinue::Output(CYOutput
&out
, CYFlags flags
) const {
281 out
<< ' ' << *label_
;
285 void CYClause::Output(CYOutput
&out
) const {
287 out
<< "case" << ' ' << *case_
;
292 code_
->Multiple(out
, CYNoFlags
);
296 const char *CYDeclaration::ForEachIn() const {
297 return identifier_
->Value();
300 void CYDeclaration::ForIn(CYOutput
&out
, CYFlags flags
) const {
302 Output(out
, CYRight(flags
));
305 void CYDeclaration::ForEachIn(CYOutput
&out
) const {
309 void CYDeclaration::Output(CYOutput
&out
, CYFlags flags
) const {
311 if (initialiser_
!= NULL
) {
312 out
<< ' ' << '=' << ' ';
313 initialiser_
->Output(out
, CYPA
, CYRight(flags
));
317 void CYDeclarations::For(CYOutput
&out
) const {
322 void CYDeclarations::Output(CYOutput
&out
) const {
323 Output(out
, CYNoFlags
);
326 void CYDeclarations::Output(CYOutput
&out
, CYFlags flags
) const {
327 const CYDeclarations
*declaration(this);
330 CYDeclarations
*next(declaration
->next_
);
331 CYFlags
jacks(first
? CYLeft(flags
) : next
== NULL
? CYRight(flags
) : CYCenter(flags
));
333 declaration
->declaration_
->Output(out
, jacks
);
342 void CYDirectMember::Output(CYOutput
&out
, CYFlags flags
) const {
343 object_
->Output(out
, Precedence(), CYLeft(flags
));
344 if (const char *word
= property_
->Word())
347 out
<< '[' << *property_
<< ']';
350 void CYDoWhile::Output(CYOutput
&out
, CYFlags flags
) const {
352 code_
->Single(out
, CYCenter(flags
));
353 out
<< "while" << ' ' << '(' << *test_
<< ')';
356 void CYElement::Output(CYOutput
&out
) const {
358 value_
->Output(out
, CYPA
, CYNoFlags
);
359 if (next_
!= NULL
|| value_
== NULL
) {
361 if (next_
!= NULL
&& next_
->value_
!= NULL
)
368 void CYEmpty::Output(CYOutput
&out
, CYFlags flags
) const {
372 void CYExpress::Output(CYOutput
&out
, CYFlags flags
) const {
373 expression_
->Output(out
, flags
| CYNoBF
);
377 void CYExpression::ClassName(CYOutput
&out
, bool object
) const {
378 Output(out
, CYPA
, CYNoFlags
);
381 const char *CYExpression::ForEachIn() const {
385 void CYExpression::For(CYOutput
&out
) const {
389 void CYExpression::ForEachIn(CYOutput
&out
) const {
390 Output(out
, CYPA
, CYNoRightHand
);
393 void CYExpression::ForIn(CYOutput
&out
, CYFlags flags
) const {
394 Output(out
, flags
| CYNoRightHand
);
397 void CYExpression::Output(CYOutput
&out
) const {
398 Output(out
, CYNoFlags
);
401 void CYExpression::Output(CYOutput
&out
, unsigned precedence
, CYFlags flags
) const {
402 if (precedence
< Precedence() || (flags
& CYNoRightHand
) != 0 && RightHand())
403 out
<< '(' << *this << ')';
408 void CYField::Output(CYOutput
&out
) const {
412 void CYFinally::Output(CYOutput
&out
) const {
413 out
<< "finally" << ' ' << '{';
415 code_
->Multiple(out
);
419 void CYFor::Output(CYOutput
&out
, CYFlags flags
) const {
420 out
<< "for" << ' ' << '(';
421 if (initialiser_
!= NULL
)
422 initialiser_
->For(out
);
428 code_
->Single(out
, CYRight(flags
));
431 void CYForEachIn::Output(CYOutput
&out
, CYFlags flags
) const {
432 out
<< "with({$cys:0,$cyt:0}){";
435 set_
->Output(out
, CYPA
, CYNoFlags
);
438 out
<< "for($cyt in $cys){";
440 initialiser_
->ForEachIn(out
);
441 out
<< "=$cys[$cyt];";
443 code_
->Multiple(out
);
450 void CYForEachInComprehension::Begin_(CYOutput
&out
) const {
451 out
<< "(function($cys){";
453 set_
->Output(out
, CYPA
, CYNoFlags
);
456 out
<< "for(" << *name_
<< " in $cys){";
457 out
<< *name_
<< "=$cys[" << *name_
<< "];";
460 void CYForEachInComprehension::End_(CYOutput
&out
) const {
464 void CYForIn::Output(CYOutput
&out
, CYFlags flags
) const {
465 out
<< "for" << ' ' << '(';
466 initialiser_
->ForIn(out
, CYNoIn
);
467 out
<< "in" << *set_
<< ')';
468 code_
->Single(out
, CYRight(flags
));
471 void CYForInComprehension::Begin_(CYOutput
&out
) const {
472 out
<< "for" << ' ' << '(' << *name_
<< "in" << *set_
<< ')';
475 void CYFunction::Output(CYOutput
&out
, CYFlags flags
) const {
476 // XXX: one could imagine using + here to save a byte
477 bool protect((flags
& CYNoFunction
) != 0);
482 out
<< ' ' << *name_
;
483 out
<< '(' << parameters_
<< ')';
484 OutputBody(out
, body_
);
489 void CYFunctionExpression::Output(CYOutput
&out
, CYFlags flags
) const {
490 CYFunction::Output(out
, flags
);
493 void CYFunctionStatement::Output(CYOutput
&out
, CYFlags flags
) const {
494 CYFunction::Output(out
, flags
);
497 void CYFunctionParameter::Output(CYOutput
&out
) const {
500 out
<< ',' << ' ' << *next_
;
503 void CYIf::Output(CYOutput
&out
, CYFlags flags
) const {
505 if (false_
== NULL
&& (flags
& CYNoDangle
) != 0) {
510 out
<< "if" << ' ' << '(' << *test_
<< ')';
512 CYFlags
right(protect
? CYNoFlags
: CYRight(flags
));
514 CYFlags
jacks(CYNoDangle
);
518 jacks
|= protect
? CYNoFlags
: CYCenter(flags
);
520 bool single(true_
->Single(out
, jacks
));
522 if (false_
!= NULL
) {
523 out
<< (single
? '\t' : ' ');
525 false_
->Single(out
, right
);
532 void CYIfComprehension::Begin_(CYOutput
&out
) const {
533 out
<< "if" << '(' << *test_
<< ')';
536 void CYIndirect::Output(CYOutput
&out
, CYFlags flags
) const {
537 rhs_
->Output(out
, 1, CYLeft(flags
));
541 void CYIndirectMember::Output(CYOutput
&out
, CYFlags flags
) const {
542 object_
->Output(out
, Precedence(), CYLeft(flags
));
544 if (const char *word
= property_
->Word())
547 out
<< '[' << *property_
<< ']';
550 void CYInfix::Output(CYOutput
&out
, CYFlags flags
) const {
551 const char *name(Operator());
552 bool protect((flags
& CYNoIn
) != 0 && strcmp(name
, "in") == 0);
555 CYFlags
left(protect
? CYNoFlags
: CYLeft(flags
));
556 lhs_
->Output(out
, Precedence(), left
);
557 out
<< ' ' << name
<< ' ';
558 CYFlags
right(protect
? CYNoFlags
: CYRight(flags
));
559 rhs_
->Output(out
, Precedence() - 1, right
);
564 void CYLet::Output(CYOutput
&out
, CYFlags flags
) const {
565 out
<< "let" << ' ' << '(' << *declarations_
<< ')' << ' ' << '{';
566 if (statements_
!= NULL
)
567 statements_
->Multiple(out
);
571 void CYMessage::Output(CYOutput
&out
, bool replace
) const {
573 next_
->Output(out
, replace
);
574 out
<< "$cyn=new Selector(\"";
575 for (CYMessageParameter
*parameter(parameter_
); parameter
!= NULL
; parameter
= parameter
->next_
)
576 if (parameter
->tag_
!= NULL
) {
577 out
<< *parameter
->tag_
;
578 if (parameter
->name_
!= NULL
)
582 out
<< "$cyt=$cyn.type($cy" << (instance_
? 's' : 'p') << ");";
583 out
<< "class_" << (replace
? "replace" : "add") << "Method($cy" << (instance_
? 'c' : 'm') << ",$cyn,";
584 out
<< "new Functor(function(self,_cmd";
585 for (CYMessageParameter
*parameter(parameter_
); parameter
!= NULL
; parameter
= parameter
->next_
)
586 if (parameter
->name_
!= NULL
)
587 out
<< ',' << *parameter
->name_
;
588 out
<< "){return function(){";
590 body_
->Multiple(out
);
591 out
<< "}.call(self);},$cyt),$cyt);";
594 void CYNew::Output(CYOutput
&out
, CYFlags flags
) const {
596 CYFlags
jacks(CYNoCall
| CYCenter(flags
));
597 constructor_
->Output(out
, Precedence(), jacks
);
598 if (arguments_
!= NULL
)
599 out
<< '(' << *arguments_
<< ')';
602 void CYNull::Output(CYOutput
&out
, CYFlags flags
) const {
606 void CYNumber::Output(CYOutput
&out
, CYFlags flags
) const {
608 sprintf(value
, "%.17g", Value());
612 void CYNumber::PropertyName(CYOutput
&out
) const {
613 Output(out
, CYNoFlags
);
616 void CYObject::Output(CYOutput
&out
, CYFlags flags
) const {
617 bool protect((flags
& CYNoBrace
) != 0);
629 void CYPostfix::Output(CYOutput
&out
, CYFlags flags
) const {
630 lhs_
->Output(out
, Precedence(), CYLeft(flags
));
634 void CYPrefix::Output(CYOutput
&out
, CYFlags flags
) const {
635 const char *name(Operator());
639 rhs_
->Output(out
, Precedence(), CYRight(flags
));
642 void CYProperty::Output(CYOutput
&out
) const {
644 name_
->PropertyName(out
);
646 value_
->Output(out
, CYPA
, CYNoFlags
);
648 out
<< ',' << '\n' << *next_
;
653 void CYRegEx::Output(CYOutput
&out
, CYFlags flags
) const {
657 void CYReturn::Output(CYOutput
&out
, CYFlags flags
) const {
660 out
<< ' ' << *value_
;
664 void CYSelector::Output(CYOutput
&out
, CYFlags flags
) const {
665 out
<< "new Selector(\"";
671 void CYSelectorPart::Output(CYOutput
&out
) const {
678 void CYSend::Output(CYOutput
&out
, CYFlags flags
) const {
679 out
<< "objc_msgSend(";
680 self_
->Output(out
, CYPA
, CYNoFlags
);
682 std::ostringstream name
;
683 for (CYArgument
*argument(arguments_
); argument
!= NULL
; argument
= argument
->next_
)
684 if (argument
->name_
!= NULL
) {
685 name
<< *argument
->name_
;
686 if (argument
->value_
!= NULL
)
689 out
.out_
<< reinterpret_cast<void *>(sel_registerName(name
.str().c_str()));
690 for (CYArgument
*argument(arguments_
); argument
!= NULL
; argument
= argument
->next_
)
691 if (argument
->value_
!= NULL
) {
693 argument
->value_
->Output(out
, CYPA
, CYNoFlags
);
698 void CYStatement::Multiple(CYOutput
&out
, CYFlags flags
) const {
700 for (const CYStatement
*next(this); next
!= NULL
; next
= next
->next_
) {
701 bool last(next
->next_
== NULL
);
702 CYFlags
jacks(first
? last
? flags
: CYLeft(flags
) : last
? CYCenter(flags
) : CYRight(flags
));
705 next
->Output(out
, jacks
);
710 bool CYStatement::Single(CYOutput
&out
, CYFlags flags
) const {
712 out
<< ' ' << '{' << '\n';
719 for (CYLabel
*label(labels_
); label
!= NULL
; label
= label
->next_
)
720 out
<< ' ' << *label
->name_
<< ':';
731 void CYString::Output(CYOutput
&out
, CYFlags flags
) const {
732 unsigned quot(0), apos(0);
733 for (const char *value(value_
), *end(value_
+ size_
); value
!= end
; ++value
)
736 else if (*value
== '\'')
739 bool single(quot
> apos
);
741 std::ostringstream str
;
743 str
<< (single
? '\'' : '"');
744 for (const char *value(value_
), *end(value_
+ size_
); value
!= end
; ++value
)
746 case '\\': str
<< "\\\\"; break;
747 case '\b': str
<< "\\b"; break;
748 case '\f': str
<< "\\f"; break;
749 case '\n': str
<< "\\n"; break;
750 case '\r': str
<< "\\r"; break;
751 case '\t': str
<< "\\t"; break;
752 case '\v': str
<< "\\v"; break;
767 if (*value
< 0x20 || *value
>= 0x7f)
768 str
<< "\\x" << std::setbase(16) << std::setw(2) << std::setfill('0') << unsigned(*value
);
772 str
<< (single
? '\'' : '"');
774 out
<< str
.str().c_str();
777 void CYString::PropertyName(CYOutput
&out
) const {
778 if (const char *word
= Word())
784 static const char *Reserved_
[] = {
785 "false", "null", "true",
787 "break", "case", "catch", "continue", "default",
788 "delete", "do", "else", "finally", "for", "function",
789 "if", "in", "instanceof", "new", "return", "switch",
790 "this", "throw", "try", "typeof", "var", "void",
795 "class", "enum", "export", "extends", "import", "super",
797 "abstract", "boolean", "byte", "char", "double", "final",
798 "float", "goto", "int", "long", "native", "short",
799 "synchronized", "throws", "transient", "volatile",
808 const char *CYString::Word() const {
809 if (size_
== 0 || !WordStartRange_
[value_
[0]])
811 for (size_t i(1); i
!= size_
; ++i
)
812 if (!WordEndRange_
[value_
[i
]])
814 const char *value(Value());
815 for (const char **reserved(Reserved_
); *reserved
!= NULL
; ++reserved
)
816 if (strcmp(*reserved
, value
) == 0)
821 void CYSwitch::Output(CYOutput
&out
, CYFlags flags
) const {
822 out
<< "switch" << ' ' << '(' << *value_
<< ')' << ' ' << '{';
827 void CYThis::Output(CYOutput
&out
, CYFlags flags
) const {
831 void CYThrow::Output(CYOutput
&out
, CYFlags flags
) const {
834 out
<< ' ' << *value_
;
838 void CYTry::Output(CYOutput
&out
, CYFlags flags
) const {
839 out
<< "try" << ' ' << '{';
841 code_
->Multiple(out
);
847 void CYVar::Output(CYOutput
&out
, CYFlags flags
) const {
849 declarations_
->Output(out
, flags
);
853 void CYVariable::Output(CYOutput
&out
, CYFlags flags
) const {
857 void CYWhile::Output(CYOutput
&out
, CYFlags flags
) const {
858 out
<< "while" << '(' << *test_
<< ')';
859 code_
->Single(out
, CYRight(flags
));
862 void CYWith::Output(CYOutput
&out
, CYFlags flags
) const {
863 out
<< "with" << '(' << *scope_
<< ')';
864 code_
->Single(out
, CYRight(flags
));
867 void CYWord::ClassName(CYOutput
&out
, bool object
) const {
869 out
<< "objc_getClass(";
870 out
<< '"' << Value() << '"';
875 void CYWord::Output(CYOutput
&out
) const {
879 void CYWord::PropertyName(CYOutput
&out
) const {