mirror of https://gitee.com/bigwinds/arangodb
364 lines
8.4 KiB
C++
364 lines
8.4 KiB
C++
////////////////////////////////////////////////////////////////////////////////
|
|
/// @brief base class for input-output tasks from sockets
|
|
///
|
|
/// @file
|
|
///
|
|
/// DISCLAIMER
|
|
///
|
|
/// Copyright 2010-2011 triagens GmbH, Cologne, Germany
|
|
///
|
|
/// Licensed under the Apache License, Version 2.0 (the "License");
|
|
/// you may not use this file except in compliance with the License.
|
|
/// You may obtain a copy of the License at
|
|
///
|
|
/// http://www.apache.org/licenses/LICENSE-2.0
|
|
///
|
|
/// Unless required by applicable law or agreed to in writing, software
|
|
/// distributed under the License is distributed on an "AS IS" BASIS,
|
|
/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
/// See the License for the specific language governing permissions and
|
|
/// limitations under the License.
|
|
///
|
|
/// Copyright holder is triAGENS GmbH, Cologne, Germany
|
|
///
|
|
/// @author Dr. Frank Celler
|
|
/// @author Achim Brandt
|
|
/// @author Copyright 2009-2011, triAGENS GmbH, Cologne, Germany
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include "SocketTask.h"
|
|
|
|
#include <errno.h>
|
|
|
|
#include <Logger/Logger.h>
|
|
#include <Basics/MutexLocker.h>
|
|
#include <Basics/StringBuffer.h>
|
|
|
|
#include "Scheduler/Scheduler.h"
|
|
|
|
using namespace triagens::basics;
|
|
|
|
namespace triagens {
|
|
namespace rest {
|
|
|
|
static Mutex lockNumWrites;
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// constructors and destructors
|
|
// -----------------------------------------------------------------------------
|
|
|
|
SocketTask::SocketTask (socket_t fd)
|
|
: Task("SocketTask"),
|
|
readWatcher(0),
|
|
writeWatcher(0),
|
|
watcher(0),
|
|
commSocket(fd),
|
|
writeBuffer(0),
|
|
ownBuffer(true),
|
|
writeLength(0) {
|
|
readBuffer = new StringBuffer();
|
|
tmpReadBuffer = new char[READ_BLOCK_SIZE];
|
|
}
|
|
|
|
|
|
|
|
SocketTask::~SocketTask () {
|
|
if (commSocket != -1) {
|
|
close(commSocket);
|
|
}
|
|
|
|
if (writeBuffer != 0) {
|
|
if (ownBuffer) {
|
|
writeBuffer->free();
|
|
delete writeBuffer;
|
|
}
|
|
}
|
|
|
|
readBuffer->free();
|
|
delete readBuffer;
|
|
|
|
delete[] tmpReadBuffer;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// Task methods
|
|
// -----------------------------------------------------------------------------
|
|
|
|
void SocketTask::setup (Scheduler* scheduler, EventLoop loop) {
|
|
this->scheduler = scheduler;
|
|
this->loop = loop;
|
|
|
|
watcher = scheduler->installAsyncEvent(loop, this);
|
|
readWatcher = scheduler->installSocketEvent(loop, EVENT_SOCKET_READ, this, commSocket);
|
|
writeWatcher = scheduler->installSocketEvent(loop, EVENT_SOCKET_WRITE, this, commSocket);
|
|
|
|
tid = Thread::currentThreadId();
|
|
}
|
|
|
|
|
|
|
|
void SocketTask::cleanup () {
|
|
scheduler->uninstallEvent(watcher);
|
|
watcher = 0;
|
|
|
|
scheduler->uninstallEvent(readWatcher);
|
|
readWatcher = 0;
|
|
|
|
scheduler->uninstallEvent(writeWatcher);
|
|
writeWatcher = 0;
|
|
}
|
|
|
|
|
|
|
|
bool SocketTask::handleEvent (EventToken token, EventType revents) {
|
|
bool result = true;
|
|
bool closed = false;
|
|
|
|
if (token == readWatcher && (revents & EVENT_SOCKET_READ)) {
|
|
result = handleRead(closed);
|
|
}
|
|
|
|
if (result && ! closed && token == writeWatcher) {
|
|
bool noWrite = false;
|
|
|
|
{
|
|
MUTEX_LOCKER(writeBufferLock);
|
|
noWrite = (writeBuffer == 0);
|
|
}
|
|
|
|
if (revents & EVENT_SOCKET_WRITE) {
|
|
result = handleWrite(closed, noWrite);
|
|
}
|
|
}
|
|
|
|
if (result) {
|
|
MUTEX_LOCKER(writeBufferLock);
|
|
|
|
if (writeBuffer == 0) {
|
|
scheduler->stopSocketEvents(writeWatcher);
|
|
}
|
|
else {
|
|
scheduler->startSocketEvents(writeWatcher);
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// protected methods
|
|
// -----------------------------------------------------------------------------
|
|
|
|
bool SocketTask::handleWrite (bool& closed, bool noWrite) {
|
|
|
|
closed = false;
|
|
|
|
if (noWrite) {
|
|
return true;
|
|
}
|
|
|
|
bool callCompletedWriteBuffer = false;
|
|
|
|
{
|
|
MUTEX_LOCKER(writeBufferLock);
|
|
|
|
size_t len = writeBuffer->length() - writeLength;
|
|
|
|
int nr = 0;
|
|
|
|
if (0 < len) {
|
|
nr = write(commSocket, writeBuffer->begin() + writeLength, (int) len);
|
|
|
|
if (nr < 0) {
|
|
if (errno == EINTR) {
|
|
return handleWrite(closed, noWrite);
|
|
}
|
|
else if (errno != EWOULDBLOCK) {
|
|
LOGGER_DEBUG << "write failed with " << errno << " (" << strerror(errno) << ")";
|
|
|
|
return false;
|
|
}
|
|
else {
|
|
nr = 0;
|
|
}
|
|
}
|
|
|
|
len -= nr;
|
|
}
|
|
|
|
if (len == 0) {
|
|
if (ownBuffer) {
|
|
writeBuffer->free();
|
|
delete writeBuffer;
|
|
}
|
|
|
|
writeBuffer = 0;
|
|
callCompletedWriteBuffer = true;
|
|
}
|
|
else {
|
|
writeLength += nr;
|
|
}
|
|
}
|
|
|
|
// we have to release the lock, before calling completedWriteBuffer
|
|
if (callCompletedWriteBuffer) {
|
|
completedWriteBuffer(closed);
|
|
|
|
if (closed) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// we might have a new write buffer or none at all
|
|
{
|
|
MUTEX_LOCKER(writeBufferLock);
|
|
|
|
if (writeBuffer == 0) {
|
|
scheduler->stopSocketEvents(writeWatcher);
|
|
}
|
|
else {
|
|
scheduler->startSocketEvents(writeWatcher);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
bool SocketTask::hasWriteBuffer () const {
|
|
MUTEX_LOCKER(writeBufferLock);
|
|
|
|
return writeBuffer != 0;
|
|
}
|
|
|
|
|
|
|
|
void SocketTask::setWriteBuffer (StringBuffer* buffer, bool ownBuffer) {
|
|
bool callCompletedWriteBuffer = false;
|
|
|
|
{
|
|
MUTEX_LOCKER(writeBufferLock);
|
|
|
|
writeLength = 0;
|
|
|
|
if (buffer->empty()) {
|
|
if (ownBuffer) {
|
|
buffer->free();
|
|
delete buffer;
|
|
}
|
|
|
|
writeBuffer = 0;
|
|
callCompletedWriteBuffer = true;
|
|
}
|
|
else {
|
|
if (writeBuffer != 0) {
|
|
if (this->ownBuffer) {
|
|
writeBuffer->free();
|
|
delete writeBuffer;
|
|
}
|
|
|
|
}
|
|
|
|
writeBuffer = buffer;
|
|
this->ownBuffer = ownBuffer;
|
|
}
|
|
}
|
|
|
|
// we have to release the lock before calling completedWriteBuffer
|
|
if (callCompletedWriteBuffer) {
|
|
bool closed;
|
|
completedWriteBuffer(closed);
|
|
}
|
|
|
|
// we might have a new write buffer or none at all
|
|
if (tid == Thread::currentThreadId()) {
|
|
MUTEX_LOCKER(writeBufferLock);
|
|
|
|
if (writeBuffer == 0) {
|
|
scheduler->stopSocketEvents(writeWatcher);
|
|
}
|
|
else {
|
|
scheduler->startSocketEvents(writeWatcher);
|
|
}
|
|
}
|
|
else {
|
|
scheduler->sendAsync(watcher);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void SocketTask::appendWriteBuffer (StringBuffer* buffer) {
|
|
if (buffer->empty()) {
|
|
return;
|
|
}
|
|
|
|
bool newBuffer = false;
|
|
|
|
{
|
|
MUTEX_LOCKER(writeBufferLock);
|
|
|
|
if (writeBuffer == 0) {
|
|
writeLength = 0;
|
|
|
|
writeBuffer = new StringBuffer;
|
|
writeBuffer->initialise();
|
|
|
|
newBuffer = true;
|
|
}
|
|
|
|
writeBuffer->appendText(*buffer);
|
|
}
|
|
|
|
// we might have a new write buffer
|
|
if (newBuffer) {
|
|
if (tid == Thread::currentThreadId()) {
|
|
MUTEX_LOCKER(writeBufferLock);
|
|
|
|
scheduler->startSocketEvents(writeWatcher);
|
|
}
|
|
else {
|
|
scheduler->sendAsync(watcher);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
bool SocketTask::fillReadBuffer (bool& closed) {
|
|
closed = false;
|
|
|
|
int nr = read(commSocket, tmpReadBuffer, READ_BLOCK_SIZE);
|
|
|
|
if (nr > 0) {
|
|
readBuffer->appendText(tmpReadBuffer, nr);
|
|
|
|
return true;
|
|
}
|
|
else if (nr == 0) {
|
|
closed = true;
|
|
|
|
LOGGER_TRACE << "read return 0 with " << errno << " (" << strerror(errno) << ")";
|
|
|
|
return false;
|
|
}
|
|
else {
|
|
if (errno == EINTR) {
|
|
return fillReadBuffer(closed);
|
|
}
|
|
else if (errno != EWOULDBLOCK) {
|
|
LOGGER_TRACE << "read failed with " << errno << " (" << strerror(errno) << ")";
|
|
|
|
return false;
|
|
}
|
|
else {
|
|
LOGGER_TRACE << "read would block with " << errno << " (" << strerror(errno) << ")";
|
|
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|