Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 1 | # -*- Mode: python; py-indent-offset: 4; indent-tabs-mode: nil; coding: utf-8; -*- |
Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 2 | |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 3 | from waflib import Configure, Logs, Utils |
Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 4 | |
| 5 | def options(opt): |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 6 | opt.add_option('--debug', '--with-debug', action='store_true', default=False, |
| 7 | help='Compile in debugging mode with minimal optimizations (-O0 or -Og)') |
Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 8 | |
| 9 | def configure(conf): |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 10 | conf.start_msg('Checking C++ compiler version') |
| 11 | |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 12 | cxx = conf.env.CXX_NAME # generic name of the compiler |
| 13 | ccver = tuple(int(i) for i in conf.env.CC_VERSION) |
| 14 | ccverstr = '.'.join(conf.env.CC_VERSION) |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 15 | errmsg = '' |
| 16 | warnmsg = '' |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 17 | if cxx == 'gcc': |
Davide Pesavento | a08dc3f | 2018-05-24 00:40:28 -0400 | [diff] [blame] | 18 | if ccver < (5, 3, 0): |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 19 | errmsg = ('The version of gcc you are using is too old.\n' |
Davide Pesavento | a08dc3f | 2018-05-24 00:40:28 -0400 | [diff] [blame] | 20 | 'The minimum supported gcc version is 5.3.0.') |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 21 | conf.flags = GccFlags() |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 22 | elif cxx == 'clang': |
Davide Pesavento | e556991 | 2019-01-29 19:39:06 -0500 | [diff] [blame^] | 23 | if ccver < (3, 6, 0): |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 24 | errmsg = ('The version of clang you are using is too old.\n' |
Davide Pesavento | e556991 | 2019-01-29 19:39:06 -0500 | [diff] [blame^] | 25 | 'The minimum supported clang version is 3.6.0.') |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 26 | conf.flags = ClangFlags() |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 27 | else: |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 28 | warnmsg = 'Note: %s compiler is unsupported' % cxx |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 29 | conf.flags = CompilerFlags() |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 30 | |
| 31 | if errmsg: |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 32 | conf.end_msg(ccverstr, color='RED') |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 33 | conf.fatal(errmsg) |
| 34 | elif warnmsg: |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 35 | conf.end_msg(ccverstr, color='YELLOW') |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 36 | Logs.warn(warnmsg) |
| 37 | else: |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 38 | conf.end_msg(ccverstr) |
Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 39 | |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 40 | conf.areCustomCxxflagsPresent = (len(conf.env.CXXFLAGS) > 0) |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 41 | |
Davide Pesavento | a08dc3f | 2018-05-24 00:40:28 -0400 | [diff] [blame] | 42 | # General flags are always applied (e.g., selecting C++ language standard) |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 43 | generalFlags = conf.flags.getGeneralFlags(conf) |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 44 | conf.add_supported_cxxflags(generalFlags['CXXFLAGS']) |
| 45 | conf.add_supported_linkflags(generalFlags['LINKFLAGS']) |
| 46 | conf.env.DEFINES += generalFlags['DEFINES'] |
| 47 | |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 48 | @Configure.conf |
| 49 | def check_compiler_flags(conf): |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 50 | # Debug or optimized CXXFLAGS and LINKFLAGS are applied only if the |
| 51 | # corresponding environment variables are not set. |
| 52 | # DEFINES are always applied. |
Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 53 | if conf.options.debug: |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 54 | extraFlags = conf.flags.getDebugFlags(conf) |
| 55 | if conf.areCustomCxxflagsPresent: |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 56 | missingFlags = [x for x in extraFlags['CXXFLAGS'] if x not in conf.env.CXXFLAGS] |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 57 | if missingFlags: |
| 58 | Logs.warn('Selected debug mode, but CXXFLAGS is set to a custom value "%s"' |
| 59 | % ' '.join(conf.env.CXXFLAGS)) |
| 60 | Logs.warn('Default flags "%s" will not be used' % ' '.join(missingFlags)) |
Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 61 | else: |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 62 | extraFlags = conf.flags.getOptimizedFlags(conf) |
Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 63 | |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 64 | if not conf.areCustomCxxflagsPresent: |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 65 | conf.add_supported_cxxflags(extraFlags['CXXFLAGS']) |
| 66 | conf.add_supported_linkflags(extraFlags['LINKFLAGS']) |
| 67 | |
| 68 | conf.env.DEFINES += extraFlags['DEFINES'] |
Vince Lehman | 0a7da61 | 2014-10-29 14:39:29 -0500 | [diff] [blame] | 69 | |
Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 70 | @Configure.conf |
| 71 | def add_supported_cxxflags(self, cxxflags): |
| 72 | """ |
| 73 | Check which cxxflags are supported by compiler and add them to env.CXXFLAGS variable |
| 74 | """ |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 75 | if len(cxxflags) == 0: |
| 76 | return |
| 77 | |
| 78 | self.start_msg('Checking supported CXXFLAGS') |
Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 79 | |
| 80 | supportedFlags = [] |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 81 | for flags in cxxflags: |
| 82 | flags = Utils.to_list(flags) |
| 83 | if self.check_cxx(cxxflags=['-Werror'] + flags, mandatory=False): |
| 84 | supportedFlags += flags |
Yingdi Yu | 40cd1c3 | 2014-04-17 15:02:17 -0700 | [diff] [blame] | 85 | |
| 86 | self.end_msg(' '.join(supportedFlags)) |
Davide Pesavento | c58cc7f | 2017-08-08 16:51:28 -0500 | [diff] [blame] | 87 | self.env.prepend_value('CXXFLAGS', supportedFlags) |
Vince Lehman | 0a7da61 | 2014-10-29 14:39:29 -0500 | [diff] [blame] | 88 | |
| 89 | @Configure.conf |
| 90 | def add_supported_linkflags(self, linkflags): |
| 91 | """ |
| 92 | Check which linkflags are supported by compiler and add them to env.LINKFLAGS variable |
| 93 | """ |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 94 | if len(linkflags) == 0: |
| 95 | return |
| 96 | |
| 97 | self.start_msg('Checking supported LINKFLAGS') |
Vince Lehman | 0a7da61 | 2014-10-29 14:39:29 -0500 | [diff] [blame] | 98 | |
| 99 | supportedFlags = [] |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 100 | for flags in linkflags: |
| 101 | flags = Utils.to_list(flags) |
| 102 | if self.check_cxx(linkflags=['-Werror'] + flags, mandatory=False): |
| 103 | supportedFlags += flags |
Vince Lehman | 0a7da61 | 2014-10-29 14:39:29 -0500 | [diff] [blame] | 104 | |
| 105 | self.end_msg(' '.join(supportedFlags)) |
Davide Pesavento | c58cc7f | 2017-08-08 16:51:28 -0500 | [diff] [blame] | 106 | self.env.prepend_value('LINKFLAGS', supportedFlags) |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 107 | |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 108 | |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 109 | class CompilerFlags(object): |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 110 | def getCompilerVersion(self, conf): |
| 111 | return tuple(int(i) for i in conf.env.CC_VERSION) |
| 112 | |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 113 | def getGeneralFlags(self, conf): |
| 114 | """Get dict of CXXFLAGS, LINKFLAGS, and DEFINES that are always needed""" |
| 115 | return {'CXXFLAGS': [], 'LINKFLAGS': [], 'DEFINES': []} |
| 116 | |
| 117 | def getDebugFlags(self, conf): |
| 118 | """Get dict of CXXFLAGS, LINKFLAGS, and DEFINES that are needed only in debug mode""" |
| 119 | return {'CXXFLAGS': [], 'LINKFLAGS': [], 'DEFINES': ['_DEBUG']} |
| 120 | |
| 121 | def getOptimizedFlags(self, conf): |
| 122 | """Get dict of CXXFLAGS, LINKFLAGS, and DEFINES that are needed only in optimized mode""" |
| 123 | return {'CXXFLAGS': [], 'LINKFLAGS': [], 'DEFINES': ['NDEBUG']} |
| 124 | |
| 125 | class GccBasicFlags(CompilerFlags): |
| 126 | """ |
| 127 | This class defines basic flags that work for both gcc and clang compilers |
| 128 | """ |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 129 | def getGeneralFlags(self, conf): |
| 130 | flags = super(GccBasicFlags, self).getGeneralFlags(conf) |
Davide Pesavento | a08dc3f | 2018-05-24 00:40:28 -0400 | [diff] [blame] | 131 | flags['CXXFLAGS'] += ['-std=c++14'] |
Davide Pesavento | e556991 | 2019-01-29 19:39:06 -0500 | [diff] [blame^] | 132 | if Utils.unversioned_sys_platform() == 'linux': |
| 133 | flags['LINKFLAGS'] += ['-fuse-ld=gold'] |
| 134 | elif Utils.unversioned_sys_platform() == 'freebsd': |
| 135 | flags['LINKFLAGS'] += ['-fuse-ld=lld'] |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 136 | return flags |
| 137 | |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 138 | def getDebugFlags(self, conf): |
| 139 | flags = super(GccBasicFlags, self).getDebugFlags(conf) |
| 140 | flags['CXXFLAGS'] += ['-O0', |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 141 | '-Og', # gcc >= 4.8, clang >= 4.0 |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 142 | '-g3', |
| 143 | '-pedantic', |
| 144 | '-Wall', |
| 145 | '-Wextra', |
| 146 | '-Werror', |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 147 | '-Wnon-virtual-dtor', |
Alexander Afanasyev | 96bf2f0 | 2016-10-11 16:37:45 -0700 | [diff] [blame] | 148 | '-Wno-error=deprecated-declarations', # Bug #3795 |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 149 | '-Wno-error=maybe-uninitialized', # Bug #1615 |
| 150 | '-Wno-unused-parameter', |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 151 | ] |
Davide Pesavento | e556991 | 2019-01-29 19:39:06 -0500 | [diff] [blame^] | 152 | flags['LINKFLAGS'] += ['-Wl,-O1'] |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 153 | return flags |
| 154 | |
| 155 | def getOptimizedFlags(self, conf): |
| 156 | flags = super(GccBasicFlags, self).getOptimizedFlags(conf) |
| 157 | flags['CXXFLAGS'] += ['-O2', |
| 158 | '-g', |
| 159 | '-pedantic', |
| 160 | '-Wall', |
| 161 | '-Wextra', |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 162 | '-Wnon-virtual-dtor', |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 163 | '-Wno-unused-parameter', |
| 164 | ] |
Davide Pesavento | e556991 | 2019-01-29 19:39:06 -0500 | [diff] [blame^] | 165 | flags['LINKFLAGS'] += ['-Wl,-O1'] |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 166 | return flags |
| 167 | |
| 168 | class GccFlags(GccBasicFlags): |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 169 | def getDebugFlags(self, conf): |
| 170 | flags = super(GccFlags, self).getDebugFlags(conf) |
Davide Pesavento | a08dc3f | 2018-05-24 00:40:28 -0400 | [diff] [blame] | 171 | flags['CXXFLAGS'] += ['-fdiagnostics-color'] |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 172 | return flags |
| 173 | |
| 174 | def getOptimizedFlags(self, conf): |
| 175 | flags = super(GccFlags, self).getOptimizedFlags(conf) |
Davide Pesavento | a08dc3f | 2018-05-24 00:40:28 -0400 | [diff] [blame] | 176 | flags['CXXFLAGS'] += ['-fdiagnostics-color'] |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 177 | return flags |
| 178 | |
| 179 | class ClangFlags(GccBasicFlags): |
| 180 | def getGeneralFlags(self, conf): |
| 181 | flags = super(ClangFlags, self).getGeneralFlags(conf) |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 182 | if Utils.unversioned_sys_platform() == 'darwin' and self.getCompilerVersion(conf) >= (9, 0, 0): |
| 183 | # Bug #4296 |
| 184 | flags['CXXFLAGS'] += [['-isystem', '/usr/local/include'], # for Homebrew |
| 185 | ['-isystem', '/opt/local/include']] # for MacPorts |
Davide Pesavento | e556991 | 2019-01-29 19:39:06 -0500 | [diff] [blame^] | 186 | if Utils.unversioned_sys_platform() == 'freebsd': |
| 187 | flags['CXXFLAGS'] += [['-isystem', '/usr/local/include']] # Bug #4790 |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 188 | return flags |
| 189 | |
| 190 | def getDebugFlags(self, conf): |
| 191 | flags = super(ClangFlags, self).getDebugFlags(conf) |
| 192 | flags['CXXFLAGS'] += ['-fcolor-diagnostics', |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 193 | '-Wextra-semi', |
| 194 | '-Wundefined-func-template', |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 195 | '-Wno-unused-local-typedef', # Bugs #2657 and #3209 |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 196 | ] |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 197 | version = self.getCompilerVersion(conf) |
| 198 | if version < (3, 9, 0) or (Utils.unversioned_sys_platform() == 'darwin' and version < (8, 1, 0)): |
| 199 | flags['CXXFLAGS'] += ['-Wno-unknown-pragmas'] |
Davide Pesavento | e556991 | 2019-01-29 19:39:06 -0500 | [diff] [blame^] | 200 | if version < (6, 0, 0): |
| 201 | flags['CXXFLAGS'] += ['-Wno-missing-braces'] # Bug #4721 |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 202 | return flags |
| 203 | |
| 204 | def getOptimizedFlags(self, conf): |
| 205 | flags = super(ClangFlags, self).getOptimizedFlags(conf) |
| 206 | flags['CXXFLAGS'] += ['-fcolor-diagnostics', |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 207 | '-Wextra-semi', |
| 208 | '-Wundefined-func-template', |
Davide Pesavento | 6ebfd7c | 2017-08-26 17:52:54 -0400 | [diff] [blame] | 209 | '-Wno-unused-local-typedef', # Bugs #2657 and #3209 |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 210 | ] |
Alexander Afanasyev | 67758b1 | 2018-03-06 18:36:44 -0500 | [diff] [blame] | 211 | version = self.getCompilerVersion(conf) |
| 212 | if version < (3, 9, 0) or (Utils.unversioned_sys_platform() == 'darwin' and version < (8, 1, 0)): |
| 213 | flags['CXXFLAGS'] += ['-Wno-unknown-pragmas'] |
Davide Pesavento | e556991 | 2019-01-29 19:39:06 -0500 | [diff] [blame^] | 214 | if version < (6, 0, 0): |
| 215 | flags['CXXFLAGS'] += ['-Wno-missing-braces'] # Bug #4721 |
Alexander Afanasyev | f9f3910 | 2015-12-01 17:43:40 -0800 | [diff] [blame] | 216 | return flags |