2019-08-21 20:42:48 -04:00
|
|
|
use super::dispatch_minimal::MinimalOp;
|
|
|
|
use crate::deno_error;
|
2019-11-07 11:11:15 -05:00
|
|
|
use crate::deno_error::bad_resource;
|
2019-10-11 14:41:54 -04:00
|
|
|
use crate::ops::minimal_op;
|
2019-08-21 20:42:48 -04:00
|
|
|
use crate::resources;
|
2019-11-07 11:11:15 -05:00
|
|
|
use crate::resources::CliResource;
|
|
|
|
use crate::resources::DenoAsyncRead;
|
|
|
|
use crate::resources::DenoAsyncWrite;
|
2019-10-11 14:41:54 -04:00
|
|
|
use crate::state::ThreadSafeState;
|
|
|
|
use deno::*;
|
2019-08-21 20:42:48 -04:00
|
|
|
use futures::Future;
|
2019-11-07 11:11:15 -05:00
|
|
|
use futures::Poll;
|
2019-08-21 20:42:48 -04:00
|
|
|
|
2019-10-11 14:41:54 -04:00
|
|
|
pub fn init(i: &mut Isolate, s: &ThreadSafeState) {
|
|
|
|
i.register_op("read", s.core_op(minimal_op(op_read)));
|
|
|
|
i.register_op("write", s.core_op(minimal_op(op_write)));
|
|
|
|
}
|
|
|
|
|
2019-11-07 11:11:15 -05:00
|
|
|
#[derive(Debug, PartialEq)]
|
|
|
|
enum IoState {
|
|
|
|
Pending,
|
|
|
|
Done,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Tries to read some bytes directly into the given `buf` in asynchronous
|
|
|
|
/// manner, returning a future type.
|
|
|
|
///
|
|
|
|
/// The returned future will resolve to both the I/O stream and the buffer
|
|
|
|
/// as well as the number of bytes read once the read operation is completed.
|
|
|
|
pub fn read<T>(rid: ResourceId, buf: T) -> Read<T>
|
|
|
|
where
|
|
|
|
T: AsMut<[u8]>,
|
|
|
|
{
|
|
|
|
Read {
|
|
|
|
rid,
|
|
|
|
buf,
|
|
|
|
state: IoState::Pending,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A future which can be used to easily read available number of bytes to fill
|
|
|
|
/// a buffer.
|
|
|
|
///
|
|
|
|
/// Created by the [`read`] function.
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Read<T> {
|
|
|
|
rid: ResourceId,
|
|
|
|
buf: T,
|
|
|
|
state: IoState,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> Future for Read<T>
|
|
|
|
where
|
|
|
|
T: AsMut<[u8]>,
|
|
|
|
{
|
|
|
|
type Item = usize;
|
|
|
|
type Error = ErrBox;
|
|
|
|
|
|
|
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
|
|
|
if self.state == IoState::Done {
|
|
|
|
panic!("poll a Read after it's done");
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut table = resources::lock_resource_table();
|
|
|
|
let resource = table
|
|
|
|
.get_mut::<CliResource>(self.rid)
|
|
|
|
.ok_or_else(bad_resource)?;
|
|
|
|
let nread = try_ready!(resource.poll_read(&mut self.buf.as_mut()[..]));
|
|
|
|
self.state = IoState::Done;
|
|
|
|
Ok(nread.into())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-21 20:42:48 -04:00
|
|
|
pub fn op_read(rid: i32, zero_copy: Option<PinnedBuf>) -> Box<MinimalOp> {
|
|
|
|
debug!("read rid={}", rid);
|
|
|
|
let zero_copy = match zero_copy {
|
|
|
|
None => {
|
|
|
|
return Box::new(futures::future::err(deno_error::no_buffer_specified()))
|
|
|
|
}
|
|
|
|
Some(buf) => buf,
|
|
|
|
};
|
2019-09-10 00:59:40 -04:00
|
|
|
|
2019-11-07 11:11:15 -05:00
|
|
|
let fut = read(rid as u32, zero_copy)
|
|
|
|
.map_err(ErrBox::from)
|
|
|
|
.and_then(move |nread| Ok(nread as i32));
|
|
|
|
|
|
|
|
Box::new(fut)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A future used to write some data to a stream.
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Write<T> {
|
|
|
|
rid: ResourceId,
|
|
|
|
buf: T,
|
|
|
|
state: IoState,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a future that will write some of the buffer `buf` to
|
|
|
|
/// the stream resource with `rid`.
|
|
|
|
///
|
|
|
|
/// Any error which happens during writing will cause both the stream and the
|
|
|
|
/// buffer to get destroyed.
|
|
|
|
pub fn write<T>(rid: ResourceId, buf: T) -> Write<T>
|
|
|
|
where
|
|
|
|
T: AsRef<[u8]>,
|
|
|
|
{
|
|
|
|
Write {
|
|
|
|
rid,
|
|
|
|
buf,
|
|
|
|
state: IoState::Pending,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This is almost the same implementation as in tokio, difference is
|
|
|
|
/// that error type is `ErrBox` instead of `std::io::Error`.
|
|
|
|
impl<T> Future for Write<T>
|
|
|
|
where
|
|
|
|
T: AsRef<[u8]>,
|
|
|
|
{
|
|
|
|
type Item = usize;
|
|
|
|
type Error = ErrBox;
|
|
|
|
|
|
|
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
|
|
|
if self.state == IoState::Done {
|
|
|
|
panic!("poll a Read after it's done");
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut table = resources::lock_resource_table();
|
|
|
|
let resource = table
|
|
|
|
.get_mut::<CliResource>(self.rid)
|
|
|
|
.ok_or_else(bad_resource)?;
|
|
|
|
let nwritten = try_ready!(resource.poll_write(self.buf.as_ref()));
|
|
|
|
self.state = IoState::Done;
|
|
|
|
Ok(nwritten.into())
|
2019-08-21 20:42:48 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn op_write(rid: i32, zero_copy: Option<PinnedBuf>) -> Box<MinimalOp> {
|
|
|
|
debug!("write rid={}", rid);
|
|
|
|
let zero_copy = match zero_copy {
|
|
|
|
None => {
|
|
|
|
return Box::new(futures::future::err(deno_error::no_buffer_specified()))
|
|
|
|
}
|
|
|
|
Some(buf) => buf,
|
|
|
|
};
|
2019-09-10 00:59:40 -04:00
|
|
|
|
2019-11-07 11:11:15 -05:00
|
|
|
let fut = write(rid as u32, zero_copy)
|
|
|
|
.map_err(ErrBox::from)
|
|
|
|
.and_then(move |nwritten| Ok(nwritten as i32));
|
|
|
|
|
|
|
|
Box::new(fut)
|
2019-08-21 20:42:48 -04:00
|
|
|
}
|