source: trunk/Decompiler.class.php @ 746

Last change on this file since 746 was 746, checked in by moo, 4 years ago

Decompiler: fix ticks, str

  • Property svn:eol-style set to native
File size: 49.8 KB
Line 
1<?php
2
3define('INDENT', "\t");
4ini_set('error_reporting', E_ALL);
5
6function color($str, $color = 33)
7{
8    return "\x1B[{$color}m$str\x1B[0m";
9}
10
11function str($code, $indent = '') // {{{
12{
13    if (is_array($code)) {
14        $array = array();
15        foreach ($code as $key => $value) {
16            $array[$key] = str($value, $indent);
17        }
18        return $array;
19    }
20    if (is_object($code)) {
21        $code = foldToCode($code, $indent);
22        return $code->toCode($indent);
23    }
24
25    return (string) $code;
26}
27// }}}
28function foldToCode($src, $indent = '') // {{{ wrap or rewrap anything to Decompiler_Code
29{
30    if (is_array($indent)) {
31        $indent = $indent['indent'];
32    }
33
34    if (!is_object($src)) {
35        return new Decompiler_Code($src);
36    }
37
38    if (!method_exists($src, 'toCode')) {
39        var_dump($src);
40        exit('no toCode');
41    }
42    if (get_class($src) != 'Decompiler_Code') {
43        // rewrap it
44        $src = new Decompiler_Code($src->toCode($indent));
45    }
46
47    return $src;
48}
49// }}}
50function value($value) // {{{
51{
52    $spec = xcache_get_special_value($value);
53    if (isset($spec)) {
54        $value = $spec;
55        if (!is_array($value)) {
56            // constant
57            return $value;
58        }
59    }
60
61    if (is_a($value, 'Decompiler_Object')) {
62        // use as is
63    }
64    else if (is_array($value)) {
65        $value = new Decompiler_ConstArray($value);
66    }
67    else {
68        $value = new Decompiler_Value($value);
69    }
70    return $value;
71}
72// }}}
73class Decompiler_Object // {{{
74{
75}
76// }}}
77class Decompiler_Value extends Decompiler_Object // {{{
78{
79    var $value;
80
81    function Decompiler_Value($value = null)
82    {
83        $this->value = $value;
84    }
85
86    function toCode($indent)
87    {
88        return var_export($this->value, true);
89    }
90}
91// }}}
92class Decompiler_Code extends Decompiler_Object // {{{
93{
94    var $src;
95
96    function Decompiler_Code($src)
97    {
98        assert('isset($src)');
99        $this->src = $src;
100    }
101
102    function toCode($indent)
103    {
104        return $this->src;
105    }
106}
107// }}}
108class Decompiler_Binop extends Decompiler_Code // {{{
109{
110    var $opc;
111    var $op1;
112    var $op2;
113    var $parent;
114    var $indent;
115
116    function Decompiler_Binop($parent, $op1, $opc, $op2)
117    {
118        $this->parent = &$parent;
119        $this->opc = $opc;
120        $this->op1 = $op1;
121        $this->op2 = $op2;
122    }
123
124    function toCode($indent)
125    {
126        $op1 = foldToCode($this->op1, $indent);
127        if (is_a($this->op1, 'Decompiler_Binop') && $this->op1->opc != $this->opc) {
128            $op1 = "($op1)";
129        }
130        $opstr = $this->parent->binops[$this->opc];
131        if ($op1 == '0' && $this->opc == XC_SUB) {
132            return $opstr . str($this->op2, $indent);
133        }
134        return str($op1) . ' ' . $opstr . ' ' . str($this->op2, $indent);
135    }
136}
137// }}}
138class Decompiler_Fetch extends Decompiler_Code // {{{
139{
140    var $src;
141    var $fetchType;
142
143    function Decompiler_Fetch($src, $type, $globalsrc)
144    {
145        $this->src = $src;
146        $this->fetchType = $type;
147        $this->globalsrc = $globalsrc;
148    }
149
150    function toCode($indent)
151    {
152        switch ($this->fetchType) {
153        case ZEND_FETCH_LOCAL:
154            return '$' . substr($this->src, 1, -1);
155        case ZEND_FETCH_STATIC:
156            die('static fetch cant to string');
157        case ZEND_FETCH_GLOBAL:
158        case ZEND_FETCH_GLOBAL_LOCK:
159            return $this->globalsrc;
160        default:
161            var_dump($this->fetchType);
162            assert(0);
163        }
164    }
165}
166// }}}
167class Decompiler_Box // {{{
168{
169    var $obj;
170
171    function Decompiler_Box(&$obj)
172    {
173        $this->obj = &$obj;
174    }
175
176    function toCode($indent)
177    {
178        return $this->obj->toCode($indent);
179    }
180}
181// }}}
182class Decompiler_Dim extends Decompiler_Value // {{{
183{
184    var $offsets = array();
185    var $isLast = false;
186    var $assign = null;
187
188    function toCode($indent)
189    {
190        if (is_a($this->value, 'Decompiler_ListBox')) {
191            $exp = foldToCode($this->value->obj->src, $indent);
192        }
193        else {
194            $exp = foldToCode($this->value, $indent);
195        }
196        foreach ($this->offsets as $dim) {
197            $exp .= '[' . foldToCode($dim, $indent) . ']';
198        }
199        return $exp;
200    }
201}
202// }}}
203class Decompiler_DimBox extends Decompiler_Box // {{{
204{
205}
206// }}}
207class Decompiler_List extends Decompiler_Code // {{{
208{
209    var $src;
210    var $dims = array();
211    var $everLocked = false;
212
213    function toCode($indent)
214    {
215        if (count($this->dims) == 1 && !$this->everLocked) {
216            $dim = $this->dims[0];
217            unset($dim->value);
218            $dim->value = $this->src;
219            if (!isset($dim->assign)) {
220                return foldToCode($dim, $indent);
221            }
222            return foldToCode($this->dims[0]->assign, $indent) . ' = ' . foldToCode($dim, $indent);
223        }
224        /* flatten dims */
225        $assigns = array();
226        foreach ($this->dims as $dim) {
227            $assign = &$assigns;
228            foreach ($dim->offsets as $offset) {
229                $assign = &$assign[$offset];
230            }
231            $assign = foldToCode($dim->assign, $indent);
232        }
233        return $this->toList($assigns) . ' = ' . foldToCode($this->src, $indent);
234    }
235
236    function toList($assigns)
237    {
238        $keys = array_keys($assigns);
239        if (count($keys) < 2) {
240            $keys[] = 0;
241        }
242        $max = call_user_func_array('max', $keys);
243        $list = 'list(';
244        for ($i = 0; $i <= $max; $i ++) {
245            if ($i) {
246                $list .= ', ';
247            }
248            if (!isset($assigns[$i])) {
249                continue;
250            }
251            if (is_array($assigns[$i])) {
252                $list .= $this->toList($assigns[$i]);
253            }
254            else {
255                $list .= $assigns[$i];
256            }
257        }
258        return $list . ')';
259    }
260}
261// }}}
262class Decompiler_ListBox extends Decompiler_Box // {{{
263{
264}
265// }}}
266class Decompiler_Array extends Decompiler_Value // {{{
267{
268    // emenets
269    function Decompiler_Array()
270    {
271        $this->value = array();
272    }
273
274    function toCode($indent)
275    {
276        $subindent = $indent . INDENT;
277
278        $elementsCode = array();
279        $index = 0;
280        foreach ($this->value as $element) {
281            list($key, $value) = $element;
282            if (!isset($key)) {
283                $key = $index++;
284            }
285            $elementsCode[] = array(str($key, $subindent), str($value, $subindent), $key, $value);
286        }
287
288        $exp = "array(";
289        $indent = $indent . INDENT;
290        $assocWidth = 0;
291        $multiline = 0;
292        $i = 0;
293        foreach ($elementsCode as $element) {
294            list($keyCode, $valueCode) = $element;
295            if ((string) $i !== $keyCode) {
296                $assocWidth = 1;
297                break;
298            }
299            ++$i;
300        }
301        foreach ($elementsCode as $element) {
302            list($keyCode, $valueCode, $key, $value) = $element;
303            if ($assocWidth) {
304                $len = strlen($keyCode);
305                if ($assocWidth < $len) {
306                    $assocWidth = $len;
307                }
308            }
309            if (is_array($value) || is_a($value, 'Decompiler_Array')) {
310                $multiline ++;
311            }
312        }
313
314        $i = 0;
315        foreach ($elementsCode as $element) {
316            list($keyCode, $value) = $element;
317            if ($multiline) {
318                if ($i) {
319                    $exp .= ",";
320                }
321                $exp .= "\n";
322                $exp .= $indent;
323            }
324            else {
325                if ($i) {
326                    $exp .= ", ";
327                }
328            }
329
330            if ($assocWidth) {
331                if ($multiline) {
332                    $exp .= sprintf("%-{$assocWidth}s => ", $keyCode);
333                }
334                else {
335                    $exp .= $keyCode . ' => ';
336                }
337            }
338
339            $exp .= $value;
340
341            $i ++;
342        }
343        if ($multiline) {
344            $exp .= "\n$indent)";
345        }
346        else {
347            $exp .= ")";
348        }
349        return $exp;
350    }
351}
352// }}}
353class Decompiler_ConstArray extends Decompiler_Array // {{{
354{
355    function Decompiler_ConstArray($array)
356    {
357        $elements = array();
358        foreach ($array as $key => $value) {
359            $elements[] = array(value($key), value($value));
360        }
361        $this->value = $elements;
362    }
363}
364// }}}
365class Decompiler_ForeachBox extends Decompiler_Box // {{{
366{
367    var $iskey;
368
369    function toCode($indent)
370    {
371        return 'foreach (' . '';
372    }
373}
374// }}}
375
376class Decompiler
377{
378    var $rName = '!^[\\w_][\\w\\d_]*$!';
379    var $rQuotedName = "!^'[\\w_][\\w\\d_]*'\$!";
380
381    function Decompiler()
382    {
383        // {{{ opinfo
384        $this->unaryops = array(
385                XC_BW_NOT   => '~',
386                XC_BOOL_NOT => '!',
387                );
388        $this->binops = array(
389                XC_ADD                 => "+",
390                XC_ASSIGN_ADD          => "+=",
391                XC_SUB                 => "-",
392                XC_ASSIGN_SUB          => "-=",
393                XC_MUL                 => "*",
394                XC_ASSIGN_MUL          => "*=",
395                XC_DIV                 => "/",
396                XC_ASSIGN_DIV          => "/=",
397                XC_MOD                 => "%",
398                XC_ASSIGN_MOD          => "%=",
399                XC_SL                  => "<<",
400                XC_ASSIGN_SL           => "<<=",
401                XC_SR                  => ">>",
402                XC_ASSIGN_SR           => ">>=",
403                XC_CONCAT              => ".",
404                XC_ASSIGN_CONCAT       => ".=",
405                XC_IS_IDENTICAL        => "===",
406                XC_IS_NOT_IDENTICAL    => "!==",
407                XC_IS_EQUAL            => "==",
408                XC_IS_NOT_EQUAL        => "!=",
409                XC_IS_SMALLER          => "<",
410                XC_IS_SMALLER_OR_EQUAL => "<=",
411                XC_BW_OR               => "|",
412                XC_ASSIGN_BW_OR        => "|=",
413                XC_BW_AND              => "&",
414                XC_ASSIGN_BW_AND       => "&=",
415                XC_BW_XOR              => "^",
416                XC_ASSIGN_BW_XOR       => "^=",
417                XC_BOOL_XOR            => "xor",
418                );
419        // }}}
420        $this->includeTypes = array( // {{{
421                ZEND_EVAL         => 'eval',
422                ZEND_INCLUDE      => 'include',
423                ZEND_INCLUDE_ONCE => 'include_once',
424                ZEND_REQUIRE      => 'require',
425                ZEND_REQUIRE_ONCE => 'require_once',
426                );
427                // }}}
428    }
429    function outputPhp(&$opcodes, $opline, $last, $indent) // {{{
430    {
431        $origindent = $indent;
432        $curticks = 0;
433        for ($i = $opline; $i <= $last; $i ++) {
434            $op = $opcodes[$i];
435            if (isset($op['php'])) {
436                $toticks = isset($op['ticks']) ? (int) str($op['ticks']) : 0;
437                if ($curticks != $toticks) {
438                    $oldticks = $curticks;
439                    $curticks = $toticks;
440                    if (!$curticks) {
441                        echo $origindent, "}\n";
442                        $indent = $origindent;
443                    }
444                    else {
445                        if ($oldticks) {
446                            echo $origindent, "}\n";
447                        }
448                        else if (!$oldticks) {
449                            $indent .= INDENT;
450                        }
451                        echo $origindent, "declare (ticks=$curticks) {\n";
452                    }
453                }
454                echo $indent, str($op['php'], $indent), ";\n";
455            }
456        }
457        if ($curticks) {
458            echo $origindent, "}\n";
459        }
460    }
461    // }}}
462    function getOpVal($op, &$EX, $tostr = true, $free = false) // {{{
463    {
464        switch ($op['op_type']) {
465        case XC_IS_CONST:
466            return foldToCode(value($op['constant']), $EX);
467
468        case XC_IS_VAR:
469        case XC_IS_TMP_VAR:
470            $T = &$EX['Ts'];
471            $ret = $T[$op['var']];
472            if ($tostr) {
473                $ret = foldToCode($ret, $EX);
474            }
475            if ($free) {
476                unset($T[$op['var']]);
477            }
478            return $ret;
479
480        case XC_IS_CV:
481            $var = $op['var'];
482            $var = $EX['op_array']['vars'][$var];
483            return '$' . $var['name'];
484
485        case XC_IS_UNUSED:
486            return null;
487        }
488    }
489    // }}}
490    function removeKeyPrefix($array, $prefix) // {{{
491    {
492        $prefixLen = strlen($prefix);
493        $ret = array();
494        foreach ($array as $key => $value) {
495            if (substr($key, 0, $prefixLen) == $prefix) {
496                $key = substr($key, $prefixLen);
497            }
498            $ret[$key] = $value;
499        }
500        return $ret;
501    }
502    // }}}
503    function &fixOpcode($opcodes, $removeTailing = false, $defaultReturnValue = null) // {{{
504    {
505        for ($i = 0, $cnt = count($opcodes); $i < $cnt; $i ++) {
506            if (function_exists('xcache_get_fixed_opcode')) {
507                $opcodes[$i]['opcode'] = xcache_get_fixed_opcode($opcodes[$i]['opcode'], $i);
508            }
509            if (isset($opcodes[$i]['op1'])) {
510                $opcodes[$i]['op1'] = $this->removeKeyPrefix($opcodes[$i]['op1'], 'u.');
511                $opcodes[$i]['op2'] = $this->removeKeyPrefix($opcodes[$i]['op2'], 'u.');
512                $opcodes[$i]['result'] = $this->removeKeyPrefix($opcodes[$i]['result'], 'u.');
513            }
514            else {
515                $op = array(
516                    'op1' => array(),
517                    'op2' => array(),
518                    'op3' => array(),
519                );
520                foreach ($opcodes[$i] as $name => $value) {
521                    if (preg_match('!^(op1|op2|result)\\.(.*)!', $name, $m)) {
522                        list(, $which, $field) = $m;
523                        $op[$which][$field] = $value;
524                    }
525                    else if (preg_match('!^(op1|op2|result)_type$!', $name, $m)) {
526                        list(, $which) = $m;
527                        $op[$which]['op_type'] = $value;
528                    }
529                    else {
530                        $op[$name] = $value;
531                    }
532                }
533                $opcodes[$i] = $op;
534            }
535        }
536
537        if ($removeTailing) {
538            $last = count($opcodes) - 1;
539            if ($opcodes[$last]['opcode'] == XC_HANDLE_EXCEPTION) {
540                unset($opcodes[$last]);
541                --$last;
542            }
543            if ($opcodes[$last]['opcode'] == XC_RETURN) {
544                $op1 = $opcodes[$last]['op1'];
545                if ($op1['op_type'] == XC_IS_CONST && array_key_exists('constant', $op1) && $op1['constant'] === $defaultReturnValue) {
546                    unset($opcodes[$last]);
547                    --$last;
548                }
549            }
550        }
551        return $opcodes;
552    }
553    // }}}
554    function &dop_array($op_array, $indent = '') // {{{
555    {
556        $op_array['opcodes'] = $this->fixOpcode($op_array['opcodes'], true, $indent == '' ? 1 : null);
557        $opcodes = &$op_array['opcodes'];
558        $EX['indent'] = '';
559        // {{{ build jmp array
560        for ($i = 0, $cnt = count($opcodes); $i < $cnt; $i ++) {
561            $op = &$opcodes[$i];
562            /*
563            if ($op['opcode'] == XC_JMPZ) {
564                $this->dumpop($op, $EX);
565                var_dump($op);
566            }
567            continue;
568            */
569            $op['line'] = $i;
570            switch ($op['opcode']) {
571            case XC_JMP:
572                $target = $op['op1']['var'];
573                $op['jmpouts'] = array($target);
574                $opcodes[$target]['jmpins'][] = $i;
575                break;
576
577            case XC_JMPZNZ:
578                $jmpz = $op['op2']['opline_num'];
579                $jmpnz = $op['extended_value'];
580                $op['jmpouts'] = array($jmpz, $jmpnz);
581                $opcodes[$jmpz]['jmpins'][] = $i;
582                $opcodes[$jmpnz]['jmpins'][] = $i;
583                break;
584
585            case XC_JMPZ:
586            case XC_JMPNZ:
587            case XC_JMPZ_EX:
588            case XC_JMPNZ_EX:
589            // case XC_FE_RESET:
590            case XC_FE_FETCH:
591            // case XC_JMP_NO_CTOR:
592                $target = $op['op2']['opline_num'];
593                //if (!isset($target)) {
594                //  $this->dumpop($op, $EX);
595                //  var_dump($op); exit;
596                //}
597                $op['jmpouts'] = array($target);
598                $opcodes[$target]['jmpins'][] = $i;
599                break;
600
601            /*
602            case XC_RETURN:
603                $op['jmpouts'] = array();
604                break;
605            */
606            }
607        }
608        unset($op);
609        // }}}
610        // build semi-basic blocks
611        $nextbbs = array();
612        $starti = 0;
613        for ($i = 1, $cnt = count($opcodes); $i < $cnt; $i ++) {
614            if (isset($opcodes[$i]['jmpins'])
615             || isset($opcodes[$i - 1]['jmpouts'])) {
616                $nextbbs[$starti] = $i;
617                $starti = $i;
618            }
619        }
620        $nextbbs[$starti] = $cnt;
621
622        $EX = array();
623        $EX['Ts'] = array();
624        $EX['indent'] = $indent;
625        $EX['nextbbs'] = $nextbbs;
626        $EX['op_array'] = &$op_array;
627        $EX['opcodes'] = &$opcodes;
628        // func call
629        $EX['object'] = null;
630        $EX['called_scope'] = null;
631        $EX['fbc'] = null;
632        $EX['argstack'] = array();
633        $EX['arg_types_stack'] = array();
634        $EX['last'] = count($opcodes) - 1;
635        $EX['silence'] = 0;
636
637        for ($next = 0, $last = $EX['last'];
638                $loop = $this->outputCode($EX, $next, $last, $indent, true);
639                list($next, $last) = $loop) {
640            // empty
641        }
642        return $EX;
643    }
644    // }}}
645    function outputCode(&$EX, $opline, $last, $indent, $loop = false) // {{{
646    {
647        $op = &$EX['opcodes'][$opline];
648        $next = $EX['nextbbs'][$opline];
649
650        $end = $next - 1;
651        if ($end > $last) {
652            $end = $last;
653        }
654
655        if (isset($op['jmpins'])) {
656            echo "\nline", $op['line'], ":\n";
657        }
658        else {
659            // echo ";;;\n";
660        }
661        $this->dasmBasicBlock($EX, $opline, $end);
662        $this->outputPhp($EX['opcodes'], $opline, $end, $indent);
663        // jmpout op
664        $op = &$EX['opcodes'][$end];
665        $op1 = $op['op1'];
666        $op2 = $op['op2'];
667        $ext = $op['extended_value'];
668        $line = $op['line'];
669
670        if (isset($EX['opcodes'][$next])) {
671            if (isset($last) && $next > $last) {
672                $next = null;
673            }
674        }
675        else {
676            $next = null;
677        }
678        if ($op['opcode'] == XC_FE_FETCH) {
679            $opline = $next;
680            $next = $op['op2']['opline_num'];
681            $end = $next - 1;
682
683            ob_start();
684            $this->outputCode($EX, $opline, $end /* - 1 skip last jmp */, $indent . INDENT);
685            $body = ob_get_clean();
686
687            $as = foldToCode($op['fe_as'], $EX);
688            if (isset($op['fe_key'])) {
689                $as = str($op['fe_key'], $EX) . ' => ' . str($as);
690            }
691            echo "{$indent}foreach (" . str($op['fe_src'], $EX) . " as $as) {\n";
692            echo $body;
693            echo "{$indent}}";
694            // $this->outputCode($EX, $next, $last, $indent);
695            // return;
696        }
697        /*
698        if ($op['opcode'] == XC_JMPZ) {
699            $target = $op2['opline_num'];
700            if ($line + 1) {
701                $nextblock = $EX['nextbbs'][$next];
702                $jmpop = end($nextblock);
703                if ($jmpop['opcode'] == XC_JMP) {
704                    $ifendline = $op2['opline_num'];
705                    if ($ifendline >= $line) {
706                        $cond = $op['cond'];
707                        echo "{$indent}if ($cond) {\n";
708                        $this->outputCode($EX, $next, $last, INDENT . $indent);
709                        echo "$indent}\n";
710                        $this->outputCode($EX, $target, $last, $indent);
711                        return;
712                    }
713                }
714            }
715        }
716        */
717        if (!isset($next)) {
718            return;
719        }
720        if (!empty($op['jmpouts']) && isset($op['isjmp'])) {
721            if (isset($op['cond'])) {
722                echo "{$indent}check (" . str($op["cond"]) . ") {\n";
723                echo INDENT;
724            }
725            echo $indent;
726            echo xcache_get_opcode($op['opcode']), ' line', $op['jmpouts'][0];
727            if (isset($op['jmpouts'][1])) {
728                echo ', line', $op['jmpouts'][1];
729            }
730            echo ";";
731            // echo ' // <- line', $op['line'];
732            echo "\n";
733            if (isset($op['cond'])) echo "$indent}\n";
734        }
735
736        // proces JMPZ_EX/JMPNZ_EX for AND,OR
737        $op = &$EX['opcodes'][$next];
738        /*
739        if (isset($op['jmpins'])) {
740            foreach (array_reverse($op['jmpins']) as $fromline) {
741                $fromop = $EX['opcodes'][$fromline];
742                switch ($fromop['opcode']) {
743                case XC_JMPZ_EX: $opstr = 'and'; break;
744                case XC_JMPNZ_EX: $opstr = 'or'; break;
745                case XC_JMPZNZ: var_dump($fromop); exit;
746                default: continue 2;
747                }
748
749                $var = $fromop['result']['var'];
750                var_dump($EX['Ts'][$var]);
751                $EX['Ts'][$var] = '(' . $fromop['and_or'] . " $opstr " . $EX['Ts'][$var] . ')';
752            }
753            #$this->outputCode($EX, $next, $last, $indent);
754            #return;
755        }
756        */
757        if (isset($op['cond_false'])) {
758            // $this->dumpop($op, $EX);
759            // any true comes here, so it's a "or"
760            $cond = implode(' and ', str($op['cond_false']));
761            // var_dump($op['cond'] = $cond);
762            /*
763            $rvalue = implode(' or ', $op['cond_true']) . ' or ' . $rvalue;
764            unset($op['cond_true']);
765            */
766        }
767
768        if ($loop) {
769            return array($next, $last);
770        }
771        $this->outputCode($EX, $next, $last, $indent);
772    }
773    // }}}
774    function unquoteName($str) // {{{
775    {
776        $str = str($str);
777        if (preg_match($this->rQuotedName, $str)) {
778            $str = substr($str, 1, -1);
779        }
780        return $str;
781    }
782    // }}}
783    function dasmBasicBlock(&$EX, $opline, $last) // {{{
784    {
785        $T = &$EX['Ts'];
786        $opcodes = &$EX['opcodes'];
787        $lastphpop = null;
788
789        for ($i = $opline, $ic = $last + 1; $i < $ic; $i ++) {
790            // {{{ prepair
791            $op = &$opcodes[$i];
792            $opc = $op['opcode'];
793            if ($opc == XC_NOP) {
794                continue;
795            }
796
797            $op1 = $op['op1'];
798            $op2 = $op['op2'];
799            $res = $op['result'];
800            $ext = $op['extended_value'];
801
802            $opname = xcache_get_opcode($opc);
803
804            if ($opname == 'UNDEF' || !isset($opname)) {
805                echo 'UNDEF OP:';
806                $this->dumpop($op, $EX);
807                continue;
808            }
809            // $this->dumpop($op, $EX); //var_dump($op);
810
811            $resvar = null;
812            if ((ZEND_ENGINE_2_4 ? ($res['op_type'] & EXT_TYPE_UNUSED) : ($res['EA.type'] & EXT_TYPE_UNUSED)) || $res['op_type'] == XC_IS_UNUSED) {
813                $istmpres = false;
814            }
815            else {
816                $istmpres = true;
817            }
818            // }}}
819            // echo $opname, "\n";
820
821            $call = array(&$this, $opname);
822            if (is_callable($call)) {
823                $this->{$opname}($op, $EX);
824            }
825            else if (isset($this->binops[$opc])) { // {{{
826                $op1val = $this->getOpVal($op1, $EX, false);
827                $op2val = $this->getOpVal($op2, $EX, false);
828                $rvalue = new Decompiler_Binop($this, $op1val, $opc, $op2val);
829                $resvar = $rvalue;
830                // }}}
831            }
832            else if (isset($this->unaryops[$opc])) { // {{{
833                $op1val = $this->getOpVal($op1, $EX);
834                $myop = $this->unaryops[$opc];
835                $rvalue = "$myop$op1val";
836                $resvar = $rvalue;
837                // }}}
838            }
839            else {
840                switch ($opc) {
841                case XC_NEW: // {{{
842                    array_push($EX['arg_types_stack'], array($EX['fbc'], $EX['object'], $EX['called_scope']));
843                    $EX['object'] = (int) $res['var'];
844                    $EX['called_scope'] = null;
845                    $EX['fbc'] = 'new ' . $this->unquoteName($this->getOpVal($op1, $EX));
846                    if (!ZEND_ENGINE_2) {
847                        $resvar = '$new object$';
848                    }
849                    break;
850                    // }}}
851                case XC_FETCH_CLASS: // {{{
852                    if ($op2['op_type'] == XC_IS_UNUSED) {
853                        switch (($ext & (defined('ZEND_FETCH_CLASS_MASK') ? ZEND_FETCH_CLASS_MASK : 0xFF))) {
854                        case ZEND_FETCH_CLASS_SELF:
855                            $class = 'self';
856                            break;
857                        case ZEND_FETCH_CLASS_PARENT:
858                            $class = 'parent';
859                            break;
860                        case ZEND_FETCH_CLASS_STATIC:
861                            $class = 'static';
862                            break;
863                        }
864                        $istmpres = true;
865                    }
866                    else {
867                        $class = $op2['constant'];
868                        if (is_object($class)) {
869                            $class = get_class($class);
870                        }
871                    }
872                    $resvar = $class;
873                    break;
874                    // }}}
875                case XC_FETCH_CONSTANT: // {{{
876                    if ($op1['op_type'] == XC_IS_CONST) {
877                        $resvar = $op1['constant'];
878                    }
879                    else if ($op1['op_type'] == XC_IS_UNUSED) {
880                        $resvar = $op2['constant'];
881                    }
882                    else {
883                        $class = $T[$op1['var']];
884                        assert($class[0] == 'class');
885                        $resvar = $class[1] . '::' . $op2['constant'];
886                    }
887                    break;
888                    // }}}
889                    // {{{ case XC_FETCH_*
890                case XC_FETCH_R:
891                case XC_FETCH_W:
892                case XC_FETCH_RW:
893                case XC_FETCH_FUNC_ARG:
894                case XC_FETCH_UNSET:
895                case XC_FETCH_IS:
896                case XC_UNSET_VAR:
897                    $rvalue = $this->getOpVal($op1, $EX);
898                    if (defined('ZEND_FETCH_TYPE_MASK')) {
899                        $fetchtype = ($ext & ZEND_FETCH_TYPE_MASK);
900                    }
901                    else {
902                        $fetchtype = $op2[!ZEND_ENGINE_2 ? 'fetch_type' : 'EA.type'];
903                    }
904                    switch ($fetchtype) {
905                    case ZEND_FETCH_STATIC_MEMBER:
906                        $class = $this->getOpVal($op2, $EX);
907                        $rvalue = $class . '::$' . $this->unquoteName($rvalue);
908                        break;
909                    default:
910                        $name = $this->unquoteName($rvalue);
911                        $globalname = xcache_is_autoglobal($name) ? "\$$name" : "\$GLOBALS[$rvalue]";
912                        $rvalue = new Decompiler_Fetch($rvalue, $fetchtype, $globalname);
913                        break;
914                    }
915                    if ($opc == XC_UNSET_VAR) {
916                        $op['php'] = "unset(" . str($rvalue, $EX) . ")";
917                        $lastphpop = &$op;
918                    }
919                    else if ($res['op_type'] != XC_IS_UNUSED) {
920                        $resvar = $rvalue;
921                    }
922                    break;
923                    // }}}
924                    // {{{ case XC_FETCH_DIM_*
925                case XC_FETCH_DIM_TMP_VAR:
926                case XC_FETCH_DIM_R:
927                case XC_FETCH_DIM_W:
928                case XC_FETCH_DIM_RW:
929                case XC_FETCH_DIM_FUNC_ARG:
930                case XC_FETCH_DIM_UNSET:
931                case XC_FETCH_DIM_IS:
932                case XC_ASSIGN_DIM:
933                case XC_UNSET_DIM:
934                case XC_UNSET_DIM_OBJ:
935                    $src = $this->getOpVal($op1, $EX, false);
936                    if (is_a($src, "Decompiler_ForeachBox")) {
937                        $src->iskey = $this->getOpVal($op2, $EX);
938                        $resvar = $src;
939                        break;
940                    }
941                    else if (is_a($src, "Decompiler_DimBox")) {
942                        $dimbox = $src;
943                    }
944                    else {
945                        if (!is_a($src, "Decompiler_ListBox")) {
946                            $list = new Decompiler_List($this->getOpVal($op1, $EX, false));
947
948                            $src = new Decompiler_ListBox($list);
949                            if (!isset($op1['var'])) {
950                                $this->dumpop($op, $EX);
951                                var_dump($op);
952                                die('missing var');
953                            }
954                            $T[$op1['var']] = $src;
955                            unset($list);
956                        }
957                        $dim = new Decompiler_Dim($src);
958                        $src->obj->dims[] = &$dim;
959
960                        $dimbox = new Decompiler_DimBox($dim);
961                    }
962                    $dim = &$dimbox->obj;
963                    $dim->offsets[] = $this->getOpVal($op2, $EX);
964                    if ($ext == ZEND_FETCH_ADD_LOCK) {
965                        $src->obj->everLocked = true;
966                    }
967                    else if ($ext == ZEND_FETCH_STANDARD) {
968                        $dim->isLast = true;
969                    }
970                    unset($dim);
971                    $rvalue = $dimbox;
972
973                    if ($opc == XC_ASSIGN_DIM) {
974                        $lvalue = $rvalue;
975                        ++ $i;
976                        $rvalue = $this->getOpVal($opcodes[$i]['op1'], $EX);
977                        $resvar = str($lvalue, $EX) . ' = ' . $rvalue;
978                    }
979                    else if ($opc == XC_UNSET_DIM) {
980                        $op['php'] = "unset(" . str($rvalue, $EX) . ")";
981                        $lastphpop = &$op;
982                    }
983                    else if ($res['op_type'] != XC_IS_UNUSED) {
984                        $resvar = $rvalue;
985                    }
986                    break;
987                    // }}}
988                case XC_ASSIGN: // {{{
989                    $lvalue = $this->getOpVal($op1, $EX);
990                    $rvalue = $this->getOpVal($op2, $EX, false);
991                    if (is_a($rvalue, 'Decompiler_ForeachBox')) {
992                        $type = $rvalue->iskey ? 'fe_key' : 'fe_as';
993                        $rvalue->obj[$type] = $lvalue;
994                        unset($T[$op2['var']]);
995                        break;
996                    }
997                    if (is_a($rvalue, "Decompiler_DimBox")) {
998                        $dim = &$rvalue->obj;
999                        $dim->assign = $lvalue;
1000                        if ($dim->isLast) {
1001                            $resvar = foldToCode($dim->value, $EX);
1002                        }
1003                        unset($dim);
1004                        break;
1005                    }
1006                    $resvar = "$lvalue = " . str($rvalue, $EX);
1007                    break;
1008                    // }}}
1009                case XC_ASSIGN_REF: // {{{
1010                    $lvalue = $this->getOpVal($op1, $EX);
1011                    $rvalue = $this->getOpVal($op2, $EX, false);
1012                    if (is_a($rvalue, 'Decompiler_Fetch')) {
1013                        $src = foldToCode($rvalue->src, $EX);
1014                        if (substr($src, 1, -1) == substr($lvalue, 1)) {
1015                            switch ($rvalue->fetchType) {
1016                            case ZEND_FETCH_GLOBAL:
1017                            case ZEND_FETCH_GLOBAL_LOCK:
1018                                $resvar = 'global ' . $lvalue;
1019                                break 2;
1020                            case ZEND_FETCH_STATIC:
1021                                $statics = &$EX['op_array']['static_variables'];
1022                                $resvar = 'static ' . $lvalue;
1023                                $name = substr($src, 1, -1);
1024                                if (isset($statics[$name])) {
1025                                    $var = $statics[$name];
1026                                    $resvar .= ' = ';
1027                                    $resvar .= str(value($var), $EX);
1028                                }
1029                                unset($statics);
1030                                break 2;
1031                            default:
1032                            }
1033                        }
1034                    }
1035                    // TODO: PHP_6 global
1036                    $rvalue = foldToCode($rvalue, $EX);
1037                    $resvar = "$lvalue = &$rvalue";
1038                    break;
1039                    // }}}
1040                // {{{ case XC_FETCH_OBJ_*
1041                case XC_FETCH_OBJ_R:
1042                case XC_FETCH_OBJ_W:
1043                case XC_FETCH_OBJ_RW:
1044                case XC_FETCH_OBJ_FUNC_ARG:
1045                case XC_FETCH_OBJ_UNSET:
1046                case XC_FETCH_OBJ_IS:
1047                case XC_ASSIGN_OBJ:
1048                    $obj = $this->getOpVal($op1, $EX);
1049                    if (!isset($obj)) {
1050                        $obj = '$this';
1051                    }
1052                    $prop = $this->getOpVal($op2, $EX);
1053                    if (preg_match($this->rQuotedName, $prop)) {
1054                        $prop = substr($prop, 1, -1);;
1055                        $rvalue = "{$obj}->$prop";
1056                    }
1057                    else {
1058                        $rvalue = "{$obj}->{" . "$prop}";
1059                    }
1060                    if ($res['op_type'] != XC_IS_UNUSED) {
1061                        $resvar = $rvalue;
1062                    }
1063                    if ($opc == XC_ASSIGN_OBJ) {
1064                        ++ $i;
1065                        $lvalue = $rvalue;
1066                        $rvalue = $this->getOpVal($opcodes[$i]['op1'], $EX);
1067                        $resvar = "$lvalue = $rvalue";
1068                    }
1069                    break;
1070                    // }}}
1071                case XC_ISSET_ISEMPTY_DIM_OBJ:
1072                case XC_ISSET_ISEMPTY_PROP_OBJ:
1073                case XC_ISSET_ISEMPTY:
1074                case XC_ISSET_ISEMPTY_VAR: // {{{
1075                    if ($opc == XC_ISSET_ISEMPTY_VAR) {
1076                        $rvalue = $this->getOpVal($op1, $EX);;
1077                        if (preg_match($this->rQuotedName, $rvalue)) {
1078                            $rvalue = '$' . substr($rvalue, 1, -1);
1079                        }
1080                        else {
1081                            $rvalue = '${' . $rvalue . '}';
1082                        }
1083                        if ($op2['EA.type'] == ZEND_FETCH_STATIC_MEMBER) {
1084                            $class = $this->getOpVal($op2, $EX);
1085                            $rvalue = $class . '::' . $rvalue;
1086                        }
1087                    }
1088                    else if ($opc == XC_ISSET_ISEMPTY) {
1089                        $rvalue = $this->getOpVal($op1, $EX);
1090                    }
1091                    else {
1092                        $container = $this->getOpVal($op1, $EX);
1093                        $dim = $this->getOpVal($op2, $EX);
1094                        if ($opc == XC_ISSET_ISEMPTY_PROP_OBJ) {
1095                            if (preg_match($this->rQuotedName, $dim)) {
1096                                $rvalue = $container . "->" . substr($dim, 1, -1);
1097                            }
1098                            else {
1099                                $rvalue = $container . "->{" . $dim . "}";
1100                            }
1101                        }
1102                        else {
1103                            $rvalue = $container . "[$dim]";
1104                        }
1105                    }
1106
1107                    switch ((!ZEND_ENGINE_2 ? $op['op2']['var'] /* constant */ : $ext) & (ZEND_ISSET|ZEND_ISEMPTY)) {
1108                    case ZEND_ISSET:
1109                        $rvalue = "isset($rvalue)";
1110                        break;
1111                    case ZEND_ISEMPTY:
1112                        $rvalue = "empty($rvalue)";
1113                        break;
1114                    }
1115                    $resvar = $rvalue;
1116                    break;
1117                    // }}}
1118                case XC_SEND_VAR_NO_REF:
1119                case XC_SEND_VAL:
1120                case XC_SEND_REF:
1121                case XC_SEND_VAR: // {{{
1122                    $ref = ($opc == XC_SEND_REF ? '&' : '');
1123                    $EX['argstack'][] = $ref . str($this->getOpVal($op1, $EX));
1124                    break;
1125                    // }}}
1126                case XC_INIT_STATIC_METHOD_CALL:
1127                case XC_INIT_METHOD_CALL:
1128                case XC_INIT_FCALL_BY_FUNC:
1129                case XC_INIT_FCALL_BY_NAME: // {{{
1130                    if (($ext & ZEND_CTOR_CALL)) {
1131                        break;
1132                    }
1133                    array_push($EX['arg_types_stack'], array($EX['fbc'], $EX['object'], $EX['called_scope']));
1134                    if ($opc == XC_INIT_STATIC_METHOD_CALL || $opc == XC_INIT_METHOD_CALL || $op1['op_type'] != XC_IS_UNUSED) {
1135                        $obj = $this->getOpVal($op1, $EX);
1136                        if (!isset($obj)) {
1137                            $obj = '$this';
1138                        }
1139                        if ($opc == XC_INIT_STATIC_METHOD_CALL || /* PHP4 */ isset($op1['constant'])) {
1140                            $EX['object'] = null;
1141                            $EX['called_scope'] = $this->unquoteName($obj);
1142                        }
1143                        else {
1144                            $EX['object'] = $obj;
1145                            $EX['called_scope'] = null;
1146                        }
1147                        if ($res['op_type'] != XC_IS_UNUSED) {
1148                            $resvar = '$obj call$';
1149                        }
1150                    }
1151                    else {
1152                        $EX['object'] = null;
1153                        $EX['called_scope'] = null;
1154                    }
1155
1156                    if ($opc == XC_INIT_FCALL_BY_FUNC) {
1157                        $which = $op1['var'];
1158                        $EX['fbc'] = $EX['op_array']['funcs'][$which]['name'];
1159                    }
1160                    else {
1161                        $EX['fbc'] = $this->getOpVal($op2, $EX, false);
1162                    }
1163                    break;
1164                    // }}}
1165                case XC_DO_FCALL_BY_FUNC:
1166                    $which = $op1['var'];
1167                    $fname = $EX['op_array']['funcs'][$which]['name'];
1168                    $args = $this->popargs($EX, $ext);
1169                    $resvar = $fname . "($args)";
1170                    break;
1171                case XC_DO_FCALL:
1172                    $fname = $this->unquoteName($this->getOpVal($op1, $EX, false));
1173                    $args = $this->popargs($EX, $ext);
1174                    $resvar = $fname . "($args)";
1175                    break;
1176                case XC_DO_FCALL_BY_NAME: // {{{
1177                    $object = null;
1178
1179                    $fname = $this->unquoteName($EX['fbc']);
1180                    if (!is_int($EX['object'])) {
1181                        $object = $EX['object'];
1182                    }
1183
1184                    $args = $this->popargs($EX, $ext);
1185
1186                    $resvar =
1187                        (isset($object) ? $object . '->' : '' )
1188                        . (isset($EX['called_scope']) ? $EX['called_scope'] . '::' : '' )
1189                        . $fname . "($args)";
1190                    unset($args);
1191
1192                    if (is_int($EX['object'])) {
1193                        $T[$EX['object']] = $resvar;
1194                        $resvar = null;
1195                    }
1196                    list($EX['fbc'], $EX['object'], $EX['called_scope']) = array_pop($EX['arg_types_stack']);
1197                    break;
1198                    // }}}
1199                case XC_VERIFY_ABSTRACT_CLASS: // {{{
1200                    //unset($T[$op1['var']]);
1201                    break;
1202                    // }}}
1203                case XC_DECLARE_CLASS: 
1204                case XC_DECLARE_INHERITED_CLASS:
1205                case XC_DECLARE_INHERITED_CLASS_DELAYED: // {{{
1206                    $key = $op1['constant'];
1207                    if (!isset($this->dc['class_table'][$key])) {
1208                        echo 'class not found: ', $key, 'existing classes are:', "\n";
1209                        var_dump(array_keys($this->dc['class_table']));
1210                        exit;
1211                    }
1212                    $class = &$this->dc['class_table'][$key];
1213                    $class['name'] = $this->unquoteName($this->getOpVal($op2, $EX));
1214                    if ($opc == XC_DECLARE_INHERITED_CLASS || $opc == XC_DECLARE_INHERITED_CLASS_DELAYED) {
1215                        $ext /= XC_SIZEOF_TEMP_VARIABLE;
1216                        $class['parent'] = $T[$ext];
1217                        unset($T[$ext]);
1218                    }
1219                    else {
1220                        $class['parent'] = null;
1221                    }
1222
1223                    while ($i + 2 < $ic
1224                     && $opcodes[$i + 2]['opcode'] == XC_ADD_INTERFACE
1225                     && $opcodes[$i + 2]['op1']['var'] == $res['var']
1226                     && $opcodes[$i + 1]['opcode'] == XC_FETCH_CLASS) {
1227                        $fetchop = &$opcodes[$i + 1];
1228                        $impl = $this->unquoteName($this->getOpVal($fetchop['op2'], $EX));
1229                        $addop = &$opcodes[$i + 2];
1230                        $class['interfaces'][$addop['extended_value']] = $impl;
1231                        unset($fetchop, $addop);
1232                        $i += 2;
1233                    }
1234                    $this->dclass($class);
1235                    unset($class);
1236                    break;
1237                    // }}}
1238                case XC_INIT_STRING: // {{{
1239                    $resvar = "''";
1240                    break;
1241                    // }}}
1242                case XC_ADD_CHAR:
1243                case XC_ADD_STRING:
1244                case XC_ADD_VAR: // {{{
1245                    $op1val = $this->getOpVal($op1, $EX);
1246                    $op2val = $this->getOpVal($op2, $EX);
1247                    switch ($opc) {
1248                    case XC_ADD_CHAR:
1249                        $op2val = foldToCode(chr($op2val), $EX);
1250                        break;
1251                    case XC_ADD_STRING:
1252                        $op2val = foldToCode($op2val, $EX);
1253                        break;
1254                    case XC_ADD_VAR:
1255                        break;
1256                    }
1257                    if ($op1val == "''") {
1258                        $rvalue = $op2val;
1259                    }
1260                    else if ($op2val == "''") {
1261                        $rvalue = $op1val;
1262                    }
1263                    else {
1264                        $rvalue = str($op1val) . ' . ' . str($op2val);
1265                    }
1266                    $resvar = $rvalue;
1267                    // }}}
1268                    break;
1269                case XC_PRINT: // {{{
1270                    $op1val = $this->getOpVal($op1, $EX);
1271                    $resvar = "print($op1val)";
1272                    break;
1273                    // }}}
1274                case XC_ECHO: // {{{
1275                    $op1val = $this->getOpVal($op1, $EX);
1276                    $resvar = "echo " . str($op1val);
1277                    break;
1278                    // }}}
1279                case XC_EXIT: // {{{
1280                    $op1val = $this->getOpVal($op1, $EX);
1281                    $resvar = "exit($op1val)";
1282                    break;
1283                    // }}}
1284                case XC_INIT_ARRAY:
1285                case XC_ADD_ARRAY_ELEMENT: // {{{
1286                    $rvalue = $this->getOpVal($op1, $EX, false, true);
1287
1288                    if ($opc == XC_ADD_ARRAY_ELEMENT) {
1289                        $assoc = $this->getOpVal($op2, $EX);
1290                        if (isset($assoc)) {
1291                            $T[$res['var']]->value[] = array($assoc, $rvalue);
1292                        }
1293                        else {
1294                            $T[$res['var']]->value[] = array(null, $rvalue);
1295                        }
1296                    }
1297                    else {
1298                        if ($opc == XC_INIT_ARRAY) {
1299                            $resvar = new Decompiler_Array();
1300                            if (!isset($rvalue)) {
1301                                continue;
1302                            }
1303                        }
1304
1305                        $assoc = $this->getOpVal($op2, $EX);
1306                        if (isset($assoc)) {
1307                            $resvar->value[] = array($assoc, $rvalue);
1308                        }
1309                        else {
1310                            $resvar->value[] = array(null, $rvalue);
1311                        }
1312                    }
1313                    break;
1314                    // }}}
1315                case XC_QM_ASSIGN: // {{{
1316                    $resvar = $this->getOpVal($op1, $EX);
1317                    break;
1318                    // }}}
1319                case XC_BOOL: // {{{
1320                    $resvar = /*'(bool) ' .*/ $this->getOpVal($op1, $EX);
1321                    break;
1322                    // }}}
1323                case XC_RETURN: // {{{
1324                    $resvar = "return " . str($this->getOpVal($op1, $EX));
1325                    break;
1326                    // }}}
1327                case XC_INCLUDE_OR_EVAL: // {{{
1328                    $type = $op2['var']; // hack
1329                    $keyword = $this->includeTypes[$type];
1330                    $resvar = "$keyword(" . str($this->getOpVal($op1, $EX)) . ")";
1331                    break;
1332                    // }}}
1333                case XC_FE_RESET: // {{{
1334                    $resvar = $this->getOpVal($op1, $EX);
1335                    break;
1336                    // }}}
1337                case XC_FE_FETCH: // {{{
1338                    $op['fe_src'] = $this->getOpVal($op1, $EX);
1339                    $fe = new Decompiler_ForeachBox($op);
1340                    $fe->iskey = false;
1341                    $T[$res['var']] = $fe;
1342
1343                    ++ $i;
1344                    if (($ext & ZEND_FE_FETCH_WITH_KEY)) {
1345                        $fe = new Decompiler_ForeachBox($op);
1346                        $fe->iskey = true;
1347
1348                        $res = $opcodes[$i]['result'];
1349                        $T[$res['var']] = $fe;
1350                    }
1351                    break;
1352                    // }}}
1353                case XC_SWITCH_FREE: // {{{
1354                    // unset($T[$op1['var']]);
1355                    break;
1356                    // }}}
1357                case XC_FREE: // {{{
1358                    $free = $T[$op1['var']];
1359                    if (!is_a($free, 'Decompiler_Array') && !is_a($free, 'Decompiler_Box')) {
1360                        $op['php'] = is_object($free) ? $free : $this->unquote($free, '(', ')');
1361                        $lastphpop = &$op;
1362                    }
1363                    unset($T[$op1['var']], $free);
1364                    break;
1365                    // }}}
1366                case XC_JMP_NO_CTOR:
1367                    break;
1368                case XC_JMPNZ: // while
1369                case XC_JMPZNZ: // for
1370                case XC_JMPZ_EX: // and
1371                case XC_JMPNZ_EX: // or
1372                case XC_JMPZ: // {{{
1373                    if ($opc == XC_JMP_NO_CTOR && $EX['object']) {
1374                        $rvalue = $EX['object'];
1375                    }
1376                    else {
1377                        $rvalue = $this->getOpVal($op1, $EX);
1378                    }
1379
1380                    if (isset($op['cond_true'])) {
1381                        // any true comes here, so it's a "or"
1382                        $rvalue = implode(' or ', $op['cond_true']) . ' or ' . $rvalue;
1383                        unset($op['cond_true']);
1384                    }
1385                    if (isset($op['cond_false'])) {
1386                        echo "TODO(cond_false):\n";
1387                        var_dump($op);// exit;
1388                    }
1389                    if ($opc == XC_JMPZ_EX || $opc == XC_JMPNZ_EX || $opc == XC_JMPZ) {
1390                        $targetop = &$EX['opcodes'][$op2['opline_num']];
1391                        if ($opc == XC_JMPNZ_EX) {
1392                            $targetop['cond_true'][] = foldToCode($rvalue, $EX);
1393                        }
1394                        else {
1395                            $targetop['cond_false'][] = foldToCode($rvalue, $EX);
1396                        }
1397                        unset($targetop);
1398                    }
1399                    else {
1400                        $op['cond'] = $rvalue; 
1401                        $op['isjmp'] = true;
1402                    }
1403                    break;
1404                    // }}}
1405                case XC_JMP: // {{{
1406                    $op['cond'] = null;
1407                    $op['isjmp'] = true;
1408                    break;
1409                    // }}}
1410                case XC_CASE:
1411                    $switchValue = $this->getOpVal($op1, $EX);
1412                    $caseValue = $this->getOpVal($op2, $EX);
1413                    $resvar = str($switchValue) . ' == ' . str($caseValue);
1414                    break;
1415                case XC_BRK:
1416                    break;
1417                case XC_RECV_INIT:
1418                case XC_RECV:
1419                    $offset = $this->getOpVal($op1, $EX);
1420                    $lvalue = $this->getOpVal($op['result'], $EX);
1421                    if ($opc == XC_RECV_INIT) {
1422                        $default = value($op['op2']['constant']);
1423                    }
1424                    else {
1425                        $default = null;
1426                    }
1427                    $EX['recvs'][str($offset)] = array($lvalue, $default);
1428                    break;
1429                case XC_POST_DEC:
1430                case XC_POST_INC:
1431                case XC_POST_DEC_OBJ:
1432                case XC_POST_INC_OBJ:
1433                case XC_PRE_DEC:
1434                case XC_PRE_INC:
1435                case XC_PRE_DEC_OBJ:
1436                case XC_PRE_INC_OBJ: // {{{
1437                    $flags = array_flip(explode('_', $opname));
1438                    if (isset($flags['OBJ'])) {
1439                        $resvar = $this->getOpVal($op1, $EX);
1440                        $prop = $this->unquoteName($this->getOpVal($op2, $EX));
1441                        if ($prop{0} == '$') {
1442                            $resvar = $resvar . "{" . $prop . "}";
1443                        }
1444                        else {
1445                            $resvar = $resvar . "->" . $prop;
1446                        }
1447                    }
1448                    else {
1449                        $resvar = $this->getOpVal($op1, $EX);
1450                    }
1451                    $opstr = isset($flags['DEC']) ? '--' : '++';
1452                    if (isset($flags['POST'])) {
1453                        $resvar .= ' ' . $opstr;
1454                    }
1455                    else {
1456                        $resvar = "$opstr $resvar";
1457                    }
1458                    break;
1459                    // }}}
1460
1461                case XC_BEGIN_SILENCE: // {{{
1462                    $EX['silence'] ++;
1463                    break;
1464                    // }}}
1465                case XC_END_SILENCE: // {{{
1466                    $EX['silence'] --;
1467                    $lastresvar = '@' . str($lastresvar, $EX);
1468                    break;
1469                    // }}}
1470                case XC_CONT: // {{{
1471                    break;
1472                    // }}}
1473                case XC_CAST: // {{{
1474                    $type = $ext;
1475                    static $type2cast = array(
1476                            IS_LONG   => '(int)',
1477                            IS_DOUBLE => '(double)',
1478                            IS_STRING => '(string)',
1479                            IS_ARRAY  => '(array)',
1480                            IS_OBJECT => '(object)',
1481                            IS_BOOL   => '(bool)',
1482                            IS_NULL   => '(unset)',
1483                            );
1484                    assert(isset($type2cast[$type]));
1485                    $cast = $type2cast[$type];
1486                    $resvar = $cast . ' ' . $this->getOpVal($op1, $EX);
1487                    break;
1488                    // }}}
1489                case XC_EXT_STMT:
1490                case XC_EXT_FCALL_BEGIN:
1491                case XC_EXT_FCALL_END:
1492                case XC_EXT_NOP:
1493                    break;
1494                case XC_DECLARE_FUNCTION_OR_CLASS:
1495                    /* always removed by compiler */
1496                    break;
1497                case XC_TICKS:
1498                    $lastphpop['ticks'] = $this->getOpVal($op1, $EX);
1499                    // $EX['tickschanged'] = true;
1500                    break;
1501                default: // {{{
1502                    echo "\x1B[31m * TODO ", $opname, "\x1B[0m\n";
1503                    // }}}
1504                }
1505            }
1506            if (isset($resvar)) {
1507                if ($istmpres) {
1508                    $T[$res['var']] = $resvar;
1509                    $lastresvar = &$T[$res['var']];
1510                }
1511                else {
1512                    $op['php'] = $resvar;
1513                    $lastphpop = &$op;
1514                    $lastresvar = &$op['php'];
1515                }
1516            }
1517        }
1518        return $T;
1519    }
1520    // }}}
1521    function unquote($str, $st, $ed) // {{{
1522    {
1523        $l1 = strlen($st);
1524        $l2 = strlen($ed);
1525        if (substr($str, 0, $l1) === $st && substr($str, -$l2) === $ed) {
1526            $str = substr($str, $l1, -$l2);
1527        }
1528        return $str;
1529    }
1530    // }}}
1531    function popargs(&$EX, $n) // {{{
1532    {
1533        $args = array();
1534        for ($i = 0; $i < $n; $i ++) {
1535            $a = array_pop($EX['argstack']);
1536            if (is_array($a)) {
1537                array_unshift($args, foldToCode($a, $EX));
1538            }
1539            else {
1540                array_unshift($args, $a);
1541            }
1542        }
1543        return implode(', ', $args);
1544    }
1545    // }}}
1546    function dumpop($op, &$EX) // {{{
1547    {
1548        $op1 = $op['op1'];
1549        $op2 = $op['op2'];
1550        $d = array('opname' => xcache_get_opcode($op['opcode']), 'opcode' => $op['opcode']);
1551
1552        foreach (array('op1' => 'op1', 'op2' => 'op2', 'result' => 'res') as $k => $kk) {
1553            switch ($op[$k]['op_type']) {
1554            case XC_IS_UNUSED:
1555                $d[$kk] = '*UNUSED* ' . $op[$k]['opline_num'];
1556                break;
1557
1558            case XC_IS_VAR:
1559                $d[$kk] = '$' . $op[$k]['var'];
1560                if ($kk != 'res') {
1561                    $d[$kk] .= ':' . $this->getOpVal($op[$k], $EX);
1562                }
1563                break;
1564
1565            case XC_IS_TMP_VAR:
1566                $d[$kk] = '#' . $op[$k]['var'];
1567                if ($kk != 'res') {
1568                    $d[$kk] .= ':' . $this->getOpVal($op[$k], $EX);
1569                }
1570                break;
1571
1572            case XC_IS_CV:
1573                $d[$kk] = $this->getOpVal($op[$k], $EX);
1574                break;
1575
1576            default:
1577                if ($kk == 'res') {
1578                    var_dump($op);
1579                    exit;
1580                    assert(0);
1581                }
1582                else {
1583                    $d[$kk] = $this->getOpVal($op[$k], $EX);
1584                }
1585            }
1586        }
1587        $d['ext'] = $op['extended_value'];
1588
1589        var_dump($d);
1590    }
1591    // }}}
1592    function dargs(&$EX, $indent) // {{{
1593    {
1594        $op_array = &$EX['op_array'];
1595
1596        if (isset($op_array['num_args'])) {
1597            $c = $op_array['num_args'];
1598        }
1599        else if ($op_array['arg_types']) {
1600            $c = count($op_array['arg_types']);
1601        }
1602        else {
1603            // php4
1604            $c = count($EX['recvs']);
1605        }
1606
1607        $refrest = false;
1608        for ($i = 0; $i < $c; $i ++) {
1609            if ($i) {
1610                echo ', ';
1611            }
1612            $arg = $EX['recvs'][$i + 1];
1613            if (isset($op_array['arg_info'])) {
1614                $ai = $op_array['arg_info'][$i];
1615                if (!empty($ai['class_name'])) {
1616                    echo $ai['class_name'], ' ';
1617                    if (!ZEND_ENGINE_2_2 && $ai['allow_null']) {
1618                        echo 'or NULL ';
1619                    }
1620                }
1621                else if (!empty($ai['array_type_hint'])) {
1622                    echo 'array ';
1623                    if (!ZEND_ENGINE_2_2 && $ai['allow_null']) {
1624                        echo 'or NULL ';
1625                    }
1626                }
1627                if ($ai['pass_by_reference']) {
1628                    echo '&';
1629                }
1630                printf("\$%s", $ai['name']);
1631            }
1632            else {
1633                if ($refrest) {
1634                    echo '&';
1635                }
1636                else if (isset($op_array['arg_types'][$i])) {
1637                    switch ($op_array['arg_types'][$i]) {
1638                    case BYREF_FORCE_REST:
1639                        $refrest = true;
1640                        /* fall */
1641                    case BYREF_FORCE:
1642                        echo '&';
1643                        break;
1644
1645                    case BYREF_NONE:
1646                    case BYREF_ALLOW:
1647                        break;
1648                    default:
1649                        assert(0);
1650                    }
1651                }
1652                echo str($arg[0], $indent);
1653            }
1654            if (isset($arg[1])) {
1655                echo ' = ', str($arg[1], $indent);
1656            }
1657        }
1658    }
1659    // }}}
1660    function dfunction($func, $indent = '', $nobody = false) // {{{
1661    {
1662        if ($nobody) {
1663            $body = ";\n";
1664            $EX = array();
1665            $EX['op_array'] = &$func['op_array'];
1666            $EX['recvs'] = array();
1667        }
1668        else {
1669            ob_start();
1670            $newindent = INDENT . $indent;
1671            $EX = &$this->dop_array($func['op_array'], $newindent);
1672            $body = ob_get_clean();
1673            if (!isset($EX['recvs'])) {
1674                $EX['recvs'] = array();
1675            }
1676        }
1677
1678        echo 'function ', $func['op_array']['function_name'], '(';
1679        $this->dargs($EX, $indent);
1680        echo ")\n";
1681        echo $indent, "{\n";
1682        echo $body;
1683        echo "$indent}\n";
1684    }
1685    // }}}
1686    function dclass($class, $indent = '') // {{{
1687    {
1688        // {{{ class decl
1689        if (!empty($class['doc_comment'])) {
1690            echo $indent;
1691            echo $class['doc_comment'];
1692            echo "\n";
1693        }
1694        $isinterface = false;
1695        if (!empty($class['ce_flags'])) {
1696            if ($class['ce_flags'] & ZEND_ACC_INTERFACE) {
1697                echo 'interface ';
1698                $isinterface = true;
1699            }
1700            else {
1701                if ($class['ce_flags'] & ZEND_ACC_IMPLICIT_ABSTRACT) {
1702                    echo "abstract ";
1703                }
1704                if ($class['ce_flags'] & ZEND_ACC_FINAL) {
1705                    echo "final ";
1706                }
1707            }
1708        }
1709        echo 'class ', $class['name'];
1710        if ($class['parent']) {
1711            echo ' extends ', $class['parent'];
1712        }
1713        /* TODO */
1714        if (!empty($class['interfaces'])) {
1715            echo ' implements ';
1716            echo implode(', ', $class['interfaces']);
1717        }
1718        echo "\n";
1719        echo $indent, "{";
1720        // }}}
1721        $newindent = INDENT . $indent;
1722        // {{{ const, static
1723        foreach (array('constants_table' => 'const '
1724                    , 'static_members' => 'static $') as $type => $prefix) {
1725            if (!empty($class[$type])) {
1726                echo "\n";
1727                // TODO: skip shadow?
1728                foreach ($class[$type] as $name => $v) {
1729                    echo $newindent;
1730                    echo $prefix, $name, ' = ';
1731                    echo str(value($v), $newindent);
1732                    echo ";\n";
1733                }
1734            }
1735        }
1736        // }}}
1737        // {{{ properties
1738        $member_variables = isset($class['properties_info']) ? $class['properties_info'] : ($class['default_static_members'] + $class['default_properties']);
1739        if ($member_variables) {
1740            echo "\n";
1741            $infos = !empty($class['properties_info']) ? $class['properties_info'] : null;
1742            foreach ($member_variables as $name => $dummy) {
1743                $info = (isset($infos) && isset($infos[$name])) ? $infos[$name] : null;
1744                if (isset($info)) {
1745                    if (!empty($info['doc_comment'])) {
1746                        echo $newindent;
1747                        echo $info['doc_comment'];
1748                        echo "\n";
1749                    }
1750                }
1751
1752                echo $newindent;
1753                $static = false;
1754                if (isset($info)) {
1755                    if ($info['flags'] & ZEND_ACC_STATIC) {
1756                        $static = true;
1757                    }
1758                }
1759                else if (isset($class['default_static_members'][$name])) {
1760                    $static = true;
1761                }
1762
1763                if ($static) {
1764                    echo "static ";
1765                }
1766
1767                $mangled = false;
1768                if (!ZEND_ENGINE_2) {
1769                    echo 'var ';
1770                }
1771                else if (!isset($info)) {
1772                    echo 'public ';
1773                }
1774                else {
1775                    if ($info['flags'] & ZEND_ACC_SHADOW) {
1776                        continue;
1777                    }
1778                    switch ($info['flags'] & ZEND_ACC_PPP_MASK) {
1779                    case ZEND_ACC_PUBLIC:
1780                        echo "public ";
1781                        break;
1782                    case ZEND_ACC_PRIVATE:
1783                        echo "private ";
1784                        $mangled = true;
1785                        break;
1786                    case ZEND_ACC_PROTECTED:
1787                        echo "protected ";
1788                        $mangled = true;
1789                        break;
1790                    }
1791                }
1792
1793                echo '$', $name;
1794
1795                if (isset($info['offset'])) {
1796                    $value = $class[$static ? 'default_static_members_table' : 'default_properties_table'][$info['offset']];
1797                }
1798                else {
1799                    $key = isset($info) ? $info['name'] . ($mangled ? "\000" : "") : $name;
1800
1801                    $value = $class[$static ? 'default_static_members' : 'default_properties'][$key];
1802                }
1803                if (isset($value)) {
1804                    echo ' = ';
1805                    echo str(value($value), $newindent);
1806                }
1807                echo ";\n";
1808            }
1809        }
1810        // }}}
1811        // {{{ function_table
1812        if (isset($class['function_table'])) {
1813            foreach ($class['function_table'] as $func) {
1814                if (!isset($func['scope']) || $func['scope'] == $class['name']) {
1815                    // TODO: skip shadow here
1816                    echo "\n";
1817                    $opa = $func['op_array'];
1818                    if (!empty($opa['doc_comment'])) {
1819                        echo $newindent;
1820                        echo $opa['doc_comment'];
1821                        echo "\n";
1822                    }
1823                    echo $newindent;
1824                    if (isset($opa['fn_flags'])) {
1825                        if ($opa['fn_flags'] & ZEND_ACC_ABSTRACT) {
1826                            echo "abstract ";
1827                        }
1828                        if ($opa['fn_flags'] & ZEND_ACC_FINAL) {
1829                            echo "final ";
1830                        }
1831                        if ($opa['fn_flags'] & ZEND_ACC_STATIC) {
1832                            echo "static ";
1833                        }
1834
1835                        switch ($opa['fn_flags'] & ZEND_ACC_PPP_MASK) {
1836                            case ZEND_ACC_PUBLIC:
1837                                echo "public ";
1838                                break;
1839                            case ZEND_ACC_PRIVATE:
1840                                echo "private ";
1841                                break;
1842                            case ZEND_ACC_PROTECTED:
1843                                echo "protected ";
1844                                break;
1845                            default:
1846                                echo "<visibility error> ";
1847                                break;
1848                        }
1849                    }
1850                    $this->dfunction($func, $newindent, $isinterface);
1851                    if ($opa['function_name'] == 'Decompiler') {
1852                        //exit;
1853                    }
1854                }
1855            }
1856        }
1857        // }}}
1858        echo $indent, "}\n";
1859    }
1860    // }}}
1861    function decompileString($string) // {{{
1862    {
1863        $this->dc = xcache_dasm_string($string);
1864        if ($this->dc === false) {
1865            echo "error compling string\n";
1866            return false;
1867        }
1868    }
1869    // }}}
1870    function decompileFile($file) // {{{
1871    {
1872        $this->dc = xcache_dasm_file($file);
1873        if ($this->dc === false) {
1874            echo "error compling $file\n";
1875            return false;
1876        }
1877    }
1878    // }}}
1879    function output() // {{{
1880    {
1881        echo "<?". "php\n";
1882        foreach ($this->dc['class_table'] as $key => $class) {
1883            if ($key{0} != "\0") {
1884                echo "\n";
1885                $this->dclass($class);
1886            }
1887        }
1888
1889        foreach ($this->dc['function_table'] as $key => $func) {
1890            if ($key{0} != "\0") {
1891                echo "\n";
1892                $this->dfunction($func);
1893            }
1894        }
1895
1896        echo "\n";
1897        $this->dop_array($this->dc['op_array']);
1898        echo "\n?" . ">\n";
1899        return true;
1900    }
1901    // }}}
1902}
1903
1904// {{{ defines
1905define('ZEND_ACC_STATIC',         0x01);
1906define('ZEND_ACC_ABSTRACT',       0x02);
1907define('ZEND_ACC_FINAL',          0x04);
1908define('ZEND_ACC_IMPLEMENTED_ABSTRACT',       0x08);
1909
1910define('ZEND_ACC_IMPLICIT_ABSTRACT_CLASS',    0x10);
1911define('ZEND_ACC_EXPLICIT_ABSTRACT_CLASS',    0x20);
1912define('ZEND_ACC_FINAL_CLASS',                0x40);
1913define('ZEND_ACC_INTERFACE',                  0x80);
1914define('ZEND_ACC_PUBLIC',     0x100);
1915define('ZEND_ACC_PROTECTED',  0x200);
1916define('ZEND_ACC_PRIVATE',    0x400);
1917define('ZEND_ACC_PPP_MASK',  (ZEND_ACC_PUBLIC | ZEND_ACC_PROTECTED | ZEND_ACC_PRIVATE));
1918
1919define('ZEND_ACC_CHANGED',    0x800);
1920define('ZEND_ACC_IMPLICIT_PUBLIC',    0x1000);
1921
1922define('ZEND_ACC_CTOR',       0x2000);
1923define('ZEND_ACC_DTOR',       0x4000);
1924define('ZEND_ACC_CLONE',      0x8000);
1925
1926define('ZEND_ACC_ALLOW_STATIC',   0x10000);
1927
1928define('ZEND_ACC_SHADOW', 0x2000);
1929
1930define('ZEND_ENGINE_2_4', PHP_VERSION >= "5.3.99");
1931define('ZEND_ENGINE_2_3', ZEND_ENGINE_2_4 || PHP_VERSION >= "5.3.");
1932define('ZEND_ENGINE_2_2', ZEND_ENGINE_2_3 || PHP_VERSION >= "5.2.");
1933define('ZEND_ENGINE_2_1', ZEND_ENGINE_2_2 || PHP_VERSION >= "5.1.");
1934define('ZEND_ENGINE_2',   ZEND_ENGINE_2_1 || PHP_VERSION >= "5.0.");
1935
1936if (ZEND_ENGINE_2_4) {
1937    define('ZEND_FETCH_GLOBAL',           0x00000000);
1938    define('ZEND_FETCH_LOCAL',            0x10000000);
1939    define('ZEND_FETCH_STATIC',           0x20000000);
1940    define('ZEND_FETCH_STATIC_MEMBER',    0x30000000);
1941    define('ZEND_FETCH_GLOBAL_LOCK',      0x40000000);
1942    define('ZEND_FETCH_LEXICAL',          0x50000000);
1943
1944    define('ZEND_FETCH_TYPE_MASK',        0x70000000);
1945}
1946else {
1947    define('ZEND_FETCH_GLOBAL',           0);
1948    define('ZEND_FETCH_LOCAL',            1);
1949    define('ZEND_FETCH_STATIC',           2);
1950    define('ZEND_FETCH_STATIC_MEMBER',    3);
1951    define('ZEND_FETCH_GLOBAL_LOCK',      4);
1952}
1953
1954define('ZEND_FETCH_CLASS_DEFAULT',    0);
1955define('ZEND_FETCH_CLASS_SELF',       1);
1956define('ZEND_FETCH_CLASS_PARENT',     2);
1957define('ZEND_FETCH_CLASS_MAIN',       3);
1958define('ZEND_FETCH_CLASS_GLOBAL',     4);
1959define('ZEND_FETCH_CLASS_AUTO',       5);
1960define('ZEND_FETCH_CLASS_INTERFACE',  6);
1961define('ZEND_FETCH_CLASS_STATIC',     7);
1962if (ZEND_ENGINE_2_4) {
1963    define('ZEND_FETCH_CLASS_TRAIT',     14);
1964}
1965if (ZEND_ENGINE_2_3) {
1966    define('ZEND_FETCH_CLASS_MASK',     0xF);
1967}
1968
1969define('ZEND_EVAL',               (1<<0));
1970define('ZEND_INCLUDE',            (1<<1));
1971define('ZEND_INCLUDE_ONCE',       (1<<2));
1972define('ZEND_REQUIRE',            (1<<3));
1973define('ZEND_REQUIRE_ONCE',       (1<<4));
1974
1975define('ZEND_ISSET',              (1<<0));
1976define('ZEND_ISEMPTY',            (1<<1));
1977if (ZEND_ENGINE_2_4) {
1978    define('EXT_TYPE_UNUSED',     (1<<5));
1979}
1980else {
1981    define('EXT_TYPE_UNUSED',     (1<<0));
1982}
1983
1984define('ZEND_FETCH_STANDARD',     0);
1985define('ZEND_FETCH_ADD_LOCK',     1);
1986
1987define('ZEND_FE_FETCH_BYREF',     1);
1988define('ZEND_FE_FETCH_WITH_KEY',  2);
1989
1990define('ZEND_MEMBER_FUNC_CALL',   1<<0);
1991define('ZEND_CTOR_CALL',          1<<1);
1992
1993define('ZEND_ARG_SEND_BY_REF',        (1<<0));
1994define('ZEND_ARG_COMPILE_TIME_BOUND', (1<<1));
1995define('ZEND_ARG_SEND_FUNCTION',      (1<<2));
1996
1997define('BYREF_NONE',       0);
1998define('BYREF_FORCE',      1);
1999define('BYREF_ALLOW',      2);
2000define('BYREF_FORCE_REST', 3);
2001define('IS_NULL',     0);
2002define('IS_LONG',     1);
2003define('IS_DOUBLE',   2);
2004define('IS_STRING',   3);
2005define('IS_ARRAY',    4);
2006define('IS_OBJECT',   5);
2007define('IS_BOOL',     6);
2008define('IS_RESOURCE', 7);
2009define('IS_CONSTANT', 8);
2010define('IS_CONSTANT_ARRAY',   9);
2011
2012@define('XC_IS_CV', 16);
2013
2014/*
2015if (preg_match_all('!XC_[A-Z_]+!', file_get_contents(__FILE__), $ms)) {
2016    $verdiff = array();
2017    foreach ($ms[0] as $k) {
2018        if (!defined($k)) {
2019            $verdiff[$k] = -1;
2020            define($k, -1);
2021        }
2022    }
2023    var_export($verdiff);
2024}
2025/*/
2026foreach (array (
2027    'XC_HANDLE_EXCEPTION' => -1,
2028    'XC_FETCH_CLASS' => -1,
2029    'XC_FETCH_' => -1,
2030    'XC_FETCH_DIM_' => -1,
2031    'XC_ASSIGN_DIM' => -1,
2032    'XC_UNSET_DIM' => -1,
2033    'XC_FETCH_OBJ_' => -1,
2034    'XC_ASSIGN_OBJ' => -1,
2035    'XC_ISSET_ISEMPTY_DIM_OBJ' => -1,
2036    'XC_ISSET_ISEMPTY_PROP_OBJ' => -1,
2037    'XC_ISSET_ISEMPTY_VAR' => -1,
2038    'XC_INIT_STATIC_METHOD_CALL' => -1,
2039    'XC_INIT_METHOD_CALL' => -1,
2040    'XC_VERIFY_ABSTRACT_CLASS' => -1,
2041    'XC_DECLARE_CLASS' => -1,
2042    'XC_DECLARE_INHERITED_CLASS' => -1,
2043    'XC_DECLARE_INHERITED_CLASS_DELAYED' => -1,
2044    'XC_ADD_INTERFACE' => -1,
2045    'XC_POST_DEC_OBJ' => -1,
2046    'XC_POST_INC_OBJ' => -1,
2047    'XC_PRE_DEC_OBJ' => -1,
2048    'XC_PRE_INC_OBJ' => -1,
2049    'XC_UNSET_OBJ' => -1,
2050    'XC_JMP_NO_CTOR' => -1,
2051    'XC_FETCH_' => -1,
2052    'XC_FETCH_DIM_' => -1,
2053    'XC_UNSET_DIM_OBJ' => -1,
2054    'XC_FETCH_OBJ_' => -1,
2055    'XC_ISSET_ISEMPTY' => -1,
2056    'XC_INIT_FCALL_BY_FUNC' => -1,
2057    'XC_DO_FCALL_BY_FUNC' => -1,
2058    'XC_DECLARE_FUNCTION_OR_CLASS' => -1,
2059) as $k => $v) {
2060    if (!defined($k)) {
2061        define($k, $v);
2062    }
2063}
2064
2065/* XC_UNDEF XC_OP_DATA
2066$content = file_get_contents(__FILE__);
2067for ($i = 0; $opname = xcache_get_opcode($i); $i ++) {
2068    if (!preg_match("/\\bXC_" . $opname . "\\b(?!')/", $content)) {
2069        echo "not done ", $opname, "\n";
2070    }
2071}
2072// */
2073// }}}
2074
Note: See TracBrowser for help on using the repository browser.