http://gcc.gnu.org/bugzilla/show_bug.cgi?id=59761

            Bug ID: 59761
           Summary: ICE: g++ segfaults in test case involving constexpr
                    default constructor with uninitialized member and
                    template type alias
           Product: gcc
           Version: 4.8.2
            Status: UNCONFIRMED
          Severity: normal
          Priority: P3
         Component: c++
          Assignee: unassigned at gcc dot gnu.org
          Reporter: mizvekov at gmail dot com

Created attachment 31803
  --> http://gcc.gnu.org/bugzilla/attachment.cgi?id=31803&action=edit
ICE/segfault when compiling this

Compiling the attached test_bad.cc with g++ results in an ICE, with the
following message:



test_bad.cc: In substitution of ‘template<bool B> using enable_if_t = typename
std::enable_if<B>::type [with bool B = is_foo0((*(const foo<0>*)(&
foo<0>())))]’:
test_bad.cc:12:22:   required from here
test_bad.cc:12:22: internal compiler error: Segmentation fault
 template<typename T, class = enable_if_t< is_foo0(T{}) >> int test() { return
0; }


               ^
This error only happens when both using the enable_if_t template type alias,
and struct foo having a data member uninitialized by the constexpr default
constructor.

If enable_if_t<...> is substituted by std::enable_if<...>::type, as in the
attached file test_good1.cc, then g++ rejects the code without any ICE. The
following is output:



test_good1.cc:12:36: error: no matching function for call to ‘test()’
 static const int fi = test<foo<0>>();
                                    ^
test_good1.cc:12:36: note: candidate is:
test_good1.cc:10:80: note: template<class T, class> int test()
 template<typename T, class = typename std::enable_if< is_foo0(T{})>::type> int
test() { return 0; }
                                                                               
^
test_good1.cc:10:80: note:   template argument deduction/substitution failed:
test_good1.cc:10:22: error: ‘constexpr foo<X>::foo() [with int X = 0]’ called
in a constant expression
 template<typename T, class = typename std::enable_if< is_foo0(T{})>::type> int
test() { return 0; }
                      ^
test_good1.cc:4:12: note: ‘constexpr foo<X>::foo() [with int X = 0]’ is not
usable as a constexpr function because:
  constexpr foo() {}
            ^
test_good1.cc:4:12: error: uninitialized member ‘foo<0>::bar’ in ‘constexpr’
constructor
test_good1.cc:10:22: note: in template argument for type ‘bool’ 
 template<typename T, class = typename std::enable_if< is_foo0(T{})>::type> int
test() { return 0; }
                      ^


If the code is modified so the default constructor leaves no members
uninitialized, as in the attached file test_good2.cc, then g++ accepts the
code.

Note that clang 3.4 quietly accepts all three versions, even with -Wall.

This is the version of g++ where the problem was found:

Using built-in specs.
COLLECT_GCC=g++
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-unknown-linux-gnu/4.8.2/lto-wrapper
Target: x86_64-unknown-linux-gnu
Configured with: /build/gcc/src/gcc-4.8-20131219/configure --prefix=/usr
--libdir=/usr/lib --libexecdir=/usr/lib --mandir=/usr/share/man
--infodir=/usr/share/info --with-bugurl=https://bugs.archlinux.org/
--enable-languages=c,c++,ada,fortran,go,lto,objc,obj-c++ --enable-shared
--enable-threads=posix --with-system-zlib --enable-__cxa_atexit
--disable-libunwind-exceptions --enable-clocale=gnu --disable-libstdcxx-pch
--disable-libssp --enable-gnu-unique-object --enable-linker-build-id
--enable-cloog-backend=isl --disable-cloog-version-check --enable-lto
--enable-plugin --enable-install-libiberty --with-linker-hash-style=gnu
--disable-multilib --disable-werror --enable-checking=release
Thread model: posix
gcc version 4.8.2 20131219 (prerelease) (GCC)

Reply via email to