2015-01-27 12:20:58 -08: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.
|
|
|
|
|
|
2015-02-06 09:42:57 -08:00
|
|
|
use io::ErrorKind;
|
2015-02-24 23:27:20 -08:00
|
|
|
use io;
|
|
|
|
|
use mem;
|
2015-05-12 11:03:49 -07:00
|
|
|
use ops::Deref;
|
2015-02-06 09:42:57 -08:00
|
|
|
use ptr;
|
2015-11-02 16:23:22 -08:00
|
|
|
use sys::c;
|
2015-02-06 09:42:57 -08:00
|
|
|
use sys::cvt;
|
2015-01-27 12:20:58 -08:00
|
|
|
|
2015-05-12 11:03:49 -07:00
|
|
|
/// An owned container for `HANDLE` object, closing them on Drop.
|
|
|
|
|
///
|
|
|
|
|
/// All methods are inherited through a `Deref` impl to `RawHandle`
|
|
|
|
|
pub struct Handle(RawHandle);
|
2015-01-27 12:20:58 -08:00
|
|
|
|
2015-05-12 11:03:49 -07:00
|
|
|
/// A wrapper type for `HANDLE` objects to give them proper Send/Sync inference
|
|
|
|
|
/// as well as Rust-y methods.
|
|
|
|
|
///
|
|
|
|
|
/// This does **not** drop the handle when it goes out of scope, use `Handle`
|
|
|
|
|
/// instead for that.
|
|
|
|
|
#[derive(Copy, Clone)]
|
2015-11-02 16:23:22 -08:00
|
|
|
pub struct RawHandle(c::HANDLE);
|
2015-05-12 11:03:49 -07:00
|
|
|
|
|
|
|
|
unsafe impl Send for RawHandle {}
|
|
|
|
|
unsafe impl Sync for RawHandle {}
|
2015-01-27 12:20:58 -08:00
|
|
|
|
|
|
|
|
impl Handle {
|
2015-11-02 16:23:22 -08:00
|
|
|
pub fn new(handle: c::HANDLE) -> Handle {
|
2015-05-12 11:03:49 -07:00
|
|
|
Handle(RawHandle::new(handle))
|
2015-01-27 12:20:58 -08:00
|
|
|
}
|
2015-02-06 09:42:57 -08:00
|
|
|
|
2015-11-02 16:23:22 -08:00
|
|
|
pub fn into_raw(self) -> c::HANDLE {
|
2015-05-12 11:03:49 -07:00
|
|
|
let ret = self.raw();
|
2015-05-07 10:49:39 -07:00
|
|
|
mem::forget(self);
|
2015-02-24 23:27:20 -08:00
|
|
|
return ret;
|
|
|
|
|
}
|
2015-05-12 11:03:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Deref for Handle {
|
|
|
|
|
type Target = RawHandle;
|
|
|
|
|
fn deref(&self) -> &RawHandle { &self.0 }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Drop for Handle {
|
|
|
|
|
fn drop(&mut self) {
|
2015-11-02 16:23:22 -08:00
|
|
|
unsafe { let _ = c::CloseHandle(self.raw()); }
|
2015-05-12 11:03:49 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl RawHandle {
|
2015-11-02 16:23:22 -08:00
|
|
|
pub fn new(handle: c::HANDLE) -> RawHandle {
|
2015-05-12 11:03:49 -07:00
|
|
|
RawHandle(handle)
|
|
|
|
|
}
|
|
|
|
|
|
2015-11-02 16:23:22 -08:00
|
|
|
pub fn raw(&self) -> c::HANDLE { self.0 }
|
2015-02-24 23:27:20 -08:00
|
|
|
|
2015-02-06 09:42:57 -08:00
|
|
|
pub fn read(&self, buf: &mut [u8]) -> io::Result<usize> {
|
2015-04-14 11:17:47 -07:00
|
|
|
let mut read = 0;
|
|
|
|
|
let res = cvt(unsafe {
|
2015-11-02 16:23:22 -08:00
|
|
|
c::ReadFile(self.0, buf.as_ptr() as c::LPVOID,
|
|
|
|
|
buf.len() as c::DWORD, &mut read,
|
2015-04-14 11:17:47 -07:00
|
|
|
ptr::null_mut())
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
match res {
|
|
|
|
|
Ok(_) => Ok(read as usize),
|
|
|
|
|
|
|
|
|
|
// The special treatment of BrokenPipe is to deal with Windows
|
|
|
|
|
// pipe semantics, which yields this error when *reading* from
|
|
|
|
|
// a pipe after the other end has closed; we interpret that as
|
|
|
|
|
// EOF on the pipe.
|
|
|
|
|
Err(ref e) if e.kind() == ErrorKind::BrokenPipe => Ok(0),
|
|
|
|
|
|
|
|
|
|
Err(e) => Err(e)
|
|
|
|
|
}
|
2015-02-06 09:42:57 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn write(&self, buf: &[u8]) -> io::Result<usize> {
|
2015-04-14 11:17:47 -07:00
|
|
|
let mut amt = 0;
|
|
|
|
|
try!(cvt(unsafe {
|
2015-11-02 16:23:22 -08:00
|
|
|
c::WriteFile(self.0, buf.as_ptr() as c::LPVOID,
|
|
|
|
|
buf.len() as c::DWORD, &mut amt,
|
2015-04-14 11:17:47 -07:00
|
|
|
ptr::null_mut())
|
|
|
|
|
}));
|
|
|
|
|
Ok(amt as usize)
|
2015-02-06 09:42:57 -08:00
|
|
|
}
|
2015-04-27 13:44:20 -07:00
|
|
|
|
2015-11-02 16:23:22 -08:00
|
|
|
pub fn duplicate(&self, access: c::DWORD, inherit: bool,
|
|
|
|
|
options: c::DWORD) -> io::Result<Handle> {
|
|
|
|
|
let mut ret = 0 as c::HANDLE;
|
2015-04-27 13:44:20 -07:00
|
|
|
try!(cvt(unsafe {
|
2015-11-02 16:23:22 -08:00
|
|
|
let cur_proc = c::GetCurrentProcess();
|
|
|
|
|
c::DuplicateHandle(cur_proc, self.0, cur_proc, &mut ret,
|
|
|
|
|
access, inherit as c::BOOL,
|
2015-04-27 13:44:20 -07:00
|
|
|
options)
|
|
|
|
|
}));
|
|
|
|
|
Ok(Handle::new(ret))
|
|
|
|
|
}
|
2015-01-27 12:20:58 -08:00
|
|
|
}
|