221 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Factor
		
	
	
		
		
			
		
	
	
			221 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Factor
		
	
	
|  | ! Copyright (C) 2004, 2008 Slava Pestov. | ||
|  | ! See http://factorcode.org/license.txt for BSD license. | ||
|  | USING: alien alien.c-types generic assocs kernel kernel.private | ||
|  | math io.ports sequences strings structs sbufs threads unix | ||
|  | vectors io.buffers io.backend io.encodings math.parser | ||
|  | continuations system libc qualified namespaces io.timeouts | ||
|  | io.encodings.utf8 destructors accessors summary combinators | ||
|  | locals ;
 | ||
|  | QUALIFIED: io | ||
|  | IN: io.unix.backend | ||
|  | 
 | ||
|  | GENERIC: handle-fd ( handle -- fd )
 | ||
|  | 
 | ||
|  | TUPLE: fd fd disposed ;
 | ||
|  | 
 | ||
|  | : init-fd ( fd -- fd )
 | ||
|  |     [ | ||
|  |         |dispose | ||
|  |         dup fd>> F_SETFL O_NONBLOCK fcntl io-error | ||
|  |         dup fd>> F_SETFD FD_CLOEXEC fcntl io-error | ||
|  |     ] with-destructors ;
 | ||
|  | 
 | ||
|  | : <fd> ( n -- fd )
 | ||
|  |     #! We drop the error code rather than calling io-error, | ||
|  |     #! since on OS X 10.3, this operation fails from init-io | ||
|  |     #! when running the Factor.app (presumably because fd 0 and | ||
|  |     #! 1 are closed). | ||
|  |     f fd boa ;
 | ||
|  | 
 | ||
|  | M: fd dispose | ||
|  |     dup disposed>> [ drop ] [ | ||
|  |         [ cancel-operation ] | ||
|  |         [ t >>disposed drop ] | ||
|  |         [ fd>> close-file ] | ||
|  |         tri
 | ||
|  |     ] if ;
 | ||
|  | 
 | ||
|  | M: fd handle-fd dup check-disposed fd>> ;
 | ||
|  | 
 | ||
|  | ! I/O multiplexers | ||
|  | TUPLE: mx fd reads writes ;
 | ||
|  | 
 | ||
|  | : new-mx ( class -- obj )
 | ||
|  |     new
 | ||
|  |         H{ } clone >>reads | ||
|  |         H{ } clone >>writes ; inline
 | ||
|  | 
 | ||
|  | GENERIC: add-input-callback ( thread fd mx -- )
 | ||
|  | 
 | ||
|  | M: mx add-input-callback reads>> push-at ;
 | ||
|  | 
 | ||
|  | GENERIC: add-output-callback ( thread fd mx -- )
 | ||
|  | 
 | ||
|  | M: mx add-output-callback writes>> push-at ;
 | ||
|  | 
 | ||
|  | GENERIC: remove-input-callbacks ( fd mx -- callbacks )
 | ||
|  | 
 | ||
|  | M: mx remove-input-callbacks reads>> delete-at* drop ;
 | ||
|  | 
 | ||
|  | GENERIC: remove-output-callbacks ( fd mx -- callbacks )
 | ||
|  | 
 | ||
|  | M: mx remove-output-callbacks writes>> delete-at* drop ;
 | ||
|  | 
 | ||
|  | GENERIC: wait-for-events ( ms mx -- )
 | ||
|  | 
 | ||
|  | : input-available ( fd mx -- )
 | ||
|  |     remove-input-callbacks [ resume ] each ;
 | ||
|  | 
 | ||
|  | : output-available ( fd mx -- )
 | ||
|  |     remove-output-callbacks [ resume ] each ;
 | ||
|  | 
 | ||
|  | M: fd cancel-operation ( fd -- )
 | ||
|  |     dup disposed>> [ drop ] [ | ||
|  |         fd>> | ||
|  |         mx get-global
 | ||
|  |         [ remove-input-callbacks [ t swap resume-with ] each ] | ||
|  |         [ remove-output-callbacks [ t swap resume-with ] each ] | ||
|  |         2bi
 | ||
|  |     ] if ;
 | ||
|  | 
 | ||
|  | SYMBOL: +retry+ ! just try the operation again without blocking | ||
|  | SYMBOL: +input+ | ||
|  | SYMBOL: +output+ | ||
|  | 
 | ||
|  | ERROR: io-timeout ;
 | ||
|  | 
 | ||
|  | M: io-timeout summary drop "I/O operation timed out" ;
 | ||
|  | 
 | ||
|  | : wait-for-fd ( handle event -- )
 | ||
|  |     dup +retry+ eq? [ 2drop ] [ | ||
|  |         [ | ||
|  |             >r | ||
|  |             swap handle-fd | ||
|  |             mx get-global
 | ||
|  |             r> { | ||
|  |                 { +input+ [ add-input-callback ] } | ||
|  |                 { +output+ [ add-output-callback ] } | ||
|  |             } case
 | ||
|  |         ] curry "I/O" suspend nip [ io-timeout ] when
 | ||
|  |     ] if ;
 | ||
|  | 
 | ||
|  | : wait-for-port ( port event -- )
 | ||
|  |     [ >r handle>> r> wait-for-fd ] curry with-timeout ;
 | ||
|  | 
 | ||
|  | ! Some general stuff | ||
|  | : file-mode OCT: 0666 ;
 | ||
|  |   | ||
|  | ! Readers | ||
|  | : (refill) ( port -- n )
 | ||
|  |     [ handle>> ] | ||
|  |     [ buffer>> buffer-end ] | ||
|  |     [ buffer>> buffer-capacity ] tri read ;
 | ||
|  | 
 | ||
|  | ! Returns an event to wait for which will ensure completion of | ||
|  | ! this request | ||
|  | GENERIC: refill ( port handle -- event/f )
 | ||
|  | 
 | ||
|  | M: fd refill | ||
|  |     fd>> over buffer>> [ buffer-end ] [ buffer-capacity ] bi read
 | ||
|  |     { | ||
|  |         { [ dup 0 >= ] [ swap buffer>> n>buffer f ] } | ||
|  |         { [ err_no EINTR = ] [ 2drop +retry+ ] } | ||
|  |         { [ err_no EAGAIN = ] [ 2drop +input+ ] } | ||
|  |         [ (io-error) ] | ||
|  |     } cond ;
 | ||
|  | 
 | ||
|  | M: unix (wait-to-read) ( port -- )
 | ||
|  |     dup
 | ||
|  |     dup handle>> dup check-disposed refill dup
 | ||
|  |     [ dupd wait-for-port (wait-to-read) ] [ 2drop ] if ;
 | ||
|  | 
 | ||
|  | ! Writers | ||
|  | GENERIC: drain ( port handle -- event/f )
 | ||
|  | 
 | ||
|  | M: fd drain | ||
|  |     fd>> over buffer>> [ buffer@ ] [ buffer-length ] bi write
 | ||
|  |     { | ||
|  |         { [ dup 0 >= ] [ | ||
|  |             over buffer>> buffer-consume | ||
|  |             buffer>> buffer-empty? f +output+ ?
 | ||
|  |         ] } | ||
|  |         { [ err_no EINTR = ] [ 2drop +retry+ ] } | ||
|  |         { [ err_no EAGAIN = ] [ 2drop +output+ ] } | ||
|  |         [ (io-error) ] | ||
|  |     } cond ;
 | ||
|  | 
 | ||
|  | M: unix (wait-to-write) ( port -- )
 | ||
|  |     dup
 | ||
|  |     dup handle>> dup check-disposed drain | ||
|  |     dup [ wait-for-port ] [ 2drop ] if ;
 | ||
|  | 
 | ||
|  | M: unix io-multiplex ( ms/f -- )
 | ||
|  |     mx get-global wait-for-events ;
 | ||
|  | 
 | ||
|  | ! On Unix, you're not supposed to set stdin to non-blocking | ||
|  | ! because the fd might be shared with another process (either | ||
|  | ! parent or child). So what we do is have the VM start a thread | ||
|  | ! which pumps data from the real stdin to a pipe. We set the | ||
|  | ! pipe to non-blocking, and read from it instead of the real | ||
|  | ! stdin. Very crufty, but it will suffice until we get native | ||
|  | ! threading support at the language level. | ||
|  | TUPLE: stdin control size data disposed ;
 | ||
|  | 
 | ||
|  | M: stdin dispose* | ||
|  |     [ | ||
|  |         [ control>> &dispose drop ] | ||
|  |         [ size>> &dispose drop ] | ||
|  |         [ data>> &dispose drop ] | ||
|  |         tri
 | ||
|  |     ] with-destructors ;
 | ||
|  | 
 | ||
|  | : wait-for-stdin ( stdin -- n )
 | ||
|  |     [ control>> CHAR: X over io:stream-write1 io:stream-flush ] | ||
|  |     [ size>> "ssize_t" heap-size swap io:stream-read *int ] | ||
|  |     bi ;
 | ||
|  | 
 | ||
|  | :: refill-stdin ( buffer stdin size -- )
 | ||
|  |     stdin data>> handle-fd buffer buffer-end size read
 | ||
|  |     dup 0 < [ | ||
|  |         drop
 | ||
|  |         err_no EINTR = [ buffer stdin size refill-stdin ] [ (io-error) ] if
 | ||
|  |     ] [ | ||
|  |         size = [ "Error reading stdin pipe" throw ] unless
 | ||
|  |         size buffer n>buffer | ||
|  |     ] if ;
 | ||
|  | 
 | ||
|  | M: stdin refill | ||
|  |     [ buffer>> ] [ dup wait-for-stdin ] bi* refill-stdin f ;
 | ||
|  | 
 | ||
|  | : control-write-fd ( -- fd ) "control_write" f dlsym *uint ;
 | ||
|  | 
 | ||
|  | : size-read-fd ( -- fd ) "size_read" f dlsym *uint ;
 | ||
|  | 
 | ||
|  | : data-read-fd ( -- fd ) "stdin_read" f dlsym *uint ;
 | ||
|  | 
 | ||
|  | : <stdin> ( -- stdin )
 | ||
|  |     stdin new
 | ||
|  |         control-write-fd <fd> <output-port> >>control | ||
|  |         size-read-fd <fd> init-fd <input-port> >>size | ||
|  |         data-read-fd <fd> >>data ;
 | ||
|  | 
 | ||
|  | M: unix (init-stdio) ( -- )
 | ||
|  |     <stdin> <input-port> | ||
|  |     1 <fd> <output-port> | ||
|  |     2 <fd> <output-port> ;
 | ||
|  | 
 | ||
|  | ! mx io-task for embedding an fd-based mx inside another mx | ||
|  | TUPLE: mx-port < port mx ;
 | ||
|  | 
 | ||
|  | : <mx-port> ( mx -- port )
 | ||
|  |     dup fd>> mx-port <port> swap >>mx ;
 | ||
|  | 
 | ||
|  | : multiplexer-error ( n -- )
 | ||
|  |     0 < [ | ||
|  |         err_no [ EAGAIN = ] [ EINTR = ] bi or
 | ||
|  |         [ (io-error) ] unless
 | ||
|  |     ] when ;
 | ||
|  | 
 | ||
|  | : ?flag ( n mask symbol -- n )
 | ||
|  |     pick rot bitand 0 > [ , ] [ drop ] if ;
 |