This is an archive of the discontinued LLVM Phabricator instance.

[clangd] Improve heuristic resolution of dependent types in TargetFinder
ClosedPublic

Authored by nridge on Jun 28 2020, 11:55 PM.

Details

Summary
  • Try to apply heuristic resolution recursively to the base expression of a CXXDependentScopeMemberExpr.
  • Try to apply heuristic resolution recursively to the callee expression in a call expression.

Fixes https://github.com/clangd/clangd/issues/441

Diff Detail

Event Timeline

nridge created this revision.Jun 28 2020, 11:55 PM

Adding some reviewers.

hokein added a comment.Jul 6 2020, 8:01 AM

looks like we use this heuristic for go-to-def, I think we might want to use it in more places, e.g. libindex (for xrefs), sema code completion (so that this->a.^ can suggest something).

clang-tools-extra/clangd/FindTarget.cpp
281

I think this is the key point of the fix?

It would be nice if you could find a way to split it into two (one for refactoring, one for the fix). The diff of this patch contains large chunk of refactoring changes which make the fix less obvious.

nridge marked an inline comment as done.Jul 7 2020, 11:57 PM

looks like we use this heuristic for go-to-def, I think we might want to use it in more places, e.g. libindex (for xrefs), sema code completion (so that this->a.^ can suggest something).

Yeah, this is a great point. I would definitely like to reuse this heuristic resolution for other features like code completion (including in https://github.com/clangd/clangd/issues/443 where I hope to build on it to improve the original STR which involves completion).

I will explore relocating this code to a place where things like code completion can reuse it, in a follow-up patch.

clang-tools-extra/clangd/FindTarget.cpp
281

Yeah, sorry about that. I've split the patch and cleaned it up further (to avoid unnecessary reordering of functions) to make the diff neater.

Split out the refactoring into D83371 which this depends on now.

nridge updated this revision to Diff 276322.Jul 8 2020, 12:06 AM

Improve patch split

hokein added a comment.Jul 8 2020, 8:15 AM

looks like we use this heuristic for go-to-def, I think we might want to use it in more places, e.g. libindex (for xrefs), sema code completion (so that this->a.^ can suggest something).

Yeah, this is a great point. I would definitely like to reuse this heuristic resolution for other features like code completion (including in https://github.com/clangd/clangd/issues/443 where I hope to build on it to improve the original STR which involves completion).

I will explore relocating this code to a place where things like code completion can reuse it, in a follow-up patch.

thanks, this sounds a good plan.

clang-tools-extra/clangd/FindTarget.cpp
196

I'm doubting whether we will miss some other exprs, since we are using it to find the decls for the base expr of CXXDependentScopeMemberExpr.

could you try the following testcase (also add it to the unittest)?

struct A {
  void foo() {}
};
struct B {
  A getA();
};

template <typename T> struct C {
  C c;

  void bar() { this->c.getA().foo(); }
};
281

it looks better now, thanks!

nridge retitled this revision from Improve heuristic resolution of dependent types in TargetFinder to [clangd] Improve heuristic resolution of dependent types in TargetFinder.Jul 8 2020, 1:34 PM
nridge edited the summary of this revision. (Show Details)
nridge updated this revision to Diff 276916.Jul 9 2020, 11:11 PM
nridge marked 4 inline comments as done.

Address review comments

clang-tools-extra/clangd/FindTarget.cpp
196

Thank you for the example! Handling this case required a bit of additional logic, which I've now added.

nridge edited the summary of this revision. (Show Details)Jul 9 2020, 11:12 PM
hokein added inline comments.Jul 13 2020, 2:24 AM
clang-tools-extra/clangd/FindTarget.cpp
74

an off-topic comment: I don't have a reproduce case, but I think we might get a nullptr, and get a null-access crash below,

99

nit: the "a dependent type" doesn't seem to be correct, the type is not always dependent, I'd use a tag-decl type.

117

this function was pretty clear prior to the patch -- given a tag-decl type, and a member name, it tries to perform a lookup in the tag-decl, and returns the result.

Now we add an extra&optional E which seems making it tangled, there is only one caller passing the actual E, so I think we should keep this helper function as-is, and handle the builtin-dependent type in the caller.

214

btw, could you try the case below? it doesn't seem to work.

struct Bar {
  int aaaa;
};
template <typename T> struct Foo {
  Bar func(int);
  void test() {
    func(1).aa^aa;
  }
};
215

we need to check whether the type is null, otherwise we get a crash below.

see:

template<typename T>
struct Foo {
  int func(int);
  void test() {
    func().a;
  }
};
216

could you add a unittest for this case?

nit: I'd remove the extra {} if the if body just contains a single statement.

277

this FIXME looks stale now, could you please update it?

nridge updated this revision to Diff 279051.Jul 18 2020, 10:08 PM
nridge marked 9 inline comments as done.

Address review comments

clang-tools-extra/clangd/FindTarget.cpp
214

Thanks for the additional example. To make this work I had to handle one more case (MemberExpr) in resolveDependentExprToDecls().

277

I think the FIXME is still relevant. While we handle DependentScopeDeclRefExpr and CXXDependentScopeMemberExpr, we still do not handle the AST node types listed in the comment.

For example, we do not handle:

template <typename>
struct A {
  typedef int [[type]];
};

template <typename T>
struct B {
  typedef typename A<T>::t^ype type;
};

(I believe that one would be DependentNameType? You can imagine similar scenarios involving using-decls etc.)

However, the example in the FIXME (which depicts a CXXDependentScopeMemberExpr) is now stale, so I've removed it and clarified the comment a bit.

hokein accepted this revision.Jul 20 2020, 3:43 AM

thanks, looks good.

clang-tools-extra/clangd/FindTarget.cpp
70

nit: remove the {}, and elsewhere.

100

I'd move resolveDependentExprToType close to resolveDependentExprToDecls (they are quite related).

consider putting resolveDependentExprToType after resolveDependentExprToDecls definition, and add a forward declaration of resolveDependentExprToType before the resolveDependentExprToDecls.

202

nit: move the Base to the if below.

206

nit: b -> by

209

this branch is redundant, we can remove it, since getMembersReferencedViaDependentName can handle a nullptr.

This revision is now accepted and ready to land.Jul 20 2020, 3:43 AM
nridge updated this revision to Diff 279426.Jul 20 2020, 11:01 PM
nridge marked 5 inline comments as done.

Address remaining comments

This revision was automatically updated to reflect the committed changes.
hokein added inline comments.Jul 28 2020, 8:00 AM
clang-tools-extra/clangd/FindTarget.cpp
194–195

@nridge, the assertion is not true anymore, since we have extended it to support possibly-non-dependent expressions (callExpr, MemberExpr).

we hit this assert when opening clang/include/clang/ASTMatchers/ASTMatchers.h. I think we probably need to remove it and rename related function names.

hokein added inline comments.Jul 29 2020, 4:34 AM
clang-tools-extra/clangd/FindTarget.cpp
194–195

a reduce case:

template<typename T>
class Foo {
public:
    int foo();
};

class Bar {
public:
static void k();
template <typename T> T convert() const;
};

void func();
template <typename T>
void foo2(Foo<T> t) {
    Bar::k(t.foo()).template convert<T>();
}