Title
Trivial unions changing existing behavior
Status
open
Section
11.4.5.2 [class.default.ctor]
Submitter
Richard Smith

Created on 2025-02-14.00:00:00 last changed 1 month ago

Messages

Date: 2025-03-04.23:01:39

Consider:

  union U { int a, b; };
  template<U u> class X {};
  constexpr U make() { U u; return u; }
  void f(X<make()>) {}

Before P3074R7, the template argument of X was a union object with no active member. Now, the default construction of U starts the lifetime of U::a because it has implicit-lifetime type (11.4.5.2 [class.default.ctor] paragraph 4).

This changes the mangling of f, because a union with no active member is different from a union with the first member active. More importantly, it is now ill-formed, because U::a is in-lifetime, but uninitialized (7.7 [expr.const] bullet 22.2).

Also consider the following implementation divergence

  struct E { };
  union U { E e; };
  template<U u> class X {};

  constexpr U make() { U u; return u; }
  constexpr U make2() { U u{}; return u; }

  void f(X<make()>) {}
  void f(X<make2()>) {}    // OK with clang; redefinition error with gcc

If any of those behavior changes are intended, an Annex C entry is needed.

History
Date User Action Args
2025-02-14 00:00:00admincreate