-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathC++11 and Beyond.txt
executable file
·64 lines (48 loc) · 3.73 KB
/
C++11 and Beyond.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
------------------ C++11 and beyond ------------------
C++11 is supported by CMake. Really. Just not in CMake 2.8, because, guess what,
C++11 didn't exist in 2009 when 2.0 was released. As long as you are using CMake 3.1 or newer,
you should be fine, there are two different ways to enable support. And as you'll soon see, there's
even better support in CMake 3.8+. I'll start with that, since this is Modern CMake.
------------------ CMake 3.8+: Meta compiler features ------------------
As long as you can require that a user install CMake, you'll have access to the newest way to
enable C++ standards. This is the most powerful way, with the nicest syntax and the best support
for new standards, and the best target behavior for mixing standards and options. Assuming you
have a target named myTarget, it looks like this:
target_compile_features(myTarget PUBLIC cxx_std_11)
set_target_properties(myTarget PROPERTIES CXX_EXTENSIONS OFF)
For the first line, we get to pick between cxx_std_11, cxx_std_14, and cxx_std_17. The second line
is optional, but will avoid extensions being added; without it you'd get things like -std=g++11
replacing -std=c++11. The first line even works on INTERFACE targets; only actual compiled targets
can use the second line.
If a target further down the dependency chain specifies a higher C++ level, this interacts nicely.
It's really just a more advanced version of the following method, so it interacts nicely with that, too.
------------------ CMake 3.1+: Compiler features ------------------
You can ask for specific compiler features to be available. This was more granular than asking for a C++
version, though it's a bit tricky to pick out just the features a package is using unless you wrote the
package and have a good memory. Since this ultimately checks against a list of options CMake knows your
compiler supports and picks the highest flag indicated in that list, you don't have to specify all the
options you need, just the rarest ones. The syntax is identical to the section above, you just have a list
of options to pick instead of cxx_std_* options. See the whole list here.
If you have optional features, you can use the list CMAKE_CXX_COMPILE_FEATURES and use if(... IN_LIST ...)
from CMake 3.3+ to see if that feature is supported, and add it conditionally. See the docs here for other use cases.
A related feature, WriteCompilerDetectionHeader, is worth checking out. It is a module that lets you make a file
with macros allowing you to check and support optional features for specific compilers. Like any header generator,
this will require that you build with CMake so that your header can be generated as part of the build process
(only important if you care about supporting multiple build systems, or if you are making a no-build header-only library).
------------------ CMake 3.1+: Global and property settings ------------------
There is another way that C++ standards are supported; a specific set of three properties (both global and target level).
The global properties are:
set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
set(CMAKE_CXX_EXTENSIONS OFF)
The first line sets a C++ standard level, and the second tells CMake to use it, and the final line is optional
and ensures -std=c++11 vs. something like -std=g++11. This method isn't bad for a final package, but shouldn't be
used by a library. You can also set these values on a target:
set_target_properties(myTarget PROPERTIES
CXX_STANDARD 11
CXX_STANDARD_REQUIRED YES
CXX_EXTENSIONS NO
)
Which is better, but still doesn't have the sort of explicit control that compiler features have for populating
PRIVATE and INTERFACE properties.
You can find more information about the final two methods on Craig Scott's useful blog post.