2014-04-20 20:14:48 -06:00
|
|
|
/* Copyright (c) 2008-2014, Avian Contributors
|
2013-11-05 00:07:43 +03:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This file implements a simple cross-platform JNI sockets API
|
|
|
|
* It is used from different classes of the default Avian classpath
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "sockets.h"
|
|
|
|
|
|
|
|
namespace avian {
|
|
|
|
namespace classpath {
|
|
|
|
namespace sockets {
|
|
|
|
|
|
|
|
int last_socket_error() {
|
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
|
|
int error = WSAGetLastError();
|
|
|
|
#else
|
|
|
|
int error = errno;
|
|
|
|
#endif
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void init(JNIEnv* ONLY_ON_WINDOWS(e)) {
|
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
|
|
static bool wsaInitialized = false;
|
|
|
|
if (not wsaInitialized) {
|
|
|
|
WSADATA data;
|
|
|
|
int r = WSAStartup(MAKEWORD(2, 2), &data);
|
|
|
|
if (r or LOBYTE(data.wVersion) != 2 or HIBYTE(data.wVersion) != 2) {
|
|
|
|
throwNew(e, "java/io/IOException", "WSAStartup failed");
|
|
|
|
} else {
|
|
|
|
wsaInitialized = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
SOCKET create(JNIEnv* e) {
|
|
|
|
SOCKET sock;
|
|
|
|
if (INVALID_SOCKET == (sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP))) {
|
|
|
|
char buf[255];
|
|
|
|
sprintf(buf, "Can't create a socket. System error: %d", last_socket_error());
|
|
|
|
throwNew(e, "java/io/IOException", buf);
|
|
|
|
return 0; // This doesn't matter cause we have risen an exception
|
|
|
|
}
|
|
|
|
return sock;
|
|
|
|
}
|
|
|
|
|
|
|
|
void connect(JNIEnv* e, SOCKET sock, long addr, short port) {
|
|
|
|
sockaddr_in adr;
|
|
|
|
adr.sin_family = AF_INET;
|
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
|
|
adr.sin_addr.S_un.S_addr = htonl(addr);
|
|
|
|
#else
|
|
|
|
adr.sin_addr.s_addr = htonl(addr);
|
|
|
|
#endif
|
|
|
|
adr.sin_port = htons (port);
|
|
|
|
|
|
|
|
if (SOCKET_ERROR == ::connect(sock, (sockaddr* )&adr, sizeof(adr)))
|
|
|
|
{
|
|
|
|
char buf[255];
|
|
|
|
sprintf(buf, "Can't connect a socket. System error: %d", last_socket_error());
|
|
|
|
throwNew(e, "java/io/IOException", buf);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void bind(JNIEnv* e, SOCKET sock, long addr, short port) {
|
|
|
|
sockaddr_in adr;
|
|
|
|
adr.sin_family = AF_INET;
|
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
|
|
adr.sin_addr.S_un.S_addr = htonl(addr);
|
|
|
|
#else
|
|
|
|
adr.sin_addr.s_addr = htonl(addr);
|
|
|
|
#endif
|
|
|
|
adr.sin_port = htons (port);
|
|
|
|
|
|
|
|
if (SOCKET_ERROR == ::bind(sock, (sockaddr* )&adr, sizeof(adr)))
|
|
|
|
{
|
|
|
|
char buf[255];
|
|
|
|
sprintf(buf, "Can't bind a socket. System error: %d", last_socket_error());
|
|
|
|
throwNew(e, "java/io/IOException", buf);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SOCKET accept(JNIEnv* e, SOCKET sock, long* client_addr, short* client_port) {
|
|
|
|
sockaddr_in adr;
|
|
|
|
SOCKET client_socket = ::accept(sock, (sockaddr* )&adr, NULL);
|
|
|
|
if (INVALID_SOCKET == client_socket) {
|
|
|
|
char buf[255];
|
|
|
|
sprintf(buf, "Can't accept the incoming connection. System error: %d", last_socket_error());
|
|
|
|
throwNew(e, "java/io/IOException", buf);
|
|
|
|
return INVALID_SOCKET;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (client_addr != NULL) {
|
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
|
|
*client_addr = ntohl(adr.sin_addr.S_un.S_addr);
|
|
|
|
#else
|
|
|
|
*client_addr = ntohl(adr.sin_addr.s_addr);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
if (client_port != NULL) {
|
|
|
|
*client_port = ntohs (adr.sin_port);
|
|
|
|
}
|
|
|
|
|
|
|
|
return client_socket;
|
|
|
|
}
|
|
|
|
|
|
|
|
void send(JNIEnv* e, SOCKET sock, const char* buff_ptr, int buff_size) {
|
|
|
|
if (SOCKET_ERROR == ::send(sock, buff_ptr, buff_size, 0)) {
|
|
|
|
char buf[255];
|
|
|
|
sprintf(buf, "Can't send data through the socket. System error: %d", last_socket_error());
|
|
|
|
throwNew(e, "java/io/IOException", buf);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int recv(JNIEnv* e, SOCKET sock, char* buff_ptr, int buff_size) {
|
|
|
|
int length = ::recv(sock, buff_ptr, buff_size, 0);
|
|
|
|
if (SOCKET_ERROR == length) {
|
|
|
|
char buf[255];
|
|
|
|
sprintf(buf, "Can't receive data through the socket. System error: %d", last_socket_error());
|
|
|
|
throwNew(e, "java/io/IOException", buf);
|
|
|
|
return 0; // This doesn't matter cause we have risen an exception
|
|
|
|
}
|
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
|
|
|
void abort(JNIEnv* e, SOCKET sock) {
|
|
|
|
if (SOCKET_ERROR == ::closesocket(sock)) {
|
|
|
|
char buf[255];
|
|
|
|
sprintf(buf, "Can't close the socket. System error: %d", last_socket_error());
|
|
|
|
throwNew(e, "java/io/IOException", buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void close(JNIEnv* e, SOCKET sock) {
|
|
|
|
if (SOCKET_ERROR == ::shutdown(sock, SD_BOTH)) {
|
|
|
|
int errcode = last_socket_error();
|
|
|
|
if (errcode != ENOTCONN) {
|
|
|
|
char buf[255];
|
|
|
|
sprintf(buf, "Can't shutdown the socket. System error: %d", errcode);
|
|
|
|
throwNew(e, "java/io/IOException", buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void close_input(JNIEnv* e, SOCKET sock) {
|
|
|
|
if (SOCKET_ERROR == ::shutdown(sock, SD_RECEIVE)) {
|
|
|
|
int errcode = last_socket_error();
|
|
|
|
if (errcode != ENOTCONN) {
|
|
|
|
char buf[255];
|
|
|
|
sprintf(buf, "Can't shutdown the socket. System error: %d", errcode);
|
|
|
|
throwNew(e, "java/io/IOException", buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void close_output(JNIEnv* e, SOCKET sock) {
|
|
|
|
if (SOCKET_ERROR == ::shutdown(sock, SD_SEND)) {
|
|
|
|
int errcode = last_socket_error();
|
|
|
|
if (errcode != ENOTCONN) {
|
|
|
|
char buf[255];
|
|
|
|
sprintf(buf, "Can't shutdown the socket. System error: %d", errcode);
|
|
|
|
throwNew(e, "java/io/IOException", buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|