summaryrefslogtreecommitdiffhomepage
path: root/ir/be/ia32/x86_cconv.c
blob: dad4887cee4db027baa757be4b5619ed93bf2951 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
/*
 * This file is part of libFirm.
 * Copyright (C) 2014 University of Karlsruhe.
 */

/**
 * @file
 * @brief   calling convention helpers
 * @author  Matthias Braun
 */
#include "x86_cconv.h"

#include "betranshlp.h"
#include "bevarargs.h"
#include "iredges_t.h"
#include "irgmod.h"
#include "irnode_t.h"
#include <stdlib.h>

void x86_free_calling_convention(x86_cconv_t *cconv)
{
	free(cconv->parameters);
	free(cconv->results);
	free(cconv->caller_saves);
	free(cconv->callee_saves);
	free(cconv);
}

void x86_create_parameter_loads(ir_graph *irg, const x86_cconv_t *cconv)
{
	ir_node *start       = get_irg_start(irg);
	ir_node *start_block = get_irg_start_block(irg);
	ir_node *nomem       = get_irg_no_mem(irg);
	ir_node *frame       = get_irg_frame(irg);
	ir_node *proj_args   = get_Proj_for_pn(start, pn_Start_T_args);
	foreach_out_edge_safe(proj_args, edge) {
		ir_node *proj = get_edge_src_irn(edge);
		if (!is_Proj(proj))
			continue;
		unsigned pn   = get_Proj_num(proj);
		const reg_or_stackslot_t *param = &cconv->parameters[pn];
		ir_entity *entity = param->entity;
		if (entity == NULL)
			continue;
		ir_type  *const type   = get_entity_type(entity);
		ir_mode  *const mode   = get_type_mode(type);
		dbg_info *const dbgi   = get_irn_dbg_info(proj);
		ir_node  *const member = new_rd_Member(dbgi, start_block, frame, entity);
		ir_node  *const load   = new_rd_Load(dbgi, start_block, nomem, member, mode, type, cons_none);
		ir_node  *const res    = new_r_Proj(load, mode, pn_Load_res);
		exchange(proj, res);
	}
}

void x86_layout_param_entities(ir_graph *const irg, x86_cconv_t *const cconv,
                               int const parameters_offset)
{
	ir_type *const frame_type = get_irg_frame_type(irg);
	/** Return address is on the stack after that comes the parameters */

	/* Create missing entities and set offsets */
	ir_entity **const param_map = be_collect_parameter_entities(irg);
	size_t      const n_params  = cconv->n_parameters;
	for (size_t p = 0; p < n_params; ++p) {
		reg_or_stackslot_t *param = &cconv->parameters[p];
		if (param->type == NULL)
			continue;

		ir_entity *entity = param_map[p];
		if (entity == NULL)
			entity = new_parameter_entity(frame_type, p, param->type);
		param->entity = entity;
		/* Adjust for return address on stack */
		int const offset = param->offset + parameters_offset;
		set_entity_offset(param->entity, offset);
	}
	free(param_map);

	ir_entity *const entity        = get_irg_entity(irg);
	ir_type   *const function_type = get_entity_type(entity);
	if (is_method_variadic(function_type)) {
		int const offset = cconv->param_stacksize + parameters_offset;
		cconv->va_start_addr = be_make_va_start_entity(frame_type, offset);
	}
}