8
8
// option. This file may not be copied, modified, or distributed
9
9
// except according to those terms.
10
10
11
- #![ unstable( feature = "allocator_api" ,
12
- reason = "the precise API and guarantees it provides may be tweaked \
13
- slightly, especially to possibly take into account the \
14
- types being stored to make room for a future \
15
- tracing garbage collector",
16
- issue = "32838" ) ]
11
+ //! Memory allocation APIs
12
+
13
+ #![ stable( feature = "alloc_module" , since = "1.28.0" ) ]
17
14
18
15
use core:: intrinsics:: { min_align_of_val, size_of_val} ;
19
16
use core:: ptr:: { NonNull , Unique } ;
20
17
use core:: usize;
21
18
19
+ #[ stable( feature = "alloc_module" , since = "1.28.0" ) ]
22
20
#[ doc( inline) ]
23
21
pub use core:: alloc:: * ;
24
22
@@ -37,67 +35,112 @@ extern "Rust" {
37
35
fn __rust_alloc_zeroed ( size : usize , align : usize ) -> * mut u8 ;
38
36
}
39
37
38
+ /// The global memory allocator.
39
+ ///
40
+ /// This type implements the [`Alloc`] trait by forwarding calls
41
+ /// to the allocator registered with the `#[global_allocator]` attribute
42
+ /// if there is one, or the `std` crate’s default.
43
+ #[ unstable( feature = "allocator_api" , issue = "32838" ) ]
40
44
#[ derive( Copy , Clone , Default , Debug ) ]
41
45
pub struct Global ;
42
46
43
- #[ unstable( feature = "allocator_api" , issue = "32838" ) ]
44
- #[ rustc_deprecated( since = "1.27.0" , reason = "type renamed to `Global`" ) ]
45
- pub type Heap = Global ;
46
-
47
- #[ unstable( feature = "allocator_api" , issue = "32838" ) ]
48
- #[ rustc_deprecated( since = "1.27.0" , reason = "type renamed to `Global`" ) ]
49
- #[ allow( non_upper_case_globals) ]
50
- pub const Heap : Global = Global ;
51
-
52
- unsafe impl GlobalAlloc for Global {
53
- #[ inline]
54
- unsafe fn alloc ( & self , layout : Layout ) -> * mut Opaque {
55
- let ptr = __rust_alloc ( layout. size ( ) , layout. align ( ) ) ;
56
- ptr as * mut Opaque
57
- }
47
+ /// Allocate memory with the global allocator.
48
+ ///
49
+ /// This function forwards calls to the [`GlobalAlloc::alloc`] method
50
+ /// of the allocator registered with the `#[global_allocator]` attribute
51
+ /// if there is one, or the `std` crate’s default.
52
+ ///
53
+ /// This function is expected to be deprecated in favor of the `alloc` method
54
+ /// of the [`Global`] type when it and the [`Alloc`] trait become stable.
55
+ ///
56
+ /// # Safety
57
+ ///
58
+ /// See [`GlobalAlloc::alloc`].
59
+ #[ stable( feature = "global_alloc" , since = "1.28.0" ) ]
60
+ #[ inline]
61
+ pub unsafe fn alloc ( layout : Layout ) -> * mut u8 {
62
+ __rust_alloc ( layout. size ( ) , layout. align ( ) )
63
+ }
58
64
59
- #[ inline]
60
- unsafe fn dealloc ( & self , ptr : * mut Opaque , layout : Layout ) {
61
- __rust_dealloc ( ptr as * mut u8 , layout. size ( ) , layout. align ( ) )
62
- }
65
+ /// Deallocate memory with the global allocator.
66
+ ///
67
+ /// This function forwards calls to the [`GlobalAlloc::dealloc`] method
68
+ /// of the allocator registered with the `#[global_allocator]` attribute
69
+ /// if there is one, or the `std` crate’s default.
70
+ ///
71
+ /// This function is expected to be deprecated in favor of the `dealloc` method
72
+ /// of the [`Global`] type when it and the [`Alloc`] trait become stable.
73
+ ///
74
+ /// # Safety
75
+ ///
76
+ /// See [`GlobalAlloc::dealloc`].
77
+ #[ stable( feature = "global_alloc" , since = "1.28.0" ) ]
78
+ #[ inline]
79
+ pub unsafe fn dealloc ( ptr : * mut u8 , layout : Layout ) {
80
+ __rust_dealloc ( ptr, layout. size ( ) , layout. align ( ) )
81
+ }
63
82
64
- #[ inline]
65
- unsafe fn realloc ( & self , ptr : * mut Opaque , layout : Layout , new_size : usize ) -> * mut Opaque {
66
- let ptr = __rust_realloc ( ptr as * mut u8 , layout. size ( ) , layout. align ( ) , new_size) ;
67
- ptr as * mut Opaque
68
- }
83
+ /// Reallocate memory with the global allocator.
84
+ ///
85
+ /// This function forwards calls to the [`GlobalAlloc::realloc`] method
86
+ /// of the allocator registered with the `#[global_allocator]` attribute
87
+ /// if there is one, or the `std` crate’s default.
88
+ ///
89
+ /// This function is expected to be deprecated in favor of the `realloc` method
90
+ /// of the [`Global`] type when it and the [`Alloc`] trait become stable.
91
+ ///
92
+ /// # Safety
93
+ ///
94
+ /// See [`GlobalAlloc::realloc`].
95
+ #[ stable( feature = "global_alloc" , since = "1.28.0" ) ]
96
+ #[ inline]
97
+ pub unsafe fn realloc ( ptr : * mut u8 , layout : Layout , new_size : usize ) -> * mut u8 {
98
+ __rust_realloc ( ptr, layout. size ( ) , layout. align ( ) , new_size)
99
+ }
69
100
70
- #[ inline]
71
- unsafe fn alloc_zeroed ( & self , layout : Layout ) -> * mut Opaque {
72
- let ptr = __rust_alloc_zeroed ( layout. size ( ) , layout. align ( ) ) ;
73
- ptr as * mut Opaque
74
- }
101
+ /// Allocate zero-initialized memory with the global allocator.
102
+ ///
103
+ /// This function forwards calls to the [`GlobalAlloc::alloc_zeroed`] method
104
+ /// of the allocator registered with the `#[global_allocator]` attribute
105
+ /// if there is one, or the `std` crate’s default.
106
+ ///
107
+ /// This function is expected to be deprecated in favor of the `alloc_zeroed` method
108
+ /// of the [`Global`] type when it and the [`Alloc`] trait become stable.
109
+ ///
110
+ /// # Safety
111
+ ///
112
+ /// See [`GlobalAlloc::alloc_zeroed`].
113
+ #[ stable( feature = "global_alloc" , since = "1.28.0" ) ]
114
+ #[ inline]
115
+ pub unsafe fn alloc_zeroed ( layout : Layout ) -> * mut u8 {
116
+ __rust_alloc_zeroed ( layout. size ( ) , layout. align ( ) )
75
117
}
76
118
119
+ #[ unstable( feature = "allocator_api" , issue = "32838" ) ]
77
120
unsafe impl Alloc for Global {
78
121
#[ inline]
79
- unsafe fn alloc ( & mut self , layout : Layout ) -> Result < NonNull < Opaque > , AllocErr > {
80
- NonNull :: new ( GlobalAlloc :: alloc ( self , layout) ) . ok_or ( AllocErr )
122
+ unsafe fn alloc ( & mut self , layout : Layout ) -> Result < NonNull < u8 > , AllocErr > {
123
+ NonNull :: new ( alloc ( layout) ) . ok_or ( AllocErr )
81
124
}
82
125
83
126
#[ inline]
84
- unsafe fn dealloc ( & mut self , ptr : NonNull < Opaque > , layout : Layout ) {
85
- GlobalAlloc :: dealloc ( self , ptr. as_ptr ( ) , layout)
127
+ unsafe fn dealloc ( & mut self , ptr : NonNull < u8 > , layout : Layout ) {
128
+ dealloc ( ptr. as_ptr ( ) , layout)
86
129
}
87
130
88
131
#[ inline]
89
132
unsafe fn realloc ( & mut self ,
90
- ptr : NonNull < Opaque > ,
133
+ ptr : NonNull < u8 > ,
91
134
layout : Layout ,
92
135
new_size : usize )
93
- -> Result < NonNull < Opaque > , AllocErr >
136
+ -> Result < NonNull < u8 > , AllocErr >
94
137
{
95
- NonNull :: new ( GlobalAlloc :: realloc ( self , ptr. as_ptr ( ) , layout, new_size) ) . ok_or ( AllocErr )
138
+ NonNull :: new ( realloc ( ptr. as_ptr ( ) , layout, new_size) ) . ok_or ( AllocErr )
96
139
}
97
140
98
141
#[ inline]
99
- unsafe fn alloc_zeroed ( & mut self , layout : Layout ) -> Result < NonNull < Opaque > , AllocErr > {
100
- NonNull :: new ( GlobalAlloc :: alloc_zeroed ( self , layout) ) . ok_or ( AllocErr )
142
+ unsafe fn alloc_zeroed ( & mut self , layout : Layout ) -> Result < NonNull < u8 > , AllocErr > {
143
+ NonNull :: new ( alloc_zeroed ( layout) ) . ok_or ( AllocErr )
101
144
}
102
145
}
103
146
@@ -111,9 +154,9 @@ unsafe fn exchange_malloc(size: usize, align: usize) -> *mut u8 {
111
154
align as * mut u8
112
155
} else {
113
156
let layout = Layout :: from_size_align_unchecked ( size, align) ;
114
- let ptr = Global . alloc ( layout) ;
157
+ let ptr = alloc ( layout) ;
115
158
if !ptr. is_null ( ) {
116
- ptr as * mut u8
159
+ ptr
117
160
} else {
118
161
oom ( layout)
119
162
}
@@ -129,10 +172,23 @@ pub(crate) unsafe fn box_free<T: ?Sized>(ptr: Unique<T>) {
129
172
// We do not allocate for Box<T> when T is ZST, so deallocation is also not necessary.
130
173
if size != 0 {
131
174
let layout = Layout :: from_size_align_unchecked ( size, align) ;
132
- Global . dealloc ( ptr as * mut Opaque , layout) ;
175
+ dealloc ( ptr as * mut u8 , layout) ;
133
176
}
134
177
}
135
178
179
+ /// Abort on memory allocation error or failure.
180
+ ///
181
+ /// Callers of memory allocation APIs wishing to abort computation
182
+ /// in response to an allocation error are encouraged to call this function,
183
+ /// rather than directly invoking `panic!` or similar.
184
+ ///
185
+ /// The default behavior of this function is to print a message to standard error
186
+ /// and abort the process.
187
+ /// It can be replaced with [`set_oom_hook`] and [`take_oom_hook`].
188
+ ///
189
+ /// [`set_oom_hook`]: ../../std/alloc/fn.set_oom_hook.html
190
+ /// [`take_oom_hook`]: ../../std/alloc/fn.take_oom_hook.html
191
+ #[ stable( feature = "global_alloc" , since = "1.28.0" ) ]
136
192
#[ rustc_allocator_nounwind]
137
193
pub fn oom ( layout : Layout ) -> ! {
138
194
#[ allow( improper_ctypes) ]
0 commit comments