blob: 386f89e4772be3879009701bae798490445e2796 [file] [log] [blame]
José Fonseca836a9f02009-09-01 12:22:52 +01001"""llvm
2
3Tool-specific initialization for LLVM
4
5"""
6
7#
8# Copyright (c) 2009 VMware, Inc.
9#
10# Permission is hereby granted, free of charge, to any person obtaining
11# a copy of this software and associated documentation files (the
12# "Software"), to deal in the Software without restriction, including
13# without limitation the rights to use, copy, modify, merge, publish,
14# distribute, sublicense, and/or sell copies of the Software, and to
15# permit persons to whom the Software is furnished to do so, subject to
16# the following conditions:
17#
18# The above copyright notice and this permission notice shall be included
19# in all copies or substantial portions of the Software.
20#
21# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
22# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
23# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
24# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
25# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
26# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
27# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
28#
29
30import os
31import os.path
José Fonseca8edc6b02010-03-05 15:07:50 +000032import re
pal1000dc099562019-03-01 12:30:15 +020033import platform as host_platform
José Fonsecabf484472009-10-22 19:11:48 +010034import sys
José Fonseca8edc6b02010-03-05 15:07:50 +000035import distutils.version
José Fonseca836a9f02009-09-01 12:22:52 +010036
José Fonseca836a9f02009-09-01 12:22:52 +010037import SCons.Errors
38import SCons.Util
39
40
José Fonseca013ff2f2014-10-22 13:09:59 +010041required_llvm_version = '3.3'
José Fonsecac0ef9a62014-05-08 13:32:07 +010042
43
José Fonseca836a9f02009-09-01 12:22:52 +010044def generate(env):
José Fonseca601498a2010-11-01 13:30:22 +000045 env['llvm'] = False
46
José Fonseca836a9f02009-09-01 12:22:52 +010047 try:
48 llvm_dir = os.environ['LLVM']
49 except KeyError:
50 # Do nothing -- use the system headers/libs
José Fonsecabf484472009-10-22 19:11:48 +010051 llvm_dir = None
José Fonseca836a9f02009-09-01 12:22:52 +010052 else:
53 if not os.path.isdir(llvm_dir):
Eric Engestrombb66af92017-09-19 14:09:01 +010054 raise SCons.Errors.InternalError("Specified LLVM directory not found")
José Fonseca836a9f02009-09-01 12:22:52 +010055
56 if env['debug']:
57 llvm_subdir = 'Debug'
58 else:
59 llvm_subdir = 'Release'
60
61 llvm_bin_dir = os.path.join(llvm_dir, llvm_subdir, 'bin')
62 if not os.path.isdir(llvm_bin_dir):
José Fonseca459ea002009-09-16 10:39:06 +010063 llvm_bin_dir = os.path.join(llvm_dir, 'bin')
64 if not os.path.isdir(llvm_bin_dir):
Eric Engestrombb66af92017-09-19 14:09:01 +010065 raise SCons.Errors.InternalError("LLVM binary directory not found")
José Fonseca836a9f02009-09-01 12:22:52 +010066
67 env.PrependENVPath('PATH', llvm_bin_dir)
68
José Fonseca9e3728c2009-11-25 18:06:12 +000069 if env['platform'] == 'windows':
José Fonsecabf484472009-10-22 19:11:48 +010070 # XXX: There is no llvm-config on Windows, so assume a standard layout
José Fonseca8edc6b02010-03-05 15:07:50 +000071 if llvm_dir is None:
Eric Engestrom7d482192017-09-19 13:56:34 +010072 print('scons: LLVM environment variable must be specified when building for windows')
José Fonseca601498a2010-11-01 13:30:22 +000073 return
José Fonseca8edc6b02010-03-05 15:07:50 +000074
75 # Try to determine the LLVM version from llvm/Config/config.h
Olivier Penab94a4e82015-04-27 10:23:58 +000076 llvm_config = os.path.join(llvm_dir, 'include/llvm/Config/llvm-config.h')
José Fonseca8edc6b02010-03-05 15:07:50 +000077 if not os.path.exists(llvm_config):
Eric Engestrom7d482192017-09-19 13:56:34 +010078 print('scons: could not find %s' % llvm_config)
José Fonseca601498a2010-11-01 13:30:22 +000079 return
Olivier Penab94a4e82015-04-27 10:23:58 +000080 llvm_version_major_re = re.compile(r'^#define LLVM_VERSION_MAJOR ([0-9]+)')
81 llvm_version_minor_re = re.compile(r'^#define LLVM_VERSION_MINOR ([0-9]+)')
José Fonseca8edc6b02010-03-05 15:07:50 +000082 llvm_version = None
Olivier Penab94a4e82015-04-27 10:23:58 +000083 llvm_version_major = None
84 llvm_version_minor = None
José Fonseca8edc6b02010-03-05 15:07:50 +000085 for line in open(llvm_config, 'rt'):
Olivier Penab94a4e82015-04-27 10:23:58 +000086 mo = llvm_version_major_re.match(line)
José Fonseca8edc6b02010-03-05 15:07:50 +000087 if mo:
Olivier Penab94a4e82015-04-27 10:23:58 +000088 llvm_version_major = mo.group(1)
89 mo = llvm_version_minor_re.match(line)
90 if mo:
91 llvm_version_minor = mo.group(1)
92 if llvm_version_major is not None and llvm_version_minor is not None:
93 llvm_version = distutils.version.LooseVersion('%s.%s' % (llvm_version_major, llvm_version_minor))
94
José Fonseca8edc6b02010-03-05 15:07:50 +000095 if llvm_version is None:
Eric Engestrom7d482192017-09-19 13:56:34 +010096 print('scons: could not determine the LLVM version from %s' % llvm_config)
José Fonseca601498a2010-11-01 13:30:22 +000097 return
José Fonsecac0ef9a62014-05-08 13:32:07 +010098 if llvm_version < distutils.version.LooseVersion(required_llvm_version):
Eric Engestrom7d482192017-09-19 13:56:34 +010099 print('scons: LLVM version %s found, but %s is required' % (llvm_version, required_llvm_version))
José Fonsecac0ef9a62014-05-08 13:32:07 +0100100 return
José Fonseca8edc6b02010-03-05 15:07:50 +0000101
102 env.Prepend(CPPPATH = [os.path.join(llvm_dir, 'include')])
José Fonseca8edc6b02010-03-05 15:07:50 +0000103 env.Prepend(LIBPATH = [os.path.join(llvm_dir, 'lib')])
pal1000a413b552019-09-06 17:34:30 +0300104
105 # LLVM 5.0 and newer requires MinGW w/ pthreads due to use of std::thread and friends.
106 if llvm_version >= distutils.version.LooseVersion('5.0') and env['crosscompile']:
107 assert env['gcc']
108 env.AppendUnique(CXXFLAGS = ['-posix'])
109
110 # LIBS should match the output of `llvm-config --libs engine mcjit bitwriter x86asmprinter irreader` for LLVM<=7.0
111 # and `llvm-config --libs engine irreader` for LLVM>=8.0
112 # LLVMAggressiveInstCombine library part of engine component can be safely omitted as it's not used.
113 if llvm_version >= distutils.version.LooseVersion('9.0'):
114 env.Prepend(LIBS = [
115 'LLVMX86Disassembler', 'LLVMX86AsmParser',
116 'LLVMX86CodeGen', 'LLVMSelectionDAG', 'LLVMAsmPrinter',
117 'LLVMDebugInfoCodeView', 'LLVMCodeGen',
118 'LLVMScalarOpts', 'LLVMInstCombine',
119 'LLVMTransformUtils',
120 'LLVMBitWriter', 'LLVMX86Desc',
121 'LLVMMCDisassembler', 'LLVMX86Info',
122 'LLVMX86Utils',
123 'LLVMMCJIT', 'LLVMExecutionEngine', 'LLVMTarget',
124 'LLVMAnalysis', 'LLVMProfileData',
125 'LLVMRuntimeDyld', 'LLVMObject', 'LLVMMCParser',
126 'LLVMBitReader', 'LLVMMC', 'LLVMCore',
127 'LLVMSupport',
128 'LLVMIRReader', 'LLVMAsmParser',
129 'LLVMDemangle', 'LLVMGlobalISel', 'LLVMDebugInfoMSF',
130 'LLVMBinaryFormat',
131 'LLVMRemarks', 'LLVMBitstreamReader', 'LLVMDebugInfoDWARF',
132 ])
133 elif llvm_version >= distutils.version.LooseVersion('5.0'):
Alexandru-Liviu Prodeac1b01372017-09-15 07:26:33 +0000134 env.Prepend(LIBS = [
135 'LLVMX86Disassembler', 'LLVMX86AsmParser',
136 'LLVMX86CodeGen', 'LLVMSelectionDAG', 'LLVMAsmPrinter',
137 'LLVMDebugInfoCodeView', 'LLVMCodeGen',
138 'LLVMScalarOpts', 'LLVMInstCombine',
139 'LLVMTransformUtils',
140 'LLVMBitWriter', 'LLVMX86Desc',
141 'LLVMMCDisassembler', 'LLVMX86Info',
142 'LLVMX86AsmPrinter', 'LLVMX86Utils',
143 'LLVMMCJIT', 'LLVMExecutionEngine', 'LLVMTarget',
144 'LLVMAnalysis', 'LLVMProfileData',
145 'LLVMRuntimeDyld', 'LLVMObject', 'LLVMMCParser',
146 'LLVMBitReader', 'LLVMMC', 'LLVMCore',
147 'LLVMSupport',
148 'LLVMIRReader', 'LLVMAsmParser',
149 'LLVMDemangle', 'LLVMGlobalISel', 'LLVMDebugInfoMSF',
150 'LLVMBinaryFormat',
151 ])
152 elif llvm_version >= distutils.version.LooseVersion('4.0'):
Ben Boeckel58f51f02017-04-27 16:31:48 -0400153 env.Prepend(LIBS = [
154 'LLVMX86Disassembler', 'LLVMX86AsmParser',
155 'LLVMX86CodeGen', 'LLVMSelectionDAG', 'LLVMAsmPrinter',
156 'LLVMDebugInfoCodeView', 'LLVMCodeGen',
157 'LLVMScalarOpts', 'LLVMInstCombine',
158 'LLVMTransformUtils',
159 'LLVMBitWriter', 'LLVMX86Desc',
160 'LLVMMCDisassembler', 'LLVMX86Info',
161 'LLVMX86AsmPrinter', 'LLVMX86Utils',
162 'LLVMMCJIT', 'LLVMExecutionEngine', 'LLVMTarget',
163 'LLVMAnalysis', 'LLVMProfileData',
164 'LLVMRuntimeDyld', 'LLVMObject', 'LLVMMCParser',
165 'LLVMBitReader', 'LLVMMC', 'LLVMCore',
166 'LLVMSupport',
167 'LLVMIRReader', 'LLVMAsmParser',
168 'LLVMDemangle', 'LLVMGlobalISel', 'LLVMDebugInfoMSF',
169 ])
170 elif llvm_version >= distutils.version.LooseVersion('3.9'):
George Kyriazis30ae2cb2016-11-09 16:35:48 -0600171 env.Prepend(LIBS = [
172 'LLVMX86Disassembler', 'LLVMX86AsmParser',
173 'LLVMX86CodeGen', 'LLVMSelectionDAG', 'LLVMAsmPrinter',
174 'LLVMDebugInfoCodeView', 'LLVMCodeGen',
175 'LLVMScalarOpts', 'LLVMInstCombine',
176 'LLVMInstrumentation', 'LLVMTransformUtils',
177 'LLVMBitWriter', 'LLVMX86Desc',
178 'LLVMMCDisassembler', 'LLVMX86Info',
179 'LLVMX86AsmPrinter', 'LLVMX86Utils',
180 'LLVMMCJIT', 'LLVMExecutionEngine', 'LLVMTarget',
181 'LLVMAnalysis', 'LLVMProfileData',
182 'LLVMRuntimeDyld', 'LLVMObject', 'LLVMMCParser',
183 'LLVMBitReader', 'LLVMMC', 'LLVMCore',
184 'LLVMSupport',
185 'LLVMIRReader', 'LLVMASMParser'
186 ])
187 elif llvm_version >= distutils.version.LooseVersion('3.7'):
Olivier Penaa5256012015-12-07 17:13:18 +0100188 env.Prepend(LIBS = [
189 'LLVMBitWriter', 'LLVMX86Disassembler', 'LLVMX86AsmParser',
190 'LLVMX86CodeGen', 'LLVMSelectionDAG', 'LLVMAsmPrinter',
191 'LLVMCodeGen', 'LLVMScalarOpts', 'LLVMProfileData',
192 'LLVMInstCombine', 'LLVMInstrumentation', 'LLVMTransformUtils', 'LLVMipa',
193 'LLVMAnalysis', 'LLVMX86Desc', 'LLVMMCDisassembler',
194 'LLVMX86Info', 'LLVMX86AsmPrinter', 'LLVMX86Utils',
195 'LLVMMCJIT', 'LLVMTarget', 'LLVMExecutionEngine',
196 'LLVMRuntimeDyld', 'LLVMObject', 'LLVMMCParser',
197 'LLVMBitReader', 'LLVMMC', 'LLVMCore', 'LLVMSupport'
198 ])
199 elif llvm_version >= distutils.version.LooseVersion('3.6'):
Olivier Penab94a4e82015-04-27 10:23:58 +0000200 env.Prepend(LIBS = [
201 'LLVMBitWriter', 'LLVMX86Disassembler', 'LLVMX86AsmParser',
202 'LLVMX86CodeGen', 'LLVMSelectionDAG', 'LLVMAsmPrinter',
203 'LLVMCodeGen', 'LLVMScalarOpts', 'LLVMProfileData',
204 'LLVMInstCombine', 'LLVMTransformUtils', 'LLVMipa',
205 'LLVMAnalysis', 'LLVMX86Desc', 'LLVMMCDisassembler',
206 'LLVMX86Info', 'LLVMX86AsmPrinter', 'LLVMX86Utils',
207 'LLVMMCJIT', 'LLVMTarget', 'LLVMExecutionEngine',
208 'LLVMRuntimeDyld', 'LLVMObject', 'LLVMMCParser',
209 'LLVMBitReader', 'LLVMMC', 'LLVMCore', 'LLVMSupport'
210 ])
211 elif llvm_version >= distutils.version.LooseVersion('3.5'):
Jose Fonseca5b941ce2015-02-03 10:16:50 +0000212 env.Prepend(LIBS = [
Jose Fonseca0db4ef92015-05-28 16:55:10 +0100213 'LLVMMCDisassembler',
Jose Fonseca5b941ce2015-02-03 10:16:50 +0000214 'LLVMBitWriter', 'LLVMMCJIT', 'LLVMRuntimeDyld',
215 'LLVMX86Disassembler', 'LLVMX86AsmParser', 'LLVMX86CodeGen',
216 'LLVMSelectionDAG', 'LLVMAsmPrinter', 'LLVMX86Desc',
217 'LLVMObject', 'LLVMMCParser', 'LLVMBitReader', 'LLVMX86Info',
218 'LLVMX86AsmPrinter', 'LLVMX86Utils', 'LLVMJIT',
219 'LLVMExecutionEngine', 'LLVMCodeGen', 'LLVMScalarOpts',
220 'LLVMInstCombine', 'LLVMTransformUtils', 'LLVMipa',
221 'LLVMAnalysis', 'LLVMTarget', 'LLVMMC', 'LLVMCore',
222 'LLVMSupport'
223 ])
224 else:
Keith Kriewallefd83112013-02-28 15:40:02 +0000225 env.Prepend(LIBS = [
Jose Fonseca0db4ef92015-05-28 16:55:10 +0100226 'LLVMMCDisassembler',
Keith Kriewallefd83112013-02-28 15:40:02 +0000227 'LLVMBitWriter', 'LLVMX86Disassembler', 'LLVMX86AsmParser',
228 'LLVMX86CodeGen', 'LLVMX86Desc', 'LLVMSelectionDAG',
229 'LLVMAsmPrinter', 'LLVMMCParser', 'LLVMX86AsmPrinter',
José Fonseca2c02f342014-05-14 12:20:14 +0100230 'LLVMX86Utils', 'LLVMX86Info', 'LLVMMCJIT', 'LLVMJIT',
Keith Kriewallefd83112013-02-28 15:40:02 +0000231 'LLVMExecutionEngine', 'LLVMCodeGen', 'LLVMScalarOpts',
232 'LLVMInstCombine', 'LLVMTransformUtils', 'LLVMipa',
233 'LLVMAnalysis', 'LLVMTarget', 'LLVMMC', 'LLVMCore',
234 'LLVMSupport', 'LLVMRuntimeDyld', 'LLVMObject'
235 ])
José Fonseca8edc6b02010-03-05 15:07:50 +0000236 env.Append(LIBS = [
237 'imagehlp',
238 'psapi',
Brian Paul3d1af782011-08-25 15:14:37 -0600239 'shell32',
Jose Fonseca8841c2c2018-06-01 19:57:31 +0100240 'advapi32',
241 'ole32',
242 'uuid',
José Fonseca8edc6b02010-03-05 15:07:50 +0000243 ])
Jose Fonseca8841c2c2018-06-01 19:57:31 +0100244
pal1000dc099562019-03-01 12:30:15 +0200245 # Mingw-w64 zlib is required when building with LLVM support in MSYS2 environment
246 if host_platform.system().lower().startswith('mingw'):
247 env.Append(LIBS = [
248 'z',
249 ])
250
José Fonseca8edc6b02010-03-05 15:07:50 +0000251 if env['msvc']:
252 # Some of the LLVM C headers use the inline keyword without
253 # defining it.
254 env.Append(CPPDEFINES = [('inline', '__inline')])
Jose Fonsecae518d972015-03-19 22:09:20 +0000255 # Match some of the warning options from llvm/cmake/modules/HandleLLVMOptions.cmake
256 env.AppendUnique(CXXFLAGS = [
257 '/wd4355', # 'this' : used in base member initializer list
258 '/wd4624', # 'derived class' : destructor could not be generated because a base class destructor is inaccessible
259 ])
José Fonsecae3a3a532010-09-29 14:24:52 +0100260 if env['build'] in ('debug', 'checked'):
José Fonseca8edc6b02010-03-05 15:07:50 +0000261 # LLVM libraries are static, build with /MT, and they
262 # automatically link agains LIBCMT. When we're doing a
263 # debug build we'll be linking against LIBCMTD, so disable
264 # that.
265 env.Append(LINKFLAGS = ['/nodefaultlib:LIBCMT'])
José Fonsecaea532f02010-04-10 02:41:39 +0100266 else:
Vinson Leef07da5a2016-11-22 17:01:35 -0800267 llvm_config = os.environ.get('LLVM_CONFIG', 'llvm-config')
268 if not env.Detect(llvm_config):
Eric Engestrom7d482192017-09-19 13:56:34 +0100269 print('scons: %s script not found' % llvm_config)
José Fonseca601498a2010-11-01 13:30:22 +0000270 return
José Fonsecaea532f02010-04-10 02:41:39 +0100271
Vinson Leef07da5a2016-11-22 17:01:35 -0800272 llvm_version = env.backtick('%s --version' % llvm_config).rstrip()
José Fonseca19a63332010-03-06 09:18:15 +0000273 llvm_version = distutils.version.LooseVersion(llvm_version)
José Fonseca836a9f02009-09-01 12:22:52 +0100274
José Fonsecac0ef9a62014-05-08 13:32:07 +0100275 if llvm_version < distutils.version.LooseVersion(required_llvm_version):
Eric Engestrom7d482192017-09-19 13:56:34 +0100276 print('scons: LLVM version %s found, but %s is required' % (llvm_version, required_llvm_version))
José Fonsecac0ef9a62014-05-08 13:32:07 +0100277 return
278
Vinson Lee79f48c92009-09-07 15:16:25 +0100279 try:
José Fonsecaacf82192011-07-11 15:36:40 +0100280 # Treat --cppflags specially to prevent NDEBUG from disabling
281 # assertion failures in debug builds.
Vinson Leef07da5a2016-11-22 17:01:35 -0800282 cppflags = env.ParseFlags('!%s --cppflags' % llvm_config)
José Fonsecaacf82192011-07-11 15:36:40 +0100283 try:
284 cppflags['CPPDEFINES'].remove('NDEBUG')
285 except ValueError:
286 pass
287 env.MergeFlags(cppflags)
288
Alexander von Gluck IV9aad1ba2013-10-16 20:24:13 -0500289 # Match llvm --fno-rtti flag
Vinson Leef07da5a2016-11-22 17:01:35 -0800290 cxxflags = env.backtick('%s --cxxflags' % llvm_config).split()
Alexander von Gluck IV9aad1ba2013-10-16 20:24:13 -0500291 if '-fno-rtti' in cxxflags:
292 env.Append(CXXFLAGS = ['-fno-rtti'])
293
Roland Scheidegger84f3f1c2019-05-24 03:46:07 +0200294 if llvm_version < distutils.version.LooseVersion('9.0'):
295 components = ['engine', 'mcjit', 'bitwriter', 'x86asmprinter', 'mcdisassembler', 'irreader']
296 else:
297 components = ['engine', 'mcjit', 'bitwriter', 'mcdisassembler', 'irreader']
José Fonseca34697152012-07-13 18:09:30 +0100298
Vinson Leef07da5a2016-11-22 17:01:35 -0800299 env.ParseConfig('%s --libs ' % llvm_config + ' '.join(components))
300 env.ParseConfig('%s --ldflags' % llvm_config)
Vinson Lee35a34142013-12-19 15:55:28 -0800301 if llvm_version >= distutils.version.LooseVersion('3.5'):
Vinson Leef07da5a2016-11-22 17:01:35 -0800302 env.ParseConfig('%s --system-libs' % llvm_config)
Kai Wasserbäch1abe8732019-08-17 10:59:43 +0200303 env.Append(CXXFLAGS = ['-std=c++14'])
Vinson Lee79f48c92009-09-07 15:16:25 +0100304 except OSError:
Eric Engestrom7d482192017-09-19 13:56:34 +0100305 print('scons: llvm-config version %s failed' % llvm_version)
José Fonseca601498a2010-11-01 13:30:22 +0000306 return
José Fonseca8edc6b02010-03-05 15:07:50 +0000307
308 assert llvm_version is not None
José Fonseca601498a2010-11-01 13:30:22 +0000309 env['llvm'] = True
José Fonseca8edc6b02010-03-05 15:07:50 +0000310
Eric Engestrom7d482192017-09-19 13:56:34 +0100311 print('scons: Found LLVM version %s' % llvm_version)
José Fonseca8edc6b02010-03-05 15:07:50 +0000312 env['LLVM_VERSION'] = llvm_version
313
314 # Define HAVE_LLVM macro with the major/minor version number (e.g., 0x0206 for 2.6)
315 llvm_version_major = int(llvm_version.version[0])
316 llvm_version_minor = int(llvm_version.version[1])
317 llvm_version_hex = '0x%02x%02x' % (llvm_version_major, llvm_version_minor)
318 env.Prepend(CPPDEFINES = [('HAVE_LLVM', llvm_version_hex)])
José Fonseca836a9f02009-09-01 12:22:52 +0100319
320def exists(env):
321 return True
322
323# vim:set ts=4 sw=4 et: