X-Git-Url: http://developer.intra2net.com/git/?p=libt2n;a=blobdiff_plain;f=src%2Fsocket_server.cpp;h=a5ec5d0d5e777cff884fceea0e86fee889d905d0;hp=38c2ea95581c23c8d92cac15612e2fec7180ad00;hb=56f3994d74dbc36d10bfa83b50b016bf269ac563;hpb=a7170401dd90dc79cc7d7a808cfe18a06c7e983b diff --git a/src/socket_server.cpp b/src/socket_server.cpp index 38c2ea9..a5ec5d0 100644 --- a/src/socket_server.cpp +++ b/src/socket_server.cpp @@ -67,7 +67,10 @@ socket_server::socket_server(int port, const std::string& ip) EXCEPTIONSTREAM(error,t2n_server_error,"failed listening on invalid ip " << ip); if (bind (sock, (struct sockaddr *) &sockaddr, sizeof (sockaddr)) < 0) + { + // FIXME: Calls virtual function socket_server::get_logstream() in constructor EXCEPTIONSTREAM(error,t2n_server_error,"error binding socket: " << strerror(errno)); + } start_listening(); } @@ -83,6 +86,8 @@ socket_server::socket_server(const std::string& path, mode_t filemode, const std { unix_path=path; + // TODO: Every EXCEPTIONSTREAM in here calls virtual function get_logstream() + /* Create the socket. */ sock = socket (PF_UNIX, SOCK_STREAM, 0); if (sock < 0) @@ -124,12 +129,27 @@ socket_server::socket_server(const std::string& path, mode_t filemode, const std start_listening(); } +/** + * Destructor + */ socket_server::~socket_server() { - socket_handler::close(); + // close all client connections + server::close(); + + // server socket will be closed by destructor of socket_handler if (get_type()==unix_s) unlink(unix_path.c_str()); + + // disconnect connection<->server pointer + std::map::iterator it, it_end = connections.end(); + for (it = connections.begin(); it != it_end; ++it) + { + socket_server_connection *conn = dynamic_cast(it->second); + if (conn) + conn->my_server = NULL; + } } /// start listening on a new server socket (called by the constructors) @@ -152,14 +172,17 @@ void socket_server::new_connection() int newsock = accept (sock,(struct sockaddr *) &clientname,&size); if (newsock < 0) { - if (errno == EAGAIN) + // return on non-fatal errors (list taken from man-page) + if (errno == EAGAIN || errno == EWOULDBLOCK || errno == ECONNABORTED || errno == EINTR || + errno == EMFILE || errno == ENFILE || errno == ENOBUFS || errno == ENOMEM || + errno == EPROTO || errno == EPERM || errno == ETIMEDOUT) { - LOGSTREAM(error,"accept error (EAGAIN): no connection waiting"); + LOGSTREAM(error,"non-fatal accept error: " << strerror(errno)); return; } - /* default: break */ - EXCEPTIONSTREAM(error,t2n_server_error,"error accepting connection: " << strerror(errno)); + /* fatal error: will usually kill or restart the server */ + EXCEPTIONSTREAM(error,t2n_server_error,"fatal error accepting connection: " << strerror(errno)); } FD_SET (newsock, &connection_set); @@ -172,7 +195,16 @@ void socket_server::new_connection() return; } -bool socket_server::fill_buffer(long long usec_timeout) +/** @brief look for new connections and new data in any of the existing connections + @param usec_timeout wait until new data is found, max timeout usecs. + -1: wait endless + 0: return instantly + @param usec_timeout_remaining if non-NULL the function will write the + not used time to the given target + @retval true if new data was found (does not mean that the received data + is a complete packet though) +*/ +bool socket_server::fill_buffer(long long usec_timeout,long long* usec_timeout_remaining) { fd_set used_fdset=connection_set; @@ -193,6 +225,10 @@ bool socket_server::fill_buffer(long long usec_timeout) int ret=select (FD_SETSIZE, &used_fdset, NULL, NULL, timeout_ptr); + // return the timeout we did not use + if (usec_timeout > 0 && usec_timeout_remaining != NULL) + *usec_timeout_remaining=(tval.tv_sec*1000000)+tval.tv_usec; + if (ret < 0) { if (errno == EINTR) @@ -224,13 +260,23 @@ bool socket_server::fill_buffer(long long usec_timeout) /// call fill_buffer() on all connections, called from fill_buffer() bool socket_server::fill_connection_buffers() { - bool data_found; + bool data_found = false; std::map::iterator ie=connections.end(); for(std::map::iterator i=connections.begin(); i != ie; i++) if (!i->second->server_connection::is_closed()) - if (i->second->fill_buffer(0)) - data_found=true; + { + // shutdown all connections which throw exceptions to protect the server + try + { + if (i->second->fill_buffer(0)) + data_found=true; + } + catch (t2n_transfer_error &e) + { i->second->close(); } + catch(...) + { throw; } + } return data_found; } @@ -241,18 +287,35 @@ void socket_server::remove_connection_socket(int sock) FD_CLR(sock, &connection_set); } +/** + * Destructor + */ +socket_server_connection::~socket_server_connection() +{ + // Only notify parent server about going down. + // The real socket will be closed by the destructor of the base classes. + if (my_server && sock != -1) + { + socket_server *srv = dynamic_cast(my_server); + if (srv) + srv->remove_connection_socket(sock); + } +} + /// close this connection. complete data waiting in the buffer can still be retrieved. void socket_server_connection::close() { - if (!server_connection::is_closed()) + if (my_server && sock != -1) { - socket_handler::close(); - server_connection::close(); + socket_server *srv = dynamic_cast(my_server); + if (srv) + srv->remove_connection_socket(sock); } - if (my_server) + if (!server_connection::is_closed()) { - dynamic_cast(my_server)->remove_connection_socket(sock); + socket_handler::close(); + server_connection::close(); } }