Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / libcxx / docs / DesignDocs / ExtendedCXX03Support.rst
blob8c18e563e819975d057c96a49fb299fc4a9279db
1 =======================
2 Extended C++03 Support
3 =======================
5 .. contents::
6    :local:
8 Overview
9 ========
11 libc++ is an implementation of the C++ standard library targeting C++11 or later.
13 In C++03, the library implements the C++11 standard using C++11 language extensions provided
14 by Clang.
16 This document tracks the C++11 extensions libc++ requires, the C++11 extensions it provides,
17 and how to write minimal C++11 inside libc++.
19 Required C++11 Compiler Extensions
20 ==================================
22 Clang provides a large subset of C++11 in C++03 as an extension. The features
23 libc++ expects Clang  to provide are:
25 * Variadic templates.
26 * RValue references and perfect forwarding.
27 * Alias templates
28 * defaulted and deleted Functions.
29 * reference qualified Functions
30 * ``auto``
32 There are also features that Clang *does not* provide as an extension in C++03
33 mode. These include:
35 * ``constexpr`` and ``noexcept``
36 *  Trailing return types.
37 * ``>>`` without a space.
40 Provided C++11 Library Extensions
41 =================================
43 .. warning::
44   The C++11 extensions libc++ provides in C++03 are currently undergoing change. Existing extensions
45   may be removed in the future. New users are strongly discouraged depending on these extension
46   in new code.
48   This section will be updated once the libc++ developer community has further discussed the
49   future of C++03 with libc++.
52 Using Minimal C++11 in libc++
53 =============================
55 This section is for developers submitting patches to libc++. It describes idioms that should be
56 used in libc++ code, even in C++03, and the reasons behind them.
59 Use Alias Templates over Class Templates
60 ----------------------------------------
62 Alias templates should be used instead of class templates in metaprogramming. Unlike class templates,
63 Alias templates do not produce a new instantiation every time they are used. This significantly
64 decreases the amount of memory used by the compiler.
66 For example, libc++ should not use ``add_const`` internally. Instead it should use an alias template
67 like
69 .. code-block:: cpp
71   template <class _Tp>
72   using _AddConst = const _Tp;
74 Use Default Template Parameters for SFINAE
75 ------------------------------------------
77 There are three places in a function declaration that SFINAE may occur: In the template parameter list,
78 in the function parameter list, and in the return type. For example:
80 .. code-block:: cpp
82   template <class _Tp, class _ = enable_if_t</*...*/ >
83   void foo(_Tp); // #1
85   template <class _Tp>
86   void bar(_Tp, enable_if_t</*...*/>* = nullptr); // # 2
88   template <class _Tp>
89   enable_if_t</*...*/> baz(_Tp); // # 3
91 Using default template parameters for SFINAE (#1) should always be preferred.
93 Option #2 has two problems. First, users can observe and accidentally pass values to the SFINAE
94 function argument. Second, the default argument creates a live variable, which causes debug
95 information to be emitted containing the text of the SFINAE.
97 Option #3 can also cause more debug information to be emitted than is needed, because the function
98 return type will appear in the debug information.
100 Use ``unique_ptr`` when allocating memory
101 ------------------------------------------
103 The standard library often needs to allocate memory and then construct a user type in it.
104 If the users constructor throws, the library needs to deallocate that memory. The idiomatic way to
105 achieve this is with ``unique_ptr``.
107 ``__builtin_new_allocator`` is an example of this idiom. Example usage would look like:
109 .. code-block:: cpp
111   template <class T>
112   T* __create() {
113     using _UniquePtr = unique_ptr<void*, __default_new_allocator::__default_new_deleter>;
114     _UniquePtr __p = __default_new_allocator::__allocate_bytes(sizeof(T), alignof(T));
115     T* __res = ::new(__p.get()) T();
116     (void)__p.release();
117     return __res;
118   }