source: trunk/Decompiler.class.php @ 749

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

decompiler: massive improvements for php 5 and 5.3

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