source: trunk/Decompiler.class.php @ 743

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

decompiler: process ZEND_CASE opcode, switch/case is not recognized as it used to be however

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