diff options
Diffstat (limited to 'ipc/unixsocket/DataSocket.cpp')
-rw-r--r-- | ipc/unixsocket/DataSocket.cpp | 136 |
1 files changed, 136 insertions, 0 deletions
diff --git a/ipc/unixsocket/DataSocket.cpp b/ipc/unixsocket/DataSocket.cpp new file mode 100644 index 000000000..057c59203 --- /dev/null +++ b/ipc/unixsocket/DataSocket.cpp @@ -0,0 +1,136 @@ +/* -*- Mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 40 -*- */ +/* vim: set ts=2 et sw=2 tw=80: */ + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + */ + +#include "DataSocket.h" +#ifdef MOZ_TASK_TRACER +#include "GeckoTaskTracer.h" +#endif +#include "nsISupportsImpl.h" // for MOZ_COUNT_CTOR, MOZ_COUNT_DTOR + +#ifdef MOZ_TASK_TRACER +using namespace mozilla::tasktracer; +#endif + +namespace mozilla { +namespace ipc { + +// +// DataSocketIO +// + +DataSocketIO::~DataSocketIO() +{ + MOZ_COUNT_DTOR_INHERITED(DataSocketIO, SocketIOBase); +} + +void +DataSocketIO::EnqueueData(UnixSocketIOBuffer* aBuffer) +{ + if (!aBuffer->GetSize()) { + delete aBuffer; // delete empty data immediately + return; + } + mOutgoingQ.AppendElement(aBuffer); +} + +bool +DataSocketIO::HasPendingData() const +{ + return !mOutgoingQ.IsEmpty(); +} + +ssize_t +DataSocketIO::ReceiveData(int aFd) +{ + MOZ_ASSERT(aFd >= 0); + + UnixSocketIOBuffer* incoming; + nsresult rv = QueryReceiveBuffer(&incoming); + if (NS_FAILED(rv)) { + /* an error occured */ + GetConsumerThread()->PostTask( + MakeAndAddRef<SocketRequestClosingTask>(this)); + return -1; + } + + ssize_t res = incoming->Receive(aFd); + if (res < 0) { + /* an I/O error occured */ + DiscardBuffer(); + GetConsumerThread()->PostTask( + MakeAndAddRef<SocketRequestClosingTask>(this)); + return -1; + } else if (!res) { + /* EOF or peer shut down sending */ + DiscardBuffer(); + GetConsumerThread()->PostTask( + MakeAndAddRef<SocketRequestClosingTask>(this)); + return 0; + } + +#ifdef MOZ_TASK_TRACER + /* Make unix socket creation events to be the source events of TaskTracer, + * and originate the rest correlation tasks from here. + */ + AutoSourceEvent taskTracerEvent(SourceEventType::Unixsocket); +#endif + + ConsumeBuffer(); + + return res; +} + +nsresult +DataSocketIO::SendPendingData(int aFd) +{ + MOZ_ASSERT(aFd >= 0); + + while (HasPendingData()) { + UnixSocketIOBuffer* outgoing = mOutgoingQ.ElementAt(0); + + ssize_t res = outgoing->Send(aFd); + if (res < 0) { + /* an I/O error occured */ + GetConsumerThread()->PostTask( + MakeAndAddRef<SocketRequestClosingTask>(this)); + return NS_ERROR_FAILURE; + } else if (!res && outgoing->GetSize()) { + /* I/O is currently blocked; try again later */ + return NS_OK; + } + if (!outgoing->GetSize()) { + mOutgoingQ.RemoveElementAt(0); + delete outgoing; + } + } + + return NS_OK; +} + +DataSocketIO::DataSocketIO(MessageLoop* aConsumerLoop) + : SocketIOBase(aConsumerLoop) +{ + MOZ_COUNT_CTOR_INHERITED(DataSocketIO, SocketIOBase); +} + +// +// DataSocket +// + +DataSocket::DataSocket() +{ + MOZ_COUNT_CTOR_INHERITED(DataSocket, SocketBase); +} + +DataSocket::~DataSocket() +{ + MOZ_COUNT_DTOR_INHERITED(DataSocket, SocketBase); +} + +} +} |