Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Forward-declared consteval function is rejected for use in a template parameter's default value #62224

Closed
cjdb opened this issue Apr 19, 2023 · 10 comments
Assignees
Labels
clang:frontend Language frontend issues, e.g. anything involving "Sema" confirmed Verified by a second party consteval C++20 consteval rejects-valid

Comments

@cjdb
Copy link
Contributor

cjdb commented Apr 19, 2023

GCC test: g++.dg/cpp2a/consteval-defarg3.C

consteval int foober();

int g(int = foober());
struct A { int i = foober(); };
template <int i = foober()> struct B { };
struct C
{
  consteval C(int = foober()) { }
};
int h(C = C());

consteval int foober() { return 42; }

int main() {
  A a;
  B<> b;
  g();
  h();
}

Diagnostic:

<source>:5:19: error: non-type template argument is not a constant expression
template <int i = foober()> struct B { };
                  ^~~~~~~~
<source>:5:19: note: undefined function 'foober' cannot be used in a constant expression
<source>:1:15: note: declared here
consteval int foober();
              ^
<source>:16:3: error: too few template arguments for class template 'B'
  B<> b;
  ^
<source>:5:36: note: template is declared here
template <int i = foober()> struct B { };
~~~~~~~~~~~~~~~~~~~~~~~~~~~        ^

Repro: https://godbolt.org/z/dPPE5TTnr

@cjdb cjdb added clang:frontend Language frontend issues, e.g. anything involving "Sema" rejects-valid consteval C++20 consteval labels Apr 19, 2023
@llvmbot
Copy link
Member

llvmbot commented Apr 19, 2023

@llvm/issue-subscribers-clang-frontend

@shafik
Copy link
Collaborator

shafik commented Apr 19, 2023

This is DR2631 I think this is on @cor3ntin list but maybe not.

@shafik shafik added the confirmed Verified by a second party label Apr 19, 2023
@shafik
Copy link
Collaborator

shafik commented Apr 19, 2023

On second thought, maybe it is not DR2631: https://godbolt.org/z/5fr9PEW99

constexpr int g();
consteval int f() {
  return g();
}

int k(int x = f()) {  // error: constexpr evaluation of undefined function g
  return x;
}

constexpr int g() {
  return 42;
}

int main() {
  return k();
}

Although it looks related.

@cor3ntin
Copy link
Contributor

@shafik at a glance, it looks like CWG2631 - I apparently forgot to implement the case where the default argument appartain to a template parameter. I'll try to put it on my infinite plate.

Thanks for reporting that @cjdb

@Fznamznon
Copy link
Contributor

@cor3ntin , if you don't have enough time for this bug ATM and don't mind, I can take a look.

@cor3ntin
Copy link
Contributor

@Fznamznon If you have time sure, that sounds great! Thanks

@Fznamznon
Copy link
Contributor

Fznamznon commented May 4, 2023

Clang evaluates and checks constant expression in default non-type template argument once it is met by parser, gcc seems to be doing that only if the default value is required. The illustrating example https://godbolt.org/z/vKofvWees
I was able to make clang do the same and the test doesn't error out anymore.
I've noticed there is some amount of work done in https://reviews.llvm.org/D136554 to make std::source_location::current have the value of location where default argument was required. So, Is my understanding correct that something like:

#include <source_location>

template <int sl = std::source_location::current().line()> struct SLS {
  static const int get = sl;
};

static_assert(SLS<>::get == __LINE__);
static_assert(SLS<>::get == __LINE__);
}

Should compile fine? gcc doesn't compile this. https://godbolt.org/z/3o3974fjT

CC @AaronBallman @cor3ntin

@cor3ntin
Copy link
Contributor

cor3ntin commented May 4, 2023

@cor3ntin
Copy link
Contributor

cor3ntin commented May 4, 2023

@Fznamznon I think GCC is correct here. http://eel.is/c++draft/support.srcloc.class#support.srcloc.cons-2 does not mention template default argument at all.

It may be worth confirming with WG21 though.

@Fznamznon
Copy link
Contributor

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
clang:frontend Language frontend issues, e.g. anything involving "Sema" confirmed Verified by a second party consteval C++20 consteval rejects-valid
Projects
Status: No status
Development

No branches or pull requests

5 participants