/************************************************************************* * * * YAP Prolog %W% %G% * * * Yap Prolog was developed at NCCUP - Universidade do Porto * * * * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997 * * * ************************************************************************** * * * File: io.h * * Last rev: 19/2/88 * * mods: * * comments: control YAP from sockets. * * * *************************************************************************/ #include "sysbits.h" #if HAVE_SOCKET #if HAVE_UNISTD_H && !defined(__MINGW32__) && !_MSC_VER #include <unistd.h> #endif #if STDC_HEADERS #include <stdlib.h> #endif #if HAVE_SYS_TYPES_H #include <sys/types.h> #endif #if HAVE_SYS_TIME_H && !defined(__MINGW32__) && !_MSC_VER #include <sys/time.h> #endif #ifdef _WIN32 #if HAVE_IO_H #include <io.h> #endif #endif #if _MSC_VER || defined(__MINGW32__) #include <io.h> #include <winsock2.h> #else #if HAVE_SYS_SOCKET_H #include <sys/socket.h> #endif #if HAVE_SYS_UN_H #include <sys/un.h> #endif #if HAVE_NETDB_H #include <netdb.h> #endif #if HAVE_NETINET_IN_H #include <netinet/in.h> #endif #if HAVE_ARPA_INET_H #include <arpa/inet.h> #endif #if HAVE_FCNTL_H #include <fcntl.h> #endif #if HAVE_STRING_H #include <string.h> #endif #if HAVE_SYS_SELECT_H #include <sys/select.h> #endif #if HAVE_SYS_PARAM_H #include <sys/param.h> #endif #endif #ifdef _WIN32 //#include <ws2tcpip.h> typedef int socklen_t; #endif /* make sure we can compile in any platform */ #ifndef AF_UNSPEC #define AF_UNSPEC 0 #endif #ifndef AF_LOCAL #define AF_LOCAL AF_UNSPEC #endif #ifndef AF_AAL5 #define AF_AAL5 AF_UNSPEC #endif #ifndef AF_APPLETALK #define AF_APPLETALK AF_UNSPEC #endif #ifndef AF_AX25 #define AF_AX25 AF_UNSPEC #endif #ifndef AF_BRIDGE #define AF_BRIDGE AF_UNSPEC #endif #ifndef AF_DECnet #define AF_DECnet AF_UNSPEC #endif #ifndef AF_FILE #define AF_FILE AF_UNSPEC #endif #ifndef AF_INET #define AF_INET AF_UNSPEC #endif #ifndef AF_INET6 #define AF_INET6 AF_UNSPEC #endif #ifndef AF_IPX #define AF_IPX AF_UNSPEC #endif #ifndef AF_LOCAL #define AF_LOCAL AF_UNSPEC #endif #ifndef AF_NETBEUI #define AF_NETBEUI AF_UNSPEC #endif #ifndef AF_NETLINK #define AF_NETLINK AF_UNSPEC #endif #ifndef AF_NETROM #define AF_NETROM AF_UNSPEC #endif #ifndef AF_OSINET #define AF_OSINET AF_UNSPEC #endif #ifndef AF_PACKET #define AF_PACKET AF_UNSPEC #endif #ifndef AF_ROSE #define AF_ROSE AF_UNSPEC #endif #ifndef AF_ROUTE #define AF_ROUTE AF_UNSPEC #endif #ifndef AF_SECURITY #define AF_SECURITY AF_UNSPEC #endif #ifndef AF_SNA #define AF_SNA AF_UNSPEC #endif #ifndef AF_UNIX #define AF_UNIX AF_UNSPEC #endif #ifndef AF_X25 #define AF_X25 AF_UNSPEC #endif #ifndef SOCK_STREAM #define SOCK_STREAM -1 #endif #ifndef SOCK_DGRAM #define SOCK_DGRAM -1 #endif #ifndef SOCK_RAW #define SOCK_RAW -1 #endif #ifndef SOCK_RDM #define SOCK_RDM -1 #endif #ifndef SOCK_SEQPACKET #define SOCK_SEQPACKET -1 #endif #ifndef SOCK_PACKET #define SOCK_PACKET -1 #endif #ifndef MAXHOSTNAMELEN #define MAXHOSTNAMELEN 256 #endif #ifndef BUFSIZ #define BUFSIZ 256 #endif #if _MSC_VER || defined(__MINGW32__) #define socket_errno WSAGetLastError() #define invalid_socket_fd(fd) (fd) == INVALID_SOCKET #else #define socket_errno errno #define invalid_socket_fd(fd) (fd) < 0 #endif void Yap_init_socks(char *host, long interface_port) { int s; int r; struct sockaddr_in soadr; struct in_addr adr; struct hostent *he; #if HAVE_SOCKET he = gethostbyname(host); if (he == NULL) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "can not get address for host %s: %s", host, strerror(h_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "can not get address for host"); #endif return; } (void)memset((void *)&soadr, '\0', sizeof(struct sockaddr_in)); soadr.sin_family = AF_INET; soadr.sin_port = htons((short)interface_port); if (he != NULL) { memmove((char *)&adr, (char *)he->h_addr_list[0], (size_t)he->h_length); } else { adr.s_addr = inet_addr(host); } soadr.sin_addr.s_addr = adr.s_addr; s = socket(AF_INET, SOCK_STREAM, 0); if (s < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "could not create socket: %s", strerror(errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "could not create socket"); #endif return; } #if ENABLE_SO_LINGER struct linger ling; /* disables socket lingering. */ ling.l_onoff = 1; ling.l_linger = 0; if (setsockopt(s, SOL_SOCKET, SO_LINGER, (void *)&ling, sizeof(ling)) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (setsockopt_linger: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (setsockopt_linger)"); #endif return; } #endif r = connect(s, (struct sockaddr *)&soadr, sizeof(soadr)); if (r < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "connect failed, could not connect to interface: %s", strerror(errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "connect failed, could not connect to interface"); #endif return; } /* now reopen stdin stdout and stderr */ #if HAVE_DUP2 && !defined(__MINGW32__) if (dup2(s, 0) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "could not dup2 stdin: %s", strerror(errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "could not dup2 stdin"); #endif return; } if (dup2(s, 1) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "could not dup2 stdout: %s", strerror(errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "could not dup2 stdout"); #endif return; } if (dup2(s, 2) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "could not dup2 stderr: %s", strerror(errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "could not dup2 stderr"); #endif return; } #elif _MSC_VER || defined(__MINGW32__) if (_dup2(s, 0) < 0) { fprintf(stderr, "could not dup2 stdin\n"); return; } if (_dup2(s, 1) < 0) { fprintf(stderr, "could not dup2 stdout\n"); return; } if (_dup2(s, 2) < 0) { fprintf(stderr, "could not dup2 stderr\n"); return; } #else if (dup2(s, 0) < 0) { fprintf(stderr, "could not dup2 stdin\n"); return; } yp_iob[0].cnt = 0; yp_iob[0].flags = _YP_IO_SOCK | _YP_IO_READ; if (dup2(s, 1) < 0) { fprintf(stderr, "could not dup2 stdout\n"); return; } yp_iob[1].cnt = 0; yp_iob[1].flags = _YP_IO_SOCK | _YP_IO_WRITE; if (dup2(s, 2) < 0) { fprintf(stderr, "could not dup2 stderr\n"); return; } yp_iob[2].cnt = 0; yp_iob[2].flags = _YP_IO_SOCK | _YP_IO_WRITE; #endif // Yap_sockets_io = 1; #if _MSC_VER || defined(__MINGW32__) _close(s); #else close(s); #endif #else /* HAVE_SOCKET */ Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "sockets not installed", strerror(errno)); #endif /* HAVE_SOCKET */ } static Int p_socket(USES_REGS1) { Term t1 = Deref(ARG1); Term t2 = Deref(ARG2); Term t3 = Deref(ARG3); char *sdomain, *stype; Int domain = AF_UNSPEC, type, protocol; int fd; Term out; if (IsVarTerm(t1)) { Yap_Error(INSTANTIATION_ERROR, t1, "socket/4"); return (FALSE); } if (!IsAtomTerm(t1)) { Yap_Error(TYPE_ERROR_ATOM, t1, "socket/4"); return (FALSE); } if (IsVarTerm(t2)) { Yap_Error(INSTANTIATION_ERROR, t2, "socket/4"); return (FALSE); } if (!IsAtomTerm(t2)) { Yap_Error(TYPE_ERROR_ATOM, t2, "socket/4"); return (FALSE); } if (IsVarTerm(t3)) { Yap_Error(INSTANTIATION_ERROR, t3, "socket/4"); return (FALSE); } if (!IsIntTerm(t3)) { Yap_Error(TYPE_ERROR_ATOM, t3, "socket/4"); return (FALSE); } sdomain = RepAtom(AtomOfTerm(t1))->StrOfAE; if (sdomain[0] != 'A' || sdomain[1] != 'F' || sdomain[2] != '_') return (FALSE); /* Error */ sdomain += 3; switch (sdomain[0]) { case 'A': if (strcmp(sdomain, "AAL5") == 0) domain = AF_AAL5; else if (strcmp(sdomain, "APPLETALK") == 0) domain = AF_APPLETALK; else if (strcmp(sdomain, "AX25") == 0) domain = AF_AX25; break; case 'B': if (strcmp(sdomain, "BRIDGE") == 0) domain = AF_APPLETALK; break; case 'D': if (strcmp(sdomain, "DECnet") == 0) domain = AF_DECnet; break; case 'F': if (strcmp(sdomain, "FILE") == 0) domain = AF_FILE; break; case 'I': if (strcmp(sdomain, "INET") == 0) domain = AF_INET; else if (strcmp(sdomain, "INET6") == 0) domain = AF_INET6; else if (strcmp(sdomain, "IPX") == 0) domain = AF_IPX; break; case 'L': if (strcmp(sdomain, "LOCAL") == 0) domain = AF_LOCAL; break; case 'N': if (strcmp(sdomain, "NETBEUI") == 0) domain = AF_NETBEUI; else if (strcmp(sdomain, "NETLINK") == 0) domain = AF_NETLINK; else if (strcmp(sdomain, "NETROM") == 0) domain = AF_NETROM; break; case 'O': if (strcmp(sdomain, "OSINET") == 0) domain = AF_OSINET; break; case 'P': if (strcmp(sdomain, "PACKET") == 0) domain = AF_PACKET; break; case 'R': if (strcmp(sdomain, "ROSE") == 0) domain = AF_ROSE; else if (strcmp(sdomain, "ROUTE") == 0) domain = AF_ROUTE; break; case 'S': if (strcmp(sdomain, "SECURITY") == 0) domain = AF_SECURITY; else if (strcmp(sdomain, "SNA") == 0) domain = AF_SNA; break; case 'U': if (strcmp(sdomain, "UNIX") == 0) domain = AF_UNIX; break; case 'X': if (strcmp(sdomain, "X25") == 0) domain = AF_X25; break; } stype = RepAtom(AtomOfTerm(t2))->StrOfAE; if (stype[0] != 'S' || stype[1] != 'O' || stype[2] != 'C' || stype[3] != 'K' || stype[4] != '_') return (FALSE); /* Error */ stype += 5; if (strcmp(stype, "STREAM") == 0) type = SOCK_STREAM; else if (strcmp(stype, "DGRAM") == 0) type = SOCK_DGRAM; else if (strcmp(stype, "RAW") == 0) type = SOCK_RAW; else if (strcmp(stype, "RDM") == 0) type = SOCK_RDM; else if (strcmp(stype, "SEQPACKET") == 0) type = SOCK_SEQPACKET; else if (strcmp(stype, "PACKET") == 0) type = SOCK_PACKET; else return (FALSE); protocol = IntOfTerm(t3); if (protocol < 0) return (FALSE); fd = socket(domain, type, protocol); if (invalid_socket_fd(fd)) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket/4 (socket: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket/4 (socket)"); #endif return (FALSE); } if (domain == AF_UNIX || domain == AF_LOCAL) out = Yap_InitSocketStream(fd, new_socket, af_unix); else if (domain == AF_INET) out = Yap_InitSocketStream(fd, new_socket, af_inet); else { /* ok, we currently don't support these sockets */ #if _MSC_VER || defined(__MINGW32__) _close(fd); #else close(fd); #endif return (FALSE); } if (out == TermNil) return (FALSE); return (Yap_unify(out, ARG4)); } Int Yap_CloseSocket(int fd, socket_info status, socket_domain domain) { #if _MSC_VER || defined(__MINGW32__) /* prevent further writing to the socket */ if (status == server_session_socket || status == client_socket) { char bfr; if (shutdown(fd, 1) != 0) { Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_close/1 (close)"); return (FALSE); } /* read all pending characters from the socket */ while (recv(fd, &bfr, 1, 0) > 0) ; /* prevent further reading from the socket */ if (shutdown(fd, 0) < 0) { Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_close/1 (close)"); return (FALSE); } /* close the socket */ if (closesocket(fd) != 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_close/1 (close: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_close/1 (close)"); #endif } #else if (status == server_session_socket || status == client_socket) { if (shutdown(fd, 2) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_close/1 (shutdown: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_close/1 (shutdown)"); #endif return (FALSE); } } if (close(fd) != 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_close/1 (close: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_close/1 (close)"); #endif #endif return (FALSE); } return (TRUE); } static Int p_socket_close(USES_REGS1) { Term t1 = Deref(ARG1); int sno; if ((sno = Yap_CheckSocketStream(t1, "socket_close/1")) < 0) { return (FALSE); } Yap_CloseStream(sno); return (TRUE); } static Int p_socket_bind(USES_REGS1) { Term t1 = Deref(ARG1); Term t2 = Deref(ARG2); int sno; Functor fun; socket_info status; int fd; if ((sno = Yap_CheckSocketStream(t1, "socket_bind/2")) < 0) { return (FALSE); } status = Yap_GetSocketStatus(sno); fd = Yap_GetStreamFd(sno); if (status != new_socket) { /* ok, this should be an error, as you are trying to bind */ return (FALSE); } if (IsVarTerm(t2)) { Yap_Error(INSTANTIATION_ERROR, t2, "socket_bind/2"); return (FALSE); } if (!IsApplTerm(t2)) { Yap_Error(DOMAIN_ERROR_STREAM, t2, "socket_bind/2"); return (FALSE); } fun = FunctorOfTerm(t2); #if HAVE_SYS_UN_H if (fun == FunctorAfUnix || fun == FunctorAfLocal) { struct sockaddr_un sock; Term taddr = ArgOfTerm(1, t2); char *s; int len; if (IsVarTerm(taddr)) { Yap_Error(INSTANTIATION_ERROR, t2, "socket_bind/2"); return (FALSE); } if (!IsAtomTerm(taddr)) { Yap_Error(TYPE_ERROR_ATOM, taddr, "socket_bind/2"); return (FALSE); } s = RepAtom(AtomOfTerm(taddr))->StrOfAE; sock.sun_family = AF_UNIX; if ((len = strlen(s)) > 107) /* hit me with a broomstick */ { Yap_Error(DOMAIN_ERROR_STREAM, taddr, "socket_bind/2"); return (FALSE); } sock.sun_family = AF_UNIX; strcpy(sock.sun_path, s); if (bind(fd, (struct sockaddr *)(&sock), ((size_t)(((struct sockaddr_un *)0)->sun_path) + len)) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_bind/2 (bind: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_bind/2 (bind)"); #endif return (FALSE); } Yap_UpdateSocketStream(sno, server_socket, af_unix); return (TRUE); } else #endif if (fun == FunctorAfInet) { Term thost = ArgOfTerm(1, t2); Term tport = ArgOfTerm(2, t2); char *shost; struct hostent *he; struct sockaddr_in saddr; Int port; memset((void *)&saddr, (int)0, sizeof(saddr)); if (IsVarTerm(thost)) { saddr.sin_addr.s_addr = INADDR_ANY; } else if (!IsAtomTerm(thost)) { Yap_Error(TYPE_ERROR_ATOM, thost, "socket_bind/2"); return (FALSE); } else { shost = RepAtom(AtomOfTerm(thost))->StrOfAE; if ((he = gethostbyname(shost)) == NULL) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_bind/2 (gethostbyname: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_bind/2 (gethostbyname)"); #endif return (FALSE); } memmove((void *)&saddr.sin_addr, (void *)he->h_addr_list[0], he->h_length); } if (IsVarTerm(tport)) { port = 0; } else { port = IntOfTerm(tport); } saddr.sin_port = htons(port); saddr.sin_family = AF_INET; if (bind(fd, (struct sockaddr *)&saddr, sizeof(saddr)) == -1) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_bind/2 (bind: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_bind/2 (bind)"); #endif return (FALSE); } if (IsVarTerm(tport)) { /* get the port number */ #if _WIN32 || defined(__MINGW32__) int namelen; #else socklen_t namelen; #endif Term t; if (getsockname(fd, (struct sockaddr *)&saddr, &namelen) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_bind/2 (getsockname: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_bind/2 (getsockname)"); #endif return (FALSE); } t = MkIntTerm(ntohs(saddr.sin_port)); Yap_unify(ArgOfTermCell(2, t2), t); } Yap_UpdateSocketStream(sno, server_socket, af_inet); return (TRUE); } else return (FALSE); } static Int p_socket_connect(USES_REGS1) { Term t1 = Deref(ARG1); Term t2 = Deref(ARG2); Functor fun; int sno; socket_info status; int fd; int flag; Term out; if ((sno = Yap_CheckSocketStream(t1, "socket_connect/3")) < 0) { return (FALSE); } if (IsVarTerm(t2)) { Yap_Error(INSTANTIATION_ERROR, t2, "socket_connect/3"); return (FALSE); } if (!IsApplTerm(t2)) { Yap_Error(DOMAIN_ERROR_STREAM, t2, "socket_connect/3"); return (FALSE); } fun = FunctorOfTerm(t2); fd = Yap_GetStreamFd(sno); status = Yap_GetSocketStatus(sno); if (status != new_socket) { /* ok, this should be an error, as you are trying to bind */ return (FALSE); } #if HAVE_SYS_UN_H if (fun == FunctorAfUnix) { struct sockaddr_un sock; Term taddr = ArgOfTerm(1, t2); char *s; int len; if (IsVarTerm(taddr)) { Yap_Error(INSTANTIATION_ERROR, t2, "socket_connect/3"); return (FALSE); } if (!IsAtomTerm(taddr)) { Yap_Error(TYPE_ERROR_ATOM, taddr, "socket_connect/3"); return (FALSE); } s = RepAtom(AtomOfTerm(taddr))->StrOfAE; sock.sun_family = AF_UNIX; if ((len = strlen(s)) > 107) /* beat me with a broomstick */ { Yap_Error(DOMAIN_ERROR_STREAM, taddr, "socket_connect/3"); return (FALSE); } sock.sun_family = AF_UNIX; strcpy(sock.sun_path, s); if ((flag = connect( fd, (struct sockaddr *)(&sock), ((size_t)(((struct sockaddr_un *)0)->sun_path) + len))) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (connect: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (connect)"); #endif return (FALSE); } Yap_UpdateSocketStream(sno, client_socket, af_unix); } else #endif if (fun == FunctorAfInet) { Term thost = ArgOfTerm(1, t2); Term tport = ArgOfTerm(2, t2); char *shost; struct hostent *he; struct sockaddr_in saddr; unsigned short int port; memset((void *)&saddr, (int)0, sizeof(saddr)); if (IsVarTerm(thost)) { Yap_Error(INSTANTIATION_ERROR, thost, "socket_connect/3"); return (FALSE); } else if (!IsAtomTerm(thost)) { Yap_Error(TYPE_ERROR_ATOM, thost, "socket_connect/3"); return (FALSE); } else { shost = RepAtom(AtomOfTerm(thost))->StrOfAE; if ((he = gethostbyname(shost)) == NULL) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (gethostbyname: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (gethostbyname)"); #endif return (FALSE); } memmove((void *)&saddr.sin_addr, (void *)he->h_addr_list[0], he->h_length); } if (IsVarTerm(tport)) { Yap_Error(INSTANTIATION_ERROR, tport, "socket_connect/3"); return (FALSE); } else if (!IsIntegerTerm(tport)) { Yap_Error(TYPE_ERROR_INTEGER, tport, "socket_connect/3"); return (FALSE); } else { port = (unsigned short int)IntegerOfTerm(tport); } saddr.sin_port = htons(port); saddr.sin_family = AF_INET; #if ENABLE_SO_LINGER { struct linger ling; /* For making sockets linger. */ /* disabled: I see why no reason why we should throw things away by * default!! */ ling.l_onoff = 1; ling.l_linger = 0; if (setsockopt(fd, SOL_SOCKET, SO_LINGER, (void *)&ling, sizeof(ling)) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (setsockopt_linger: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (setsockopt_linger)"); #endif return FALSE; } } #endif { int one = 1; /* code by David MW Powers */ if (setsockopt(fd, SOL_SOCKET, SO_BROADCAST, (void *)&one, sizeof(one))) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (setsockopt_broadcast: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (setsockopt_broadcast)"); #endif return FALSE; } } flag = connect(fd, (struct sockaddr *)&saddr, sizeof(saddr)); if (flag < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (connect: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_connect/3 (connect)"); #endif return FALSE; } Yap_UpdateSocketStream(sno, client_socket, af_inet); } else return (FALSE); out = t1; return (Yap_unify(out, ARG3)); } static Int p_socket_listen(USES_REGS1) { Term t1 = Deref(ARG1); Term t2 = Deref(ARG2); int sno; socket_info status; int fd; Int j; if ((sno = Yap_CheckSocketStream(t1, "socket_listen/2")) < 0) { return (FALSE); } if (IsVarTerm(t2)) { Yap_Error(INSTANTIATION_ERROR, t2, "socket_listen/2"); return (FALSE); } if (!IsIntTerm(t2)) { Yap_Error(TYPE_ERROR_INTEGER, t2, "socket_listen/2"); return (FALSE); } j = IntOfTerm(t2); if (j < 0) { Yap_Error(DOMAIN_ERROR_STREAM, t1, "socket_listen/2"); return (FALSE); } fd = Yap_GetStreamFd(sno); status = Yap_GetSocketStatus(sno); if (status != server_socket) { /* ok, this should be an error, as you are trying to bind */ return (FALSE); } if (listen(fd, j) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_listen/2 (listen: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_listen/2 (listen)"); #endif } return (TRUE); } static Int p_socket_accept(USES_REGS1) { Term t1 = Deref(ARG1); int sno; socket_info status; socket_domain domain; int ofd, fd; Term out; if ((sno = Yap_CheckSocketStream(t1, "socket_accept/3")) < 0) { return (FALSE); } ofd = Yap_GetStreamFd(sno); status = Yap_GetSocketStatus(sno); if (status != server_socket) { /* ok, this should be an error, as you are trying to bind */ return (FALSE); } domain = Yap_GetSocketDomain(sno); #if HAVE_SYS_UN_H if (domain == af_unix) { struct sockaddr_un caddr; socklen_t len; memset((void *)&caddr, (int)0, sizeof(caddr)); if ((fd = accept(ofd, (struct sockaddr *)&caddr, &len)) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_accept/3 (accept: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_accept/3 (accept)"); #endif } /* ignore 2nd argument */ out = Yap_InitSocketStream(fd, server_session_socket, af_unix); } else #endif if (domain == af_inet) { struct sockaddr_in caddr; Term tcli; char *s; socklen_t len; len = sizeof(caddr); memset((void *)&caddr, (int)0, sizeof(caddr)); if (invalid_socket_fd(fd = accept(ofd, (struct sockaddr *)&caddr, &len))) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_accept/3 (accept: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_accept/3 (accept)"); #endif return (FALSE); } if ((s = inet_ntoa(caddr.sin_addr)) == NULL) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_accept/3 (inet_ntoa: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_accept/3 (inet_ntoa)"); #endif } tcli = MkAtomTerm(Yap_LookupAtom(s)); if (!Yap_unify(ARG2, tcli)) return (FALSE); out = Yap_InitSocketStream(fd, server_session_socket, af_inet); } else return (FALSE); if (out == TermNil) return (FALSE); return (Yap_unify(out, ARG3)); } static Int p_socket_buffering(USES_REGS1) { Term t1 = Deref(ARG1); Term t2 = Deref(ARG2); Term t4 = Deref(ARG4); Atom mode; int fd; int writing; #if _WIN32 || defined(__MINGW32__) int bufsize; #else unsigned int bufsize; #endif int sno; socklen_t len; if ((sno = Yap_CheckSocketStream(t1, "socket_buffering/4")) < 0) { return (FALSE); } if (IsVarTerm(t2)) { Yap_Error(INSTANTIATION_ERROR, t2, "socket_buffering/4"); return (FALSE); } if (!IsAtomTerm(t2)) { Yap_Error(TYPE_ERROR_ATOM, t2, "socket_buffering/4"); return (FALSE); } mode = AtomOfTerm(t2); if (mode == AtomRead) writing = FALSE; else if (mode == AtomWrite) writing = TRUE; else { Yap_Error(DOMAIN_ERROR_IO_MODE, t2, "socket_buffering/4"); return (FALSE); } fd = Yap_GetStreamFd(sno); if (writing) { getsockopt(fd, SOL_SOCKET, SO_SNDBUF, (void *)&bufsize, &len); } else { getsockopt(fd, SOL_SOCKET, SO_RCVBUF, (void *)&bufsize, &len); } if (!Yap_unify(ARG3, MkIntegerTerm(bufsize))) return (FALSE); if (IsVarTerm(t4)) { bufsize = BUFSIZ; } else { Int siz; if (!IsIntegerTerm(t4)) { Yap_Error(TYPE_ERROR_INTEGER, t4, "socket_buffering/4"); return (FALSE); } siz = IntegerOfTerm(t4); if (siz < 0) { Yap_Error(DOMAIN_ERROR_NOT_LESS_THAN_ZERO, t4, "socket_buffering/4"); return (FALSE); } bufsize = siz; } if (writing) { setsockopt(fd, SOL_SOCKET, SO_SNDBUF, (void *)&bufsize, sizeof(bufsize)); } else { setsockopt(fd, SOL_SOCKET, SO_RCVBUF, (void *)&bufsize, sizeof(bufsize)); } return (TRUE); } static Term select_out_list(Term t1, fd_set *readfds_ptr USES_REGS) { if (t1 == TermNil) { return (TermNil); } else { int fd; int sno; Term next = select_out_list(TailOfTerm(t1), readfds_ptr PASS_REGS); Term Head = HeadOfTerm(t1); sno = Yap_CheckIOStream(Head, "stream_select/5"); fd = Yap_GetStreamFd(sno); if (FD_ISSET(fd, readfds_ptr)) return (MkPairTerm(Head, next)); else return (MkPairTerm(TermNil, next)); } } static Int p_socket_select(USES_REGS1) { Term t1 = Deref(ARG1); Term t2 = Deref(ARG2); Term t3 = Deref(ARG3); fd_set readfds, writefds, exceptfds; struct timeval timeout, *ptime; #if _MSC_VER || defined(__MINGW32__) u_int fdmax = 0; #else int fdmax = 0; #endif Int tsec, tusec; Term tout = TermNil, ti, Head; if (IsVarTerm(t1)) { Yap_Error(INSTANTIATION_ERROR, t1, "socket_select/5"); return (FALSE); } if (!IsPairTerm(t1)) { Yap_Error(TYPE_ERROR_LIST, t1, "socket_select/5"); return (FALSE); } if (IsVarTerm(t2)) { Yap_Error(INSTANTIATION_ERROR, t2, "socket_select/5"); return (FALSE); } if (!IsIntegerTerm(t2)) { Yap_Error(TYPE_ERROR_INTEGER, t2, "socket_select/5"); return (FALSE); } if (IsVarTerm(t3)) { Yap_Error(INSTANTIATION_ERROR, t3, "socket_select/5"); return (FALSE); } if (!IsIntegerTerm(t3)) { Yap_Error(TYPE_ERROR_INTEGER, t3, "socket_select/5"); return (FALSE); } FD_ZERO(&readfds); FD_ZERO(&writefds); FD_ZERO(&exceptfds); /* fetch the input streams */ ti = t1; while (ti != TermNil) { #if _MSC_VER || defined(__MINGW32__) u_int fd; #else int fd; #endif int sno; Head = HeadOfTerm(ti); sno = Yap_CheckIOStream(Head, "stream_select/5"); if (sno < 0) return (FALSE); fd = Yap_GetStreamFd(sno); FD_SET(fd, &readfds); if (fd > fdmax) fdmax = fd; ti = TailOfTerm(ti); } /* now, check the time */ tsec = IntegerOfTerm(t2); tusec = IntegerOfTerm(t3); if (tsec < 0) /* off */ { ptime = NULL; } else { timeout.tv_sec = tsec; timeout.tv_usec = tusec; ptime = &timeout; } /* do the real work */ if (select(fdmax + 1, &readfds, &writefds, &exceptfds, ptime) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_select/5 (select: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "socket_select/5 (select)"); #endif } tout = select_out_list(t1, &readfds PASS_REGS); /* we're done, just pass the info back */ return (Yap_unify(ARG4, tout)); } static Int p_current_host(USES_REGS1) { char oname[MAXHOSTNAMELEN], *name; Term t1 = Deref(ARG1), out; if (!IsVarTerm(t1) && !IsAtomTerm(t1)) { Yap_Error(TYPE_ERROR_ATOM, t1, "current_host/2"); return (FALSE); } name = oname; if (gethostname(name, sizeof(oname)) < 0) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "current_host/2 (gethostname: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "current_host/2 (gethostname)"); #endif return (FALSE); } if ((strrchr(name, '.') == NULL)) { struct hostent *he; /* not a fully qualified name, ask the name server */ if ((he = gethostbyname(name)) == NULL) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "current_host/2 (gethostbyname: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "current_host/2 (gethostbyname)"); #endif return (FALSE); } name = (char *)(he->h_name); } if (IsAtomTerm(t1)) { char *sin = RepAtom(AtomOfTerm(t1))->StrOfAE; int faq = (strrchr(sin, '.') != NULL); if (faq) #if _MSC_VER || defined(__MINGW32__) return (_stricmp(name, sin) == 0); #else return (strcasecmp(name, sin) == 0); #endif else { int isize = strlen(sin); if (isize >= 256) { Yap_Error(SYSTEM_ERROR_INTERNAL, ARG1, "current_host/2 (input longer than longest FAQ host name)"); return (FALSE); } if (name[isize] != '.') return (FALSE); name[isize] = '\0'; #if _MSC_VER || defined(__MINGW32__) return (_stricmp(name, sin) == 0); #else return (strcasecmp(name, sin) == 0); #endif } } else { out = MkAtomTerm(Yap_LookupAtom(name)); return (Yap_unify(ARG1, out)); } } static Int p_hostname_address(USES_REGS1) { char *s; Term t1 = Deref(ARG1); Term t2 = Deref(ARG2); Term tin, out; struct hostent *he; if (!IsVarTerm(t1)) { if (!IsAtomTerm(t1)) { Yap_Error(TYPE_ERROR_ATOM, t1, "hostname_address/2"); return (FALSE); } else tin = t1; } else if (IsVarTerm(t2)) { Yap_Error(INSTANTIATION_ERROR, t1, "hostname_address/5"); return (FALSE); } else if (!IsAtomTerm(t2)) { Yap_Error(TYPE_ERROR_ATOM, t2, "hostname_address/2"); return (FALSE); } else tin = t2; s = RepAtom(AtomOfTerm(tin))->StrOfAE; if (IsVarTerm(t1)) { if ((he = gethostbyaddr(s, strlen(s), AF_INET)) == NULL) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "hostname_address/2 (gethostbyname: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "hostname_address/2 (gethostbyname)"); #endif } out = MkAtomTerm(Yap_LookupAtom((char *)(he->h_name))); return (Yap_unify(out, ARG1)); } else { struct in_addr adr; if ((he = gethostbyname(s)) == NULL) { #if HAVE_STRERROR Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "hostname_address/2 (gethostbyname: %s)", strerror(socket_errno)); #else Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "hostname_address/2 (gethostbyname)"); #endif return false; } memmove((char *)&adr, (char *)he->h_addr_list[0], (size_t)he->h_length); out = MkAtomTerm(Yap_LookupAtom(inet_ntoa(adr))); return (Yap_unify(out, ARG2)); } } #endif void Yap_InitSocketLayer(void) { #ifdef HAVE_SOCKET Yap_InitCPred("socket", 4, p_socket, SafePredFlag | SyncPredFlag); Yap_InitCPred("socket_close", 1, p_socket_close, SafePredFlag | SyncPredFlag); Yap_InitCPred("socket_bind", 2, p_socket_bind, SafePredFlag | SyncPredFlag); Yap_InitCPred("socket_connect", 3, p_socket_connect, SafePredFlag | SyncPredFlag); Yap_InitCPred("socket_listen", 2, p_socket_listen, SafePredFlag | SyncPredFlag); Yap_InitCPred("socket_accept", 3, p_socket_accept, SafePredFlag | SyncPredFlag); Yap_InitCPred("$socket_buffering", 4, p_socket_buffering, SafePredFlag | SyncPredFlag | HiddenPredFlag); Yap_InitCPred("$socket_select", 4, p_socket_select, SafePredFlag | SyncPredFlag | HiddenPredFlag); Yap_InitCPred("current_host", 1, p_current_host, SafePredFlag); Yap_InitCPred("hostname_address", 2, p_hostname_address, SafePredFlag); #if _MSC_VER || defined(__MINGW32__) { WSADATA info; if (WSAStartup(MAKEWORD(2, 1), &info) != 0) exit(1); } #endif #endif }