2016-02-27 21:06:46 -08:00
|
|
|
// This file is in a package which has the root source file exposed as "@root".
|
2017-03-22 08:26:30 -07:00
|
|
|
// It is included in the compilation unit when exporting an executable.
|
2015-12-10 14:34:38 -08:00
|
|
|
|
2016-02-27 21:06:46 -08:00
|
|
|
const root = @import("@root");
|
2017-02-06 00:10:32 -08:00
|
|
|
const std = @import("std");
|
2017-05-01 10:12:38 -07:00
|
|
|
const builtin = @import("builtin");
|
2016-02-15 22:30:05 -08:00
|
|
|
|
2017-04-03 15:11:57 -07:00
|
|
|
var argc_ptr: &usize = undefined;
|
2016-01-13 17:15:51 -08:00
|
|
|
|
2017-12-18 06:59:57 -08:00
|
|
|
comptime {
|
2017-12-18 23:39:43 -08:00
|
|
|
const strong_linkage = builtin.GlobalLinkage.Strong;
|
2017-12-18 06:59:57 -08:00
|
|
|
if (builtin.link_libc) {
|
2017-12-18 23:39:43 -08:00
|
|
|
@export("main", main, strong_linkage);
|
2018-01-07 01:43:08 -08:00
|
|
|
} else if (builtin.os == builtin.Os.zen) {
|
|
|
|
@export("main", zenMain, strong_linkage);
|
2017-12-18 06:59:57 -08:00
|
|
|
} else if (builtin.os == builtin.Os.windows) {
|
2017-12-18 23:39:43 -08:00
|
|
|
@export("WinMainCRTStartup", WinMainCRTStartup, strong_linkage);
|
2017-12-18 06:59:57 -08:00
|
|
|
} else {
|
2017-12-18 23:39:43 -08:00
|
|
|
@export("_start", _start, strong_linkage);
|
2017-03-22 23:59:58 -07:00
|
|
|
}
|
2017-12-18 06:59:57 -08:00
|
|
|
}
|
2016-09-27 23:33:32 -07:00
|
|
|
|
2018-01-07 01:43:08 -08:00
|
|
|
extern fn zenMain() -> noreturn {
|
2018-01-14 21:01:02 -08:00
|
|
|
std.os.posix.exit(callMain());
|
2018-01-07 01:43:08 -08:00
|
|
|
}
|
|
|
|
|
2017-12-18 06:59:57 -08:00
|
|
|
nakedcc fn _start() -> noreturn {
|
2017-05-01 10:12:38 -07:00
|
|
|
switch (builtin.arch) {
|
|
|
|
builtin.Arch.x86_64 => {
|
2017-04-30 08:28:11 -07:00
|
|
|
argc_ptr = asm("lea (%%rsp), %[argc]": [argc] "=r" (-> &usize));
|
2016-02-12 01:04:46 -08:00
|
|
|
},
|
2017-05-01 10:12:38 -07:00
|
|
|
builtin.Arch.i386 => {
|
2017-04-30 08:28:11 -07:00
|
|
|
argc_ptr = asm("lea (%%esp), %[argc]": [argc] "=r" (-> &usize));
|
2016-02-12 01:04:46 -08:00
|
|
|
},
|
2016-09-05 14:01:54 -07:00
|
|
|
else => @compileError("unsupported arch"),
|
2016-02-12 01:04:46 -08:00
|
|
|
}
|
2017-12-06 15:12:05 -08:00
|
|
|
// If LLVM inlines stack variables into _start, they will overwrite
|
|
|
|
// the command line argument data.
|
|
|
|
@noInlineCall(posixCallMainAndExit);
|
2017-06-13 21:04:34 -07:00
|
|
|
}
|
|
|
|
|
2017-12-18 06:59:57 -08:00
|
|
|
extern fn WinMainCRTStartup() -> noreturn {
|
2017-10-02 19:00:42 -07:00
|
|
|
@setAlignStack(16);
|
2017-10-02 21:27:14 -07:00
|
|
|
|
2018-01-14 21:01:02 -08:00
|
|
|
std.os.windows.ExitProcess(callMain());
|
2016-01-15 23:07:20 -08:00
|
|
|
}
|
2016-01-13 17:15:51 -08:00
|
|
|
|
2017-06-13 21:04:34 -07:00
|
|
|
fn posixCallMainAndExit() -> noreturn {
|
2017-04-03 15:11:57 -07:00
|
|
|
const argc = *argc_ptr;
|
2017-04-21 07:39:13 -07:00
|
|
|
const argv = @ptrCast(&&u8, &argc_ptr[1]);
|
|
|
|
const envp = @ptrCast(&?&u8, &argv[argc + 1]);
|
2018-01-14 21:01:02 -08:00
|
|
|
std.os.posix.exit(callMainWithArgs(argc, argv, envp));
|
2017-04-03 15:11:57 -07:00
|
|
|
}
|
|
|
|
|
2018-01-14 21:01:02 -08:00
|
|
|
fn callMainWithArgs(argc: usize, argv: &&u8, envp: &?&u8) -> u8 {
|
2017-10-11 07:16:13 -07:00
|
|
|
std.os.ArgIteratorPosix.raw = argv[0..argc];
|
2017-04-02 16:14:23 -07:00
|
|
|
|
|
|
|
var env_count: usize = 0;
|
2017-05-03 15:12:07 -07:00
|
|
|
while (envp[env_count] != null) : (env_count += 1) {}
|
2017-10-14 22:23:10 -07:00
|
|
|
std.os.posix_environ_raw = @ptrCast(&&u8, envp)[0..env_count];
|
2017-04-02 16:14:23 -07:00
|
|
|
|
2018-01-14 21:01:02 -08:00
|
|
|
return callMain();
|
2016-02-27 21:06:46 -08:00
|
|
|
}
|
|
|
|
|
2017-12-18 06:59:57 -08:00
|
|
|
extern fn main(c_argc: i32, c_argv: &&u8, c_envp: &?&u8) -> i32 {
|
2018-01-14 21:01:02 -08:00
|
|
|
return callMainWithArgs(usize(c_argc), c_argv, c_envp);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn callMain() -> u8 {
|
|
|
|
switch (@typeId(@typeOf(root.main).ReturnType)) {
|
|
|
|
builtin.TypeId.NoReturn => {
|
|
|
|
root.main();
|
|
|
|
},
|
|
|
|
builtin.TypeId.Void => {
|
|
|
|
root.main();
|
|
|
|
return 0;
|
|
|
|
},
|
|
|
|
builtin.TypeId.Int => {
|
|
|
|
if (@typeOf(root.main).ReturnType.bit_count != 8) {
|
|
|
|
@compileError("expected return type of main to be 'u8', 'noreturn', 'void', or '%void'");
|
|
|
|
}
|
|
|
|
return root.main();
|
|
|
|
},
|
|
|
|
builtin.TypeId.ErrorUnion => {
|
|
|
|
root.main() catch |err| {
|
|
|
|
std.debug.warn("error: {}\n", @errorName(err));
|
|
|
|
if (@errorReturnTrace()) |trace| {
|
|
|
|
std.debug.dumpStackTrace(trace);
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
};
|
|
|
|
return 0;
|
|
|
|
},
|
|
|
|
else => @compileError("expected return type of main to be 'u8', 'noreturn', 'void', or '%void'"),
|
|
|
|
}
|
2016-02-15 22:30:05 -08:00
|
|
|
}
|