| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | /***********************************************************
 | 
					
						
							| 
									
										
										
										
											1993-03-16 12:15:04 +00:00
										 |  |  | Copyright 1991, 1992, 1993 by Stichting Mathematisch Centrum, | 
					
						
							|  |  |  | Amsterdam, The Netherlands. | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                         All Rights Reserved | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Permission to use, copy, modify, and distribute this software and its  | 
					
						
							|  |  |  | documentation for any purpose and without fee is hereby granted,  | 
					
						
							|  |  |  | provided that the above copyright notice appear in all copies and that | 
					
						
							|  |  |  | both that copyright notice and this permission notice appear in  | 
					
						
							|  |  |  | supporting documentation, and that the names of Stichting Mathematisch | 
					
						
							|  |  |  | Centrum or CWI not be used in advertising or publicity pertaining to | 
					
						
							|  |  |  | distribution of the software without specific, written prior permission. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | STICHTING MATHEMATISCH CENTRUM DISCLAIMS ALL WARRANTIES WITH REGARD TO | 
					
						
							|  |  |  | THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND | 
					
						
							|  |  |  | FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH CENTRUM BE LIABLE | 
					
						
							|  |  |  | FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | 
					
						
							|  |  |  | WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | 
					
						
							|  |  |  | ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT | 
					
						
							|  |  |  | OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ******************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Socket module */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  | This module provides an interface to Berkeley socket IPC. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Limitations: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | - only AF_INET and AF_UNIX address families are supported | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | - no asynchronous I/O (but read polling: avail) | 
					
						
							|  |  |  | - no read/write operations (use send/recv or makefile instead) | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | - setsockopt() and getsockopt() only support integer options | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | Interface: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | - socket.gethostname() --> host name (string) | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | - socket.gethostbyname(hostname) --> host IP address (string: 'dd.dd.dd.dd') | 
					
						
							| 
									
										
										
										
											1991-07-01 18:51:33 +00:00
										 |  |  | - socket.getservbyname(servername, protocolname) --> port number | 
					
						
							|  |  |  | - socket.socket(family, type [, proto]) --> new socket object | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | - family and type constants from <socket.h> are accessed as socket.AF_INET etc. | 
					
						
							|  |  |  | - errors are reported as the exception socket.error | 
					
						
							|  |  |  | - an Internet socket address is a pair (hostname, port) | 
					
						
							|  |  |  |   where hostname can be anything recognized by gethostbyname() | 
					
						
							|  |  |  |   (including the dd.dd.dd.dd notation) and port is in host byte order | 
					
						
							|  |  |  | - where a hostname is returned, the dd.dd.dd.dd notation is used | 
					
						
							|  |  |  | - a UNIX domain socket is a string specifying the pathname | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | Socket methods: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | - s.accept() --> new socket object, sockaddr | 
					
						
							|  |  |  | - s.avail() --> boolean | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | - s.setsockopt(level, optname, flag) --> None | 
					
						
							|  |  |  | - s.getsockopt(level, optname) --> flag | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | - s.bind(sockaddr) --> None | 
					
						
							|  |  |  | - s.connect(sockaddr) --> None | 
					
						
							| 
									
										
										
										
											1992-11-26 08:54:07 +00:00
										 |  |  | - s.getsockname() --> sockaddr | 
					
						
							|  |  |  | - s.getpeername() --> sockaddr | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | - s.listen(n) --> None | 
					
						
							|  |  |  | - s.makefile(mode) --> file object | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | - s.recv(nbytes [,flags]) --> string | 
					
						
							| 
									
										
										
										
											1993-05-25 12:16:29 +00:00
										 |  |  | - s.recvfrom(nbytes [,flags]) --> string, sockaddr | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | - s.send(string [,flags]) --> None | 
					
						
							|  |  |  | - s.sendto(string, [flags,] sockaddr) --> None | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | - s.shutdown(how) --> None | 
					
						
							|  |  |  | - s.close() --> None | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "allobjects.h"
 | 
					
						
							|  |  |  | #include "modsupport.h"
 | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | #include "ceval.h"
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-06-23 09:07:03 +00:00
										 |  |  | #include "myselect.h" /* Implies <sys/types.h>, <sys/time.h>, <sys/param.h> */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | #include <signal.h>
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | #include <sys/socket.h>
 | 
					
						
							|  |  |  | #include <netinet/in.h>
 | 
					
						
							|  |  |  | #include <sys/un.h>
 | 
					
						
							|  |  |  | #include <netdb.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-04-07 14:06:14 +00:00
										 |  |  | #ifdef i860
 | 
					
						
							|  |  |  | /* Cray APP doesn't have getpeername() */ | 
					
						
							|  |  |  | #define NO_GETPEERNAME
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Global variable holding the exception type for errors detected
 | 
					
						
							|  |  |  |    by this module (but not argument type or memory errors, etc.). */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object *SocketError; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Convenience function to raise an error according to errno
 | 
					
						
							|  |  |  |    and return a NULL pointer from a function. */ | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | socket_error() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return err_errno(SocketError); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* The object holding a socket.  It holds some extra information,
 | 
					
						
							|  |  |  |    like the address family, which is used to decode socket address | 
					
						
							|  |  |  |    arguments properly. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | typedef struct { | 
					
						
							|  |  |  | 	OB_HEAD | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	int sock_fd;		/* Socket file descriptor */ | 
					
						
							|  |  |  | 	int sock_family;	/* Address family, e.g., AF_INET */ | 
					
						
							|  |  |  | 	int sock_type;		/* Socket type, e.g., SOCK_STREAM */ | 
					
						
							|  |  |  | 	int sock_proto;		/* Protocol type, usually 0 */ | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | } sockobject; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* A forward reference to the Socktype type object.
 | 
					
						
							|  |  |  |    The Socktype variable contains pointers to various functions, | 
					
						
							|  |  |  |    some of which call newsocobject(), which uses Socktype, so | 
					
						
							| 
									
										
										
										
											1991-09-10 14:57:12 +00:00
										 |  |  |    there has to be a circular reference. */ | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | extern typeobject Socktype; /* Forward */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Create a new socket object.
 | 
					
						
							|  |  |  |    This just creates the object and initializes it. | 
					
						
							|  |  |  |    If the creation fails, return NULL and set an exception (implicit | 
					
						
							|  |  |  |    in NEWOBJ()). */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static sockobject * | 
					
						
							|  |  |  | newsockobject(fd, family, type, proto) | 
					
						
							|  |  |  | 	int fd, family, type, proto; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	s = NEWOBJ(sockobject, &Socktype); | 
					
						
							|  |  |  | 	if (s != NULL) { | 
					
						
							|  |  |  | 		s->sock_fd = fd; | 
					
						
							|  |  |  | 		s->sock_family = family; | 
					
						
							|  |  |  | 		s->sock_type = type; | 
					
						
							|  |  |  | 		s->sock_proto = proto; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return s; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Convert a string specifying a host name or one of a few symbolic
 | 
					
						
							|  |  |  |    names to a numeric IP address.  This usually calls gethostbyname() | 
					
						
							|  |  |  |    to do the work; the names "" and "<broadcast>" are special. | 
					
						
							|  |  |  |    Return the length (should always be 4 bytes), or negative if | 
					
						
							|  |  |  |    an error occurred; then an exception is raised. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | setipaddr(name, addr_ret) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	char *name; | 
					
						
							|  |  |  | 	struct sockaddr_in *addr_ret; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct hostent *hp; | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	int d1, d2, d3, d4; | 
					
						
							|  |  |  | 	char ch; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	if (name[0] == '\0') { | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		addr_ret->sin_addr.s_addr = INADDR_ANY; | 
					
						
							|  |  |  | 		return 4; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	if (name[0] == '<' && strcmp(name, "<broadcast>") == 0) { | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		addr_ret->sin_addr.s_addr = INADDR_BROADCAST; | 
					
						
							|  |  |  | 		return 4; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	if (sscanf(name, "%d.%d.%d.%d%c", &d1, &d2, &d3, &d4, &ch) == 4 && | 
					
						
							|  |  |  | 	    0 <= d1 && d1 <= 255 && 0 <= d2 && d2 <= 255 && | 
					
						
							|  |  |  | 	    0 <= d3 && d3 <= 255 && 0 <= d4 && d4 <= 255) { | 
					
						
							|  |  |  | 		addr_ret->sin_addr.s_addr = htonl( | 
					
						
							|  |  |  | 			((long) d1 << 24) | ((long) d2 << 16) | | 
					
						
							|  |  |  | 			((long) d3 << 8) | ((long) d4 << 0)); | 
					
						
							|  |  |  | 		return 4; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	hp = gethostbyname(name); | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	END_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (hp == NULL) { | 
					
						
							|  |  |  | 		err_setstr(SocketError, "host not found"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	memcpy((char *) &addr_ret->sin_addr, hp->h_addr, hp->h_length); | 
					
						
							|  |  |  | 	return hp->h_length; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Create a string object representing an IP address.
 | 
					
						
							|  |  |  |    This is always a string of the form 'dd.dd.dd.dd' (with variable | 
					
						
							|  |  |  |    size numbers). */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | makeipaddr(addr) | 
					
						
							|  |  |  | 	struct sockaddr_in *addr; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	long x = ntohl(addr->sin_addr.s_addr); | 
					
						
							|  |  |  | 	char buf[100]; | 
					
						
							|  |  |  | 	sprintf(buf, "%d.%d.%d.%d", | 
					
						
							|  |  |  | 		(int) (x>>24) & 0xff, (int) (x>>16) & 0xff, | 
					
						
							|  |  |  | 		(int) (x>> 8) & 0xff, (int) (x>> 0) & 0xff); | 
					
						
							|  |  |  | 	return newstringobject(buf); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Create an object representing the given socket address,
 | 
					
						
							|  |  |  |    suitable for passing it back to bind(), connect() etc. | 
					
						
							|  |  |  |    The family field of the sockaddr structure is inspected | 
					
						
							|  |  |  |    to determine what kind of address it really is. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | /*ARGSUSED*/ | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | makesockaddr(addr, addrlen) | 
					
						
							|  |  |  | 	struct sockaddr *addr; | 
					
						
							|  |  |  | 	int addrlen; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1992-08-05 19:00:45 +00:00
										 |  |  | 	if (addrlen == 0) { | 
					
						
							|  |  |  | 		/* No address -- may be recvfrom() from known socket */ | 
					
						
							|  |  |  | 		INCREF(None); | 
					
						
							|  |  |  | 		return None; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	switch (addr->sa_family) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case AF_INET: | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		struct sockaddr_in *a = (struct sockaddr_in *) addr; | 
					
						
							| 
									
										
										
										
											1993-02-05 09:46:15 +00:00
										 |  |  | 		object *addr = makeipaddr(a); | 
					
						
							|  |  |  | 		object *ret = mkvalue("Oi", addr, ntohs(a->sin_port)); | 
					
						
							|  |  |  | 		XDECREF(addr); | 
					
						
							|  |  |  | 		return ret; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	case AF_UNIX: | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		struct sockaddr_un *a = (struct sockaddr_un *) addr; | 
					
						
							|  |  |  | 		return newstringobject(a->sun_path); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* More cases here... */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		err_setstr(SocketError, "return unknown socket address type"); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:00:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Parse a socket address argument according to the socket object's
 | 
					
						
							|  |  |  |    address family.  Return 1 if the address was in the proper format, | 
					
						
							|  |  |  |    0 of not.  The address is returned through addr_ret, its length | 
					
						
							|  |  |  |    through len_ret. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | getsockaddrarg(s, args, addr_ret, len_ret) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | 	struct sockaddr **addr_ret; | 
					
						
							|  |  |  | 	int *len_ret; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	switch (s->sock_family) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case AF_UNIX: | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		static struct sockaddr_un addr; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 		char *path; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		int len; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 		if (!getargs(args, "s#", &path, &len)) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 			return 0; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 		if (len > sizeof addr.sun_path) { | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 			err_setstr(SocketError, "AF_UNIX path too long"); | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		addr.sun_family = AF_UNIX; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 		memcpy(addr.sun_path, path, len); | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		*addr_ret = (struct sockaddr *) &addr; | 
					
						
							|  |  |  | 		*len_ret = len + sizeof addr.sun_family; | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	case AF_INET: | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		static struct sockaddr_in addr; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 		char *host; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		int port; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 		if (!getargs(args, "(si)", &host, &port)) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 			return 0; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 		if (setipaddr(host, &addr) < 0) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 			return 0; | 
					
						
							|  |  |  | 		addr.sin_family = AF_INET; | 
					
						
							|  |  |  | 		addr.sin_port = htons(port); | 
					
						
							|  |  |  | 		*addr_ret = (struct sockaddr *) &addr; | 
					
						
							|  |  |  | 		*len_ret = sizeof addr; | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	/* More cases here... */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		err_setstr(SocketError, "getsockaddrarg: bad family"); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-06-12 10:39:36 +00:00
										 |  |  | /* Get the address length according to the socket object's address family. 
 | 
					
						
							|  |  |  |    Return 1 if the family is known, 0 otherwise.  The length is returned | 
					
						
							|  |  |  |    through len_ret. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | getsockaddrlen(s, len_ret) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	int *len_ret; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	switch (s->sock_family) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case AF_UNIX: | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		*len_ret = sizeof (struct sockaddr_un); | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case AF_INET: | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		*len_ret = sizeof (struct sockaddr_in); | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* More cases here... */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		err_setstr(SocketError, "getsockaddrarg: bad family"); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | /* s.accept() method */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | sock_accept(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char addrbuf[256]; | 
					
						
							|  |  |  | 	int addrlen, newfd; | 
					
						
							| 
									
										
										
										
											1993-02-05 09:46:15 +00:00
										 |  |  | 	object *sock, *addr, *res; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (!getnoarg(args)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-06-12 10:39:36 +00:00
										 |  |  | 	if (!getsockaddrlen(s, &addrlen)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	newfd = accept(s->sock_fd, (struct sockaddr *) addrbuf, &addrlen); | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	END_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (newfd < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	/* Create the new object with unspecified family,
 | 
					
						
							|  |  |  | 	   to avoid calls to bind() etc. on it. */ | 
					
						
							| 
									
										
										
										
											1993-02-05 09:46:15 +00:00
										 |  |  | 	sock = (object *) newsockobject(newfd, | 
					
						
							|  |  |  | 					s->sock_family, | 
					
						
							|  |  |  | 					s->sock_type, | 
					
						
							|  |  |  | 					s->sock_proto); | 
					
						
							|  |  |  | 	if (sock == NULL) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		close(newfd); | 
					
						
							| 
									
										
										
										
											1993-02-05 09:46:15 +00:00
										 |  |  | 	addr = makesockaddr((struct sockaddr *) addrbuf, addrlen); | 
					
						
							|  |  |  | 	res = mkvalue("OO", sock, addr); | 
					
						
							|  |  |  | 	XDECREF(sock); | 
					
						
							|  |  |  | 	XDECREF(addr); | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-10-20 20:22:28 +00:00
										 |  |  | /* s.allowbroadcast() method */ | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | /* XXX obsolete -- will disappear in next release */ | 
					
						
							| 
									
										
										
										
											1991-10-20 20:22:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | sock_allowbroadcast(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int flag; | 
					
						
							|  |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	if (!getargs(args, "i", &flag)) | 
					
						
							| 
									
										
										
										
											1991-10-20 20:22:28 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	res = setsockopt(s->sock_fd, SOL_SOCKET, SO_BROADCAST, | 
					
						
							|  |  |  | 			 &flag, sizeof flag); | 
					
						
							|  |  |  | 	if (res < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	INCREF(None); | 
					
						
							|  |  |  | 	return None; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-09-08 09:05:33 +00:00
										 |  |  | /* s.setsockopt() method.
 | 
					
						
							|  |  |  |    With an integer third argument, sets an integer option. | 
					
						
							|  |  |  |    With a string third argument, sets an option from a buffer; | 
					
						
							|  |  |  |    use optional built-in module 'struct' to encode the string. */ | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | sock_setsockopt(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int level; | 
					
						
							|  |  |  | 	int optname; | 
					
						
							|  |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											1992-09-08 09:05:33 +00:00
										 |  |  | 	char *buf; | 
					
						
							|  |  |  | 	int buflen; | 
					
						
							|  |  |  | 	int flag; | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-09-08 09:05:33 +00:00
										 |  |  | 	if (getargs(args, "(iii)", &level, &optname, &flag)) { | 
					
						
							|  |  |  | 		buf = (char *) &flag; | 
					
						
							|  |  |  | 		buflen = sizeof flag; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		err_clear(); | 
					
						
							|  |  |  | 		if (!getargs(args, "(iis#)", &level, &optname, &buf, &buflen)) | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	res = setsockopt(s->sock_fd, level, optname, buf, buflen); | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | 	if (res < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	INCREF(None); | 
					
						
							|  |  |  | 	return None; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-09-08 09:05:33 +00:00
										 |  |  | /* s.getsockopt() method.
 | 
					
						
							|  |  |  |    With two arguments, retrieves an integer option. | 
					
						
							|  |  |  |    With a third integer argument, retrieves a string buffer of that size; | 
					
						
							|  |  |  |    use optional built-in module 'struct' to decode the string. */ | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | sock_getsockopt(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int level; | 
					
						
							|  |  |  | 	int optname; | 
					
						
							|  |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											1992-09-08 09:05:33 +00:00
										 |  |  | 	object *buf; | 
					
						
							|  |  |  | 	int buflen; | 
					
						
							|  |  |  | 	int flag; | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-09-08 09:05:33 +00:00
										 |  |  | 	if (getargs(args, "(ii)", &level, &optname)) { | 
					
						
							|  |  |  | 		int flag = 0; | 
					
						
							|  |  |  | 		int flagsize = sizeof flag; | 
					
						
							|  |  |  | 		res = getsockopt(s->sock_fd, level, optname, &flag, &flagsize); | 
					
						
							|  |  |  | 		if (res < 0) | 
					
						
							|  |  |  | 			return socket_error(); | 
					
						
							|  |  |  | 		return newintobject(flag); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	err_clear(); | 
					
						
							|  |  |  | 	if (!getargs(args, "(iii)", &level, &optname, &buflen)) | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-09-08 09:05:33 +00:00
										 |  |  | 	if (buflen <= 0 || buflen > 1024) { | 
					
						
							|  |  |  | 		err_setstr(SocketError, "getsockopt buflen out of range"); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	buf = newsizedstringobject((char *)NULL, buflen); | 
					
						
							|  |  |  | 	if (buf == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	res = getsockopt(s->sock_fd, level, optname, getstringvalue(buf), | 
					
						
							|  |  |  | 			 &buflen); | 
					
						
							|  |  |  | 	if (res < 0) { | 
					
						
							|  |  |  | 		DECREF(buf); | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | 		return socket_error(); | 
					
						
							| 
									
										
										
										
											1992-09-08 09:05:33 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	resizestring(&buf, buflen); | 
					
						
							|  |  |  | 	return buf; | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | /* s.avail() method */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | sock_avail(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct timeval timeout; | 
					
						
							|  |  |  | 	fd_set readers; | 
					
						
							|  |  |  | 	int n; | 
					
						
							| 
									
										
										
										
											1992-03-27 17:22:00 +00:00
										 |  |  | 	if (!getnoarg(args)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | 	timeout.tv_sec = 0; | 
					
						
							|  |  |  | 	timeout.tv_usec = 0; | 
					
						
							|  |  |  | 	FD_ZERO(&readers); | 
					
						
							|  |  |  | 	FD_SET(s->sock_fd, &readers); | 
					
						
							|  |  |  | 	n = select(s->sock_fd+1, &readers, (fd_set *)0, (fd_set *)0, &timeout); | 
					
						
							|  |  |  | 	if (n < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	return newintobject((long) (n != 0)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | /* s.bind(sockaddr) method */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | sock_bind(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct sockaddr *addr; | 
					
						
							|  |  |  | 	int addrlen; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (!getsockaddrarg(s, args, &addr, &addrlen)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							|  |  |  | 	res = bind(s->sock_fd, addr, addrlen); | 
					
						
							|  |  |  | 	END_SAVE | 
					
						
							|  |  |  | 	if (res < 0) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	INCREF(None); | 
					
						
							|  |  |  | 	return None; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* s.close() method.
 | 
					
						
							|  |  |  |    Set the file descriptor to -1 so operations tried subsequently | 
					
						
							|  |  |  |    will surely fail. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | sock_close(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!getnoarg(args)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	(void) close(s->sock_fd); | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	END_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	s->sock_fd = -1; | 
					
						
							|  |  |  | 	INCREF(None); | 
					
						
							|  |  |  | 	return None; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* s.connect(sockaddr) method */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | sock_connect(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct sockaddr *addr; | 
					
						
							|  |  |  | 	int addrlen; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (!getsockaddrarg(s, args, &addr, &addrlen)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							|  |  |  | 	res = connect(s->sock_fd, addr, addrlen); | 
					
						
							|  |  |  | 	END_SAVE | 
					
						
							|  |  |  | 	if (res < 0) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	INCREF(None); | 
					
						
							|  |  |  | 	return None; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-06-23 09:07:03 +00:00
										 |  |  | /* s.fileno() method */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | sock_fileno(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!getnoarg(args)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	return newintobject((long) s->sock_fd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-11-26 08:54:07 +00:00
										 |  |  | /* s.getsockname() method */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | sock_getsockname(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char addrbuf[256]; | 
					
						
							|  |  |  | 	int addrlen, res; | 
					
						
							|  |  |  | 	if (!getnoarg(args)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	if (!getsockaddrlen(s, &addrlen)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	BGN_SAVE | 
					
						
							|  |  |  | 	res = getsockname(s->sock_fd, (struct sockaddr *) addrbuf, &addrlen); | 
					
						
							|  |  |  | 	END_SAVE | 
					
						
							|  |  |  | 	if (res < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	return makesockaddr((struct sockaddr *) addrbuf, addrlen); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-04-07 14:06:14 +00:00
										 |  |  | #ifndef NO_GETPEERNAME
 | 
					
						
							| 
									
										
										
										
											1992-11-26 08:54:07 +00:00
										 |  |  | /* s.getpeername() method */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | sock_getpeername(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char addrbuf[256]; | 
					
						
							|  |  |  | 	int addrlen, res; | 
					
						
							|  |  |  | 	if (!getnoarg(args)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	if (!getsockaddrlen(s, &addrlen)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	BGN_SAVE | 
					
						
							|  |  |  | 	res = getpeername(s->sock_fd, (struct sockaddr *) addrbuf, &addrlen); | 
					
						
							|  |  |  | 	END_SAVE | 
					
						
							|  |  |  | 	if (res < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	return makesockaddr((struct sockaddr *) addrbuf, addrlen); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1993-04-07 14:06:14 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1992-11-26 08:54:07 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | /* s.listen(n) method */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | sock_listen(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int backlog; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (!getintarg(args, &backlog)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							|  |  |  | 	res = listen(s->sock_fd, backlog); | 
					
						
							|  |  |  | 	END_SAVE | 
					
						
							|  |  |  | 	if (res < 0) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	INCREF(None); | 
					
						
							|  |  |  | 	return None; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* s.makefile(mode) method.
 | 
					
						
							|  |  |  |    Create a new open file object referring to a dupped version of | 
					
						
							|  |  |  |    the socket's file descriptor.  (The dup() call is necessary so | 
					
						
							|  |  |  |    that the open file and socket objects may be closed independent | 
					
						
							|  |  |  |    of each other.) | 
					
						
							|  |  |  |    The mode argument specifies 'r' or 'w' passed to fdopen(). */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static object * | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | sock_makefile(s, args) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	extern int fclose PROTO((FILE *)); | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	char *mode; | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	int fd; | 
					
						
							|  |  |  | 	FILE *fp; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	if (!getargs(args, "s", &mode)) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	if ((fd = dup(s->sock_fd)) < 0 || | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	    (fp = fdopen(fd, mode)) == NULL) | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 		return socket_error(); | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	return newopenfileobject(fp, "<socket>", mode, fclose); | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | /* s.recv(nbytes [,flags]) method */ | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | sock_recv(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int len, n, flags; | 
					
						
							|  |  |  | 	object *buf; | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 	flags = 0; | 
					
						
							|  |  |  | 	if (!getargs(args, "i", &len)) { | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 		err_clear(); | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 		if (!getargs(args, "(ii)", &len, &flags)) | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 			return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	buf = newsizedstringobject((char *) 0, len); | 
					
						
							|  |  |  | 	if (buf == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	n = recv(s->sock_fd, getstringvalue(buf), len, flags); | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	END_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (n < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	if (resizestring(&buf, n) < 0) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	return buf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | /* s.recvfrom(nbytes [,flags]) method */ | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | sock_recvfrom(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char addrbuf[256]; | 
					
						
							| 
									
										
										
										
											1993-02-05 09:46:15 +00:00
										 |  |  | 	object *buf, *addr, *ret; | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 	int addrlen, len, n, flags; | 
					
						
							|  |  |  | 	flags = 0; | 
					
						
							|  |  |  | 	if (!getargs(args, "i", &len)) { | 
					
						
							|  |  |  | 		err_clear(); | 
					
						
							|  |  |  | 		if (!getargs(args, "(ii)", &len, &flags)) | 
					
						
							|  |  |  | 		    return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1993-05-25 12:16:29 +00:00
										 |  |  | 	if (!getsockaddrlen(s, &addrlen)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	buf = newsizedstringobject((char *) 0, len); | 
					
						
							| 
									
										
										
										
											1993-05-25 12:16:29 +00:00
										 |  |  | 	if (buf == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 	n = recvfrom(s->sock_fd, getstringvalue(buf), len, flags, | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		     addrbuf, &addrlen); | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	END_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (n < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	if (resizestring(&buf, n) < 0) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1993-02-05 09:46:15 +00:00
										 |  |  | 	addr = makesockaddr((struct sockaddr *)addrbuf, addrlen); | 
					
						
							|  |  |  | 	ret = mkvalue("OO", buf, addr); | 
					
						
							|  |  |  | 	XDECREF(addr); | 
					
						
							|  |  |  | 	XDECREF(buf); | 
					
						
							|  |  |  | 	return ret; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | /* s.send(data [,flags]) method */ | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | sock_send(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	char *buf; | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	int len, n, flags; | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 	flags = 0; | 
					
						
							| 
									
										
										
										
											1993-06-17 12:35:49 +00:00
										 |  |  | 	if (!getargs(args, "s#", &buf, &len)) { | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 		err_clear(); | 
					
						
							| 
									
										
										
										
											1993-06-17 12:35:49 +00:00
										 |  |  | 		if (!getargs(args, "(s#i)", &buf, &len, &flags)) | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 			return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							|  |  |  | 	n = send(s->sock_fd, buf, len, flags); | 
					
						
							|  |  |  | 	END_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	if (n < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	INCREF(None); | 
					
						
							|  |  |  | 	return None; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | /* s.sendto(data, [flags,] sockaddr) method */ | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | sock_sendto(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	object *addro; | 
					
						
							|  |  |  | 	char *buf; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	struct sockaddr *addr; | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 	int addrlen, len, n, flags; | 
					
						
							|  |  |  | 	flags = 0; | 
					
						
							|  |  |  | 	if (!getargs(args, "(s#O)", &buf, &len, &addro)) { | 
					
						
							|  |  |  | 		err_clear(); | 
					
						
							|  |  |  | 		if (!getargs(args, "(s#iO)", &buf, &len, &flags, &addro)) | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 	if (!getsockaddrarg(s, addro, &addr, &addrlen)) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							| 
									
										
										
										
											1993-05-25 09:38:27 +00:00
										 |  |  | 	n = sendto(s->sock_fd, buf, len, flags, addr, addrlen); | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	END_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (n < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	INCREF(None); | 
					
						
							|  |  |  | 	return None; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* s.shutdown(how) method */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | sock_shutdown(s, args) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int how; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (!getintarg(args, &how)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							|  |  |  | 	res = shutdown(s->sock_fd, how); | 
					
						
							|  |  |  | 	END_SAVE | 
					
						
							|  |  |  | 	if (res < 0) | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	INCREF(None); | 
					
						
							|  |  |  | 	return None; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* List of methods for socket objects */ | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct methodlist sock_methods[] = { | 
					
						
							|  |  |  | 	{"accept",	sock_accept}, | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | 	{"avail",	sock_avail}, | 
					
						
							| 
									
										
										
										
											1991-10-20 20:22:28 +00:00
										 |  |  | 	{"allowbroadcast",	sock_allowbroadcast}, | 
					
						
							| 
									
										
										
										
											1992-06-05 15:11:30 +00:00
										 |  |  | 	{"setsockopt",	sock_setsockopt}, | 
					
						
							|  |  |  | 	{"getsockopt",	sock_getsockopt}, | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	{"bind",	sock_bind}, | 
					
						
							|  |  |  | 	{"close",	sock_close}, | 
					
						
							|  |  |  | 	{"connect",	sock_connect}, | 
					
						
							| 
									
										
										
										
											1992-06-23 09:07:03 +00:00
										 |  |  | 	{"fileno",	sock_fileno}, | 
					
						
							| 
									
										
										
										
											1992-11-26 08:54:07 +00:00
										 |  |  | 	{"getsockname",	sock_getsockname}, | 
					
						
							| 
									
										
										
										
											1993-04-07 14:06:14 +00:00
										 |  |  | #ifndef NO_GETPEERNAME
 | 
					
						
							| 
									
										
										
										
											1992-11-26 08:54:07 +00:00
										 |  |  | 	{"getpeername",	sock_getpeername}, | 
					
						
							| 
									
										
										
										
											1993-04-07 14:06:14 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	{"listen",	sock_listen}, | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	{"makefile",	sock_makefile}, | 
					
						
							|  |  |  | 	{"recv",	sock_recv}, | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	{"recvfrom",	sock_recvfrom}, | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	{"send",	sock_send}, | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	{"sendto",	sock_sendto}, | 
					
						
							|  |  |  | 	{"shutdown",	sock_shutdown}, | 
					
						
							|  |  |  | 	{NULL,		NULL}		/* sentinel */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Deallocate a socket object in response to the last DECREF().
 | 
					
						
							|  |  |  |    First close the file description. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | sock_dealloc(s) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	(void) close(s->sock_fd); | 
					
						
							|  |  |  | 	DEL(s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Return a socket object's named attribute. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static object * | 
					
						
							|  |  |  | sock_getattr(s, name) | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	char *name; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return findmethod(sock_methods, (object *) s, name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Type object for socket objects.
 | 
					
						
							| 
									
										
										
										
											1991-09-10 14:57:12 +00:00
										 |  |  |    XXX This should be static, but some compilers don't grok the | 
					
						
							|  |  |  |    XXX forward reference to it in that case... */ | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-09-10 14:57:12 +00:00
										 |  |  | typeobject Socktype = { | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	OB_HEAD_INIT(&Typetype) | 
					
						
							|  |  |  | 	0, | 
					
						
							|  |  |  | 	"socket", | 
					
						
							|  |  |  | 	sizeof(sockobject), | 
					
						
							|  |  |  | 	0, | 
					
						
							|  |  |  | 	sock_dealloc,	/*tp_dealloc*/ | 
					
						
							|  |  |  | 	0,		/*tp_print*/ | 
					
						
							|  |  |  | 	sock_getattr,	/*tp_getattr*/ | 
					
						
							|  |  |  | 	0,		/*tp_setattr*/ | 
					
						
							|  |  |  | 	0,		/*tp_compare*/ | 
					
						
							|  |  |  | 	0,		/*tp_repr*/ | 
					
						
							|  |  |  | 	0,		/*tp_as_number*/ | 
					
						
							|  |  |  | 	0,		/*tp_as_sequence*/ | 
					
						
							|  |  |  | 	0,		/*tp_as_mapping*/ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | /* Python interface to gethostname(). */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*ARGSUSED*/ | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | socket_gethostname(self, args) | 
					
						
							|  |  |  | 	object *self; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char buf[1024]; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | 	if (!getnoarg(args)) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							|  |  |  | 	res = gethostname(buf, (int) sizeof buf - 1); | 
					
						
							|  |  |  | 	END_SAVE | 
					
						
							|  |  |  | 	if (res < 0) | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	buf[sizeof buf - 1] = '\0'; | 
					
						
							|  |  |  | 	return newstringobject(buf); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | /* Python interface to gethostbyname(name). */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*ARGSUSED*/ | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | socket_gethostbyname(self, args) | 
					
						
							|  |  |  | 	object *self; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *name; | 
					
						
							|  |  |  | 	struct sockaddr_in addrbuf; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	if (!getargs(args, "s", &name)) | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	if (setipaddr(name, &addrbuf) < 0) | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	return makeipaddr(&addrbuf); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Python interface to getservbyname(name).
 | 
					
						
							|  |  |  |    This only returns the port number, since the other info is already | 
					
						
							|  |  |  |    known or not useful (like the list of aliases). */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*ARGSUSED*/ | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | socket_getservbyname(self, args) | 
					
						
							|  |  |  | 	object *self; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	char *name, *proto; | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	struct servent *sp; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	if (!getargs(args, "(ss)", &name, &proto)) | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							|  |  |  | 	sp = getservbyname(name, proto); | 
					
						
							|  |  |  | 	END_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	if (sp == NULL) { | 
					
						
							|  |  |  | 		err_setstr(SocketError, "service/proto not found"); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return newintobject((long) ntohs(sp->s_port)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Python interface to socket(family, type, proto).
 | 
					
						
							|  |  |  |    The third (protocol) argument is optional. | 
					
						
							|  |  |  |    Return a new socket object. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | /*ARGSUSED*/ | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | socket_socket(self, args) | 
					
						
							|  |  |  | 	object *self; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							| 
									
										
										
										
											1992-12-08 13:38:24 +00:00
										 |  |  | 	int fd, family, type, proto; | 
					
						
							|  |  |  | 	proto = 0; | 
					
						
							|  |  |  | 	if (!getargs(args, "(ii)", &family, &type)) { | 
					
						
							|  |  |  | 		err_clear(); | 
					
						
							|  |  |  | 		if (!getargs(args, "(iii)", &family, &type, &proto)) | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 			return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	BGN_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	fd = socket(family, type, proto); | 
					
						
							| 
									
										
										
										
											1992-08-05 19:58:53 +00:00
										 |  |  | 	END_SAVE | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (fd < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							|  |  |  | 	s = newsockobject(fd, family, type, proto); | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	/* If the object can't be created, don't forget to close the
 | 
					
						
							|  |  |  | 	   file descriptor again! */ | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	if (s == NULL) | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 		(void) close(fd); | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | 	/* From now on, ignore SIGPIPE and let the error checking
 | 
					
						
							|  |  |  | 	   do the work. */ | 
					
						
							|  |  |  | 	(void) signal(SIGPIPE, SIG_IGN); | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | 	return (object *) s; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-12-08 13:38:24 +00:00
										 |  |  | /* Create a socket object from a numeric file description.
 | 
					
						
							|  |  |  |    Useful e.g. if stdin is a socket. | 
					
						
							|  |  |  |    Additional arguments as for socket(). */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*ARGSUSED*/ | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | socket_fromfd(self, args) | 
					
						
							|  |  |  | 	object *self; | 
					
						
							|  |  |  | 	object *args; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	sockobject *s; | 
					
						
							|  |  |  | 	int fd, family, type, proto; | 
					
						
							|  |  |  | 	proto = 0; | 
					
						
							|  |  |  | 	if (!getargs(args, "(iii)", &fd, &family, &type)) { | 
					
						
							|  |  |  | 		err_clear(); | 
					
						
							|  |  |  | 		if (!getargs(args, "(iiii)", &fd, &family, &type, &proto)) | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
											
												* mymalloc.h: always allocate one extra byte, since some malloc's
  return NULL for malloc(0) or realloc(p, 0).  (This should be done
  differently than wasting one byte, but alas...)
* Moved "add'l libraries" option in Makefile to an earlier place.
* Remove argument compatibility hacks (b) and (c).
* Add grey2mono, dither2mono and mono2grey to imageop.
* Dup the fd in socket.fromfd().
* Added new modules mpz, md5 (by JH, requiring GNU MP 1.2).  Affects
  Makefile and config.c.
* socketmodule.c: added socket.fromfd(fd, family, type, [proto]),
  converted socket() to use of getargs().
											
										 
											1992-12-14 16:59:51 +00:00
										 |  |  | 	/* Dup the fd so it and the socket can be closed independently */ | 
					
						
							|  |  |  | 	fd = dup(fd); | 
					
						
							|  |  |  | 	if (fd < 0) | 
					
						
							|  |  |  | 		return socket_error(); | 
					
						
							| 
									
										
										
										
											1992-12-08 13:38:24 +00:00
										 |  |  | 	s = newsockobject(fd, family, type, proto); | 
					
						
							|  |  |  | 	/* From now on, ignore SIGPIPE and let the error checking
 | 
					
						
							|  |  |  | 	   do the work. */ | 
					
						
							|  |  |  | 	(void) signal(SIGPIPE, SIG_IGN); | 
					
						
							|  |  |  | 	return (object *) s; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | /* List of functions exported by this module. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static struct methodlist socket_methods[] = { | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	{"gethostbyname",	socket_gethostbyname}, | 
					
						
							| 
									
										
										
										
											1991-07-27 21:42:02 +00:00
										 |  |  | 	{"gethostname",		socket_gethostname}, | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	{"getservbyname",	socket_getservbyname}, | 
					
						
							|  |  |  | 	{"socket",		socket_socket}, | 
					
						
							| 
									
										
										
										
											1992-12-08 13:38:24 +00:00
										 |  |  | 	{"fromfd",		socket_fromfd}, | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 	{NULL,			NULL}		 /* Sentinel */ | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Convenience routine to export an integer value.
 | 
					
						
							|  |  |  |    For simplicity, errors (which are unlikely anyway) are ignored. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | insint(d, name, value) | 
					
						
							|  |  |  | 	object *d; | 
					
						
							|  |  |  | 	char *name; | 
					
						
							|  |  |  | 	int value; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *v = newintobject((long) value); | 
					
						
							|  |  |  | 	if (v == NULL) { | 
					
						
							|  |  |  | 		/* Don't bother reporting this error */ | 
					
						
							|  |  |  | 		err_clear(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		dictinsert(d, name, v); | 
					
						
							|  |  |  | 		DECREF(v); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-27 15:51:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Initialize this module.
 | 
					
						
							|  |  |  |    This is called when the first 'import socket' is done, | 
					
						
							|  |  |  |    via a table in config.c, if config.c is compiled with USE_SOCKET | 
					
						
							|  |  |  |    defined. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1991-06-25 21:36:08 +00:00
										 |  |  | void | 
					
						
							|  |  |  | initsocket() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *m, *d; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	m = initmodule("socket", socket_methods); | 
					
						
							|  |  |  | 	d = getmoduledict(m); | 
					
						
							|  |  |  | 	SocketError = newstringobject("socket.error"); | 
					
						
							|  |  |  | 	if (SocketError == NULL || dictinsert(d, "error", SocketError) != 0) | 
					
						
							|  |  |  | 		fatal("can't define socket.error"); | 
					
						
							|  |  |  | 	insint(d, "AF_INET", AF_INET); | 
					
						
							|  |  |  | 	insint(d, "AF_UNIX", AF_UNIX); | 
					
						
							|  |  |  | 	insint(d, "SOCK_STREAM", SOCK_STREAM); | 
					
						
							|  |  |  | 	insint(d, "SOCK_DGRAM", SOCK_DGRAM); | 
					
						
							|  |  |  | 	insint(d, "SOCK_RAW", SOCK_RAW); | 
					
						
							|  |  |  | 	insint(d, "SOCK_SEQPACKET", SOCK_SEQPACKET); | 
					
						
							|  |  |  | 	insint(d, "SOCK_RDM", SOCK_RDM); | 
					
						
							|  |  |  | } |