<div dir="ltr"><div dir="ltr"><div dir="ltr"><div>An example for the duplication please check the <a href="https://github.com/ghc/ghc/blob/master/libraries/integer-simple/GHC/Integer/Type.hs#L373-L380">divModInteger</a> function from integer-simple GHC.Integer.Type.<br></div><div><div>The STG (GHC 8.2.2) generated from <b><span class="gmail-pl-en">divModInteger</span> </b><i><span class="gmail-pl-k">::</span> <span class="gmail-pl-en"><span class="gmail-pl-c1">Integer</span></span> <span class="gmail-pl-k">-></span> <span class="gmail-pl-en"><span class="gmail-pl-c1">Integer</span></span> <span class="gmail-pl-k">-></span> (<span class="gmail-pl-k">#</span> <span class="gmail-pl-en"><span class="gmail-pl-c1">Integer</span></span>, <span class="gmail-pl-en"><span class="gmail-pl-c1">Integer</span></span> <span class="gmail-pl-k">#</span>) </i>contains duplications in a closure binder list.</div><br></div><div>Using my custom STG printer it looks like:</div><div style="margin-left:40px"><font size="1"><font face="monospace,monospace">module GHC.Integer.Type where<br></font></font></div><div style="margin-left:40px"><font size="1"><font face="monospace,monospace"><br></font></font></div><div style="margin-left:40px"><font size="1"><font face="monospace,monospace">using GHC.Prim<br>using GHC.Tuple<br>using GHC.Types<br></font></font></div><div style="margin-left:40px"><font size="1"><font face="monospace,monospace"><br></font></font></div><div style="margin-left:40px"><font size="1"><font face="monospace,monospace">GHC.Integer.Type.divModInteger {-083-} =<br>  closure (F:) (B:<br>  n.s84123 {-s84123-}<br>  d.s84124 {-s84124-}) {<br>  case GHC.Integer.Type.quotRemInteger {-084-}<br>         n.s84123 {-s84123-}<br>         d.s84124 {-s84124-}<br>  of qr.s84125 {-s84125-} {<br>    GHC.Prim.(#,#) {-86-} ipv.s84126 {-s84126-} ipv1.s84127 {-s84127-} -><br>      let $j.s84128 {-s84128-} =<br>            closure (F:<br>            d.s84124 {-s84124-}<br><span style="background-color:rgb(244,204,204)"><span style="color:rgb(0,0,0)"><b>            ipv.s84126 {-s84126-}</b></span></span><br><span style="background-color:rgb(207,226,243)"><span style="color:rgb(0,0,0)"><b>            ipv1.s84127 {-s84127-}</b></span></span><br><span style="background-color:rgb(244,204,204)"><span style="color:rgb(0,0,0)"><b>            ipv.s84126 {-s84126-}</b></span></span><br><span style="color:rgb(0,0,0)"><span style="background-color:rgb(207,226,243)">            <b>ipv1.s84127 {-s84127-}</b></span>) (B:</span><br>            wild.s84129 {-s84129-}) {<br>            let $j1.s84130 {-s84130-} =<br>                  closure (F:<br>                  d.s84124 {-s84124-}<br>                  ipv.s84126 {-s84126-}<br>                  ipv1.s84127 {-s84127-}<br>                  ipv.s84126 {-s84126-}<br>                  ipv1.s84127 {-s84127-}<br>                  wild.s84129 {-s84129-}) (B:<br>                  wild1.s84131 {-s84131-}) {<br>                  case _stg_prim_negateInt#<br>                         wild.s84129 {-s84129-}<br>                  of sat.s84132 {-s84132-} {<br>                    DEFAULT -><br>                      case _stg_prim_==#<br>                             wild1.s84131 {-s84131-}<br>                             sat.s84132 {-s84132-}<br>                      of sat.s84133 {-s84133-} {<br>                        DEFAULT -><br>                          case _stg_prim_tagToEnum#<br>                                 sat.s84133 {-s84133-}<br>                          of wild2.s84134 {-s84134-} {<br>                            GHC.Types.False {-612-} -><br>                              GHC.Prim.(#,#) {-86-}<br>                                ipv.s84126 {-s84126-}<br>                                ipv1.s84127 {-s84127-}<br>                            GHC.Types.True {-645-} -><br>                              case GHC.Integer.Type.plusInteger {-066-}<br>                                     ipv1.s84127 {-s84127-}<br>                                     d.s84124 {-s84124-}<br>                              of r'.s84135 {-s84135-} {<br>                                DEFAULT -><br>                                  case GHC.Integer.Type.plusInteger {-066-}<br>                                         ipv.s84126 {-s84126-}<br>                                         GHC.Integer.Type.lvl11 {-r50574-}<br>                                  of q'.s84136 {-s84136-} {<br>                                    DEFAULT -><br>                                      GHC.Prim.(#,#) {-86-}<br>                                        q'.s84136 {-s84136-}<br>                                        r'.s84135 {-s84135-}<br>                                  }<br>                              }<br>                          }<br>                      }<br>                  }}<br>                <br>            in case ipv1.s84127 {-s84127-} <br>               of wild1.s84137 {-s84137-} {<br>                 GHC.Integer.Type.S# {-621-} i#.s84138 {-s84138-} -><br>                   case _stg_prim_<#<br>                          i#.s84138 {-s84138-} 0#<br>                   of sat.s84140 {-s84140-} {<br>                     DEFAULT -><br>                       case _stg_prim_>#<br>                              i#.s84138 {-s84138-} 0#<br>                       of sat.s84139 {-s84139-} {<br>                         DEFAULT -><br>                           case _stg_prim_-#<br>                                  sat.s84139 {-s84139-}<br>                                  sat.s84140 {-s84140-}<br>                           of sat.s84141 {-s84141-} {<br>                             DEFAULT -><br>                               $j1.s84130 {-s84130-}<br>                                 sat.s84141 {-s84141-}<br>                           }<br>                       }<br>                   }<br>                 <a href="http://GHC.Integer.Type.Jp#">GHC.Integer.Type.Jp#</a> {-r5813-} dt.s84142 {-s84142-} -><br>                   $j1.s84130 {-s84130-} 1#<br>                 GHC.Integer.Type.Jn# {-r5814-} dt.s84143 {-s84143-} -><br>                   $j1.s84130 {-s84130-} -1#<br>               }}<br>          <br>      in case d.s84124 {-s84124-} <br>         of wild.s84144 {-s84144-} {<br>           GHC.Integer.Type.S# {-621-} i#.s84145 {-s84145-} -><br>             case _stg_prim_<#<br>                    i#.s84145 {-s84145-} 0#<br>             of sat.s84147 {-s84147-} {<br>               DEFAULT -><br>                 case _stg_prim_>#<br>                        i#.s84145 {-s84145-} 0#<br>                 of sat.s84146 {-s84146-} {<br>                   DEFAULT -><br>                     case _stg_prim_-#<br>                            sat.s84146 {-s84146-}<br>                            sat.s84147 {-s84147-}<br>                     of sat.s84148 {-s84148-} {<br>                       DEFAULT -><br>                         $j.s84128 {-s84128-}<br>                           sat.s84148 {-s84148-}<br>                     }<br>                 }<br>             }<br>           <a href="http://GHC.Integer.Type.Jp#">GHC.Integer.Type.Jp#</a> {-r5813-} dt.s84149 {-s84149-} -><br>             $j.s84128 {-s84128-} 1#<br>           GHC.Integer.Type.Jn# {-r5814-} dt.s84150 {-s84150-} -><br>             $j.s84128 {-s84128-} -1#<br>         }<br>  }}<br></font></font><br></div></div></div></div><br><div class="gmail_quote"><div dir="ltr">On Mon, Nov 5, 2018 at 2:08 PM Simon Peyton Jones <<a href="mailto:simonpj@microsoft.com">simonpj@microsoft.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">





<div lang="EN-GB" link="blue" vlink="purple">
<div class="m_-3312551194861425370WordSection1">
<p class="MsoNormal"><span style="font-size:12.0pt">I don’t think there should be duplicates in either. Do you have a test case that shows duplicates?<u></u><u></u></span></p>
<p class="MsoNormal"><span style="font-size:12.0pt"><u></u> <u></u></span></p>
<p class="MsoNormal"><span style="font-size:12.0pt">Simon<u></u><u></u></span></p>
<p class="MsoNormal"><span style="font-size:12.0pt"><u></u> <u></u></span></p>
<div style="border:none;border-left:solid blue 1.5pt;padding:0cm 0cm 0cm 4.0pt">
<div>
<div style="border:none;border-top:solid #e1e1e1 1.0pt;padding:3.0pt 0cm 0cm 0cm">
<p class="MsoNormal"><b><span lang="EN-US">From:</span></b><span lang="EN-US"> ghc-devs <<a href="mailto:ghc-devs-bounces@haskell.org" target="_blank">ghc-devs-bounces@haskell.org</a>>
<b>On Behalf Of </b>Csaba Hruska<br>
<b>Sent:</b> 04 November 2018 11:22<br>
<b>To:</b> <a href="mailto:ghc-devs@haskell.org" target="_blank">ghc-devs@haskell.org</a><br>
<b>Subject:</b> Re: StgRhsClosure freevar and argument name duplicates<u></u><u></u></span></p>
</div>
</div>
<p class="MsoNormal"><u></u> <u></u></p>
<div>
<div>
<div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
Is it possible that GHC generates STG with invalid binding semantics for certain cases that the Cmm codegen fix or ignore?<u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
This could explain my observations.<u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
I've checked the Stg linter source (StgLint.hs ; GHC 8.2.2 and github master) and it does not check StgRhsClosure free var and binder list at all.<u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
And the scope checker function (addInScopeVars) does not check for duplicates.<u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
<u></u> <u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
Any thoughts?<u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
<u></u> <u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
Cheers,<u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
Csaba<u></u><u></u></p>
</div>
</div>
</div>
</div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
<u></u> <u></u></p>
<div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
On Sat, Nov 3, 2018 at 9:53 AM Csaba Hruska <<a href="mailto:csaba.hruska@gmail.com" target="_blank">csaba.hruska@gmail.com</a>> wrote:<u></u><u></u></p>
</div>
<blockquote style="border:none;border-left:solid #cccccc 1.0pt;padding:0cm 0cm 0cm 6.0pt;margin-left:4.8pt;margin-right:0cm">
<div>
<div>
<div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
Hi,<u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
<u></u> <u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
Can StgRhsClosure's freevar list ([occ]) or argument list ([bndr]) contain duplicates?<u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
<u></u> <u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
Cheers,<u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
Csaba<u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
<u></u> <u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
<span style="font-size:7.5pt;font-family:"Courier New"">data GenStgRhs bndr occ</span><br>
<span style="font-size:7.5pt;font-family:"Courier New"">  = StgRhsClosure</span><br>
<span style="font-size:7.5pt;font-family:"Courier New"">        CostCentreStack         -- CCS to be attached (default is CurrentCCS)</span><br>
<span style="font-size:7.5pt;font-family:"Courier New"">        StgBinderInfo           -- Info about how this binder is used (see below)</span><br>
<span style="font-size:7.5pt;font-family:"Courier New"">        <b>[occ]</b>                   -- non-global free vars; a list, rather than</span><br>
<span style="font-size:7.5pt;font-family:"Courier New"">                                -- a set, because order is important</span><br>
<span style="font-size:7.5pt;font-family:"Courier New"">        !UpdateFlag             -- ReEntrant | Updatable | SingleEntry</span><br>
<span style="font-size:7.5pt;font-family:"Courier New"">        <b>[bndr]</b>                  -- arguments; if empty, then not a function;</span><br>
<span style="font-size:7.5pt;font-family:"Courier New"">                                -- as above, order is important.</span><br>
<span style="font-size:7.5pt;font-family:"Courier New"">        (GenStgExpr bndr occ)   -- body</span><u></u><u></u></p>
</div>
<div>
<p class="MsoNormal" style="margin-right:0cm;margin-bottom:6.0pt;margin-left:0cm">
<u></u> <u></u></p>
</div>
</div>
</div>
</div>
</blockquote>
</div>
</div>
</div>
</div>

</blockquote></div>