/*
* psocksxx - A C++ wrapper for POSIX sockets
* Copyright (C) 2013 Uditha Atukorala
*
* This software library is free software; you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This software library 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 Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this software library. If not, see .
*
*/
#include "lecho.h"
#include
#include
#include
#include
#include
#include
lecho::lecho( const char * path ) :
_sockfd( -1 ), _lsock_path( path ) {
// fork
_cpid = fork();
if ( _cpid == -1 ) {
std::cerr << "failed to fork out child process" << std::endl;
} else if ( _cpid == 0 ) {
// child - initialise echo server on a local socket
init_listener();
// serve requests
serve_requests();
// exit - should never get here
exit( 0 );
} else {
// parent - wait for the child to create the socket and return
wait_connect();
}
}
lecho::~lecho() {
// cleanup
shutdown();
}
void lecho::init_listener() {
// initialise socket
if ( ( _sockfd = socket( PF_LOCAL, SOCK_STREAM, 0 ) ) < 0 ) {
std::cerr << "failed to setup local socket" << std::endl;
exit( -1 );
}
// socket address
sockaddr_un saddr;
bzero( (void *) &saddr, sizeof( saddr ) );
saddr.sun_family = AF_LOCAL;
strcpy( saddr.sun_path, _lsock_path );
// bind
if ( bind( _sockfd, (::sockaddr *) &saddr,
sizeof( sockaddr_un ) ) != 0 ) {
std::cerr << "failed to bind to local socket" << std::endl;
exit( -1 );
}
// listen
if ( listen( _sockfd, 2 ) != 0 ) {
std::cerr << "failed to listen on local socket" << std::endl;
exit( -1 );
}
}
void lecho::serve_requests() {
int peer_sockfd;
sockaddr_un peer_addr;
socklen_t addrlen = sizeof( peer_addr );
int n = 0;
char cbuffer[64];
bool close_peer;
for (;;) {
if ( ( peer_sockfd = accept( _sockfd, (sockaddr *) &peer_addr, &addrlen ) ) == -1 ) {
continue;
}
close_peer = false;
do {
// read from the socket
n = recv( peer_sockfd, cbuffer, sizeof( cbuffer ), 0 );
// sanity check - did we receive anything?
if ( n <= 0 ) {
close_peer = true;
} else {
// send back what we received
if ( send( peer_sockfd, cbuffer, n, 0 ) < 0 ) {
close_peer = true;
}
}
} while (! close_peer );
// close peer
close( peer_sockfd );
}
}
void lecho::shutdown() {
if ( _cpid == 0 ) {
// child - should never get here
} else {
// parent - send kill to child
kill( _cpid, SIGTERM );
}
// unlink
unlink( _lsock_path );
}
void lecho::wait_connect() {
int sockfd = -1;
unsigned int max_tries = 10;
sockaddr_un saddr;
// initialise socket
if ( ( sockfd = socket( PF_LOCAL, SOCK_STREAM, 0 ) ) < 0 ) {
std::cerr << "failed to setup local socket" << std::endl;
exit( -1 );
}
// socket address structure
bzero( (void *) &saddr, sizeof( saddr ) );
saddr.sun_family = AF_LOCAL;
strcpy( saddr.sun_path, _lsock_path );
// loop until we can establish a connection or
// we exhaust our attempts
for ( int i = 0; i < max_tries; i++ ) {
// try to connect
if ( ( connect( sockfd, (sockaddr *) &saddr, sizeof( sockaddr_un ) ) == -1 ) ) {
// connect failed - sleep for a bit
usleep( 500 );
} else {
// connect successful - break out of the loop
break;
}
}
// close socket
close( sockfd );
}