|
| 1 | +// TODO: A lot of this file is very likely wrong. |
| 2 | + |
| 3 | +const builtin = @import("builtin"); |
| 4 | +const std = @import("../../std.zig"); |
| 5 | +const SYS = std.os.linux.SYS; |
| 6 | + |
| 7 | +pub fn syscall0(number: SYS) u32 { |
| 8 | + return asm volatile ("syscall" |
| 9 | + : [ret] "={rax}" (-> u32), |
| 10 | + : [number] "{rax}" (@intFromEnum(number)), |
| 11 | + : .{ .rcx = true, .r11 = true, .memory = true }); |
| 12 | +} |
| 13 | + |
| 14 | +pub fn syscall1(number: SYS, arg1: u32) u32 { |
| 15 | + return asm volatile ("syscall" |
| 16 | + : [ret] "={rax}" (-> u32), |
| 17 | + : [number] "{rax}" (@intFromEnum(number)), |
| 18 | + [arg1] "{rdi}" (arg1), |
| 19 | + : .{ .rcx = true, .r11 = true, .memory = true }); |
| 20 | +} |
| 21 | + |
| 22 | +pub fn syscall2(number: SYS, arg1: u32, arg2: u32) u32 { |
| 23 | + return asm volatile ("syscall" |
| 24 | + : [ret] "={rax}" (-> u32), |
| 25 | + : [number] "{rax}" (@intFromEnum(number)), |
| 26 | + [arg1] "{rdi}" (arg1), |
| 27 | + [arg2] "{rsi}" (arg2), |
| 28 | + : .{ .rcx = true, .r11 = true, .memory = true }); |
| 29 | +} |
| 30 | + |
| 31 | +pub fn syscall3(number: SYS, arg1: u32, arg2: u32, arg3: u32) u32 { |
| 32 | + return asm volatile ("syscall" |
| 33 | + : [ret] "={rax}" (-> u32), |
| 34 | + : [number] "{rax}" (@intFromEnum(number)), |
| 35 | + [arg1] "{rdi}" (arg1), |
| 36 | + [arg2] "{rsi}" (arg2), |
| 37 | + [arg3] "{rdx}" (arg3), |
| 38 | + : .{ .rcx = true, .r11 = true, .memory = true }); |
| 39 | +} |
| 40 | + |
| 41 | +pub fn syscall4(number: SYS, arg1: u32, arg2: u32, arg3: u32, arg4: u32) u32 { |
| 42 | + return asm volatile ("syscall" |
| 43 | + : [ret] "={rax}" (-> u32), |
| 44 | + : [number] "{rax}" (@intFromEnum(number)), |
| 45 | + [arg1] "{rdi}" (arg1), |
| 46 | + [arg2] "{rsi}" (arg2), |
| 47 | + [arg3] "{rdx}" (arg3), |
| 48 | + [arg4] "{r10}" (arg4), |
| 49 | + : .{ .rcx = true, .r11 = true, .memory = true }); |
| 50 | +} |
| 51 | + |
| 52 | +pub fn syscall5(number: SYS, arg1: u32, arg2: u32, arg3: u32, arg4: u32, arg5: u32) u32 { |
| 53 | + return asm volatile ("syscall" |
| 54 | + : [ret] "={rax}" (-> u32), |
| 55 | + : [number] "{rax}" (@intFromEnum(number)), |
| 56 | + [arg1] "{rdi}" (arg1), |
| 57 | + [arg2] "{rsi}" (arg2), |
| 58 | + [arg3] "{rdx}" (arg3), |
| 59 | + [arg4] "{r10}" (arg4), |
| 60 | + [arg5] "{r8}" (arg5), |
| 61 | + : .{ .rcx = true, .r11 = true, .memory = true }); |
| 62 | +} |
| 63 | + |
| 64 | +pub fn syscall6( |
| 65 | + number: SYS, |
| 66 | + arg1: u32, |
| 67 | + arg2: u32, |
| 68 | + arg3: u32, |
| 69 | + arg4: u32, |
| 70 | + arg5: u32, |
| 71 | + arg6: u32, |
| 72 | +) u32 { |
| 73 | + return asm volatile ("syscall" |
| 74 | + : [ret] "={rax}" (-> u32), |
| 75 | + : [number] "{rax}" (@intFromEnum(number)), |
| 76 | + [arg1] "{rdi}" (arg1), |
| 77 | + [arg2] "{rsi}" (arg2), |
| 78 | + [arg3] "{rdx}" (arg3), |
| 79 | + [arg4] "{r10}" (arg4), |
| 80 | + [arg5] "{r8}" (arg5), |
| 81 | + [arg6] "{r9}" (arg6), |
| 82 | + : .{ .rcx = true, .r11 = true, .memory = true }); |
| 83 | +} |
| 84 | + |
| 85 | +pub fn clone() callconv(.naked) u32 { |
| 86 | + asm volatile ( |
| 87 | + \\ movl $56,%%eax // SYS_clone |
| 88 | + \\ movq %%rdi,%%r11 |
| 89 | + \\ movq %%rdx,%%rdi |
| 90 | + \\ movq %%r8,%%rdx |
| 91 | + \\ movq %%r9,%%r8 |
| 92 | + \\ movq 8(%%rsp),%%r10 |
| 93 | + \\ movq %%r11,%%r9 |
| 94 | + \\ andq $-16,%%rsi |
| 95 | + \\ subq $8,%%rsi |
| 96 | + \\ movq %%rcx,(%%rsi) |
| 97 | + \\ syscall |
| 98 | + \\ testq %%rax,%%rax |
| 99 | + \\ jz 1f |
| 100 | + \\ retq |
| 101 | + \\ |
| 102 | + \\1: |
| 103 | + ); |
| 104 | + if (builtin.unwind_tables != .none or !builtin.strip_debug_info) asm volatile ( |
| 105 | + \\ .cfi_undefined %%rip |
| 106 | + ); |
| 107 | + asm volatile ( |
| 108 | + \\ xorl %%ebp,%%ebp |
| 109 | + \\ |
| 110 | + \\ popq %%rdi |
| 111 | + \\ callq *%%r9 |
| 112 | + \\ movl %%eax,%%edi |
| 113 | + \\ movl $60,%%eax // SYS_exit |
| 114 | + \\ syscall |
| 115 | + \\ |
| 116 | + ); |
| 117 | +} |
| 118 | + |
| 119 | +pub const restore = restore_rt; |
| 120 | + |
| 121 | +pub fn restore_rt() callconv(.naked) noreturn { |
| 122 | + switch (builtin.zig_backend) { |
| 123 | + .stage2_c => asm volatile ( |
| 124 | + \\ movl %[number], %%eax |
| 125 | + \\ syscall |
| 126 | + : |
| 127 | + : [number] "i" (@intFromEnum(SYS.rt_sigreturn)), |
| 128 | + ), |
| 129 | + else => asm volatile ( |
| 130 | + \\ syscall |
| 131 | + : |
| 132 | + : [number] "{rax}" (@intFromEnum(SYS.rt_sigreturn)), |
| 133 | + ), |
| 134 | + } |
| 135 | +} |
| 136 | + |
| 137 | +pub const mode_t = u32; |
| 138 | +pub const time_t = i32; |
| 139 | +pub const nlink_t = u32; |
| 140 | +pub const blksize_t = i32; |
| 141 | +pub const blkcnt_t = i32; |
| 142 | +pub const off_t = i64; |
| 143 | +pub const ino_t = u64; |
| 144 | +pub const dev_t = u64; |
| 145 | + |
| 146 | +pub const VDSO = struct { |
| 147 | + pub const CGT_SYM = "__vdso_clock_gettime"; |
| 148 | + pub const CGT_VER = "LINUX_2.6"; |
| 149 | + |
| 150 | + pub const GETCPU_SYM = "__vdso_getcpu"; |
| 151 | + pub const GETCPU_VER = "LINUX_2.6"; |
| 152 | +}; |
| 153 | + |
| 154 | +pub const ARCH = struct { |
| 155 | + pub const SET_GS = 0x1001; |
| 156 | + pub const SET_FS = 0x1002; |
| 157 | + pub const GET_FS = 0x1003; |
| 158 | + pub const GET_GS = 0x1004; |
| 159 | +}; |
| 160 | + |
| 161 | +// The `stat` definition used by the Linux kernel. |
| 162 | +pub const Stat = extern struct { |
| 163 | + dev: dev_t, |
| 164 | + ino: ino_t, |
| 165 | + nlink: nlink_t, |
| 166 | + |
| 167 | + mode: mode_t, |
| 168 | + uid: std.os.linux.uid_t, |
| 169 | + gid: std.os.linux.gid_t, |
| 170 | + __pad0: u32, |
| 171 | + rdev: dev_t, |
| 172 | + size: off_t, |
| 173 | + blksize: blksize_t, |
| 174 | + blocks: i64, |
| 175 | + |
| 176 | + atim: std.os.linux.timespec, |
| 177 | + mtim: std.os.linux.timespec, |
| 178 | + ctim: std.os.linux.timespec, |
| 179 | + __unused: [3]i32, |
| 180 | + |
| 181 | + pub fn atime(self: @This()) std.os.linux.timespec { |
| 182 | + return self.atim; |
| 183 | + } |
| 184 | + |
| 185 | + pub fn mtime(self: @This()) std.os.linux.timespec { |
| 186 | + return self.mtim; |
| 187 | + } |
| 188 | + |
| 189 | + pub fn ctime(self: @This()) std.os.linux.timespec { |
| 190 | + return self.ctim; |
| 191 | + } |
| 192 | +}; |
0 commit comments