<div dir="ltr"><div class="gmail_quote"><div dir="ltr" class="gmail_attr">On Fri, Mar 8, 2019 at 9:30 AM Juan A. Suarez Romero <<a href="mailto:jasuarez@igalia.com">jasuarez@igalia.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">On Thu, 2019-03-07 at 07:15 -0600, Jason Ekstrand wrote:<br>
> Woah, is this legal SPIR-V? I think a second OpSelectionMerge is required.<br>
<br>
I'd say it is legal. The spec does not mandate that each branch has its own<br>
merge instruction; only that the control flow must be structured for shaders.<br></blockquote><div><br></div><div>That's exactly the problem.  It says how merge instructions work and how they have to be nested but never that or where you have to use them. :-(  This is a spec bug.  The closest I can find is this line from the structured control flow section: "These explicitly declare a <a href="https://www.khronos.org/registry/spir-v/specs/unified1/SPIRV.html#HeaderBlock">header block</a> before the control flow diverges and a <a href="https://www.khronos.org/registry/spir-v/specs/unified1/SPIRV.html#MergeBlock">merge block</a> where control flow subsequently converges."  If you read that as "you must declare divergence" then it would imply that every OpBranchConditional must be preceded by an OpSelectionMerge.  If we don't require this, there are lots of weird cases where you can get into diamond patters and other things that aren't trivially structurizable.</div><div><br></div><div>--Jason<br></div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
In section 2.11 ("Structured Control Flow"), about structured control-flow<br>
constructs:<br>
<br>
<br>
- A structured control-flow construct is then defined as one of:<br>
  - a selection construct: the set of blocks dominated by a selection header,<br>
minus the set of blocks dominated by the header’s merge block<br>
  - [...]<br>
<br>
- The above structured control-flow constructs must satisfy the following rules:<br>
  - [...]<br>
  - the only blocks in a construct that can branch outside the construct are<br>
    - a block branching to the construct’s merge block<br>
    - a block branching from one case construct to another, for the same<br>
OpSwitch<br>
    - a back-edge block<br>
    - a continue block for the innermost loop it is nested inside of<br>
    - a break block for the innermost loop it is nested inside of<br>
    - a return block<br>
<br>
<br>
Our selection construct, which contains the two nested conditional branches,<br>
satisfies the rules, as both conditionals branches to the construct's merge<br>
block.<br>
<br>
<br>
        J.A.<br>
<br>
<br>
<br>
<br>
<br>
<br>
> <br>
<br>
<br>
> --Jason<br>
> <br>
> <br>
> On March 6, 2019 05:25:26 "Juan A. Suarez Romero" <<a href="mailto:jasuarez@igalia.com" target="_blank">jasuarez@igalia.com</a>> wrote:<br>
> <br>
> > This fixes the case when the SPIR-V code has two nested conditional<br>
> > branches, but only one selection merge:<br>
> > <br>
> > <br>
> > [...]<br>
> > %1 = OpLabel<br>
> >     OpSelectionMerge %2 None<br>
> >     OpBranchConditional %3 %4 %2<br>
> > %4 = OpLabel<br>
> >     OpBranchConditional %3 %5 %2<br>
> > %5 = OpLabel<br>
> >     OpBranch %2<br>
> > %2 = OpLabel<br>
> > [...]<br>
> > <br>
> > <br>
> > In the second OpBranchConditional, as the else-part is the end<br>
> > block (started in the first OpBranchConditional) we can just follow the<br>
> > then-part.<br>
> > <br>
> > <br>
> > This fixes dEQP-VK.vkrunner.controlflow.2-obc-triangle-triangle<br>
> > <br>
> > <br>
> > CC: Jason Ekstrand <<a href="mailto:jason@jlekstrand.net" target="_blank">jason@jlekstrand.net</a>><br>
> > ---<br>
> > src/compiler/spirv/vtn_cfg.c | 11 ++++++++++-<br>
> > 1 file changed, 10 insertions(+), 1 deletion(-)<br>
> > <br>
> > <br>
> > diff --git a/src/compiler/spirv/vtn_cfg.c b/src/compiler/spirv/vtn_cfg.c<br>
> > index 7868eeb60bc..f749118efbe 100644<br>
> > --- a/src/compiler/spirv/vtn_cfg.c<br>
> > +++ b/src/compiler/spirv/vtn_cfg.c<br>
> > @@ -605,7 +605,16 @@ vtn_cfg_walk_blocks(struct vtn_builder *b, struct <br>
> > list_head *cf_list,<br>
> >             }<br>
> >          } else if (if_stmt->then_type == vtn_branch_type_none &&<br>
> >                     if_stmt->else_type == vtn_branch_type_none) {<br>
> > -            /* Neither side of the if is something we can short-circuit. */<br>
> > +            /* Neither side of the if is something we can short-circuit,<br>
> > +             * unless one of the blocks is the end block. */<br>
> > +            if (then_block == end) {<br>
> > +               block = else_block;<br>
> > +               continue;<br>
> > +            } else if (else_block == end) {<br>
> > +               block = then_block;<br>
> > +               continue;<br>
> > +            }<br>
> > +<br>
> >             vtn_assert((*block->merge & SpvOpCodeMask) == SpvOpSelectionMerge);<br>
> >             struct vtn_block *merge_block =<br>
> >                vtn_value(b, block->merge[1], vtn_value_type_block)->block;<br>
> > --<br>
> > 2.20.1<br>
> <br>
> <br>
> <br>
<br>
</blockquote></div></div>