zig/std/special/bootstrap.zig

110 lines
3.6 KiB
Zig
Raw Normal View History

// This file is in a package which has the root source file exposed as "@root".
// It is included in the compilation unit when exporting an executable.
2015-12-10 14:34:38 -08:00
const root = @import("@root");
const std = @import("std");
const builtin = @import("builtin");
var argc_ptr: &usize = undefined;
2016-01-13 17:15:51 -08:00
2017-12-18 06:59:57 -08:00
comptime {
const strong_linkage = builtin.GlobalLinkage.Strong;
2017-12-18 06:59:57 -08:00
if (builtin.link_libc) {
@export("main", main, strong_linkage);
2017-12-18 06:59:57 -08:00
} else if (builtin.os == builtin.Os.windows) {
@export("WinMainCRTStartup", WinMainCRTStartup, strong_linkage);
2018-03-10 16:59:28 -08:00
} else if (builtin.os == builtin.Os.zen) {
@export("_start", zen_start, strong_linkage);
2017-12-18 06:59:57 -08:00
} else {
@export("_start", _start, strong_linkage);
}
2017-12-18 06:59:57 -08:00
}
2018-03-10 16:59:28 -08:00
extern fn zen_start() noreturn {
std.os.posix.exit(@inlineCall(callMain));
2018-01-07 01:43:08 -08:00
}
nakedcc fn _start() noreturn {
switch (builtin.arch) {
builtin.Arch.x86_64 => {
argc_ptr = asm ("lea (%%rsp), %[argc]" : [argc] "=r" (-> &usize));
},
builtin.Arch.i386 => {
argc_ptr = asm ("lea (%%esp), %[argc]" : [argc] "=r" (-> &usize));
},
else => @compileError("unsupported arch"),
}
// If LLVM inlines stack variables into _start, they will overwrite
// the command line argument data.
@noInlineCall(posixCallMainAndExit);
}
extern fn WinMainCRTStartup() noreturn {
2017-10-02 19:00:42 -07:00
@setAlignStack(16);
std.os.windows.ExitProcess(callMain());
}
2016-01-13 17:15:51 -08:00
fn posixCallMainAndExit() noreturn {
const argc = argc_ptr.*;
const argv = @ptrCast(&&u8, &argc_ptr[1]);
const envp_nullable = @ptrCast(&?&u8, &argv[argc + 1]);
var envp_count: usize = 0;
while (envp_nullable[envp_count]) |_| : (envp_count += 1) {}
const envp = @ptrCast(&&u8, envp_nullable)[0..envp_count];
if (builtin.os == builtin.Os.linux) {
const auxv = &@ptrCast(&usize, envp.ptr)[envp_count + 1];
var i: usize = 0;
while (auxv[i] != 0) : (i += 2) {
if (auxv[i] < std.os.linux_aux_raw.len) std.os.linux_aux_raw[auxv[i]] = auxv[i + 1];
}
std.debug.assert(std.os.linux_aux_raw[std.elf.AT_PAGESZ] == std.os.page_size);
}
std.os.posix.exit(callMainWithArgs(argc, argv, envp));
}
fn callMainWithArgs(argc: usize, argv: &&u8, envp: []&u8) u8 {
std.os.ArgIteratorPosix.raw = argv[0..argc];
std.os.posix_environ_raw = envp;
return callMain();
}
extern fn main(c_argc: i32, c_argv: &&u8, c_envp: &?&u8) i32 {
var env_count: usize = 0;
while (c_envp[env_count] != null) : (env_count += 1) {}
const envp = @ptrCast(&&u8, c_envp)[0..env_count];
return callMainWithArgs(usize(c_argc), c_argv, 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) {
2018-02-07 23:08:45 -08:00
@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));
2018-03-20 13:09:30 -07:00
if (builtin.os != builtin.Os.zen) {
if (@errorReturnTrace()) |trace| {
std.debug.dumpStackTrace(trace);
}
}
return 1;
};
return 0;
},
2018-02-07 23:08:45 -08:00
else => @compileError("expected return type of main to be 'u8', 'noreturn', 'void', or '!void'"),
}
}