summaryrefslogtreecommitdiffstats
path: root/ipc/unixsocket/DataSocket.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'ipc/unixsocket/DataSocket.cpp')
-rw-r--r--ipc/unixsocket/DataSocket.cpp136
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);
+}
+
+}
+}