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

<functional>: is compatible with /Zc:alignedNew- now #2712

Merged
merged 9 commits into from
May 24, 2022

Conversation

fsb4000
Copy link
Contributor

@fsb4000 fsb4000 commented May 11, 2022

fixes #2711

I also found that:

#include <iostream>

using namespace std;

int main() {
#ifdef __cpp_aligned_new
    cout << "has aligned new\n";
#else
    cout << "nothing\n";
#endif
}

works differently on cl.exe and clang-cl.exe

$ clang-cl main.cpp /std:c++14 /Zc:alignedNew /EHsc
$ main
has aligned new
$ cl main.cpp /std:c++14 /Zc:alignedNew /EHsc
$ main
nothing

this is why I replaced all

#ifdef __cpp_aligned_new

to

#if defined(__cpp_aligned_new) && _HAS_CXX17

Should I report about it to LLVM and add "Transition" comment?

@fsb4000 fsb4000 requested a review from a team as a code owner May 11, 2022 13:52
Copy link
Contributor

@AlexGuteniev AlexGuteniev left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm concerned about changing existing code to rely on C++17 mode

@fsb4000
Copy link
Contributor Author

fsb4000 commented May 11, 2022

cl defines __cpp_aligned_new for C++17 and up (with /Zc:alignedNew)
clang-cl defines __cpp_aligned_new for C++14 and up (with /Zc:alignedNew)
I added _HAS_CXX17 for compatibility between clang-cl and cl
Am I wrong?

@AlexGuteniev

This comment was marked as resolved.

@fsb4000
Copy link
Contributor Author

fsb4000 commented May 11, 2022

Does clang-cl compile with /std:c++14 and /Zc:alignedNew before this change?

No, it doesn't compile as far as I can see.
I got the error(and other similar):

Command: "C:\Program Files\Microsoft Visual Studio\2022\Preview\VC\Tools\Llvm\x64\bin\clang-cl.EXE" "C:\Dev\STL\tests\std\tests\GH_000690_overaligned_function\test.cpp" "-IC:\Dev\STL\out\build\x64\out\inc" "-IC:\Dev\STL\llvm-project\libcxx\test\support" "-IC:\Dev\STL\tests\std\include" "/nologo" "/Od" "/W4" "/w14061" "/w14242" "/w14265" "/w14582" "/w14583" "/w14587" "/w14588" "/w14749" "/w14841" "/w14842" "/w15038" "/w15214" "/w15215" "/w15216" "/w15217" "/sdl" "/WX" "/Zc:strictStrings" "/D_ENABLE_STL_INTERNAL_CHECK" "/bigobj" "/FIforce_include.hpp" "/w14365" "/D_ENFORCE_FACET_SPECIALIZATIONS=1" "/D_STL_CALL_ABORT_INSTEAD_OF_INVALID_PARAMETER" "-fno-ms-compatibility" "-fno-delayed-template-parsing" "/EHsc" "/MD" "/std:c++14" "/w14640" "/Zc:threadSafeInit-" "/Zc:alignedNew" "-m64" "-FeC:\Dev\STL\out\build\x64\tests\std\tests\GH_000690_overaligned_function\Output\55\GH_000690_overaligned_function.exe" "-link" "-LIBPATH:C:\Dev\STL\out\build\x64\out\lib\amd64" "-LIBPATH:C:\Program Files\Microsoft Visual Studio\2022\Preview\VC\Tools\MSVC\14.33.31424\lib\x64" "/MANIFEST:EMBED"
Exit Code: 1
Standard Error:
--
In file included from C:\Dev\STL\tests\std\tests\GH_000690_overaligned_function\test.cpp:9:
In file included from C:\Dev\STL\out\build\x64\out\inc\functional:14:
C:\Dev\STL\out\build\x64\out\inc\xmemory(1425,60): error: cannot initialize a value of type 'std::align_val_t' with an rvalue of type 'unsigned long long'
                _Pbuf = ::operator new (_Size, align_val_t{alignof(_Ty)}, nothrow);
                                                           ^~~~~~~~~~~~
C:\Dev\STL\out\build\x64\out\inc\xmemory(1445,47): error: cannot initialize a value of type 'std::align_val_t' with an rvalue of type 'unsigned long long'
        ::operator delete (_Pbuf, align_val_t{alignof(_Ty)});
                                              ^~~~~~~~~~~~
2 errors generated.

only with clang-cl and c++14 configuration.

So every #ifdef __cpp_aligned_new which contained align_val_t didn't compile.

Copy link
Contributor

@AlexGuteniev AlexGuteniev left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok then I agree

@CaseyCarter CaseyCarter added the bug Something isn't working label May 11, 2022
@CaseyCarter
Copy link
Contributor

works differently on cl.exe and clang-cl.exe

This is a separate issue, which should be handled in a separate PR (if at all). Feel free to report upstream to the clang-cl devs that cl ignores /Zc:alignedNew in pre-C++17 modes and that they could do so as well if they like.

@fsb4000
Copy link
Contributor Author

fsb4000 commented May 11, 2022

I created: LLVM-55389

Copy link
Contributor

@strega-nil-ms strega-nil-ms left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't have an opinion on the testing changes (though they seem reasonable to me), and this makes the existing situation better; there may be some discussion left to be had wrt how we want to test /Zc:alignedNew-.

@strega-nil-ms strega-nil-ms removed their assignment May 12, 2022
@StephanTLavavej
Copy link
Member

FYI @strega-nil-ms I pushed minor changes after you approved.

I was initially concerned about mismatch - adding __cpp_aligned_new logic to otherwise unguarded code would create an incompatibility between C++14 and C++17 mode (as C++17 implies this by default). However, the behavioral change here is specific to C++23 mode. Therefore, only mixing /std:c++latest and /std:c++latest /Zc:alignedNew- would be a problem, and this is already true for our other __cpp_aligned_new guards.

@StephanTLavavej

This comment was marked as outdated.

StephanTLavavej added a commit to StephanTLavavej/STL that referenced this pull request May 16, 2022
@StephanTLavavej StephanTLavavej removed their assignment May 16, 2022
@StephanTLavavej StephanTLavavej added the decision needed We need to choose something before working on this label May 16, 2022
@Nick-Kooij
Copy link

Hello,
Thanks all for working on this issue. As the developer who raise the issue in the first place, I'd just like to clarify that my purpose in using /Zc:alignedNew- is really a workaround to eliminate the overhead of calling overaligned new/delete.

The ideal solution for me would be to add a compiler switch /Zc:defaultNewAlignment={std::align_val_t} where the __STDCPP_DEFAULT_NEW_ALIGNMENT__ compile-time alignment threshold where C++17 overaligned new/delete are called:

  • void* operator new ( std::size_t count, std::align_val_t al );
  • void operator delete ( void* ptr, std::align_val_t al ) noexcept;
  • And similar array/noexcept overloads--anything taking an additional std::align_val_t parameter.

In other words, implement a switch symmetric to Clang's -fnew-alignment. See Add compiler switch to change STDCPP_DEFAULT_NEW_ALIGNMENT.


Some background:

  1. I have overloaded global operator new/delete providing stronger alignment guarantees than the default __STDCPP_DEFAULT_NEW_ALIGNMENT__. Specifically alignment beyond a single 64 byte cache line is aligned to 128 bytes (two cache lines). (This meshes well with x64 prefetching logic and avoids false sharing.)

  2. By default with C++17 compilation (without /Zc:alignedNew) new/delete expressions for types with alignment beyond __STDCPP_DEFAULT_NEW_ALIGNMENT__ will call the C++17 overaligned operator new/delete overloads:

  • void* operator new ( std::size_t count, std::align_val_t al );
  • void operator delete ( void* ptr, std::align_val_t al ) noexcept;
  • ...

This is a fantastic new C++17 feature allowing over-aligned types to be dynamically allocated. Ideally I would like this feature enabled, rather than disabling it using /Zc:alignedNew-.

Consider allocation of a 256 byte type, with 128 byte alignment, by default without /Zc:alignedNew-. As __STDCPP_DEFAULT_NEW_ALIGNMENT__ can't be we have we now have runtime overhead which I wish to avoid:

  • bloated operator new and operator delete call sites
  • runtime overhead of calling an overaligned operator new

The goal is to eliminate this runtime overhead.

While the runtime overhead within the overloaded overaligned new/delete can be somewhat mitigated using [[likely]]/[[unlikely]] on the alignment check, nevertheless there is a runtime overhead due to unnecessary branches.

<public: static void __cdecl TaskSpawn<struct ChainTask>::spawn<int>(int &&)>:
               	push	rbx
               	sub	rsp, 32
               	mov	rbx, rcx
               	mov	edx, 128 // additional std::align_val_t parameter
               	mov	ecx, 256 // object size
               	call	 ??2@YAPEAX_KW4align_val_t@std@@@Z // overaligned operator new call
                ...

This bloats each over-aligned operator new/delete call site, and unnecessarily calls an overload that contains additional alignment logic.

Ideally we want this:

<public: static void __cdecl TaskSpawn<struct ChainTask>::spawn<int>(int &&)>:
               	push	rbx
               	sub	rsp, 32
               	mov	rbx, rcx
               	mov	ecx, 256 // object size
               	call	 ??2@YAPEAX_K@Z // regular operator new called
                ...

Best Regards,
Nick

@StephanTLavavej StephanTLavavej removed the decision needed We need to choose something before working on this label May 18, 2022
@StephanTLavavej
Copy link
Member

We talked about this at the weekly maintainer meeting - we believe that this PR should be changed to follow the same strategy as the rest of the STL currently uses for /Zc:alignedNew-, which is to directly call operator new and not attempt to add additional alignment. As @frederick-vs-ja mentioned, this will potentially not handle overaligned types correctly (in the case where operator new hasn't been replaced to provide such alignment), but this is consistent with existing practice, and attempting to add additional alignment would interfere with scenarios like @Nick-Kooij's. We conclude that if users want the STL to automatically handle overaligned types correctly, they should be using the C++17 Standard setting - by setting /Zc:alignedNew- they are responsible for their own destiny.

@CaseyCarter CaseyCarter self-requested a review May 19, 2022 17:24
@CaseyCarter CaseyCarter self-assigned this May 19, 2022
@StephanTLavavej StephanTLavavej self-assigned this May 21, 2022
@StephanTLavavej
Copy link
Member

I'm speculatively mirroring this to the MSVC-internal repo - further changes can be pushed, but please notify me.

Copy link
Contributor

@CaseyCarter CaseyCarter left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Applied my suggested changes after testing locally.

@CaseyCarter CaseyCarter removed their assignment May 22, 2022
@StephanTLavavej StephanTLavavej merged commit 39f8c34 into microsoft:main May 24, 2022
@StephanTLavavej
Copy link
Member

Thanks for aligning these new features with our existing tolerance of this option! 😹 🚀 🎉

@fsb4000 fsb4000 deleted the fix2711 branch May 24, 2022 05:15
fsb4000 added a commit to fsb4000/STL that referenced this pull request Aug 13, 2022
)

Co-authored-by: Stephan T. Lavavej <[email protected]>
Co-authored-by: Casey Carter <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
None yet
Development

Successfully merging this pull request may close these issues.

<functional>: incompatible with /std:c++latest and /Zc:alignedNew-
7 participants