erichkeane added inline comments.

================
Comment at: clang/lib/Sema/SemaConcept.cpp:773
+  // ConstrExpr for the inner template will properly adjust the depths.
+  if (isa<CXXRecordDecl>(ND) && isa<CXXRecordDecl>(OtherND))
+    ForConstraintInstantiation = true;
----------------
erichkeane wrote:
> erichkeane wrote:
> > alexander-shaposhnikov wrote:
> > > erichkeane wrote:
> > > > alexander-shaposhnikov wrote:
> > > > > erichkeane wrote:
> > > > > > alexander-shaposhnikov wrote:
> > > > > > > erichkeane wrote:
> > > > > > > > Hmm... this seems really strange to have to do. 
> > > > > > > > `ForConstraintInstantiation` shouldn't be used here, the point 
> > > > > > > > of that is to make sure we 'keep looking upward' once we hit a 
> > > > > > > > spot we normally stop with.  What exactly is the issue that you 
> > > > > > > > end up running into here?  Perhaps I can spend some time 
> > > > > > > > debugging what we should really be doign.
> > > > > > > yeah, I agree. I haven't found a proper solution or at least a 
> > > > > > > better workaround (but would be happy to).
> > > > > > > This kicks in for the case 
> > > > > > > 
> > > > > > > ```
> > > > > > > template <class T0>
> > > > > > > concept Constraint = true;
> > > > > > > 
> > > > > > > 
> > > > > > > template<Constraint T1>
> > > > > > > struct Iterator {
> > > > > > >     template <Constraint T2>
> > > > > > >     friend class Iterator;
> > > > > > >     void operator*();
> > > > > > > };
> > > > > > > 
> > > > > > > Iterator<char*> I2;
> > > > > > > ```
> > > > > > > yeah, I agree. I haven't found a proper solution or at least a 
> > > > > > > better workaround (but would be happy to).
> > > > > > > This kicks in for the case 
> > > > > > > 
> > > > > > > ```
> > > > > > > template <class T0>
> > > > > > > concept Constraint = true;
> > > > > > > 
> > > > > > > 
> > > > > > > template<Constraint T1>
> > > > > > > struct Iterator {
> > > > > > >     template <Constraint T2>
> > > > > > >     friend class Iterator;
> > > > > > >     void operator*();
> > > > > > > };
> > > > > > > 
> > > > > > > Iterator<char*> I2;
> > > > > > > ```
> > > > > > 
> > > > > > Alright, well, I should have time later in the week to poke at 
> > > > > > this, perhaps I can come up with something better?  I DO remember 
> > > > > > self-friend is a little wacky, and I spent a bunch of time on it 
> > > > > > last time.
> > > > > Ok, sounds good + maybe Richard will give it another look.
> > > > So IMO, `ForConstraintInstantiation` should be 'true' always, and that 
> > > > makes those examples pass.  However, I'm now seeing that it causes a 
> > > > failure in the concepts-out-of-line-def.cpp file.  
> > > > 
> > > > I took the example of `foo3`:
> > > > 
> > > > ```
> > > >    template<typename T0> concept C = true;
> > > >    template <typename T1>
> > > >    struct S {
> > > >      template <typename F3> requires C<F3>
> > > >      void foo3(F3 f); // #1
> > > >    };
> > > >    template <typename T4>
> > > >    template <typename F6> requires C<F6>
> > > >    void S<T4>::foo3(F6 f) {} // #3
> > > > ```
> > > > 
> > > > Which, seems to require `ForConstraintInstantiation` to be false to 
> > > > pass.  However, I don't think this is correct.  This is only working 
> > > > because when evaluating the in-line one (#1 above!) its skipping the 
> > > > application of `T1`, which is wrong.  
> > > > 
> > > > However, I think the problem here is that the `out of line` version 
> > > > (#3) is not applying the T4 like it should be. SO, I think the 
> > > > `HandleFunctionTemplateDecl` I provided you earlier needs modification.
> > > > 
> > > > FIRST, though not related to this, I think we might need to add 
> > > > `FunctionTemplateDecl::getInjectedTemplateArgs` to the `Result`, but 
> > > > only because that 'sounds right' to me?  IDK what other problem that 
> > > > would cause, but it is worth evaluating/saving for later.  It might 
> > > > just not matter, since we're treating them as equal at the moment, I 
> > > > don't think injecting them would cause anything.
> > > > 
> > > > Secondly: I see that the we can get to the `T4` via the 
> > > > `FTD->getTemplatedDecl()->getQualifier()->getAsType()->getAs<TemplateSpecializationType>()->template_arguments()`.
> > > >   
> > > > 
> > > > HOWEVER, the problem that comes with picking THOSE up, is that it ALSO 
> > > > applies with a `FunctionTemplateDecl` inside of an out-of-line 
> > > > `ClassTemplateSpecializationDecl` (which doesn't NEED the 
> > > > specialization's template args).  
> > > > 
> > > > SO I think the fix here is something in `HandleFunctionTemplateDecl` to 
> > > > make sure we pick up the right list of template arguments.  I haven't 
> > > > figured out the magic incantation to make it work unfortunately, 
> > > > perhaps @rsmith has some hints?  Else I'll keep poking at it.
> > > > 
> > > > BUT, my debugging has shown me that I'm quite convinced the setting 
> > > > `ForConstraintInstantiation ` to false is incorrect.
> > > p.s. Richard is on vacation.
> > > 
> > > to quote his comment above (where the story began):
> > > 
> > > >The right way to fix that and the issue being addressed here is that, 
> > > >rather than adjusting the depths, we ?>should substitute the outer 
> > > >template arguments from the scope specifier (A<int>::) into the 
> > > >constraint before >performing the comparison. (In the special case where 
> > > >none of the outer template parameters are used by the >inner template, 
> > > >that does effectively just adjust the depths of any inner template 
> > > >parameters.)
> > > 
> > > A. the formal meaning of ForConstraintInstantiation=true/false is 
> > > unclear, @eirchkean - if you happen to understand it - mind expanding the 
> > > comment a bit ? (and we can add this to the documentation)
> > > 
> > > B. <related to A> T1 - maybe it should be collected as a retained layer ? 
> > > 
> > I also notice that the work in `HandleRecordDecl` inside of the 
> > `ForConstraintInstantiation` is a little odd, I don't recall why I put that 
> > in place, but it DOES result in 5 failed tests if I remove it.  Perhaps we 
> > just need to be more careful when we substitute in these arguments?  It 
> > seems to me that putting those arguments in place is perhaps not the right 
> > thing to do, since they don't have any meaningful values yet?  Perhaps 
> > something else for us to poke at.
> Ah, didn't know he was away :)  
> 
> My understanding is that is what you're doing (the quote).  By collecting the 
> instantiation args correctly, that is what should be happening here.
> 
> A:
> The `ForConstraintInstantiation`s intent is to make it so we collect 
> arguments 'all the way up' rather than stopping at some points where normal 
> instantiation would.  The idea is that a Constraint is never stored in the 
> AST as substituted, so the depth is always relative to the TU.  Other 
> situations where we collect arguments is relative to the 'last thing that has 
> been instantiated'.  So thats the difference (is where we are substituting 
> 'relative' to).
> 
> B: At this point, we COULD and probably SHOULD collect T1 as a retained outer 
> layer, but that would still cause a problem with that example.  I think the 
> issue here is that we're not ALSO collecting `T4`.
To answer this question:  

```
namespace DelayedInst {
template<unsigned I>
struct AAA {
  template<typename T>
  requires (sizeof(T) == I)
  struct B {
    static_constexpr int a = 0;
  };

  static constexpr auto foo() {
    return B<int>::a;
  }
};
```
Ends up instantiating `B<int>` to get `a` despite `I` not being filled in yet, 
but we still try instantiation into the requires clause for various syntactic 
checking.  So we need something to take up the space of the `I`.  We cant just 
do the retained layer, since `AAA` might be inside of ANOTHER class template 
(perhaps a specialization?), so we'd need that one as well.

So that code goes to try to introduce the `I`.  So I think this solution ends 
up coming down to 'how do we get `T4` into `foo3`'s MLTAL.


Repository:
  rG LLVM Github Monorepo

CHANGES SINCE LAST ACTION
  https://reviews.llvm.org/D146178/new/

https://reviews.llvm.org/D146178

_______________________________________________
cfe-commits mailing list
cfe-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits

Reply via email to