mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
Bug 1171994: Use |RilSocket| to handle RIL messages on the RIL worker, r=htsai
This patch finally breaks up forwarding received RIL messages to the main thread before they go to the RIL worker. Any RIL message that is received on th I/O thread is forwarded directly to the RIL worker thread and handed over to the RIL worker JS code. The patch includes a number of changes. They all depend on each other, so there's no good way of landing them one-by-one. * |RilConsumer| now runs on the RIL worker thread. * |RilWorker| uses tasks to register/unregister |RilConsumer| in the worker. * |RilConsumer| uses |RilSocket| instead of |StreamSocket|. * With |RilSocket|, received RIL messages do not go through main. They are forwared to the RIL worker and handed over to JS immediately.
This commit is contained in:
parent
4f939a0d7b
commit
da78986754
240
ipc/ril/Ril.cpp
240
ipc/ril/Ril.cpp
@ -5,16 +5,15 @@
|
|||||||
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
#include "mozilla/ipc/Ril.h"
|
#include "mozilla/ipc/Ril.h"
|
||||||
|
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
#include <sys/un.h>
|
#include <sys/un.h>
|
||||||
#include <netdb.h> // For gethostbyname.
|
|
||||||
#include "jsfriendapi.h"
|
#include "jsfriendapi.h"
|
||||||
#include "mozilla/ArrayUtils.h"
|
#include "mozilla/ArrayUtils.h"
|
||||||
|
#include "mozilla/dom/ScriptSettings.h"
|
||||||
#include "mozilla/dom/workers/Workers.h"
|
#include "mozilla/dom/workers/Workers.h"
|
||||||
#include "mozilla/ipc/StreamSocket.h"
|
#include "mozilla/ipc/RilSocket.h"
|
||||||
#include "mozilla/ipc/StreamSocketConsumer.h"
|
#include "mozilla/ipc/RilSocketConsumer.h"
|
||||||
#include "nsThreadUtils.h" // For NS_IsMainThread.
|
#include "nsThreadUtils.h" // For NS_IsMainThread.
|
||||||
#include "RilConnector.h"
|
#include "RilConnector.h"
|
||||||
|
|
||||||
@ -29,11 +28,11 @@
|
|||||||
#define CHROMIUM_LOG(args...) printf(args);
|
#define CHROMIUM_LOG(args...) printf(args);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
USING_WORKERS_NAMESPACE
|
|
||||||
|
|
||||||
namespace mozilla {
|
namespace mozilla {
|
||||||
namespace ipc {
|
namespace ipc {
|
||||||
|
|
||||||
|
USING_WORKERS_NAMESPACE
|
||||||
|
|
||||||
class RilConsumer;
|
class RilConsumer;
|
||||||
|
|
||||||
static const char RIL_SOCKET_NAME[] = "/dev/socket/rilproxy";
|
static const char RIL_SOCKET_NAME[] = "/dev/socket/rilproxy";
|
||||||
@ -44,29 +43,30 @@ static nsTArray<nsAutoPtr<RilConsumer>> sRilConsumers;
|
|||||||
// RilConsumer
|
// RilConsumer
|
||||||
//
|
//
|
||||||
|
|
||||||
class RilConsumer final : public StreamSocketConsumer
|
class RilConsumer final : public RilSocketConsumer
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
friend class RilWorker;
|
RilConsumer();
|
||||||
|
|
||||||
RilConsumer(unsigned long aClientId,
|
|
||||||
WorkerCrossThreadDispatcher* aDispatcher);
|
|
||||||
|
|
||||||
void Send(UnixSocketRawData* aRawData);
|
void Send(UnixSocketRawData* aRawData);
|
||||||
void Close();
|
void Close();
|
||||||
|
|
||||||
// Methods for |StreamSocketConsumer|
|
nsresult Register(unsigned long aClientId,
|
||||||
|
WorkerCrossThreadDispatcher* aDispatcher);
|
||||||
|
void Unregister();
|
||||||
|
|
||||||
|
// Methods for |RilSocketConsumer|
|
||||||
//
|
//
|
||||||
|
|
||||||
void ReceiveSocketData(int aIndex,
|
void ReceiveSocketData(JSContext* aCx,
|
||||||
|
int aIndex,
|
||||||
nsAutoPtr<UnixSocketBuffer>& aBuffer) override;
|
nsAutoPtr<UnixSocketBuffer>& aBuffer) override;
|
||||||
void OnConnectSuccess(int aIndex) override;
|
void OnConnectSuccess(int aIndex) override;
|
||||||
void OnConnectError(int aIndex) override;
|
void OnConnectError(int aIndex) override;
|
||||||
void OnDisconnect(int aIndex) override;
|
void OnDisconnect(int aIndex) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
nsRefPtr<StreamSocket> mSocket;
|
nsRefPtr<RilSocket> mSocket;
|
||||||
nsRefPtr<mozilla::dom::workers::WorkerCrossThreadDispatcher> mDispatcher;
|
|
||||||
nsCString mAddress;
|
nsCString mAddress;
|
||||||
bool mShutdown;
|
bool mShutdown;
|
||||||
};
|
};
|
||||||
@ -77,39 +77,10 @@ public:
|
|||||||
bool RunTask(JSContext* aCx) override;
|
bool RunTask(JSContext* aCx) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class SendRilSocketDataTask final : public nsRunnable
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
SendRilSocketDataTask(unsigned long aClientId,
|
|
||||||
UnixSocketRawData* aRawData)
|
|
||||||
: mRawData(aRawData)
|
|
||||||
, mClientId(aClientId)
|
|
||||||
{ }
|
|
||||||
|
|
||||||
NS_IMETHOD Run() override
|
|
||||||
{
|
|
||||||
MOZ_ASSERT(NS_IsMainThread());
|
|
||||||
|
|
||||||
if (sRilConsumers.Length() <= mClientId || !sRilConsumers[mClientId]) {
|
|
||||||
// Probably shutting down.
|
|
||||||
delete mRawData;
|
|
||||||
return NS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
sRilConsumers[mClientId]->Send(mRawData);
|
|
||||||
return NS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
UnixSocketRawData* mRawData;
|
|
||||||
unsigned long mClientId;
|
|
||||||
};
|
|
||||||
|
|
||||||
static bool
|
static bool
|
||||||
PostToRIL(JSContext* aCx, unsigned aArgc, JS::Value* aVp)
|
PostToRIL(JSContext* aCx, unsigned aArgc, JS::Value* aVp)
|
||||||
{
|
{
|
||||||
JS::CallArgs args = JS::CallArgsFromVp(aArgc, aVp);
|
JS::CallArgs args = JS::CallArgsFromVp(aArgc, aVp);
|
||||||
NS_ASSERTION(!NS_IsMainThread(), "Expecting to be on the worker thread");
|
|
||||||
|
|
||||||
if (args.length() != 2) {
|
if (args.length() != 2) {
|
||||||
JS_ReportError(aCx, "Expecting two arguments with the RIL message");
|
JS_ReportError(aCx, "Expecting two arguments with the RIL message");
|
||||||
@ -119,7 +90,7 @@ PostToRIL(JSContext* aCx, unsigned aArgc, JS::Value* aVp)
|
|||||||
int clientId = args[0].toInt32();
|
int clientId = args[0].toInt32();
|
||||||
JS::Value v = args[1];
|
JS::Value v = args[1];
|
||||||
|
|
||||||
UnixSocketRawData* raw = nullptr;
|
nsAutoPtr<UnixSocketRawData> raw;
|
||||||
|
|
||||||
if (v.isString()) {
|
if (v.isString()) {
|
||||||
JSAutoByteString abs;
|
JSAutoByteString abs;
|
||||||
@ -159,9 +130,13 @@ PostToRIL(JSContext* aCx, unsigned aArgc, JS::Value* aVp)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
nsRefPtr<SendRilSocketDataTask> task = new SendRilSocketDataTask(clientId,
|
if ((ssize_t)sRilConsumers.Length() <= clientId || !sRilConsumers[clientId]) {
|
||||||
raw);
|
// Probably shutting down.
|
||||||
NS_DispatchToMainThread(task);
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
sRilConsumers[clientId]->Send(raw.forget());
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -170,7 +145,7 @@ ConnectWorkerToRIL::RunTask(JSContext* aCx)
|
|||||||
{
|
{
|
||||||
// Set up the postRILMessage on the function for worker -> RIL thread
|
// Set up the postRILMessage on the function for worker -> RIL thread
|
||||||
// communication.
|
// communication.
|
||||||
NS_ASSERTION(!NS_IsMainThread(), "Expecting to be on the worker thread");
|
//NS_ASSERTION(!NS_IsMainThread(), "Expecting to be on the worker thread");
|
||||||
NS_ASSERTION(!JS_IsRunning(aCx), "Are we being called somehow?");
|
NS_ASSERTION(!JS_IsRunning(aCx), "Are we being called somehow?");
|
||||||
JS::Rooted<JSObject*> workerGlobal(aCx, JS::CurrentGlobalOrNull(aCx));
|
JS::Rooted<JSObject*> workerGlobal(aCx, JS::CurrentGlobalOrNull(aCx));
|
||||||
|
|
||||||
@ -191,49 +166,13 @@ ConnectWorkerToRIL::RunTask(JSContext* aCx)
|
|||||||
PostToRIL, 2, 0);
|
PostToRIL, 2, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
class DispatchRILEvent final : public WorkerTask
|
RilConsumer::RilConsumer()
|
||||||
{
|
: mShutdown(false)
|
||||||
public:
|
{ }
|
||||||
DispatchRILEvent(unsigned long aClient, UnixSocketBuffer* aBuffer)
|
|
||||||
: mClientId(aClient)
|
|
||||||
, mBuffer(aBuffer)
|
|
||||||
{ }
|
|
||||||
|
|
||||||
bool RunTask(JSContext* aCx) override;
|
nsresult
|
||||||
|
RilConsumer::Register(unsigned long aClientId,
|
||||||
private:
|
WorkerCrossThreadDispatcher* aDispatcher)
|
||||||
unsigned long mClientId;
|
|
||||||
nsAutoPtr<UnixSocketBuffer> mBuffer;
|
|
||||||
};
|
|
||||||
|
|
||||||
bool
|
|
||||||
DispatchRILEvent::RunTask(JSContext* aCx)
|
|
||||||
{
|
|
||||||
JS::Rooted<JSObject*> obj(aCx, JS::CurrentGlobalOrNull(aCx));
|
|
||||||
|
|
||||||
JS::Rooted<JSObject*> array(aCx,
|
|
||||||
JS_NewUint8Array(aCx, mBuffer->GetSize()));
|
|
||||||
if (!array) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
{
|
|
||||||
JS::AutoCheckCannotGC nogc;
|
|
||||||
memcpy(JS_GetArrayBufferViewData(array, nogc),
|
|
||||||
mBuffer->GetData(), mBuffer->GetSize());
|
|
||||||
}
|
|
||||||
|
|
||||||
JS::AutoValueArray<2> args(aCx);
|
|
||||||
args[0].setNumber((uint32_t)mClientId);
|
|
||||||
args[1].setObject(*array);
|
|
||||||
|
|
||||||
JS::Rooted<JS::Value> rval(aCx);
|
|
||||||
return JS_CallFunctionName(aCx, obj, "onRILMessage", args, &rval);
|
|
||||||
}
|
|
||||||
|
|
||||||
RilConsumer::RilConsumer(unsigned long aClientId,
|
|
||||||
WorkerCrossThreadDispatcher* aDispatcher)
|
|
||||||
: mDispatcher(aDispatcher)
|
|
||||||
, mShutdown(false)
|
|
||||||
{
|
{
|
||||||
// Only append client id after RIL_SOCKET_NAME when it's not connected to
|
// Only append client id after RIL_SOCKET_NAME when it's not connected to
|
||||||
// the first(0) rilproxy for compatibility.
|
// the first(0) rilproxy for compatibility.
|
||||||
@ -246,8 +185,21 @@ RilConsumer::RilConsumer(unsigned long aClientId,
|
|||||||
mAddress = addr_un.sun_path;
|
mAddress = addr_un.sun_path;
|
||||||
}
|
}
|
||||||
|
|
||||||
mSocket = new StreamSocket(this, aClientId);
|
mSocket = new RilSocket(aDispatcher, this, aClientId);
|
||||||
mSocket->Connect(new RilConnector(mAddress, aClientId));
|
|
||||||
|
nsresult rv = mSocket->Connect(new RilConnector(mAddress, aClientId));
|
||||||
|
if (NS_FAILED(rv)) {
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
RilConsumer::Unregister()
|
||||||
|
{
|
||||||
|
mShutdown = true;
|
||||||
|
Close();
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -270,16 +222,31 @@ RilConsumer::Close()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// |StreamSocketConnector|
|
// |RilSocketConnector|
|
||||||
|
|
||||||
void
|
void
|
||||||
RilConsumer::ReceiveSocketData(int aIndex,
|
RilConsumer::ReceiveSocketData(JSContext* aCx,
|
||||||
|
int aIndex,
|
||||||
nsAutoPtr<UnixSocketBuffer>& aBuffer)
|
nsAutoPtr<UnixSocketBuffer>& aBuffer)
|
||||||
{
|
{
|
||||||
MOZ_ASSERT(NS_IsMainThread());
|
JS::Rooted<JSObject*> obj(aCx, JS::CurrentGlobalOrNull(aCx));
|
||||||
|
|
||||||
nsRefPtr<DispatchRILEvent> dre(new DispatchRILEvent(aIndex, aBuffer.forget()));
|
JS::Rooted<JSObject*> array(aCx, JS_NewUint8Array(aCx, aBuffer->GetSize()));
|
||||||
mDispatcher->PostTask(dre);
|
if (!array) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
{
|
||||||
|
JS::AutoCheckCannotGC nogc;
|
||||||
|
memcpy(JS_GetArrayBufferViewData(array, nogc),
|
||||||
|
aBuffer->GetData(), aBuffer->GetSize());
|
||||||
|
}
|
||||||
|
|
||||||
|
JS::AutoValueArray<2> args(aCx);
|
||||||
|
args[0].setNumber((uint32_t)aIndex);
|
||||||
|
args[1].setObject(*array);
|
||||||
|
|
||||||
|
JS::Rooted<JS::Value> rval(aCx);
|
||||||
|
JS_CallFunctionName(aCx, obj, "onRILMessage", args, &rval);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -357,6 +324,39 @@ RilWorker::RilWorker(WorkerCrossThreadDispatcher* aDispatcher)
|
|||||||
MOZ_ASSERT(mDispatcher);
|
MOZ_ASSERT(mDispatcher);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class RilWorker::RegisterConsumerTask : public WorkerTask
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
RegisterConsumerTask(unsigned int aClientId,
|
||||||
|
WorkerCrossThreadDispatcher* aDispatcher)
|
||||||
|
: mClientId(aClientId)
|
||||||
|
, mDispatcher(aDispatcher)
|
||||||
|
{
|
||||||
|
MOZ_ASSERT(mDispatcher);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool RunTask(JSContext* aCx) override
|
||||||
|
{
|
||||||
|
sRilConsumers.EnsureLengthAtLeast(mClientId + 1);
|
||||||
|
|
||||||
|
MOZ_ASSERT(!sRilConsumers[mClientId]);
|
||||||
|
|
||||||
|
nsAutoPtr<RilConsumer> rilConsumer(new RilConsumer());
|
||||||
|
|
||||||
|
nsresult rv = rilConsumer->Register(mClientId, mDispatcher);
|
||||||
|
if (NS_FAILED(rv)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
sRilConsumers[mClientId] = rilConsumer;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
unsigned int mClientId;
|
||||||
|
nsRefPtr<WorkerCrossThreadDispatcher> mDispatcher;
|
||||||
|
};
|
||||||
|
|
||||||
nsresult
|
nsresult
|
||||||
RilWorker::RegisterConsumer(unsigned int aClientId)
|
RilWorker::RegisterConsumer(unsigned int aClientId)
|
||||||
{
|
{
|
||||||
@ -366,24 +366,48 @@ RilWorker::RegisterConsumer(unsigned int aClientId)
|
|||||||
return NS_ERROR_UNEXPECTED;
|
return NS_ERROR_UNEXPECTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
sRilConsumers.EnsureLengthAtLeast(aClientId + 1);
|
nsRefPtr<RegisterConsumerTask> task = new RegisterConsumerTask(aClientId,
|
||||||
|
mDispatcher);
|
||||||
MOZ_ASSERT(!sRilConsumers[aClientId]);
|
if (!mDispatcher->PostTask(task)) {
|
||||||
|
NS_WARNING("Failed to post register-consumer task.");
|
||||||
sRilConsumers[aClientId] = new RilConsumer(aClientId, mDispatcher);
|
return NS_ERROR_UNEXPECTED;
|
||||||
|
}
|
||||||
|
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class RilWorker::UnregisterConsumerTask : public WorkerTask
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
UnregisterConsumerTask(unsigned int aClientId)
|
||||||
|
: mClientId(aClientId)
|
||||||
|
{ }
|
||||||
|
|
||||||
|
bool RunTask(JSContext* aCx) override
|
||||||
|
{
|
||||||
|
MOZ_ASSERT(mClientId < sRilConsumers.Length());
|
||||||
|
MOZ_ASSERT(sRilConsumers[mClientId]);
|
||||||
|
|
||||||
|
sRilConsumers[mClientId]->Unregister();
|
||||||
|
sRilConsumers[mClientId] = nullptr;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
unsigned int mClientId;
|
||||||
|
};
|
||||||
|
|
||||||
void
|
void
|
||||||
RilWorker::UnregisterConsumer(unsigned int aClientId)
|
RilWorker::UnregisterConsumer(unsigned int aClientId)
|
||||||
{
|
{
|
||||||
MOZ_ASSERT(aClientId < sRilConsumers.Length());
|
nsRefPtr<UnregisterConsumerTask> task =
|
||||||
MOZ_ASSERT(sRilConsumers[aClientId]);
|
new UnregisterConsumerTask(aClientId);
|
||||||
|
|
||||||
sRilConsumers[aClientId]->mShutdown = true;
|
if (!mDispatcher->PostTask(task)) {
|
||||||
sRilConsumers[aClientId]->Close();
|
NS_WARNING("Failed to post unregister-consumer task.");
|
||||||
sRilConsumers[aClientId] = nullptr;
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace ipc
|
} // namespace ipc
|
||||||
|
@ -35,6 +35,9 @@ public:
|
|||||||
static void Shutdown();
|
static void Shutdown();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
class RegisterConsumerTask;
|
||||||
|
class UnregisterConsumerTask;
|
||||||
|
|
||||||
RilWorker(mozilla::dom::workers::WorkerCrossThreadDispatcher* aDispatcher);
|
RilWorker(mozilla::dom::workers::WorkerCrossThreadDispatcher* aDispatcher);
|
||||||
|
|
||||||
nsresult RegisterConsumer(unsigned int aClientId);
|
nsresult RegisterConsumer(unsigned int aClientId);
|
||||||
|
Loading…
Reference in New Issue
Block a user