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.
|
|
|
|
|
|
2016-02-12 10:29:25 -08:00
|
|
|
#![unstable(issue = "0", feature = "windows_handle")]
|
|
|
|
|
|
2016-02-12 00:17:24 -08:00
|
|
|
use prelude::v1::*;
|
|
|
|
|
|
2016-02-23 00:04:24 +00:00
|
|
|
use cmp;
|
2016-02-12 00:17:24 -08:00
|
|
|
use io::{ErrorKind, Read};
|
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;
|
2016-02-12 00:17:24 -08:00
|
|
|
use sys_common::io::read_to_end_uninitialized;
|
2016-02-23 00:04:24 +00:00
|
|
|
use u32;
|
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
|
|
|
|
2016-02-12 10:29:25 -08:00
|
|
|
pub fn new_event(manual: bool, init: bool) -> io::Result<Handle> {
|
|
|
|
|
unsafe {
|
|
|
|
|
let event = c::CreateEventW(0 as *mut _,
|
|
|
|
|
manual as c::BOOL,
|
|
|
|
|
init as c::BOOL,
|
|
|
|
|
0 as *const _);
|
|
|
|
|
if event.is_null() {
|
|
|
|
|
Err(io::Error::last_os_error())
|
|
|
|
|
} else {
|
|
|
|
|
Ok(Handle::new(event))
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
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;
|
2016-02-23 00:04:24 +00:00
|
|
|
// ReadFile takes a DWORD (u32) for the length so it only supports
|
|
|
|
|
// reading u32::MAX bytes at a time.
|
|
|
|
|
let len = cmp::min(buf.len(), u32::MAX as usize) as c::DWORD;
|
2015-04-14 11:17:47 -07:00
|
|
|
let res = cvt(unsafe {
|
2016-02-23 00:04:24 +00:00
|
|
|
c::ReadFile(self.0, buf.as_mut_ptr() as c::LPVOID,
|
|
|
|
|
len, &mut read, ptr::null_mut())
|
2015-04-14 11:17:47 -07:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
2016-02-12 10:29:25 -08:00
|
|
|
pub unsafe fn read_overlapped(&self,
|
|
|
|
|
buf: &mut [u8],
|
|
|
|
|
overlapped: *mut c::OVERLAPPED)
|
|
|
|
|
-> io::Result<Option<usize>> {
|
|
|
|
|
let len = cmp::min(buf.len(), <c::DWORD>::max_value() as usize) as c::DWORD;
|
|
|
|
|
let mut amt = 0;
|
|
|
|
|
let res = cvt({
|
|
|
|
|
c::ReadFile(self.0, buf.as_ptr() as c::LPVOID,
|
|
|
|
|
len, &mut amt, overlapped)
|
|
|
|
|
});
|
|
|
|
|
match res {
|
|
|
|
|
Ok(_) => Ok(Some(amt as usize)),
|
|
|
|
|
Err(e) => {
|
|
|
|
|
if e.raw_os_error() == Some(c::ERROR_IO_PENDING as i32) {
|
|
|
|
|
Ok(None)
|
|
|
|
|
} else if e.raw_os_error() == Some(c::ERROR_BROKEN_PIPE as i32) {
|
|
|
|
|
Ok(Some(0))
|
|
|
|
|
} else {
|
|
|
|
|
Err(e)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn overlapped_result(&self,
|
|
|
|
|
overlapped: *mut c::OVERLAPPED,
|
|
|
|
|
wait: bool) -> io::Result<usize> {
|
|
|
|
|
unsafe {
|
|
|
|
|
let mut bytes = 0;
|
|
|
|
|
let wait = if wait {c::TRUE} else {c::FALSE};
|
|
|
|
|
let res = cvt({
|
|
|
|
|
c::GetOverlappedResult(self.raw(), overlapped, &mut bytes, wait)
|
|
|
|
|
});
|
|
|
|
|
match res {
|
|
|
|
|
Ok(_) => Ok(bytes as usize),
|
|
|
|
|
Err(e) => {
|
|
|
|
|
if e.raw_os_error() == Some(c::ERROR_HANDLE_EOF as i32) ||
|
|
|
|
|
e.raw_os_error() == Some(c::ERROR_BROKEN_PIPE as i32) {
|
|
|
|
|
Ok(0)
|
|
|
|
|
} else {
|
|
|
|
|
Err(e)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn cancel_io(&self) -> io::Result<()> {
|
|
|
|
|
unsafe {
|
|
|
|
|
cvt(c::CancelIo(self.raw())).map(|_| ())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-12 00:17:24 -08:00
|
|
|
pub fn read_to_end(&self, buf: &mut Vec<u8>) -> io::Result<usize> {
|
|
|
|
|
let mut me = self;
|
|
|
|
|
(&mut me).read_to_end(buf)
|
|
|
|
|
}
|
|
|
|
|
|
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;
|
2016-02-23 00:04:24 +00:00
|
|
|
// WriteFile takes a DWORD (u32) for the length so it only supports
|
|
|
|
|
// writing u32::MAX bytes at a time.
|
|
|
|
|
let len = cmp::min(buf.len(), u32::MAX as usize) as c::DWORD;
|
2015-04-14 11:17:47 -07:00
|
|
|
try!(cvt(unsafe {
|
2015-11-02 16:23:22 -08:00
|
|
|
c::WriteFile(self.0, buf.as_ptr() as c::LPVOID,
|
2016-02-23 00:04:24 +00:00
|
|
|
len, &mut amt, ptr::null_mut())
|
2015-04-14 11:17:47 -07:00
|
|
|
}));
|
|
|
|
|
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
|
|
|
}
|
2016-02-12 00:17:24 -08:00
|
|
|
|
|
|
|
|
impl<'a> Read for &'a RawHandle {
|
|
|
|
|
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
|
|
|
|
|
(**self).read(buf)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn read_to_end(&mut self, buf: &mut Vec<u8>) -> io::Result<usize> {
|
|
|
|
|
unsafe { read_to_end_uninitialized(self, buf) }
|
|
|
|
|
}
|
|
|
|
|
}
|