[Mesa-dev] [PATCH 08/11] glsl: lower unconditional returns and continues in loops.
Ian Romanick
idr at freedesktop.org
Wed Jul 6 12:18:32 PDT 2011
-----BEGIN PGP SIGNED MESSAGE-----
Hash: SHA1
On 07/05/2011 03:07 PM, Paul Berry wrote:
> Previously, lower_jumps.cpp would only lower return and continue
> statements that appeared inside conditionals. This patch makes it
> lower unconditional returns and continue statements that occur inside
> a loop.
>
> Such unconditional flow control statements would be unlikely to be
> explicitly coded by a reasonable user, however they might arise as a
> result of other optimizations.
>
> Without this patch, lower_jumps.cpp might not lower certain return and
> continue statements, causing some backends to fail.
>
> Fixes unit tests test_lower_return_void_at_end_of_loop and
> test_remove_continue_at_end_of_loop.
> ---
> src/glsl/lower_jumps.cpp | 64 ++++++++++++++++++++++++++++++++++++++++------
> 1 files changed, 56 insertions(+), 8 deletions(-)
>
> diff --git a/src/glsl/lower_jumps.cpp b/src/glsl/lower_jumps.cpp
> index 73b5d57..51a4cf0 100644
> --- a/src/glsl/lower_jumps.cpp
> +++ b/src/glsl/lower_jumps.cpp
> @@ -304,6 +304,45 @@ struct ir_lower_jumps_visitor : public ir_control_flow_visitor {
> }
> }
>
> + /**
> + * Insert the instructions necessary to lower a return statement,
> + * before the given return instruction.
> + */
> + void insert_lowered_return(ir_return *ir)
> + {
> + ir_variable* return_flag = this->function.get_return_flag();
> + if(!this->function.signature->return_type->is_void()) {
> + ir_variable* return_value = this->function.get_return_value();
> + ir->insert_before(
> + new(ir) ir_assignment(
> + new (ir) ir_dereference_variable(return_value),
> + ir->value,
> + NULL));
The NULL is unnecessary. The constructor has a default value.
> + }
> + ir->insert_before(
> + new(ir) ir_assignment(
> + new (ir) ir_dereference_variable(return_flag),
> + new (ir) ir_constant(true),
> + NULL));
Same as above.
> + this->loop.may_set_return_flag = true;
> + }
> +
> + /**
> + * If the given instruction is a return, lower it to instructions
> + * that store the return value (if there is one), set the return
> + * flag, and then break.
> + *
> + * It is safe to pass NULL to this function.
> + */
> + void lower_return_unconditionally(ir_instruction *ir)
> + {
> + if (get_jump_strength(ir) != strength_return) {
> + return;
> + }
> + insert_lowered_return((ir_return*)ir);
> + ir->replace_with(new(ir) ir_loop_jump(ir_loop_jump::jump_break));
> + }
> +
> virtual void visit(class ir_loop_jump * ir)
> {
> /* Eliminate all instructions after each one, since they are
> @@ -532,13 +571,7 @@ retry: /* we get here if we put code after the if inside a branch */
> * that: 1. store the return value (if this function has a
> * non-void return) and 2. set the return flag
> */
> - ir_variable* return_flag = this->function.get_return_flag();
> - if(!this->function.signature->return_type->is_void()) {
> - ir_variable* return_value = this->function.get_return_value();
> - jumps[lower]->insert_before(new(ir) ir_assignment(new (ir) ir_dereference_variable(return_value), ((ir_return*)jumps[lower])->value, NULL));
> - }
> - jumps[lower]->insert_before(new(ir) ir_assignment(new (ir) ir_dereference_variable(return_flag), new (ir) ir_constant(true), NULL));
> - this->loop.may_set_return_flag = true;
> + insert_lowered_return((ir_return*)jumps[lower]);
> if(this->loop.loop) {
> /* If we are in a loop, replace the return instruction
> * with a break instruction, and then loop so that the
> @@ -761,10 +794,25 @@ lower_continue:
> /* Recursively lower nested jumps. This satisfies the
> * CONTAINED_JUMPS_LOWERED postcondition, except in the case of
> * an unconditional continue or return at the bottom of the
> - * loop.
> + * loop, which are handled below.
> */
> block_record body = visit_block(&ir->body_instructions);
>
> + /* If the loop ends in an unconditional continue, eliminate it
> + * because it is redundant.
> + */
> + ir_instruction *ir_last
> + = (ir_instruction *) ir->body_instructions.get_tail();
> + if (get_jump_strength(ir_last) == strength_continue) {
> + ir_last->remove();
> + }
> +
> + /* If the loop ends in an unconditional return, and we are
> + * lowering returns, lower it.
> + */
> + if (this->function.lower_return)
> + lower_return_unconditionally(ir_last);
> +
> if(body.min_strength >= strength_break) {
> /* FINISHME: If the min_strength of the loop body is
> * strength_break or strength_return, that means that it
-----BEGIN PGP SIGNATURE-----
Version: GnuPG v1.4.11 (GNU/Linux)
Comment: Using GnuPG with Fedora - http://enigmail.mozdev.org/
iEYEARECAAYFAk4UtQgACgkQX1gOwKyEAw/AmwCgggO8uVW8xT5mpyaOcjbJ/STS
1ygAoJuoQvJAX2vrjGMgeFd0H0I8MSbF
=zXzB
-----END PGP SIGNATURE-----
More information about the mesa-dev
mailing list