mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-24 13:21:05 +00:00
5c12ebb9b7
If we get a connection error, mSocket can end up being null in RilConsumer::Close().
321 lines
7.9 KiB
C++
321 lines
7.9 KiB
C++
/* -*- 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 "mozilla/ipc/Ril.h"
|
|
|
|
#include <fcntl.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/un.h>
|
|
#include <netdb.h> // For gethostbyname.
|
|
|
|
#undef CHROMIUM_LOG
|
|
#if defined(MOZ_WIDGET_GONK)
|
|
#include <android/log.h>
|
|
#define CHROMIUM_LOG(args...) __android_log_print(ANDROID_LOG_INFO, "Gonk", args)
|
|
#else
|
|
#define CHROMIUM_LOG(args...) printf(args);
|
|
#endif
|
|
|
|
#include "jsfriendapi.h"
|
|
#include "mozilla/ArrayUtils.h"
|
|
#include "nsTArray.h"
|
|
#include "nsThreadUtils.h" // For NS_IsMainThread.
|
|
#include "RilConnector.h"
|
|
|
|
USING_WORKERS_NAMESPACE
|
|
using namespace mozilla::ipc;
|
|
|
|
namespace {
|
|
|
|
static const char RIL_SOCKET_NAME[] = "/dev/socket/rilproxy";
|
|
|
|
static nsTArray<nsAutoPtr<mozilla::ipc::RilConsumer>> sRilConsumers;
|
|
|
|
class ConnectWorkerToRIL final : public WorkerTask
|
|
{
|
|
public:
|
|
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
|
|
PostToRIL(JSContext* aCx, unsigned aArgc, JS::Value* aVp)
|
|
{
|
|
JS::CallArgs args = JS::CallArgsFromVp(aArgc, aVp);
|
|
NS_ASSERTION(!NS_IsMainThread(), "Expecting to be on the worker thread");
|
|
|
|
if (args.length() != 2) {
|
|
JS_ReportError(aCx, "Expecting two arguments with the RIL message");
|
|
return false;
|
|
}
|
|
|
|
int clientId = args[0].toInt32();
|
|
JS::Value v = args[1];
|
|
|
|
UnixSocketRawData* raw = nullptr;
|
|
|
|
if (v.isString()) {
|
|
JSAutoByteString abs;
|
|
JS::Rooted<JSString*> str(aCx, v.toString());
|
|
if (!abs.encodeUtf8(aCx, str)) {
|
|
return false;
|
|
}
|
|
|
|
raw = new UnixSocketRawData(abs.ptr(), abs.length());
|
|
} else if (!v.isPrimitive()) {
|
|
JSObject* obj = v.toObjectOrNull();
|
|
if (!JS_IsTypedArrayObject(obj)) {
|
|
JS_ReportError(aCx, "Object passed in wasn't a typed array");
|
|
return false;
|
|
}
|
|
|
|
uint32_t type = JS_GetArrayBufferViewType(obj);
|
|
if (type != js::Scalar::Int8 &&
|
|
type != js::Scalar::Uint8 &&
|
|
type != js::Scalar::Uint8Clamped) {
|
|
JS_ReportError(aCx, "Typed array data is not octets");
|
|
return false;
|
|
}
|
|
|
|
JS::AutoCheckCannotGC nogc;
|
|
size_t size = JS_GetTypedArrayByteLength(obj);
|
|
void* data = JS_GetArrayBufferViewData(obj, nogc);
|
|
raw = new UnixSocketRawData(data, size);
|
|
} else {
|
|
JS_ReportError(
|
|
aCx, "Incorrect argument. Expecting a string or a typed array");
|
|
return false;
|
|
}
|
|
|
|
if (!raw) {
|
|
JS_ReportError(aCx, "Unable to post to RIL");
|
|
return false;
|
|
}
|
|
|
|
nsRefPtr<SendRilSocketDataTask> task = new SendRilSocketDataTask(clientId,
|
|
raw);
|
|
NS_DispatchToMainThread(task);
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
ConnectWorkerToRIL::RunTask(JSContext* aCx)
|
|
{
|
|
// Set up the postRILMessage on the function for worker -> RIL thread
|
|
// communication.
|
|
NS_ASSERTION(!NS_IsMainThread(), "Expecting to be on the worker thread");
|
|
NS_ASSERTION(!JS_IsRunning(aCx), "Are we being called somehow?");
|
|
JS::Rooted<JSObject*> workerGlobal(aCx, JS::CurrentGlobalOrNull(aCx));
|
|
|
|
// Check whether |postRILMessage| has been defined. No one but this class
|
|
// should ever define |postRILMessage| in a RIL worker.
|
|
JS::Rooted<JS::Value> val(aCx);
|
|
if (!JS_GetProperty(aCx, workerGlobal, "postRILMessage", &val)) {
|
|
JS_ReportPendingException(aCx);
|
|
return false;
|
|
}
|
|
|
|
// Make sure that |postRILMessage| is a function.
|
|
if (JSTYPE_FUNCTION == JS_TypeOfValue(aCx, val)) {
|
|
return true;
|
|
}
|
|
|
|
return !!JS_DefineFunction(aCx, workerGlobal, "postRILMessage",
|
|
PostToRIL, 2, 0);
|
|
}
|
|
|
|
class DispatchRILEvent final : public WorkerTask
|
|
{
|
|
public:
|
|
DispatchRILEvent(unsigned long aClient, UnixSocketBuffer* aBuffer)
|
|
: mClientId(aClient)
|
|
, mBuffer(aBuffer)
|
|
{ }
|
|
|
|
bool RunTask(JSContext* aCx) override;
|
|
|
|
private:
|
|
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);
|
|
}
|
|
|
|
} // anonymous namespace
|
|
|
|
namespace mozilla {
|
|
namespace ipc {
|
|
|
|
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
|
|
// the first(0) rilproxy for compatibility.
|
|
if (!aClientId) {
|
|
mAddress = RIL_SOCKET_NAME;
|
|
} else {
|
|
struct sockaddr_un addr_un;
|
|
snprintf(addr_un.sun_path, sizeof addr_un.sun_path, "%s%lu",
|
|
RIL_SOCKET_NAME, aClientId);
|
|
mAddress = addr_un.sun_path;
|
|
}
|
|
|
|
mSocket = new StreamSocket(this, aClientId);
|
|
mSocket->Connect(new RilConnector(mAddress, aClientId));
|
|
}
|
|
|
|
nsresult
|
|
RilConsumer::Register(unsigned int aClientId,
|
|
WorkerCrossThreadDispatcher* aDispatcher)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
sRilConsumers.EnsureLengthAtLeast(aClientId + 1);
|
|
|
|
if (sRilConsumers[aClientId]) {
|
|
NS_WARNING("RilConsumer already registered");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsRefPtr<ConnectWorkerToRIL> connection = new ConnectWorkerToRIL();
|
|
if (!aDispatcher->PostTask(connection)) {
|
|
NS_WARNING("Failed to connect worker to ril");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
// Now that we're set up, connect ourselves to the RIL thread.
|
|
sRilConsumers[aClientId] = new RilConsumer(aClientId, aDispatcher);
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
RilConsumer::Shutdown()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
for (unsigned long i = 0; i < sRilConsumers.Length(); i++) {
|
|
nsAutoPtr<RilConsumer> instance(sRilConsumers[i]);
|
|
if (!instance) {
|
|
continue;
|
|
}
|
|
|
|
instance->mShutdown = true;
|
|
instance->Close();
|
|
instance = nullptr;
|
|
}
|
|
}
|
|
|
|
void
|
|
RilConsumer::Send(UnixSocketRawData* aRawData)
|
|
{
|
|
if (!mSocket || mSocket->GetConnectionStatus() != SOCKET_CONNECTED) {
|
|
// Probably shutting down.
|
|
delete aRawData;
|
|
return;
|
|
}
|
|
mSocket->SendSocketData(aRawData);
|
|
}
|
|
|
|
void
|
|
RilConsumer::Close()
|
|
{
|
|
if (mSocket) {
|
|
mSocket->Close();
|
|
mSocket = nullptr;
|
|
}
|
|
}
|
|
|
|
// |StreamSocketConnector|
|
|
|
|
void
|
|
RilConsumer::ReceiveSocketData(int aIndex,
|
|
nsAutoPtr<UnixSocketBuffer>& aBuffer)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
nsRefPtr<DispatchRILEvent> dre(new DispatchRILEvent(aIndex, aBuffer.forget()));
|
|
mDispatcher->PostTask(dre);
|
|
}
|
|
|
|
void
|
|
RilConsumer::OnConnectSuccess(int aIndex)
|
|
{
|
|
// Nothing to do here.
|
|
CHROMIUM_LOG("RIL[%d]: %s\n", aIndex, __FUNCTION__);
|
|
}
|
|
|
|
void
|
|
RilConsumer::OnConnectError(int aIndex)
|
|
{
|
|
CHROMIUM_LOG("RIL[%d]: %s\n", aIndex, __FUNCTION__);
|
|
Close();
|
|
}
|
|
|
|
void
|
|
RilConsumer::OnDisconnect(int aIndex)
|
|
{
|
|
CHROMIUM_LOG("RIL[%d]: %s\n", aIndex, __FUNCTION__);
|
|
if (mShutdown) {
|
|
return;
|
|
}
|
|
mSocket->Connect(new RilConnector(mAddress, aIndex),
|
|
mSocket->GetSuggestedConnectDelayMs());
|
|
}
|
|
|
|
} // namespace ipc
|
|
} // namespace mozilla
|