This source file includes following definitions.
- gf_net_mobileip_set_callback
- gf_net_mobileip_ctrl
- gf_net_has_ipv6
- gf_net_is_ipv6
- gf_sk_get_ipv6_addr
- gf_sk_ipv6_set_remote_address
- gf_sk_get_host_name
- gf_sk_get_local_ip
- gf_sk_new
- gf_sk_set_buffer_size
- gf_sk_set_block_mode
- gf_sk_free
- gf_sk_del
- gf_sk_reset
- gf_sk_get_handle
- gf_sk_connect
- gf_sk_bind
- gf_sk_send
- gf_sk_is_multicast_address
- gf_sk_setup_multicast
- gf_sk_receive
- gf_sk_listen
- gf_sk_accept
- gf_sk_get_local_info
- gf_sk_server_mode
- gf_sk_get_remote_address
- gf_sk_send_to
- gf_sk_receive_wait
- gf_sk_send_wait
- gf_htonl
- gf_ntohl
- gf_htons
- gf_tohs
#ifndef GPAC_DISABLE_CORE_TOOLS
#if defined(WIN32) || defined(_WIN32_WCE)
#define _WINSOCK_DEPRECATED_NO_WARNINGS
#ifdef _WIN32_WCE
#include <winsock.h>
#if !defined(__GNUC__)
#pragma comment(lib, "winsock")
#endif
#else
#include <sys/timeb.h>
#include <winsock2.h>
#include <ws2tcpip.h>
#if !defined(__GNUC__)
#pragma comment(lib, "ws2_32")
#endif
#endif
#include <windows.h>
#if !defined(__GNUC__)
#if defined(IPV6_MULTICAST_IF)
#define GPAC_HAS_IPV6 1
#pragma message("Using WinSock IPV6")
#else
#undef GPAC_HAS_IPV6
#pragma message("Using WinSock IPV4")
#endif
#endif
#ifndef EISCONN
#undef EAGAIN
#define EAGAIN WSAEWOULDBLOCK
#define EISCONN WSAEISCONN
#define ENOTCONN WSAENOTCONN
#define ECONNRESET WSAECONNRESET
#define EMSGSIZE WSAEMSGSIZE
#define ECONNABORTED WSAECONNABORTED
#define ENETDOWN WSAENETDOWN
#undef EINTR
#define EINTR WSAEINTR
#undef EBADF
#define EBADF WSAEBADF
#endif
#define LASTSOCKERROR WSAGetLastError()
static int wsa_init = 0;
#include <gpac/network.h>
#else
#include <unistd.h>
#include <fcntl.h>
#include <netdb.h>
#ifndef __BEOS__
#include <errno.h>
#endif
#ifndef __DARWIN__
#include <sys/time.h>
#endif
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <gpac/network.h>
#if !defined(INADDR_NONE)
# if (defined(sun) && defined(__SVR4))
# define INADDR_NONE -1
# else
# define INADDR_NONE ((unsigned long)-1)
# endif
#endif
#define INVALID_SOCKET -1
#define SOCKET_ERROR -1
#define LASTSOCKERROR errno
typedef s32 SOCKET;
#define closesocket(v) close(v)
#endif
#ifdef GPAC_HAS_IPV6
# ifndef IPV6_ADD_MEMBERSHIP
# define IPV6_ADD_MEMBERSHIP IPV6_JOIN_GROUP
# endif
# ifndef IPV6_DROP_MEMBERSHIP
# define IPV6_DROP_MEMBERSHIP IPV6_LEAVE_GROUP
# endif
#endif
#ifdef __SYMBIAN32__
#define SSO_CAST
#else
#define SSO_CAST (const char *)
#endif
#define SOCK_MICROSEC_WAIT 500
#ifdef GPAC_HAS_IPV6
static u32 ipv6_check_state = 0;
#endif
#ifdef _LP64
#define NULL_SOCKET 0
#else
#define NULL_SOCKET (SOCKET)NULL
#endif
enum
{
GF_SOCK_IS_TCP = 1<<9,
GF_SOCK_IS_IPV6 = 1<<10,
GF_SOCK_NON_BLOCKING = 1<<11,
GF_SOCK_IS_MULTICAST = 1<<12,
GF_SOCK_IS_LISTENING = 1<<13,
GF_SOCK_HAS_PEER = 1<<14,
GF_SOCK_IS_MIP = 1<<15
};
struct __tag_socket
{
u32 flags;
SOCKET socket;
#ifdef GPAC_HAS_IPV6
struct sockaddr_storage dest_addr;
#else
struct sockaddr_in dest_addr;
#endif
u32 dest_addr_len;
};
static gf_net_mobileip_ctrl_cbk mobip_cbk = NULL;
static const char *MobileIPAdd = NULL;
GF_EXPORT
void gf_net_mobileip_set_callback(gf_net_mobileip_ctrl_cbk _mobip_cbk, const char *mip)
{
mobip_cbk = _mobip_cbk;
MobileIPAdd = _mobip_cbk ? mip : NULL;
}
static GF_Err gf_net_mobileip_ctrl(Bool start)
{
if (mobip_cbk) return mobip_cbk(start);
return GF_NOT_SUPPORTED;
}
u32 gf_net_has_ipv6()
{
#ifdef GPAC_HAS_IPV6
if (!ipv6_check_state) {
SOCKET s;
#ifdef WIN32
if (!wsa_init) {
WSADATA Data;
if (WSAStartup(0x0202, &Data)!=0) {
ipv6_check_state = 1;
return 0;
}
}
#endif
s = socket(PF_INET6, SOCK_STREAM, 0);
if (!s) ipv6_check_state = 1;
else {
ipv6_check_state = 2;
closesocket(s);
}
#ifdef WIN32
if (!wsa_init) WSACleanup();
#endif
}
return (ipv6_check_state==2);
#else
return 0;
#endif
}
GF_EXPORT
Bool gf_net_is_ipv6(const char *address)
{
char *sep;
if (!address) return GF_FALSE;
sep = strchr(address, ':');
if (sep) sep = strchr(address, ':');
return sep ? GF_TRUE : GF_FALSE;
}
#ifdef GPAC_HAS_IPV6
#define MAX_PEER_NAME_LEN 1024
static struct addrinfo *gf_sk_get_ipv6_addr(const char *PeerName, u16 PortNumber, int family, int flags, int sock_type)
{
struct addrinfo *res=NULL;
struct addrinfo hints;
char node[MAX_PEER_NAME_LEN], portstring[20];
char *service, *dest;
service = dest = NULL;
#ifdef WIN32
if (!wsa_init) {
WSADATA Data;
if (WSAStartup(0x0202, &Data)!=0) return NULL;
wsa_init = 1;
}
#endif
service = dest = NULL;
memset(&hints, 0, sizeof(hints));
hints.ai_socktype = sock_type;
hints.ai_family = family;
hints.ai_flags = flags;
if (PortNumber) {
sprintf (portstring, "%d", PortNumber);
service = (char *)portstring;
}
if (PeerName) {
strncpy(node, PeerName, MAX_PEER_NAME_LEN);
if (node[0]=='[') {
node[strlen(node)-1] = 0;
strncpy(node, &node[1], MAX_PEER_NAME_LEN);
}
node[MAX_PEER_NAME_LEN - 1] = 0;
dest = (char *) node;
}
if (getaddrinfo((const char *)dest, (const char *)service, &hints, &res) != 0) return NULL;
return res;
}
static Bool gf_sk_ipv6_set_remote_address(GF_Socket *sock, const char *address, u16 PortNumber)
{
struct addrinfo *res = gf_sk_get_ipv6_addr(address, PortNumber, AF_UNSPEC, 0, (sock->flags & GF_SOCK_IS_TCP) ? SOCK_STREAM : SOCK_DGRAM);
if (!res) return GF_FALSE;
memcpy(&sock->dest_addr, res->ai_addr, res->ai_addrlen);
sock->dest_addr_len = (u32) res->ai_addrlen;
freeaddrinfo(res);
return GF_TRUE;
}
#endif
GF_EXPORT
GF_Err gf_sk_get_host_name(char *buffer)
{
s32 ret = gethostname(buffer, GF_MAX_IP_NAME_LEN);
return (ret == SOCKET_ERROR) ? GF_IP_ADDRESS_NOT_FOUND : GF_OK;
}
GF_Err gf_sk_get_local_ip(GF_Socket *sock, char *buffer)
{
#ifdef GPAC_HAS_IPV6
char clienthost[NI_MAXHOST];
if (sock->flags & GF_SOCK_HAS_PEER) {
if (getnameinfo((struct sockaddr *)&sock->dest_addr, sock->dest_addr_len, clienthost, sizeof(clienthost), NULL, 0, NI_NUMERICHOST))
return GF_IP_NETWORK_FAILURE;
} else {
struct sockaddr_storage clientaddr;
socklen_t addrlen = sizeof(clientaddr);
if (getsockname(sock->socket, (struct sockaddr *)&clientaddr, &addrlen)) return GF_IP_NETWORK_FAILURE;
if (getnameinfo((struct sockaddr *)&clientaddr, addrlen, clienthost, sizeof(clienthost), NULL, 0, NI_NUMERICHOST))
return GF_IP_NETWORK_FAILURE;
}
strcpy(buffer, clienthost);
#else
char *ip;
if (sock->flags & GF_SOCK_HAS_PEER) {
ip = inet_ntoa(sock->dest_addr.sin_addr);
} else {
struct sockaddr_in name;
u32 len = sizeof(struct sockaddr_in);
if (getsockname(sock->socket, (struct sockaddr*) &name, &len)) return GF_IP_NETWORK_FAILURE;
ip = inet_ntoa(name.sin_addr);
}
if (!ip) return GF_IP_NETWORK_FAILURE;
strcpy(buffer, ip);
#endif
return GF_OK;
}
GF_EXPORT
GF_Socket *gf_sk_new(u32 SocketType)
{
GF_Socket *tmp;
#ifdef WIN32
WSADATA Data;
if (!wsa_init && (WSAStartup(0x0202, &Data)!=0) ) return NULL;
#endif
if ((SocketType != GF_SOCK_TYPE_UDP) && (SocketType != GF_SOCK_TYPE_TCP)) return NULL;
GF_SAFEALLOC(tmp, GF_Socket);
if (!tmp) return NULL;
if (SocketType == GF_SOCK_TYPE_TCP) tmp->flags |= GF_SOCK_IS_TCP;
#ifdef GPAC_HAS_IPV6
memset(&tmp->dest_addr, 0, sizeof(struct sockaddr_storage));
#else
memset(&tmp->dest_addr, 0, sizeof(struct sockaddr_in));
tmp->dest_addr_len = sizeof(struct sockaddr);
#endif
#ifdef WIN32
wsa_init ++;
#endif
return tmp;
}
GF_EXPORT
GF_Err gf_sk_set_buffer_size(GF_Socket *sock, Bool SendBuffer, u32 NewSize)
{
s32 res;
if (!sock || !sock->socket) return GF_BAD_PARAM;
if (SendBuffer) {
res = setsockopt(sock->socket, SOL_SOCKET, SO_SNDBUF, (char *) &NewSize, sizeof(u32) );
} else {
res = setsockopt(sock->socket, SOL_SOCKET, SO_RCVBUF, (char *) &NewSize, sizeof(u32) );
}
if (res<0) {
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[Socket] Couldn't set socket %s buffer size to %d: %d\n", SendBuffer ? "send" : "receive", NewSize, res));
} else {
GF_LOG(GF_LOG_DEBUG, GF_LOG_NETWORK, ("[Socket] Set socket %s buffer size to %d\n", SendBuffer ? "send" : "receive", NewSize));
}
return GF_OK;
}
GF_EXPORT
GF_Err gf_sk_set_block_mode(GF_Socket *sock, Bool NonBlockingOn)
{
s32 res;
#ifdef WIN32
long val = NonBlockingOn;
if (sock->socket) {
res = ioctlsocket(sock->socket, FIONBIO, &val);
if (res) return GF_SERVICE_ERROR;
}
#else
s32 flag = fcntl(sock->socket, F_GETFL, 0);
if (sock->socket) {
res = fcntl(sock->socket, F_SETFL, flag | O_NONBLOCK);
if (res) return GF_SERVICE_ERROR;
}
#endif
if (NonBlockingOn) {
sock->flags |= GF_SOCK_NON_BLOCKING;
} else {
sock->flags &= ~GF_SOCK_NON_BLOCKING;
}
return GF_OK;
}
#include <assert.h>
static void gf_sk_free(GF_Socket *sock)
{
assert( sock );
if (sock->socket && (sock->flags & GF_SOCK_IS_MULTICAST) ) {
struct ip_mreq mreq;
#ifdef GPAC_HAS_IPV6
struct sockaddr *addr = (struct sockaddr *)&sock->dest_addr;
if (addr->sa_family==AF_INET6) {
struct ipv6_mreq mreq6;
memcpy(&mreq6.ipv6mr_multiaddr, &(((struct sockaddr_in6 *)addr)->sin6_addr), sizeof(struct in6_addr));
mreq6.ipv6mr_interface= 0;
setsockopt(sock->socket, IPPROTO_IPV6, IPV6_ADD_MEMBERSHIP, (char *) &mreq6, sizeof(mreq6));
} else {
mreq.imr_multiaddr.s_addr = ((struct sockaddr_in *)addr)->sin_addr.s_addr;
mreq.imr_interface.s_addr = INADDR_ANY;
setsockopt(sock->socket, IPPROTO_IP, IP_DROP_MEMBERSHIP, (char *) &mreq, sizeof(mreq));
}
#else
mreq.imr_multiaddr.s_addr = sock->dest_addr.sin_addr.s_addr;
mreq.imr_interface.s_addr = INADDR_ANY;
setsockopt(sock->socket, IPPROTO_IP, IP_DROP_MEMBERSHIP, (char *) &mreq, sizeof(mreq));
#endif
}
if (sock->socket) closesocket(sock->socket);
sock->socket = (SOCKET) 0L;
if (sock->flags & GF_SOCK_IS_MIP) {
sock->flags &= ~GF_SOCK_IS_MIP;
gf_net_mobileip_ctrl(GF_FALSE);
}
}
GF_EXPORT
void gf_sk_del(GF_Socket *sock)
{
assert( sock );
gf_sk_free(sock);
#ifdef WIN32
wsa_init --;
if (!wsa_init) WSACleanup();
#endif
gf_free(sock);
}
void gf_sk_reset(GF_Socket *sock)
{
u32 clear;
if (sock) setsockopt(sock->socket, SOL_SOCKET, SO_ERROR, (char *) &clear, sizeof(u32) );
}
s32 gf_sk_get_handle(GF_Socket *sock)
{
return (s32) sock->socket;
}
GF_Err gf_sk_connect(GF_Socket *sock, const char *PeerName, u16 PortNumber, const char *local_ip)
{
s32 ret;
#ifdef GPAC_HAS_IPV6
u32 type = (sock->flags & GF_SOCK_IS_TCP) ? SOCK_STREAM : SOCK_DGRAM;
struct addrinfo *res, *aip, *lip;
gf_sk_free(sock);
GF_LOG(GF_LOG_INFO, GF_LOG_NETWORK, ("[Sock_IPV6] Solving %s address\n", PeerName));
res = gf_sk_get_ipv6_addr(PeerName, PortNumber, AF_UNSPEC, AI_PASSIVE, type);
if (!res) return GF_IP_CONNECTION_FAILURE;
GF_LOG(GF_LOG_INFO, GF_LOG_NETWORK, ("[Sock_IPV6] Host %s found\n", PeerName));
if (local_ip && MobileIPAdd && !strcmp(MobileIPAdd, local_ip) ) {
if (gf_net_mobileip_ctrl(GF_TRUE)==GF_OK) {
sock->flags |= GF_SOCK_IS_MIP;
} else {
local_ip = NULL;
}
}
lip = NULL;
if (local_ip) {
lip = gf_sk_get_ipv6_addr(local_ip, PortNumber, AF_UNSPEC, AI_PASSIVE, type);
if (!lip && local_ip) {
lip = gf_sk_get_ipv6_addr(NULL, PortNumber, AF_UNSPEC, AI_PASSIVE, type);
local_ip = NULL;
}
}
for (aip=res; aip!=NULL; aip=aip->ai_next) {
if (type != (u32) aip->ai_socktype) continue;
sock->socket = socket(aip->ai_family, aip->ai_socktype, aip->ai_protocol);
if (sock->socket == INVALID_SOCKET) {
sock->socket = NULL_SOCKET;
continue;
}
if (sock->flags & GF_SOCK_NON_BLOCKING) gf_sk_set_block_mode(sock, GF_TRUE);
if (aip->ai_family==PF_INET6) sock->flags |= GF_SOCK_IS_IPV6;
else sock->flags &= ~GF_SOCK_IS_IPV6;
if (lip) {
ret = bind(sock->socket, lip->ai_addr, (int) lip->ai_addrlen);
if (ret == SOCKET_ERROR) {
closesocket(sock->socket);
sock->socket = NULL_SOCKET;
continue;
}
}
GF_LOG(GF_LOG_INFO, GF_LOG_NETWORK, ("[Sock_IPV6] Connecting to %s:%d\n", PeerName, PortNumber));
ret = connect(sock->socket, aip->ai_addr, (int) aip->ai_addrlen);
if (ret == SOCKET_ERROR) {
closesocket(sock->socket);
sock->socket = NULL_SOCKET;
continue;
}
GF_LOG(GF_LOG_INFO, GF_LOG_NETWORK, ("[Sock_IPV6] Connected to %s:%d\n", PeerName, PortNumber));
memcpy(&sock->dest_addr, aip->ai_addr, aip->ai_addrlen);
sock->dest_addr_len = (u32) aip->ai_addrlen;
freeaddrinfo(res);
if (lip) freeaddrinfo(lip);
return GF_OK;
}
freeaddrinfo(res);
if (lip) freeaddrinfo(lip);
return GF_IP_CONNECTION_FAILURE;
#else
struct hostent *Host;
if (local_ip) {
GF_Err e = gf_sk_bind(sock, local_ip, PortNumber, PeerName, PortNumber, GF_SOCK_REUSE_PORT);
if (e) return e;
}
if (!sock->socket) {
sock->socket = socket(AF_INET, (sock->flags & GF_SOCK_IS_TCP) ? SOCK_STREAM : SOCK_DGRAM, 0);
if (sock->flags & GF_SOCK_NON_BLOCKING)
gf_sk_set_block_mode(sock, 1);
}
sock->dest_addr.sin_family = AF_INET;
sock->dest_addr.sin_port = htons(PortNumber);
sock->dest_addr.sin_addr.s_addr = inet_addr(PeerName);
if (sock->dest_addr.sin_addr.s_addr==INADDR_NONE) {
GF_LOG(GF_LOG_INFO, GF_LOG_NETWORK, ("[Sock_IPV4] Solving %s address\n", PeerName));
Host = gethostbyname(PeerName);
if (Host == NULL) {
switch (LASTSOCKERROR) {
#ifndef __SYMBIAN32__
case ENETDOWN:
return GF_IP_NETWORK_FAILURE;
#endif
default:
return GF_IP_NETWORK_FAILURE;
}
}
GF_LOG(GF_LOG_INFO, GF_LOG_NETWORK, ("[Sock_IPV4] Host %s found\n", PeerName));
memcpy((char *) &sock->dest_addr.sin_addr, Host->h_addr_list[0], sizeof(u32));
}
if (sock->flags & GF_SOCK_IS_TCP) {
GF_LOG(GF_LOG_INFO, GF_LOG_NETWORK, ("[Sock_IPV4] Connecting to %s:%d\n", PeerName, PortNumber));
ret = connect(sock->socket, (struct sockaddr *) &sock->dest_addr, sizeof(struct sockaddr));
if (ret == SOCKET_ERROR) {
u32 res = LASTSOCKERROR;
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[Sock_IPV4] Couldn't connect socket - last sock error %d\n", res));
switch (res) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
#ifdef WIN32
case WSAEINVAL:
if (sock->flags & GF_SOCK_NON_BLOCKING)
return GF_IP_SOCK_WOULD_BLOCK;
#endif
case EISCONN:
return GF_OK;
case ENOTCONN:
return GF_IP_CONNECTION_FAILURE;
case ECONNRESET:
return GF_IP_CONNECTION_FAILURE;
case EMSGSIZE:
return GF_IP_CONNECTION_FAILURE;
case ECONNABORTED:
return GF_IP_CONNECTION_FAILURE;
case ENETDOWN:
return GF_IP_CONNECTION_FAILURE;
default:
return GF_IP_CONNECTION_FAILURE;
}
}
GF_LOG(GF_LOG_INFO, GF_LOG_NETWORK, ("[Sock_IPV4] Connected to %s:%d\n", PeerName, PortNumber));
}
#endif
return GF_OK;
}
GF_EXPORT
GF_Err gf_sk_bind(GF_Socket *sock, const char *local_ip, u16 port, const char *peer_name, u16 peer_port, u32 options)
{
#ifdef GPAC_HAS_IPV6
struct addrinfo *res, *aip;
int af;
u32 type;
#else
u32 ip_add;
size_t addrlen;
struct sockaddr_in LocalAdd;
struct hostent *Host;
#if 0
char buf[GF_MAX_IP_NAME_LEN];
#endif
#endif
s32 ret;
s32 optval;
if (!sock || sock->socket) return GF_BAD_PARAM;
#ifndef WIN32
if(!local_ip) {
if(!peer_name || !strcmp(peer_name,"localhost")) {
peer_name="127.0.0.1";
}
}
#endif
#ifdef GPAC_HAS_IPV6
type = (sock->flags & GF_SOCK_IS_TCP) ? SOCK_STREAM : SOCK_DGRAM;
af = (options & GF_SOCK_FORCE_IPV6) ? PF_INET6 : PF_UNSPEC;
if (!gf_net_has_ipv6()) af = PF_INET;
if (peer_name && peer_port) {
res = gf_sk_get_ipv6_addr(peer_name, peer_port, af, AI_PASSIVE, type);
if (!res) {
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[Socket] Cannot get IPV6 host name for %s:%d\n", peer_name, peer_port));
return GF_IP_ADDRESS_NOT_FOUND;
}
#ifdef WIN32
af = res->ai_family;
#endif
memcpy(&sock->dest_addr, res->ai_addr, res->ai_addrlen);
sock->dest_addr_len = (u32) res->ai_addrlen;
freeaddrinfo(res);
}
if (local_ip && MobileIPAdd && !strcmp(MobileIPAdd, local_ip) ) {
if (gf_net_mobileip_ctrl(GF_TRUE)==GF_OK) {
sock->flags |= GF_SOCK_IS_MIP;
} else {
gf_sk_get_ipv6_addr(NULL, port, af, AI_PASSIVE, type);
local_ip = NULL;
}
}
res = gf_sk_get_ipv6_addr(local_ip, port, af, AI_PASSIVE, type);
if (!res) {
if (local_ip) {
res = gf_sk_get_ipv6_addr(NULL, port, af, AI_PASSIVE, type);
local_ip = NULL;
}
if (!res) {
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[Socket] Cannot get IPV6 host name for %s:%d\n", local_ip, port));
return GF_IP_ADDRESS_NOT_FOUND;
}
}
for (aip=res; aip!=NULL; aip=aip->ai_next) {
if (type != (u32) aip->ai_socktype) continue;
if (aip->ai_next && (aip->ai_next->ai_family==PF_INET) && !gf_net_is_ipv6(peer_name)) continue;
sock->socket = socket(aip->ai_family, aip->ai_socktype, aip->ai_protocol);
if (sock->socket == INVALID_SOCKET) {
sock->socket = NULL_SOCKET;
continue;
}
if (options & GF_SOCK_REUSE_PORT) {
optval = 1;
setsockopt(sock->socket, SOL_SOCKET, SO_REUSEADDR, (const char *) &optval, sizeof(optval));
#ifdef SO_REUSEPORT
optval = 1;
setsockopt(sock->socket, SOL_SOCKET, SO_REUSEPORT, SSO_CAST &optval, sizeof(optval));
#endif
}
if (sock->flags & GF_SOCK_NON_BLOCKING) gf_sk_set_block_mode(sock, GF_TRUE);
if (peer_name && peer_port)
sock->flags |= GF_SOCK_HAS_PEER;
ret = bind(sock->socket, aip->ai_addr, (int) aip->ai_addrlen);
if (ret == SOCKET_ERROR) {
closesocket(sock->socket);
sock->socket = NULL_SOCKET;
continue;
}
if (aip->ai_family==PF_INET6) sock->flags |= GF_SOCK_IS_IPV6;
else sock->flags &= ~GF_SOCK_IS_IPV6;
freeaddrinfo(res);
return GF_OK;
}
freeaddrinfo(res);
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[Socket] Cannot bind to host %s port %d\n", local_ip, port));
return GF_IP_CONNECTION_FAILURE;
#else
sock->socket = socket(AF_INET, (sock->flags & GF_SOCK_IS_TCP) ? SOCK_STREAM : SOCK_DGRAM, 0);
if (sock->flags & GF_SOCK_NON_BLOCKING) gf_sk_set_block_mode(sock, 1);
sock->flags &= ~GF_SOCK_IS_IPV6;
memset((void *) &LocalAdd, 0, sizeof(LocalAdd));
if (local_ip && MobileIPAdd && !strcmp(MobileIPAdd, local_ip) ) {
if (gf_net_mobileip_ctrl(1)==GF_OK) {
sock->flags |= GF_SOCK_IS_MIP;
} else {
local_ip = NULL;
}
}
ip_add = 0;
if (local_ip) ip_add = inet_addr(local_ip);
if (!ip_add) {
#if 0
buf[0] = 0;
ret = gethostname(buf, GF_MAX_IP_NAME_LEN);
Host = gethostbyname(buf);
if (Host != NULL) {
memcpy((char *) &LocalAdd.sin_addr, Host->h_addr_list[0], sizeof(LocalAdd.sin_addr));
ip_add = LocalAdd.sin_addr.s_addr;
} else {
ip_add = INADDR_ANY;
}
#else
ip_add = INADDR_ANY;
#endif
}
if (peer_name && peer_port) {
#ifdef WIN32
if ((inet_addr(peer_name)== ip_add)) {
optval = 1;
setsockopt(sock->socket, SOL_SOCKET, SO_USELOOPBACK, SSO_CAST &optval, sizeof(optval));
}
#endif
}
LocalAdd.sin_family = AF_INET;
LocalAdd.sin_port = htons(port);
LocalAdd.sin_addr.s_addr = ip_add;
addrlen = sizeof(struct sockaddr_in);
if (options & GF_SOCK_REUSE_PORT) {
optval = 1;
setsockopt(sock->socket, SOL_SOCKET, SO_REUSEADDR, SSO_CAST &optval, sizeof(optval));
#ifdef SO_REUSEPORT
optval = 1;
setsockopt(sock->socket, SOL_SOCKET, SO_REUSEPORT, SSO_CAST &optval, sizeof(optval));
#endif
}
ret = bind(sock->socket, (struct sockaddr *) &LocalAdd, (int) addrlen);
if (ret == SOCKET_ERROR) {
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[socket] cannot bind socket - socket error %x\n", LASTSOCKERROR));
ret = GF_IP_CONNECTION_FAILURE;
}
if (peer_name && peer_port) {
sock->dest_addr.sin_port = htons(peer_port);
sock->dest_addr.sin_family = AF_INET;
sock->dest_addr.sin_addr.s_addr = inet_addr(peer_name);
if (sock->dest_addr.sin_addr.s_addr == INADDR_NONE) {
Host = gethostbyname(peer_name);
if (Host == NULL) ret = GF_IP_ADDRESS_NOT_FOUND;
else memcpy((char *) &sock->dest_addr.sin_addr, Host->h_addr_list[0], sizeof(u32));
}
sock->flags |= GF_SOCK_HAS_PEER;
}
if (sock->flags & GF_SOCK_HAS_PEER) {
GF_LOG(GF_LOG_INFO, GF_LOG_NETWORK, ("[socket] socket bound to %08X - port %d - remote peer: %s:%d\n", ip_add, port, peer_name, peer_port));
} else {
GF_LOG(GF_LOG_INFO, GF_LOG_NETWORK, ("[socket] socket bound to %08X - port %d\n", ip_add, port));
}
return ret;
#endif
}
GF_EXPORT
GF_Err gf_sk_send(GF_Socket *sock, const char *buffer, u32 length)
{
u32 count;
s32 res;
Bool not_ready = GF_FALSE;
#ifndef __SYMBIAN32__
int ready;
struct timeval timeout;
fd_set Group;
#endif
if (!sock || !sock->socket) return GF_BAD_PARAM;
#ifndef __SYMBIAN32__
FD_ZERO(&Group);
FD_SET(sock->socket, &Group);
timeout.tv_sec = 0;
timeout.tv_usec = SOCK_MICROSEC_WAIT;
ready = select((int) sock->socket+1, NULL, &Group, NULL, &timeout);
if (ready == SOCKET_ERROR) {
switch (LASTSOCKERROR) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
default:
return GF_IP_NETWORK_FAILURE;
}
}
if (!ready || !FD_ISSET(sock->socket, &Group)) {
not_ready = GF_TRUE;
}
#endif
count = 0;
while (count < length) {
if (sock->flags & GF_SOCK_HAS_PEER) {
res = (s32) sendto(sock->socket, (char *) buffer+count, length - count, 0, (struct sockaddr *) &sock->dest_addr, sock->dest_addr_len);
} else {
res = (s32) send(sock->socket, (char *) buffer+count, length - count, 0);
}
if (res == SOCKET_ERROR) {
if (not_ready)
return GF_IP_NETWORK_EMPTY;
switch (res = LASTSOCKERROR) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
#ifndef __SYMBIAN32__
case ENOTCONN:
case ECONNRESET:
return GF_IP_CONNECTION_CLOSED;
#endif
default:
return GF_IP_NETWORK_FAILURE;
}
}
count += res;
}
return GF_OK;
}
GF_EXPORT
u32 gf_sk_is_multicast_address(const char *multi_IPAdd)
{
#ifdef GPAC_HAS_IPV6
u32 val;
char *sep;
struct addrinfo *res;
if (!multi_IPAdd) return 0;
sep = strchr(multi_IPAdd, ':');
if (sep) sep = strchr(multi_IPAdd, ':');
if (sep && !strnicmp(multi_IPAdd, "ff", 2)) return 1;
res = gf_sk_get_ipv6_addr((char*)multi_IPAdd, 7000, AF_UNSPEC, AI_PASSIVE, SOCK_DGRAM);
if (!res) return 0;
val = 0;
if (res->ai_addr->sa_family == AF_INET) {
val = IN_MULTICAST(ntohl(((struct sockaddr_in *)res->ai_addr)->sin_addr.s_addr));
} else if (res->ai_addr->sa_family == AF_INET6) {
val = IN6_IS_ADDR_MULTICAST(& ((struct sockaddr_in6 *)res->ai_addr)->sin6_addr);
}
freeaddrinfo(res);
return val;
#else
if (!multi_IPAdd) return 0;
return ((htonl(inet_addr(multi_IPAdd)) >> 8) & 0x00f00000) == 0x00e00000;
#endif
}
GF_EXPORT
GF_Err gf_sk_setup_multicast(GF_Socket *sock, const char *multi_IPAdd, u16 MultiPortNumber, u32 TTL, Bool NoBind, char *local_interface_ip)
{
s32 ret;
u32 flag;
struct ip_mreq M_req;
u32 optval;
#ifdef GPAC_HAS_IPV6
struct sockaddr *addr;
struct addrinfo *res, *aip;
Bool is_ipv6 = GF_FALSE;
u32 type;
#endif
unsigned long local_add_id;
if (!sock || sock->socket) return GF_BAD_PARAM;
if (TTL > 255) TTL = 255;
if (!gf_sk_is_multicast_address(multi_IPAdd)) return GF_BAD_PARAM;
if (local_interface_ip && MobileIPAdd && !strcmp(MobileIPAdd, local_interface_ip) ) {
if (gf_net_mobileip_ctrl(GF_TRUE)==GF_OK) {
sock->flags |= GF_SOCK_IS_MIP;
} else {
local_interface_ip = NULL;
}
}
#ifdef GPAC_HAS_IPV6
is_ipv6 = gf_net_is_ipv6(multi_IPAdd) || gf_net_is_ipv6(local_interface_ip) ? GF_TRUE : GF_FALSE;
type = (sock->flags & GF_SOCK_IS_TCP) ? SOCK_STREAM : SOCK_DGRAM;
if (is_ipv6) {
res = gf_sk_get_ipv6_addr(local_interface_ip, MultiPortNumber, AF_UNSPEC, AI_PASSIVE, type);
if (!res) {
if (local_interface_ip) {
res = gf_sk_get_ipv6_addr(NULL, MultiPortNumber, AF_UNSPEC, AI_PASSIVE, type);
local_interface_ip = NULL;
}
if (!res) return GF_IP_CONNECTION_FAILURE;
}
for (aip=res; aip!=NULL; aip=aip->ai_next) {
if (type != (u32) aip->ai_socktype) continue;
sock->socket = socket(aip->ai_family, aip->ai_socktype, aip->ai_protocol);
if (sock->socket == INVALID_SOCKET) {
sock->socket = NULL_SOCKET;
continue;
}
if ((aip->ai_family!=PF_INET) && aip->ai_next && (aip->ai_next->ai_family==PF_INET) && !gf_net_is_ipv6(multi_IPAdd)) continue;
optval = 1;
setsockopt(sock->socket, SOL_SOCKET, SO_REUSEADDR, (const char *) &optval, sizeof(optval));
#ifdef SO_REUSEPORT
optval = 1;
setsockopt(sock->socket, SOL_SOCKET, SO_REUSEPORT, SSO_CAST &optval, sizeof(optval));
#endif
if (sock->flags & GF_SOCK_NON_BLOCKING) gf_sk_set_block_mode(sock, GF_TRUE);
memcpy(&sock->dest_addr, aip->ai_addr, aip->ai_addrlen);
sock->dest_addr_len = (u32) aip->ai_addrlen;
if (!NoBind) {
ret = bind(sock->socket, aip->ai_addr, (int) aip->ai_addrlen);
if (ret == SOCKET_ERROR) {
closesocket(sock->socket);
sock->socket = NULL_SOCKET;
continue;
}
}
if (aip->ai_family==PF_INET6) sock->flags |= GF_SOCK_IS_IPV6;
else sock->flags &= ~GF_SOCK_IS_IPV6;
break;
}
freeaddrinfo(res);
if (!sock->socket) return GF_IP_CONNECTION_FAILURE;
if (!gf_sk_ipv6_set_remote_address(sock, multi_IPAdd, MultiPortNumber))
return GF_IP_CONNECTION_FAILURE;
addr = (struct sockaddr *)&sock->dest_addr;
if (addr->sa_family == AF_INET) {
M_req.imr_multiaddr.s_addr = ((struct sockaddr_in *)addr)->sin_addr.s_addr;
M_req.imr_interface.s_addr = INADDR_ANY;
ret = setsockopt(sock->socket, IPPROTO_IP, IP_ADD_MEMBERSHIP, (char *) &M_req, sizeof(M_req));
if (ret == SOCKET_ERROR) return GF_IP_CONNECTION_FAILURE;
ret = setsockopt(sock->socket, IPPROTO_IP, IP_MULTICAST_TTL, (char *) &TTL, sizeof(TTL));
if (ret == SOCKET_ERROR) return GF_IP_CONNECTION_FAILURE;
flag = 1;
ret = setsockopt(sock->socket, IPPROTO_IP, IP_MULTICAST_LOOP, (char *) &flag, sizeof(flag));
if (ret == SOCKET_ERROR) return GF_IP_CONNECTION_FAILURE;
}
if (addr->sa_family == AF_INET6) {
struct ipv6_mreq M_reqV6;
memcpy(&M_reqV6.ipv6mr_multiaddr, &(((struct sockaddr_in6 *)addr)->sin6_addr), sizeof(struct in6_addr));
M_reqV6.ipv6mr_interface = 0;
ret = setsockopt(sock->socket, IPPROTO_IPV6, IPV6_MULTICAST_HOPS, (char *) &TTL, sizeof(TTL));
if (ret == SOCKET_ERROR) return GF_IP_CONNECTION_FAILURE;
flag = 1;
ret = setsockopt(sock->socket, IPPROTO_IPV6, IPV6_MULTICAST_LOOP, (char *) &flag, sizeof(flag));
if (ret == SOCKET_ERROR) {
GF_LOG(GF_LOG_WARNING, GF_LOG_NETWORK, ("[Socket] Cannot disale multicast loop: error %d\n", LASTSOCKERROR));
}
ret = setsockopt(sock->socket, IPPROTO_IPV6, IPV6_ADD_MEMBERSHIP, (char *) &M_reqV6, sizeof(M_reqV6));
if (ret == SOCKET_ERROR) return GF_IP_CONNECTION_FAILURE;
}
sock->flags |= GF_SOCK_IS_MULTICAST | GF_SOCK_HAS_PEER;
return GF_OK;
}
#endif
sock->socket = socket(AF_INET, (sock->flags & GF_SOCK_IS_TCP) ? SOCK_STREAM : SOCK_DGRAM, 0);
if (sock->flags & GF_SOCK_NON_BLOCKING) gf_sk_set_block_mode(sock, GF_TRUE);
sock->flags &= ~GF_SOCK_IS_IPV6;
optval = 1;
ret = setsockopt(sock->socket, SOL_SOCKET, SO_REUSEADDR, SSO_CAST &optval, sizeof(optval));
if (ret == SOCKET_ERROR) {
GF_LOG(GF_LOG_WARNING, GF_LOG_NETWORK, ("[core] Failed to set SO_REUSEADDR: error %d\n", LASTSOCKERROR));
}
#ifdef SO_REUSEPORT
optval = 1;
ret = setsockopt(sock->socket, SOL_SOCKET, SO_REUSEPORT, SSO_CAST &optval, sizeof(optval));
if (ret == SOCKET_ERROR) {
GF_LOG(GF_LOG_WARNING, GF_LOG_NETWORK, ("[core] Failed to set SO_REUSEPORT: error %d\n", LASTSOCKERROR));
}
#endif
if (local_interface_ip) local_add_id = inet_addr(local_interface_ip);
else local_add_id = htonl(INADDR_ANY);
if (!NoBind) {
struct sockaddr_in local_address;
memset(&local_address, 0, sizeof(struct sockaddr_in ));
local_address.sin_family = AF_INET;
local_address.sin_addr.s_addr = (u32) local_add_id;
local_address.sin_port = htons( MultiPortNumber);
ret = bind(sock->socket, (struct sockaddr *) &local_address, sizeof(local_address));
if (ret == SOCKET_ERROR) {
local_address.sin_addr.s_addr = local_add_id = htonl(INADDR_ANY);
local_interface_ip = NULL;
ret = bind(sock->socket, (struct sockaddr *) &local_address, sizeof(local_address));
if (ret == SOCKET_ERROR) return GF_IP_CONNECTION_FAILURE;
}
if (local_interface_ip) {
ret = setsockopt(sock->socket, IPPROTO_IP, IP_MULTICAST_IF, (char *) &local_add_id, sizeof(local_add_id));
if (ret == SOCKET_ERROR) return GF_IP_CONNECTION_FAILURE;
}
}
M_req.imr_multiaddr.s_addr = inet_addr(multi_IPAdd);
M_req.imr_interface.s_addr = (u32) local_add_id;
ret = setsockopt(sock->socket, IPPROTO_IP, IP_ADD_MEMBERSHIP, (char *) &M_req, sizeof(M_req));
if (ret == SOCKET_ERROR) {
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[core] cannot join multicast: error %d\n", LASTSOCKERROR));
return GF_IP_CONNECTION_FAILURE;
}
if (TTL) {
ret = setsockopt(sock->socket, IPPROTO_IP, IP_MULTICAST_TTL, (char *)&TTL, sizeof(TTL));
if (ret == SOCKET_ERROR) return GF_IP_CONNECTION_FAILURE;
}
flag = 1;
ret = setsockopt(sock->socket, IPPROTO_IP, IP_MULTICAST_LOOP, (char *) &flag, sizeof(flag));
if (ret == SOCKET_ERROR) {
GF_LOG(GF_LOG_WARNING, GF_LOG_NETWORK, ("[Socket] Cannot disale multicast loop: error %d\n", LASTSOCKERROR));
}
#ifdef GPAC_HAS_IPV6
((struct sockaddr_in *) &sock->dest_addr)->sin_family = AF_INET;
((struct sockaddr_in *) &sock->dest_addr)->sin_addr.s_addr = M_req.imr_multiaddr.s_addr;
((struct sockaddr_in *) &sock->dest_addr)->sin_port = htons( MultiPortNumber);
sock->dest_addr_len = sizeof(struct sockaddr);
#else
sock->dest_addr.sin_family = AF_INET;
sock->dest_addr.sin_addr.s_addr = M_req.imr_multiaddr.s_addr;
sock->dest_addr.sin_port = htons( MultiPortNumber);
#endif
sock->flags |= GF_SOCK_IS_MULTICAST | GF_SOCK_HAS_PEER;
return GF_OK;
}
GF_EXPORT
GF_Err gf_sk_receive(GF_Socket *sock, char *buffer, u32 length, u32 startFrom, u32 *BytesRead)
{
s32 res;
#ifndef __SYMBIAN32__
s32 ready;
struct timeval timeout;
fd_set Group;
#endif
*BytesRead = 0;
if (!sock->socket) return GF_BAD_PARAM;
if (startFrom >= length) return GF_IO_ERR;
#ifndef __SYMBIAN32__
FD_ZERO(&Group);
FD_SET(sock->socket, &Group);
timeout.tv_sec = 0;
timeout.tv_usec = SOCK_MICROSEC_WAIT;
ready = select((int) sock->socket+1, &Group, NULL, NULL, &timeout);
if (ready == SOCKET_ERROR) {
switch (LASTSOCKERROR) {
case EBADF:
GF_LOG(GF_LOG_WARNING, GF_LOG_NETWORK, ("[socket] cannot select, BAD descriptor\n"));
return GF_IP_CONNECTION_CLOSED;
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
case EINTR:
GF_LOG(GF_LOG_WARNING, GF_LOG_NETWORK, ("[socket] network is lost\n"));
return GF_IP_NETWORK_EMPTY;
default:
GF_LOG(GF_LOG_WARNING, GF_LOG_NETWORK, ("[socket] cannot select (error %d)\n", LASTSOCKERROR));
return GF_IP_NETWORK_FAILURE;
}
}
if (!ready || !FD_ISSET(sock->socket, &Group)) {
GF_LOG(GF_LOG_DEBUG, GF_LOG_NETWORK, ("[socket] nothing to be read - ready %d\n", ready));
return GF_IP_NETWORK_EMPTY;
}
#endif
if (sock->flags & GF_SOCK_HAS_PEER)
res = (s32) recvfrom(sock->socket, (char *) buffer + startFrom, length - startFrom, 0, (struct sockaddr *)&sock->dest_addr, &sock->dest_addr_len);
else {
res = (s32) recv(sock->socket, (char *) buffer + startFrom, length - startFrom, 0);
if (res == 0)
return GF_IP_CONNECTION_CLOSED;
}
if (res == SOCKET_ERROR) {
res = LASTSOCKERROR;
switch (res) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
#ifndef __SYMBIAN32__
case EMSGSIZE:
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[socket] error reading - socket error %d\n", res));
return GF_OUT_OF_MEM;
case ENOTCONN:
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[socket] error reading - not connected\n"));
return GF_IP_CONNECTION_CLOSED;
case ECONNRESET:
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[socket] error reading - connection reset\n"));
return GF_IP_CONNECTION_CLOSED;
case ECONNABORTED:
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[socket] error reading - connection aborted\n"));
return GF_IP_CONNECTION_CLOSED;
#endif
default:
GF_LOG(GF_LOG_ERROR, GF_LOG_NETWORK, ("[socket] error reading - socket error %d\n", res));
return GF_IP_NETWORK_FAILURE;
}
}
if (!res) return GF_IP_NETWORK_EMPTY;
*BytesRead = res;
return GF_OK;
}
GF_Err gf_sk_listen(GF_Socket *sock, u32 MaxConnection)
{
s32 i;
if (!sock || !sock->socket) return GF_BAD_PARAM;
if (MaxConnection >= SOMAXCONN) MaxConnection = SOMAXCONN;
i = listen(sock->socket, MaxConnection);
if (i == SOCKET_ERROR) return GF_IP_NETWORK_FAILURE;
sock->flags |= GF_SOCK_IS_LISTENING;
return GF_OK;
}
GF_Err gf_sk_accept(GF_Socket *sock, GF_Socket **newConnection)
{
u32 client_address_size;
SOCKET sk;
#ifndef __SYMBIAN32__
s32 ready;
struct timeval timeout;
fd_set Group;
#endif
*newConnection = NULL;
if (!sock || !(sock->flags & GF_SOCK_IS_LISTENING) ) return GF_BAD_PARAM;
#ifndef __SYMBIAN32__
FD_ZERO(&Group);
FD_SET(sock->socket, &Group);
timeout.tv_sec = 0;
timeout.tv_usec = SOCK_MICROSEC_WAIT;
ready = select((int) sock->socket+1, &Group, NULL, NULL, &timeout);
if (ready == SOCKET_ERROR) {
switch (LASTSOCKERROR) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
default:
return GF_IP_NETWORK_FAILURE;
}
}
if (!ready || !FD_ISSET(sock->socket, &Group)) return GF_IP_NETWORK_EMPTY;
#endif
#ifdef GPAC_HAS_IPV6
client_address_size = sizeof(struct sockaddr_in6);
#else
client_address_size = sizeof(struct sockaddr_in);
#endif
sk = accept(sock->socket, (struct sockaddr *) &sock->dest_addr, &client_address_size);
if (sk == INVALID_SOCKET) {
switch (LASTSOCKERROR) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
default:
return GF_IP_NETWORK_FAILURE;
}
}
(*newConnection) = (GF_Socket *) gf_malloc(sizeof(GF_Socket));
(*newConnection)->socket = sk;
(*newConnection)->flags = sock->flags & ~GF_SOCK_IS_LISTENING;
#ifdef GPAC_HAS_IPV6
memcpy( &(*newConnection)->dest_addr, &sock->dest_addr, client_address_size);
memset(&sock->dest_addr, 0, sizeof(struct sockaddr_in6));
#else
memcpy( &(*newConnection)->dest_addr, &sock->dest_addr, client_address_size);
memset(&sock->dest_addr, 0, sizeof(struct sockaddr_in));
#endif
#if defined(WIN32) || defined(_WIN32_WCE)
wsa_init++;
#endif
(*newConnection)->dest_addr_len = client_address_size;
return GF_OK;
}
GF_Err gf_sk_get_local_info(GF_Socket *sock, u16 *Port, u32 *Familly)
{
#ifdef GPAC_HAS_IPV6
struct sockaddr_in6 the_add;
#else
struct sockaddr_in the_add;
#endif
u32 size;
if (!sock || !sock->socket) return GF_BAD_PARAM;
if (Port) {
#ifdef GPAC_HAS_IPV6
size = sizeof(struct sockaddr_in6);
if (getsockname(sock->socket, (struct sockaddr *) &the_add, &size) == SOCKET_ERROR) return GF_IP_NETWORK_FAILURE;
*Port = (u32) ntohs(the_add.sin6_port);
#else
size = sizeof(struct sockaddr_in);
if (getsockname(sock->socket, (struct sockaddr *) &the_add, &size) == SOCKET_ERROR) return GF_IP_NETWORK_FAILURE;
*Port = (u32) ntohs(the_add.sin_port);
#endif
}
if (Familly) {
if (sock->flags & GF_SOCK_IS_TCP) *Familly = GF_SOCK_TYPE_TCP;
else *Familly = GF_SOCK_TYPE_UDP;
}
return GF_OK;
}
GF_Err gf_sk_server_mode(GF_Socket *sock, Bool serverOn)
{
u32 one;
if (!sock || !(sock->flags & GF_SOCK_IS_TCP) || !sock->socket)
return GF_BAD_PARAM;
one = serverOn ? 1 : 0;
setsockopt(sock->socket, IPPROTO_TCP, TCP_NODELAY, SSO_CAST &one, sizeof(u32));
#ifndef __SYMBIAN32__
setsockopt(sock->socket, SOL_SOCKET, SO_KEEPALIVE, (char *) &one, sizeof(u32));
#endif
return GF_OK;
}
GF_Err gf_sk_get_remote_address(GF_Socket *sock, char *buf)
{
#ifdef GPAC_HAS_IPV6
char clienthost[NI_MAXHOST];
struct sockaddr_in6 * addrptr = (struct sockaddr_in6 *)(&sock->dest_addr_len);
if (!sock || sock->socket) return GF_BAD_PARAM;
if (getnameinfo((struct sockaddr *)addrptr, sock->dest_addr_len, clienthost, sizeof(clienthost), NULL, 0, NI_NUMERICHOST))
return GF_IP_ADDRESS_NOT_FOUND;
strcpy(buf, clienthost);
#else
if (!sock || !sock->socket) return GF_BAD_PARAM;
strcpy(buf, inet_ntoa(sock->dest_addr.sin_addr));
#endif
return GF_OK;
}
GF_Err gf_sk_send_to(GF_Socket *sock, const char *buffer, u32 length, char *remoteHost, u16 remotePort)
{
u32 count, remote_add_len;
s32 res;
#ifdef GPAC_HAS_IPV6
struct sockaddr_storage remote_add;
#else
struct sockaddr_in remote_add;
struct hostent *Host;
#endif
#ifndef __SYMBIAN32__
s32 ready;
struct timeval timeout;
fd_set Group;
#endif
if (!sock || !sock->socket) return GF_BAD_PARAM;
if (remoteHost && !remotePort) return GF_BAD_PARAM;
#ifndef __SYMBIAN32__
FD_ZERO(&Group);
FD_SET(sock->socket, &Group);
timeout.tv_sec = 0;
timeout.tv_usec = SOCK_MICROSEC_WAIT;
ready = select((int) sock->socket+1, NULL, &Group, NULL, &timeout);
if (ready == SOCKET_ERROR) {
switch (LASTSOCKERROR) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
default:
return GF_IP_NETWORK_FAILURE;
}
}
if (!ready || !FD_ISSET(sock->socket, &Group)) return GF_IP_NETWORK_EMPTY;
#endif
#ifdef GPAC_HAS_IPV6
remote_add.ss_family = AF_INET6;
if (remoteHost) {
struct addrinfo *res = gf_sk_get_ipv6_addr(remoteHost, remotePort, AF_UNSPEC, 0, (sock->flags & GF_SOCK_IS_TCP) ? SOCK_STREAM : SOCK_DGRAM);
if (!res) return GF_IP_ADDRESS_NOT_FOUND;
memcpy(&remote_add, res->ai_addr, res->ai_addrlen);
remote_add_len = (u32) res->ai_addrlen;
freeaddrinfo(res);
} else {
struct sockaddr_in6 *remotePtr = (struct sockaddr_in6 *)&remote_add;
struct sockaddr_in6 * addrptr = (struct sockaddr_in6 *)(&sock->dest_addr);
remotePtr->sin6_port = addrptr->sin6_port;
remotePtr->sin6_addr = addrptr->sin6_addr;
remote_add_len = sock->dest_addr_len;
}
#else
remote_add_len = sizeof(struct sockaddr_in);
remote_add.sin_family = AF_INET;
if (remoteHost) {
remote_add.sin_port = htons(remotePort);
Host = gethostbyname(remoteHost);
if (Host == NULL) return GF_IP_ADDRESS_NOT_FOUND;
memcpy((char *) &remote_add.sin_addr, Host->h_addr_list[0], sizeof(u32));
} else {
remote_add.sin_port = sock->dest_addr.sin_port;
remote_add.sin_addr.s_addr = sock->dest_addr.sin_addr.s_addr;
}
#endif
count = 0;
while (count < length) {
res = (s32) sendto(sock->socket, (char *) buffer+count, length - count, 0, (struct sockaddr *) &remote_add, remote_add_len);
if (res == SOCKET_ERROR) {
switch (LASTSOCKERROR) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
default:
return GF_IP_NETWORK_FAILURE;
}
}
count += res;
}
return GF_OK;
}
GF_Err gf_sk_receive_wait(GF_Socket *sock, char *buffer, u32 length, u32 startFrom, u32 *BytesRead, u32 Second )
{
s32 res;
#ifndef __SYMBIAN32__
s32 ready;
struct timeval timeout;
fd_set Group;
#endif
*BytesRead = 0;
if (startFrom >= length) return GF_OK;
#ifndef __SYMBIAN32__
FD_ZERO(&Group);
FD_SET(sock->socket, &Group);
timeout.tv_sec = Second;
timeout.tv_usec = SOCK_MICROSEC_WAIT;
ready = select((int) sock->socket+1, &Group, NULL, NULL, &timeout);
if (ready == SOCKET_ERROR) {
switch (LASTSOCKERROR) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
default:
return GF_IP_NETWORK_FAILURE;
}
}
if (!FD_ISSET(sock->socket, &Group)) {
return GF_IP_NETWORK_EMPTY;
}
#endif
res = (s32) recv(sock->socket, (char *) buffer + startFrom, length - startFrom, 0);
if (res == SOCKET_ERROR) {
switch (LASTSOCKERROR) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
default:
return GF_IP_NETWORK_FAILURE;
}
}
*BytesRead = res;
return GF_OK;
}
GF_Err gf_sk_send_wait(GF_Socket *sock, const char *buffer, u32 length, u32 Second )
{
u32 count;
s32 res;
#ifndef __SYMBIAN32__
s32 ready;
struct timeval timeout;
fd_set Group;
#endif
if (!sock || !sock->socket) return GF_BAD_PARAM;
#ifndef __SYMBIAN32__
FD_ZERO(&Group);
FD_SET(sock->socket, &Group);
timeout.tv_sec = Second;
timeout.tv_usec = SOCK_MICROSEC_WAIT;
ready = select((int) sock->socket+1, NULL, &Group, NULL, &timeout);
if (ready == SOCKET_ERROR) {
switch (LASTSOCKERROR) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
default:
return GF_IP_NETWORK_FAILURE;
}
}
if (!ready || !FD_ISSET(sock->socket, &Group)) {
return GF_IP_NETWORK_EMPTY;
}
#endif
count = 0;
while (count < length) {
res = (s32) send(sock->socket, (char *) buffer+count, length - count, 0);
if (res == SOCKET_ERROR) {
switch (LASTSOCKERROR) {
case EAGAIN:
return GF_IP_SOCK_WOULD_BLOCK;
#ifndef __SYMBIAN32__
case ECONNRESET:
return GF_IP_CONNECTION_CLOSED;
#endif
default:
return GF_IP_NETWORK_FAILURE;
}
}
count += res;
}
return GF_OK;
}
GF_EXPORT
u32 gf_htonl(u32 val)
{
return htonl(val);
}
GF_EXPORT
u32 gf_ntohl(u32 val)
{
return htonl(val);
}
GF_EXPORT
u16 gf_htons(u16 val)
{
return htons(val);
}
GF_EXPORT
u16 gf_tohs(u16 val)
{
return htons(val);
}
#endif