irnode.c 22 KB
Newer Older
Christian Würdig's avatar
Christian Würdig committed
1
2
/*
 * This file is part of libFirm.
3
 * Copyright (C) 2012 University of Karlsruhe.
Christian Würdig's avatar
Christian Würdig committed
4
5
 */

Matthias Braun's avatar
Matthias Braun committed
6
7
8
9
/**
 * @file
 * @brief   Representation of an intermediate operation.
 * @author  Martin Trapp, Christian Schaefer, Goetz Lindenmaier, Michael Beck
Götz Lindenmaier's avatar
Götz Lindenmaier committed
10
 */
11
#include <string.h>
Boris Boesler's avatar
added    
Boris Boesler committed
12

13
#include "pset_new.h"
14
#include "ident.h"
Götz Lindenmaier's avatar
Götz Lindenmaier committed
15
#include "irnode_t.h"
Götz Lindenmaier's avatar
Götz Lindenmaier committed
16
#include "irgraph_t.h"
Götz Lindenmaier's avatar
Götz Lindenmaier committed
17
#include "irmode_t.h"
18
#include "irbackedge_t.h"
19
#include "irdump.h"
20
#include "irop_t.h"
21
#include "irprog_t.h"
Sebastian Hack's avatar
Sebastian Hack committed
22
#include "iredgekinds.h"
Sebastian Hack's avatar
Sebastian Hack committed
23
#include "iredges_t.h"
Matthias Braun's avatar
Matthias Braun committed
24
#include "ircons.h"
25
#include "irprintf.h"
Matthias Braun's avatar
Matthias Braun committed
26
#include "panic.h"
27
#include "irverify.h"
Götz Lindenmaier's avatar
Götz Lindenmaier committed
28

29
#include "irhooks.h"
30
#include "util.h"
Michael Beck's avatar
Michael Beck committed
31

32
33
#include "beinfo.h"

Götz Lindenmaier's avatar
Götz Lindenmaier committed
34
35
/* some constants fixing the positions of nodes predecessors
   in the in array */
Michael Beck's avatar
Michael Beck committed
36
#define END_KEEPALIVE_OFFSET  0
Götz Lindenmaier's avatar
Götz Lindenmaier committed
37

38
const char *get_relation_string(ir_relation relation)
39
{
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
	static char const *const relation_names[] = {
		"false",
		"equal",
		"less",
		"less_equal",
		"greater",
		"greater_equal",
		"less_greater",
		"less_equal_greater",
		"unordered",
		"unordered_equal",
		"unordered_less",
		"unordered_less_equal",
		"unordered_greater",
		"unordered_greater_equal",
		"not_equal",
		"true"
	};

	assert((size_t)relation < ARRAY_SIZE(relation_names));
60
	return relation_names[relation];
Christian Schäfer's avatar
Christian Schäfer committed
61
62
}

63
ir_relation get_negated_relation(ir_relation relation)
64
{
65
	return relation ^ ir_relation_true;
Christian Schäfer's avatar
Christian Schäfer committed
66
67
}

68
ir_relation get_inversed_relation(ir_relation relation)
69
{
70
71
72
	ir_relation code    = relation & ~(ir_relation_less|ir_relation_greater);
	bool        less    = relation & ir_relation_less;
	bool        greater = relation & ir_relation_greater;
73
74
	code |= (less ? ir_relation_greater : ir_relation_false)
	      | (greater ? ir_relation_less : ir_relation_false);
75
	return code;
Michael Beck's avatar
Michael Beck committed
76
77
}

78
ir_node *new_ir_node(dbg_info *db, ir_graph *irg, ir_node *block, ir_op *op,
79
                     ir_mode *mode, int arity, ir_node *const *in)
Christian Schäfer's avatar
Christian Schäfer committed
80
{
Matthias Braun's avatar
Matthias Braun committed
81
	assert(mode != NULL);
82
83

	size_t   const node_size = offsetof(ir_node, attr) + op->attr_size;
84
	ir_node *const res       = (ir_node*)OALLOCNZ(get_irg_obstack(irg), char, node_size);
85
86
87
88

	res->kind     = k_ir_node;
	res->op       = op;
	res->mode     = mode;
89
	res->irg      = irg;
90
91
92
	res->node_idx = irg_register_node_idx(irg, res);

	if (arity < 0) {
93
		res->in = NEW_ARR_F(ir_node *, 1);  /* 1: space for block */
94
	} else {
95
96
		/* Nodes with dynamic arity must always have a flexible array. */
		if (op->opar == oparity_dynamic)
97
98
			res->in = NEW_ARR_F(ir_node *, (arity+1));
		else
99
			res->in = NEW_ARR_D(ir_node*, get_irg_obstack(irg), arity + 1);
Christoph Mallon's avatar
Christoph Mallon committed
100
		MEMCPY(&res->in[1], in, arity);
101
102
	}

103
	res->in[0]   = block;
104
105
	set_irn_dbg_info(res, db);
	res->node_nr = get_irp_new_node_nr();
Götz Lindenmaier's avatar
Götz Lindenmaier committed
106

107
	for (ir_edge_kind_t i = EDGE_KIND_FIRST; i <= EDGE_KIND_LAST; ++i) {
108
		INIT_LIST_HEAD(&res->edge_info[i].outs_head);
109
		/* Edges will be built immediately. */
Michael Beck's avatar
Michael Beck committed
110
111
112
		res->edge_info[i].edges_built = 1;
		res->edge_info[i].out_count = 0;
	}
113

114
	/* don't put this into the for loop, arity is -1 for some nodes! */
Matthias Braun's avatar
Matthias Braun committed
115
116
	if (block != NULL)
		edges_notify_edge(res, -1, block, NULL, irg);
117
118
	for (int i = 0; i < arity; ++i)
		edges_notify_edge(res, i, res->in[i+1], NULL, irg);
Sebastian Hack's avatar
Sebastian Hack committed
119

120
	hook_new_node(res);
Matthias Braun's avatar
Matthias Braun committed
121
	if (irg_is_constrained(irg, IR_GRAPH_CONSTRAINT_BACKEND))
122
		be_info_new_node(irg, res);
Michael Beck's avatar
Michael Beck committed
123

124
	return res;
Christian Schäfer's avatar
Christian Schäfer committed
125
126
}

127
128
129
ir_node *new_similar_node(ir_node *const old, ir_node *const block, ir_node **const in)
{
	dbg_info *const dbgi  = get_irn_dbg_info(old);
130
	ir_graph *const irg   = get_irn_irg(old);
131
132
133
134
135
136
137
138
	ir_op    *const op    = get_irn_op(old);
	ir_mode  *const mode  = get_irn_mode(old);
	int       const arity = get_irn_arity(old);
	ir_node  *const n     = new_ir_node(dbgi, irg, block, op, mode, arity, in);
	copy_node_attr(irg, old, n);
	return n;
}

139
140
int (get_irn_arity)(const ir_node *node)
{
141
	return get_irn_arity_(node);
142
143
}

144
void set_irn_in(ir_node *const node, int const arity, ir_node *const *const in)
145
{
146
147
	assert(node != NULL && node->kind == k_ir_node);
	assert(arity >= 0);
Matthias Braun's avatar
Matthias Braun committed
148
149
#ifndef NDEBUG
	for (int i = 0; i < arity; ++i) {
yb9976's avatar
yb9976 committed
150
		assert(in[i] != NULL && in[i]->kind == k_ir_node);
151
152
	}
#endif
153

Matthias Braun's avatar
Matthias Braun committed
154
155
156
	ir_graph  *irg     = get_irn_irg(node);
	ir_node ***pOld_in = &node->in;
	int        i;
157
	for (i = 0; i < arity; i++) {
158
		if (i < (int)ARR_LEN(*pOld_in)-1)
Michael Beck's avatar
Michael Beck committed
159
			edges_notify_edge(node, i, in[i], (*pOld_in)[i+1], irg);
160
		else
Michael Beck's avatar
Michael Beck committed
161
			edges_notify_edge(node, i, in[i], NULL,            irg);
162
	}
163
	for (;i < (int)ARR_LEN(*pOld_in)-1; i++) {
Michael Beck's avatar
Michael Beck committed
164
		edges_notify_edge(node, i, NULL, (*pOld_in)[i+1], irg);
165
166
	}

167
	if (arity != (int)ARR_LEN(*pOld_in) - 1) {
Michael Beck's avatar
Michael Beck committed
168
		ir_node * block = (*pOld_in)[0];
169
		*pOld_in = NEW_ARR_D(ir_node*, get_irg_obstack(irg), arity + 1);
Michael Beck's avatar
Michael Beck committed
170
		(*pOld_in)[0] = block;
171
	}
172
	fix_backedges(get_irg_obstack(irg), node);
173

Christoph Mallon's avatar
Christoph Mallon committed
174
	MEMCPY(*pOld_in + 1, in, arity);
175
176

	/* update irg flags */
177
	clear_irg_properties(irg, IR_GRAPH_PROPERTY_CONSISTENT_OUTS | IR_GRAPH_PROPERTY_CONSISTENT_LOOPINFO);
178
179
}

180
181
ir_node *(get_irn_n)(const ir_node *node, int n)
{
182
	return get_irn_n_(node, n);
Christian Schäfer's avatar
Christian Schäfer committed
183
184
}

185
186
void set_irn_n(ir_node *node, int n, ir_node *in)
{
187
	ir_graph *irg = get_irn_irg(node);
188
189
190
191
	assert(node && node->kind == k_ir_node);
	assert(-1 <= n);
	assert(n < get_irn_arity(node));
	assert(in && in->kind == k_ir_node);
192
	assert(!is_Deleted(in));
193
194
195
196
197

	/* Call the hook */
	hook_set_irn_n(node, n, in, node->in[n + 1]);

	/* Here, we rely on src and tgt being in the current ir graph */
198
	edges_notify_edge(node, n, in, node->in[n + 1], irg);
199
200

	node->in[n + 1] = in;
201
202

	/* update irg flags */
203
	clear_irg_properties(irg, IR_GRAPH_PROPERTY_CONSISTENT_OUTS | IR_GRAPH_PROPERTY_CONSISTENT_LOOPINFO);
Christian Schäfer's avatar
Christian Schäfer committed
204
205
}

206
207
int add_irn_n(ir_node *node, ir_node *in)
{
208
209
	ir_graph *irg = get_irn_irg(node);

210
	assert(is_irn_dynamic(node));
Matthias Braun's avatar
Matthias Braun committed
211
	int pos = ARR_LEN(node->in) - 1;
212
213
214
	ARR_APP1(ir_node *, node->in, in);
	edges_notify_edge(node, pos, node->in[pos + 1], NULL, irg);

215
216
217
	/* Call the hook */
	hook_set_irn_n(node, pos, node->in[pos + 1], NULL);

Matthias Braun's avatar
Matthias Braun committed
218
219
220
	/* update irg flags */
	clear_irg_properties(irg, IR_GRAPH_PROPERTY_CONSISTENT_OUTS);

221
222
223
	return pos;
}

224
static void remove_irn_n(ir_node *node, int n)
Matthias Braun's avatar
Matthias Braun committed
225
{
226
227
228
229
230
231
232
233
	ir_graph *const irg   = get_irn_irg(node);
	int       const arity = get_irn_arity(node);
	ir_node  *const last  = node->in[arity];
	if (n != arity - 1) {
		/* Replace by last edge. */
		ir_node **const slot = &node->in[n + 1];
		ir_node  *const pred = *slot;
		*slot = last;
234
		edges_notify_edge(node, n, last, pred, irg);
Matthias Braun's avatar
Matthias Braun committed
235
	}
236
237
	/* Remove last edge. */
	edges_notify_edge(node, arity - 1, NULL, last, irg);
Matthias Braun's avatar
Matthias Braun committed
238
239
240
241
242
243
	ARR_SHRINKLEN(node->in, arity);

	/* update irg flags */
	clear_irg_properties(irg, IR_GRAPH_PROPERTY_CONSISTENT_OUTS);
}

244
void remove_Sync_n(ir_node *n, int i)
Christoph Mallon's avatar
Christoph Mallon committed
245
{
246
	remove_irn_n(n, i);
Christoph Mallon's avatar
Christoph Mallon committed
247
248
}

Sebastian Hack's avatar
Sebastian Hack committed
249

250
251
ir_mode *(get_irn_mode)(const ir_node *node)
{
252
	return get_irn_mode_(node);
Christian Schäfer's avatar
Christian Schäfer committed
253
254
}

255
256
void (set_irn_mode)(ir_node *node, ir_mode *mode)
{
257
	set_irn_mode_(node, mode);
Till Riedel's avatar
Till Riedel committed
258
259
}

260
261
ir_op *(get_irn_op)(const ir_node *node)
{
262
	return get_irn_op_(node);
Christian Schäfer's avatar
Christian Schäfer committed
263
264
}

265
266
unsigned (get_irn_opcode)(const ir_node *node)
{
267
	return get_irn_opcode_(node);
Götz Lindenmaier's avatar
Götz Lindenmaier committed
268
269
}

270
271
const char *get_irn_opname(const ir_node *node)
{
272
	return get_id_str(node->op->name);
Götz Lindenmaier's avatar
Götz Lindenmaier committed
273
274
}

275
276
ident *get_irn_opident(const ir_node *node)
{
277
278
	assert(node);
	return node->op->name;
279
280
}

281
282
ir_visited_t (get_irn_visited)(const ir_node *node)
{
283
	return get_irn_visited_(node);
284
285
}

286
287
void (set_irn_visited)(ir_node *node, ir_visited_t visited)
{
288
	set_irn_visited_(node, visited);
Christian Schäfer's avatar
Christian Schäfer committed
289
}
290

291
292
void (mark_irn_visited)(ir_node *node)
{
293
	mark_irn_visited_(node);
294
295
}

296
297
int (irn_visited)(const ir_node *node)
{
298
	return irn_visited_(node);
299
300
}

301
302
int (irn_visited_else_mark)(ir_node *node)
{
303
	return irn_visited_else_mark_(node);
304
305
}

306
307
void (set_irn_link)(ir_node *node, void *link)
{
308
	set_irn_link_(node, link);
Christian Schäfer's avatar
Christian Schäfer committed
309
310
}

311
312
void *(get_irn_link)(const ir_node *node)
{
313
	return get_irn_link_(node);
Christian Schäfer's avatar
Christian Schäfer committed
314
315
}

316
int (get_irn_pinned)(const ir_node *node)
317
{
318
	return get_irn_pinned_(node);
319
320
}

321
void set_irn_pinned(ir_node *node, int pinned)
322
{
Matthias Braun's avatar
Matthias Braun committed
323
	assert(get_op_pinned(get_irn_op(node)) >= op_pin_state_exc_pinned);
Michael Beck's avatar
Michael Beck committed
324

325
	node->attr.except.pinned = (pinned != 0);
Michael Beck's avatar
Michael Beck committed
326
}
Götz Lindenmaier's avatar
Götz Lindenmaier committed
327

328
329
long get_irn_node_nr(const ir_node *node)
{
Matthias Braun's avatar
Matthias Braun committed
330
	assert(node->kind == k_ir_node);
331
	return node->node_nr;
332
}
Götz Lindenmaier's avatar
Götz Lindenmaier committed
333

334
335
void *(get_irn_generic_attr)(ir_node *node)
{
336
	assert(node->kind == k_ir_node);
337
	return get_irn_generic_attr_(node);
338
339
}

340
341
const void *(get_irn_generic_attr_const)(const ir_node *node)
{
342
	assert(node->kind == k_ir_node);
343
	return get_irn_generic_attr_const_(node);
344
345
}

346
347
unsigned (get_irn_idx)(const ir_node *node)
{
348
	assert(node->kind == k_ir_node);
349
	return get_irn_idx_(node);
350
351
}

352
ir_node *(get_nodes_block)(const ir_node *node)
353
{
354
	return get_nodes_block_(node);
Christian Schäfer's avatar
Christian Schäfer committed
355
356
}

357
358
void set_nodes_block(ir_node *node, ir_node *block)
{
359
	assert(!is_Block(node));
360
	set_irn_n(node, -1, block);
Christian Schäfer's avatar
Christian Schäfer committed
361
362
}

363
364
ir_node *(get_Block_cfgpred_block)(const ir_node *node, int pos)
{
365
	return get_Block_cfgpred_block_(node, pos);
366
367
}

368
369
int get_Block_matured(const ir_node *node)
{
370
	assert(is_Block(node));
Michael Beck's avatar
Michael Beck committed
371
	return (int)node->attr.block.is_matured;
Christian Schäfer's avatar
Christian Schäfer committed
372
373
}

374
375
void set_Block_matured(ir_node *node, int matured)
{
376
	assert(is_Block(node));
Michael Beck's avatar
Michael Beck committed
377
	node->attr.block.is_matured = matured;
Christian Schäfer's avatar
Christian Schäfer committed
378
}
Michael Beck's avatar
Michael Beck committed
379

380
381
ir_visited_t (get_Block_block_visited)(const ir_node *node)
{
382
	return get_Block_block_visited_(node);
Christian Schäfer's avatar
Christian Schäfer committed
383
384
}

385
386
void (set_Block_block_visited)(ir_node *node, ir_visited_t visit)
{
387
	set_Block_block_visited_(node, visit);
Christian Schäfer's avatar
Christian Schäfer committed
388
389
}

390
391
void (mark_Block_block_visited)(ir_node *node)
{
392
	mark_Block_block_visited_(node);
Götz Lindenmaier's avatar
Götz Lindenmaier committed
393
394
}

395
396
int (Block_block_visited)(const ir_node *node)
{
397
	return Block_block_visited_(node);
Matthias Braun's avatar
Matthias Braun committed
398
399
}

400
401
ir_entity *create_Block_entity(ir_node *block)
{
402
	ir_entity *entity;
Michael Beck's avatar
Michael Beck committed
403
	assert(is_Block(block));
404
405
406

	entity = block->attr.block.entity;
	if (entity == NULL) {
Matthias Braun's avatar
Matthias Braun committed
407
408
		ir_label_t nr = get_irp_next_label_nr();
		entity = new_label_entity(nr);
409
410
		set_entity_visibility(entity, ir_visibility_local);
		set_entity_linkage(entity, IR_LINKAGE_CONSTANT);
411
412
413
414
415
		set_entity_compiler_generated(entity, 1);

		block->attr.block.entity = entity;
	}
	return entity;
Michael Beck's avatar
Michael Beck committed
416
417
}

418
419
ir_node *(get_Block_phis)(const ir_node *block)
{
420
	return get_Block_phis_(block);
421
422
}

423
424
void (set_Block_phis)(ir_node *block, ir_node *phi)
{
425
	set_Block_phis_(block, phi);
426
427
}

428
429
void (add_Block_phi)(ir_node *block, ir_node *phi)
{
430
	add_Block_phi_(block, phi);
Michael Beck's avatar
Michael Beck committed
431
432
}

433
434
unsigned (get_Block_mark)(const ir_node *block)
{
435
	return get_Block_mark_(block);
Michael Beck's avatar
Michael Beck committed
436
437
}

438
439
void (set_Block_mark)(ir_node *block, unsigned mark)
{
440
	set_Block_mark_(block, mark);
Michael Beck's avatar
Michael Beck committed
441
442
}

443
444
void add_End_keepalive(ir_node *end, ir_node *ka)
{
445
	assert(is_End(end));
446
	add_irn_n(end, ka);
447
448
}

449
450
void set_End_keepalives(ir_node *end, int n, ir_node *in[])
{
451
	/* notify that edges are deleted */
Matthias Braun's avatar
Matthias Braun committed
452
453
	ir_graph *irg = get_irn_irg(end);
	for (size_t e = END_KEEPALIVE_OFFSET; e < ARR_LEN(end->in) - 1; ++e) {
454
		edges_notify_edge(end, e, NULL, end->in[e + 1], irg);
455
456
	}
	ARR_RESIZE(ir_node *, end->in, n + 1 + END_KEEPALIVE_OFFSET);
457

Matthias Braun's avatar
Matthias Braun committed
458
	for (int i = 0; i < n; ++i) {
459
		end->in[1 + END_KEEPALIVE_OFFSET + i] = in[i];
460
		edges_notify_edge(end, END_KEEPALIVE_OFFSET + i, end->in[1 + END_KEEPALIVE_OFFSET + i], NULL, irg);
461
	}
462
463

	/* update irg flags */
464
	clear_irg_properties(irg, IR_GRAPH_PROPERTY_CONSISTENT_OUTS);
465
}
466

467
468
469
470
471
void remove_End_n(ir_node *n, int idx)
{
	remove_irn_n(n, idx);
}

Matthias Braun's avatar
Matthias Braun committed
472
void remove_End_keepalive(ir_node *end, const ir_node *irn)
473
{
474
	assert(END_KEEPALIVE_OFFSET == 0);
475
	foreach_irn_in_r(end, i, ka) {
476
477
		if (ka == irn) {
			remove_irn_n(end, i);
Christoph Mallon's avatar
Christoph Mallon committed
478
			return;
479
		}
480
	}
481
}
482

483
void remove_keep_alive(const ir_node *irn)
484
485
486
487
488
489
{
	ir_graph *irg = get_irn_irg(irn);
	ir_node  *end = get_irg_end(irg);

	for (int i = get_End_n_keepalives(end);;) {
		if (i-- == 0)
490
			return;
491
492
493
494

		ir_node *old_ka = end->in[1 + END_KEEPALIVE_OFFSET + i];

		/* find irn */
495
		if (old_ka == irn)
496
497
498
499
			set_irn_n(end, END_KEEPALIVE_OFFSET+i, new_r_Bad(irg, get_irn_mode(irn)));
	}
}

500
501
void remove_End_Bads_and_doublets(ir_node *end)
{
Matthias Braun's avatar
Matthias Braun committed
502
	int n = get_End_n_keepalives(end);
503
504
505
	if (n <= 0)
		return;

Matthias Braun's avatar
Matthias Braun committed
506
507
	ir_graph *irg = get_irn_irg(end);
	pset_new_t keeps;
508
509
	pset_new_init(&keeps);

Matthias Braun's avatar
Matthias Braun committed
510
511
	bool changed = false;
	for (int idx = n; idx-- > 0; ) {
512
513
		ir_node *ka = get_End_keepalive(end, idx);

Michael Beck's avatar
Michael Beck committed
514
		if (is_Bad(ka) || is_NoMem(ka) || pset_new_contains(&keeps, ka)) {
515
			changed = true;
516
			remove_irn_n(end, idx - END_KEEPALIVE_OFFSET);
517
518
519
520
521
522
			--n;
		} else {
			pset_new_insert(&keeps, ka);
		}
	}
	pset_new_destroy(&keeps);
523

Matthias Braun's avatar
Matthias Braun committed
524
	if (changed)
525
		clear_irg_properties(irg, IR_GRAPH_PROPERTY_CONSISTENT_OUTS);
526
527
}

528
529
void free_End(ir_node *end)
{
530
	assert(is_End(end));
531
532
533
534
	end->kind = k_BAD;
	DEL_ARR_F(end->in);
	end->in = NULL;   /* @@@ make sure we get an error if we use the
	                     in array afterwards ... */
535
536
}

537
538
int (is_Const_null)(const ir_node *node)
{
539
	return is_Const_null_(node);
540
541
}

542
543
int (is_Const_one)(const ir_node *node)
{
544
	return is_Const_one_(node);
545
546
}

547
548
int (is_Const_all_one)(const ir_node *node)
{
549
	return is_Const_all_one_(node);
Sebastian Hack's avatar
Sebastian Hack committed
550
551
}

552
553
const char *get_builtin_kind_name(ir_builtin_kind kind)
{
554
#define X(a)    case a: return #a
555
	switch (kind) {
Michael Beck's avatar
Michael Beck committed
556
		X(ir_bk_trap);
557
		X(ir_bk_debugbreak);
558
		X(ir_bk_return_address);
559
		X(ir_bk_frame_address);
560
		X(ir_bk_prefetch);
Michael Beck's avatar
Michael Beck committed
561
562
563
564
565
566
		X(ir_bk_ffs);
		X(ir_bk_clz);
		X(ir_bk_ctz);
		X(ir_bk_popcount);
		X(ir_bk_parity);
		X(ir_bk_bswap);
567
568
		X(ir_bk_inport);
		X(ir_bk_outport);
569
		X(ir_bk_saturating_increment);
570
		X(ir_bk_compare_swap);
Matthias Braun's avatar
Matthias Braun committed
571
		X(ir_bk_may_alias);
572
573
		X(ir_bk_va_start);
		X(ir_bk_va_arg);
574
575
576
577
578
	}
	return "<unknown>";
#undef X
}

579
580
581
ir_entity *get_Call_callee(const ir_node *node)
{
	ir_node *ptr = get_Call_ptr(node);
582
	if (!is_Address(ptr))
583
		return NULL;
584
	ir_entity *entity = get_Address_entity(ptr);
585
586
	/* some (corner case/pointless) graphs can have non-method entities as
	 * call pointers */
587
	if (!is_method_entity(entity) && !is_alias_entity(entity))
588
589
590
591
		return NULL;
	return entity;
}

592
ir_node *get_binop_left(const ir_node *node)
593
{
594
	assert(is_binop(node));
595
	return get_irn_n(node, node->op->op_index);
596
597
}

598
void set_binop_left(ir_node *node, ir_node *left)
599
{
600
	assert(is_binop(node));
601
	set_irn_n(node, node->op->op_index, left);
602
603
}

604
ir_node *get_binop_right(const ir_node *node)
605
{
606
	assert(is_binop(node));
607
	return get_irn_n(node, node->op->op_index + 1);
608
609
}

610
void set_binop_right(ir_node *node, ir_node *right)
611
{
612
	assert(is_binop(node));
613
	set_irn_n(node, node->op->op_index + 1, right);
614
615
}

616
617
ir_node *(get_Phi_next)(const ir_node *phi)
{
618
	return get_Phi_next_(phi);
619
620
}

621
622
void (set_Phi_next)(ir_node *phi, ir_node *next)
{
623
	set_Phi_next_(phi, next);
624
}
625

626
627
int is_memop(const ir_node *node)
{
628
	return is_op_uses_memory(get_irn_op(node));
629
630
}

631
632
ir_node *get_memop_mem(const ir_node *node)
{
633
	const ir_op *op = get_irn_op(node);
634
	assert(is_memop(node));
635
	return get_irn_n(node, op->memory_index);
636
637
}

638
639
void set_memop_mem(ir_node *node, ir_node *mem)
{
640
	const ir_op *op = get_irn_op(node);
641
	assert(is_memop(node));
642
	set_irn_n(node, op->memory_index, mem);
643
644
}

645
646
void add_Sync_pred(ir_node *node, ir_node *pred)
{
647
	assert(is_Sync(node));
648
	add_irn_n(node, pred);
Michael Beck's avatar
Michael Beck committed
649
650
}

Matthias Braun's avatar
Matthias Braun committed
651
652
653
654
int is_x_except_Proj(const ir_node *node)
{
	if (!is_Proj(node))
		return false;
Matthias Braun's avatar
Matthias Braun committed
655
	ir_node *pred = get_Proj_pred(node);
Matthias Braun's avatar
Matthias Braun committed
656
657
	if (!is_fragile_op(pred))
		return false;
658
	return get_Proj_num(node) == pred->op->pn_x_except;
Matthias Braun's avatar
Matthias Braun committed
659
660
661
662
663
664
}

int is_x_regular_Proj(const ir_node *node)
{
	if (!is_Proj(node))
		return false;
Matthias Braun's avatar
Matthias Braun committed
665
	ir_node *pred = get_Proj_pred(node);
Matthias Braun's avatar
Matthias Braun committed
666
667
	if (!is_fragile_op(pred))
		return false;
668
	return get_Proj_num(node) == pred->op->pn_x_regular;
Matthias Braun's avatar
Matthias Braun committed
669
670
}

671
672
673
674
675
676
677
678
679
680
681
682
683
684
void ir_set_throws_exception(ir_node *node, int throws_exception)
{
	except_attr *attr = &node->attr.except;
	assert(is_fragile_op(node));
	attr->throws_exception = throws_exception;
}

int ir_throws_exception(const ir_node *node)
{
	const except_attr *attr = &node->attr.except;
	assert(is_fragile_op(node));
	return attr->throws_exception;
}

685
size_t get_ASM_n_output_constraints(const ir_node *node)
686
{
687
	assert(is_ASM(node));
688
	return ARR_LEN(node->attr.assem.output_constraints);
Michael Beck's avatar
Michael Beck committed
689
}
690

691
size_t get_ASM_n_clobbers(const ir_node *node)
692
{
693
	assert(is_ASM(node));
694
	return ARR_LEN(node->attr.assem.clobbers);
Michael Beck's avatar
Michael Beck committed
695
696
}

697
ir_graph *(get_irn_irg)(const ir_node *node)
698
{
699
	return get_irn_irg_(node);
700
701
}

702
ir_node *skip_Proj(ir_node *node)
703
{
704
705
706
	if (is_Proj(node))
		node = get_Proj_pred(node);
	return node;
Matthias Braun's avatar
Matthias Braun committed
707
708
}

Matthias Braun's avatar
Matthias Braun committed
709
const ir_node *skip_Proj_const(const ir_node *node)
710
{
711
712
713
	if (is_Proj(node))
		node = get_Proj_pred(node);
	return node;
Christian Schäfer's avatar
Christian Schäfer committed
714
715
}

716
ir_node *skip_Tuple(ir_node *node)
717
{
718
restart:
719
	if (is_Proj(node)) {
Matthias Braun's avatar
Matthias Braun committed
720
		ir_node *pred = get_Proj_pred(node);
721

722
		if (is_Proj(pred)) { /* nested Tuple ? */
723
724
		    pred = skip_Tuple(pred);

725
			if (is_Tuple(pred)) {
726
				node = get_Tuple_pred(pred, get_Proj_num(node));
727
728
				goto restart;
			}
729
		} else if (is_Tuple(pred)) {
730
			node = get_Tuple_pred(pred, get_Proj_num(node));
731
732
733
734
			goto restart;
		}
	}
	return node;
735
736
}

737
738
ir_node *skip_Pin(ir_node *node)
{
Michael Beck's avatar
Michael Beck committed
739
740
741
742
743
	if (is_Pin(node))
		return get_Pin_op(node);
	return node;
}

744
745
ir_node *skip_Confirm(ir_node *node)
{
746
	if (is_Confirm(node))
747
748
		return get_Confirm_value(node);
	return node;
Michael Beck's avatar
Michael Beck committed
749
750
}

751
ir_node *skip_Id(ir_node *node)
752
{
753
754
	/* This should compact Id-cycles to self-cycles. It has the same (or less?) complexity
	 * than any other approach, as Id chains are resolved and all point to the real node, or
Matthias Braun's avatar
Matthias Braun committed
755
756
757
	 * all id's are self loops. */
	if (node->op != op_Id)
		return node;
758

759
760
	/* Don't use get_Id_pred():  We get into an endless loop for
	   self-referencing Ids. */
Matthias Braun's avatar
Matthias Braun committed
761
762
763
	ir_node *pred = node->in[0+1];
	if (pred->op != op_Id)
		return pred;
764

765
	if (node != pred) {  /* not a self referencing Id. Resolve Id chain. */
Matthias Braun's avatar
Matthias Braun committed
766
767
768
		if (pred->op != op_Id)
			return pred; /* shortcut */
		ir_node *rem_pred = pred;
769

770
		assert(get_irn_arity (node) > 0);
771

772
		node->in[0+1] = node;   /* turn us into a self referencing Id:  shorten Id cycles. */
Matthias Braun's avatar
Matthias Braun committed
773
774
775
776
		ir_node *res = skip_Id(rem_pred);
		/* self-loop */
		if (is_Id(res))
			return node;
777

778
779
780
781
782
		node->in[0+1] = res;    /* Turn Id chain into Ids all referencing the chain end. */
		return res;
	} else {
		return node;
	}
783
784
}

785
786
int is_cfop(const ir_node *node)
{
787
788
789
	if (is_fragile_op(node) && ir_throws_exception(node))
		return true;

790
	return is_op_cfopcode(get_irn_op(node));
Christian Schäfer's avatar
Christian Schäfer committed
791
792
}

793
794
795
796
797
int is_unknown_jump(const ir_node *node)
{
	return is_op_unknown_jump(get_irn_op(node));
}

798
int is_fragile_op(const ir_node *node)
799
{
800
	return is_op_fragile(get_irn_op(node));
Christian Schäfer's avatar
Christian Schäfer committed
801
}
802