Socket6
view release on metacpan or search on metacpan
if (strEQ(name, "NI_NUMERICHOST"))
#ifdef NI_NUMERICHOST
return NI_NUMERICHOST;
#else
goto not_there;
#endif
if (strEQ(name, "NI_NAMEREQD"))
#ifdef NI_NAMEREQD
return NI_NAMEREQD;
#else
goto not_there;
#endif
if (strEQ(name, "NI_NUMERICSERV"))
#ifdef NI_NUMERICSERV
return NI_NUMERICSERV;
#else
goto not_there;
#endif
if (strEQ(name, "NI_DGRAM"))
#ifdef NI_DGRAM
return NI_DGRAM;
#else
goto not_there;
#endif
if (strEQ(name, "NI_WITHSCOPEID"))
#ifdef NI_WITHSCOPEID
return NI_WITHSCOPEID;
#else
goto not_there;
#endif
break;
case 'P':
if (strEQ(name, "PF_INET6"))
#ifdef PF_INET6
return PF_INET6;
#else
goto not_there;
#endif
break;
}
errno = EINVAL;
return 0;
not_there:
errno = ENOENT;
return 0;
}
MODULE = Socket6 PACKAGE = Socket6
double
constant(name,arg)
char * name
int arg
void
gethostbyname2(host, af)
char * host;
int af;
PPCODE:
{
#ifdef HAVE_GETHOSTBYNAME2
struct hostent *phe;
int count, i;
if ((phe = gethostbyname2(host, af)) != NULL) {
for (count = 0; phe->h_addr_list[count]; ++count);
EXTEND(sp, 4 + count);
PUSHs(sv_2mortal(newSVpv((char *) phe->h_name,
strlen(phe->h_name))));
PUSHs(sv_2mortal(newSVpv((char *) phe->h_aliases,
sizeof(char *))));
PUSHs(sv_2mortal(newSViv((IV) phe->h_addrtype)));
PUSHs(sv_2mortal(newSViv((IV) phe->h_length)));
for (i = 0; i < count; ++i) {
PUSHs(sv_2mortal(newSVpv((char *)phe->h_addr_list[i],
phe->h_length)));
}
}
#else
ST(0) = (SV *) not_here("gethostbyname2");
#endif
}
void
inet_pton(af, host)
int af
char * host
CODE:
{
#ifdef HAVE_INET_PTON
union {
#ifdef INET6_ADDRSTRLEN
struct in6_addr addr6;
#endif
struct in_addr addr4;
} ip_address;
int len;
int ok;
switch (af) {
#ifdef INET6_ADDRSTRLEN
case AF_INET6:
len = sizeof(struct in6_addr);
break;
#endif
case AF_INET:
len = sizeof(struct in_addr);
break;
default:
croak("Bad address family for %s, got %d",
"Socket6::inet_pton", af);
break;
}
ok = inet_pton(af, host, &ip_address);
ST(0) = sv_newmortal();
if (ok == 1) {
sv_setpvn( ST(0), (char *)&ip_address, len );
}
#else
ST(0) = (SV *) not_here("inet_ntop");
#endif
}
void
pack_sockaddr_in6(port,ip6_address)
unsigned short port
char * ip6_address
CODE:
{
#ifdef INET6_ADDRSTRLEN
struct sockaddr_in6 sin;
Zero( &sin, sizeof sin, char );
#ifdef SIN6_LEN
sin.sin6_len = sizeof sin;
#endif
sin.sin6_family = AF_INET6;
sin.sin6_port = htons(port);
Copy( ip6_address, &sin.sin6_addr, sizeof sin.sin6_addr, char );
ST(0) = sv_2mortal(newSVpv((char *)&sin, sizeof sin));
#else
ST(0) = (SV *) not_here("pack_sockaddr_in6");
#endif
}
void
pack_sockaddr_in6_all(port,flowinfo,ip6_address,scope_id)
unsigned short port
unsigned long flowinfo
char * ip6_address
unsigned long scope_id
CODE:
{
#ifdef INET6_ADDRSTRLEN
struct sockaddr_in6 sin;
Zero( &sin, sizeof sin, char );
#ifdef SIN6_LEN
sin.sin6_len = sizeof sin;
#endif
sin.sin6_family = AF_INET6;
sin.sin6_port = htons(port);
sin.sin6_flowinfo = htonl(flowinfo);
Copy( ip6_address, &sin.sin6_addr, sizeof sin.sin6_addr, char );
#ifdef HAVE_SOCKADDR_IN6_SIN6_SCOPE_ID
sin.sin6_scope_id = scope_id;
#endif
ST(0) = sv_2mortal(newSVpv((char *)&sin, sizeof sin));
#else
ST(0) = (SV *) not_here("pack_sockaddr_in6_all");
#endif
}
void
unpack_sockaddr_in6(sin_sv)
SV * sin_sv
PPCODE:
{
#ifdef INET6_ADDRSTRLEN
STRLEN sockaddrlen;
struct sockaddr_in6 addr;
unsigned short port;
struct in6_addr ip6_address;
char * sin = SvPV(sin_sv,sockaddrlen);
if (sockaddrlen != sizeof(addr)) {
croak("Bad arg length for %s, length is %d, should be %d",
"Socket6::unpack_sockaddr_in6",
sockaddrlen, sizeof(addr));
}
Copy( sin, &addr,sizeof addr, char );
if ( addr.sin6_family != AF_INET6 ) {
croak("Bad address family for %s, got %d, should be %d",
"Socket6::unpack_sockaddr_in6",
addr.sin6_family,
AF_INET6);
}
port = ntohs(addr.sin6_port);
ip6_address = addr.sin6_addr;
EXTEND(sp, 2);
PUSHs(sv_2mortal(newSViv((IV) port)));
PUSHs(sv_2mortal(newSVpv((char *)&ip6_address,sizeof ip6_address)));
#else
ST(0) = (SV *) not_here("unpack_sockaddr_in6");
#endif
}
void
unpack_sockaddr_in6_all(sin_sv)
SV * sin_sv
PPCODE:
{
#ifdef INET6_ADDRSTRLEN
STRLEN sockaddrlen;
struct sockaddr_in6 addr;
unsigned short port;
unsigned long flowinfo;
struct in6_addr ip6_address;
unsigned long scope_id;
char * sin = SvPV(sin_sv,sockaddrlen);
if (sockaddrlen != sizeof(addr)) {
croak("Bad arg length for %s, length is %d, should be %d",
"Socket6::unpack_sockaddr_in6",
sockaddrlen, sizeof(addr));
}
Copy( sin, &addr,sizeof addr, char );
if ( addr.sin6_family != AF_INET6 ) {
croak("Bad address family for %s, got %d, should be %d",
"Socket6::unpack_sockaddr_in6",
addr.sin6_family,
AF_INET6);
}
port = ntohs(addr.sin6_port);
flowinfo = ntohl(addr.sin6_flowinfo);
ip6_address = addr.sin6_addr;
#ifdef HAVE_SOCKADDR_IN6_SIN6_SCOPE_ID
scope_id = addr.sin6_scope_id;
#else
scope_id = 0;
#endif
EXTEND(sp, 5);
PUSHs(sv_2mortal(newSViv((IV) port)));
PUSHs(sv_2mortal(newSViv((IV) flowinfo)));
PUSHs(sv_2mortal(newSVpv((char *)&ip6_address,sizeof ip6_address)));
PUSHs(sv_2mortal(newSViv((IV) scope_id)));
#else
ST(0) = (SV *) not_here("unpack_sockaddr_in6_all");
#endif
}
void
in6addr_any()
CODE:
{
#ifdef INET6_ADDRSTRLEN
ST(0) = sv_2mortal(newSVpv((char *)&in6addr_any, sizeof in6addr_any));
#else
ST(0) = (SV *) not_here("in6addr_any");
#endif
}
void
in6addr_loopback()
CODE:
{
#ifdef INET6_ADDRSTRLEN
ST(0) = sv_2mortal(newSVpv((char *)&in6addr_loopback,
sizeof in6addr_loopback));
#else
ST(0) = (SV *) not_here("in6addr_loopback");
#endif
}
void
getaddrinfo(host,port,family=0,socktype=0,protocol=0,flags=0)
char * host
char * port
int family
int socktype
int protocol
int flags
PPCODE:
{
#ifdef HAVE_GETADDRINFO
struct addrinfo hints, * res;
int err;
int count;
const char *error;
WSA_DECLARE;
Zero( &hints, sizeof hints, char );
hints.ai_flags = flags;
hints.ai_family = family;
hints.ai_socktype = socktype;
hints.ai_protocol = protocol;
WSA_STARTUP();
err = getaddrinfo(*host ? host : 0, *port ? port : 0, &hints, &res);
WSA_CLEANUP();
if (err == 0) {
struct addrinfo * p;
count = 0;
for (p = res; p; p = p->ai_next)
++count;
EXTEND(sp, 5 * count);
for (p = res; p; p = p->ai_next) {
PUSHs(sv_2mortal(newSViv((IV) p->ai_family)));
PUSHs(sv_2mortal(newSViv((IV) p->ai_socktype)));
PUSHs(sv_2mortal(newSViv((IV) p->ai_protocol)));
PUSHs(sv_2mortal(newSVpv((char *)p->ai_addr,
p->ai_addrlen)));
if (p->ai_canonname)
PUSHs(sv_2mortal(newSVpv((char *)p->ai_canonname,
strlen(p->ai_canonname))));
else
PUSHs(&PL_sv_undef);
}
freeaddrinfo(res);
} else {
SV *error_sv = sv_newmortal();
SvUPGRADE(error_sv, SVt_PVNV);
error = gai_strerror(err);
sv_setpv(error_sv, error);
SvIV_set(error_sv, err); SvIOK_on(error_sv);
PUSHs(error_sv);
}
#else
ST(0) = (SV *) not_here("getaddrinfo");
#endif
}
void
getnameinfo(sin_sv, flags = 0)
SV * sin_sv
int flags;
PPCODE:
{
#ifdef HAVE_GETNAMEINFO
STRLEN sockaddrlen;
struct sockaddr * sin = (struct sockaddr *)SvPV(sin_sv,sockaddrlen);
char host[NI_MAXHOST];
char port[NI_MAXSERV];
int err;
const char *error;
WSA_DECLARE;
WSA_STARTUP();
if (items < 2) {
err = getnameinfo(sin, sockaddrlen, host, sizeof host,
port, sizeof port, 0);
if (err)
err = getnameinfo(sin, sockaddrlen, host, sizeof host,
port, sizeof port, NI_NUMERICSERV);
if (err)
err = getnameinfo(sin, sockaddrlen, host, sizeof host,
port, sizeof port, NI_NUMERICHOST);
if (err)
err = getnameinfo(sin, sockaddrlen, host, sizeof host,
port, sizeof port,
NI_NUMERICHOST|NI_NUMERICSERV);
} else {
err = getnameinfo(sin, sockaddrlen, host, sizeof host,
port, sizeof port, flags);
}
WSA_CLEANUP();
if (err == 0) {
EXTEND(sp, 2);
PUSHs(sv_2mortal(newSVpv(host, strlen(host))));
PUSHs(sv_2mortal(newSVpv(port, strlen(port))));
} else {
SV *error_sv = sv_newmortal();
SvUPGRADE(error_sv, SVt_PVNV);
error = gai_strerror(err);
sv_setpv(error_sv, error);
SvIV_set(error_sv, err); SvIOK_on(error_sv);
PUSHs(error_sv);
}
#else
ST(0) = (SV *) not_here("getnameinfo");
#endif
}
char *
gai_strerror(errcode = 0)
int errcode;
CODE:
RETVAL = (char *)gai_strerror(errcode);
OUTPUT:
RETVAL
void
getipnodebyname(hostname, family=0, flags=0)
char * hostname
int family
int flags
PREINIT:
#ifdef HAVE_GETIPNODEBYNAME
struct hostent *he;
int err;
char **p;
SV *temp, *address_ref, *alias_ref;
AV *address_list, *alias_list;
#endif
PPCODE:
{
#ifdef HAVE_GETIPNODEBYNAME
he = getipnodebyname(hostname, family, flags, &err);
if (err == 0) {
XPUSHs(sv_2mortal(newSVpv(he->h_name, strlen(he->h_name))));
XPUSHs(sv_2mortal(newSViv(he->h_addrtype)));
XPUSHs(sv_2mortal(newSViv(he->h_length)));
address_list = newAV();
for(p = he->h_addr_list; *p != NULL; p++) {
temp = newSVpv(*p, he->h_length);
av_push(address_list, temp);
}
address_ref = newRV_noinc((SV*) address_list);
XPUSHs(address_ref);
alias_list = newAV();
for(p = he->h_aliases; *p != NULL; p++) {
temp = newSVpv(*p, strlen(*p));
av_push(alias_list, temp);
}
alias_ref = newRV_noinc((SV*) alias_list);
XPUSHs(alias_ref);
freehostent(he);
} else {
XPUSHs(sv_2mortal(newSViv(err)));
}
#else
ST(0) = (SV *) not_here("getipnodebyname");
#endif
}
void
getipnodebyaddr(family, address_sv)
int family
SV * address_sv
PREINIT:
#ifdef HAVE_GETIPNODEBYADDR
STRLEN addrlen;
struct hostent *he;
int err, alen;
char **p;
SV *temp, *address_ref, *alias_ref;
AV *address_list, *alias_list;
struct in6_addr addr;
char *addr_buffer;
#endif
PPCODE:
{
#ifdef HAVE_GETIPNODEBYADDR
addr_buffer = SvPV(address_sv, addrlen);
switch(family) {
case AF_INET:
alen = sizeof(struct in_addr);
break;
case AF_INET6:
alen = sizeof(struct in6_addr);
break;
default:
croak("Unsupported address family for %s, af is %d",
"Socket6::getipnodebyaddr", family);
}
if (alen > sizeof(addr) || alen != addrlen) {
croak("Arg length mismatch in %s, length is %d, should be %d\n",
"Socket6::getipnodebyaddr", addrlen, alen);
}
Copy(addr_buffer, &addr, sizeof(addr), char);
he = getipnodebyaddr(addr_buffer, alen, family, &err);
if (err == 0) {
XPUSHs(sv_2mortal(newSVpv(he->h_name, strlen(he->h_name))));
XPUSHs(sv_2mortal(newSViv(he->h_addrtype)));
XPUSHs(sv_2mortal(newSViv(he->h_length)));
address_list = newAV();
for(p = he->h_addr_list; *p != NULL; p++) {
temp = newSVpv(*p, he->h_length);
av_push(address_list, temp);
}
address_ref = newRV_noinc((SV*) address_list);
XPUSHs(address_ref);
alias_list = newAV();
for(p = he->h_aliases; *p != NULL; p++) {
temp = newSVpv(*p, strlen(*p));
av_push(alias_list, temp);
}
alias_ref = newRV_noinc((SV*) alias_list);
XPUSHs(alias_ref);
freehostent(he);
} else {
XPUSHs(sv_2mortal(newSViv(err)));
}
#else
ST(0) = (SV *) not_here("getipnodebyaddr");
#endif
}
( run in 2.130 seconds using v1.01-cache-2.11-cpan-5511b514fd6 )