9
9
// except according to those terms.
10
10
11
11
//! Failure support for libcore
12
+ //!
13
+ //! The core library cannot define failure, but it does *declare* failure. This
14
+ //! means that the functions inside of libcore are allowed to fail, but to be
15
+ //! useful an upstream crate must define failure for libcore to use. The current
16
+ //! interface for failure is:
17
+ //!
18
+ //! fn begin_unwind(fmt: &fmt::Arguments, file: &str, line: uint) -> !;
19
+ //!
20
+ //! This definition allows for failing with any general message, but it does not
21
+ //! allow for failing with a `~Any` value. The reason for this is that libcore
22
+ //! is not allowed to allocate.
23
+ //!
24
+ //! This module contains a few other failure functions, but these are just the
25
+ //! necessary lang items for the compiler. All failure is funneled through this
26
+ //! one function. Currently, the actual symbol is declared in the standard
27
+ //! library, but the location of this may change over time.
12
28
13
29
#![ allow( dead_code, missing_doc) ]
14
30
15
31
#[ cfg( not( test) ) ]
16
32
use str:: raw:: c_str_to_static_slice;
17
-
18
- // FIXME: Once std::fmt is in libcore, all of these functions should delegate
19
- // to a common failure function with this signature:
20
- //
21
- // extern {
22
- // fn rust_unwind(f: &fmt::Arguments, file: &str, line: uint) -> !;
23
- // }
24
- //
25
- // Each of these functions can create a temporary fmt::Arguments
26
- // structure to pass to this function.
33
+ use fmt;
27
34
28
35
#[ cold] #[ inline( never) ] // this is the slow path, always
29
36
#[ lang="fail_" ]
@@ -32,24 +39,31 @@ fn fail_(expr: *u8, file: *u8, line: uint) -> ! {
32
39
unsafe {
33
40
let expr = c_str_to_static_slice ( expr as * i8 ) ;
34
41
let file = c_str_to_static_slice ( file as * i8 ) ;
35
- begin_unwind ( expr, file, line)
42
+ format_args ! ( |args| -> ( ) {
43
+ begin_unwind( args, file, line) ;
44
+ } , "{}" , expr) ;
45
+
46
+ loop { }
36
47
}
37
48
}
38
49
39
50
#[ cold]
40
51
#[ lang="fail_bounds_check" ]
41
52
#[ cfg( not( test) ) ]
42
53
fn fail_bounds_check ( file : * u8 , line : uint , index : uint , len : uint ) -> ! {
43
- #[ allow( ctypes) ]
44
- extern { fn rust_fail_bounds_check ( file : * u8 , line : uint ,
45
- index : uint , len : uint , ) -> !; }
46
- unsafe { rust_fail_bounds_check ( file, line, index, len) }
54
+ let file = unsafe { c_str_to_static_slice ( file as * i8 ) } ;
55
+ format_args ! ( |args| -> ( ) {
56
+ begin_unwind( args, file, line) ;
57
+ } , "index out of bounds: the len is {} but the index is {}" , len, index) ;
58
+ loop { }
47
59
}
48
60
49
61
#[ cold]
50
- pub fn begin_unwind ( msg : & str , file : & ' static str , line : uint ) -> ! {
62
+ pub fn begin_unwind ( fmt : & fmt:: Arguments , file : & ' static str , line : uint ) -> ! {
63
+ // FIXME: this should be a proper lang item, it should not just be some
64
+ // undefined symbol sitting in the middle of nowhere.
51
65
#[ allow( ctypes) ]
52
- extern { fn rust_begin_unwind ( msg : & str , file : & ' static str ,
66
+ extern { fn rust_begin_unwind ( fmt : & fmt :: Arguments , file : & ' static str ,
53
67
line : uint ) -> !; }
54
- unsafe { rust_begin_unwind ( msg , file, line) }
68
+ unsafe { rust_begin_unwind ( fmt , file, line) }
55
69
}
0 commit comments