2 * Copyright © 2010 Intel Corporation
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
11 * The above copyright notice and this permission notice (including the next
12 * paragraph) shall be included in all copies or substantial portions of the
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
21 * DEALINGS IN THE SOFTWARE.
25 * \file ir_function_can_inline.cpp
27 * Determines if we can inline a function call using ir_function_inlining.cpp.
29 * The primary restriction is that we can't return from the function
30 * other than as the last instruction. We could potentially work
31 * around this for some constructs by flattening control flow and
32 * moving the return to the end, or by using breaks from a do {} while
33 * (0) loop surrounding the function body.
38 #include "ir_visitor.h"
39 #include "ir_function_inlining.h"
40 #include "ir_expression_flattening.h"
41 #include "glsl_types.h"
43 class ir_function_can_inline_visitor
: public ir_visitor
{
45 ir_function_can_inline_visitor()
47 this->can_inline
= true;
48 this->num_returns
= 0;
54 * As typical for the visitor pattern, there must be one \c visit method for
55 * each concrete subclass of \c ir_instruction. Virtual base classes within
56 * the hierarchy should not have \c visit methods.
59 virtual void visit(ir_variable
*);
60 virtual void visit(ir_label
*);
61 virtual void visit(ir_loop
*);
62 virtual void visit(ir_loop_jump
*);
63 virtual void visit(ir_function_signature
*);
64 virtual void visit(ir_function
*);
65 virtual void visit(ir_expression
*);
66 virtual void visit(ir_swizzle
*);
67 virtual void visit(ir_dereference
*);
68 virtual void visit(ir_assignment
*);
69 virtual void visit(ir_constant
*);
70 virtual void visit(ir_call
*);
71 virtual void visit(ir_return
*);
72 virtual void visit(ir_if
*);
80 ir_function_can_inline_visitor::visit(ir_variable
*ir
)
86 ir_function_can_inline_visitor::visit(ir_label
*ir
)
92 ir_function_can_inline_visitor::visit(ir_loop
*ir
)
94 /* FINISHME: Implement loop cloning in ir_function_inlining.cpp */
95 this->can_inline
= false;
98 ir
->from
->accept(this);
100 ir
->to
->accept(this);
102 ir
->increment
->accept(this);
104 foreach_iter(exec_list_iterator
, iter
, ir
->body_instructions
) {
105 ir_instruction
*inner_ir
= (ir_instruction
*)iter
.get();
106 inner_ir
->accept(this);
111 ir_function_can_inline_visitor::visit(ir_loop_jump
*ir
)
118 ir_function_can_inline_visitor::visit(ir_function_signature
*ir
)
125 ir_function_can_inline_visitor::visit(ir_function
*ir
)
131 ir_function_can_inline_visitor::visit(ir_expression
*ir
)
133 unsigned int operand
;
135 for (operand
= 0; operand
< ir
->get_num_operands(); operand
++) {
136 ir
->operands
[operand
]->accept(this);
142 ir_function_can_inline_visitor::visit(ir_swizzle
*ir
)
144 ir
->val
->accept(this);
148 ir_function_can_inline_visitor::visit(ir_dereference
*ir
)
150 ir
->var
->accept(this);
151 if (ir
->mode
== ir_dereference::ir_reference_array
)
152 ir
->selector
.array_index
->accept(this);
156 ir_function_can_inline_visitor::visit(ir_assignment
*ir
)
158 ir
->lhs
->accept(this);
159 ir
->rhs
->accept(this);
161 ir
->condition
->accept(this);
166 ir_function_can_inline_visitor::visit(ir_constant
*ir
)
173 ir_function_can_inline_visitor::visit(ir_call
*ir
)
175 foreach_iter(exec_list_iterator
, iter
, *ir
) {
176 ir_rvalue
*param
= (ir_rvalue
*)iter
.get();
184 ir_function_can_inline_visitor::visit(ir_return
*ir
)
186 ir
->get_value()->accept(this);
193 ir_function_can_inline_visitor::visit(ir_if
*ir
)
195 /* FINISHME: Implement if cloning in ir_function_inlining.cpp. */
196 this->can_inline
= false;
198 ir
->condition
->accept(this);
200 foreach_iter(exec_list_iterator
, iter
, ir
->then_instructions
) {
201 ir_instruction
*inner_ir
= (ir_instruction
*)iter
.get();
202 inner_ir
->accept(this);
205 foreach_iter(exec_list_iterator
, iter
, ir
->else_instructions
) {
206 ir_instruction
*inner_ir
= (ir_instruction
*)iter
.get();
207 inner_ir
->accept(this);
212 can_inline(ir_call
*call
)
214 ir_function_can_inline_visitor v
;
215 const ir_function_signature
*callee
= call
->get_callee();
217 foreach_iter(exec_list_iterator
, iter
, callee
->body
) {
218 ir_instruction
*ir
= (ir_instruction
*)iter
.get();
222 ir_instruction
*last
= (ir_instruction
*)callee
->body
.get_tail();
223 if (last
&& !last
->as_return())
226 return v
.can_inline
&& v
.num_returns
== 1;