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

False positive error: pack expansion contains parameter pack '_UTypes' that has a different length (1 vs. 2) from outer parameter packs #61415

Closed
Romain-Geissler-1A opened this issue Mar 14, 2023 · 18 comments · Fixed by #120380
Labels
c++23 clang:frontend Language frontend issues, e.g. anything involving "Sema" false-positive Warning fires when it should not

Comments

@Romain-Geissler-1A
Copy link
Member

Romain-Geissler-1A commented Mar 14, 2023

Hi,

This is an issue similar #58452, actually the snippet I posted in this issue is the same. Using libstc++'s 13 tuple implementation is still not possible when using -std=c++2b:

#include <tuple>

void g(std::tuple<std::tuple<int, int>>);

void f() 
{
    g(std::forward_as_tuple<std::tuple<int, int>>({0, 0}));
}

Godbolt output (https://godbolt.org/z/Meca376M6):

In file included from <source>:1:
/opt/compiler-explorer/gcc-snapshot/lib/gcc/x86_64-linux-gnu/13.0.1/../../../../include/c++/13.0.1/tuple:691:2: error: pack expansion contains parameter pack '_UTypes' that has a different length (1 vs. 2) from outer parameter packs
        using __convertible = __and_<is_convertible<_UTypes, _Types>...>;
        ^~~~~
/opt/compiler-explorer/gcc-snapshot/lib/gcc/x86_64-linux-gnu/13.0.1/../../../../include/c++/13.0.1/tuple:853:27: note: in instantiation of template type alias '__convertible' requested here
          = _TCC<true>::template __convertible<_Args...>::value;
                                 ^
/opt/compiler-explorer/gcc-snapshot/lib/gcc/x86_64-linux-gnu/13.0.1/../../../../include/c++/13.0.1/tuple:948:12: note: in instantiation of static data member 'std::tuple<std::tuple<int, int> &&>::__convertible<int &, int &>' requested here
        explicit(!__convertible<_UElements&...>)
                  ^
/opt/compiler-explorer/gcc-snapshot/lib/gcc/x86_64-linux-gnu/13.0.1/../../../../include/c++/13.0.1/tuple:2000:36: note: while substituting deduced template arguments into function template 'tuple' [with _UElements = <int, int>]
    { return tuple<_Elements&&...>(std::forward<_Elements>(__args)...); }
                                   ^
<source>:7:12: note: in instantiation of function template specialization 'std::forward_as_tuple<std::tuple<int, int>>' requested here
    g(std::forward_as_tuple<std::tuple<int, int>>({0, 0}));
           ^
In file included from <source>:1:
/opt/compiler-explorer/gcc-snapshot/lib/gcc/x86_64-linux-gnu/13.0.1/../../../../include/c++/13.0.1/tuple:691:2: error: pack expansion contains parameter pack '_UTypes' that has a different length (1 vs. 2) from outer parameter packs
        using __convertible = __and_<is_convertible<_UTypes, _Types>...>;
        ^~~~~
/opt/compiler-explorer/gcc-snapshot/lib/gcc/x86_64-linux-gnu/13.0.1/../../../../include/c++/13.0.1/tuple:853:27: note: in instantiation of template type alias '__convertible' requested here
          = _TCC<true>::template __convertible<_Args...>::value;
                                 ^
/opt/compiler-explorer/gcc-snapshot/lib/gcc/x86_64-linux-gnu/13.0.1/../../../../include/c++/13.0.1/tuple:959:12: note: in instantiation of static data member 'std::tuple<std::tuple<int, int> &&>::__convertible<const int, const int>' requested here
        explicit(!__convertible<const _UElements...>)
                  ^
/opt/compiler-explorer/gcc-snapshot/lib/gcc/x86_64-linux-gnu/13.0.1/../../../../include/c++/13.0.1/tuple:2000:36: note: while substituting deduced template arguments into function template 'tuple' [with _UElements = <int, int>]
    { return tuple<_Elements&&...>(std::forward<_Elements>(__args)...); }
                                   ^
<source>:7:12: note: in instantiation of function template specialization 'std::forward_as_tuple<std::tuple<int, int>>' requested here
    g(std::forward_as_tuple<std::tuple<int, int>>({0, 0}));
           ^
2 errors generated.
Compiler returned: 1
@shafik
Copy link
Collaborator

shafik commented Mar 14, 2023

CC @erichkeane

@EugeneZelenko EugeneZelenko added clang:frontend Language frontend issues, e.g. anything involving "Sema" c++23 and removed new issue labels Mar 14, 2023
@llvmbot
Copy link
Member

llvmbot commented Mar 14, 2023

@llvm/issue-subscribers-c-2b

@llvmbot
Copy link
Member

llvmbot commented Mar 14, 2023

@llvm/issue-subscribers-clang-frontend

@stbergmann
Copy link
Collaborator

(might be a duplicate of #17042)

@erichkeane
Copy link
Collaborator

I'll need a minimal example here, I got lost on the initial one trying to figure out what is going on inside of Tuple. Also, would be nice to know if perhaps this is just a libc++ bug instead :)

@Romain-Geissler-1A
Copy link
Member Author

Ok I will try to reduce it today, so we stop relying on libstc++ headers.

@AaronBallman AaronBallman added the needs-reduction Large reproducer that should be reduced into a simpler form label Mar 15, 2023
@stbergmann
Copy link
Collaborator

stbergmann commented Mar 15, 2023

I think a reduced example exhibiting the same underlying issue is still as discussed at Pack expansion bug?,

$ cat test.cc
template<typename, typename> struct Pair;
template<typename...> struct Tuple;
template<typename... Ts> struct A {
    template<typename... Us> using B = Tuple<Pair<Ts, Us>...>;
};
template<typename... Ts> struct C {
    template<typename... Us> using D = typename A<Ts...>::template B<Us...>;
};
using E = C<int, int>::D<int, int>;
$ clang++ -fsyntax-only test.cc
test.cc:4:30: error: pack expansion contains parameter pack 'Us' that has a different length (2 vs. 1) from outer parameter packs
    template<typename... Us> using B = Tuple<Pair<Ts, Us>...>;
                             ^~~~~
test.cc:7:68: note: in instantiation of template type alias 'B' requested here
    template<typename... Us> using D = typename A<Ts...>::template B<Us...>;
                                                                   ^
test.cc:9:11: note: in instantiation of template class 'C<int, int>' requested here
using E = C<int, int>::D<int, int>;
          ^
1 error generated.

@Romain-Geissler-1A
Copy link
Member Author

Note, I didn't try to reduce my case since @stbergmann posted his own reduced C++ snippet. Apparently it's not a regression in clang, as it seems clang never accepted this code in any version (checking on Compiler Explorer). In practice, given the updates made on libstdc++ side, it makes the just released clang 16 error when using libstdc++ tuples with C++23.

@AaronBallman AaronBallman removed the needs-reduction Large reproducer that should be reduced into a simpler form label Mar 20, 2023
@Romain-Geissler-1A
Copy link
Member Author

The problem still happens with clang 17 rc1.

@Romain-Geissler-1A
Copy link
Member Author

Note: with gcc 13 now being released, this bug is more likely to be triggered in real life code. Anyone using libstdc++ >= 13 and C++ >= 23 and trying to use maps using std::tuple will trigger this bug.

@shafik
Copy link
Collaborator

shafik commented Aug 22, 2023

Maybe related: #32252

@vient
Copy link
Member

vient commented Oct 5, 2023

It triggers in real life code with clang 17.0.1 and libstdc++ 13 indeed.

@Romain-Geissler-1A
Copy link
Member Author

Hi,

The original reproducer in the first message of this issue seems to no longer happen with the clang trunk (and is still broken with clang 17, so it's not some hidden clang workaround on libstdc++ side which "fixed" the issue).

However the reduced reproducer from @stbergmann here #61415 (comment) is still broken right now with clang trunk.

Cheers,
Romain

@Yanpas
Copy link

Yanpas commented Mar 6, 2024

Did anyone try clang 18.1?

@Romain-Geissler-1A
Copy link
Member Author

No, and Compiler explorer doesn't have clang 18 yet. However it works with clang trunk since some months, so there is no reason it shouldn't work with clang 18 (the reproducer from the first message).

However it seems the reduced case from @stbergmann in #61415 (comment) is still broken on trunk.

@shafik
Copy link
Collaborator

shafik commented Mar 6, 2024

I think we should close this issue.

CC @stbergmann can you confirm if your reduced case is the same as #32252

pgellert added a commit to redpanda-data/redpanda that referenced this issue May 10, 2024
The combination of libstdc++ >= 13, c++ >= 23 and clang 16 has this bug
that caused compilation issues on ubuntu:
 * llvm/llvm-project#61415

This does not cause any compilation errors on fedora because fedora has
a patch version higher gcc and libstdc++ version.

Seastar recently upgraded to using c++23, so this is why we are seeing
this error now.
pgellert added a commit to redpanda-data/redpanda that referenced this issue May 10, 2024
The combination of libstdc++ >= 13, c++ >= 23 and clang 16 has this bug
that caused compilation issues on ubuntu:
 * llvm/llvm-project#61415

This does not cause any compilation errors on fedora because fedora has
a patch version higher gcc and libstdc++ version.

Seastar recently upgraded to using c++23, so this is why we are seeing
this error now.
@vient
Copy link
Member

vient commented Jun 3, 2024

Original test case works fine with Clang 18.1.
Both #61415 (comment) and #32252 still get errors with Clang 18.1 though.

@cor3ntin
Copy link
Contributor

@mizvekov

zyn0217 added a commit that referenced this issue Dec 19, 2024
…acks (#120380)

CheckParameterPacksForExpansion() previously assumed that template
arguments don't include PackExpansion types when attempting another pack
expansion (i.e. when NumExpansions is present). However, this assumption
doesn't hold for type aliases, whose substitution might involve
unexpanded packs. This can lead to incorrect diagnostics during
substitution because the pack size is not yet determined.

To address this, this patch calculates the minimum pack size (ignoring
unexpanded PackExpansionTypes) and compares it to the previously
expanded size. If the minimum pack size is smaller, then there's still a
chance for future substitution to expand it to a correct size, so we
don't diagnose it too eagerly.

Fixes #61415
Fixes #32252
Fixes #17042
@EugeneZelenko EugeneZelenko added the false-positive Warning fires when it should not label Dec 19, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
c++23 clang:frontend Language frontend issues, e.g. anything involving "Sema" false-positive Warning fires when it should not
Projects
None yet
Development

Successfully merging a pull request may close this issue.

10 participants