#include "EPoll.h" #include "Socket.h" #include "Exception.h" #include "ZString.h" #include "Log.h" namespace core { void sigpipe_handler(int unused) {} Socket::Socket(EPoll &ePoll, std::string text) : ePoll(ePoll), text(text) { buffer = (char *)malloc(4096); length = 4096; } Socket::~Socket() { shutDown = true; onUnregister(); ePoll.unregisterSocket(this); // coreutils::Log(coreutils::LOG_DEBUG_4) << "BMAXFree on socket " << descriptor; free(buffer); if(descriptor == -1) return; close(descriptor); coreutils::Log(coreutils::LOG_DEBUG_1) << text << " has ended (" << descriptor << ")."; sleep(1); } void Socket::setDescriptor(int descriptor) { if((descriptor == -1) && (errno == 24)) { shutdown("Too many files open"); throw coreutils::Exception("Too many files open. Refusing connection."); } coreutils::Log(coreutils::LOG_DEBUG_1) << text << " has started (" << descriptor << ")."; // coreutils::Log(coreutils::LOG_DEBUG_3) << "Descriptor set to " << descriptor << " for Socket."; if(descriptor < 3) throw coreutils::Exception("Descriptor out of range", __FILE__, __LINE__); this->descriptor = descriptor; onRegister(); ePoll.registerSocket(this); onRegistered(); } int Socket::getDescriptor() { return descriptor; } void Socket::setBufferSize(int length) { this->length = length; buffer = (char *)realloc(buffer, length); } int Socket::getBufferSize() { return length; } void Socket::onRegister() {} void Socket::onRegistered() {} void Socket::onUnregister() {} void Socket::onUnregistered() {} bool Socket::eventReceived(struct epoll_event event, long long eventId) { // coreutils::Log(coreutils::LOG_DEBUG_3) << "eventReceived " << eventId << ": " << descriptor << ":" << event.events; inHandler = true; if(event.events & EPOLLRDHUP) { readHangup = true; shutdown("hangup received"); } if(event.events & EPOLLIN) { coreutils::ZString zbuffer(buffer, length); receiveData(zbuffer); } if(event.events & EPOLLWRNORM) { writeSocket(); } if(event.events & EPOLLHUP) { shutdown(); } inHandler = false; return !shutDown; } void Socket::onDataReceived(std::string data) { throw coreutils::Exception("Need to override onDataReceived.", __FILE__, __LINE__, -1); } void Socket::onDataReceived(coreutils::ZString &data) { onDataReceived(std::string(data.getData(), data.getLength())); } void Socket::receiveData(coreutils::ZString &buffer) { coreutils::ZString blank(""); if(buffer.getLength() <= 0) throw coreutils::Exception("Request to receive data with a zero buffer length.", __FILE__, __LINE__, -1); int len; int error = -1; if((len = ::read(getDescriptor(), buffer.getData(), buffer.getLength())) >= 0) { coreutils::ZString zbuffer(buffer.getData(), len); coreutils::Log(coreutils::LOG_DEBUG_1) << zbuffer; onDataReceived(zbuffer); } else { error = errno; switch (error) { // When a listening socket receives a connection // request we get one of these. // case ENOTCONN: onDataReceived(blank); break; case ECONNRESET: break; default: throw coreutils::Exception("Error in read of data from socket.", __FILE__, __LINE__, error); } } } void Socket::writeSocket() { if(fifo.size() > 0) { outlock.lock(); if(!shutDown) int rc = ::write(descriptor, fifo.front().c_str(), fifo.front().length()); fifo.pop(); outlock.unlock(); } } int Socket::write(std::string data, Socket *sender) { outlock.lock(); fifo.emplace(data); outlock.unlock(); // coreutils::Log(coreutils::LOG_DEBUG_4) << "inHandler " << descriptor << " " << inHandler << ":" << shutDown << ":[" << data << "]"; if(!inHandler && sender != this) ePoll.resetSocket(this); return 1; } void Socket::output(std::stringstream &out) { out << "|" << descriptor << "|"; } bool Socket::needsToWrite() { return fifo.size() > 0; } void Socket::shutdown(std::string text) { coreutils::Log(coreutils::LOG_DEBUG_2) << "Shutdown requested from " << this->text << " (" << descriptor << ") with reason: " << text << "."; shutDown = true; reset = false; } }