On Thu, 22 Jul 2021 at 11:03, Jonathan Wakely wrote: > > On Thu, 22 Jul 2021 at 10:23, Stephan Bergmann via Libstdc++ > <libstd...@gcc.gnu.org> wrote: > > > > Compared to GCC 11 (at least gcc-c++-11.1.1-3.fc34.x86_64), recent GCC > > 12 trunk emits two "unhelpful" -Wmaybe-uninitialized for > > > > > $ cat test.cc > > > #include <functional> > > > using fn = std::function<void()>; > > > fn f(fn x) { > > > fn a; > > > a = x; > > > return x; > > > } > > > > > $ ~/gcc/trunk/inst/bin/g++ -c -Wmaybe-uninitialized -O2 test.cc > > > In file included from > > > ~/gcc/trunk/inst/include/c++/12.0.0/bits/stl_function.h:60, > > > from ~/gcc/trunk/inst/include/c++/12.0.0/functional:49, > > > from test.cc:1: > > > In function ‘std::_Require<std::__not_<std::__is_tuple_like<_Tp> >, > > > std::is_move_constructible<_Tp>, std::is_move_assignable<_Tp> > > > > std::swap(_Tp&, _Tp&) [with _Tp = void (*)(const std::_Any_data&)]’, > > > inlined from ‘void std::function<_Res(_ArgTypes > > > ...)>::swap(std::function<_Res(_ArgTypes ...)>&) [with _Res = void; > > > _ArgTypes = {}]’ at > > > ~/gcc/trunk/inst/include/c++/12.0.0/bits/std_function.h:529:11, > > > inlined from ‘std::function<_Res(_ArgTypes ...)>& > > > std::function<_Res(_ArgTypes ...)>::operator=(const > > > std::function<_Res(_ArgTypes ...)>&) [with _Res = void; _ArgTypes = {}]’ > > > at ~/gcc/trunk/inst/include/c++/12.0.0/bits/std_function.h:442:20, > > > inlined from ‘fn f(fn)’ at test.cc:5:9: > > > ~/gcc/trunk/inst/include/c++/12.0.0/bits/move.h:204:11: warning: > > > ‘<unnamed>.std::function<void()>::_M_invoker’ may be used uninitialized > > > [-Wmaybe-uninitialized] > > > 204 | _Tp __tmp = _GLIBCXX_MOVE(__a); > > > | ^~~~~ > > > In file included from ~/gcc/trunk/inst/include/c++/12.0.0/functional:59, > > > from test.cc:1: > > > ~/gcc/trunk/inst/include/c++/12.0.0/bits/std_function.h: In function ‘fn > > > f(fn)’: > > > ~/gcc/trunk/inst/include/c++/12.0.0/bits/std_function.h:442:9: note: > > > ‘<anonymous>’ declared here > > > 442 | function(__x).swap(*this); > > > | ^~~~~~~~~~~~~ > > > In file included from > > > ~/gcc/trunk/inst/include/c++/12.0.0/bits/stl_function.h:60, > > > from ~/gcc/trunk/inst/include/c++/12.0.0/functional:49, > > > from test.cc:1: > > > In function ‘std::_Require<std::__not_<std::__is_tuple_like<_Tp> >, > > > std::is_move_constructible<_Tp>, std::is_move_assignable<_Tp> > > > > std::swap(_Tp&, _Tp&) [with _Tp = std::_Any_data]’, > > > inlined from ‘void std::function<_Res(_ArgTypes > > > ...)>::swap(std::function<_Res(_ArgTypes ...)>&) [with _Res = void; > > > _ArgTypes = {}]’ at > > > ~/gcc/trunk/inst/include/c++/12.0.0/bits/std_function.h:527:11, > > > inlined from ‘std::function<_Res(_ArgTypes ...)>& > > > std::function<_Res(_ArgTypes ...)>::operator=(const > > > std::function<_Res(_ArgTypes ...)>&) [with _Res = void; _ArgTypes = {}]’ > > > at ~/gcc/trunk/inst/include/c++/12.0.0/bits/std_function.h:442:20, > > > inlined from ‘fn f(fn)’ at test.cc:5:9: > > > ~/gcc/trunk/inst/include/c++/12.0.0/bits/move.h:204:11: warning: > > > ‘*(std::_Any_data*)((char*)&<unnamed> + offsetof(std::function, > > > std::function<void()>::<unnamed>))’ may be used uninitialized > > > [-Wmaybe-uninitialized] > > > 204 | _Tp __tmp = _GLIBCXX_MOVE(__a); > > > | ^~~~~ > > > In file included from ~/gcc/trunk/inst/include/c++/12.0.0/functional:59, > > > from test.cc:1: > > > ~/gcc/trunk/inst/include/c++/12.0.0/bits/std_function.h: In function ‘fn > > > f(fn)’: > > > ~/gcc/trunk/inst/include/c++/12.0.0/bits/std_function.h:442:9: note: > > > ‘<anonymous>’ declared here > > > 442 | function(__x).swap(*this); > > > | ^~~~~~~~~~~~~ > > > > This appears to be an issue with more aggressive -Wmaybe-uninitialized > > in GCC 12 vs. 11, rather than an issue with changes to <functional> in > > libstdc++ 12 vs. 11, as effectively the same warnings are emitted when I > > use GCC 12 with libstdc++ 11 with > > > > > $ ~/gcc/trunk/inst/bin/g++ -c -Wmaybe-uninitialized -O2 -nostdinc++ > > > -isystem /usr/include/c++/11 -isystem > > > /usr/include/c++/11/x86_64-redhat-linux test.cc > > > > The warnings may technically be correct, and I'm not sure whether this > > is something that should be addressed in the GCC code emitting the > > warnings or in the libstdc++ <functional> implementation. > > > > (I found this when building LibreOffice with recent GCC 12 trunk.) > > The problem is that the _Function_base default constructor is > user-provided, so when std::function value-initializes its base class, > that doesn't do zero-init first. It just calls the default ctor, which > doesn't initialize the _M_fucntor member.
I've pushed the attached patch to trunk, after testing on powerpc64le-linux.
commit c22bcfd2f7dc9bb5ad394720f4a612327dc898ba Author: Jonathan Wakely <jwak...@redhat.com> Date: Thu Jul 22 11:57:38 2021 libstdc++: Initialize all subobjects of std::function The std::function::swap member swaps each data member unconditionally, resulting in -Wmaybe-uninitialized warnings for a default constructed object. This happens because the _M_invoker and _M_functor members are only initialized if the function has a target. This change ensures that all subobjects are zero-initialized on construction. Signed-off-by: Jonathan Wakely <jwak...@redhat.com> libstdc++-v3/ChangeLog: * include/bits/std_function.h (_Function_base): Add default member initializers and define constructor as defaulted. (function::_M_invoker): Add default member initializer. diff --git a/libstdc++-v3/include/bits/std_function.h b/libstdc++-v3/include/bits/std_function.h index 31eba2b822c..c08484465c9 100644 --- a/libstdc++-v3/include/bits/std_function.h +++ b/libstdc++-v3/include/bits/std_function.h @@ -237,7 +237,7 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION { __functor._M_access<_Functor*>() = new _Functor(std::move(__f)); } }; - _Function_base() : _M_manager(nullptr) { } + _Function_base() = default; ~_Function_base() { @@ -247,11 +247,11 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION bool _M_empty() const { return !_M_manager; } - typedef bool (*_Manager_type)(_Any_data&, const _Any_data&, - _Manager_operation); + using _Manager_type + = bool (*)(_Any_data&, const _Any_data&, _Manager_operation); - _Any_data _M_functor; - _Manager_type _M_manager; + _Any_data _M_functor{}; + _Manager_type _M_manager{}; }; template<typename _Signature, typename _Functor> @@ -261,7 +261,7 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION class _Function_handler<_Res(_ArgTypes...), _Functor> : public _Function_base::_Base_manager<_Functor> { - typedef _Function_base::_Base_manager<_Functor> _Base; + using _Base = _Function_base::_Base_manager<_Functor>; public: static bool @@ -414,7 +414,7 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION function(_Functor __f) : _Function_base() { - typedef _Function_handler<_Res(_ArgTypes...), _Functor> _My_handler; + using _My_handler = _Function_handler<_Res(_ArgTypes...), _Functor>; if (_My_handler::_M_not_empty_function(__f)) { @@ -634,8 +634,8 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION private: using _Invoker_type = _Res (*)(const _Any_data&, _ArgTypes&&...); - _Invoker_type _M_invoker; - }; + _Invoker_type _M_invoker = nullptr; + }; #if __cpp_deduction_guides >= 201606 template<typename>