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

Clang: modules and std::initializer_list do not like each other #118218

Closed
jijjijj opened this issue Dec 1, 2024 · 8 comments · Fixed by #118537
Closed

Clang: modules and std::initializer_list do not like each other #118218

jijjijj opened this issue Dec 1, 2024 · 8 comments · Fixed by #118537
Labels
clang:modules C++20 modules and Clang Header Modules

Comments

@jijjijj
Copy link
Contributor

jijjijj commented Dec 1, 2024

Clang version: 19.1.3
C++ version: 23
STL: from msvc

I was trying to compile nlohmann json using modules and I encountered an error (posting the original error for others who might google it):

C:\Users\username\.conan2\p\nlohm0567ffc90cfc1\p\include\nlohmann\detail\input\lexer.hpp:748:66: error: no matching constructor for initialization of 'std::initializer_list<char_int_type>' (aka 'initializer_list<int>')
  748 |                     if (JSON_HEDLEY_UNLIKELY(!next_byte_in_range({0x80, 0xBF})))

I tried to reduce code to the minimum reproducible example and here's what I got (pay attention to the highlighted lines changing which kind of fixes the issue):

// main.cpp

import std; // Workaround 1: if I delete this line everything works fine
import mod;

int main() {
    lexer{}.func({1,1});
    return 0;
}
// mod.cppm

module;

#include "inc.hpp"

export module mod;

export using ::lexer;
export using ::adapter;
// inc.hpp

#pragma once

#include <initializer_list>
#include <string>

struct adapter {
    using char_type = char;
};

template<typename InputAdapterType = adapter>
class lexer
{
    using char_type = typename InputAdapterType::char_type; // Workaround 2: if I replace "InputAdapterType" here with "adapter" everything works fine
    using char_int_type = typename std::char_traits<char_type>::int_type;

public:
    void func(std::initializer_list<char_int_type> ranges) {}
};
// std.cppm

module;

#define _BUILD_STD_MODULE // Workaround 3: if I delete this line everything works fine

#include <intrin.h>

export module std;

#include <initializer_list>
#include <regex> // Workaround 4: if I delete this line everything works fine
The full error I get
C:/Users/username/CLionProjects/untitled23_1/std/modules/std.msvc.cppm:9:10: warning: '#include <filename>' attaches the declarations to the named module 'std', which is not usually intended; consider moving that directive before the module declaration [-Winclude-angled-in-module-purview]
    9 | #include <initializer_list>
      |          ^
C:/Users/username/CLionProjects/untitled23_1/std/modules/std.msvc.cppm:7:15: warning: 'std' is a reserved name for a module [-Wreserved-module-identifier]
    7 | export module std;
      |               ^
C:/Users/username/CLionProjects/untitled23_1/std/modules/std.msvc.cppm:10:10: warning: '#include <filename>' attaches the declarations to the named module 'std', which is not usually intended; consider moving that directive before the module declaration [-Winclude-angled-in-module-purview]
   10 | #include <regex>
      |          ^
3 warnings generated.
[4/6] Building CXX object CMakeFiles/untitled23_1.dir/main.cpp.obj
FAILED: CMakeFiles/untitled23_1.dir/main.cpp.obj 
C:\Tools\llvm\bin\clang++.exe  -IC:/Users/username/CLionProjects/untitled23_1/include -IC:/Users/username/CLionProjects/untitled23_1/mod/../include -IC:/Users/username/CLionProjects/untitled23_1/std/include -O0 -std=c++23 -D_DEBUG -D_DLL -D_MT -Xclang --dependent-lib=msvcrtd -g -Xclang -gcodeview -MD -MT CMakeFiles/untitled23_1.dir/main.cpp.obj -MF CMakeFiles\untitled23_1.dir\main.cpp.obj.d @CMakeFiles\untitled23_1.dir\main.cpp.obj.modmap -o CMakeFiles/untitled23_1.dir/main.cpp.obj -c C:/Users/username/CLionProjects/untitled23_1/main.cpp
C:/Users/username/CLionProjects/untitled23_1/main.cpp:6:18: error: no matching constructor for initialization of 'std::initializer_list<char_int_type>' (aka 'initializer_list<int>')
    6 |     lexer{}.func({1,1});
      |                  ^~~~~
C:\Program Files\Microsoft Visual Studio\2022\Community\VC\Tools\MSVC\14.42.34433\include\initializer_list:33:15: note: candidate constructor not viable: no known conversion from 'int' to 'const int *' for 1st argument
   33 |     constexpr initializer_list(const _Elem* _First_arg, const _Elem* _Last_arg) noexcept
      |               ^                ~~~~~~~~~~~~~~~~~~~~~~~
C:\Program Files\Microsoft Visual Studio\2022\Community\VC\Tools\MSVC\14.42.34433\include\initializer_list:21:7: note: candidate constructor (the implicit copy constructor) not viable: requires 1 argument, but 2 were provided
   21 | class initializer_list {
      |       ^~~~~~~~~~~~~~~~
C:\Program Files\Microsoft Visual Studio\2022\Community\VC\Tools\MSVC\14.42.34433\include\initializer_list:21:7: note: candidate constructor (the implicit move constructor) not viable: requires 1 argument, but 2 were provided
   21 | class initializer_list {
      |       ^~~~~~~~~~~~~~~~
C:\Program Files\Microsoft Visual Studio\2022\Community\VC\Tools\MSVC\14.42.34433\include\initializer_list:31:15: note: candidate constructor not viable: requires 0 arguments, but 2 were provided
   31 |     constexpr initializer_list() noexcept : _First(nullptr), _Last(nullptr) {}
      |               ^
C:\Users\username\CLionProjects\untitled23_1\mod\..\include\inc.hpp:17:52: note: passing argument to parameter 'ranges' here
   17 |     void func(std::initializer_list<char_int_type> ranges) {}
      |                                                    ^
1 error generated.
@github-actions github-actions bot added the clang Clang issues not falling into any other category label Dec 1, 2024
@EugeneZelenko EugeneZelenko added clang:modules C++20 modules and Clang Header Modules and removed clang Clang issues not falling into any other category labels Dec 1, 2024
@llvmbot
Copy link
Member

llvmbot commented Dec 1, 2024

@llvm/issue-subscribers-clang-modules

Author: None (jijjijj)

Clang version: 19.1.3 C++ version: 23 STL: from msvc

I was trying to compile nlohmann json using modules and I encountered an error (posting the original error for others who might google it):

C:\Users\username\.conan2\p\nlohm0567ffc90cfc1\p\include\nlohmann\detail\input\lexer.hpp:748:66: error: no matching constructor for initialization of 'std::initializer_list&lt;char_int_type&gt;' (aka 'initializer_list&lt;int&gt;')
  748 |                     if (JSON_HEDLEY_UNLIKELY(!next_byte_in_range({0x80, 0xBF})))

I tried to reduce code to the minimum reproducible example and here's what I got (pay attention to the highlighted lines changing which kind of fixes the issue):

// main.cpp

import std;
import mod;

int main() {
    lexer{}.func({1,1});
    return 0;
}
// mod.cppm

module;

#include "inc.hpp"

export module mod;

export using ::lexer;
export using ::adapter;

<pre>
// inc.hpp

#pragma once

#include &lt;initializer_list>
#include &lt;string>

struct adapter {
using char_type = char;
};

template&lt;typename InputAdapterType = adapter>
class lexer
{
using char_type = typename InputAdapterType::char_type; // <b><i>if I replace "InputAdapterType" here with "adapter" everything works fine</i></b>
using char_int_type = typename std::char_traits&lt;char_type>::int_type;

public:
void func(std::initializer_list&lt;char_int_type> ranges) {}
};
</pre>

<pre>
// std.cppm

module;

#define _BUILD_STD_MODULE // <b><i>if I delete this line everything works fine</i></b>

#include &lt;intrin.h>

export module std;

#include &lt;initializer_list>
#include &lt;regex> // <b><i>if I delete this line everything works fine</i></b>
</pre>

<details>
<summary>The full error I get</summary>

C:/Users/username/CLionProjects/untitled23_1/std/modules/std.msvc.cppm:9:10: warning: '#include &lt;filename&gt;' attaches the declarations to the named module 'std', which is not usually intended; consider moving that directive before the module declaration [-Winclude-angled-in-module-purview]
    9 | #include &lt;initializer_list&gt;
      |          ^
C:/Users/username/CLionProjects/untitled23_1/std/modules/std.msvc.cppm:7:15: warning: 'std' is a reserved name for a module [-Wreserved-module-identifier]
    7 | export module std;
      |               ^
C:/Users/username/CLionProjects/untitled23_1/std/modules/std.msvc.cppm:10:10: warning: '#include &lt;filename&gt;' attaches the declarations to the named module 'std', which is not usually intended; consider moving that directive before the module declaration [-Winclude-angled-in-module-purview]
   10 | #include &lt;regex&gt;
      |          ^
3 warnings generated.
[4/6] Building CXX object CMakeFiles/untitled23_1.dir/main.cpp.obj
FAILED: CMakeFiles/untitled23_1.dir/main.cpp.obj 
C:\Tools\llvm\bin\clang++.exe  -IC:/Users/username/CLionProjects/untitled23_1/include -IC:/Users/username/CLionProjects/untitled23_1/mod/../include -IC:/Users/username/CLionProjects/untitled23_1/std/include -O0 -std=c++23 -D_DEBUG -D_DLL -D_MT -Xclang --dependent-lib=msvcrtd -g -Xclang -gcodeview -MD -MT CMakeFiles/untitled23_1.dir/main.cpp.obj -MF CMakeFiles\untitled23_1.dir\main.cpp.obj.d @<!-- -->CMakeFiles\untitled23_1.dir\main.cpp.obj.modmap -o CMakeFiles/untitled23_1.dir/main.cpp.obj -c C:/Users/username/CLionProjects/untitled23_1/main.cpp
C:/Users/username/CLionProjects/untitled23_1/main.cpp:6:18: error: no matching constructor for initialization of 'std::initializer_list&lt;char_int_type&gt;' (aka 'initializer_list&lt;int&gt;')
    6 |     lexer{}.func({1,1});
      |                  ^~~~~
C:\Program Files\Microsoft Visual Studio\2022\Community\VC\Tools\MSVC\14.42.34433\include\initializer_list:33:15: note: candidate constructor not viable: no known conversion from 'int' to 'const int *' for 1st argument
   33 |     constexpr initializer_list(const _Elem* _First_arg, const _Elem* _Last_arg) noexcept
      |               ^                ~~~~~~~~~~~~~~~~~~~~~~~
C:\Program Files\Microsoft Visual Studio\2022\Community\VC\Tools\MSVC\14.42.34433\include\initializer_list:21:7: note: candidate constructor (the implicit copy constructor) not viable: requires 1 argument, but 2 were provided
   21 | class initializer_list {
      |       ^~~~~~~~~~~~~~~~
C:\Program Files\Microsoft Visual Studio\2022\Community\VC\Tools\MSVC\14.42.34433\include\initializer_list:21:7: note: candidate constructor (the implicit move constructor) not viable: requires 1 argument, but 2 were provided
   21 | class initializer_list {
      |       ^~~~~~~~~~~~~~~~
C:\Program Files\Microsoft Visual Studio\2022\Community\VC\Tools\MSVC\14.42.34433\include\initializer_list:31:15: note: candidate constructor not viable: requires 0 arguments, but 2 were provided
   31 |     constexpr initializer_list() noexcept : _First(nullptr), _Last(nullptr) {}
      |               ^
C:\Users\username\CLionProjects\untitled23_1\mod\..\include\inc.hpp:17:52: note: passing argument to parameter 'ranges' here
   17 |     void func(std::initializer_list&lt;char_int_type&gt; ranges) {}
      |                                                    ^
1 error generated.

</details>

@shafik
Copy link
Collaborator

shafik commented Dec 2, 2024

CC @ChuanqiXu9 @Bigcheese

@jijjijj
Copy link
Contributor Author

jijjijj commented Dec 2, 2024

Seems to happen because bool Sema::isStdInitializerList cannot recognize std::initializer_list as initializer_list if it's defined in <implicit global module> and getStdNamespace() returns the std namespace defined in <global module>

UPD: probably I don't know what I'm talking about

@jijjijj
Copy link
Contributor Author

jijjijj commented Dec 3, 2024

@ChuanqiXu9 Anyway what's the benefit of splitting <global> and <implicit global>? There's no such thing in the standard, and it looks like it complicates the implementation with corner cases like the one above

UPD: I found #60405 and https://reviews.llvm.org/D144367 it clarifies things a little

@ChuanqiXu9
Copy link
Member

@jijjijj the reason why we split and is, the contents in <global> may be exported but <implicit global> may not. It relates to some internal implementation logics.

ChuanqiXu9 pushed a commit that referenced this issue Dec 12, 2024
…ed out of a module (#118537)

If the std::initializer_list is exported out of module, its
`DeclContext` is not a namespace as `Sema::isStdInitializerList`
expects, but an `Decl::Kind::Export` and only its parent is a namespace.
So this commit makes `Sema::isStdInitializerList` account for that.

I'm really new to clang so I'm not 100% sure that was the issue, it
seems so and it fixes compilation. Also I probably need to add tests but
I'd like someone to approve the idea first.

Fixes #118218
@ChuanqiXu9
Copy link
Member

@jijjijj And out of curiosity, since I don't have environment for MSVC, does it mean we can use clang to compile the std module from MSSTL?

@jijjijj
Copy link
Contributor Author

jijjijj commented Dec 17, 2024

@ChuanqiXu9 it was compilable already, but as we saw with this issue, the problems may arise when you use this module

@ChuanqiXu9
Copy link
Member

@ChuanqiXu9 it was compilable already, but as we saw with this issue, the problems may arise when you use this module

Got it. Fine enough.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
clang:modules C++20 modules and Clang Header Modules
Projects
None yet
5 participants