Issue 171074
Summary [Clang] Confusing diagnostic about implicit use of `this` involving immediate-escalating function
Labels clang:diagnostics
Assignees
Reporter Sirraide
    Consider (https://godbolt.org/z/e1qnxsMeY):
```c++
struct S {
    int value = 42;
    void f(auto visitor) {
        visitor(value);
    }
};

void not_constexpr();
consteval void g(bool b) {
    if (b) not_constexpr();
}

void f() {
    S s;
    s.f([](int value) { g(true); });
}
```
For this, we emit the following diagnostics:
```console
<source>:4:9: error: call to immediate function 'f()::(anonymous class)::operator()' is not a constant _expression_
    4 |         visitor(value);
      |         ^
<source>:15:7: note: in instantiation of function template specialization 'S::f<(lambda at <source>:15:9)>' requested here
   15 |     s.f([](int value) { g(true); });
      |       ^
<source>:4:17: note: implicit use of 'this' pointer is only allowed within the evaluation of a call to a 'constexpr' member function
    4 |         visitor(value);
      |          
```

The wording ‘implicit use of 'this' pointer is only allowed within the evaluation of a call to a 'constexpr' member function’ (which is `note_constexpr_this` in DiagnosticASTKinds.ts) is... not great. What this is probably trying to say is that `this` is not a constant _expression_ here (this diagnostic is emitted if we can’t find a `this` pointer in the evaluation stack during constant evaluation), but it doesn’t exactly communicate that—and the wording is outright wrong in that it implies that implicit `this` requires a function to be `constexpr`, which is just not the case.

There’s also something else that confuses me here: Removing the call to `not_constexpr()` in `g()` (or just passing `false` to `g()`) causes the evaluation to succeed (and we don’t even complain about `this`; instead we compile the program just fine), and GCC does the same (in fact, GCC also complains about `*(S*)this` not being a constant _expression_). 

CC @cor3ntin Is this some weird consequence of how immediate escalation works? Because to me it really seems that we should be complaining about the call to `not_constexpr()`... not being `constexpr`, considering that we fail evaluation iff that call is reached.

For a more real-world example, if that call to `not_constexpr()` is instead a call to `std::format()`/`std::print()` with a missing format argument (e.g. https://godbolt.org/z/cavKqnYY4), then we never actually tell the user about that, and instead the only diagnostics you get are the ones seen above, which is rather poor QOI. I ran into this exact situation earlier, and it took me a while to figure out what the problem was.

Either way, I still think that `note_constexpr_this` needs to be reworded a bit.
_______________________________________________
llvm-bugs mailing list
[email protected]
https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-bugs

Reply via email to