X-Git-Url: http://git.xonotic.org/?a=blobdiff_plain;ds=sidebyside;f=ast.c;h=50b6e872dd540342c94037460174b3e74546db3d;hb=99bab760064aebe3e2dd9e3ecaabc2ab125367e2;hp=c3dfe21a281acff9561fb1addd31e9e14cd0bd9d;hpb=40fe52e665d902a83e545145dfedfacf0cc863a2;p=xonotic%2Fgmqcc.git diff --git a/ast.c b/ast.c index c3dfe21..50b6e87 100644 --- a/ast.c +++ b/ast.c @@ -91,7 +91,6 @@ static void ast_expression_delete_full(ast_expression *self) MEM_VEC_FUNCTIONS(ast_expression_common, ast_value*, params) -static ast_expression* ast_type_copy(lex_ctx ctx, const ast_expression *ex); ast_value* ast_value_copy(const ast_value *self) { size_t i; @@ -117,6 +116,27 @@ ast_value* ast_value_copy(const ast_value *self) return cp; } +bool ast_type_adopt_impl(ast_expression *self, const ast_expression *other) +{ + size_t i; + const ast_expression_common *fromex; + ast_expression_common *selfex; + self->expression.vtype = other->expression.vtype; + if (other->expression.next) { + self->expression.next = (ast_expression*)ast_type_copy(ast_ctx(self), other->expression.next); + if (!self->expression.next) + return false; + } + fromex = &other->expression; + selfex = &self->expression; + for (i = 0; i < fromex->params_count; ++i) { + ast_value *v = ast_value_copy(fromex->params[i]); + if (!v || !ast_expression_common_params_add(selfex, v)) + return false; + } + return true; +} + static ast_expression* ast_shallow_type(lex_ctx ctx, int vtype) { ast_instantiate(ast_expression, ctx, ast_expression_delete_full); @@ -127,7 +147,7 @@ static ast_expression* ast_shallow_type(lex_ctx ctx, int vtype) return self; } -static ast_expression* ast_type_copy(lex_ctx ctx, const ast_expression *ex) +ast_expression* ast_type_copy(lex_ctx ctx, const ast_expression *ex) { size_t i; const ast_expression_common *fromex; @@ -322,6 +342,11 @@ ast_unary* ast_unary_new(lex_ctx ctx, int op, self->op = op; self->operand = expr; + if (op >= INSTR_NOT_F && op <= INSTR_NOT_FNC) { + self->expression.vtype = TYPE_FLOAT; + } else + asterror(ctx, "cannot determine type of unary operation %s", asm_instr[op].m); + return self; } @@ -370,12 +395,14 @@ ast_entfield* ast_entfield_new(lex_ctx ctx, ast_expression *entity, ast_expressi ast_expression_init((ast_expression*)self, (ast_expression_codegen*)&ast_entfield_codegen); - self->expression.vtype = outtype->expression.vtype; - self->expression.next = ast_type_copy(ctx, outtype->expression.next); - self->entity = entity; self->field = field; + if (!ast_type_adopt(self, outtype)) { + ast_entfield_delete(self); + return NULL; + } + return self; } @@ -749,6 +776,7 @@ bool ast_global_codegen(ast_value *self, ir_builder *ir) ir_function *func = ir_builder_create_function(ir, self->name, self->expression.next->expression.vtype); if (!func) return false; + func->context = ast_ctx(self); self->constval.vfunc->ir_func = func; self->ir_v = func->value; @@ -760,6 +788,7 @@ bool ast_global_codegen(ast_value *self, ir_builder *ir) v = ir_builder_create_field(ir, self->name, self->expression.next->expression.vtype); if (!v) return false; + v->context = ast_ctx(self); if (self->isconst) { asterror(ast_ctx(self), "TODO: constant field pointers with value\n"); goto error; @@ -773,6 +802,7 @@ bool ast_global_codegen(ast_value *self, ir_builder *ir) asterror(ast_ctx(self), "ir_builder_create_global failed\n"); return false; } + v->context = ast_ctx(self); if (self->isconst) { switch (self->expression.vtype) @@ -824,6 +854,7 @@ bool ast_local_codegen(ast_value *self, ir_function *func, bool param) v = ir_function_create_local(func, self->name, self->expression.vtype, param); if (!v) return false; + v->context = ast_ctx(self); /* A constant local... hmmm... * I suppose the IR will have to deal with this @@ -906,6 +937,9 @@ bool ast_function_codegen(ast_function *self, ir_builder *ir) /* TODO: check return types */ if (!self->curblock->is_return) { + return ir_block_create_return(self->curblock, NULL); + /* From now on the parser has to handle this situation */ +#if 0 if (!self->vtype->expression.next || self->vtype->expression.next->expression.vtype == TYPE_VOID) { @@ -917,6 +951,7 @@ bool ast_function_codegen(ast_function *self, ir_builder *ir) asterror(ast_ctx(self), "function `%s` missing return value", self->name); return false; } +#endif } return true; } @@ -1305,9 +1340,9 @@ bool ast_ifthen_codegen(ast_ifthen *self, ast_function *func, bool lvalue, ir_va return false; /* add jumps ot the merge block */ - if (ontrue && !ir_block_create_jump(ontrue, merge)) + if (ontrue && !ontrue->final && !ir_block_create_jump(ontrue, merge)) return false; - if (onfalse && !ir_block_create_jump(onfalse, merge)) + if (onfalse && !onfalse->final && !ir_block_create_jump(onfalse, merge)) return false; /* we create the if here, that way all blocks are ordered :) @@ -1625,7 +1660,7 @@ bool ast_loop_codegen(ast_loop *self, ast_function *func, bool lvalue, ir_value else if (bpostcond) tmpblock = bpostcond; else if (bprecond) tmpblock = bprecond; else tmpblock = bout; - if (!ir_block_create_jump(end_bbody, tmpblock)) + if (!end_bbody->final && !ir_block_create_jump(end_bbody, tmpblock)) return false; }