@@ -7,7 +7,7 @@ use crate::fmt;
7
7
use crate :: fs;
8
8
use crate :: io;
9
9
use crate :: marker:: PhantomData ;
10
- use crate :: mem:: forget;
10
+ use crate :: mem:: { forget, ManuallyDrop } ;
11
11
use crate :: ptr;
12
12
use crate :: sys;
13
13
use crate :: sys:: cvt;
@@ -91,7 +91,7 @@ pub struct OwnedHandle {
91
91
#[ repr( transparent) ]
92
92
#[ stable( feature = "io_safety" , since = "1.63.0" ) ]
93
93
#[ derive( Debug ) ]
94
- pub struct HandleOrNull ( OwnedHandle ) ;
94
+ pub struct HandleOrNull ( RawHandle ) ;
95
95
96
96
/// FFI type for handles in return values or out parameters, where `INVALID_HANDLE_VALUE` is used
97
97
/// as a sentry value to indicate errors, such as in the return value of `CreateFileW`. This uses
@@ -110,7 +110,7 @@ pub struct HandleOrNull(OwnedHandle);
110
110
#[ repr( transparent) ]
111
111
#[ stable( feature = "io_safety" , since = "1.63.0" ) ]
112
112
#[ derive( Debug ) ]
113
- pub struct HandleOrInvalid ( OwnedHandle ) ;
113
+ pub struct HandleOrInvalid ( RawHandle ) ;
114
114
115
115
// The Windows [`HANDLE`] type may be transferred across and shared between
116
116
// thread boundaries (despite containing a `*mut void`, which in general isn't
@@ -163,15 +163,24 @@ impl TryFrom<HandleOrNull> for OwnedHandle {
163
163
164
164
#[ inline]
165
165
fn try_from ( handle_or_null : HandleOrNull ) -> Result < Self , NullHandleError > {
166
- let owned_handle = handle_or_null. 0 ;
167
- if owned_handle. handle . is_null ( ) {
168
- // Don't call `CloseHandle`; it'd be harmless, except that it could
169
- // overwrite the `GetLastError` error.
170
- forget ( owned_handle) ;
171
-
172
- Err ( NullHandleError ( ( ) ) )
166
+ let handle_or_null = ManuallyDrop :: new ( handle_or_null) ;
167
+ if handle_or_null. is_valid ( ) {
168
+ // SAFETY: The handle is not null.
169
+ Ok ( unsafe { OwnedHandle :: from_raw_handle ( handle_or_null. 0 ) } )
173
170
} else {
174
- Ok ( owned_handle)
171
+ Err ( NullHandleError ( ( ) ) )
172
+ }
173
+ }
174
+ }
175
+
176
+ #[ stable( feature = "handle_or_drop" , since = "1.75.0" ) ]
177
+ impl Drop for HandleOrNull {
178
+ #[ inline]
179
+ fn drop ( & mut self ) {
180
+ if self . is_valid ( ) {
181
+ unsafe {
182
+ let _ = sys:: c:: CloseHandle ( self . 0 ) ;
183
+ }
175
184
}
176
185
}
177
186
}
@@ -232,15 +241,24 @@ impl TryFrom<HandleOrInvalid> for OwnedHandle {
232
241
233
242
#[ inline]
234
243
fn try_from ( handle_or_invalid : HandleOrInvalid ) -> Result < Self , InvalidHandleError > {
235
- let owned_handle = handle_or_invalid. 0 ;
236
- if owned_handle. handle == sys:: c:: INVALID_HANDLE_VALUE {
237
- // Don't call `CloseHandle`; it'd be harmless, except that it could
238
- // overwrite the `GetLastError` error.
239
- forget ( owned_handle) ;
240
-
241
- Err ( InvalidHandleError ( ( ) ) )
244
+ let handle_or_invalid = ManuallyDrop :: new ( handle_or_invalid) ;
245
+ if handle_or_invalid. is_valid ( ) {
246
+ // SAFETY: The handle is not invalid.
247
+ Ok ( unsafe { OwnedHandle :: from_raw_handle ( handle_or_invalid. 0 ) } )
242
248
} else {
243
- Ok ( owned_handle)
249
+ Err ( InvalidHandleError ( ( ) ) )
250
+ }
251
+ }
252
+ }
253
+
254
+ #[ stable( feature = "handle_or_drop" , since = "1.75.0" ) ]
255
+ impl Drop for HandleOrInvalid {
256
+ #[ inline]
257
+ fn drop ( & mut self ) {
258
+ if self . is_valid ( ) {
259
+ unsafe {
260
+ let _ = sys:: c:: CloseHandle ( self . 0 ) ;
261
+ }
244
262
}
245
263
}
246
264
}
@@ -333,7 +351,11 @@ impl HandleOrNull {
333
351
#[ stable( feature = "io_safety" , since = "1.63.0" ) ]
334
352
#[ inline]
335
353
pub unsafe fn from_raw_handle ( handle : RawHandle ) -> Self {
336
- Self ( OwnedHandle :: from_raw_handle ( handle) )
354
+ Self ( handle)
355
+ }
356
+
357
+ fn is_valid ( & self ) -> bool {
358
+ !self . 0 . is_null ( )
337
359
}
338
360
}
339
361
@@ -356,7 +378,11 @@ impl HandleOrInvalid {
356
378
#[ stable( feature = "io_safety" , since = "1.63.0" ) ]
357
379
#[ inline]
358
380
pub unsafe fn from_raw_handle ( handle : RawHandle ) -> Self {
359
- Self ( OwnedHandle :: from_raw_handle ( handle) )
381
+ Self ( handle)
382
+ }
383
+
384
+ fn is_valid ( & self ) -> bool {
385
+ self . 0 != sys:: c:: INVALID_HANDLE_VALUE
360
386
}
361
387
}
362
388
0 commit comments