PHPCompiler\Backend\PHP7\PECL::compileExpr PHP Method

compileExpr() protected method

protected compileExpr ( Expr $op, $indent )
$op PHPCfg\Op\Expr
    protected function compileExpr(Op\Expr $op, $indent)
    {
        $phi = '';
        foreach ($op->result->usages as $usage) {
            if ($usage instanceof Op\Phi) {
                $phi .= $indent . $this->getVarName($usage->result) . " = " . $this->getVarName($op->result) . ";\n";
            }
        }
        $result = '';
        switch ($op->getType()) {
            case 'Expr_ArrayDimFetch':
                $var = $this->getVarName($op->var);
                $dim = $this->getVarName($op->dim);
                $result = $this->getVarName($op->result);
                switch ($this->mapToCType($op->var->type)) {
                    case 'zend_string*':
                        assert($this->mapToCType($op->dim->type) === 'zend_long');
                        $safety = $indent . "if ({$dim} < 0 || {$dim} >= ZSTR_LEN({$var})) {\n";
                        $safety .= $indent . "\t{$result} = ZSTR_EMPTY_ALLOC();\n";
                        $safety .= $indent . "\tzend_error(E_NOTICE, \"Uninitialized string offset: %pd\", {$dim});\n";
                        $safety .= $indent . "} else if (CG(one_char_string)[(unsigned char) ZSTR_VAL({$var})[{$dim}]]) {\n";
                        $safety .= $indent . "\t{$result} = CG(one_char_string)[(unsigned char) ZSTR_VAL({$var})[{$dim}]];\n";
                        $safety .= $indent . "} else {\n";
                        $safety .= $indent . "\t{$result} = zend_string_init(ZSTR_VAL({$var}) + {$dim}, 1, 0);\n";
                        $safety .= $indent . "\tfree_{$result} = 1;\n";
                        $safety .= $indent . "}\n{$phi}";
                        return $safety;
                    case 'HashTable*':
                        $dimType = $this->mapToCType($op->dim->type);
                        $resultType = $this->mapToCType($op->result->type);
                        if ($dimType === 'zend_long') {
                            $safety = $indent . "do {\n";
                            $safety .= $indent . "\tzval* tmp = zend_hash_index_find({$var}, {$dim});\n";
                            $safety .= $indent . "\tif (tmp == NULL) {\n";
                            if ($resultType === 'zval') {
                                $safety .= $indent . "\t\tZVAL_NULL(&{$result});\n";
                                $safety .= $indent . "\t\tzend_error(E_NOTICE, \"Uninitialized offset: %pd\", {$dim});\n";
                                $safety .= $indent . "\t} else {\n";
                                $safety .= $indent . "\t\t{$result} = *tmp;\n";
                                $safety .= $indent . "\t}\n";
                                $safety .= $indent . "} while(0);\n";
                                return $safety;
                            } else {
                                $typeInfo = $this->getTypeInfo($resultType);
                                $safety .= $indent . "\t\t" . $typeInfo['default']($result) . ";\n";
                                $safety .= $indent . "\t\tzend_error(E_NOTICE, \"Uninitialized offset: %pd\", {$dim});\n";
                                $safety .= $indent . "\t} else if (Z_TYPE_P(tmp) != {$typeInfo['ztype']}) {\n";
                                $safety .= $indent . "\t\tzend_throw_error(NULL, \"Offset is not an {$typeInfo['stringtype']}: %pd\", {$dim});\n";
                                $safety .= $indent . "\t} else {\n";
                                $safety .= $indent . "\t\t{$result} = {$typeInfo['ztypefetch']}(tmp);\n";
                                $safety .= $indent . "\t}\n";
                                $safety .= $indent . "} while(0); \n";
                                return $safety;
                            }
                        }
                    default:
                        throw new \LogicException("Unknown array dim fetch type {$op->var->type}");
                }
                break;
            case 'Expr_Assign':
                $result = $this->getVarName($op->var) . ' = ' . $this->getVarName($op->expr);
                foreach ($op->var->usages as $usage) {
                    if ($usage instanceof Op\Phi) {
                        $phi .= $indent . $this->getVarName($usage->result) . " = " . $this->getVarName($op->var) . ";\n";
                    }
                }
                break;
            case 'Expr_BinaryOp_BitwiseAnd':
                $result = $this->getVarName($op->left) . " & " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_BitwiseOr':
                $result = $this->getVarName($op->left) . " | " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_BitwiseXor':
                $result = $this->getVarName($op->left) . " ^ " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_Coalesce':
                throw new \LogicException("TODO");
                break;
            case 'Expr_BinaryOp_Concat':
                $result = $this->getVarName($op->left) . " . " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_Div':
                $result = $this->getVarName($op->left) . " / " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_Equal':
                throw new \LogicException("TODO");
                break;
            case 'Expr_BinaryOp_Greater':
                $result = $this->getVarName($op->left) . " > " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_GreaterOrEqual':
                $result = $this->getVarName($op->left) . " >= " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_Identical':
                throw new \LogicException("TODO");
                break;
            case 'Expr_BinaryOp_LogicalXor':
                throw new \LogicException("TODO");
                break;
            case 'Expr_BinaryOp_Minus':
                $result = $this->getVarName($op->left) . " - " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_Mod':
                $result = $this->getVarName($op->left) . " % " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_Mul':
                $result = $this->getVarName($op->left) . " * " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_NotEqual':
                throw new \LogicException("TODO");
                break;
            case 'Expr_BinaryOp_NotIdentical':
                throw new \LogicException("TODO");
                break;
            case 'Expr_BinaryOp_Plus':
                $result = $this->getVarName($op->left) . " + " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_Pow':
                throw new \LogicException("TODO");
                break;
            case 'Expr_BinaryOp_ShiftLeft':
                $result = $this->getVarName($op->left) . " << " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_ShiftRight':
                $result = $this->getVarName($op->left) . " >> " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_Smaller':
                $result = $this->getVarName($op->left) . " < " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_SmallerOrEqual':
                $result = $this->getVarName($op->left) . " <= " . $this->getVarName($op->right);
                break;
            case 'Expr_BinaryOp_Spaceship':
                throw new \LogicException("TODO");
                break;
            case 'Expr_Print':
                $result = "1;\n" . $this->compilePrintStatement($op, $indent);
                break;
            default:
                throw new \RuntimeException("Unknown expression found: " . $op->getType());
        }
        foreach ($op->result->usages as $usage) {
            if ($usage instanceof Op\Phi) {
                $phi .= $indent . $this->getVarName($usage->result) . " = " . $this->getVarName($op->result) . ";\n";
            }
        }
        if (count($op->result->usages) === 0) {
            return $indent . $result . ";\n" . $phi;
        }
        return $indent . $this->getVarName($op->result) . " = " . $result . ";\n" . $phi;
    }