Skip to content

Commit a147f06

Browse files
committed
zig puts temporary object files in zig-cache folder
See #298
1 parent 458afb0 commit a147f06

12 files changed

+124
-17
lines changed

src/all_types.hpp

+2
Original file line numberDiff line numberDiff line change
@@ -1486,6 +1486,8 @@ struct CodeGen {
14861486
Buf *test_name_prefix;
14871487

14881488
ZigList<TimeEvent> timing_events;
1489+
1490+
Buf *cache_dir;
14891491
};
14901492

14911493
enum VarLinkage {

src/codegen.cpp

+14-4
Original file line numberDiff line numberDiff line change
@@ -204,6 +204,10 @@ void codegen_set_out_name(CodeGen *g, Buf *out_name) {
204204
g->root_out_name = out_name;
205205
}
206206

207+
void codegen_set_cache_dir(CodeGen *g, Buf *cache_dir) {
208+
g->cache_dir = cache_dir;
209+
}
210+
207211
void codegen_set_libc_lib_dir(CodeGen *g, Buf *libc_lib_dir) {
208212
g->libc_lib_dir = libc_lib_dir;
209213
}
@@ -3910,16 +3914,22 @@ static void do_code_gen(CodeGen *g) {
39103914
codegen_add_time_event(g, "LLVM Emit Object");
39113915

39123916
char *err_msg = nullptr;
3913-
Buf *out_file_o = buf_create_from_buf(g->root_out_name);
3917+
Buf *o_basename = buf_create_from_buf(g->root_out_name);
39143918
const char *o_ext = target_o_file_ext(&g->zig_target);
3915-
buf_append_str(out_file_o, o_ext);
3916-
if (ZigLLVMTargetMachineEmitToFile(g->target_machine, g->module, buf_ptr(out_file_o),
3919+
buf_append_str(o_basename, o_ext);
3920+
Buf *output_path = buf_alloc();
3921+
os_path_join(g->cache_dir, o_basename, output_path);
3922+
int err;
3923+
if ((err = os_make_path(g->cache_dir))) {
3924+
zig_panic("unable to make cache dir: %s", err_str(err));
3925+
}
3926+
if (ZigLLVMTargetMachineEmitToFile(g->target_machine, g->module, buf_ptr(output_path),
39173927
LLVMObjectFile, &err_msg, !g->is_release_build))
39183928
{
39193929
zig_panic("unable to write object file: %s", err_msg);
39203930
}
39213931

3922-
g->link_objects.append(out_file_o);
3932+
g->link_objects.append(output_path);
39233933
}
39243934

39253935
static const uint8_t int_sizes_in_bits[] = {

src/codegen.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,7 @@ void codegen_set_omit_zigrt(CodeGen *g, bool omit_zigrt);
4747
void codegen_set_test_filter(CodeGen *g, Buf *filter);
4848
void codegen_set_test_name_prefix(CodeGen *g, Buf *prefix);
4949
void codegen_set_lib_version(CodeGen *g, size_t major, size_t minor, size_t patch);
50+
void codegen_set_cache_dir(CodeGen *g, Buf *cache_dir);
5051
void codegen_add_time_event(CodeGen *g, const char *name);
5152
void codegen_print_timing_report(CodeGen *g, FILE *f);
5253
void codegen_build(CodeGen *g);

src/error.cpp

+2
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,8 @@ const char *err_str(int err) {
2121
case ErrorFileTooBig: return "file too big";
2222
case ErrorDivByZero: return "division by zero";
2323
case ErrorOverflow: return "overflow";
24+
case ErrorPathAlreadyExists: return "path already exists";
25+
case ErrorUnexpected: return "unexpected error";
2426
}
2527
return "(invalid error)";
2628
}

src/error.hpp

+3-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,9 @@ enum Error {
2020
ErrorFileSystem,
2121
ErrorFileTooBig,
2222
ErrorDivByZero,
23-
ErrorOverflow
23+
ErrorOverflow,
24+
ErrorPathAlreadyExists,
25+
ErrorUnexpected,
2426
};
2527

2628
const char *err_str(int err);

src/link.cpp

+7-3
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,8 @@ static Buf *build_o(CodeGen *parent_gen, const char *oname) {
4848
codegen_set_omit_zigrt(child_gen, true);
4949
child_gen->want_h_file = false;
5050

51+
codegen_set_cache_dir(child_gen, parent_gen->cache_dir);
52+
5153
codegen_set_is_release(child_gen, parent_gen->is_release_build);
5254

5355
codegen_set_strip(child_gen, parent_gen->strip_debug_symbols);
@@ -64,10 +66,12 @@ static Buf *build_o(CodeGen *parent_gen, const char *oname) {
6466

6567
codegen_build(child_gen);
6668
const char *o_ext = target_o_file_ext(&child_gen->zig_target);
67-
Buf *o_out = buf_sprintf("%s%s", oname, o_ext);
68-
codegen_link(child_gen, buf_ptr(o_out));
69+
Buf *o_out_name = buf_sprintf("%s%s", oname, o_ext);
70+
Buf *output_path = buf_alloc();
71+
os_path_join(parent_gen->cache_dir, o_out_name, output_path);
72+
codegen_link(child_gen, buf_ptr(output_path));
6973

70-
return o_out;
74+
return output_path;
7175
}
7276

7377
static const char *get_exe_file_extension(CodeGen *g) {

src/main.cpp

+23-3
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ static int usage(const char *arg0) {
2929
" version print version number and exit\n"
3030
"Compile Options:\n"
3131
" --assembly [source] add assembly file to build\n"
32+
" --cache-dir [path] override the cache directory\n"
3233
" --color [auto|off|on] enable or disable colored error messages\n"
3334
" --enable-timing-info print timing diagnostics\n"
3435
" --libc-include-dir [path] directory where libc stdlib.h resides\n"
@@ -162,6 +163,7 @@ int main(int argc, char **argv) {
162163
size_t ver_minor = 0;
163164
size_t ver_patch = 0;
164165
bool timing_info = false;
166+
const char *cache_dir = "zig-cache";
165167

166168
if (argc >= 2 && strcmp(argv[1], "build") == 0) {
167169
const char *zig_exe_path = arg0;
@@ -180,6 +182,7 @@ int main(int argc, char **argv) {
180182
ZigList<const char *> args = {0};
181183
args.append(zig_exe_path);
182184
args.append(NULL); // placeholder
185+
args.append(NULL); // placeholder
183186
for (int i = 2; i < argc; i += 1) {
184187
if (strcmp(argv[i], "--debug-build-verbose") == 0) {
185188
verbose = true;
@@ -189,6 +192,9 @@ int main(int argc, char **argv) {
189192
} else if (i + 1 < argc && strcmp(argv[i], "--build-file") == 0) {
190193
build_file = argv[i + 1];
191194
i += 1;
195+
} else if (i + 1 < argc && strcmp(argv[i], "--cache-dir") == 0) {
196+
cache_dir = argv[i + 1];
197+
i += 1;
192198
} else {
193199
args.append(argv[i]);
194200
}
@@ -205,7 +211,14 @@ int main(int argc, char **argv) {
205211
Buf build_file_dirname = BUF_INIT;
206212
os_path_split(&build_file_abs, &build_file_dirname, &build_file_basename);
207213

214+
Buf *full_cache_dir = buf_alloc();
215+
os_path_resolve(buf_create_from_str("."), buf_create_from_str(cache_dir), full_cache_dir);
216+
Buf *path_to_build_exe = buf_alloc();
217+
os_path_join(full_cache_dir, buf_create_from_str("build"), path_to_build_exe);
218+
codegen_set_cache_dir(g, full_cache_dir);
219+
208220
args.items[1] = buf_ptr(&build_file_dirname);
221+
args.items[2] = buf_ptr(full_cache_dir);
209222

210223
bool build_file_exists;
211224
if ((err = os_file_exists(&build_file_abs, &build_file_exists))) {
@@ -221,6 +234,7 @@ int main(int argc, char **argv) {
221234
"General Options:\n"
222235
" --help Print this help and exit\n"
223236
" --build-file [file] Override path to build.zig\n"
237+
" --cache-dir [path] Override path to cache directory\n"
224238
" --verbose Print commands before executing them\n"
225239
" --debug-build-verbose Print verbose debugging information for the build system itself\n"
226240
" --prefix [prefix] Override default install prefix\n"
@@ -245,13 +259,13 @@ int main(int argc, char **argv) {
245259
build_pkg->package_table.put(buf_create_from_str("std"), g->std_package);
246260
g->root_package->package_table.put(buf_create_from_str("@build"), build_pkg);
247261
codegen_build(g);
248-
codegen_link(g, "build");
262+
codegen_link(g, buf_ptr(path_to_build_exe));
249263

250264
Termination term;
251-
os_spawn_process("./build", args, &term);
265+
os_spawn_process(buf_ptr(path_to_build_exe), args, &term);
252266
if (term.how != TerminationIdClean || term.code != 0) {
253267
fprintf(stderr, "\nBuild failed. Use the following command to reproduce the failure:\n");
254-
fprintf(stderr, "./build");
268+
fprintf(stderr, "%s", buf_ptr(path_to_build_exe));
255269
for (size_t i = 0; i < args.length; i += 1) {
256270
fprintf(stderr, " %s", args.at(i));
257271
}
@@ -331,6 +345,8 @@ int main(int argc, char **argv) {
331345
objects.append(argv[i]);
332346
} else if (strcmp(arg, "--assembly") == 0) {
333347
asm_files.append(argv[i]);
348+
} else if (strcmp(arg, "--cache-dir") == 0) {
349+
cache_dir = argv[i];
334350
} else if (strcmp(arg, "--target-arch") == 0) {
335351
target_arch = argv[i];
336352
} else if (strcmp(arg, "--target-os") == 0) {
@@ -478,12 +494,16 @@ int main(int argc, char **argv) {
478494

479495
Buf *zig_root_source_file = (cmd == CmdParseH) ? nullptr : in_file_buf;
480496

497+
Buf *full_cache_dir = buf_alloc();
498+
os_path_resolve(buf_create_from_str("."), buf_create_from_str(cache_dir), full_cache_dir);
499+
481500
CodeGen *g = codegen_create(zig_root_source_file, target);
482501
codegen_set_out_name(g, buf_out_name);
483502
codegen_set_lib_version(g, ver_major, ver_minor, ver_patch);
484503
codegen_set_is_release(g, is_release_build);
485504
codegen_set_is_test(g, cmd == CmdTest);
486505
codegen_set_linker_script(g, linker_script);
506+
codegen_set_cache_dir(g, full_cache_dir);
487507
if (each_lib_rpath)
488508
codegen_set_each_lib_rpath(g, each_lib_rpath);
489509

src/os.cpp

+48
Original file line numberDiff line numberDiff line change
@@ -723,3 +723,51 @@ double os_get_time(void) {
723723
return seconds;
724724
#endif
725725
}
726+
727+
int os_make_path(Buf *path) {
728+
Buf *resolved_path = buf_alloc();
729+
os_path_resolve(buf_create_from_str("."), path, resolved_path);
730+
731+
size_t end_index = buf_len(resolved_path);
732+
int err;
733+
while (true) {
734+
if ((err = os_make_dir(buf_slice(resolved_path, 0, end_index)))) {
735+
if (err == ErrorPathAlreadyExists) {
736+
if (end_index == buf_len(resolved_path))
737+
return 0;
738+
} else if (err == ErrorFileNotFound) {
739+
// march end_index backward until next path component
740+
while (true) {
741+
end_index -= 1;
742+
if (buf_ptr(resolved_path)[end_index] == '/')
743+
break;
744+
}
745+
continue;
746+
} else {
747+
return err;
748+
}
749+
}
750+
if (end_index == buf_len(resolved_path))
751+
return 0;
752+
// march end_index forward until next path component
753+
while (true) {
754+
end_index += 1;
755+
if (end_index == buf_len(resolved_path) || buf_ptr(resolved_path)[end_index] == '/')
756+
break;
757+
}
758+
}
759+
return 0;
760+
}
761+
762+
int os_make_dir(Buf *path) {
763+
if (mkdir(buf_ptr(path), 0755) == -1) {
764+
if (errno == EEXIST)
765+
return ErrorPathAlreadyExists;
766+
if (errno == ENOENT)
767+
return ErrorFileNotFound;
768+
if (errno == EACCES)
769+
return ErrorAccess;
770+
return ErrorUnexpected;
771+
}
772+
return 0;
773+
}

src/os.hpp

+3
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,9 @@ int os_path_real(Buf *rel_path, Buf *out_abs_path);
4040
void os_path_resolve(Buf *ref_path, Buf *target_path, Buf *out_abs_path);
4141
bool os_path_is_absolute(Buf *path);
4242

43+
int os_make_path(Buf *path);
44+
int os_make_dir(Buf *path);
45+
4346
void os_write_file(Buf *full_path, Buf *contents);
4447
int os_copy_file(Buf *src_path, Buf *dest_path);
4548

std/build.zig

+8-4
Original file line numberDiff line numberDiff line change
@@ -37,9 +37,10 @@ pub const Builder = struct {
3737
top_level_steps: List(&TopLevelStep),
3838
prefix: []const u8,
3939
lib_dir: []const u8,
40-
out_dir: []u8,
40+
out_dir: []u8, // TODO get rid of this
4141
installed_files: List([]const u8),
4242
build_root: []const u8,
43+
cache_root: []const u8,
4344

4445
const UserInputOptionsMap = HashMap([]const u8, UserInputOption, mem.hash_slice_u8, mem.eql_slice_u8);
4546
const AvailableOptionsMap = HashMap([]const u8, AvailableOption, mem.hash_slice_u8, mem.eql_slice_u8);
@@ -75,10 +76,13 @@ pub const Builder = struct {
7576
description: []const u8,
7677
};
7778

78-
pub fn init(allocator: &Allocator, zig_exe: []const u8, build_root: []const u8) -> Builder {
79+
pub fn init(allocator: &Allocator, zig_exe: []const u8, build_root: []const u8,
80+
cache_root: []const u8) -> Builder
81+
{
7982
var self = Builder {
8083
.zig_exe = zig_exe,
8184
.build_root = build_root,
85+
.cache_root = cache_root,
8286
.verbose = false,
8387
.invalid_user_input = false,
8488
.allocator = allocator,
@@ -768,7 +772,7 @@ pub const LibExeObjStep = struct {
768772
explicit_out_path
769773
} else {
770774
// TODO make it so we always know where this will be
771-
%%os.path.join(self.builder.allocator, self.builder.out_dir,
775+
%%os.path.join(self.builder.allocator, self.builder.cache_root,
772776
self.builder.fmt("{}{}", obj.name, obj.target.oFileExt()))
773777
};
774778
%%self.object_files.append(path_to_obj);
@@ -1217,7 +1221,7 @@ pub const CExecutable = struct {
12171221
self.step.dependOn(&obj.step);
12181222

12191223
// TODO make it so we always know where this will be
1220-
%%self.object_files.append(%%os.path.join(self.builder.allocator, self.builder.out_dir,
1224+
%%self.object_files.append(%%os.path.join(self.builder.allocator, self.builder.cache_root,
12211225
self.builder.fmt("{}{}", obj.name, obj.target.oFileExt())));
12221226

12231227
// TODO should be some kind of isolated directory that only has this header in it

std/special/build_runner.zig

+12-1
Original file line numberDiff line numberDiff line change
@@ -32,13 +32,23 @@ pub fn main() -> %void {
3232
result
3333
};
3434

35+
const cache_root = {
36+
if (arg_i >= os.args.count()) {
37+
%%io.stderr.printf("Expected third argument to be cache root directory path\n");
38+
return error.InvalidArgs;
39+
}
40+
const result = os.args.at(arg_i);
41+
arg_i += 1;
42+
result
43+
};
44+
3545
// TODO use a more general purpose allocator here
3646
var inc_allocator = %%mem.IncrementingAllocator.init(10 * 1024 * 1024);
3747
defer inc_allocator.deinit();
3848

3949
const allocator = &inc_allocator.allocator;
4050

41-
var builder = Builder.init(allocator, zig_exe, build_root);
51+
var builder = Builder.init(allocator, zig_exe, build_root, cache_root);
4252
defer builder.deinit();
4353

4454
var targets = List([]const u8).init(allocator);
@@ -113,6 +123,7 @@ fn usage(builder: &Builder, already_ran_build: bool, out_stream: &io.OutStream)
113123
\\General Options:
114124
\\ --help Print this help and exit
115125
\\ --build-file [file] Override path to build.zig
126+
\\ --cache-dir [path] Override path to cache directory
116127
\\ --verbose Print commands before executing them
117128
\\ --debug-build-verbose Print verbose debugging information for the build system itself
118129
\\ --prefix [prefix] Override default install prefix

test/tests.zig

+1-1
Original file line numberDiff line numberDiff line change
@@ -675,7 +675,7 @@ pub const BuildExamplesContext = struct {
675675
%%zig_args.append("--verbose");
676676
}
677677

678-
const run_cmd = b.addCommand(b.out_dir, b.env_map, b.zig_exe, zig_args.toSliceConst());
678+
const run_cmd = b.addCommand(b.cache_root, b.env_map, b.zig_exe, zig_args.toSliceConst());
679679

680680
const log_step = b.addLog("PASS {}\n", annotated_case_name);
681681
log_step.step.dependOn(&run_cmd.step);

0 commit comments

Comments
 (0)