v1nh1shungry added inline comments.

================
Comment at: clang-tools-extra/clangd/Hover.cpp:994
       HI.CalleeArgInfo.emplace(toHoverInfoParam(PVD, PP));
+      PassType.PassBy = getPassMode(PVD->getType());
+    }
----------------
kadircet wrote:
> v1nh1shungry wrote:
> > kadircet wrote:
> > > v1nh1shungry wrote:
> > > > kadircet wrote:
> > > > > v1nh1shungry wrote:
> > > > > > kadircet wrote:
> > > > > > > sorry for showing up late to the party but instead of changing 
> > > > > > > rest of the code, can we apply this logic only when there are no 
> > > > > > > implicit casts/conversions between the arg and callexpr (i.e `N 
> > > > > > > == &OuterNode`)?
> > > > > > To make sure I understand it correctly, do you mean I should give 
> > > > > > up any other code changes I made but keep this logic, and put this 
> > > > > > logic into the `N == &OuterNode` branch?
> > > > > > 
> > > > > > Sorry if I misunderstood anything! Shame for not being a good 
> > > > > > reader :(
> > > > > > To make sure I understand it correctly, do you mean I should give 
> > > > > > up any other code changes I made but keep this logic, and put this 
> > > > > > logic into the N == &OuterNode branch?
> > > > > 
> > > > > Somewhat.
> > > > > 
> > > > > Basically this code had the assumption that if we don't see any 
> > > > > casts/conversions between the expression creating the argument and 
> > > > > the expression getting passed to the callexpr, it must be passed by 
> > > > > reference, and this was wrong. Even before the patch that added 
> > > > > handling for literals.
> > > > > 
> > > > > The rest of the handling for casts/conversions/constructors in 
> > > > > between have been working so far and was constructed based on what 
> > > > > each particular cast does, not for specific cases hence they're 
> > > > > easier (for the lack of a better word) to reason about. Hence I'd 
> > > > > rather keep them as is, especially the changes in handling 
> > > > > `MaterializeTemporaryExpr` don't sound right. I can see the example 
> > > > > you've at hand, but we shouldn't be producing "converted" results for 
> > > > > it anyway (the AST should have a NoOp implicit cast to `const int` 
> > > > > and then a `MaterializeTemporaryExpr`, which shouldn't generated any 
> > > > > converted signals with the existing code already).
> > > > > 
> > > > > Hence the my proposal is getting rid of the assumption around "if we 
> > > > > don't see any casts/conversions between the expression creating the 
> > > > > argument and the expression getting passed to the callexpr, it must 
> > > > > be passed by reference", and use the type information in 
> > > > > `ParmVarDecl` directly when we don't have any implicit nodes in 
> > > > > between to infer `PassBy`.
> > > > > This should imply also getting rid of the special case for literals 
> > > > > (`if (isLiteral(E) && N->Parent == OuterNode.Parent)`).
> > > > > 
> > > > > Does that make sense?
> > > > Thanks for the detailed explanation! But before we go ahead here, what 
> > > > do you think about the new test case I'm talking about above? Do you 
> > > > agree with my conclusion?
> > > i suppose you mean:
> > > 
> > > ```
> > > void foobar(const float &);
> > > int main() {
> > >   foobar(0);
> > >               ^
> > > }
> > > ```
> > > 
> > > first of all the version of the patch that i propose doesn't involve any 
> > > changes in behaviour here (as we actually have an implicit cast in 
> > > between, and i am suggesting finding out passby based on type of the 
> > > parmvardecl only when there are no casts in between).
> > > 
> > > i can see @nridge 's reasoning about indicating copies by saying pass by 
> > > value vs ref, which unfortunately doesn't align with C++ semantics 
> > > directly (as what we have here is a prvalue, and it is indeed passed by 
> > > value, without any copies to the callee).
> > > 
> > > it isn't very obvious anywhere but the main functionality we wanted to 
> > > provide to the developer was help them figure out if a function call can 
> > > mutate a parameter they were passing in, therefore it didn't prioritise 
> > > literals at all. we probably should've made better wording choices in the 
> > > UI and talked about "immutability". hence from functionality point of 
> > > view calling this pass by `value` vs `const ref` doesn't make a huge 
> > > difference (but that's probably only in my mind and people are already 
> > > using it to infer other things like whether we're going to trigger 
> > > copies).
> > > 
> > > so i'd actually leave this case as-is, and think about what we're 
> > > actually trying to provide by showing arg info on literals. as it's 
> > > currently trying to overload the meaning of `passby` and causing 
> > > confusions. since the initial intent was to just convey "immutability" 
> > > one suggestion would be to just hide the `passby` information for 
> > > literals.
> > > otherwise from value categories point of view, these are always passed by 
> > > value, but this is going to create confusion for people that are using it 
> > > to infer "copies" and getting that right, while preserving the semantics 
> > > around "is this mutable" just complicates things.
> > > 
> > > best thing moving forward would probably be to just have two separate 
> > > fields, one indicating mutability and another indicating copies and not 
> > > talking about pass by type at all.
> > > 
> > > ---
> > > 
> > > sorry for the lengthy answer, LMK if it needs clarification.
> > > hence from functionality point of view calling this pass by value vs 
> > > const ref doesn't make a huge difference
> > 
> > Agree. But since we now choose to provide information in such a way, we 
> > should keep it as correct as we can. But for this case, I'm good with both 
> > two (indeed I now prefer `value` because of the prvalue, I was confused 
> > again :/ )
> > 
> > However, when I try to follow your proposal, I find the current code is 
> > somehow complicated for me to understand. You said we can get rid of `if 
> > (isLiteral(E) && N->Parent == OuterNode.Parent)`, that's right, but how 
> > about the rest? 
> > 
> > ```
> >   if (const auto *E = N->ASTNode.get<Expr>()) {
> >     if (E->getType().isConstQualified())
> >       PassType.PassBy = HoverInfo::PassType::ConstRef;
> >   }
> > ```
> > 
> > Whether it exists the tests won't break. If we remove it, the `CK_NoOp` 
> > cases seem to rely on the default value of `HoverInfo::PassType` when 
> > there's an implicit cast. If we keep this, why we're saying if the 
> > expression under the cursor is const-qualified then the `PassType.PassBy` 
> > will be `ConstRef`? What about `const int`?
> > 
> > So I think if we want to distinguish the cast case and the non-cast case, 
> > it needs a better starting point for the cast case. Or we get the 
> > `PassType::PassBy` through the parameter's type, which seems to be more 
> > intuitive, and then do the correction according to the cast kind if we'd 
> > like. I think it's kind of the same as the original implementation. They 
> > both are based on the cast kinds.
> > 
> > > especially the changes in handling MaterializeTemporaryExpr don't sound 
> > > right. I can see the example you've at hand, but we shouldn't be 
> > > producing "converted" results for it anyway
> > 
> > Remove the code inside the branch and it'll behave as the same as the 
> > original implementation.
> > 
> > > best thing moving forward would probably be to just have two separate 
> > > fields, one indicating mutability and another indicating copies and not 
> > > talking about pass by type at all.
> > 
> > +1
> > Whether it exists the tests won't break. If we remove it, the CK_NoOp cases 
> > seem to rely on the default value of HoverInfo::PassType when there's an 
> > implicit cast. If we keep this, why we're saying if the expression under 
> > the cursor is const-qualified then the PassType.PassBy will be ConstRef? 
> > What about const int?
> 
> It's unfortunate that we don't have an existing test case, but I guess 
> something like the following should demonstrate it's need:
> ```
> struct Foo {
>   Foo(const int &);
> };
> void foo(Foo);
> void bar() {
>   const int x = 0;
>   foo(^x);
> }
> ```
> 
> In the absence of that logic, we'd say `passed by reference` and not `by 
> const reference`. 
> 
> > So I think if we want to distinguish the cast case and the non-cast case, 
> > it needs a better starting point for the cast case. Or we get the 
> > PassType::PassBy through the parameter's type, which seems to be more 
> > intuitive, and then do the correction according to the cast kind if we'd 
> > like. I think it's kind of the same as the original implementation. They 
> > both are based on the cast kinds.
> 
> I think this logic is setting up the right starting point for the case when 
> we have implicit nodes in between. We should just make sure we bail out early 
> when we don't have any implicit nodes in between and don't execute this 
> logic. I agree that it just complicates things for that case.
Thanks for the test case! Since @kadircet supposes to keep it as-is, I'm happy 
to follow the proposal. I'm going to restore the code change.


Repository:
  rG LLVM Github Monorepo

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

https://reviews.llvm.org/D142014

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

Reply via email to