2015-07-13 20:11:44 -05:00
|
|
|
// 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.
|
|
|
|
|
|
|
|
//! Unwinding implementation of top of native Win64 SEH,
|
|
|
|
//! however the unwind handler data (aka LSDA) uses GCC-compatible encoding.
|
|
|
|
|
|
|
|
#![allow(bad_style)]
|
|
|
|
#![allow(private_no_mangle_fns)]
|
|
|
|
|
|
|
|
use prelude::v1::*;
|
|
|
|
|
|
|
|
use any::Any;
|
2015-09-08 17:53:46 -05:00
|
|
|
use sys_common::dwarf::eh;
|
2015-07-13 20:11:44 -05:00
|
|
|
use core::mem;
|
|
|
|
use core::ptr;
|
2015-11-02 18:23:22 -06:00
|
|
|
use sys::c;
|
2015-07-13 20:11:44 -05:00
|
|
|
|
|
|
|
// Define our exception codes:
|
|
|
|
// according to http://msdn.microsoft.com/en-us/library/het71c37(v=VS.80).aspx,
|
|
|
|
// [31:30] = 3 (error), 2 (warning), 1 (info), 0 (success)
|
|
|
|
// [29] = 1 (user-defined)
|
|
|
|
// [28] = 0 (reserved)
|
|
|
|
// we define bits:
|
|
|
|
// [24:27] = type
|
|
|
|
// [0:23] = magic
|
2015-11-02 18:23:22 -06:00
|
|
|
const ETYPE: c::DWORD = 0b1110_u32 << 28;
|
|
|
|
const MAGIC: c::DWORD = 0x525354; // "RST"
|
2015-07-13 20:11:44 -05:00
|
|
|
|
2015-11-02 18:23:22 -06:00
|
|
|
const RUST_PANIC: c::DWORD = ETYPE | (1 << 24) | MAGIC;
|
2015-07-13 20:11:44 -05:00
|
|
|
|
|
|
|
#[repr(C)]
|
|
|
|
struct PanicData {
|
|
|
|
data: Box<Any + Send + 'static>
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn panic(data: Box<Any + Send + 'static>) -> ! {
|
|
|
|
let panic_ctx = Box::new(PanicData { data: data });
|
2015-11-02 18:23:22 -06:00
|
|
|
let params = [Box::into_raw(panic_ctx) as c::ULONG_PTR];
|
|
|
|
c::RaiseException(RUST_PANIC,
|
|
|
|
c::EXCEPTION_NONCONTINUABLE,
|
|
|
|
params.len() as c::DWORD,
|
|
|
|
¶ms as *const c::ULONG_PTR);
|
2015-07-13 20:11:44 -05:00
|
|
|
rtabort!("could not unwind stack");
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn cleanup(ptr: *mut u8) -> Box<Any + Send + 'static> {
|
|
|
|
let panic_ctx = Box::from_raw(ptr as *mut PanicData);
|
|
|
|
return panic_ctx.data;
|
|
|
|
}
|
|
|
|
|
|
|
|
// SEH doesn't support resuming unwinds after calling a landing pad like
|
|
|
|
// libunwind does. For this reason, MSVC compiler outlines landing pads into
|
|
|
|
// separate functions that can be called directly from the personality function
|
|
|
|
// but are nevertheless able to find and modify stack frame of the "parent"
|
|
|
|
// function.
|
|
|
|
//
|
|
|
|
// Since this cannot be done with libdwarf-style landing pads,
|
|
|
|
// rust_eh_personality instead catches RUST_PANICs, runs the landing pad, then
|
|
|
|
// reraises the exception.
|
|
|
|
//
|
|
|
|
// Note that it makes certain assumptions about the exception:
|
|
|
|
//
|
|
|
|
// 1. That RUST_PANIC is non-continuable, so no lower stack frame may choose to
|
|
|
|
// resume execution.
|
|
|
|
// 2. That the first parameter of the exception is a pointer to an extra data
|
|
|
|
// area (PanicData).
|
|
|
|
// Since these assumptions do not generally hold true for foreign exceptions
|
|
|
|
// (system faults, C++ exceptions, etc), we make no attempt to invoke our
|
|
|
|
// landing pads (and, thus, destructors!) for anything other than RUST_PANICs.
|
|
|
|
// This is considered acceptable, because the behavior of throwing exceptions
|
|
|
|
// through a C ABI boundary is undefined.
|
|
|
|
|
|
|
|
#[lang = "eh_personality_catch"]
|
|
|
|
#[cfg(not(test))]
|
|
|
|
unsafe extern fn rust_eh_personality_catch(
|
2015-11-02 18:23:22 -06:00
|
|
|
exceptionRecord: *mut c::EXCEPTION_RECORD,
|
|
|
|
establisherFrame: c::LPVOID,
|
|
|
|
contextRecord: *mut c::CONTEXT,
|
|
|
|
dispatcherContext: *mut c::DISPATCHER_CONTEXT
|
|
|
|
) -> c::EXCEPTION_DISPOSITION
|
2015-07-13 20:11:44 -05:00
|
|
|
{
|
|
|
|
rust_eh_personality(exceptionRecord, establisherFrame,
|
|
|
|
contextRecord, dispatcherContext)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[lang = "eh_personality"]
|
|
|
|
#[cfg(not(test))]
|
|
|
|
unsafe extern fn rust_eh_personality(
|
2015-11-02 18:23:22 -06:00
|
|
|
exceptionRecord: *mut c::EXCEPTION_RECORD,
|
|
|
|
establisherFrame: c::LPVOID,
|
|
|
|
contextRecord: *mut c::CONTEXT,
|
|
|
|
dispatcherContext: *mut c::DISPATCHER_CONTEXT
|
|
|
|
) -> c::EXCEPTION_DISPOSITION
|
2015-07-13 20:11:44 -05:00
|
|
|
{
|
|
|
|
let er = &*exceptionRecord;
|
|
|
|
let dc = &*dispatcherContext;
|
|
|
|
|
2015-11-02 18:23:22 -06:00
|
|
|
if er.ExceptionFlags & c::EXCEPTION_UNWIND == 0 { // we are in the dispatch phase
|
2015-07-13 20:11:44 -05:00
|
|
|
if er.ExceptionCode == RUST_PANIC {
|
|
|
|
if let Some(lpad) = find_landing_pad(dc) {
|
2015-11-02 18:23:22 -06:00
|
|
|
c::RtlUnwindEx(establisherFrame,
|
|
|
|
lpad as c::LPVOID,
|
|
|
|
exceptionRecord,
|
|
|
|
er.ExceptionInformation[0] as c::LPVOID, // pointer to PanicData
|
|
|
|
contextRecord,
|
|
|
|
dc.HistoryTable);
|
2015-07-13 20:11:44 -05:00
|
|
|
rtabort!("could not unwind");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-11-02 18:23:22 -06:00
|
|
|
c::ExceptionContinueSearch
|
2015-07-13 20:11:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(test))]
|
2015-10-18 16:28:47 -05:00
|
|
|
#[lang = "eh_unwind_resume"]
|
2015-09-11 13:10:43 -05:00
|
|
|
#[unwind]
|
2015-11-02 18:23:22 -06:00
|
|
|
unsafe extern fn rust_eh_unwind_resume(panic_ctx: c::LPVOID) -> ! {
|
|
|
|
let params = [panic_ctx as c::ULONG_PTR];
|
|
|
|
c::RaiseException(RUST_PANIC,
|
|
|
|
c::EXCEPTION_NONCONTINUABLE,
|
|
|
|
params.len() as c::DWORD,
|
|
|
|
¶ms as *const c::ULONG_PTR);
|
2015-07-13 20:11:44 -05:00
|
|
|
rtabort!("could not resume unwind");
|
|
|
|
}
|
|
|
|
|
2015-11-02 18:23:22 -06:00
|
|
|
unsafe fn find_landing_pad(dc: &c::DISPATCHER_CONTEXT) -> Option<usize> {
|
2015-07-13 20:11:44 -05:00
|
|
|
let eh_ctx = eh::EHContext {
|
|
|
|
ip: dc.ControlPc as usize,
|
|
|
|
func_start: dc.ImageBase as usize + (*dc.FunctionEntry).BeginAddress as usize,
|
|
|
|
text_start: dc.ImageBase as usize,
|
|
|
|
data_start: 0
|
|
|
|
};
|
|
|
|
eh::find_landing_pad(dc.HandlerData, &eh_ctx)
|
|
|
|
}
|