#define WIN32IO_IS_STDIO
#define WIN32SCK_IS_STDSCK
#define WIN32_LEAN_AND_MEAN
#define PERLIO_NOT_STDIO 0
#ifdef __GNUC__
#define Win32_Winsock
#endif
#include <wchar.h>
#include <windows.h>
#include <ws2spi.h>
#include "EXTERN.h"
#include "perl.h"
#include "Win32iop.h"
#include <sys/socket.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <assert.h>
#include <io.h>
#define OPEN_SOCKET(x) win32_open_osfhandle(x,O_RDWR|O_BINARY)
#define TO_SOCKET(x) _get_osfhandle(x)
#define SOCKET_TEST(x, y) \
STMT_START { \
if
((x) == (y)) \
{ \
int
wsaerr = WSAGetLastError(); \
errno
= convert_wsa_error_to_errno(wsaerr); \
SetLastError(wsaerr); \
} \
} STMT_END
#define SOCKET_TEST_ERROR(x) SOCKET_TEST(x, SOCKET_ERROR)
static
struct
servent* win32_savecopyservent(
struct
servent*d,
struct
servent*s,
const
char
*proto);
#ifdef WIN32_DYN_IOINFO_SIZE
EXTERN_C Size_t w32_ioinfo_size;
#endif
EXTERN_C
void
EndSockets(
void
)
{
WSACleanup();
}
int
convert_wsa_error_to_errno(
int
wsaerr)
{
switch
(wsaerr) {
case
WSAEINTR:
return
EINTR;
case
WSAEBADF:
return
EBADF;
case
WSAEACCES:
return
EACCES;
case
WSAEFAULT:
return
EFAULT;
case
WSAEINVAL:
return
EINVAL;
case
WSAEMFILE:
return
EMFILE;
case
WSAEWOULDBLOCK:
return
EWOULDBLOCK;
case
WSAEINPROGRESS:
return
EINPROGRESS;
case
WSAEALREADY:
return
EALREADY;
case
WSAENOTSOCK:
return
ENOTSOCK;
case
WSAEDESTADDRREQ:
return
EDESTADDRREQ;
case
WSAEMSGSIZE:
return
EMSGSIZE;
case
WSAEPROTOTYPE:
return
EPROTOTYPE;
case
WSAENOPROTOOPT:
return
ENOPROTOOPT;
case
WSAEPROTONOSUPPORT:
return
EPROTONOSUPPORT;
case
WSAESOCKTNOSUPPORT:
return
ESOCKTNOSUPPORT;
case
WSAEOPNOTSUPP:
return
EOPNOTSUPP;
case
WSAEPFNOSUPPORT:
return
EPFNOSUPPORT;
case
WSAEAFNOSUPPORT:
return
EAFNOSUPPORT;
case
WSAEADDRINUSE:
return
EADDRINUSE;
case
WSAEADDRNOTAVAIL:
return
EADDRNOTAVAIL;
case
WSAENETDOWN:
return
ENETDOWN;
case
WSAENETUNREACH:
return
ENETUNREACH;
case
WSAENETRESET:
return
ENETRESET;
case
WSAECONNABORTED:
return
ECONNABORTED;
case
WSAECONNRESET:
return
ECONNRESET;
case
WSAENOBUFS:
return
ENOBUFS;
case
WSAEISCONN:
return
EISCONN;
case
WSAENOTCONN:
return
ENOTCONN;
case
WSAESHUTDOWN:
return
ESHUTDOWN;
case
WSAETOOMANYREFS:
return
ETOOMANYREFS;
case
WSAETIMEDOUT:
return
ETIMEDOUT;
case
WSAECONNREFUSED:
return
ECONNREFUSED;
case
WSAELOOP:
return
ELOOP;
case
WSAENAMETOOLONG:
return
ENAMETOOLONG;
case
WSAEHOSTDOWN:
return
WSAEHOSTDOWN;
case
WSAEHOSTUNREACH:
return
EHOSTUNREACH;
case
WSAENOTEMPTY:
return
ENOTEMPTY;
case
WSAEPROCLIM:
return
EPROCLIM;
case
WSAEUSERS:
return
EUSERS;
case
WSAEDQUOT:
return
EDQUOT;
case
WSAESTALE:
return
ESTALE;
case
WSAEREMOTE:
return
EREMOTE;
case
WSAEDISCON:
return
WSAEDISCON;
case
WSAENOMORE:
return
WSAENOMORE;
#ifdef WSAECANCELLED
case
WSAECANCELLED:
return
ECANCELED;
#endif
case
WSAEINVALIDPROCTABLE:
return
WSAEINVALIDPROCTABLE;
case
WSAEINVALIDPROVIDER:
return
WSAEINVALIDPROVIDER;
case
WSAEPROVIDERFAILEDINIT:
return
WSAEPROVIDERFAILEDINIT;
case
WSAEREFUSED:
return
WSAEREFUSED;
}
return
wsaerr;
}
#ifdef ERRNO_HAS_POSIX_SUPPLEMENT
int
convert_errno_to_wsa_error(
int
err)
{
switch
(err) {
case
EADDRINUSE:
return
WSAEADDRINUSE;
case
EADDRNOTAVAIL:
return
WSAEADDRNOTAVAIL;
case
EAFNOSUPPORT:
return
WSAEAFNOSUPPORT;
case
EALREADY:
return
WSAEALREADY;
#ifdef EBADMSG
case
EBADMSG:
return
ERROR_INVALID_FUNCTION;
#endif
case
ECANCELED:
#ifdef WSAECANCELLED
return
WSAECANCELLED;
#else
return
ERROR_INVALID_FUNCTION;
#endif
case
ECONNABORTED:
return
WSAECONNABORTED;
case
ECONNREFUSED:
return
WSAECONNREFUSED;
case
ECONNRESET:
return
WSAECONNRESET;
case
EDESTADDRREQ:
return
WSAEDESTADDRREQ;
case
EHOSTUNREACH:
return
WSAEHOSTUNREACH;
#ifdef EIDRM
case
EIDRM:
return
ERROR_INVALID_FUNCTION;
#endif
case
EINPROGRESS:
return
WSAEINPROGRESS;
case
EISCONN:
return
WSAEISCONN;
case
ELOOP:
return
WSAELOOP;
case
EMSGSIZE:
return
WSAEMSGSIZE;
case
ENETDOWN:
return
WSAENETDOWN;
case
ENETRESET:
return
WSAENETRESET;
case
ENETUNREACH:
return
WSAENETUNREACH;
case
ENOBUFS:
return
WSAENOBUFS;
#ifdef ENODATA
case
ENODATA:
return
ERROR_INVALID_FUNCTION;
#endif
#ifdef ENOLINK
case
ENOLINK:
return
ERROR_INVALID_FUNCTION;
#endif
#ifdef ENOMSG
case
ENOMSG:
return
ERROR_INVALID_FUNCTION;
#endif
case
ENOPROTOOPT:
return
WSAENOPROTOOPT;
#ifdef ENOSR
case
ENOSR:
return
ERROR_INVALID_FUNCTION;
#endif
#ifdef ENOSTR
case
ENOSTR:
return
ERROR_INVALID_FUNCTION;
#endif
case
ENOTCONN:
return
WSAENOTCONN;
#ifdef ENOTRECOVERABLE
case
ENOTRECOVERABLE:
return
ERROR_INVALID_FUNCTION;
#endif
case
ENOTSOCK:
return
WSAENOTSOCK;
case
ENOTSUP:
return
ERROR_INVALID_FUNCTION;
case
EOPNOTSUPP:
return
WSAEOPNOTSUPP;
#ifdef EOTHER
case
EOTHER:
return
ERROR_INVALID_FUNCTION;
#endif
case
EOVERFLOW:
return
ERROR_INVALID_FUNCTION;
case
EOWNERDEAD:
return
ERROR_INVALID_FUNCTION;
case
EPROTO:
return
ERROR_INVALID_FUNCTION;
case
EPROTONOSUPPORT:
return
WSAEPROTONOSUPPORT;
case
EPROTOTYPE:
return
WSAEPROTOTYPE;
#ifdef ETIME
case
ETIME:
return
ERROR_INVALID_FUNCTION;
#endif
case
ETIMEDOUT:
return
WSAETIMEDOUT;
#ifdef ETXTBSY
case
ETXTBSY:
return
ERROR_INVALID_FUNCTION;
#endif
case
EWOULDBLOCK:
return
WSAEWOULDBLOCK;
}
return
err;
}
#endif /* ERRNO_HAS_POSIX_SUPPLEMENT */
u_long
win32_htonl(u_long hostlong)
{
return
htonl(hostlong);
}
u_short
win32_htons(u_short hostshort)
{
return
htons(hostshort);
}
u_long
win32_ntohl(u_long netlong)
{
return
ntohl(netlong);
}
u_short
win32_ntohs(u_short netshort)
{
return
ntohs(netshort);
}
SOCKET
win32_accept(SOCKET s,
struct
sockaddr *addr,
int
*addrlen)
{
SOCKET r;
SOCKET_TEST((r = accept(TO_SOCKET(s), addr, addrlen)), INVALID_SOCKET);
return
OPEN_SOCKET(r);
}
int
win32_bind(SOCKET s,
const
struct
sockaddr *addr,
int
addrlen)
{
int
r;
SOCKET_TEST_ERROR(r = bind(TO_SOCKET(s), addr, addrlen));
return
r;
}
int
win32_connect(SOCKET s,
const
struct
sockaddr *addr,
int
addrlen)
{
int
r;
SOCKET_TEST_ERROR(r = connect(TO_SOCKET(s), addr, addrlen));
return
r;
}
int
win32_getpeername(SOCKET s,
struct
sockaddr *addr,
int
*addrlen)
{
int
r;
SOCKET_TEST_ERROR(r = getpeername(TO_SOCKET(s), addr, addrlen));
return
r;
}
int
win32_getsockname(SOCKET s,
struct
sockaddr *addr,
int
*addrlen)
{
int
r;
SOCKET_TEST_ERROR(r = getsockname(TO_SOCKET(s), addr, addrlen));
return
r;
}
int
win32_getsockopt(SOCKET s,
int
level,
int
optname,
char
*optval,
int
*optlen)
{
int
r;
SOCKET_TEST_ERROR(r = getsockopt(TO_SOCKET(s), level, optname, optval, optlen));
return
r;
}
int
win32_ioctlsocket(SOCKET s,
long
cmd, u_long *argp)
{
int
r;
SOCKET_TEST_ERROR(r = ioctlsocket(TO_SOCKET(s), cmd, argp));
return
r;
}
int
win32_listen(SOCKET s,
int
backlog)
{
int
r;
SOCKET_TEST_ERROR(r = listen(TO_SOCKET(s), backlog));
return
r;
}
int
win32_recv(SOCKET s,
char
*buf,
int
len,
int
flags)
{
int
r;
SOCKET_TEST_ERROR(r = recv(TO_SOCKET(s), buf, len, flags));
return
r;
}
int
win32_recvfrom(SOCKET s,
char
*buf,
int
len,
int
flags,
struct
sockaddr *from,
int
*fromlen)
{
int
r;
int
frombufsize = *fromlen;
SOCKET_TEST_ERROR(r = recvfrom(TO_SOCKET(s), buf, len, flags, from, fromlen));
if
(r != SOCKET_ERROR && frombufsize == *fromlen)
(
void
)win32_getpeername(s, from, fromlen);
return
r;
}
int
win32_select(
int
nfds, Perl_fd_set* rd, Perl_fd_set* wr, Perl_fd_set* ex,
const
struct
timeval* timeout)
{
int
r;
int
i, fd, save_errno =
errno
;
FD_SET nrd, nwr, nex;
bool
just_sleep = TRUE;
FD_ZERO(&nrd);
FD_ZERO(&nwr);
FD_ZERO(&nex);
for
(i = 0; i < nfds; i++) {
if
(rd && PERL_FD_ISSET(i,rd)) {
fd = TO_SOCKET(i);
FD_SET((unsigned)fd, &nrd);
just_sleep = FALSE;
}
if
(wr && PERL_FD_ISSET(i,wr)) {
fd = TO_SOCKET(i);
FD_SET((unsigned)fd, &nwr);
just_sleep = FALSE;
}
if
(ex && PERL_FD_ISSET(i,ex)) {
fd = TO_SOCKET(i);
FD_SET((unsigned)fd, &nex);
just_sleep = FALSE;
}
}
if
(just_sleep) {
if
(timeout)
Sleep(timeout->tv_sec * 1000 +
timeout->tv_usec / 1000);
else
Sleep(UINT_MAX);
return
0;
}
errno
= save_errno;
SOCKET_TEST_ERROR(r = select(nfds, &nrd, &nwr, &nex, (PTIMEVAL)timeout));
save_errno =
errno
;
for
(i = 0; i < nfds; i++) {
if
(rd && PERL_FD_ISSET(i,rd)) {
fd = TO_SOCKET(i);
if
(!FD_ISSET(fd, &nrd))
PERL_FD_CLR(i,rd);
}
if
(wr && PERL_FD_ISSET(i,wr)) {
fd = TO_SOCKET(i);
if
(!FD_ISSET(fd, &nwr))
PERL_FD_CLR(i,wr);
}
if
(ex && PERL_FD_ISSET(i,ex)) {
fd = TO_SOCKET(i);
if
(!FD_ISSET(fd, &nex))
PERL_FD_CLR(i,ex);
}
}
errno
= save_errno;
return
r;
}
int
win32_send(SOCKET s,
const
char
*buf,
int
len,
int
flags)
{
int
r;
SOCKET_TEST_ERROR(r = send(TO_SOCKET(s), buf, len, flags));
return
r;
}
int
win32_sendto(SOCKET s,
const
char
*buf,
int
len,
int
flags,
const
struct
sockaddr *to,
int
tolen)
{
int
r;
SOCKET_TEST_ERROR(r = sendto(TO_SOCKET(s), buf, len, flags, to, tolen));
return
r;
}
int
win32_setsockopt(SOCKET s,
int
level,
int
optname,
const
char
*optval,
int
optlen)
{
int
r;
SOCKET_TEST_ERROR(r = setsockopt(TO_SOCKET(s), level, optname, optval, optlen));
return
r;
}
int
win32_shutdown(SOCKET s,
int
how)
{
int
r;
SOCKET_TEST_ERROR(r = shutdown(TO_SOCKET(s), how));
return
r;
}
int
win32_closesocket(SOCKET s)
{
int
r;
SOCKET_TEST_ERROR(r = closesocket(TO_SOCKET(s)));
return
r;
}
void
convert_proto_info_w2a(WSAPROTOCOL_INFOW *in, WSAPROTOCOL_INFOA *out)
{
Copy(in, out, 1, WSAPROTOCOL_INFOA);
wcstombs
(out->szProtocol, in->szProtocol,
sizeof
(out->szProtocol));
}
SOCKET
open_ifs_socket(
int
af,
int
type,
int
protocol)
{
dTHX;
char
*s;
unsigned
long
proto_buffers_len = 0;
int
error_code, found = 0;
SOCKET out = INVALID_SOCKET;
if
((s = PerlEnv_getenv(
"PERL_ALLOW_NON_IFS_LSP"
)) &&
atoi
(s))
return
WSASocket(af, type, protocol, NULL, 0, 0);
if
(WSCEnumProtocols(NULL, NULL, &proto_buffers_len, &error_code) == SOCKET_ERROR
&& error_code == WSAENOBUFS)
{
WSAPROTOCOL_INFOW *proto_buffers;
int
protocols_available = 0;
Newx(proto_buffers, proto_buffers_len /
sizeof
(WSAPROTOCOL_INFOW),
WSAPROTOCOL_INFOW);
if
((protocols_available = WSCEnumProtocols(NULL, proto_buffers,
&proto_buffers_len, &error_code)) != SOCKET_ERROR)
{
int
i;
for
(i = 0; i < protocols_available; i++)
{
WSAPROTOCOL_INFOA proto_info;
if
((af != AF_UNSPEC && af != proto_buffers[i].iAddressFamily)
|| (type != proto_buffers[i].iSocketType)
|| (protocol != 0 && proto_buffers[i].iProtocol != 0 &&
protocol != proto_buffers[i].iProtocol))
continue
;
if
((proto_buffers[i].dwServiceFlags1 & XP1_IFS_HANDLES) == 0)
continue
;
found = 1;
convert_proto_info_w2a(&(proto_buffers[i]), &proto_info);
out = WSASocket(af, type, protocol, &proto_info, 0, 0);
break
;
}
if
(!found)
WSASetLastError(WSAEPROTONOSUPPORT);
}
Safefree(proto_buffers);
}
return
out;
}
SOCKET
win32_socket(
int
af,
int
type,
int
protocol)
{
SOCKET s;
if
((s = open_ifs_socket(af, type, protocol)) == INVALID_SOCKET)
{
int
wsaerr = WSAGetLastError();
errno
= convert_wsa_error_to_errno(wsaerr);
SetLastError(wsaerr);
}
else
s = OPEN_SOCKET(s);
return
s;
}
struct
hostent *
win32_gethostbyaddr(
const
char
*addr,
int
len,
int
type)
{
struct
hostent *r;
SOCKET_TEST(r = gethostbyaddr(addr, len, type), NULL);
return
r;
}
struct
hostent *
win32_gethostbyname(
const
char
*name)
{
struct
hostent *r;
SOCKET_TEST(r = gethostbyname(name), NULL);
return
r;
}
int
win32_gethostname(
char
*name,
int
len)
{
int
r;
SOCKET_TEST_ERROR(r = gethostname(name, len));
return
r;
}
struct
protoent *
win32_getprotobyname(
const
char
*name)
{
struct
protoent *r;
SOCKET_TEST(r = getprotobyname(name), NULL);
return
r;
}
struct
protoent *
win32_getprotobynumber(
int
num)
{
struct
protoent *r;
SOCKET_TEST(r = getprotobynumber(num), NULL);
return
r;
}
struct
servent *
win32_getservbyname(
const
char
*name,
const
char
*proto)
{
dTHXa(NULL);
struct
servent *r;
SOCKET_TEST(r = getservbyname(name, proto), NULL);
if
(r) {
aTHXa(PERL_GET_THX);
r = win32_savecopyservent(&w32_servent, r, proto);
}
return
r;
}
struct
servent *
win32_getservbyport(
int
port,
const
char
*proto)
{
dTHXa(NULL);
struct
servent *r;
SOCKET_TEST(r = getservbyport(port, proto), NULL);
if
(r) {
aTHXa(PERL_GET_THX);
r = win32_savecopyservent(&w32_servent, r, proto);
}
return
r;
}
int
win32_ioctl(
int
i, unsigned
int
u,
char
*data)
{
u_long u_long_arg;
int
retval;
memcpy
(&u_long_arg, data,
sizeof
u_long_arg);
retval = ioctlsocket(TO_SOCKET(i), (
long
)u, &u_long_arg);
memcpy
(data, &u_long_arg,
sizeof
u_long_arg);
if
(retval == SOCKET_ERROR) {
int
wsaerr = WSAGetLastError();
int
err = convert_wsa_error_to_errno(wsaerr);
if
(err == ENOTSOCK) {
Perl_croak_nocontext(
"ioctl implemented only on sockets"
);
}
errno
= err;
SetLastError(wsaerr);
}
return
retval;
}
char
FAR *
win32_inet_ntoa(
struct
in_addr in)
{
return
inet_ntoa(in);
}
unsigned
long
win32_inet_addr(
const
char
FAR *cp)
{
return
inet_addr(cp);
}
void
win32_endhostent()
{
win32_croak_not_implemented(
"endhostent"
);
}
void
win32_endnetent()
{
win32_croak_not_implemented(
"endnetent"
);
}
void
win32_endprotoent()
{
win32_croak_not_implemented(
"endprotoent"
);
}
void
win32_endservent()
{
win32_croak_not_implemented(
"endservent"
);
}
struct
netent *
win32_getnetent(
void
)
{
win32_croak_not_implemented(
"getnetent"
);
return
(
struct
netent *) NULL;
}
struct
netent *
win32_getnetbyname(
char
*name)
{
win32_croak_not_implemented(
"getnetbyname"
);
return
(
struct
netent *)NULL;
}
struct
netent *
win32_getnetbyaddr(
long
net,
int
type)
{
win32_croak_not_implemented(
"getnetbyaddr"
);
return
(
struct
netent *)NULL;
}
struct
protoent *
win32_getprotoent(
void
)
{
win32_croak_not_implemented(
"getprotoent"
);
return
(
struct
protoent *) NULL;
}
struct
servent *
win32_getservent(
void
)
{
win32_croak_not_implemented(
"getservent"
);
return
(
struct
servent *) NULL;
}
void
win32_sethostent(
int
stayopen)
{
win32_croak_not_implemented(
"sethostent"
);
}
void
win32_setnetent(
int
stayopen)
{
win32_croak_not_implemented(
"setnetent"
);
}
void
win32_setprotoent(
int
stayopen)
{
win32_croak_not_implemented(
"setprotoent"
);
}
void
win32_setservent(
int
stayopen)
{
win32_croak_not_implemented(
"setservent"
);
}
static
char
tcp_proto[] =
"tcp"
;
static
struct
servent*
win32_savecopyservent(
struct
servent*d,
struct
servent*s,
const
char
*proto)
{
d->s_name = s->s_name;
d->s_aliases = s->s_aliases;
d->s_port = s->s_port;
if
(s->s_proto &&
strlen
(s->s_proto))
d->s_proto = s->s_proto;
else
if
(proto &&
strlen
(proto))
d->s_proto = (
char
*)proto;
else
d->s_proto = tcp_proto;
return
d;
}