0
0
mirror of https://github.com/yse/easy_profiler.git synced 2024-12-28 01:04:41 +08:00
easy_profiler/src/easy_socket.cpp

258 lines
5.8 KiB
C++
Raw Normal View History

2016-09-18 19:11:22 +03:00
/**
Lightweight profiler library for c++
Copyright(C) 2016 Sergey Yagovtsev
This program is free software : you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program.If not, see <http://www.gnu.org/licenses/>.
**/
#include "profiler/easy_socket.h"
2016-09-19 23:59:55 +03:00
#include <string.h>
2016-09-18 19:11:22 +03:00
2016-09-19 23:59:55 +03:00
#ifdef _WIN32
#pragma comment (lib, "Ws2_32.lib")
#pragma comment (lib, "Mswsock.lib")
#pragma comment (lib, "AdvApi32.lib")
#else
2016-09-18 19:11:22 +03:00
#include <errno.h>
2016-09-19 23:59:55 +03:00
#include <sys/ioctl.h>
#endif
bool EasySocket::checkSocket(socket_t s) const
{
return s > 0;
}
int EasySocket::_close(EasySocket::socket_t s)
{
#ifdef _WIN32
return ::closesocket(s);
#else
//TODO
//return close(s);
#endif
}
void EasySocket::setBlocking(EasySocket::socket_t s, bool blocking)
{
#ifdef _WIN32
u_long iMode = blocking ? 0 : 1;//0 - blocking, 1 - non blocking
ioctlsocket(s, FIONBIO, &iMode);
#else
const int iMode = blocking ? 0 : 1;//0 - blocking, 1 - non blocking
ioctl(s, FIONBIO, (char *)&iMode);
#endif
}
2016-09-18 19:11:22 +03:00
int EasySocket::bind(uint16_t portno)
{
2016-09-19 23:59:55 +03:00
if (!checkSocket(m_socket)) return -1;
2016-09-18 19:11:22 +03:00
struct sockaddr_in serv_addr;
2016-09-19 23:59:55 +03:00
memset(&serv_addr, 0, sizeof(serv_addr));
2016-09-18 19:11:22 +03:00
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = htons(portno);
2016-09-19 23:59:55 +03:00
auto res = ::bind(m_socket, (struct sockaddr *) &serv_addr, sizeof(serv_addr));
return res;
2016-09-18 19:11:22 +03:00
}
void EasySocket::flush()
{
if (m_socket){
2016-09-19 23:59:55 +03:00
_close(m_socket);
2016-09-18 19:11:22 +03:00
}
if (m_replySocket != m_socket){
2016-09-19 23:59:55 +03:00
_close(m_replySocket);
2016-09-18 19:11:22 +03:00
}
2016-09-19 23:59:55 +03:00
#ifdef _WIN32
2016-09-18 19:11:22 +03:00
m_socket = 0;
m_replySocket = 0;
2016-09-19 23:59:55 +03:00
#endif
2016-09-18 19:11:22 +03:00
}
2016-09-19 23:59:55 +03:00
void EasySocket::checkResult(int result)
{
// printf("Errno: %s\n", strerror(errno));
if(result >= 0){
m_state = CONNECTION_STATE_SUCCESS;
return;
}else if(result == -1){
int error_code = 0;
#ifdef _WIN32
error_code = WSAGetLastError();
const int CONNECTION_ABORTED = WSAECONNABORTED;
const int CONNECTION_RESET = WSAECONNRESET;
#else
error_code = errno;
const int CONNECTION_ABORTED = ECONNABORTED;
const int CONNECTION_RESET = ECONNRESET;
#endif
switch(error_code)
{
case CONNECTION_ABORTED:
case CONNECTION_RESET:
m_state = CONNECTION_STATE_DISCONNECTED;
break;
default:
break;
}
}
}
2016-09-18 19:11:22 +03:00
void EasySocket::init()
{
2016-09-19 23:59:55 +03:00
if (wsaret == 0)
{
int protocol = 0;
#ifdef _WIN32
protocol = IPPROTO_TCP;
#endif
m_socket = socket(AF_INET, SOCK_STREAM, protocol);
if (!checkSocket(m_socket)) {
return;
}
}else
return;
setBlocking(m_socket,true);
int opt = 1;
setsockopt(m_socket, SOL_SOCKET, SO_REUSEADDR, (const char *)&opt, sizeof(opt));
2016-09-18 19:11:22 +03:00
}
EasySocket::EasySocket()
{
2016-09-19 23:59:55 +03:00
#ifdef _WIN32
WSADATA wsaData;
wsaret = WSAStartup(0x101, &wsaData);
#else
wsaret = 0;
#endif
2016-09-18 19:11:22 +03:00
init();
2016-09-19 23:59:55 +03:00
#ifndef _WIN32
wsaret = 1;
#endif
2016-09-18 19:11:22 +03:00
}
EasySocket::~EasySocket()
{
flush();
2016-09-19 23:59:55 +03:00
#ifdef _WIN32
if (wsaret == 0)
WSACleanup();
#endif
2016-09-18 19:11:22 +03:00
}
int EasySocket::send(const void *buf, size_t nbyte)
{
2016-09-19 23:59:55 +03:00
if(!checkSocket(m_replySocket)) return -1;
int res = 0;
#ifdef _WIN32
2016-09-20 23:25:13 +03:00
res = ::send(m_replySocket, (const char*)buf, (int)nbyte, 0);
2016-09-19 23:59:55 +03:00
#else
res = ::write(m_replySocket,buf,nbyte);
#endif
2016-09-18 19:11:22 +03:00
checkResult(res);
return res;
}
int EasySocket::receive(void *buf, size_t nbyte)
{
2016-09-19 23:59:55 +03:00
if(!checkSocket(m_replySocket)) return -1;
int res = 0;
#ifdef _WIN32
2016-09-20 23:25:13 +03:00
res = ::recv(m_replySocket, (char*)buf, (int)nbyte, 0);
2016-09-19 23:59:55 +03:00
#else
res = ::read(m_replySocket,buf,nbyte);
#endif
2016-09-18 19:11:22 +03:00
checkResult(res);
if (res == 0){
m_state = CONNECTION_STATE_DISCONNECTED;
}
return res;
}
int EasySocket::listen(int count)
{
2016-09-19 23:59:55 +03:00
if(!checkSocket(m_socket)) return -1;
2016-09-18 19:11:22 +03:00
int res = ::listen(m_socket,count);
checkResult(res);
return res;
}
int EasySocket::accept()
{
2016-09-19 23:59:55 +03:00
if(!checkSocket(m_socket)) return -1;
2016-09-18 19:11:22 +03:00
m_replySocket = ::accept(m_socket,nullptr,nullptr);
2016-09-20 23:25:13 +03:00
checkResult((int)m_replySocket);
2016-09-19 23:59:55 +03:00
if(checkSocket(m_replySocket))
{
int send_buffer = 64*1024*1024;
int send_buffer_sizeof = sizeof(int);
setsockopt(m_replySocket, SOL_SOCKET, SO_SNDBUF, (char*)&send_buffer, send_buffer_sizeof);
2016-09-20 23:25:13 +03:00
2016-09-19 23:59:55 +03:00
//int flag = 1;
//int result = setsockopt(m_replySocket,IPPROTO_TCP,TCP_NODELAY,(char *)&flag,sizeof(int));
2016-09-18 19:11:22 +03:00
2016-09-19 23:59:55 +03:00
//setBlocking(m_replySocket,true);
}
2016-09-20 23:25:13 +03:00
return (int)m_replySocket;
2016-09-18 19:11:22 +03:00
}
2016-09-19 23:59:55 +03:00
bool EasySocket::setAddress(const char *serv, uint16_t portno)
2016-09-18 19:11:22 +03:00
{
2016-09-19 23:59:55 +03:00
server = gethostbyname(serv);
if (server == NULL) {
return false;
//fprintf(stderr,"ERROR, no such host\n");
2016-09-18 19:11:22 +03:00
}
2016-09-20 00:54:56 +03:00
memset((char *)&serv_addr, 0, sizeof(serv_addr));
2016-09-19 23:59:55 +03:00
serv_addr.sin_family = AF_INET;
2016-09-20 00:54:56 +03:00
memcpy((char *)&serv_addr.sin_addr.s_addr, (char *)server->h_addr, server->h_length);
2016-09-18 19:11:22 +03:00
2016-09-20 00:54:56 +03:00
serv_addr.sin_port = htons(portno);
2016-09-18 19:11:22 +03:00
return true;
}
int EasySocket::connect()
{
2016-09-20 00:54:56 +03:00
if (server == NULL || m_socket <=0 ) {
2016-09-18 19:11:22 +03:00
return -1;
2016-09-20 00:54:56 +03:00
//fprintf(stderr,"ERROR, no such host\n");
2016-09-18 19:11:22 +03:00
}
2016-09-20 00:54:56 +03:00
int res = ::connect(m_socket,(struct sockaddr *) &serv_addr,sizeof(serv_addr));
checkResult(res);
if(res == 0){
2016-09-18 19:11:22 +03:00
2016-09-20 00:54:56 +03:00
struct timeval tv;
2016-09-18 19:11:22 +03:00
2016-09-20 00:54:56 +03:00
tv.tv_sec = 1;
tv.tv_usec = 0;
2016-09-18 19:11:22 +03:00
2016-09-20 00:54:56 +03:00
setsockopt(m_socket, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv,sizeof(struct timeval));
2016-09-18 19:11:22 +03:00
2016-09-20 00:54:56 +03:00
m_replySocket = m_socket;
}
return res;
2016-09-18 19:11:22 +03:00
}