Skip to content

Commit 58b5c61

Browse files
committed
auto merge of rust-lang#10457 : alexcrichton/rust/native-io, r=brson
This commit re-organizes the io::native module slightly in order to have a working implementation of rtio::IoFactory which uses native implementations. The goal is to seamlessly multiplex among libuv/native implementations wherever necessary. Right now most of the native I/O is unimplemented, but we have existing bindings for file descriptors and processes which have been hooked up. What this means is that you can now invoke println!() from libstd with no local task, no local scheduler, and even without libuv. There's still plenty of work to do on the native I/O factory, but this is the first steps into making it an official portion of the standard library. I don't expect anyone to reach into io::native directly, but rather only std::io primitives will be used. Each std::io interface seamlessly falls back onto the native I/O implementation if the local scheduler doesn't have a libuv one (hurray trait ojects!)
2 parents 8a3b35f + 9bcf557 commit 58b5c61

13 files changed

+476
-356
lines changed

src/librustuv/macros.rs

+5-5
Original file line numberDiff line numberDiff line change
@@ -33,9 +33,9 @@ macro_rules! get_handle_to_current_scheduler(
3333
)
3434

3535
pub fn dumb_println(args: &fmt::Arguments) {
36-
use std::io::native::stdio::stderr;
37-
use std::io::Writer;
38-
39-
let mut out = stderr();
40-
fmt::writeln(&mut out as &mut Writer, args);
36+
use std::io::native::file::FileDesc;
37+
use std::io;
38+
use std::libc;
39+
let mut out = FileDesc::new(libc::STDERR_FILENO, false);
40+
fmt::writeln(&mut out as &mut io::Writer, args);
4141
}

src/libstd/io/mod.rs

+1-18
Original file line numberDiff line numberDiff line change
@@ -306,24 +306,7 @@ pub mod timer;
306306
/// Buffered I/O wrappers
307307
pub mod buffered;
308308

309-
/// Thread-blocking implementations
310-
pub mod native {
311-
/// Posix file I/O
312-
pub mod file;
313-
/// Process spawning and child management
314-
pub mod process;
315-
/// Posix stdio
316-
pub mod stdio;
317-
318-
/// Sockets
319-
/// # XXX - implement this
320-
pub mod net {
321-
pub mod tcp { }
322-
pub mod udp { }
323-
#[cfg(unix)]
324-
pub mod unix { }
325-
}
326-
}
309+
pub mod native;
327310

328311
/// Signal handling
329312
pub mod signal;

src/libstd/io/native/file.rs

+132-100
Original file line numberDiff line numberDiff line change
@@ -12,42 +12,16 @@
1212
1313
#[allow(non_camel_case_types)];
1414

15+
use io::IoError;
16+
use io;
1517
use libc;
18+
use ops::Drop;
19+
use option::{Some, None, Option};
1620
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;
5125

5226
fn keep_going(data: &[u8], f: &fn(*u8, uint) -> i64) -> i64 {
5327
#[cfg(windows)] static eintr: int = 0; // doesn't matter
@@ -95,10 +69,8 @@ impl FileDesc {
9569
pub fn new(fd: fd_t, close_on_drop: bool) -> FileDesc {
9670
FileDesc { fd: fd, close_on_drop: close_on_drop }
9771
}
98-
}
9972

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> {
10274
#[cfg(windows)] type rlen = libc::c_uint;
10375
#[cfg(not(windows))] type rlen = libc::size_t;
10476
let ret = do keep_going(buf) |buf, len| {
@@ -107,20 +79,14 @@ impl Reader for FileDesc {
10779
}
10880
};
10981
if ret == 0 {
110-
None
82+
Err(io::standard_error(io::EndOfFile))
11183
} else if ret < 0 {
112-
raise_error();
113-
None
84+
Err(super::last_error())
11485
} else {
115-
Some(ret as uint)
86+
Ok(ret as uint)
11687
}
11788
}
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> {
12490
#[cfg(windows)] type wlen = libc::c_uint;
12591
#[cfg(not(windows))] type wlen = libc::size_t;
12692
let ret = do keep_going(buf) |buf, len| {
@@ -129,14 +95,84 @@ impl Writer for FileDesc {
12995
}
13096
};
13197
if ret < 0 {
132-
raise_error();
98+
Err(super::last_error())
99+
} else {
100+
Ok(())
133101
}
134102
}
135103
}
136104

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+
137172
impl Drop for FileDesc {
138173
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 {
140176
unsafe { libc::close(self.fd); }
141177
}
142178
}
@@ -154,67 +190,72 @@ impl CFile {
154190
pub fn new(file: *libc::FILE) -> CFile { CFile { file: file } }
155191
}
156192

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> {
159195
let ret = do keep_going(buf) |buf, len| {
160196
unsafe {
161197
libc::fread(buf as *mut libc::c_void, 1, len as libc::size_t,
162198
self.file) as i64
163199
}
164200
};
165201
if ret == 0 {
166-
None
202+
Err(io::standard_error(io::EndOfFile))
167203
} else if ret < 0 {
168-
raise_error();
169-
None
204+
Err(super::last_error())
170205
} else {
171-
Some(ret as uint)
206+
Ok(ret as int)
172207
}
173208
}
174209

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> {
182211
let ret = do keep_going(buf) |buf, len| {
183212
unsafe {
184213
libc::fwrite(buf as *libc::c_void, 1, len as libc::size_t,
185214
self.file) as i64
186215
}
187216
};
188217
if ret < 0 {
189-
raise_error();
218+
Err(super::last_error())
219+
} else {
220+
Ok(())
190221
}
191222
}
192223

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)
196241
}
197242
}
198-
}
199-
200-
impl Seek for CFile {
201-
fn tell(&self) -> u64 {
243+
fn tell(&self) -> Result<u64, IoError> {
202244
let ret = unsafe { libc::ftell(self.file) };
203245
if ret < 0 {
204-
raise_error();
246+
Err(super::last_error())
247+
} else {
248+
Ok(ret as u64)
205249
}
206-
return ret as u64;
207250
}
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())
218259
}
219260
}
220261

@@ -228,9 +269,9 @@ impl Drop for CFile {
228269
mod tests {
229270
use libc;
230271
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};
234275

235276
#[ignore(cfg(target_os = "freebsd"))] // hmm, maybe pipes have a tiny buffer
236277
fn test_file_desc() {
@@ -241,10 +282,10 @@ mod tests {
241282
let mut reader = FileDesc::new(input, true);
242283
let mut writer = FileDesc::new(out, true);
243284

244-
writer.write(bytes!("test"));
285+
writer.inner_write(bytes!("test"));
245286
let mut buf = [0u8, ..4];
246-
match reader.read(buf) {
247-
Some(4) => {
287+
match reader.inner_read(buf) {
288+
Ok(4) => {
248289
assert_eq!(buf[0], 't' as u8);
249290
assert_eq!(buf[1], 'e' as u8);
250291
assert_eq!(buf[2], 's' as u8);
@@ -253,17 +294,8 @@ mod tests {
253294
r => fail!("invalid read: {:?}", r)
254295
}
255296

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());
267299
}
268300
}
269301

@@ -278,7 +310,7 @@ mod tests {
278310
let mut buf = [0u8, ..4];
279311
file.seek(0, SeekSet);
280312
match file.read(buf) {
281-
Some(4) => {
313+
Ok(4) => {
282314
assert_eq!(buf[0], 't' as u8);
283315
assert_eq!(buf[1], 'e' as u8);
284316
assert_eq!(buf[2], 's' as u8);

0 commit comments

Comments
 (0)