mirror of
https://github.com/9fans/plan9port.git
synced 2025-01-30 11:54:57 +00:00
75 lines
1.1 KiB
C
75 lines
1.1 KiB
C
|
#include "threadimpl.h"
|
||
|
|
||
|
enum
|
||
|
{
|
||
|
STACK = 8192,
|
||
|
};
|
||
|
|
||
|
void
|
||
|
iointerrupt(Ioproc *io)
|
||
|
{
|
||
|
if(!io->inuse)
|
||
|
return;
|
||
|
threadint(io->tid);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
xioproc(void *a)
|
||
|
{
|
||
|
Ioproc *io, *x;
|
||
|
io = a;
|
||
|
/*
|
||
|
* first recvp acquires the ioproc.
|
||
|
* second tells us that the data is ready.
|
||
|
*/
|
||
|
for(;;){
|
||
|
while(recv(io->c, &x) == -1)
|
||
|
;
|
||
|
if(x == 0) /* our cue to leave */
|
||
|
break;
|
||
|
assert(x == io);
|
||
|
|
||
|
/* caller is now committed -- even if interrupted he'll return */
|
||
|
while(recv(io->creply, &x) == -1)
|
||
|
;
|
||
|
if(x == 0) /* caller backed out */
|
||
|
continue;
|
||
|
assert(x == io);
|
||
|
|
||
|
io->ret = io->op(&io->arg);
|
||
|
if(io->ret < 0)
|
||
|
rerrstr(io->err, sizeof io->err);
|
||
|
while(send(io->creply, &io) == -1)
|
||
|
;
|
||
|
while(recv(io->creply, &x) == -1)
|
||
|
;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Ioproc*
|
||
|
ioproc(void)
|
||
|
{
|
||
|
Ioproc *io;
|
||
|
|
||
|
io = mallocz(sizeof(*io), 1);
|
||
|
if(io == nil)
|
||
|
sysfatal("ioproc malloc: %r");
|
||
|
io->c = chancreate(sizeof(void*), 0);
|
||
|
io->creply = chancreate(sizeof(void*), 0);
|
||
|
io->tid = proccreate(xioproc, io, STACK);
|
||
|
return io;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
closeioproc(Ioproc *io)
|
||
|
{
|
||
|
if(io == nil)
|
||
|
return;
|
||
|
iointerrupt(io);
|
||
|
while(send(io->c, 0) == -1)
|
||
|
;
|
||
|
chanfree(io->c);
|
||
|
chanfree(io->creply);
|
||
|
free(io);
|
||
|
}
|