static void check_arg_types( funcargs *args_from_decl, expr **exprargs, symtable *stab, char *sp, where *const exprloc) { if(exprargs && args_from_decl->arglist){ int i; char buf[64]; int finished_expr_args = 0; for(i = 0; ; i++){ decl *decl_arg = args_from_decl->arglist[i]; if(!decl_arg) break; if(!type_is_complete(decl_arg->ref)){ warn_at_print_error(&decl_arg->where, "incomplete parameter type '%s'", type_to_str(decl_arg->ref)); fold_had_error = 1; note_at(exprloc, "in call here"); } /* exprargs[i] may be NULL - old style function */ if(finished_expr_args || !exprargs[i]){ finished_expr_args = 1; continue; } ARG_BUF(buf, i, sp); fold_type_chk_and_cast_ty( decl_arg->ref, &exprargs[i], stab, &exprargs[i]->where, buf); /* f(int [static 5]) check */ static_array_check(decl_arg, exprargs[i]); } } }
void fold_expr_assign(expr *e, symtable *stab) { sym *lhs_sym = NULL; int is_struct_cpy = 0; lhs_sym = fold_inc_writes_if_sym(e->lhs, stab); fold_expr_nodecay(e->lhs, stab); fold_expr_nodecay(e->rhs, stab); if(lhs_sym) lhs_sym->nreads--; /* cancel the read that fold_ident thinks it got */ is_struct_cpy = !!type_is_s_or_u(e->lhs->tree_type); if(!is_struct_cpy) FOLD_EXPR(e->rhs, stab); /* lval2rval the rhs */ if(type_is_primitive(e->rhs->tree_type, type_void)){ fold_had_error = 1; warn_at_print_error(&e->where, "assignment from void expression"); e->tree_type = type_nav_btype(cc1_type_nav, type_int); return; } expr_must_lvalue(e->lhs, "assignment"); if(!e->assign_is_init) expr_assign_const_check(e->lhs, &e->where); fold_check_restrict(e->lhs, e->rhs, "assignment", &e->where); /* this makes sense, but it's also critical for code-gen: * if we assign to a volatile lvalue, we don't want the volatile-ness * to propagate, as we are now an rvalue, and don't want our value read * as we decay */ e->tree_type = type_unqualify(e->lhs->tree_type); /* type check */ fold_type_chk_and_cast_ty( e->lhs->tree_type, &e->rhs, stab, &e->where, "assignment"); /* the only way to get a value into a bitfield (aside from memcpy / indirection) is via this * hence we're fine doing the truncation check here */ { decl *mem; if(expr_kind(e->lhs, struct) && (mem = e->lhs->bits.struct_mem.d) /* maybe null from s->non_present_memb */ && mem->bits.var.field_width) { bitfield_trunc_check(mem, e->rhs); } } if(is_struct_cpy){ e->expr = builtin_new_memcpy( e->lhs, e->rhs, type_size(e->rhs->tree_type, &e->rhs->where)); FOLD_EXPR(e->expr, stab); /* set is_lval, so we can participate in struct-copy chains * FIXME: don't interpret as an lvalue, e.g. (a = b) = c; * this is currently special cased in expr_is_lval() * * CHECK THIS */ if(cc1_backend == BACKEND_ASM) e->f_gen = lea_assign_lhs; e->f_islval = expr_is_lval_struct; } }