Esempio n. 1
0
def _exp_value_array(code: Code, exp):
    items = exp['items']
    t = exp['type']

    if t.dim > 1:
        desc = _create_field_descriptor(TypeArray(t.dim - 1, t.inner))
    else:
        desc = _create_field_descriptor(t.inner)

    code.const_int(len(items))
    code.new_array(desc)

    for (i, item) in enumerate(items):
        code.dup()
        code.const_int(i)
        item_type = item['type']
        _expression(code, item)
        if isinstance(item_type, TypeInt):
            code.array_store_int()
        elif isinstance(item_type, TypeReal):
            code.array_store_double()
        elif isinstance(item_type, TypeBool):
            code.array_store_boolean()
        elif isinstance(item_type, TypeStr):
            code.array_store_reference()
        elif isinstance(item_type, TypeArray):
            code.array_store_reference()
        else:
            raise NotImplementedError()
Esempio n. 2
0
def _exp_ge(code: Code, exp):
    left = exp['left']
    right = exp['right']
    left_type = exp['left']['type']
    _expression(code, left)
    _expression(code, right)

    if isinstance(left_type, TypeInt) or isinstance(left_type, TypeReal):
        if isinstance(left_type, TypeReal):
            code.cmp_double_g()
            cmp_pos = code.pos()
            code.if_ge()
        else:
            cmp_pos = code.pos()
            code.if_cmp_int_ge()
        code.const_int(0)
        goto_pos = code.pos()
        code.goto()
        true_pos = code.pos()
        code.const_int(1)
        end_pos = code.pos()

        code.update_jump(cmp_pos, true_pos)
        code.update_jump(goto_pos, end_pos)
    else:
        raise NotImplementedError()
Esempio n. 3
0
def _exp_eq(code: Code, exp):
    left = exp['left']
    right = exp['right']
    left_type = exp['left']['type']
    _expression(code, left)
    _expression(code, right)

    if isinstance(left_type, TypeInt) or isinstance(
            left_type, TypeBool) or isinstance(left_type, TypeReal):
        if isinstance(left_type, TypeReal):
            code.cmp_double_l()
            cmp_pos = code.pos()
            code.if_eq()
        else:
            cmp_pos = code.pos()
            code.if_cmp_int_eq()
        code.const_int(0)
        goto_pos = code.pos()
        code.goto()
        true_pos = code.pos()
        code.const_int(1)
        end_pos = code.pos()

        code.update_jump(cmp_pos, true_pos)
        code.update_jump(goto_pos, end_pos)
    elif isinstance(left_type, TypeStr):
        code.invoke_virtual(*JM_STRING_EQUALS)
    else:
        raise NotImplementedError()
Esempio n. 4
0
def _expression(code: Code, expression):
    node_type = expression['node']

    if node_type == Node.UMINUS:
        _exp_uminus(code, expression)
    elif node_type == Node.UPLUS:
        _exp_uplus(code, expression)
    elif node_type == Node.MUL:
        _exp_mul(code, expression)
    elif node_type == Node.DIV:
        _exp_div(code, expression)
    elif node_type == Node.PLUS:
        _exp_plus(code, expression)
    elif node_type == Node.MINUS:
        _exp_minus(code, expression)
    elif node_type == Node.EQ:
        _exp_eq(code, expression)
    elif node_type == Node.NE:
        _exp_ne(code, expression)
    elif node_type == Node.LT:
        _exp_lt(code, expression)
    elif node_type == Node.GT:
        _exp_gt(code, expression)
    elif node_type == Node.LE:
        _exp_le(code, expression)
    elif node_type == Node.GE:
        _exp_ge(code, expression)
    elif node_type == Node.NOT:
        _exp_not(code, expression)
    elif node_type == Node.AND:
        _exp_and(code, expression)
    elif node_type == Node.OR:
        _exp_or(code, expression)
    elif node_type == Node.VARIABLE_LOAD:
        _exp_var_load(code, expression)
    elif node_type == Node.ARRAY_LOAD:
        _exp_array_load(code, expression)
    elif node_type == Node.VARIABLE_ASSIGNMENT:
        _exp_var_assign(code, expression)
    elif node_type == Node.ARRAY_ASSIGNMENT:
        _exp_array_assign(code, expression)
    elif node_type == Node.VALUE_INT:
        code.const_int(expression['value'])
    elif node_type == Node.VALUE_REAL:
        code.const_double(expression['value'])
    elif node_type == Node.VALUE_BOOL:
        code.const_int(int(expression['value']))
    elif node_type == Node.VALUE_STR:
        code.const_string(expression['value'])
    elif node_type == Node.VALUE_ARRAY:
        _exp_value_array(code, expression)
    elif node_type == Node.FUNCTION_CALL_VALUE:
        _exp_function_call(code, expression)
    else:
        raise NotImplementedError()
Esempio n. 5
0
def _exp_not(code: Code, exp):
    expression = exp['expression']
    exp_type = exp['expression']['type']
    _expression(code, expression)

    if isinstance(exp_type, TypeBool):
        cmp_pos = code.pos()
        code.if_eq()
        code.const_int(0)
        goto_pos = code.pos()
        code.goto()
        false_pos = code.pos()
        code.const_int(1)
        end_pos = code.pos()

        code.update_jump(cmp_pos, false_pos)
        code.update_jump(goto_pos, end_pos)
    else:
        raise NotImplementedError()
Esempio n. 6
0
def _exp_or(code: Code, exp):
    left = exp['left']
    right = exp['right']
    t = exp['type']
    _expression(code, left)
    _expression(code, right)

    if isinstance(t, TypeBool):
        cmp1_pos = code.pos()
        code.if_ne()
        cmp2_pos = code.pos()
        code.if_ne()
        code.const_int(0)
        goto_pos = code.pos()
        code.goto()
        true_pos = code.pos()
        code.const_int(1)
        end_pos = code.pos()

        code.update_jump(cmp1_pos, true_pos)
        code.update_jump(cmp2_pos, true_pos)
        code.update_jump(goto_pos, end_pos)
    else:
        raise NotImplementedError()
Esempio n. 7
0
def _exp_and(code: Code, exp):
    left = exp['left']
    right = exp['right']
    left_type = exp['left']['type']
    _expression(code, left)
    _expression(code, right)

    if isinstance(left_type, TypeBool):
        cmp1_pos = code.pos()
        code.if_eq()
        cmp2_pos = code.pos()
        code.if_eq()
        code.const_int(1)
        goto_pos = code.pos()
        code.goto()
        false_pos = code.pos()
        code.const_int(0)
        end_pos = code.pos()

        code.update_jump(cmp1_pos, false_pos)
        code.update_jump(cmp2_pos, false_pos)
        code.update_jump(goto_pos, end_pos)
    else:
        raise NotImplementedError()
Esempio n. 8
0
def _exp_function_call(code: Code, exp):
    name = exp['name']
    args_exps = exp['arguments']
    params = exp['parameters']
    ret = exp['type']

    for e in args_exps:
        _expression(code, e)

    # check for predefined functions
    if name == FN_LEN and len(params) == 1:
        if isinstance(params[0], TypeStr):
            code.invoke_virtual(*JM_STRING_LENGTH)
            return
        elif isinstance(params[0], TypeArray):
            code.array_length()
            return

    if name == FN_INT and len(params) == 1:
        if isinstance(params[0], TypeInt):
            # nothing to do
            return
        elif isinstance(params[0], TypeReal):
            code.double_to_int()
            return
        elif isinstance(params[0], TypeBool):
            # nothing to do
            return
        elif isinstance(params[0], TypeStr):
            code.invoke_static(*JSM_INT_PARSE)
            return

    if name == FN_REAL and len(params) == 1:
        if isinstance(params[0], TypeInt):
            code.int_to_double()
            return
        elif isinstance(params[0], TypeReal):
            # nothing to do
            return
        elif isinstance(params[0], TypeBool):
            code.int_to_double()
            return
        elif isinstance(params[0], TypeStr):
            code.invoke_static(*JSM_DOUBLE_PARSE)
            return

    if name == FN_BOOL and len(params) == 1:
        if isinstance(params[0], TypeInt):
            code.const_int(1)
            code.and_int()
            return
        elif isinstance(params[0], TypeReal):
            code.const_double(0.0)
            code.cmp_double_l()
            cmp_pos = code.pos()
            code.if_eq()
            code.const_int(1)
            goto_pos = code.pos()
            code.goto()
            false_pos = code.pos()
            code.const_int(0)
            end_pos = code.pos()

            code.update_jump(cmp_pos, false_pos)
            code.update_jump(goto_pos, end_pos)
            return
        elif isinstance(params[0], TypeBool):
            # nothing to do
            return
        elif isinstance(params[0], TypeStr):
            code.invoke_static(*JSM_BOOLEAN_PARSE)
            return

    if name == FN_STR and len(params) == 1:
        if isinstance(params[0], TypeInt):
            code.invoke_static(*JSM_INT_TO_STRING)
            return
        elif isinstance(params[0], TypeReal):
            code.invoke_static(*JSM_DOUBLE_TO_STRING)
            return
        elif isinstance(params[0], TypeBool):
            code.invoke_static(*JSM_BOOLEAN_TO_STRING)
            return

    if name == FN_SUBSTRING and len(params) == 3:
        if isinstance(params[0], TypeStr) and isinstance(
                params[1], TypeInt) and isinstance(params[2], TypeInt):
            code.invoke_static(*JM_STRING_SUBSTRING)
            return

    if name == FN_WRITE and len(params) == 1:
        if isinstance(params[0], TypeStr):
            code.load_static_field(*JSF_STDOUT)
            code.swap()
            code.invoke_virtual(*JM_PRINT)
            return

    if name == FN_READ_LINE and len(params) == 0:
        code.load_static_field(_class_name, BUFF_READER_FIELD,
                               ClassDesc(JC_BUFF_READER))
        code.invoke_virtual(*JM_READLINE)

        # if result null, set the eof field to true and load empty string
        code.dup()
        cmp_pos = code.pos()
        code.if_non_null()
        code.const_int(1)
        code.store_static_field(_class_name, EOF_FIELD, BooleanDesc())
        code.const_string('')
        end_pos = code.pos()
        code.update_jump(cmp_pos, end_pos)
        return

    if name == FN_EOF and len(params) == 0:
        code.load_static_field(_class_name, EOF_FIELD, BooleanDesc())
        return

    # custom function
    name = PREFIX + name
    desc = _create_method_descriptor(params, ret)
    code.invoke_static(_class_name, name, desc)