mirror of
https://github.com/m-labs/artiq.git
synced 2025-01-06 17:13:36 +08:00
2648b1b7a1
This also updates / is a prerequisite for updating smoltcp. Rationale for changes made: * compiler_builtins is now shipped in the rust prefix. * rustc's libpanic_unwind no longer works for us because it has a hard dependency on Box (and it's a horrible hack); fortunately, we only ever needed a personality function from it. * panic and oom handlers are now set in a completely different way. * allocators are quite different (and finally stable). * NLL caused internal compiler errors in runtime, so code using NLL was rewritten to not rely on it and it was turned off.
89 lines
3.8 KiB
Rust
89 lines
3.8 KiB
Rust
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
|
|
// file at the top-level directory of this distribution and at
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
// option. This file may not be copied, modified, or distributed
|
|
// except according to those terms.
|
|
|
|
// This is the Rust personality function, adapted for use in ARTIQ. We never actually panic
|
|
// from Rust or recover from Rust exceptions (there's nothing to catch the panics), but we
|
|
// need a personality function to step back through Rust frames in order to make a backtrace.
|
|
//
|
|
// By design, this personality function is only ever called in the search phase, although
|
|
// to keep things simple and close to upstream, it is not modified
|
|
#![allow(private_no_mangle_fns)]
|
|
|
|
use unwind as uw;
|
|
use libc::{c_int, uintptr_t};
|
|
use cslice::AsCSlice;
|
|
|
|
use dwarf::{self, EHAction};
|
|
|
|
// Register ids were lifted from LLVM's TargetLowering::getExceptionPointerRegister()
|
|
// and TargetLowering::getExceptionSelectorRegister() for each architecture,
|
|
// then mapped to DWARF register numbers via register definition tables
|
|
// (typically <arch>RegisterInfo.td, search for "DwarfRegNum").
|
|
// See also http://llvm.org/docs/WritingAnLLVMBackend.html#defining-a-register.
|
|
|
|
#[cfg(target_arch = "x86")]
|
|
const UNWIND_DATA_REG: (i32, i32) = (0, 2); // EAX, EDX
|
|
|
|
#[cfg(target_arch = "x86_64")]
|
|
const UNWIND_DATA_REG: (i32, i32) = (0, 1); // RAX, RDX
|
|
|
|
#[cfg(any(target_arch = "or1k"))]
|
|
const UNWIND_DATA_REG: (i32, i32) = (3, 4); // R3, R4
|
|
|
|
// The following code is based on GCC's C and C++ personality routines. For reference, see:
|
|
// https://github.com/gcc-mirror/gcc/blob/master/libstdc++-v3/libsupc++/eh_personality.cc
|
|
// https://github.com/gcc-mirror/gcc/blob/trunk/libgcc/unwind-c.c
|
|
#[lang = "eh_personality"]
|
|
#[no_mangle]
|
|
#[allow(unused)]
|
|
unsafe extern "C" fn rust_eh_personality(version: c_int,
|
|
actions: uw::_Unwind_Action,
|
|
exception_class: uw::_Unwind_Exception_Class,
|
|
exception_object: *mut uw::_Unwind_Exception,
|
|
context: *mut uw::_Unwind_Context)
|
|
-> uw::_Unwind_Reason_Code {
|
|
if version != 1 {
|
|
return uw::_URC_FATAL_PHASE1_ERROR;
|
|
}
|
|
let eh_action = match find_eh_action(context) {
|
|
Ok(action) => action,
|
|
Err(_) => return uw::_URC_FATAL_PHASE1_ERROR,
|
|
};
|
|
if actions as i32 & uw::_UA_SEARCH_PHASE as i32 != 0 {
|
|
match eh_action {
|
|
EHAction::None |
|
|
EHAction::Cleanup(_) => return uw::_URC_CONTINUE_UNWIND,
|
|
EHAction::Catch(_) => return uw::_URC_HANDLER_FOUND,
|
|
EHAction::Terminate => return uw::_URC_FATAL_PHASE1_ERROR,
|
|
}
|
|
} else {
|
|
match eh_action {
|
|
EHAction::None => return uw::_URC_CONTINUE_UNWIND,
|
|
EHAction::Cleanup(lpad) |
|
|
EHAction::Catch(lpad) => {
|
|
uw::_Unwind_SetGR(context, UNWIND_DATA_REG.0, exception_object as uintptr_t);
|
|
uw::_Unwind_SetGR(context, UNWIND_DATA_REG.1, 0);
|
|
uw::_Unwind_SetIP(context, lpad);
|
|
return uw::_URC_INSTALL_CONTEXT;
|
|
}
|
|
EHAction::Terminate => return uw::_URC_FATAL_PHASE2_ERROR,
|
|
}
|
|
}
|
|
}
|
|
|
|
unsafe fn find_eh_action(context: *mut uw::_Unwind_Context)
|
|
-> Result<EHAction, ()>
|
|
{
|
|
let lsda = uw::_Unwind_GetLanguageSpecificData(context) as *const u8;
|
|
let func = uw::_Unwind_GetRegionStart(context);
|
|
let ip = uw::_Unwind_GetIP(context);
|
|
Ok(dwarf::find_eh_action(lsda, func, ip, [].as_c_slice()))
|
|
}
|