@@ -22,42 +22,43 @@ use sys::cvt;
22
22
use sys:: handle:: Handle ;
23
23
use sys_common:: io:: read_to_end_uninitialized;
24
24
25
- pub struct NoClose ( Option < Handle > ) ;
26
-
27
25
pub enum Output {
28
- Console ( NoClose ) ,
29
- Pipe ( NoClose ) ,
26
+ Console ( c :: HANDLE ) ,
27
+ Pipe ( c :: HANDLE ) ,
30
28
}
31
29
32
30
pub struct Stdin {
33
- handle : Output ,
34
31
utf8 : Mutex < io:: Cursor < Vec < u8 > > > ,
35
32
}
36
- pub struct Stdout ( Output ) ;
37
- pub struct Stderr ( Output ) ;
33
+ pub struct Stdout ;
34
+ pub struct Stderr ;
38
35
39
36
pub fn get ( handle : c:: DWORD ) -> io:: Result < Output > {
40
37
let handle = unsafe { c:: GetStdHandle ( handle) } ;
41
38
if handle == c:: INVALID_HANDLE_VALUE {
42
39
Err ( io:: Error :: last_os_error ( ) )
43
40
} else if handle. is_null ( ) {
44
- Err ( io:: Error :: new ( io:: ErrorKind :: Other ,
45
- "no stdio handle available for this process" ) )
41
+ Err ( io:: Error :: from_raw_os_error ( c:: ERROR_INVALID_HANDLE as i32 ) )
46
42
} else {
47
- let ret = NoClose :: new ( handle) ;
48
43
let mut out = 0 ;
49
44
match unsafe { c:: GetConsoleMode ( handle, & mut out) } {
50
- 0 => Ok ( Output :: Pipe ( ret ) ) ,
51
- _ => Ok ( Output :: Console ( ret ) ) ,
45
+ 0 => Ok ( Output :: Pipe ( handle ) ) ,
46
+ _ => Ok ( Output :: Console ( handle ) ) ,
52
47
}
53
48
}
54
49
}
55
50
56
- fn write ( out : & Output , data : & [ u8 ] ) -> io:: Result < usize > {
57
- let handle = match * out {
58
- Output :: Console ( ref c) => c. get ( ) . raw ( ) ,
59
- Output :: Pipe ( ref p) => return p. get ( ) . write ( data) ,
51
+ fn write ( handle : c:: DWORD , data : & [ u8 ] ) -> io:: Result < usize > {
52
+ let handle = match try!( get ( handle) ) {
53
+ Output :: Console ( c) => c,
54
+ Output :: Pipe ( p) => {
55
+ let handle = Handle :: new ( p) ;
56
+ let ret = handle. write ( data) ;
57
+ handle. into_raw ( ) ;
58
+ return ret
59
+ }
60
60
} ;
61
+
61
62
// As with stdin on windows, stdout often can't handle writes of large
62
63
// sizes. For an example, see #14940. For this reason, don't try to
63
64
// write the entire output buffer on windows.
@@ -93,18 +94,20 @@ fn write(out: &Output, data: &[u8]) -> io::Result<usize> {
93
94
94
95
impl Stdin {
95
96
pub fn new ( ) -> io:: Result < Stdin > {
96
- get ( c:: STD_INPUT_HANDLE ) . map ( |handle| {
97
- Stdin {
98
- handle : handle,
99
- utf8 : Mutex :: new ( Cursor :: new ( Vec :: new ( ) ) ) ,
100
- }
97
+ Ok ( Stdin {
98
+ utf8 : Mutex :: new ( Cursor :: new ( Vec :: new ( ) ) ) ,
101
99
} )
102
100
}
103
101
104
102
pub fn read ( & self , buf : & mut [ u8 ] ) -> io:: Result < usize > {
105
- let handle = match self . handle {
106
- Output :: Console ( ref c) => c. get ( ) . raw ( ) ,
107
- Output :: Pipe ( ref p) => return p. get ( ) . read ( buf) ,
103
+ let handle = match try!( get ( c:: STD_INPUT_HANDLE ) ) {
104
+ Output :: Console ( c) => c,
105
+ Output :: Pipe ( p) => {
106
+ let handle = Handle :: new ( p) ;
107
+ let ret = handle. read ( buf) ;
108
+ handle. into_raw ( ) ;
109
+ return ret
110
+ }
108
111
} ;
109
112
let mut utf8 = self . utf8 . lock ( ) . unwrap ( ) ;
110
113
// Read more if the buffer is empty
@@ -125,11 +128,9 @@ impl Stdin {
125
128
Ok ( utf8) => utf8. into_bytes ( ) ,
126
129
Err ( ..) => return Err ( invalid_encoding ( ) ) ,
127
130
} ;
128
- if let Output :: Console ( _) = self . handle {
129
- if let Some ( & last_byte) = data. last ( ) {
130
- if last_byte == CTRL_Z {
131
- data. pop ( ) ;
132
- }
131
+ if let Some ( & last_byte) = data. last ( ) {
132
+ if last_byte == CTRL_Z {
133
+ data. pop ( ) ;
133
134
}
134
135
}
135
136
* utf8 = Cursor :: new ( data) ;
@@ -158,11 +159,11 @@ impl<'a> Read for &'a Stdin {
158
159
159
160
impl Stdout {
160
161
pub fn new ( ) -> io:: Result < Stdout > {
161
- get ( c :: STD_OUTPUT_HANDLE ) . map ( Stdout )
162
+ Ok ( Stdout )
162
163
}
163
164
164
165
pub fn write ( & self , data : & [ u8 ] ) -> io:: Result < usize > {
165
- write ( & self . 0 , data)
166
+ write ( c :: STD_OUTPUT_HANDLE , data)
166
167
}
167
168
168
169
pub fn flush ( & self ) -> io:: Result < ( ) > {
@@ -172,11 +173,11 @@ impl Stdout {
172
173
173
174
impl Stderr {
174
175
pub fn new ( ) -> io:: Result < Stderr > {
175
- get ( c :: STD_ERROR_HANDLE ) . map ( Stderr )
176
+ Ok ( Stderr )
176
177
}
177
178
178
179
pub fn write ( & self , data : & [ u8 ] ) -> io:: Result < usize > {
179
- write ( & self . 0 , data)
180
+ write ( c :: STD_ERROR_HANDLE , data)
180
181
}
181
182
182
183
pub fn flush ( & self ) -> io:: Result < ( ) > {
@@ -197,27 +198,12 @@ impl io::Write for Stderr {
197
198
}
198
199
}
199
200
200
- impl NoClose {
201
- fn new ( handle : c:: HANDLE ) -> NoClose {
202
- NoClose ( Some ( Handle :: new ( handle) ) )
203
- }
204
-
205
- fn get ( & self ) -> & Handle { self . 0 . as_ref ( ) . unwrap ( ) }
206
- }
207
-
208
- impl Drop for NoClose {
209
- fn drop ( & mut self ) {
210
- self . 0 . take ( ) . unwrap ( ) . into_raw ( ) ;
211
- }
212
- }
213
-
214
201
impl Output {
215
- pub fn handle ( & self ) -> & Handle {
216
- let nc = match * self {
217
- Output :: Console ( ref c) => c,
218
- Output :: Pipe ( ref c) => c,
219
- } ;
220
- nc. 0 . as_ref ( ) . unwrap ( )
202
+ pub fn handle ( & self ) -> c:: HANDLE {
203
+ match * self {
204
+ Output :: Console ( c) => c,
205
+ Output :: Pipe ( c) => c,
206
+ }
221
207
}
222
208
}
223
209
0 commit comments