12
12
13
13
#[ allow( non_camel_case_types) ] ;
14
14
15
+ use io:: IoError ;
16
+ use io;
15
17
use libc;
18
+ use ops:: Drop ;
19
+ use option:: { Some , None , Option } ;
16
20
use os;
17
- use prelude:: * ;
18
- use super :: super :: * ;
19
-
20
- #[ cfg( windows) ]
21
- fn get_err ( errno : i32 ) -> ( IoErrorKind , & ' static str ) {
22
- match errno {
23
- libc:: EOF => ( EndOfFile , "end of file" ) ,
24
- _ => ( OtherIoError , "unknown error" ) ,
25
- }
26
- }
27
-
28
- #[ cfg( not( windows) ) ]
29
- fn get_err ( errno : i32 ) -> ( IoErrorKind , & ' static str ) {
30
- // XXX: this should probably be a bit more descriptive...
31
- match errno {
32
- libc:: EOF => ( EndOfFile , "end of file" ) ,
33
-
34
- // These two constants can have the same value on some systems, but
35
- // different values on others, so we can't use a match clause
36
- x if x == libc:: EAGAIN || x == libc:: EWOULDBLOCK =>
37
- ( ResourceUnavailable , "resource temporarily unavailable" ) ,
38
-
39
- _ => ( OtherIoError , "unknown error" ) ,
40
- }
41
- }
42
-
43
- fn raise_error ( ) {
44
- let ( kind, desc) = get_err ( os:: errno ( ) as i32 ) ;
45
- io_error:: cond. raise ( IoError {
46
- kind : kind,
47
- desc : desc,
48
- detail : Some ( os:: last_os_error ( ) )
49
- } ) ;
50
- }
21
+ use ptr:: RawPtr ;
22
+ use result:: { Result , Ok , Err } ;
23
+ use rt:: rtio;
24
+ use vec:: ImmutableVector ;
51
25
52
26
fn keep_going ( data : & [ u8 ] , f : & fn ( * u8 , uint ) -> i64 ) -> i64 {
53
27
#[ cfg( windows) ] static eintr: int = 0 ; // doesn't matter
@@ -95,10 +69,8 @@ impl FileDesc {
95
69
pub fn new ( fd : fd_t , close_on_drop : bool ) -> FileDesc {
96
70
FileDesc { fd : fd, close_on_drop : close_on_drop }
97
71
}
98
- }
99
72
100
- impl Reader for FileDesc {
101
- fn read ( & mut self , buf : & mut [ u8 ] ) -> Option < uint > {
73
+ fn inner_read ( & mut self , buf : & mut [ u8 ] ) -> Result < uint , IoError > {
102
74
#[ cfg( windows) ] type rlen = libc:: c_uint ;
103
75
#[ cfg( not( windows) ) ] type rlen = libc:: size_t ;
104
76
let ret = do keep_going ( buf) |buf, len| {
@@ -107,20 +79,14 @@ impl Reader for FileDesc {
107
79
}
108
80
} ;
109
81
if ret == 0 {
110
- None
82
+ Err ( io :: standard_error ( io :: EndOfFile ) )
111
83
} else if ret < 0 {
112
- raise_error ( ) ;
113
- None
84
+ Err ( super :: last_error ( ) )
114
85
} else {
115
- Some ( ret as uint )
86
+ Ok ( ret as uint )
116
87
}
117
88
}
118
-
119
- fn eof ( & mut self ) -> bool { false }
120
- }
121
-
122
- impl Writer for FileDesc {
123
- fn write ( & mut self , buf : & [ u8 ] ) {
89
+ fn inner_write ( & mut self , buf : & [ u8 ] ) -> Result < ( ) , IoError > {
124
90
#[ cfg( windows) ] type wlen = libc:: c_uint ;
125
91
#[ cfg( not( windows) ) ] type wlen = libc:: size_t ;
126
92
let ret = do keep_going ( buf) |buf, len| {
@@ -129,14 +95,84 @@ impl Writer for FileDesc {
129
95
}
130
96
} ;
131
97
if ret < 0 {
132
- raise_error ( ) ;
98
+ Err ( super :: last_error ( ) )
99
+ } else {
100
+ Ok ( ( ) )
133
101
}
134
102
}
135
103
}
136
104
105
+ impl io:: Reader for FileDesc {
106
+ fn read ( & mut self , buf : & mut [ u8 ] ) -> Option < uint > {
107
+ match self . inner_read ( buf) { Ok ( n) => Some ( n) , Err ( * ) => None }
108
+ }
109
+ fn eof ( & mut self ) -> bool { false }
110
+ }
111
+
112
+ impl io:: Writer for FileDesc {
113
+ fn write ( & mut self , buf : & [ u8 ] ) {
114
+ self . inner_write ( buf) ;
115
+ }
116
+ }
117
+
118
+ impl rtio:: RtioFileStream for FileDesc {
119
+ fn read ( & mut self , buf : & mut [ u8 ] ) -> Result < int , IoError > {
120
+ self . inner_read ( buf) . map ( |i| i as int )
121
+ }
122
+ fn write ( & mut self , buf : & [ u8 ] ) -> Result < ( ) , IoError > {
123
+ self . inner_write ( buf)
124
+ }
125
+ fn pread ( & mut self , _buf : & mut [ u8 ] , _offset : u64 ) -> Result < int , IoError > {
126
+ Err ( super :: unimpl ( ) )
127
+ }
128
+ fn pwrite ( & mut self , _buf : & [ u8 ] , _offset : u64 ) -> Result < ( ) , IoError > {
129
+ Err ( super :: unimpl ( ) )
130
+ }
131
+ fn seek ( & mut self , _pos : i64 , _whence : io:: SeekStyle ) -> Result < u64 , IoError > {
132
+ Err ( super :: unimpl ( ) )
133
+ }
134
+ fn tell ( & self ) -> Result < u64 , IoError > {
135
+ Err ( super :: unimpl ( ) )
136
+ }
137
+ fn fsync ( & mut self ) -> Result < ( ) , IoError > {
138
+ Err ( super :: unimpl ( ) )
139
+ }
140
+ fn datasync ( & mut self ) -> Result < ( ) , IoError > {
141
+ Err ( super :: unimpl ( ) )
142
+ }
143
+ fn truncate ( & mut self , _offset : i64 ) -> Result < ( ) , IoError > {
144
+ Err ( super :: unimpl ( ) )
145
+ }
146
+ }
147
+
148
+ impl rtio:: RtioPipe for FileDesc {
149
+ fn read ( & mut self , buf : & mut [ u8 ] ) -> Result < uint , IoError > {
150
+ self . inner_read ( buf)
151
+ }
152
+ fn write ( & mut self , buf : & [ u8 ] ) -> Result < ( ) , IoError > {
153
+ self . inner_write ( buf)
154
+ }
155
+ }
156
+
157
+ impl rtio:: RtioTTY for FileDesc {
158
+ fn read ( & mut self , buf : & mut [ u8 ] ) -> Result < uint , IoError > {
159
+ self . inner_read ( buf)
160
+ }
161
+ fn write ( & mut self , buf : & [ u8 ] ) -> Result < ( ) , IoError > {
162
+ self . inner_write ( buf)
163
+ }
164
+ fn set_raw ( & mut self , _raw : bool ) -> Result < ( ) , IoError > {
165
+ Err ( super :: unimpl ( ) )
166
+ }
167
+ fn get_winsize ( & mut self ) -> Result < ( int , int ) , IoError > {
168
+ Err ( super :: unimpl ( ) )
169
+ }
170
+ }
171
+
137
172
impl Drop for FileDesc {
138
173
fn drop ( & mut self ) {
139
- if self . close_on_drop {
174
+ // closing stdio file handles makes no sense, so never do it
175
+ if self . close_on_drop && self . fd > libc:: STDERR_FILENO {
140
176
unsafe { libc:: close ( self . fd ) ; }
141
177
}
142
178
}
@@ -154,67 +190,72 @@ impl CFile {
154
190
pub fn new ( file : * libc:: FILE ) -> CFile { CFile { file : file } }
155
191
}
156
192
157
- impl Reader for CFile {
158
- fn read ( & mut self , buf : & mut [ u8 ] ) -> Option < uint > {
193
+ impl rtio :: RtioFileStream for CFile {
194
+ fn read ( & mut self , buf : & mut [ u8 ] ) -> Result < int , IoError > {
159
195
let ret = do keep_going ( buf) |buf, len| {
160
196
unsafe {
161
197
libc:: fread ( buf as * mut libc:: c_void , 1 , len as libc:: size_t ,
162
198
self . file ) as i64
163
199
}
164
200
} ;
165
201
if ret == 0 {
166
- None
202
+ Err ( io :: standard_error ( io :: EndOfFile ) )
167
203
} else if ret < 0 {
168
- raise_error ( ) ;
169
- None
204
+ Err ( super :: last_error ( ) )
170
205
} else {
171
- Some ( ret as uint )
206
+ Ok ( ret as int )
172
207
}
173
208
}
174
209
175
- fn eof ( & mut self ) -> bool {
176
- unsafe { libc:: feof ( self . file ) != 0 }
177
- }
178
- }
179
-
180
- impl Writer for CFile {
181
- fn write ( & mut self , buf : & [ u8 ] ) {
210
+ fn write ( & mut self , buf : & [ u8 ] ) -> Result < ( ) , IoError > {
182
211
let ret = do keep_going ( buf) |buf, len| {
183
212
unsafe {
184
213
libc:: fwrite ( buf as * libc:: c_void , 1 , len as libc:: size_t ,
185
214
self . file ) as i64
186
215
}
187
216
} ;
188
217
if ret < 0 {
189
- raise_error ( ) ;
218
+ Err ( super :: last_error ( ) )
219
+ } else {
220
+ Ok ( ( ) )
190
221
}
191
222
}
192
223
193
- fn flush ( & mut self ) {
194
- if unsafe { libc:: fflush ( self . file ) } < 0 {
195
- raise_error ( ) ;
224
+ fn pread ( & mut self , _buf : & mut [ u8 ] , _offset : u64 ) -> Result < int , IoError > {
225
+ Err ( super :: unimpl ( ) )
226
+ }
227
+ fn pwrite ( & mut self , _buf : & [ u8 ] , _offset : u64 ) -> Result < ( ) , IoError > {
228
+ Err ( super :: unimpl ( ) )
229
+ }
230
+ fn seek ( & mut self , pos : i64 , style : io:: SeekStyle ) -> Result < u64 , IoError > {
231
+ let whence = match style {
232
+ io:: SeekSet => libc:: SEEK_SET ,
233
+ io:: SeekEnd => libc:: SEEK_END ,
234
+ io:: SeekCur => libc:: SEEK_CUR ,
235
+ } ;
236
+ let n = unsafe { libc:: fseek ( self . file , pos as libc:: c_long , whence) } ;
237
+ if n < 0 {
238
+ Err ( super :: last_error ( ) )
239
+ } else {
240
+ Ok ( n as u64 )
196
241
}
197
242
}
198
- }
199
-
200
- impl Seek for CFile {
201
- fn tell ( & self ) -> u64 {
243
+ fn tell ( & self ) -> Result < u64 , IoError > {
202
244
let ret = unsafe { libc:: ftell ( self . file ) } ;
203
245
if ret < 0 {
204
- raise_error ( ) ;
246
+ Err ( super :: last_error ( ) )
247
+ } else {
248
+ Ok ( ret as u64 )
205
249
}
206
- return ret as u64 ;
207
250
}
208
-
209
- fn seek ( & mut self , pos : i64 , style : SeekStyle ) {
210
- let whence = match style {
211
- SeekSet => libc:: SEEK_SET ,
212
- SeekEnd => libc:: SEEK_END ,
213
- SeekCur => libc:: SEEK_CUR ,
214
- } ;
215
- if unsafe { libc:: fseek ( self . file , pos as libc:: c_long , whence) } < 0 {
216
- raise_error ( ) ;
217
- }
251
+ fn fsync ( & mut self ) -> Result < ( ) , IoError > {
252
+ Err ( super :: unimpl ( ) )
253
+ }
254
+ fn datasync ( & mut self ) -> Result < ( ) , IoError > {
255
+ Err ( super :: unimpl ( ) )
256
+ }
257
+ fn truncate ( & mut self , _offset : i64 ) -> Result < ( ) , IoError > {
258
+ Err ( super :: unimpl ( ) )
218
259
}
219
260
}
220
261
@@ -228,9 +269,9 @@ impl Drop for CFile {
228
269
mod tests {
229
270
use libc;
230
271
use os;
231
- use prelude :: * ;
232
- use io :: { io_error , SeekSet } ;
233
- use super :: * ;
272
+ use io :: { io_error , SeekSet , Writer , Reader } ;
273
+ use result :: Ok ;
274
+ use super :: { CFile , FileDesc } ;
234
275
235
276
#[ ignore( cfg( target_os = "freebsd" ) ) ] // hmm, maybe pipes have a tiny buffer
236
277
fn test_file_desc ( ) {
@@ -241,10 +282,10 @@ mod tests {
241
282
let mut reader = FileDesc :: new ( input, true ) ;
242
283
let mut writer = FileDesc :: new ( out, true ) ;
243
284
244
- writer. write ( bytes ! ( "test" ) ) ;
285
+ writer. inner_write ( bytes ! ( "test" ) ) ;
245
286
let mut buf = [ 0u8 , ..4 ] ;
246
- match reader. read ( buf) {
247
- Some ( 4 ) => {
287
+ match reader. inner_read ( buf) {
288
+ Ok ( 4 ) => {
248
289
assert_eq ! ( buf[ 0 ] , 't' as u8 ) ;
249
290
assert_eq ! ( buf[ 1 ] , 'e' as u8 ) ;
250
291
assert_eq ! ( buf[ 2 ] , 's' as u8 ) ;
@@ -253,17 +294,8 @@ mod tests {
253
294
r => fail ! ( "invalid read: {:?}" , r)
254
295
}
255
296
256
- let mut raised = false ;
257
- do io_error:: cond. trap ( |_| { raised = true ; } ) . inside {
258
- writer. read ( buf) ;
259
- }
260
- assert ! ( raised) ;
261
-
262
- raised = false ;
263
- do io_error:: cond. trap ( |_| { raised = true ; } ) . inside {
264
- reader. write ( buf) ;
265
- }
266
- assert ! ( raised) ;
297
+ assert ! ( writer. inner_read( buf) . is_err( ) ) ;
298
+ assert ! ( reader. inner_write( buf) . is_err( ) ) ;
267
299
}
268
300
}
269
301
@@ -278,7 +310,7 @@ mod tests {
278
310
let mut buf = [ 0u8 , ..4 ] ;
279
311
file. seek ( 0 , SeekSet ) ;
280
312
match file. read ( buf) {
281
- Some ( 4 ) => {
313
+ Ok ( 4 ) => {
282
314
assert_eq ! ( buf[ 0 ] , 't' as u8 ) ;
283
315
assert_eq ! ( buf[ 1 ] , 'e' as u8 ) ;
284
316
assert_eq ! ( buf[ 2 ] , 's' as u8 ) ;
0 commit comments