2017-12-26 16:44:08 -08:00
|
|
|
const builtin = @import("builtin");
|
2017-12-11 20:34:59 -08:00
|
|
|
const std = @import("std");
|
|
|
|
const Builder = std.build.Builder;
|
2017-04-18 22:13:15 -07:00
|
|
|
const tests = @import("test/tests.zig");
|
2017-12-11 20:34:59 -08:00
|
|
|
const os = std.os;
|
|
|
|
const BufMap = std.BufMap;
|
|
|
|
const warn = std.debug.warn;
|
|
|
|
const mem = std.mem;
|
|
|
|
const ArrayList = std.ArrayList;
|
|
|
|
const Buffer = std.Buffer;
|
|
|
|
const io = std.io;
|
2017-04-18 22:13:15 -07:00
|
|
|
|
2018-05-31 07:56:59 -07:00
|
|
|
pub fn build(b: *Builder) !void {
|
2017-10-21 14:31:06 -07:00
|
|
|
const mode = b.standardReleaseOptions();
|
|
|
|
|
2017-11-07 00:22:27 -08:00
|
|
|
var docgen_exe = b.addExecutable("docgen", "doc/docgen.zig");
|
|
|
|
|
2018-01-16 22:50:35 -08:00
|
|
|
const rel_zig_exe = try os.path.relative(b.allocator, b.build_root, b.zig_exe);
|
2018-05-17 20:21:44 -07:00
|
|
|
var docgen_cmd = b.addCommand(null, b.env_map, [][]const u8{
|
2017-11-07 00:22:27 -08:00
|
|
|
docgen_exe.getOutputPath(),
|
2018-01-16 22:50:35 -08:00
|
|
|
rel_zig_exe,
|
2017-11-07 00:22:27 -08:00
|
|
|
"doc/langref.html.in",
|
2018-01-08 21:07:01 -08:00
|
|
|
os.path.join(b.allocator, b.cache_root, "langref.html") catch unreachable,
|
2017-11-07 00:22:27 -08: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 15:25:17 -08:00
|
|
|
const test_step = b.step("test", "Run all the tests");
|
|
|
|
|
2018-01-03 18:32:50 -08:00
|
|
|
// find the stage0 build artifacts because we're going to re-use config.h and zig_cpp library
|
2018-05-17 20:21:44 -07:00
|
|
|
const build_info = try b.exec([][]const u8{
|
|
|
|
b.zig_exe,
|
|
|
|
"BUILD_INFO",
|
|
|
|
});
|
2018-01-03 18:32:50 -08:00
|
|
|
var index: usize = 0;
|
2018-07-10 17:18:43 -07:00
|
|
|
var ctx = Context{
|
|
|
|
.cmake_binary_dir = nextValue(&index, build_info),
|
|
|
|
.cxx_compiler = nextValue(&index, build_info),
|
|
|
|
.llvm_config_exe = nextValue(&index, build_info),
|
|
|
|
.lld_include_dir = nextValue(&index, build_info),
|
|
|
|
.lld_libraries = nextValue(&index, build_info),
|
|
|
|
.std_files = nextValue(&index, build_info),
|
|
|
|
.c_header_files = nextValue(&index, build_info),
|
|
|
|
.dia_guids_lib = nextValue(&index, build_info),
|
|
|
|
.llvm = undefined,
|
|
|
|
};
|
|
|
|
ctx.llvm = try findLLVM(b, ctx.llvm_config_exe);
|
2018-01-03 18:32:50 -08:00
|
|
|
|
2018-07-10 17:18:43 -07:00
|
|
|
var test_stage2 = b.addTest("src-self-hosted/test.zig");
|
|
|
|
test_stage2.setBuildMode(builtin.Mode.Debug);
|
2018-01-03 18:32:50 -08:00
|
|
|
|
|
|
|
var exe = b.addExecutable("zig", "src-self-hosted/main.zig");
|
|
|
|
exe.setBuildMode(mode);
|
2018-03-31 13:34:55 -07:00
|
|
|
|
2018-07-10 17:18:43 -07:00
|
|
|
try configureStage2(b, test_stage2, ctx);
|
|
|
|
try configureStage2(b, exe, ctx);
|
2018-01-03 15:25:17 -08:00
|
|
|
|
2018-01-03 18:32:50 -08:00
|
|
|
b.default_step.dependOn(&exe.step);
|
2018-01-04 20:30:03 -08:00
|
|
|
|
2018-07-11 11:09:05 -07:00
|
|
|
const skip_release = b.option(bool, "skip-release", "Main test suite skips release builds") orelse false;
|
2018-06-09 22:13:51 -07:00
|
|
|
const skip_self_hosted = b.option(bool, "skip-self-hosted", "Main test suite skips building self hosted compiler") orelse false;
|
2018-01-04 20:30:03 -08:00
|
|
|
if (!skip_self_hosted) {
|
|
|
|
test_step.dependOn(&exe.step);
|
|
|
|
}
|
2018-06-09 22:13:51 -07:00
|
|
|
const verbose_link_exe = b.option(bool, "verbose-link", "Print link command for self hosted compiler") orelse false;
|
2018-01-04 20:43:46 -08:00
|
|
|
exe.setVerboseLink(verbose_link_exe);
|
2017-10-21 14:31:06 -07:00
|
|
|
|
2018-01-03 18:32:50 -08:00
|
|
|
b.installArtifact(exe);
|
2018-07-10 17:18:43 -07:00
|
|
|
installStdLib(b, ctx.std_files);
|
|
|
|
installCHeaders(b, ctx.c_header_files);
|
2017-10-21 14:31:06 -07:00
|
|
|
|
2017-04-18 22:13:15 -07:00
|
|
|
const test_filter = b.option([]const u8, "test-filter", "Skip tests that do not match filter");
|
|
|
|
|
2018-07-10 17:18:43 -07:00
|
|
|
const test_stage2_step = b.step("test-stage2", "Run the stage2 compiler tests");
|
|
|
|
test_stage2_step.dependOn(&test_stage2.step);
|
|
|
|
test_step.dependOn(test_stage2_step);
|
|
|
|
|
2018-07-11 11:09:05 -07:00
|
|
|
const all_modes = []builtin.Mode{
|
|
|
|
builtin.Mode.Debug,
|
|
|
|
builtin.Mode.ReleaseSafe,
|
|
|
|
builtin.Mode.ReleaseFast,
|
|
|
|
builtin.Mode.ReleaseSmall,
|
|
|
|
};
|
|
|
|
const modes = if (skip_release) []builtin.Mode{builtin.Mode.Debug} else all_modes;
|
2017-11-07 00:22:27 -08:00
|
|
|
|
2018-07-11 11:09:05 -07:00
|
|
|
test_step.dependOn(tests.addPkgTests(b, test_filter, "test/behavior.zig", "behavior", "Run the behavior tests", modes));
|
2017-04-19 01:12:22 -07:00
|
|
|
|
2018-07-11 11:09:05 -07:00
|
|
|
test_step.dependOn(tests.addPkgTests(b, test_filter, "std/index.zig", "std", "Run the standard library tests", modes));
|
2017-04-19 23:26:36 -07:00
|
|
|
|
2018-07-11 11:09:05 -07:00
|
|
|
test_step.dependOn(tests.addPkgTests(b, test_filter, "std/special/compiler_rt/index.zig", "compiler-rt", "Run the compiler_rt tests", modes));
|
2017-08-18 10:51:16 -07:00
|
|
|
|
2018-07-11 16:38:01 -07:00
|
|
|
test_step.dependOn(tests.addCompareOutputTests(b, test_filter, modes));
|
2018-07-18 01:28:14 -07:00
|
|
|
test_step.dependOn(tests.addBuildExampleTests(b, test_filter, modes));
|
2018-07-11 16:38:01 -07:00
|
|
|
test_step.dependOn(tests.addCompileErrorTests(b, test_filter, modes));
|
|
|
|
test_step.dependOn(tests.addAssembleAndLinkTests(b, test_filter, modes));
|
|
|
|
test_step.dependOn(tests.addRuntimeSafetyTests(b, test_filter, modes));
|
2017-11-24 11:56:05 -08:00
|
|
|
test_step.dependOn(tests.addTranslateCTests(b, test_filter));
|
2018-01-22 19:24:07 -08:00
|
|
|
test_step.dependOn(tests.addGenHTests(b, test_filter));
|
2018-07-11 11:09:05 -07:00
|
|
|
test_step.dependOn(docs_step);
|
2017-04-18 22:13:15 -07:00
|
|
|
}
|
2017-12-11 20:34:59 -08:00
|
|
|
|
2018-07-10 17:18:43 -07:00
|
|
|
fn dependOnLib(lib_exe_obj: var, dep: *const LibraryDep) void {
|
2017-12-11 20:34:59 -08:00
|
|
|
for (dep.libdirs.toSliceConst()) |lib_dir| {
|
|
|
|
lib_exe_obj.addLibPath(lib_dir);
|
|
|
|
}
|
2017-12-23 17:21:57 -08:00
|
|
|
for (dep.system_libs.toSliceConst()) |lib| {
|
2017-12-11 20:34:59 -08:00
|
|
|
lib_exe_obj.linkSystemLibrary(lib);
|
|
|
|
}
|
2017-12-23 17:21:57 -08:00
|
|
|
for (dep.libs.toSliceConst()) |lib| {
|
|
|
|
lib_exe_obj.addObjectFile(lib);
|
|
|
|
}
|
2017-12-11 20:34:59 -08:00
|
|
|
for (dep.includes.toSliceConst()) |include_path| {
|
|
|
|
lib_exe_obj.addIncludeDir(include_path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-10 17:18:43 -07:00
|
|
|
fn addCppLib(b: *Builder, lib_exe_obj: var, cmake_binary_dir: []const u8, lib_name: []const u8) void {
|
2018-01-03 01:55:16 -08:00
|
|
|
const lib_prefix = if (lib_exe_obj.target.isWindows()) "" else "lib";
|
2018-05-17 20:21:44 -07:00
|
|
|
lib_exe_obj.addObjectFile(os.path.join(b.allocator, cmake_binary_dir, "zig_cpp", b.fmt("{}{}{}", lib_prefix, lib_name, lib_exe_obj.target.libFileExt())) catch unreachable);
|
2017-12-26 16:44:08 -08:00
|
|
|
}
|
|
|
|
|
2017-12-11 20:34:59 -08:00
|
|
|
const LibraryDep = struct {
|
|
|
|
libdirs: ArrayList([]const u8),
|
|
|
|
libs: ArrayList([]const u8),
|
2017-12-23 17:21:57 -08:00
|
|
|
system_libs: ArrayList([]const u8),
|
2017-12-11 20:34:59 -08:00
|
|
|
includes: ArrayList([]const u8),
|
|
|
|
};
|
|
|
|
|
2018-05-31 07:56:59 -07:00
|
|
|
fn findLLVM(b: *Builder, llvm_config_exe: []const u8) !LibraryDep {
|
2018-05-17 20:21:44 -07:00
|
|
|
const libs_output = try b.exec([][]const u8{
|
|
|
|
llvm_config_exe,
|
|
|
|
"--libs",
|
|
|
|
"--system-libs",
|
|
|
|
});
|
|
|
|
const includes_output = try b.exec([][]const u8{
|
|
|
|
llvm_config_exe,
|
|
|
|
"--includedir",
|
|
|
|
});
|
|
|
|
const libdir_output = try b.exec([][]const u8{
|
|
|
|
llvm_config_exe,
|
|
|
|
"--libdir",
|
|
|
|
});
|
2017-12-11 20:34:59 -08:00
|
|
|
|
2018-05-17 20:21:44 -07:00
|
|
|
var result = LibraryDep{
|
2017-12-11 20:34:59 -08:00
|
|
|
.libs = ArrayList([]const u8).init(b.allocator),
|
2017-12-23 17:21:57 -08:00
|
|
|
.system_libs = ArrayList([]const u8).init(b.allocator),
|
2017-12-11 20:34:59 -08:00
|
|
|
.includes = ArrayList([]const u8).init(b.allocator),
|
|
|
|
.libdirs = ArrayList([]const u8).init(b.allocator),
|
|
|
|
};
|
|
|
|
{
|
2018-01-03 01:55:16 -08:00
|
|
|
var it = mem.split(libs_output, " \r\n");
|
2017-12-11 20:34:59 -08:00
|
|
|
while (it.next()) |lib_arg| {
|
|
|
|
if (mem.startsWith(u8, lib_arg, "-l")) {
|
2018-01-08 21:07:01 -08:00
|
|
|
try result.system_libs.append(lib_arg[2..]);
|
2017-12-23 17:21:57 -08:00
|
|
|
} else {
|
2017-12-23 18:19:48 -08:00
|
|
|
if (os.path.isAbsolute(lib_arg)) {
|
2018-01-08 21:07:01 -08:00
|
|
|
try result.libs.append(lib_arg);
|
2017-12-23 18:19:48 -08:00
|
|
|
} else {
|
2018-01-08 21:07:01 -08:00
|
|
|
try result.system_libs.append(lib_arg);
|
2017-12-23 18:19:48 -08:00
|
|
|
}
|
2017-12-11 20:34:59 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
{
|
2018-01-03 01:55:16 -08:00
|
|
|
var it = mem.split(includes_output, " \r\n");
|
2017-12-11 20:34:59 -08:00
|
|
|
while (it.next()) |include_arg| {
|
|
|
|
if (mem.startsWith(u8, include_arg, "-I")) {
|
2018-01-08 21:07:01 -08:00
|
|
|
try result.includes.append(include_arg[2..]);
|
2017-12-11 20:34:59 -08:00
|
|
|
} else {
|
2018-01-08 21:07:01 -08:00
|
|
|
try result.includes.append(include_arg);
|
2017-12-11 20:34:59 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
{
|
2018-01-03 01:55:16 -08:00
|
|
|
var it = mem.split(libdir_output, " \r\n");
|
2017-12-11 20:34:59 -08:00
|
|
|
while (it.next()) |libdir| {
|
|
|
|
if (mem.startsWith(u8, libdir, "-L")) {
|
2018-01-08 21:07:01 -08:00
|
|
|
try result.libdirs.append(libdir[2..]);
|
2017-12-11 20:34:59 -08:00
|
|
|
} else {
|
2018-01-08 21:07:01 -08:00
|
|
|
try result.libdirs.append(libdir);
|
2017-12-11 20:34:59 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2017-12-22 21:29:39 -08:00
|
|
|
|
2018-05-31 07:56:59 -07:00
|
|
|
pub fn installStdLib(b: *Builder, stdlib_files: []const u8) void {
|
2018-01-03 16:39:04 -08:00
|
|
|
var it = mem.split(stdlib_files, ";");
|
|
|
|
while (it.next()) |stdlib_file| {
|
2018-01-08 21:07:01 -08:00
|
|
|
const src_path = os.path.join(b.allocator, "std", stdlib_file) catch unreachable;
|
|
|
|
const dest_path = os.path.join(b.allocator, "lib", "zig", "std", stdlib_file) catch unreachable;
|
2017-12-22 21:29:39 -08:00
|
|
|
b.installFile(src_path, dest_path);
|
|
|
|
}
|
|
|
|
}
|
2018-01-03 01:55:16 -08:00
|
|
|
|
2018-05-31 07:56:59 -07:00
|
|
|
pub fn installCHeaders(b: *Builder, c_header_files: []const u8) void {
|
2018-01-03 16:39:04 -08:00
|
|
|
var it = mem.split(c_header_files, ";");
|
|
|
|
while (it.next()) |c_header_file| {
|
2018-01-08 21:07:01 -08:00
|
|
|
const src_path = os.path.join(b.allocator, "c_headers", c_header_file) catch unreachable;
|
|
|
|
const dest_path = os.path.join(b.allocator, "lib", "zig", "include", c_header_file) catch unreachable;
|
2018-01-03 16:39:04 -08:00
|
|
|
b.installFile(src_path, dest_path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-31 07:56:59 -07:00
|
|
|
fn nextValue(index: *usize, build_info: []const u8) []const u8 {
|
2018-05-17 20:21:44 -07:00
|
|
|
const start = index.*;
|
|
|
|
while (true) : (index.* += 1) {
|
|
|
|
switch (build_info[index.*]) {
|
2018-01-03 19:38:13 -08:00
|
|
|
'\n' => {
|
2018-05-17 20:21:44 -07:00
|
|
|
const result = build_info[start..index.*];
|
|
|
|
index.* += 1;
|
2018-01-03 19:38:13 -08:00
|
|
|
return result;
|
|
|
|
},
|
|
|
|
'\r' => {
|
2018-05-17 20:21:44 -07:00
|
|
|
const result = build_info[start..index.*];
|
|
|
|
index.* += 2;
|
2018-01-03 19:38:13 -08:00
|
|
|
return result;
|
|
|
|
},
|
|
|
|
else => continue,
|
|
|
|
}
|
|
|
|
}
|
2018-01-03 01:55:16 -08:00
|
|
|
}
|
2018-07-10 17:18:43 -07:00
|
|
|
|
|
|
|
fn configureStage2(b: *Builder, exe: var, ctx: Context) !void {
|
|
|
|
// This is for finding /lib/libz.a on alpine linux.
|
|
|
|
// TODO turn this into -Dextra-lib-path=/lib option
|
|
|
|
exe.addLibPath("/lib");
|
|
|
|
|
|
|
|
exe.addIncludeDir("src");
|
|
|
|
exe.addIncludeDir(ctx.cmake_binary_dir);
|
|
|
|
addCppLib(b, exe, ctx.cmake_binary_dir, "zig_cpp");
|
|
|
|
if (ctx.lld_include_dir.len != 0) {
|
|
|
|
exe.addIncludeDir(ctx.lld_include_dir);
|
|
|
|
var it = mem.split(ctx.lld_libraries, ";");
|
|
|
|
while (it.next()) |lib| {
|
|
|
|
exe.addObjectFile(lib);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
addCppLib(b, exe, ctx.cmake_binary_dir, "embedded_lld_wasm");
|
|
|
|
addCppLib(b, exe, ctx.cmake_binary_dir, "embedded_lld_elf");
|
|
|
|
addCppLib(b, exe, ctx.cmake_binary_dir, "embedded_lld_coff");
|
|
|
|
addCppLib(b, exe, ctx.cmake_binary_dir, "embedded_lld_lib");
|
|
|
|
}
|
|
|
|
dependOnLib(exe, ctx.llvm);
|
|
|
|
|
|
|
|
if (exe.target.getOs() == builtin.Os.linux) {
|
|
|
|
const libstdcxx_path_padded = try b.exec([][]const u8{
|
|
|
|
ctx.cxx_compiler,
|
|
|
|
"-print-file-name=libstdc++.a",
|
|
|
|
});
|
|
|
|
const libstdcxx_path = mem.split(libstdcxx_path_padded, "\r\n").next().?;
|
|
|
|
if (mem.eql(u8, libstdcxx_path, "libstdc++.a")) {
|
|
|
|
warn(
|
|
|
|
\\Unable to determine path to libstdc++.a
|
|
|
|
\\On Fedora, install libstdc++-static and try again.
|
|
|
|
\\
|
|
|
|
);
|
|
|
|
return error.RequiredLibraryNotFound;
|
|
|
|
}
|
|
|
|
exe.addObjectFile(libstdcxx_path);
|
|
|
|
|
|
|
|
exe.linkSystemLibrary("pthread");
|
|
|
|
} else if (exe.target.isDarwin()) {
|
|
|
|
exe.linkSystemLibrary("c++");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx.dia_guids_lib.len != 0) {
|
|
|
|
exe.addObjectFile(ctx.dia_guids_lib);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (exe.target.getOs() != builtin.Os.windows) {
|
|
|
|
exe.linkSystemLibrary("xml2");
|
|
|
|
}
|
|
|
|
exe.linkSystemLibrary("c");
|
|
|
|
}
|
|
|
|
|
|
|
|
const Context = struct {
|
|
|
|
cmake_binary_dir: []const u8,
|
|
|
|
cxx_compiler: []const u8,
|
|
|
|
llvm_config_exe: []const u8,
|
|
|
|
lld_include_dir: []const u8,
|
|
|
|
lld_libraries: []const u8,
|
|
|
|
std_files: []const u8,
|
|
|
|
c_header_files: []const u8,
|
|
|
|
dia_guids_lib: []const u8,
|
|
|
|
llvm: LibraryDep,
|
|
|
|
};
|