rust/src/lib/aio.rs

185 lines
4.9 KiB
Rust
Raw Normal View History

2011-06-16 00:04:31 -05:00
import str::sbuf;
import vec::vbuf;
native "rust" mod rustrt {
type socket;
type server;
fn aio_init();
fn aio_run();
fn aio_stop();
fn aio_connect(host: sbuf, port: int, connected: chan[socket]);
fn aio_serve(host: sbuf, port: int, acceptChan: chan[socket]) -> server;
fn aio_writedata(s: socket, buf: *u8, size: uint, status: chan[bool]);
fn aio_read(s: socket, reader: chan[u8[]]);
fn aio_close_server(s: server, status: chan[bool]);
fn aio_close_socket(s: socket);
fn aio_is_null_client(s: socket) -> bool;
}
type server = rustrt::server;
type client = rustrt::socket;
tag pending_connection {
remote(str,int);
incoming(server);
}
tag socket_event {
connected(client);
closed;
received(u8[]);
}
tag server_event {
pending(chan[chan[socket_event]]);
}
tag request {
quit;
connect(pending_connection,chan[socket_event]);
serve(str,int,chan[server_event],chan[server]);
write(client,u8[],chan[bool]);
close_server(server, chan[bool]);
close_client(client);
}
type ctx = chan[request];
fn connect_task(ip: str, portnum: int, evt: chan[socket_event]) {
let connecter: port[client] = port();
rustrt::aio_connect(str::buf(ip), portnum, chan(connecter));
let client: client;
connecter |> client;
new_client(client, evt);
}
fn new_client(client: client, evt: chan[socket_event]) {
// Start the read before notifying about the connect. This avoids a race
// condition where the receiver can close the socket before we start
// reading.
let reader: port[u8[]] = port();
rustrt::aio_read(client, chan(reader));
evt <| connected(client);
while (true) {
log "waiting for bytes";
let data: u8[];
reader |> data;
log "got some bytes";
log ivec::len[u8](data);
if (ivec::len[u8](data) == 0u) {
log "got empty buffer, bailing";
break;
}
log "got non-empty buffer, sending";
evt <| received(data);
log "sent non-empty buffer";
}
log "done reading";
evt <| closed;
log "close message sent";
}
fn accept_task(client: client, events: chan[server_event]) {
log "accept task was spawned";
let p: port[chan[socket_event]] = port();
events <| pending(chan(p));
let evt: chan[socket_event];
p |> evt;
new_client(client, evt);
log "done accepting";
}
fn server_task(ip: str, portnum: int, events: chan[server_event],
server: chan[server]) {
let accepter: port[client] = port();
server <| rustrt::aio_serve(str::buf(ip), portnum, chan(accepter));
let client: client;
while (true) {
log "preparing to accept a client";
accepter |> client;
if (rustrt::aio_is_null_client(client)) {
log "client was actually null, returning";
ret;
} else {
spawn accept_task(client, events);
}
}
}
fn request_task(c: chan[ctx]) {
// Create a port to accept IO requests on
let p: port[request] = port();
// Hand of its channel to our spawner
c <| chan(p);
log "uv run task spawned";
// Spin for requests
let req: request;
while (true) {
p |> req;
alt req {
quit. {
log "got quit message";
log "stopping libuv";
rustrt::aio_stop();
ret;
}
connect(remote(ip,portnum),client) {
spawn connect_task(ip, portnum, client);
}
serve(ip,portnum,events,server) {
spawn server_task(ip, portnum, events, server);
}
write(socket,v,status) {
rustrt::aio_writedata(socket,
ivec::to_ptr[u8](v), ivec::len[u8](v),
status);
}
close_server(server,status) {
log "closing server";
rustrt::aio_close_server(server,status);
}
close_client(client) {
log "closing client";
rustrt::aio_close_socket(client);
}
}
}
}
fn iotask(c: chan[ctx]) {
log "io task spawned";
// Initialize before accepting requests
rustrt::aio_init();
log "io task init";
// Spawn our request task
let reqtask: task = spawn request_task(c);
log "uv run task init";
// Enter IO loop. This never returns until aio_stop is called.
rustrt::aio_run();
log "waiting for request task to finish";
task::join(reqtask);
}
fn new() -> ctx {
let p: port[ctx] = port();
let t: task = spawn iotask(chan(p));
let cx: ctx;
p |> cx;
ret cx;
}
// Local Variables:
// mode: rust;
// fill-column: 78;
// indent-tabs-mode: nil
// c-basic-offset: 4
// buffer-file-coding-system: utf-8-unix
// compile-command: "make -k -C .. 2>&1 | sed -e 's/\\/x\\//x:\\//g'";
// End: