rust/src/comp/back/x86.rs

370 lines
11 KiB
Rust
Raw Normal View History

import lib::llvm::llvm;
import lib::llvm::llvm::ModuleRef;
import std::_str;
import std::_vec;
import std::os::target_os;
import util::common::istr;
const int wordsz = 4;
fn wstr(int i) -> str {
ret istr(i * wordsz);
}
fn start() -> vec[str] {
ret [".cfi_startproc"];
}
fn end() -> vec[str] {
ret [".cfi_endproc"];
}
fn save_callee_saves() -> vec[str] {
ret ["pushl %ebp",
"pushl %edi",
"pushl %esi",
"pushl %ebx"];
}
fn save_callee_saves_with_cfi() -> vec[str] {
auto offset = 8;
auto t;
t = ["pushl %ebp"];
t += [".cfi_def_cfa_offset " + istr(offset)];
t += [".cfi_offset %ebp, -" + istr(offset)];
t += ["pushl %edi"];
offset += 4;
t += [".cfi_def_cfa_offset " + istr(offset)];
t += ["pushl %esi"];
offset += 4;
t += [".cfi_def_cfa_offset " + istr(offset)];
t += ["pushl %ebx"];
offset += 4;
t += [".cfi_def_cfa_offset " + istr(offset)];
ret t;
}
fn restore_callee_saves() -> vec[str] {
ret ["popl %ebx",
"popl %esi",
"popl %edi",
"popl %ebp"];
}
2011-03-09 14:05:22 -06:00
fn load_esp_from_rust_sp_first_arg() -> vec[str] {
ret ["movl " + wstr(abi::task_field_rust_sp) + "(%ecx), %esp"];
}
2011-03-09 14:05:22 -06:00
fn load_esp_from_runtime_sp_first_arg() -> vec[str] {
ret ["movl " + wstr(abi::task_field_runtime_sp) + "(%ecx), %esp"];
}
2011-03-09 14:05:22 -06:00
fn store_esp_to_rust_sp_first_arg() -> vec[str] {
ret ["movl %esp, " + wstr(abi::task_field_rust_sp) + "(%ecx)"];
}
2011-03-09 14:05:22 -06:00
fn store_esp_to_runtime_sp_first_arg() -> vec[str] {
ret ["movl %esp, " + wstr(abi::task_field_runtime_sp) + "(%ecx)"];
}
2011-03-09 14:05:22 -06:00
fn load_esp_from_rust_sp_second_arg() -> vec[str] {
ret ["movl " + wstr(abi::task_field_rust_sp) + "(%edx), %esp"];
2011-03-09 14:05:22 -06:00
}
fn load_esp_from_runtime_sp_second_arg() -> vec[str] {
ret ["movl " + wstr(abi::task_field_runtime_sp) + "(%edx), %esp"];
2011-03-09 14:05:22 -06:00
}
fn store_esp_to_rust_sp_second_arg() -> vec[str] {
ret ["movl %esp, " + wstr(abi::task_field_rust_sp) + "(%edx)"];
2011-03-09 14:05:22 -06:00
}
fn store_esp_to_runtime_sp_second_arg() -> vec[str] {
ret ["movl %esp, " + wstr(abi::task_field_runtime_sp) + "(%edx)"];
2011-03-09 14:05:22 -06:00
}
/*
* This is a bit of glue-code. It should be emitted once per
* compilation unit.
*
* - save regs on C stack
* - align sp on a 16-byte boundary
* - save sp to task.runtime_sp (runtime_sp is thus always aligned)
* - load saved task sp (switch stack)
* - restore saved task regs
* - return to saved task pc
*
* Our incoming stack looks like this:
*
* *esp+4 = [arg1 ] = task ptr
* *esp = [retpc ]
*/
fn rust_activate_glue() -> vec[str] {
ret ["movl 4(%esp), %ecx # ecx = rust_task"]
+ save_callee_saves()
2011-03-09 14:05:22 -06:00
+ store_esp_to_runtime_sp_first_arg()
+ load_esp_from_rust_sp_first_arg()
/*
* There are two paths we can arrive at this code from:
*
*
* 1. We are activating a task for the first time. When we switch
* into the task stack and 'ret' to its first instruction, we'll
* start doing whatever the first instruction says. Probably
* saving registers and starting to establish a frame. Harmless
* stuff, doesn't look at task->rust_sp again except when it
* clobbers it during a later native call.
*
*
* 2. We are resuming a task that was descheduled by the yield glue
* below. When we switch into the task stack and 'ret', we'll be
* ret'ing to a very particular instruction:
*
* "esp <- task->rust_sp"
*
* this is the first instruction we 'ret' to after this glue,
* because it is the first instruction following *any* native
* call, and the task we are activating was descheduled
* mid-native-call.
*
* Unfortunately for us, we have already restored esp from
* task->rust_sp and are about to eat the 5 words off the top of
* it.
*
*
* | ... | <-- where esp will be once we restore + ret, below,
* | retpc | and where we'd *like* task->rust_sp to wind up.
* | ebp |
* | edi |
* | esi |
* | ebx | <-- current task->rust_sp == current esp
*
*
* This is a problem. If we return to "esp <- task->rust_sp" it
* will push esp back down by 5 words. This manifests as a rust
* stack that grows by 5 words on each yield/reactivate. Not
* good.
*
* So what we do here is just adjust task->rust_sp up 5 words as
* well, to mirror the movement in esp we're about to
* perform. That way the "esp <- task->rust_sp" we 'ret' to below
* will be a no-op. Esp won't move, and the task's stack won't
* grow.
*/
+ ["addl $20, " + wstr(abi::task_field_rust_sp) + "(%ecx)"]
/*
* In most cases, the function we're returning to (activating)
* will have saved any caller-saves before it yielded via native call,
* so no work to do here. With one exception: when we're initially
* activating, the task needs to be in the fastcall 2nd parameter
* expected by the rust main function. That's edx.
*/
+ ["mov %ecx, %edx"]
+ restore_callee_saves()
+ ["ret"];
}
/* More glue code, this time the 'bottom half' of yielding.
*
* We arrived here because an native call decided to deschedule the
* running task. So the native call's return address got patched to the
* first instruction of this glue code.
*
* When the native call does 'ret' it will come here, and its esp will be
* pointing to the last argument pushed on the C stack before making
* the native call: the 0th argument to the native call, which is always
* the task ptr performing the native call. That's where we take over.
*
* Our goal is to complete the descheduling
*
* - Switch over to the task stack temporarily.
*
* - Save the task's callee-saves onto the task stack.
* (the task is now 'descheduled', safe to set aside)
*
* - Switch *back* to the C stack.
*
* - Restore the C-stack callee-saves.
*
* - Return to the caller on the C stack that activated the task.
*
*/
fn rust_yield_glue() -> vec[str] {
ret ["movl 0(%esp), %ecx # ecx = rust_task"]
2011-03-09 14:05:22 -06:00
+ load_esp_from_rust_sp_first_arg()
+ save_callee_saves()
2011-03-09 14:05:22 -06:00
+ store_esp_to_rust_sp_first_arg()
+ load_esp_from_runtime_sp_first_arg()
+ restore_callee_saves()
+ ["ret"];
}
fn native_glue(int n_args, abi::native_glue_type ngt) -> vec[str] {
let bool pass_task;
alt (ngt) {
case (abi::ngt_rust) { pass_task = true; }
case (abi::ngt_pure_rust) { pass_task = true; }
case (abi::ngt_cdecl) { pass_task = false; }
}
/*
* 0, 4, 8, 12 are callee-saves
* 16 is retpc
* 20 .. (5+i) * 4 are args
*
* ecx is taskptr
* edx is callee
*
*/
fn copy_arg(bool pass_task, uint i) -> str {
if (i == 0u && pass_task) {
ret "movl %edx, (%esp)";
}
auto dst_off = wstr(0 + (i as int));
auto src_off;
if (pass_task) {
src_off = wstr(4 + (i as int));
} else {
src_off = wstr(5 + (i as int));
}
auto m = ["movl " + src_off + "(%ebp),%eax",
"movl %eax," + dst_off + "(%esp)"];
ret _str::connect(m, "\n\t");
}
auto carg = bind copy_arg(pass_task, _);
ret
start()
+ save_callee_saves_with_cfi()
+ ["movl %esp, %ebp # ebp = rust_sp"]
+ [".cfi_def_cfa_register %ebp"]
2011-03-09 14:05:22 -06:00
+ store_esp_to_rust_sp_second_arg()
+ load_esp_from_runtime_sp_second_arg()
+ ["subl $" + wstr(n_args) + ", %esp # esp -= args",
"andl $~0xf, %esp # align esp down"]
+ _vec::init_fn[str](carg, (n_args) as uint)
+ ["movl %edx, %edi # save task from edx to edi",
2011-03-09 14:05:22 -06:00
"call *%ecx # call *%ecx",
"movl %edi, %edx # restore edi-saved task to edx"]
2011-03-09 14:05:22 -06:00
+ load_esp_from_rust_sp_second_arg()
+ restore_callee_saves()
+ ["ret"]
+ end();
}
fn decl_glue(int align, str prefix, str name, vec[str] insns) -> str {
auto sym = prefix + name;
ret "\t.globl " + sym + "\n" +
"\t.balign " + istr(align) + "\n" +
sym + ":\n" +
"\t" + _str::connect(insns, "\n\t");
}
fn decl_native_glue(int align, str prefix, abi::native_glue_type ngt, uint n)
-> str {
let int i = n as int;
ret decl_glue(align, prefix,
abi::native_glue_name(i, ngt),
native_glue(i, ngt));
}
fn get_symbol_prefix() -> str {
if (_str::eq(target_os(), "macos") ||
_str::eq(target_os(), "win32")) {
ret "_";
} else {
ret "";
}
}
fn get_module_asm() -> str {
auto align = 4;
auto prefix = get_symbol_prefix();
auto glues =
[decl_glue(align, prefix,
abi::activate_glue_name(),
rust_activate_glue()),
decl_glue(align, prefix,
abi::yield_glue_name(),
rust_yield_glue())]
+ _vec::init_fn[str](bind decl_native_glue(align, prefix,
abi::ngt_rust, _), (abi::n_native_glues + 1) as uint)
+ _vec::init_fn[str](bind decl_native_glue(align, prefix,
abi::ngt_pure_rust, _), (abi::n_native_glues + 1) as uint)
+ _vec::init_fn[str](bind decl_native_glue(align, prefix,
abi::ngt_cdecl, _), (abi::n_native_glues + 1) as uint);
ret _str::connect(glues, "\n\n");
}
fn get_meta_sect_name() -> str {
if (_str::eq(target_os(), "macos")) {
ret "__DATA,__note.rustc";
}
if (_str::eq(target_os(), "win32")) {
ret ".note.rustc";
}
ret ".note.rustc";
}
fn get_data_layout() -> str {
if (_str::eq(target_os(), "macos")) {
ret "e-p:32:32:32-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:32:64" +
"-f32:32:32-f64:32:64-v64:64:64-v128:128:128-a0:0:64-f80:128:128" +
"-n8:16:32";
}
if (_str::eq(target_os(), "win32")) {
ret "e-p:32:32-f64:64:64-i64:64:64-f80:32:32-n8:16:32";
}
ret "e-p:32:32-f64:32:64-i64:32:64-f80:32:32-n8:16:32";
}
fn get_target_triple() -> str {
if (_str::eq(target_os(), "macos")) {
ret "i686-apple-darwin";
}
if (_str::eq(target_os(), "win32")) {
ret "i686-pc-mingw32";
}
2011-05-05 19:20:27 -05:00
ret "i686-unknown-linux-gnu";
}
//
// Local Variables:
// mode: rust
// fill-column: 78;
// indent-tabs-mode: nil
// c-basic-offset: 4
// buffer-file-coding-system: utf-8-unix
// compile-command: "make -k -C $RBUILD 2>&1 | sed -e 's/\\/x\\//x:\\//g'";
// End:
//