648 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			648 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|   | /*  $Id$
 | ||
|  | 
 | ||
|  |     Part of SWI-Prolog | ||
|  | 
 | ||
|  |     Author:        Jan Wielemaker | ||
|  |     E-mail:        jan@swi.psy.uva.nl | ||
|  |     WWW:           http://www.swi-prolog.org
 | ||
|  |     Copyright (C): 1985-2002, University of Amsterdam | ||
|  | 
 | ||
|  |     This library is free software; you can redistribute it and/or | ||
|  |     modify it under the terms of the GNU Lesser General Public | ||
|  |     License as published by the Free Software Foundation; either | ||
|  |     version 2.1 of the License, or (at your option) any later version. | ||
|  | 
 | ||
|  |     This library is distributed in the hope that it will be useful, | ||
|  |     but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
|  |     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | ||
|  |     Lesser General Public License for more details. | ||
|  | 
 | ||
|  |     You should have received a copy of the GNU Lesser General Public | ||
|  |     License along with this library; if not, write to the Free Software | ||
|  |     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||
|  | */ | ||
|  | 
 | ||
|  | #define O_DEBUG 1
 | ||
|  | 
 | ||
|  | #ifdef HAVE_CONFIG_H
 | ||
|  | #include <config.h>
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | #ifdef __GNUC__
 | ||
|  | # define alloca __builtin_alloca
 | ||
|  | #else
 | ||
|  | # if HAVE_ALLOCA_H
 | ||
|  | #  include <alloca.h>
 | ||
|  | # endif
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | #ifdef __CYGWIN__
 | ||
|  | #undef HAVE_H_ERRNO
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | #include "nonblockio.h"
 | ||
|  | 
 | ||
|  | #include <SWI-Stream.h>
 | ||
|  | #include "clib.h"
 | ||
|  | #include "error.h"
 | ||
|  | 
 | ||
|  | #include <stdio.h>
 | ||
|  | #include <stdlib.h>
 | ||
|  | #include <errno.h>
 | ||
|  | #include <sys/types.h>
 | ||
|  | #include <assert.h>
 | ||
|  | #include <string.h>
 | ||
|  | #ifdef __WINDOWS__
 | ||
|  | #include <malloc.h>
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | static atom_t ATOM_reuseaddr;		/* "reuseaddr" */ | ||
|  | static atom_t ATOM_broadcast;		/* "broadcast" */ | ||
|  | static atom_t ATOM_nodelay;		/* "nodelay" */ | ||
|  | static atom_t ATOM_dispatch;		/* "dispatch" */ | ||
|  | static atom_t ATOM_nonblock;		/* "nonblock" */ | ||
|  | static atom_t ATOM_infinite;		/* "infinite" */ | ||
|  | static atom_t ATOM_as;			/* "as" */ | ||
|  | static atom_t ATOM_atom;		/* "atom" */ | ||
|  | static atom_t ATOM_string;		/* "string" */ | ||
|  | static atom_t ATOM_codes;		/* "codes" */ | ||
|  | 
 | ||
|  | static functor_t FUNCTOR_socket1;	/* $socket(Id) */ | ||
|  | 
 | ||
|  | 
 | ||
|  | 		 /*******************************
 | ||
|  | 		 *	     CONVERSION		* | ||
|  | 		 *******************************/ | ||
|  | 
 | ||
|  | NBIO_EXPORT(int) | ||
|  | tcp_get_socket(term_t Socket, int *id) | ||
|  | { IOSTREAM *s; | ||
|  |   int socket; | ||
|  | 
 | ||
|  |   if ( PL_is_functor(Socket, FUNCTOR_socket1) ) | ||
|  |   { term_t a = PL_new_term_ref(); | ||
|  | 
 | ||
|  |     _PL_get_arg(1, Socket, a); | ||
|  |     if ( PL_get_integer(a, id) ) | ||
|  |       return TRUE; | ||
|  |   } | ||
|  | 
 | ||
|  |   if ( PL_get_stream_handle(Socket, &s) ) | ||
|  |   { socket = (int)(intptr_t)s->handle; | ||
|  | 
 | ||
|  |     *id = socket; | ||
|  |     return TRUE; | ||
|  |   } | ||
|  | 
 | ||
|  |   return pl_error(NULL, 0, NULL, ERR_ARGTYPE, -1, Socket, "socket"); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static int | ||
|  | tcp_unify_socket(term_t Socket, int id) | ||
|  | { return PL_unify_term(Socket, | ||
|  | 		       PL_FUNCTOR, FUNCTOR_socket1, | ||
|  | 		         IntArg(id)); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | pl_host_to_address(term_t Host, term_t Ip) | ||
|  | { struct in_addr ip; | ||
|  |   struct hostent *host; | ||
|  |   char *host_name; | ||
|  | 
 | ||
|  |   if ( PL_get_atom_chars(Host, &host_name) ) | ||
|  |   { if ( (host = gethostbyname(host_name)) ) | ||
|  |     { if ( sizeof(ip) == host->h_length ) | ||
|  |       { memcpy(&ip, host->h_addr, host->h_length); | ||
|  | 	return nbio_unify_ip4(Ip, ntohl(ip.s_addr)); | ||
|  |       } else | ||
|  | 	return PL_warning("tcp_host_to_address/2: length mismatch in address"); | ||
|  |     } else | ||
|  |       return nbio_error(h_errno, TCP_HERRNO); | ||
|  |   } else if ( nbio_get_ip(Ip, &ip) ) | ||
|  |   { if ( (host = gethostbyaddr((char *)&ip, sizeof(ip), AF_INET)) ) | ||
|  |       return PL_unify_atom_chars(Host, host->h_name); | ||
|  |     else | ||
|  |       return nbio_error(h_errno, TCP_HERRNO); | ||
|  |   } | ||
|  | 
 | ||
|  |   return FALSE; | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | pl_setopt(term_t Socket, term_t opt) | ||
|  | { int socket; | ||
|  |   atom_t a; | ||
|  |   int arity; | ||
|  | 
 | ||
|  |   if ( !tcp_get_socket(Socket, &socket) ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   if ( PL_get_name_arity(opt, &a, &arity) ) | ||
|  |   { if ( a == ATOM_reuseaddr && arity == 0 ) | ||
|  |     { if ( nbio_setopt(socket, TCP_REUSEADDR, TRUE) == 0 ) | ||
|  | 	return TRUE; | ||
|  | 
 | ||
|  |       return FALSE; | ||
|  |     } else if ( a == ATOM_nodelay && arity <= 1 ) | ||
|  |     { int enable, rc; | ||
|  | 
 | ||
|  |       if ( arity == 0 ) | ||
|  |       { enable = TRUE; | ||
|  |       } else /*if ( arity == 1 )*/ | ||
|  |       { term_t a = PL_new_term_ref(); | ||
|  | 
 | ||
|  | 	_PL_get_arg(1, opt, a); | ||
|  | 	if ( !PL_get_bool(a, &enable) ) | ||
|  | 	  return pl_error(NULL, 0, NULL, ERR_DOMAIN, a, "boolean"); | ||
|  |       } | ||
|  | 
 | ||
|  |       if ( (rc=nbio_setopt(socket, TCP_NO_DELAY, enable) == 0) ) | ||
|  | 	return TRUE; | ||
|  |       if ( rc == -2 ) | ||
|  | 	goto not_implemented; | ||
|  | 
 | ||
|  |       return FALSE; | ||
|  |     } else if ( a == ATOM_broadcast && arity == 0 ) | ||
|  |     { if ( nbio_setopt(socket, UDP_BROADCAST, TRUE) == 0 ) | ||
|  | 	return TRUE; | ||
|  | 
 | ||
|  |       return FALSE; | ||
|  |     } else if ( a == ATOM_dispatch && arity == 1 ) | ||
|  |     { int val; | ||
|  |       term_t a1 = PL_new_term_ref(); | ||
|  | 
 | ||
|  |       if ( PL_get_arg(1, opt, a1) && PL_get_bool(a1, &val) ) | ||
|  |       { if ( nbio_setopt(socket, TCP_DISPATCH, val) == 0 ) | ||
|  | 	  return TRUE; | ||
|  | 	return FALSE; | ||
|  |       } | ||
|  |     } else if ( a == ATOM_nonblock && arity == 0 ) | ||
|  |     { if ( nbio_setopt(socket, TCP_NONBLOCK) == 0 ) | ||
|  | 	return TRUE; | ||
|  |       return FALSE; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  | not_implemented: | ||
|  |   return pl_error(NULL, 0, NULL, ERR_DOMAIN, opt, "socket_option"); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | #include "sockcommon.c"
 | ||
|  | 
 | ||
|  | 		 /*******************************
 | ||
|  | 		 *	    UDP SOCKETS		* | ||
|  | 		 *******************************/ | ||
|  | 
 | ||
|  | /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
 | ||
|  | udp_receive(+Socket, -String, -From, +Options) | ||
|  | udp_send(+String, +String, +To, +Options) | ||
|  | 
 | ||
|  | From/To are of the format <Host>:<Port> | ||
|  | - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */ | ||
|  | 
 | ||
|  | #define UDP_MAXDATA 4096
 | ||
|  | 
 | ||
|  | static int | ||
|  | unify_address(term_t t, struct sockaddr_in *addr) | ||
|  | { term_t av = PL_new_term_refs(2); | ||
|  | 
 | ||
|  |   if ( !nbio_unify_ip4(av+0, ntohl(addr->sin_addr.s_addr)) || | ||
|  |        !PL_unify_integer(av+1, ntohs(addr->sin_port)) ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   return PL_unify_term(t, PL_FUNCTOR_CHARS, ":", 2, | ||
|  | 		       PL_TERM, av+0, | ||
|  | 		       PL_TERM, av+1); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | udp_receive(term_t Socket, term_t Data, term_t From, term_t options) | ||
|  | { struct sockaddr_in sockaddr; | ||
|  | #ifdef __WINDOWS__
 | ||
|  |   int alen = sizeof(sockaddr); | ||
|  | #else
 | ||
|  |   socklen_t alen = sizeof(sockaddr); | ||
|  | #endif
 | ||
|  |   int socket; | ||
|  |   int flags = 0; | ||
|  |   char buf[UDP_MAXDATA]; | ||
|  |   ssize_t n; | ||
|  |   int as = PL_STRING; | ||
|  | 
 | ||
|  |   if ( !PL_get_nil(options) ) | ||
|  |   { term_t tail = PL_copy_term_ref(options); | ||
|  |     term_t head = PL_new_term_ref(); | ||
|  |     term_t arg  = PL_new_term_ref(); | ||
|  | 
 | ||
|  |     while(PL_get_list(tail, head, tail)) | ||
|  |     { atom_t name; | ||
|  |       int arity; | ||
|  | 
 | ||
|  |       if ( PL_get_name_arity(head, &name, &arity) && arity == 1 ) | ||
|  |       { _PL_get_arg(1, head, arg); | ||
|  | 
 | ||
|  | 	if ( name == ATOM_as ) | ||
|  | 	{ atom_t a; | ||
|  | 
 | ||
|  | 	  if ( !PL_get_atom(arg, &a) ) | ||
|  | 	    return pl_error(NULL, 0, NULL, ERR_TYPE, head, "atom"); | ||
|  | 	  if ( a == ATOM_atom ) | ||
|  | 	    as = PL_ATOM; | ||
|  | 	  else if ( a == ATOM_codes ) | ||
|  | 	    as = PL_CODE_LIST; | ||
|  | 	  else if ( a == ATOM_string ) | ||
|  | 	    as = PL_STRING; | ||
|  | 	  else | ||
|  | 	    return pl_error(NULL, 0, NULL, ERR_DOMAIN, arg, "as_option"); | ||
|  | 	} | ||
|  | 
 | ||
|  |       } else | ||
|  | 	return pl_error(NULL, 0, NULL, ERR_TYPE, head, "option"); | ||
|  |     } | ||
|  |     if ( !PL_get_nil(tail) ) | ||
|  |       return pl_error(NULL, 0, NULL, ERR_TYPE, tail, "list"); | ||
|  |   } | ||
|  | 
 | ||
|  | 
 | ||
|  |   if ( !tcp_get_socket(Socket, &socket) || | ||
|  |        !nbio_get_sockaddr(From, &sockaddr) ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   if ( (n=nbio_recvfrom(socket, buf, sizeof(buf), flags, | ||
|  | 			(struct sockaddr*)&sockaddr, &alen)) == -1 ) | ||
|  |     return nbio_error(errno, TCP_ERRNO); | ||
|  |   if ( !PL_unify_chars(Data, as, n, buf) ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   return unify_address(From, &sockaddr); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | udp_send(term_t Socket, term_t Data, term_t To, term_t Options) | ||
|  | { struct sockaddr_in sockaddr; | ||
|  | #ifdef __WINDOWS__
 | ||
|  |   int alen = sizeof(sockaddr); | ||
|  | #else
 | ||
|  |   int alen = sizeof(sockaddr); | ||
|  | #endif
 | ||
|  |   int socket; | ||
|  |   int flags = 0L; | ||
|  |   char *data; | ||
|  |   size_t dlen; | ||
|  |   ssize_t n; | ||
|  | 
 | ||
|  |   if ( !PL_get_nchars(Data, &dlen, &data, CVT_ALL|CVT_EXCEPTION) ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   if ( !tcp_get_socket(Socket, &socket) || | ||
|  |        !nbio_get_sockaddr(To, &sockaddr) ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   if ( (n=nbio_sendto(socket, data, | ||
|  | 		      (int)dlen, | ||
|  | 		      flags, | ||
|  | 		      (struct sockaddr*)&sockaddr, alen)) == -1 ) | ||
|  |     return nbio_error(errno, TCP_ERRNO); | ||
|  | 
 | ||
|  |   return TRUE; | ||
|  | } | ||
|  | 
 | ||
|  | 		 /*******************************
 | ||
|  | 		 *	PROLOG CONNECTION	* | ||
|  | 		 *******************************/ | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | create_socket(term_t socket, int type) | ||
|  | { int sock; | ||
|  | 
 | ||
|  |   sock = nbio_socket(AF_INET, type, 0); | ||
|  |   if ( sock < 0 ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   return tcp_unify_socket(socket, sock); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | tcp_socket(term_t socket) | ||
|  | { return create_socket(socket, SOCK_STREAM); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | udp_socket(term_t socket) | ||
|  | { return create_socket(socket, SOCK_DGRAM); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | pl_connect(term_t Socket, term_t Address) | ||
|  | { int sock; | ||
|  |   struct sockaddr_in sockaddr; | ||
|  | 
 | ||
|  |   if ( !tcp_get_socket(Socket, &sock) || | ||
|  |        !nbio_get_sockaddr(Address, &sockaddr) ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   if ( nbio_connect(sock, (struct sockaddr*)&sockaddr, sizeof(sockaddr)) == 0 ) | ||
|  |     return TRUE; | ||
|  | 
 | ||
|  |   return FALSE; | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | pl_bind(term_t Socket, term_t Address) | ||
|  | { struct sockaddr_in sockaddr; | ||
|  |   int socket; | ||
|  | 
 | ||
|  |   memset(&sockaddr, 0, sizeof(sockaddr)); | ||
|  | 
 | ||
|  |   if ( !tcp_get_socket(Socket, &socket) || | ||
|  |        !nbio_get_sockaddr(Address, &sockaddr) ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   if ( nbio_bind(socket, (struct sockaddr*)&sockaddr, sizeof(sockaddr)) < 0 ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   if ( PL_is_variable(Address) ) | ||
|  |   { SOCKET fd = nbio_fd(socket); | ||
|  |     struct sockaddr_in addr; | ||
|  | #ifdef __WINDOWS__
 | ||
|  |     int len = sizeof(addr); | ||
|  | #else
 | ||
|  |     socklen_t len = sizeof(addr); | ||
|  | #endif
 | ||
|  | 
 | ||
|  |     if ( getsockname(fd, (struct sockaddr *) &addr, &len) ) | ||
|  |       return nbio_error(errno, TCP_ERRNO); | ||
|  |     return PL_unify_integer(Address, ntohs(addr.sin_port)); | ||
|  |   } | ||
|  | 
 | ||
|  |   return TRUE; | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | pl_accept(term_t Master, term_t Slave, term_t Peer) | ||
|  | { int master, slave; | ||
|  |   struct sockaddr_in addr; | ||
|  |   socklen_t addrlen = sizeof(addr); | ||
|  | 
 | ||
|  |   if ( !tcp_get_socket(Master, &master) ) | ||
|  |     return FALSE; | ||
|  | 
 | ||
|  |   if ( (slave = nbio_accept(master, (struct sockaddr*)&addr, &addrlen)) < 0 ) | ||
|  |     return FALSE; | ||
|  | 					/* TBD: close on failure */ | ||
|  |   if ( nbio_unify_ip4(Peer, ntohl(addr.sin_addr.s_addr)) && | ||
|  |        tcp_unify_socket(Slave, slave) ) | ||
|  |     return TRUE; | ||
|  | 
 | ||
|  |   return FALSE; | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | pl_gethostname(term_t name) | ||
|  | { static atom_t hname; | ||
|  | 
 | ||
|  |   if ( !hname ) | ||
|  |   { char buf[256]; | ||
|  | 
 | ||
|  |     if ( gethostname(buf, sizeof(buf)) == 0 ) | ||
|  |     { struct hostent *he; | ||
|  | 
 | ||
|  |       if ( (he = gethostbyname(buf)) ) | ||
|  | 	hname = PL_new_atom(he->h_name); | ||
|  |       else | ||
|  | 	hname = PL_new_atom(buf); | ||
|  |     } else | ||
|  |     { return nbio_error(h_errno, TCP_HERRNO); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return PL_unify_atom(name, hname); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | 		 /*******************************
 | ||
|  | 		 *	       SELECT		* | ||
|  | 		 *******************************/ | ||
|  | 
 | ||
|  | typedef struct fdentry | ||
|  | { int fd; | ||
|  |   term_t stream; | ||
|  |   struct fdentry *next; | ||
|  | } fdentry; | ||
|  | 
 | ||
|  | 
 | ||
|  | static term_t | ||
|  | findmap(fdentry *map, int fd) | ||
|  | { for( ; map; map = map->next ) | ||
|  |   { if ( map->fd == fd ) | ||
|  |       return map->stream; | ||
|  |   } | ||
|  |   assert(0); | ||
|  |   return 0; | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static int | ||
|  | is_socket_stream(IOSTREAM *s) | ||
|  | { if ( s->functions == &readFunctions /* ||
 | ||
|  |        s->functions == &writeFunctions */ ) | ||
|  |     return TRUE; | ||
|  | 
 | ||
|  |   return FALSE; | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | static foreign_t | ||
|  | tcp_select(term_t Streams, term_t Available, term_t timeout) | ||
|  | { fd_set fds; | ||
|  |   struct timeval t, *to; | ||
|  |   double time; | ||
|  |   int n, max = 0, ret, min = 1000000; | ||
|  |   fdentry *map     = NULL; | ||
|  |   term_t head      = PL_new_term_ref(); | ||
|  |   term_t streams   = PL_copy_term_ref(Streams); | ||
|  |   term_t available = PL_copy_term_ref(Available); | ||
|  |   term_t ahead     = PL_new_term_ref(); | ||
|  |   int from_buffer  = 0; | ||
|  |   atom_t a; | ||
|  | 
 | ||
|  |   FD_ZERO(&fds); | ||
|  |   while( PL_get_list(streams, head, streams) ) | ||
|  |   { IOSTREAM *s; | ||
|  | #ifdef __WINDOWS__
 | ||
|  |     nbio_sock_t fd; | ||
|  | #else
 | ||
|  |     int fd; | ||
|  | #endif
 | ||
|  |     fdentry *e; | ||
|  | 
 | ||
|  |     if ( !PL_get_stream_handle(head, &s) ) | ||
|  |       return FALSE; | ||
|  | 
 | ||
|  | #ifdef __WINDOWS__
 | ||
|  |     fd = fdFromHandle(s->handle); | ||
|  | #else
 | ||
|  |     fd = Sfileno(s); | ||
|  | #endif
 | ||
|  | 
 | ||
|  |     PL_release_stream(s); | ||
|  |     if ( fd < 0 || !is_socket_stream(s) ) | ||
|  |     { return pl_error("tcp_select", 3, NULL, ERR_DOMAIN, | ||
|  | 		      head, "socket_stream"); | ||
|  |     } | ||
|  | 					/* check for input in buffer */ | ||
|  |     if ( s->bufp < s->limitp ) | ||
|  |     { if ( !PL_unify_list(available, ahead, available) || | ||
|  | 	   !PL_unify(ahead, head) ) | ||
|  | 	return FALSE; | ||
|  |       from_buffer++; | ||
|  |     } | ||
|  | 
 | ||
|  |     e         = alloca(sizeof(*e)); | ||
|  |     e->fd     = fd; | ||
|  |     e->stream = PL_copy_term_ref(head); | ||
|  |     e->next   = map; | ||
|  |     map       = e; | ||
|  | 
 | ||
|  | #ifdef __WINDOWS__
 | ||
|  |     FD_SET((SOCKET)fd, &fds); | ||
|  | #else
 | ||
|  |     FD_SET(fd, &fds); | ||
|  | #endif
 | ||
|  | 
 | ||
|  |     if ( fd > max ) | ||
|  |       max = fd; | ||
|  |     if( fd < min ) | ||
|  |       min = fd; | ||
|  |   } | ||
|  |   if ( !PL_get_nil(streams) ) | ||
|  |     return pl_error("tcp_select", 3, NULL, ERR_TYPE, Streams, "list"); | ||
|  | 
 | ||
|  |   if ( from_buffer > 0 ) | ||
|  |     return PL_unify_nil(available); | ||
|  | 
 | ||
|  |   if ( PL_get_atom(timeout, &a) && a == ATOM_infinite ) | ||
|  |   { to = NULL; | ||
|  |   } else | ||
|  |   { if ( !PL_get_float(timeout, &time) ) | ||
|  |       return pl_error("tcp_select", 3, NULL, | ||
|  | 		      ERR_TYPE, timeout, "number"); | ||
|  | 
 | ||
|  |     if ( time >= 0.0 ) | ||
|  |     { t.tv_sec  = (int)time; | ||
|  |       t.tv_usec = ((int)(time * 1000000) % 1000000); | ||
|  |     } else | ||
|  |     { t.tv_sec  = 0; | ||
|  |       t.tv_usec = 0; | ||
|  |     } | ||
|  |     to = &t; | ||
|  |   } | ||
|  | 
 | ||
|  |   while( (ret=nbio_select(max+1, &fds, NULL, NULL, to)) == -1 && | ||
|  | 	 errno == EINTR ) | ||
|  |   { fdentry *e; | ||
|  | 
 | ||
|  |     if ( PL_handle_signals() < 0 ) | ||
|  |       return FALSE;			/* exception */ | ||
|  | 
 | ||
|  |     FD_ZERO(&fds);			/* EINTR may leave fds undefined */ | ||
|  |     for(e=map; e; e=e->next)		/* so we rebuild it to be safe */ | ||
|  |     { FD_SET((SOCKET)e->fd, &fds); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   switch(ret) | ||
|  |   { case -1: | ||
|  |       return pl_error("tcp_select", 3, NULL, ERR_ERRNO, errno, "select", "streams", Streams); | ||
|  | 
 | ||
|  |     case 0: /* Timeout */ | ||
|  |       break; | ||
|  | 
 | ||
|  |     default: /* Something happened -> check fds */ | ||
|  |       for(n=min; n <= max; n++) | ||
|  |       { if ( FD_ISSET(n, &fds) ) | ||
|  | 	{ if ( !PL_unify_list(available, ahead, available) || | ||
|  | 	       !PL_unify(ahead, findmap(map, n)) ) | ||
|  | 	    return FALSE; | ||
|  | 	} | ||
|  |       } | ||
|  |       break; | ||
|  |   } | ||
|  | 
 | ||
|  |   return PL_unify_nil(available); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | #ifdef O_DEBUG
 | ||
|  | static foreign_t | ||
|  | pl_debug(term_t val) | ||
|  | { int dbg; | ||
|  | 
 | ||
|  |   if ( PL_get_integer(val, &dbg) ) | ||
|  |   { nbio_debug(dbg); | ||
|  |     return TRUE; | ||
|  |   } | ||
|  | 
 | ||
|  |   return FALSE; | ||
|  | } | ||
|  | #endif
 | ||
|  | 
 | ||
|  | install_t | ||
|  | install_socket() | ||
|  | { nbio_init("socket"); | ||
|  | 
 | ||
|  |   ATOM_reuseaddr      =	PL_new_atom("reuseaddr"); | ||
|  |   ATOM_broadcast      =	PL_new_atom("broadcast"); | ||
|  |   ATOM_nodelay	      =	PL_new_atom("nodelay"); | ||
|  |   ATOM_dispatch	      =	PL_new_atom("dispatch"); | ||
|  |   ATOM_nonblock	      =	PL_new_atom("nonblock"); | ||
|  |   ATOM_infinite	      =	PL_new_atom("infinite"); | ||
|  |   ATOM_as	      =	PL_new_atom("as"); | ||
|  |   ATOM_atom	      =	PL_new_atom("atom"); | ||
|  |   ATOM_string	      =	PL_new_atom("string"); | ||
|  |   ATOM_codes	      =	PL_new_atom("codes"); | ||
|  | 
 | ||
|  |   FUNCTOR_socket1 = PL_new_functor(PL_new_atom("$socket"), 1); | ||
|  | 
 | ||
|  | #ifdef O_DEBUG
 | ||
|  |   PL_register_foreign_in_module("user", "tcp_debug", 1, pl_debug, 0); | ||
|  | #endif
 | ||
|  |   PL_register_foreign("tcp_accept",           3, pl_accept,           0); | ||
|  |   PL_register_foreign("tcp_bind",             2, pl_bind,             0); | ||
|  |   PL_register_foreign("tcp_connect",          2, pl_connect,	      0); | ||
|  |   PL_register_foreign("tcp_listen",           2, pl_listen,           0); | ||
|  |   PL_register_foreign("tcp_open_socket",      3, pl_open_socket,      0); | ||
|  |   PL_register_foreign("tcp_socket",           1, tcp_socket,          0); | ||
|  |   PL_register_foreign("tcp_close_socket",     1, pl_close_socket,     0); | ||
|  |   PL_register_foreign("tcp_setopt",           2, pl_setopt,           0); | ||
|  |   PL_register_foreign("tcp_host_to_address",  2, pl_host_to_address,  0); | ||
|  |   PL_register_foreign("gethostname",          1, pl_gethostname,      0); | ||
|  |   PL_register_foreign("tcp_select",           3, tcp_select,          0); | ||
|  | 
 | ||
|  |   PL_register_foreign("udp_socket",           1, udp_socket,          0); | ||
|  |   PL_register_foreign("udp_receive",	      4, udp_receive,	      0); | ||
|  |   PL_register_foreign("udp_send",	      4, udp_send,	      0); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | install_t | ||
|  | uninstall_socket() | ||
|  | { nbio_cleanup(); | ||
|  | } |