2014-02-12 22:54:17 +08:00
#!/usr/bin/python
2017-08-02 23:00:59 +08:00
from __future__ import print_function
# We only need this for int() cast, which works by default in python 2
try :
from builtins import int
except :
pass
2012-01-09 06:09:58 +08:00
def c_compiler_rule ( b , name , description , compiler , flags ) :
command = " %s -MMD -MF $out.d %s -c -o $out $in " % ( compiler , flags )
b . rule ( name , command , description + " $out " , depfile = " $out.d " )
2013-06-27 02:20:03 +08:00
version_major = 0 ;
2015-08-07 16:31:37 +08:00
version_minor = 2 ;
2014-12-31 23:27:53 +08:00
version_patch = 0 ;
2013-06-27 02:20:03 +08:00
2012-01-09 06:09:58 +08:00
from optparse import OptionParser
import os
2013-12-30 00:39:53 +08:00
import string
2012-01-09 06:09:58 +08:00
from subprocess import *
import sys
srcdir = os . path . dirname ( sys . argv [ 0 ] )
sys . path . insert ( 0 , os . path . join ( srcdir , ' build ' ) )
import metabuild
p = OptionParser ( )
p . add_option ( ' --with-llvm-config ' , metavar = ' PATH ' ,
help = ' use given llvm-config script ' )
2013-10-23 10:49:27 +08:00
p . add_option ( ' --with-cxx-compiler ' , metavar = ' PATH ' ,
help = ' use given C++ compiler ' )
2012-06-02 01:29:59 +08:00
p . add_option ( ' --prefix ' , metavar = ' PATH ' ,
help = ' install to given prefix ' )
2013-06-27 02:20:03 +08:00
p . add_option ( ' --libexecdir ' , metavar = ' PATH ' ,
help = ' install *.bc to given dir ' )
p . add_option ( ' --includedir ' , metavar = ' PATH ' ,
help = ' install include files to given dir ' )
p . add_option ( ' --pkgconfigdir ' , metavar = ' PATH ' ,
help = ' install clc.pc to given dir ' )
2012-01-09 06:09:58 +08:00
p . add_option ( ' -g ' , metavar = ' GENERATOR ' , default = ' make ' ,
help = ' use given generator (default: make) ' )
2015-04-21 02:49:50 +08:00
p . add_option ( ' --enable-runtime-subnormal ' , action = " store_true " , default = False ,
help = ' Allow runtimes to choose subnormal support ' )
2012-01-09 06:09:58 +08:00
( options , args ) = p . parse_args ( )
llvm_config_exe = options . with_llvm_config or " llvm-config "
2013-06-27 02:20:03 +08:00
prefix = options . prefix
if not prefix :
prefix = ' /usr/local '
libexecdir = options . libexecdir
if not libexecdir :
libexecdir = os . path . join ( prefix , ' lib/clc ' )
includedir = options . includedir
if not includedir :
includedir = os . path . join ( prefix , ' include ' )
pkgconfigdir = options . pkgconfigdir
if not pkgconfigdir :
2013-09-06 07:27:58 +08:00
pkgconfigdir = os . path . join ( prefix , ' share/pkgconfig ' )
2013-06-27 02:20:03 +08:00
2012-01-09 06:09:58 +08:00
def llvm_config ( args ) :
try :
2017-08-02 23:00:59 +08:00
# Universal newlines translate different newline formats to '\n'
# it also force the input to be string instead of bytes in python 3
proc = Popen ( [ llvm_config_exe ] + args , stdout = PIPE , universal_newlines = True )
2012-01-09 06:09:58 +08:00
return proc . communicate ( ) [ 0 ] . rstrip ( ) . replace ( ' \n ' , ' ' )
except OSError :
2017-08-02 23:00:59 +08:00
print ( " Error executing llvm-config. " )
print ( " Please ensure that llvm-config is in your $PATH, or use --with-llvm-config. " )
2012-01-09 06:09:58 +08:00
sys . exit ( 1 )
2017-08-02 23:00:59 +08:00
llvm_version = llvm_config ( [ ' --version ' ] ) . replace ( ' svn ' , ' ' ) . split ( ' . ' )
2015-08-07 16:31:37 +08:00
llvm_int_version = int ( llvm_version [ 0 ] ) * 100 + int ( llvm_version [ 1 ] ) * 10
2015-04-29 23:37:06 +08:00
llvm_string_version = ' LLVM ' + llvm_version [ 0 ] + ' . ' + llvm_version [ 1 ]
2016-11-12 05:34:47 +08:00
if llvm_int_version < 400 :
2017-08-02 23:00:59 +08:00
print ( " libclc requires LLVM >= 4.0 " )
2015-08-07 16:31:37 +08:00
sys . exit ( 1 )
2014-12-31 23:27:53 +08:00
llvm_system_libs = llvm_config ( [ ' --system-libs ' ] )
2012-01-09 06:09:58 +08:00
llvm_bindir = llvm_config ( [ ' --bindir ' ] )
2012-12-05 15:39:02 +08:00
llvm_core_libs = llvm_config ( [ ' --libs ' , ' core ' , ' bitreader ' , ' bitwriter ' ] ) + ' ' + \
2013-12-30 00:39:53 +08:00
llvm_system_libs + ' ' + \
2012-12-05 15:39:02 +08:00
llvm_config ( [ ' --ldflags ' ] )
2012-01-09 06:09:58 +08:00
llvm_cxxflags = llvm_config ( [ ' --cxxflags ' ] ) + ' -fno-exceptions -fno-rtti '
2014-09-03 01:54:59 +08:00
llvm_libdir = llvm_config ( [ ' --libdir ' ] )
2012-01-09 06:09:58 +08:00
llvm_clang = os . path . join ( llvm_bindir , ' clang ' )
llvm_link = os . path . join ( llvm_bindir , ' llvm-link ' )
llvm_opt = os . path . join ( llvm_bindir , ' opt ' )
2013-10-23 10:49:27 +08:00
cxx_compiler = options . with_cxx_compiler
if not cxx_compiler :
cxx_compiler = os . path . join ( llvm_bindir , ' clang++ ' )
2013-06-27 02:20:38 +08:00
available_targets = {
' r600-- ' : { ' devices ' :
2016-06-18 04:30:50 +08:00
[ { ' gpu ' : ' cedar ' , ' aliases ' : [ ' palm ' , ' sumo ' , ' sumo2 ' , ' redwood ' , ' juniper ' ] } ,
{ ' gpu ' : ' cypress ' , ' aliases ' : [ ' hemlock ' ] } ,
{ ' gpu ' : ' barts ' , ' aliases ' : [ ' turks ' , ' caicos ' ] } ,
{ ' gpu ' : ' cayman ' , ' aliases ' : [ ' aruba ' ] } ] } ,
2015-01-07 04:42:12 +08:00
' amdgcn-- ' : { ' devices ' :
2016-08-31 02:00:41 +08:00
[ { ' gpu ' : ' tahiti ' , ' aliases ' : [ ' pitcairn ' , ' verde ' , ' oland ' , ' hainan ' , ' bonaire ' , ' kabini ' , ' kaveri ' , ' hawaii ' , ' mullins ' , ' tonga ' , ' carrizo ' , ' iceland ' , ' fiji ' , ' stoney ' , ' polaris10 ' , ' polaris11 ' ] } ] } ,
2016-04-08 03:54:19 +08:00
' amdgcn--amdhsa ' : { ' devices ' :
2016-08-31 02:00:41 +08:00
[ { ' gpu ' : ' ' , ' aliases ' : [ ' bonaire ' , ' hawaii ' , ' kabini ' , ' kaveri ' , ' mullins ' , ' carrizo ' , ' stoney ' , ' fiji ' , ' iceland ' , ' tonga ' , ' polaris10 ' , ' polaris11 ' ] } ] } ,
2016-06-18 04:30:50 +08:00
' nvptx-- ' : { ' devices ' : [ { ' gpu ' : ' ' , ' aliases ' : [ ] } ] } ,
' nvptx64-- ' : { ' devices ' : [ { ' gpu ' : ' ' , ' aliases ' : [ ] } ] } ,
' nvptx--nvidiacl ' : { ' devices ' : [ { ' gpu ' : ' ' , ' aliases ' : [ ] } ] } ,
' nvptx64--nvidiacl ' : { ' devices ' : [ { ' gpu ' : ' ' , ' aliases ' : [ ] } ] } ,
2013-06-27 02:20:38 +08:00
}
2016-09-17 06:43:33 +08:00
available_targets [ ' amdgcn-mesa-mesa3d ' ] = available_targets [ ' amdgcn-- ' ]
default_targets = [ ' nvptx--nvidiacl ' , ' nvptx64--nvidiacl ' , ' r600-- ' , ' amdgcn-- ' , ' amdgcn--amdhsa ' , ' amdgcn-mesa-mesa3d ' ]
2012-01-09 06:09:58 +08:00
targets = args
if not targets :
targets = default_targets
b = metabuild . from_name ( options . g )
b . rule ( " LLVM_AS " , " %s -o $out $in " % os . path . join ( llvm_bindir , " llvm-as " ) ,
' LLVM-AS $out ' )
b . rule ( " LLVM_LINK " , command = llvm_link + " -o $out $in " ,
description = ' LLVM-LINK $out ' )
b . rule ( " OPT " , command = llvm_opt + " -O3 -o $out $in " ,
description = ' OPT $out ' )
2013-10-23 10:49:27 +08:00
c_compiler_rule ( b , " LLVM_TOOL_CXX " , ' CXX ' , cxx_compiler , llvm_cxxflags )
2014-09-03 01:54:59 +08:00
b . rule ( " LLVM_TOOL_LINK " , cxx_compiler + " -o $out $in %s " % llvm_core_libs + " -Wl,-rpath %s " % llvm_libdir , ' LINK $out ' )
2012-01-09 06:09:58 +08:00
prepare_builtins = os . path . join ( ' utils ' , ' prepare-builtins ' )
b . build ( os . path . join ( ' utils ' , ' prepare-builtins.o ' ) , " LLVM_TOOL_CXX " ,
os . path . join ( srcdir , ' utils ' , ' prepare-builtins.cpp ' ) )
b . build ( prepare_builtins , " LLVM_TOOL_LINK " ,
os . path . join ( ' utils ' , ' prepare-builtins.o ' ) )
b . rule ( " PREPARE_BUILTINS " , " %s -o $out $in " % prepare_builtins ,
' PREPARE-BUILTINS $out ' )
2013-10-11 03:09:01 +08:00
b . rule ( " PYTHON_GEN " , " python < $in > $out " , " PYTHON_GEN $out " )
b . build ( ' generic/lib/convert.cl ' , " PYTHON_GEN " , [ ' generic/lib/gen_convert.py ' ] )
2012-01-09 06:09:58 +08:00
manifest_deps = set ( [ sys . argv [ 0 ] , os . path . join ( srcdir , ' build ' , ' metabuild.py ' ) ,
os . path . join ( srcdir , ' build ' , ' ninja_syntax.py ' ) ] )
2013-06-27 02:20:03 +08:00
install_files_bc = [ ]
2012-06-02 01:29:59 +08:00
install_deps = [ ]
2015-04-21 02:49:50 +08:00
# Create rules for subnormal helper objects
for src in [ ' subnormal_disable.ll ' , ' subnormal_use_default.ll ' ] :
obj_name = src [ : - 2 ] + ' bc '
obj = os . path . join ( ' generic-- ' , ' lib ' , obj_name )
src_file = os . path . join ( ' generic ' , ' lib ' , src )
b . build ( obj , ' LLVM_AS ' , src_file )
b . default ( obj )
install_files_bc . append ( ( obj , obj ) )
install_deps . append ( obj )
2013-06-27 02:20:03 +08:00
# Create libclc.pc
clc = open ( ' libclc.pc ' , ' w ' )
clc . write ( ' includedir= %(inc)s \n libexecdir= %(lib)s \n \n Name: libclc \n Description: Library requirements of the OpenCL C programming language \n Version: %(maj)s . %(min)s . %(pat)s \n Cflags: -I$ {includedir} \n Libs: -L$ {libexecdir} ' %
{ ' inc ' : includedir , ' lib ' : libexecdir , ' maj ' : version_major , ' min ' : version_minor , ' pat ' : version_patch } )
clc . close ( )
2012-01-09 06:09:58 +08:00
for target in targets :
( t_arch , t_vendor , t_os ) = target . split ( ' - ' )
archs = [ t_arch ]
2012-05-29 04:42:15 +08:00
if t_arch == ' nvptx ' or t_arch == ' nvptx64 ' :
2012-01-09 06:09:58 +08:00
archs . append ( ' ptx ' )
archs . append ( ' generic ' )
subdirs = [ ]
for arch in archs :
subdirs . append ( " %s - %s - %s " % ( arch , t_vendor , t_os ) )
subdirs . append ( " %s - %s " % ( arch , t_os ) )
2016-09-17 06:43:33 +08:00
if t_os == ' mesa3d ' :
subdirs . append ( ' amdgcn-amdhsa ' )
2012-01-09 06:09:58 +08:00
subdirs . append ( arch )
2016-02-13 09:01:59 +08:00
if arch == ' amdgcn ' or arch == ' r600 ' :
subdirs . append ( ' amdgpu ' )
2012-01-09 06:09:58 +08:00
2012-06-02 01:29:59 +08:00
incdirs = filter ( os . path . isdir ,
[ os . path . join ( srcdir , subdir , ' include ' ) for subdir in subdirs ] )
libdirs = filter ( lambda d : os . path . isfile ( os . path . join ( d , ' SOURCES ' ) ) ,
[ os . path . join ( srcdir , subdir , ' lib ' ) for subdir in subdirs ] )
2012-01-09 06:09:58 +08:00
2017-08-02 23:00:59 +08:00
# The above are iterables in python3 but we might use them multiple times
# if more then one device is supported.
incdirs = list ( incdirs )
libdirs = list ( libdirs )
2012-06-02 01:29:59 +08:00
clang_cl_includes = ' ' . join ( [ " -I %s " % incdir for incdir in incdirs ] )
2012-01-09 06:09:58 +08:00
2013-06-27 02:20:38 +08:00
for device in available_targets [ target ] [ ' devices ' ] :
# The rule for building a .bc file for the specified architecture using clang.
clang_bc_flags = " -target %s -I`dirname $in` %s " \
2013-12-30 00:39:55 +08:00
" -fno-builtin " \
2013-10-11 03:08:51 +08:00
" -D__CLC_INTERNAL " \
2016-06-18 04:30:50 +08:00
" -emit-llvm " % ( target , clang_cl_includes )
2013-06-27 02:20:38 +08:00
if device [ ' gpu ' ] != ' ' :
clang_bc_flags + = ' -mcpu= ' + device [ ' gpu ' ]
2014-01-30 04:03:27 +08:00
clang_bc_rule = " CLANG_CL_BC_ " + target + " _ " + device [ ' gpu ' ]
2013-06-27 02:20:38 +08:00
c_compiler_rule ( b , clang_bc_rule , " LLVM-CC " , llvm_clang , clang_bc_flags )
objects = [ ]
sources_seen = set ( )
2015-04-29 23:37:06 +08:00
compats_seen = set ( )
2013-06-27 02:20:38 +08:00
if device [ ' gpu ' ] == ' ' :
full_target_name = target
obj_suffix = ' '
else :
full_target_name = device [ ' gpu ' ] + ' - ' + target
obj_suffix = ' . ' + device [ ' gpu ' ]
for libdir in libdirs :
subdir_list_file = os . path . join ( libdir , ' SOURCES ' )
manifest_deps . add ( subdir_list_file )
override_list_file = os . path . join ( libdir , ' OVERRIDES ' )
2015-04-29 23:37:06 +08:00
compat_list_file = os . path . join ( libdir ,
' SOURCES_ ' + llvm_string_version )
# Build compat list
if os . path . exists ( compat_list_file ) :
for compat in open ( compat_list_file ) . readlines ( ) :
compat = compat . rstrip ( )
compats_seen . add ( compat )
2013-06-27 02:20:38 +08:00
# Add target overrides
if os . path . exists ( override_list_file ) :
for override in open ( override_list_file ) . readlines ( ) :
override = override . rstrip ( )
sources_seen . add ( override )
for src in open ( subdir_list_file ) . readlines ( ) :
src = src . rstrip ( )
if src not in sources_seen :
sources_seen . add ( src )
obj = os . path . join ( target , ' lib ' , src + obj_suffix + ' .bc ' )
objects . append ( obj )
2015-04-29 23:37:06 +08:00
src_path = libdir
if src in compats_seen :
src_path = os . path . join ( libdir , llvm_string_version )
src_file = os . path . join ( src_path , src )
2013-06-27 02:20:38 +08:00
ext = os . path . splitext ( src ) [ 1 ]
if ext == ' .ll ' :
b . build ( obj , ' LLVM_AS ' , src_file )
else :
b . build ( obj , clang_bc_rule , src_file )
2015-04-21 02:49:50 +08:00
obj = os . path . join ( ' generic-- ' , ' lib ' , ' subnormal_use_default.bc ' )
if not options . enable_runtime_subnormal :
objects . append ( obj )
2013-06-27 02:20:38 +08:00
builtins_link_bc = os . path . join ( target , ' lib ' , ' builtins.link ' + obj_suffix + ' .bc ' )
builtins_opt_bc = os . path . join ( target , ' lib ' , ' builtins.opt ' + obj_suffix + ' .bc ' )
builtins_bc = os . path . join ( ' built_libs ' , full_target_name + ' .bc ' )
b . build ( builtins_link_bc , " LLVM_LINK " , objects )
b . build ( builtins_opt_bc , " OPT " , builtins_link_bc )
b . build ( builtins_bc , " PREPARE_BUILTINS " , builtins_opt_bc , prepare_builtins )
install_files_bc . append ( ( builtins_bc , builtins_bc ) )
install_deps . append ( builtins_bc )
for alias in device [ ' aliases ' ] :
2014-01-30 04:03:27 +08:00
# Ninja cannot have multiple rules with same name so append suffix
ruleName = " CREATE_ALIAS_ {0} _for_ {1} " . format ( alias , device [ ' gpu ' ] )
b . rule ( ruleName , " ln -fs %s $out " % os . path . basename ( builtins_bc )
2013-06-27 02:20:38 +08:00
, " CREATE-ALIAS $out " )
alias_file = os . path . join ( ' built_libs ' , alias + ' - ' + target + ' .bc ' )
2014-01-30 04:03:27 +08:00
b . build ( alias_file , ruleName , builtins_bc )
2013-06-27 02:20:38 +08:00
install_files_bc . append ( ( alias_file , alias_file ) )
install_deps . append ( alias_file )
b . default ( builtins_bc )
2012-06-02 01:29:59 +08:00
2013-06-27 02:20:03 +08:00
2014-01-30 04:03:26 +08:00
install_cmd = ' && ' . join ( [ ' mkdir -p $ {DESTDIR} / %(dst)s && cp -r %(src)s $ {DESTDIR} / %(dst)s ' %
2013-06-27 02:20:03 +08:00
{ ' src ' : file ,
' dst ' : libexecdir }
for ( file , dest ) in install_files_bc ] )
2014-01-30 04:03:26 +08:00
install_cmd = ' && ' . join ( [ ' %(old)s && mkdir -p $ {DESTDIR} / %(dst)s && cp -r %(srcdir)s /generic/include/clc $ {DESTDIR} / %(dst)s ' %
2013-06-27 02:20:03 +08:00
{ ' old ' : install_cmd ,
2013-06-27 02:20:35 +08:00
' dst ' : includedir ,
' srcdir ' : srcdir } ] )
2014-01-30 04:03:26 +08:00
install_cmd = ' && ' . join ( [ ' %(old)s && mkdir -p $ {DESTDIR} / %(dst)s && cp -r libclc.pc $ {DESTDIR} / %(dst)s ' %
2013-06-27 02:20:03 +08:00
{ ' old ' : install_cmd ,
' dst ' : pkgconfigdir } ] )
b . rule ( ' install ' , command = install_cmd , description = ' INSTALL ' )
b . build ( ' install ' , ' install ' , install_deps )
2012-01-09 06:09:58 +08:00
b . rule ( " configure " , command = ' ' . join ( sys . argv ) , description = ' CONFIGURE ' ,
generator = True )
b . build ( b . output_filename ( ) , ' configure ' , list ( manifest_deps ) )
b . finish ( )