2017-12-27 00:44:08 +00:00
const builtin = @import ( " builtin " ) ;
2017-12-12 04:34:59 +00:00
const std = @import ( " std " ) ;
const Builder = std . build . Builder ;
2017-04-19 05:13:15 +00:00
const tests = @import ( " test/tests.zig " ) ;
2017-12-12 04:34:59 +00:00
const BufMap = std . BufMap ;
const warn = std . debug . warn ;
const mem = std . mem ;
const ArrayList = std . ArrayList ;
const io = std . io ;
2019-05-26 17:17:34 +00:00
const fs = std . fs ;
2019-07-14 07:06:20 +00:00
const InstallDirectoryOptions = std . build . InstallDirectoryOptions ;
2020-09-04 03:23:00 +00:00
const assert = std . debug . assert ;
2017-04-19 05:13:15 +00:00
2020-08-15 21:17:40 +00:00
const zig_version = std . builtin . Version { . major = 0 , . minor = 6 , . patch = 0 } ;
2018-05-31 14:56:59 +00:00
pub fn build ( b : * Builder ) ! void {
2019-07-04 21:43:56 +00:00
b . setPreferredReleaseMode ( . ReleaseFast ) ;
2017-10-21 21:31:06 +00:00
const mode = b . standardReleaseOptions ( ) ;
2020-07-21 18:18:51 +00:00
const target = b . standardTargetOptions ( . { } ) ;
2017-10-21 21:31:06 +00:00
2017-11-07 08:22:27 +00:00
var docgen_exe = b . addExecutable ( " docgen " , " doc/docgen.zig " ) ;
2019-05-26 17:17:34 +00:00
const rel_zig_exe = try fs . path . relative ( b . allocator , b . build_root , b . zig_exe ) ;
const langref_out_path = fs . path . join (
2019-02-07 05:42:41 +00:00
b . allocator ,
2019-11-27 08:30:39 +00:00
& [ _ ] [ ] const u8 { b . cache_root , " langref.html " } ,
2019-02-07 05:42:41 +00:00
) catch unreachable ;
breaking changes to zig build API and improved caching
* in Zig build scripts, getOutputPath() is no longer a valid function
to call, unless setOutputDir() was used, or within a custom make()
function. Instead there is more convenient API to use which takes
advantage of the caching system. Search this commit diff for
`exe.run()` for an example.
* Zig build by default enables caching. All build artifacts will go
into zig-cache. If you want to access build artifacts in a convenient
location, it is recommended to add an `install` step. Otherwise
you can use the `run()` API mentioned above to execute programs
directly from their location in the cache. Closes #330.
`addSystemCommand` is available for programs not built with Zig
build.
* Please note that Zig does no cache evicting yet. You may have to
manually delete zig-cache directories periodically to keep disk
usage down. It's planned for this to be a simple Least Recently
Used eviction system eventually.
* `--output`, `--output-lib`, and `--output-h` are removed. Instead,
use `--output-dir` which defaults to the current working directory.
Or take advantage of `--cache on`, which will print the main output
path to stdout, and the other artifacts will be in the same directory
with predictable file names. `--disable-gen-h` is available when
one wants to prevent .h file generation.
* `@cImport` is always independently cached now. Closes #2015.
It always writes the generated Zig code to disk which makes debug
info and compile errors better. No more "TODO: remember C source
location to display here"
* Fix .d file parsing. (Fixes the MacOS CI failure)
* Zig no longer creates "temporary files" other than inside a
zig-cache directory.
This breaks the CLI API that Godbolt uses. The suggested new invocation
can be found in this commit diff, in the changes to `test/cli.zig`.
2019-03-09 03:53:35 +00:00
var docgen_cmd = docgen_exe . run ( ) ;
2019-11-27 08:30:39 +00:00
docgen_cmd . addArgs ( & [ _ ] [ ] const u8 {
2018-01-17 06:50:35 +00:00
rel_zig_exe ,
2019-05-26 17:17:34 +00:00
" doc " + + fs . path . sep_str + + " langref.html.in " ,
2018-09-12 15:49:46 +00:00
langref_out_path ,
2017-11-07 08:22:27 +00:00
} ) ;
docgen_cmd . step . dependOn ( & docgen_exe . step ) ;
const docs_step = b . step ( " docs " , " Build documentation " ) ;
docs_step . dependOn ( & docgen_cmd . step ) ;
2018-01-03 23:25:17 +00:00
const test_step = b . step ( " test " , " Run all the tests " ) ;
2018-07-11 00:18:43 +00:00
var test_stage2 = b . addTest ( " src-self-hosted/test.zig " ) ;
2020-08-20 20:16:04 +00:00
test_stage2 . setBuildMode ( mode ) ;
2019-11-23 20:56:05 +00:00
test_stage2 . addPackagePath ( " stage2_tests " , " test/stage2/test.zig " ) ;
2018-01-04 02:32:50 +00:00
2019-11-27 08:30:39 +00:00
const fmt_build_zig = b . addFmt ( & [ _ ] [ ] const u8 { " build.zig " } ) ;
2019-02-26 23:10:40 +00:00
2018-07-11 18:09:05 +00:00
const skip_release = b . option ( bool , " skip-release " , " Main test suite skips release builds " ) orelse false ;
2018-09-12 18:50:26 +00:00
const skip_release_small = b . option ( bool , " skip-release-small " , " Main test suite skips release-small builds " ) orelse skip_release ;
const skip_release_fast = b . option ( bool , " skip-release-fast " , " Main test suite skips release-fast builds " ) orelse skip_release ;
const skip_release_safe = b . option ( bool , " skip-release-safe " , " Main test suite skips release-safe builds " ) orelse skip_release ;
2019-05-15 01:21:59 +00:00
const skip_non_native = b . option ( bool , " skip-non-native " , " Main test suite skips non-native builds " ) orelse false ;
2019-09-22 03:55:56 +00:00
const skip_libc = b . option ( bool , " skip-libc " , " Main test suite skips tests that link libc " ) orelse false ;
2020-08-31 21:54:05 +00:00
const skip_compile_errors = b . option ( bool , " skip-compile-errors " , " Main test suite skips compile error tests " ) orelse false ;
2017-10-21 21:31:06 +00:00
2019-07-14 07:06:20 +00:00
const only_install_lib_files = b . option ( bool , " lib-files-only " , " Only install library files " ) orelse false ;
2020-05-15 19:20:42 +00:00
const enable_llvm = b . option ( bool , " enable-llvm " , " Build self-hosted compiler with LLVM backend enabled " ) orelse false ;
2020-07-04 01:15:01 +00:00
const config_h_path_option = b . option ( [ ] const u8 , " config_h " , " Path to the generated config.h " ) ;
2020-04-06 17:07:19 +00:00
2020-05-16 05:22:56 +00:00
if ( ! only_install_lib_files ) {
2020-07-01 21:56:30 +00:00
var exe = b . addExecutable ( " zig " , " src-self-hosted/main.zig " ) ;
2020-09-04 03:23:00 +00:00
exe . install ( ) ;
2020-07-01 21:56:30 +00:00
exe . setBuildMode ( mode ) ;
2020-07-21 18:18:51 +00:00
exe . setTarget ( target ) ;
2020-07-01 21:56:30 +00:00
test_step . dependOn ( & exe . step ) ;
b . default_step . dependOn ( & exe . step ) ;
2020-09-04 03:23:00 +00:00
exe . addBuildOption ( bool , " have_llvm " , enable_llvm ) ;
2020-07-01 21:56:30 +00:00
if ( enable_llvm ) {
2020-07-04 01:15:01 +00:00
const config_h_text = if ( config_h_path_option ) | config_h_path |
try std . fs . cwd ( ) . readFileAlloc ( b . allocator , toNativePathSep ( b , config_h_path ) , max_config_h_bytes )
else
try findAndReadConfigH ( b ) ;
2020-07-01 21:56:30 +00:00
var ctx = parseConfigH ( b , config_h_text ) ;
ctx . llvm = try findLLVM ( b , ctx . llvm_config_exe ) ;
try configureStage2 ( b , exe , ctx ) ;
}
const tracy = b . option ( [ ] const u8 , " tracy " , " Enable Tracy integration. Supply path to Tracy source " ) ;
2020-09-04 03:23:00 +00:00
const link_libc = b . option ( bool , " force-link-libc " , " Force self-hosted compiler to link libc " ) orelse enable_llvm ;
2020-08-18 22:08:43 +00:00
if ( link_libc ) {
exe . linkLibC ( ) ;
test_stage2 . linkLibC ( ) ;
}
2020-07-01 21:56:30 +00:00
2020-08-04 04:56:21 +00:00
const log_scopes = b . option ( [ ] const [ ] const u8 , " log " , " Which log scopes to enable " ) orelse & [ 0 ] [ ] const u8 { } ;
2020-08-26 04:31:03 +00:00
const zir_dumps = b . option ( [ ] const [ ] const u8 , " dump-zir " , " Which functions to dump ZIR for before codegen " ) orelse & [ 0 ] [ ] const u8 { } ;
2020-08-04 04:56:21 +00:00
2020-08-15 21:17:40 +00:00
const opt_version_string = b . option ( [ ] const u8 , " version-string " , " Override Zig version string. Default is to find out with git. " ) ;
const version = if ( opt_version_string ) | version | version else v : {
var code : u8 = undefined ;
const version_untrimmed = b . execAllowFail ( & [ _ ] [ ] const u8 {
" git " , " -C " , b . build_root , " name-rev " , " HEAD " ,
" --tags " , " --name-only " , " --no-undefined " , " --always " ,
} , & code , . Ignore ) catch | err | {
std . debug . print (
\\Unable to determine zig version string: {}
\\Provide the zig version string explicitly using the `version-string` build option.
, . { err } ) ;
std . process . exit ( 1 ) ;
} ;
const trimmed = mem . trim ( u8 , version_untrimmed , " \n \r " ) ;
break : v b . fmt ( " {}.{}.{}+{} " , . { zig_version . major , zig_version . minor , zig_version . patch , trimmed } ) ;
} ;
exe . addBuildOption ( [ ] const u8 , " version " , version ) ;
2020-08-04 04:56:21 +00:00
exe . addBuildOption ( [ ] const [ ] const u8 , " log_scopes " , log_scopes ) ;
2020-08-26 04:31:03 +00:00
exe . addBuildOption ( [ ] const [ ] const u8 , " zir_dumps " , zir_dumps ) ;
2020-07-01 21:56:30 +00:00
exe . addBuildOption ( bool , " enable_tracy " , tracy ! = null ) ;
if ( tracy ) | tracy_path | {
const client_cpp = fs . path . join (
b . allocator ,
& [ _ ] [ ] const u8 { tracy_path , " TracyClient.cpp " } ,
) catch unreachable ;
exe . addIncludeDir ( tracy_path ) ;
exe . addCSourceFile ( client_cpp , & [ _ ] [ ] const u8 { " -DTRACY_ENABLE=1 " , " -fno-sanitize=undefined " } ) ;
exe . linkSystemLibraryName ( " c++ " ) ;
exe . linkLibC ( ) ;
}
2020-06-18 23:03:41 +00:00
}
2019-07-14 07:06:20 +00:00
b . installDirectory ( InstallDirectoryOptions {
2019-07-15 21:54:50 +00:00
. source_dir = " lib " ,
2019-07-14 07:06:20 +00:00
. install_dir = . Lib ,
2019-07-15 21:54:50 +00:00
. install_subdir = " zig " ,
2020-09-07 22:15:48 +00:00
. exclude_extensions = & [ _ ] [ ] const u8 {
" test.zig " ,
" README.md " ,
" .z.0 " ,
" .z.9 " ,
" rfc1951.txt " ,
} ,
2019-07-14 07:06:20 +00:00
} ) ;
2017-10-21 21:31:06 +00:00
2017-04-19 05:13:15 +00:00
const test_filter = b . option ( [ ] const u8 , " test-filter " , " Skip tests that do not match filter " ) ;
2019-09-22 03:55:56 +00:00
const is_wine_enabled = b . option ( bool , " enable-wine " , " Use Wine to run cross compiled Windows tests " ) orelse false ;
const is_qemu_enabled = b . option ( bool , " enable-qemu " , " Use QEMU to run cross compiled foreign architecture tests " ) orelse false ;
2020-05-13 06:08:16 +00:00
const is_wasmtime_enabled = b . option ( bool , " enable-wasmtime " , " Use Wasmtime to enable and run WASI libstd tests " ) orelse false ;
2019-09-22 03:55:56 +00:00
const glibc_multi_dir = b . option ( [ ] const u8 , " enable-foreign-glibc " , " Provide directory with glibc installations to run cross compiled tests that link glibc " ) ;
2020-09-08 08:11:10 +00:00
test_stage2 . addBuildOption ( bool , " have_llvm " , enable_llvm ) ;
2020-08-04 22:32:16 +00:00
test_stage2 . addBuildOption ( bool , " enable_qemu " , is_qemu_enabled ) ;
test_stage2 . addBuildOption ( bool , " enable_wine " , is_wine_enabled ) ;
test_stage2 . addBuildOption ( bool , " enable_wasmtime " , is_wasmtime_enabled ) ;
test_stage2 . addBuildOption ( ? [ ] const u8 , " glibc_multi_install_dir " , glibc_multi_dir ) ;
2018-07-11 00:18:43 +00:00
const test_stage2_step = b . step ( " test-stage2 " , " Run the stage2 compiler tests " ) ;
test_stage2_step . dependOn ( & test_stage2 . step ) ;
2019-11-23 20:56:05 +00:00
test_step . dependOn ( test_stage2_step ) ;
2018-07-11 00:18:43 +00:00
2018-09-12 18:50:26 +00:00
var chosen_modes : [ 4 ] builtin . Mode = undefined ;
var chosen_mode_index : usize = 0 ;
chosen_modes [ chosen_mode_index ] = builtin . Mode . Debug ;
chosen_mode_index + = 1 ;
if ( ! skip_release_safe ) {
chosen_modes [ chosen_mode_index ] = builtin . Mode . ReleaseSafe ;
chosen_mode_index + = 1 ;
}
if ( ! skip_release_fast ) {
chosen_modes [ chosen_mode_index ] = builtin . Mode . ReleaseFast ;
chosen_mode_index + = 1 ;
}
if ( ! skip_release_small ) {
chosen_modes [ chosen_mode_index ] = builtin . Mode . ReleaseSmall ;
chosen_mode_index + = 1 ;
}
const modes = chosen_modes [ 0 . . chosen_mode_index ] ;
2017-11-07 08:22:27 +00:00
2019-02-26 23:10:40 +00:00
// run stage1 `zig fmt` on this build.zig file just to make sure it works
test_step . dependOn ( & fmt_build_zig . step ) ;
const fmt_step = b . step ( " test-fmt " , " Run zig fmt against build.zig to make sure it works " ) ;
fmt_step . dependOn ( & fmt_build_zig . step ) ;
Add/fix missing WASI functionality to pass libstd tests
This rather large commit adds/fixes missing WASI functionality
in `libstd` needed to pass the `libstd` tests. As such, now by
default tests targeting `wasm32-wasi` target are enabled in
`test/tests.zig` module. However, they can be disabled by passing
the `-Dskip-wasi=true` flag when invoking the `zig build test`
command. When the flag is set to `false`, i.e., when WASI tests are
included, `wasmtime` with `--dir=.` is used as the default testing
command.
Since the majority of `libstd` tests were relying on `fs.cwd()`
call to get current working directory handle wrapped in `Dir`
struct, in order to make the tests WASI-friendly, `fs.cwd()`
call was replaced with `testing.getTestDir()` function which
resolved to either `fs.cwd()` for non-WASI targets, or tries to
fetch the preopen list from the WASI runtime and extract a
preopen for '.' path.
The summary of changes introduced by this commit:
* implement `Dir.makeDir` and `Dir.openDir` targeting WASI
* implement `Dir.deleteFile` and `Dir.deleteDir` targeting WASI
* fix `os.close` and map errors in `unlinkat`
* move WASI-specific `mkdirat` and `unlinkat` from `std.fs.wasi`
to `std.os` module
* implement `lseek_{SET, CUR, END}` targeting WASI
* implement `futimens` targeting WASI
* implement `ftruncate` targeting WASI
* implement `readv`, `writev`, `pread{v}`, `pwrite{v}` targeting WASI
* make sure ANSI escape codes are _not_ used in stderr or stdout
in WASI, as WASI always sanitizes stderr, and sanitizes stdout if
fd is a TTY
* fix specifying WASI rights when opening/creating files/dirs
* tweak `AtomicFile` to be WASI-compatible
* implement `os.renameatWasi` for WASI-compliant `os.renameat` function
* implement sleep() targeting WASI
* fix `process.getEnvMap` targeting WASI
2020-05-05 15:23:49 +00:00
// TODO for the moment, skip wasm32-wasi until bugs are sorted out.
2020-05-18 15:45:06 +00:00
test_step . dependOn ( tests . addPkgTests ( b , test_filter , " test/stage1/behavior.zig " , " behavior " , " Run the behavior tests " , modes , false , skip_non_native , skip_libc , is_wine_enabled , is_qemu_enabled , is_wasmtime_enabled , glibc_multi_dir ) ) ;
2017-04-19 08:12:22 +00:00
2020-05-13 06:08:16 +00:00
test_step . dependOn ( tests . addPkgTests ( b , test_filter , " lib/std/std.zig " , " std " , " Run the standard library tests " , modes , false , skip_non_native , skip_libc , is_wine_enabled , is_qemu_enabled , is_wasmtime_enabled , glibc_multi_dir ) ) ;
2017-04-20 06:26:36 +00:00
2020-05-13 06:08:16 +00:00
test_step . dependOn ( tests . addPkgTests ( b , test_filter , " lib/std/special/compiler_rt.zig " , " compiler-rt " , " Run the compiler_rt tests " , modes , true , skip_non_native , true , is_wine_enabled , is_qemu_enabled , is_wasmtime_enabled , glibc_multi_dir ) ) ;
2017-08-18 17:51:16 +00:00
2018-07-11 23:38:01 +00:00
test_step . dependOn ( tests . addCompareOutputTests ( b , test_filter , modes ) ) ;
2019-07-16 16:15:46 +00:00
test_step . dependOn ( tests . addStandaloneTests ( b , test_filter , modes ) ) ;
2019-09-03 14:08:39 +00:00
test_step . dependOn ( tests . addStackTraceTests ( b , test_filter , modes ) ) ;
2020-06-21 03:50:28 +00:00
const test_cli = tests . addCliTests ( b , test_filter , modes ) ;
const test_cli_step = b . step ( " test-cli " , " Run zig cli tests " ) ;
test_cli_step . dependOn ( test_cli ) ;
test_step . dependOn ( test_cli ) ;
2018-07-11 23:38:01 +00:00
test_step . dependOn ( tests . addAssembleAndLinkTests ( b , test_filter , modes ) ) ;
test_step . dependOn ( tests . addRuntimeSafetyTests ( b , test_filter , modes ) ) ;
2017-11-24 19:56:05 +00:00
test_step . dependOn ( tests . addTranslateCTests ( b , test_filter ) ) ;
2020-01-03 03:45:48 +00:00
test_step . dependOn ( tests . addRunTranslatedCTests ( b , test_filter ) ) ;
2020-03-20 22:33:36 +00:00
// tests for this feature are disabled until we have the self-hosted compiler available
Add/fix missing WASI functionality to pass libstd tests
This rather large commit adds/fixes missing WASI functionality
in `libstd` needed to pass the `libstd` tests. As such, now by
default tests targeting `wasm32-wasi` target are enabled in
`test/tests.zig` module. However, they can be disabled by passing
the `-Dskip-wasi=true` flag when invoking the `zig build test`
command. When the flag is set to `false`, i.e., when WASI tests are
included, `wasmtime` with `--dir=.` is used as the default testing
command.
Since the majority of `libstd` tests were relying on `fs.cwd()`
call to get current working directory handle wrapped in `Dir`
struct, in order to make the tests WASI-friendly, `fs.cwd()`
call was replaced with `testing.getTestDir()` function which
resolved to either `fs.cwd()` for non-WASI targets, or tries to
fetch the preopen list from the WASI runtime and extract a
preopen for '.' path.
The summary of changes introduced by this commit:
* implement `Dir.makeDir` and `Dir.openDir` targeting WASI
* implement `Dir.deleteFile` and `Dir.deleteDir` targeting WASI
* fix `os.close` and map errors in `unlinkat`
* move WASI-specific `mkdirat` and `unlinkat` from `std.fs.wasi`
to `std.os` module
* implement `lseek_{SET, CUR, END}` targeting WASI
* implement `futimens` targeting WASI
* implement `ftruncate` targeting WASI
* implement `readv`, `writev`, `pread{v}`, `pwrite{v}` targeting WASI
* make sure ANSI escape codes are _not_ used in stderr or stdout
in WASI, as WASI always sanitizes stderr, and sanitizes stdout if
fd is a TTY
* fix specifying WASI rights when opening/creating files/dirs
* tweak `AtomicFile` to be WASI-compatible
* implement `os.renameatWasi` for WASI-compliant `os.renameat` function
* implement sleep() targeting WASI
* fix `process.getEnvMap` targeting WASI
2020-05-05 15:23:49 +00:00
// test_step.dependOn(tests.addGenHTests(b, test_filter));
2020-08-31 21:54:05 +00:00
if ( ! skip_compile_errors ) {
test_step . dependOn ( tests . addCompileErrorTests ( b , test_filter , modes ) ) ;
}
2018-07-11 18:09:05 +00:00
test_step . dependOn ( docs_step ) ;
2017-04-19 05:13:15 +00:00
}
2017-12-12 04:34:59 +00:00
2020-07-11 11:09:04 +00:00
fn dependOnLib ( b : * Builder , lib_exe_obj : anytype , dep : LibraryDep ) void {
2020-04-11 19:06:56 +00:00
for ( dep . libdirs . items ) | lib_dir | {
2017-12-12 04:34:59 +00:00
lib_exe_obj . addLibPath ( lib_dir ) ;
}
2019-05-26 17:17:34 +00:00
const lib_dir = fs . path . join (
2019-02-07 05:42:41 +00:00
b . allocator ,
2019-11-27 08:30:39 +00:00
& [ _ ] [ ] const u8 { dep . prefix , " lib " } ,
2019-02-07 05:42:41 +00:00
) catch unreachable ;
2020-04-11 19:06:56 +00:00
for ( dep . system_libs . items ) | lib | {
2018-11-02 04:07:43 +00:00
const static_bare_name = if ( mem . eql ( u8 , lib , " curses " ) )
2019-11-08 20:56:21 +00:00
@as ( [ ] const u8 , " libncurses.a " )
2018-11-02 04:07:43 +00:00
else
2019-12-09 03:53:51 +00:00
b . fmt ( " lib{}.a " , . { lib } ) ;
2019-05-26 17:17:34 +00:00
const static_lib_name = fs . path . join (
2019-02-07 05:42:41 +00:00
b . allocator ,
2019-11-27 08:30:39 +00:00
& [ _ ] [ ] const u8 { lib_dir , static_bare_name } ,
2019-02-07 05:42:41 +00:00
) catch unreachable ;
2018-11-02 04:07:43 +00:00
const have_static = fileExists ( static_lib_name ) catch unreachable ;
if ( have_static ) {
lib_exe_obj . addObjectFile ( static_lib_name ) ;
} else {
lib_exe_obj . linkSystemLibrary ( lib ) ;
}
2017-12-12 04:34:59 +00:00
}
2020-04-11 19:06:56 +00:00
for ( dep . libs . items ) | lib | {
2017-12-24 01:21:57 +00:00
lib_exe_obj . addObjectFile ( lib ) ;
}
2020-04-11 19:06:56 +00:00
for ( dep . includes . items ) | include_path | {
2017-12-12 04:34:59 +00:00
lib_exe_obj . addIncludeDir ( include_path ) ;
}
}
2018-11-02 04:07:43 +00:00
fn fileExists ( filename : [ ] const u8 ) ! bool {
2020-02-16 18:25:30 +00:00
fs . cwd ( ) . access ( filename , . { } ) catch | err | switch ( err ) {
2019-05-27 03:35:26 +00:00
error . FileNotFound = > return false ,
2018-11-02 04:07:43 +00:00
else = > return err ,
} ;
return true ;
}
2020-07-11 11:09:04 +00:00
fn addCppLib ( b : * Builder , lib_exe_obj : anytype , cmake_binary_dir : [ ] const u8 , lib_name : [ ] const u8 ) void {
2019-11-27 08:30:39 +00:00
lib_exe_obj . addObjectFile ( fs . path . join ( b . allocator , & [ _ ] [ ] const u8 {
2019-02-07 05:42:41 +00:00
cmake_binary_dir ,
" zig_cpp " ,
2019-12-09 03:53:51 +00:00
b . fmt ( " {}{}{} " , . { lib_exe_obj . target . libPrefix ( ) , lib_name , lib_exe_obj . target . staticLibSuffix ( ) } ) ,
2019-02-07 05:42:41 +00:00
} ) catch unreachable ) ;
2017-12-27 00:44:08 +00:00
}
2018-11-13 13:08:37 +00:00
const LibraryDep = struct {
2018-11-02 04:07:43 +00:00
prefix : [ ] const u8 ,
2017-12-12 04:34:59 +00:00
libdirs : ArrayList ( [ ] const u8 ) ,
libs : ArrayList ( [ ] const u8 ) ,
2017-12-24 01:21:57 +00:00
system_libs : ArrayList ( [ ] const u8 ) ,
2017-12-12 04:34:59 +00:00
includes : ArrayList ( [ ] const u8 ) ,
} ;
2018-05-31 14:56:59 +00:00
fn findLLVM ( b : * Builder , llvm_config_exe : [ ] const u8 ) ! LibraryDep {
2019-11-27 08:30:39 +00:00
const shared_mode = try b . exec ( & [ _ ] [ ] const u8 { llvm_config_exe , " --shared-mode " } ) ;
2018-11-02 04:07:43 +00:00
const is_static = mem . startsWith ( u8 , shared_mode , " static " ) ;
const libs_output = if ( is_static )
2019-11-27 08:30:39 +00:00
try b . exec ( & [ _ ] [ ] const u8 {
2018-11-02 04:07:43 +00:00
llvm_config_exe ,
" --libfiles " ,
" --system-libs " ,
} )
else
2019-11-27 08:30:39 +00:00
try b . exec ( & [ _ ] [ ] const u8 {
2018-11-02 04:07:43 +00:00
llvm_config_exe ,
" --libs " ,
} ) ;
2019-11-27 08:30:39 +00:00
const includes_output = try b . exec ( & [ _ ] [ ] const u8 { llvm_config_exe , " --includedir " } ) ;
const libdir_output = try b . exec ( & [ _ ] [ ] const u8 { llvm_config_exe , " --libdir " } ) ;
const prefix_output = try b . exec ( & [ _ ] [ ] const u8 { llvm_config_exe , " --prefix " } ) ;
2017-12-12 04:34:59 +00:00
2018-11-13 13:08:37 +00:00
var result = LibraryDep {
2019-02-04 20:24:06 +00:00
. prefix = mem . tokenize ( prefix_output , " \r \n " ) . next ( ) . ? ,
2017-12-12 04:34:59 +00:00
. libs = ArrayList ( [ ] const u8 ) . init ( b . allocator ) ,
2017-12-24 01:21:57 +00:00
. system_libs = ArrayList ( [ ] const u8 ) . init ( b . allocator ) ,
2017-12-12 04:34:59 +00:00
. includes = ArrayList ( [ ] const u8 ) . init ( b . allocator ) ,
. libdirs = ArrayList ( [ ] const u8 ) . init ( b . allocator ) ,
} ;
{
2019-02-04 20:24:06 +00:00
var it = mem . tokenize ( libs_output , " \r \n " ) ;
2017-12-12 04:34:59 +00:00
while ( it . next ( ) ) | lib_arg | {
if ( mem . startsWith ( u8 , lib_arg , " -l " ) ) {
2018-01-09 05:07:01 +00:00
try result . system_libs . append ( lib_arg [ 2 . . ] ) ;
2020-02-04 20:04:11 +00:00
} else {
2020-03-03 14:17:37 +00:00
if ( fs . path . isAbsolute ( lib_arg ) ) {
try result . libs . append ( lib_arg ) ;
} else {
2020-04-07 20:34:30 +00:00
var lib_arg_copy = lib_arg ;
2020-03-03 14:17:37 +00:00
if ( mem . endsWith ( u8 , lib_arg , " .lib " ) ) {
2020-04-07 20:34:30 +00:00
lib_arg_copy = lib_arg [ 0 . . lib_arg . len - 4 ] ;
2020-03-03 14:17:37 +00:00
}
2020-04-07 20:34:30 +00:00
try result . system_libs . append ( lib_arg_copy ) ;
2017-12-24 02:19:48 +00:00
}
2017-12-12 04:34:59 +00:00
}
}
}
{
2019-02-04 20:24:06 +00:00
var it = mem . tokenize ( includes_output , " \r \n " ) ;
2017-12-12 04:34:59 +00:00
while ( it . next ( ) ) | include_arg | {
if ( mem . startsWith ( u8 , include_arg , " -I " ) ) {
2018-01-09 05:07:01 +00:00
try result . includes . append ( include_arg [ 2 . . ] ) ;
2017-12-12 04:34:59 +00:00
} else {
2018-01-09 05:07:01 +00:00
try result . includes . append ( include_arg ) ;
2017-12-12 04:34:59 +00:00
}
}
}
{
2019-02-04 20:24:06 +00:00
var it = mem . tokenize ( libdir_output , " \r \n " ) ;
2017-12-12 04:34:59 +00:00
while ( it . next ( ) ) | libdir | {
if ( mem . startsWith ( u8 , libdir , " -L " ) ) {
2018-01-09 05:07:01 +00:00
try result . libdirs . append ( libdir [ 2 . . ] ) ;
2017-12-12 04:34:59 +00:00
} else {
2018-01-09 05:07:01 +00:00
try result . libdirs . append ( libdir ) ;
2017-12-12 04:34:59 +00:00
}
}
}
return result ;
}
2017-12-23 05:29:39 +00:00
2020-07-11 11:09:04 +00:00
fn configureStage2 ( b : * Builder , exe : anytype , ctx : Context ) ! void {
2018-07-11 00:18:43 +00:00
exe . addIncludeDir ( " src " ) ;
exe . addIncludeDir ( ctx . cmake_binary_dir ) ;
addCppLib ( b , exe , ctx . cmake_binary_dir , " zig_cpp " ) ;
2020-09-04 03:23:00 +00:00
assert ( ctx . lld_include_dir . len ! = 0 ) ;
exe . addIncludeDir ( ctx . lld_include_dir ) ;
{
2019-02-04 20:24:06 +00:00
var it = mem . tokenize ( ctx . lld_libraries , " ; " ) ;
2018-07-11 00:18:43 +00:00
while ( it . next ( ) ) | lib | {
exe . addObjectFile ( lib ) ;
}
}
2020-02-04 08:39:20 +00:00
{
var it = mem . tokenize ( ctx . clang_libraries , " ; " ) ;
while ( it . next ( ) ) | lib | {
exe . addObjectFile ( lib ) ;
}
}
2018-11-02 04:07:43 +00:00
dependOnLib ( b , exe , ctx . llvm ) ;
2018-07-11 00:18:43 +00:00
2020-09-04 03:23:00 +00:00
// Boy, it sure would be nice to simply linkSystemLibrary("c++") and rely on zig's
// ability to provide libc++ right? Well thanks to C++ not having a stable ABI this
// will cause linker errors. It would work in the situation when `zig cc` is used to
// build LLVM, Clang, and LLD, however when depending on them as system libraries, system
// libc++ must be used.
const cross_compile = false ; // TODO
if ( cross_compile ) {
// In this case we assume that zig cc was used to build the LLVM, Clang, LLD dependencies.
exe . linkSystemLibrary ( " c++ " ) ;
} else {
if ( exe . target . getOsTag ( ) = = . linux ) {
// First we try to static link against gcc libstdc++. If that doesn't work,
// we fall back to -lc++ and cross our fingers.
addCxxKnownPath ( b , ctx , exe , " libstdc++.a " , " " ) catch | err | switch ( err ) {
error . RequiredLibraryNotFound = > {
exe . linkSystemLibrary ( " c++ " ) ;
} ,
else = > | e | return e ,
} ;
2018-07-11 00:18:43 +00:00
2018-11-02 04:07:43 +00:00
exe . linkSystemLibrary ( " pthread " ) ;
2020-09-04 03:23:00 +00:00
} else if ( exe . target . isFreeBSD ( ) ) {
try addCxxKnownPath ( b , ctx , exe , " libc++.a " , null ) ;
exe . linkSystemLibrary ( " pthread " ) ;
} else if ( exe . target . isDarwin ( ) ) {
if ( addCxxKnownPath ( b , ctx , exe , " libgcc_eh.a " , " " ) ) {
// Compiler is GCC.
try addCxxKnownPath ( b , ctx , exe , " libstdc++.a " , null ) ;
exe . linkSystemLibrary ( " pthread " ) ;
// TODO LLD cannot perform this link.
// See https://github.com/ziglang/zig/issues/1535
exe . enableSystemLinkerHack ( ) ;
} else | err | switch ( err ) {
error . RequiredLibraryNotFound = > {
// System compiler, not gcc.
exe . linkSystemLibrary ( " c++ " ) ;
} ,
else = > | e | return e ,
}
2018-11-02 04:07:43 +00:00
}
2018-07-11 00:18:43 +00:00
2020-09-04 03:23:00 +00:00
if ( ctx . dia_guids_lib . len ! = 0 ) {
exe . addObjectFile ( ctx . dia_guids_lib ) ;
}
2018-07-11 00:18:43 +00:00
}
}
2018-11-02 04:07:43 +00:00
fn addCxxKnownPath (
b : * Builder ,
ctx : Context ,
2020-07-11 11:09:04 +00:00
exe : anytype ,
2018-11-02 04:07:43 +00:00
objname : [ ] const u8 ,
errtxt : ? [ ] const u8 ,
) ! void {
2019-11-27 08:30:39 +00:00
const path_padded = try b . exec ( & [ _ ] [ ] const u8 {
2018-11-02 04:07:43 +00:00
ctx . cxx_compiler ,
2019-12-09 03:53:51 +00:00
b . fmt ( " -print-file-name={} " , . { objname } ) ,
2018-11-02 04:07:43 +00:00
} ) ;
2019-02-04 20:24:06 +00:00
const path_unpadded = mem . tokenize ( path_padded , " \r \n " ) . next ( ) . ? ;
2018-11-02 04:07:43 +00:00
if ( mem . eql ( u8 , path_unpadded , objname ) ) {
if ( errtxt ) | msg | {
2019-12-09 03:53:51 +00:00
warn ( " {} " , . { msg } ) ;
2018-11-02 04:07:43 +00:00
} else {
2019-12-09 03:53:51 +00:00
warn ( " Unable to determine path to {} \n " , . { objname } ) ;
2018-11-02 04:07:43 +00:00
}
return error . RequiredLibraryNotFound ;
}
exe . addObjectFile ( path_unpadded ) ;
}
2018-11-13 13:08:37 +00:00
const Context = struct {
2018-07-11 00:18:43 +00:00
cmake_binary_dir : [ ] const u8 ,
cxx_compiler : [ ] const u8 ,
llvm_config_exe : [ ] const u8 ,
lld_include_dir : [ ] const u8 ,
lld_libraries : [ ] const u8 ,
2020-02-04 08:39:20 +00:00
clang_libraries : [ ] const u8 ,
2018-07-11 00:18:43 +00:00
dia_guids_lib : [ ] const u8 ,
llvm : LibraryDep ,
} ;
2020-04-04 15:57:28 +00:00
2020-04-06 17:07:19 +00:00
const max_config_h_bytes = 1 * 1024 * 1024 ;
fn findAndReadConfigH ( b : * Builder ) ! [ ] const u8 {
2020-04-04 15:57:28 +00:00
var check_dir = fs . path . dirname ( b . zig_exe ) . ? ;
2020-04-06 17:07:19 +00:00
while ( true ) {
2020-04-04 15:57:28 +00:00
var dir = try fs . cwd ( ) . openDir ( check_dir , . { } ) ;
defer dir . close ( ) ;
2020-04-06 17:07:19 +00:00
const config_h_text = dir . readFileAlloc ( b . allocator , " config.h " , max_config_h_bytes ) catch | err | switch ( err ) {
2020-04-04 15:57:28 +00:00
error . FileNotFound = > {
2020-04-04 18:05:49 +00:00
const new_check_dir = fs . path . dirname ( check_dir ) ;
if ( new_check_dir = = null or mem . eql ( u8 , new_check_dir . ? , check_dir ) ) {
2020-04-04 15:57:28 +00:00
std . debug . warn ( " Unable to find config.h file relative to Zig executable. \n " , . { } ) ;
std . debug . warn ( " `zig build` must be run using a Zig executable within the source tree. \n " , . { } ) ;
std . process . exit ( 1 ) ;
2020-04-04 18:05:49 +00:00
}
check_dir = new_check_dir . ? ;
2020-04-04 15:57:28 +00:00
continue ;
} ,
else = > | e | return e ,
} ;
2020-04-06 17:07:19 +00:00
return config_h_text ;
2020-04-04 15:57:28 +00:00
} else unreachable ; // TODO should not need `else unreachable`.
2020-04-06 17:07:19 +00:00
}
2020-04-04 15:57:28 +00:00
2020-04-06 22:35:49 +00:00
fn parseConfigH ( b : * Builder , config_h_text : [ ] const u8 ) Context {
2020-04-04 15:57:28 +00:00
var ctx : Context = . {
. cmake_binary_dir = undefined ,
. cxx_compiler = undefined ,
. llvm_config_exe = undefined ,
. lld_include_dir = undefined ,
. lld_libraries = undefined ,
. clang_libraries = undefined ,
. dia_guids_lib = undefined ,
. llvm = undefined ,
} ;
const mappings = [ _ ] struct { prefix : [ ] const u8 , field : [ ] const u8 } {
. {
. prefix = " #define ZIG_CMAKE_BINARY_DIR " ,
. field = " cmake_binary_dir " ,
} ,
. {
. prefix = " #define ZIG_CXX_COMPILER " ,
. field = " cxx_compiler " ,
} ,
. {
. prefix = " #define ZIG_LLD_INCLUDE_PATH " ,
. field = " lld_include_dir " ,
} ,
. {
. prefix = " #define ZIG_LLD_LIBRARIES " ,
. field = " lld_libraries " ,
} ,
. {
. prefix = " #define ZIG_CLANG_LIBRARIES " ,
. field = " clang_libraries " ,
} ,
. {
. prefix = " #define ZIG_LLVM_CONFIG_EXE " ,
. field = " llvm_config_exe " ,
} ,
. {
. prefix = " #define ZIG_DIA_GUIDS_LIB " ,
. field = " dia_guids_lib " ,
} ,
} ;
var lines_it = mem . tokenize ( config_h_text , " \r \n " ) ;
while ( lines_it . next ( ) ) | line | {
inline for ( mappings ) | mapping | {
if ( mem . startsWith ( u8 , line , mapping . prefix ) ) {
2020-04-04 17:15:08 +00:00
var it = mem . split ( line , " \" " ) ;
2020-04-04 15:57:28 +00:00
_ = it . next ( ) . ? ; // skip the stuff before the quote
2020-04-06 22:35:49 +00:00
const quoted = it . next ( ) . ? ; // the stuff inside the quote
@field ( ctx , mapping . field ) = toNativePathSep ( b , quoted ) ;
2020-04-04 15:57:28 +00:00
}
}
}
return ctx ;
}
2020-04-06 22:35:49 +00:00
fn toNativePathSep ( b : * Builder , s : [ ] const u8 ) [ ] u8 {
const duplicated = mem . dupe ( b . allocator , u8 , s ) catch unreachable ;
for ( duplicated ) | * byte | switch ( byte . * ) {
'/' = > byte . * = fs . path . sep ,
else = > { } ,
} ;
return duplicated ;
}