<html>
    <head>
      <base href="https://bugs.llvm.org/">
    </head>
    <body><table border="1" cellspacing="0" cellpadding="8">
        <tr>
          <th>Bug ID</th>
          <td><a class="bz_bug_link 
          bz_status_NEW "
   title="NEW - LLVM's ManagedStatic.h fails with VS 2019 <atomic> and clang-cl due to init-order-fiasco and constexpr confusion"
   href="https://bugs.llvm.org/show_bug.cgi?id=41367">41367</a>
          </td>
        </tr>

        <tr>
          <th>Summary</th>
          <td>LLVM's ManagedStatic.h fails with VS 2019 <atomic> and clang-cl due to init-order-fiasco and constexpr confusion
          </td>
        </tr>

        <tr>
          <th>Product</th>
          <td>clang
          </td>
        </tr>

        <tr>
          <th>Version</th>
          <td>unspecified
          </td>
        </tr>

        <tr>
          <th>Hardware</th>
          <td>PC
          </td>
        </tr>

        <tr>
          <th>OS</th>
          <td>Windows NT
          </td>
        </tr>

        <tr>
          <th>Status</th>
          <td>NEW
          </td>
        </tr>

        <tr>
          <th>Severity</th>
          <td>enhancement
          </td>
        </tr>

        <tr>
          <th>Priority</th>
          <td>P
          </td>
        </tr>

        <tr>
          <th>Component</th>
          <td>C++
          </td>
        </tr>

        <tr>
          <th>Assignee</th>
          <td>unassignedclangbugs@nondot.org
          </td>
        </tr>

        <tr>
          <th>Reporter</th>
          <td>rnk@google.com
          </td>
        </tr>

        <tr>
          <th>CC</th>
          <td>billy.oneal@gmail.com, blitzrakete@gmail.com, dgregor@apple.com, erik.pilkington@gmail.com, llvm-bugs@lists.llvm.org, richard-llvm@metafoo.co.uk, sfinae@hotmail.com
          </td>
        </tr></table>
      <p>
        <div>
        <pre>I installed VS 2019 and tried to self-host clang with the new headers, but it
didn't work. The symptom was this:

[1 processes, 17/531 @ 3.4/s : 4.984s ] Building AttributesCompatFunc.inc...
FAILED: lib/IR/AttributesCompatFunc.inc
cmd.exe /C "cd /D C:\src\llvm-project\build &&
C:\src\llvm-project\build\bin\llvm-tblgen.exe -gen-attrs -I
C:/src/llvm-project/llvm/lib/IR -I C:/src/llvm-project/llvm/include
C:/src/llvm-project/llvm/lib/IR/AttributesCompatFunc.td -o
lib/IR/AttributesCompatFunc.inc -d lib/IR/AttributesCompatFunc.inc.d"
llvm-tblgen.exe: Unknown command line argument '-gen-attrs'.  Try:
'C:\src\llvm-project\build\bin\llvm-tblgen.exe -help'
llvm-tblgen.exe: Did you mean '-stats'?
ninja: build stopped: subcommand failed.

The issue here is that command line options aren't being registered. This is
because TopLevelSubCommand is a ManagedStatic<SubCommand>. What happens is that
the cl::opt initializers run first, then ManagedStatic is incorrectly *not*
linker initialized, we run the ManagedStatic ctor, and we zero out the pointer
to the previously constructed list of command line flags.

The issue is that ManagedStaticBase wants to be linker initialized, but the way
it achieves that is by not having a constructor at all. It has members that
look like this:

class ManagedStaticBase {
protected:
  // This should only be used as a static variable, which guarantees that this
  // will be zero initialized.
  mutable std::atomic<void *> Ptr;
  mutable void (*DeleterFn)(void*);
  mutable const ManagedStaticBase *Next;

With clang-cl, for some reason the default constructor for this class isn't
considered to be constexpr. When I try to explicitly make it constexpr, clang
complains:
C:\src\llvm-project\llvm\include\llvm/Support/ManagedStatic.h(47,3): error:
defaulted definition of default constructor is not constexpr
  constexpr ManagedStaticBase() = default;
  ^
1 error generated.

So, clang thinks the defaulted one won't be constexpr, but it doesn't say why.
Even if it's correct, it should probably point the user at the uninitialized
members. If I initialize the members explicitly like this:

  mutable std::atomic<void *> Ptr{nullptr};
  mutable void (*DeleterFn)(void*) = nullptr;
  mutable const ManagedStaticBase *Next = nullptr;

Then the defaulted default ctor becomes constexpr and everything works as
expected.

The atomic headers have some interesting ifdefs that seem to allude to this
issue:

struct _Atomic_base : _Atomic_impl<_Bytes> {
  ...
#ifdef __clang__ // TRANSITION, VSO#406237
    constexpr _Atomic_base()
_NOEXCEPT_COND(is_nothrow_default_constructible_v<_Ty>) : _My_val() {}
#else // ^^^ no workaround ^^^ // vvv workaround vvv
    _Atomic_base() = default;
#endif // VSO#406237

The odd thing is that if this constexpr default ctor didn't exist, we might be
in business? Hard to say.

Another way of looking at this bug, if it's a clang bug, is, why is the
following an error?

struct atomic_ptr {
  constexpr atomic_ptr() = default;
  void *Ptr = nullptr;
};
struct Foo {
  atomic_ptr Ptr;
  int y;
  constexpr Foo() = default; // why an error?
};
extern Foo obj;
Foo obj;

Just because 'y' has no initializer, Foo can't be constexpr, even if I ask for
it to be?

In the end, I *think* this is a clang bug, but it might be an STL bug, so I
added Billy and STL.

Anyway, there's an obvious workaround, which is to add a real constexpr
constructor to ManagedStatic, assuming that works with all the toolchains we
support. I'm going to try that.</pre>
        </div>
      </p>


      <hr>
      <span>You are receiving this mail because:</span>

      <ul>
          <li>You are on the CC list for the bug.</li>
      </ul>
    </body>
</html>