mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
81d61fa324
- Bug 1265036 - Use NS_ABORT_OOM() if try_realloc() fails. r=billm (c30f4f83d5) - Bug 1263292 - Handle calling realloc(0) (r=jld) (f292859ee9) - Bug 1256366 - Remove linear and exponential stats collection from histogram.cc. r=gfritzsche (f9a1c869a1) - Bug 1263953 - Reduce the growth rate of Pickle. r=wmccloskey (6eb5228490) - Bug 1233275 - Copy environment for IPC using NSPR. r=jld (2004db748e) - Bug 1261094 - Improve how MessageChannel::mInterruptStack is used in IPC code, r=jld (56e2c114a4) - Bug 1246931: Include dbus.h in DBus IPC headers, r=shuang (43e797c2d8) - Bug 1264887: Make DBus helpers available on desktop builds, r=shuang (58bff1f640) - Bug 1268130, part 1 - Reimplement ByteLengthIsValid using CheckedInt. r=froydnj (6018e22ae0) - Bug 1268130, part 2 - Make ByteLengthIsValid failures fatal in release builds. r=froydnj (f9d934a498) - Bug 1269365, part 1 - Swap fallible and infallible TArray ParamTraits. r=froydnj (ad423bc04d) - Bug 1269365, part 2 - Make ParamTraits<nsTArray<E>>::Read use infallible allocation. r=froydnj (9b902a5bc4) - Bug 1269365, part 3 - Use infallible array allocation in implementSpecialArrayPickling. r=froydnj (592fe648d3) - Bug 1264820 - Measure IPC reply size in telemetry (r=mccr8) (62c54d3141) - Bug 1268938 - Use the name of the original message in Send for reply telemetry. r=billm (a2de5c6a91) - Bug 1266954: Remove temporary |ScopedClose| from PDU receive code, r=jacheng (cb06315c33) - Bug 1142109 - Fix IPDL tests (r=dvander) (df3f0cda32) - Bug 1177013 - Fix IPDL tests for not allowing CPOWs during sync (r=dvander) (5da0a8a4c9) - Bug 1261307: Convert RIL sockets to |UniquePtr|, r=nfroyd (08609783b3) - Bug 1253622 - Move the mozilla-trace.h generation into moz.build; r=ted (f01dc418bc) - Bug 1267318 ignore cert expiration for mozilla-signed packages, r=dkeeler (7a1ddd6090) - Bug 1029173 - Clean up nsDataSignatureVerifier. r=keeler (f9602341ea) - bug 1267463 - add a more nuanced subject common name fallback option for prerelease channels r=Cykesiopka,jcj (9b55320c9b) - Bug 1253108 - Enable ESLint "strict" rule for PSM. r=keeler (54802bdc38) - Bug 1255425 - part 1 - clearly delineate steps when outputting HSTS preload list; r=keeler (79f73189c8) - Bug 1251801 - Fully implement nsNSSShutDownObject and obviate manual NSS resource management. r=keeler (af32315d3f) - Bug 1251801 - Improve handling of PK11_* function error codes. r=keeler (9f2c8ac64b) - Fix unified-build bustage from bug 1264706. r=bustage (11bc0417c7) - Bug 1265164 - Always use nsCOMPtrs with getNSSDialogs(). r=keeler (ce5a703972)
450 lines
10 KiB
C++
450 lines
10 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 "jsfriendapi.h"
|
|
#include "mozilla/ArrayUtils.h"
|
|
#include "mozilla/dom/ScriptSettings.h"
|
|
#include "mozilla/dom/workers/Workers.h"
|
|
#include "mozilla/ipc/RilSocket.h"
|
|
#include "mozilla/ipc/RilSocketConsumer.h"
|
|
#include "nsThreadUtils.h" // For NS_IsMainThread.
|
|
#include "RilConnector.h"
|
|
|
|
#ifdef CHROMIUM_LOG
|
|
#undef CHROMIUM_LOG
|
|
#endif
|
|
|
|
#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
|
|
|
|
namespace mozilla {
|
|
namespace ipc {
|
|
|
|
USING_WORKERS_NAMESPACE;
|
|
using namespace JS;
|
|
|
|
class RilConsumer;
|
|
|
|
static const char RIL_SOCKET_NAME[] = "/dev/socket/rilproxy";
|
|
|
|
static nsTArray<UniquePtr<RilConsumer>> sRilConsumers;
|
|
|
|
//
|
|
// RilConsumer
|
|
//
|
|
|
|
class RilConsumer final : public RilSocketConsumer
|
|
{
|
|
public:
|
|
RilConsumer();
|
|
|
|
nsresult ConnectWorkerToRIL(JSContext* aCx);
|
|
|
|
nsresult Register(unsigned long aClientId,
|
|
WorkerCrossThreadDispatcher* aDispatcher);
|
|
void Unregister();
|
|
|
|
// Methods for |RilSocketConsumer|
|
|
//
|
|
|
|
void ReceiveSocketData(JSContext* aCx,
|
|
int aIndex,
|
|
UniquePtr<UnixSocketBuffer>& aBuffer) override;
|
|
void OnConnectSuccess(int aIndex) override;
|
|
void OnConnectError(int aIndex) override;
|
|
void OnDisconnect(int aIndex) override;
|
|
|
|
protected:
|
|
static bool PostRILMessage(JSContext* aCx, unsigned aArgc, Value* aVp);
|
|
|
|
nsresult Send(JSContext* aCx, const CallArgs& aArgs);
|
|
nsresult Receive(JSContext* aCx,
|
|
uint32_t aClientId,
|
|
const UnixSocketBuffer* aBuffer);
|
|
void Close();
|
|
|
|
private:
|
|
RefPtr<RilSocket> mSocket;
|
|
nsCString mAddress;
|
|
bool mShutdown;
|
|
};
|
|
|
|
RilConsumer::RilConsumer()
|
|
: mShutdown(false)
|
|
{ }
|
|
|
|
nsresult
|
|
RilConsumer::ConnectWorkerToRIL(JSContext* aCx)
|
|
{
|
|
// Set up the postRILMessage on the function for worker -> RIL thread
|
|
// communication.
|
|
Rooted<JSObject*> workerGlobal(aCx, CurrentGlobalOrNull(aCx));
|
|
|
|
// Check whether |postRILMessage| has been defined. No one but this class
|
|
// should ever define |postRILMessage| in a RIL worker.
|
|
Rooted<Value> val(aCx);
|
|
if (!JS_GetProperty(aCx, workerGlobal, "postRILMessage", &val)) {
|
|
JS_ReportPendingException(aCx);
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
// Make sure that |postRILMessage| is a function.
|
|
if (JSTYPE_FUNCTION == JS_TypeOfValue(aCx, val)) {
|
|
return NS_OK;
|
|
}
|
|
|
|
JSFunction* postRILMessage = JS_DefineFunction(aCx, workerGlobal,
|
|
"postRILMessage",
|
|
PostRILMessage, 2, 0);
|
|
if (NS_WARN_IF(!postRILMessage)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
RilConsumer::Register(unsigned long aClientId,
|
|
WorkerCrossThreadDispatcher* aDispatcher)
|
|
{
|
|
// 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 RilSocket(aDispatcher, this, aClientId);
|
|
|
|
nsresult rv = mSocket->Connect(new RilConnector(mAddress, aClientId));
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
RilConsumer::Unregister()
|
|
{
|
|
mShutdown = true;
|
|
Close();
|
|
}
|
|
|
|
bool
|
|
RilConsumer::PostRILMessage(JSContext* aCx, unsigned aArgc, Value* aVp)
|
|
{
|
|
CallArgs args = CallArgsFromVp(aArgc, aVp);
|
|
|
|
if (args.length() != 2) {
|
|
JS_ReportError(aCx, "Expecting two arguments with the RIL message");
|
|
return false;
|
|
}
|
|
|
|
int clientId = args[0].toInt32();
|
|
|
|
if ((ssize_t)sRilConsumers.Length() <= clientId || !sRilConsumers[clientId]) {
|
|
// Probably shutting down.
|
|
return true;
|
|
}
|
|
|
|
nsresult rv = sRilConsumers[clientId]->Send(aCx, args);
|
|
if (NS_FAILED(rv)) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
nsresult
|
|
RilConsumer::Send(JSContext* aCx, const CallArgs& aArgs)
|
|
{
|
|
if (NS_WARN_IF(!mSocket) ||
|
|
NS_WARN_IF(mSocket->GetConnectionStatus() == SOCKET_DISCONNECTED)) {
|
|
// Probably shutting down.
|
|
return NS_OK;
|
|
}
|
|
|
|
UniquePtr<UnixSocketRawData> raw;
|
|
|
|
Value v = aArgs[1];
|
|
|
|
if (v.isString()) {
|
|
JSAutoByteString abs;
|
|
Rooted<JSString*> str(aCx, v.toString());
|
|
if (!abs.encodeUtf8(aCx, str)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
raw = MakeUnique<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 NS_ERROR_FAILURE;
|
|
}
|
|
|
|
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 NS_ERROR_FAILURE;
|
|
}
|
|
|
|
size_t size = JS_GetTypedArrayByteLength(obj);
|
|
bool isShared;
|
|
void* data;
|
|
{
|
|
AutoCheckCannotGC nogc;
|
|
data = JS_GetArrayBufferViewData(obj, &isShared, nogc);
|
|
}
|
|
if (isShared) {
|
|
JS_ReportError(
|
|
aCx, "Incorrect argument. Shared memory not supported");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
raw = MakeUnique<UnixSocketRawData>(data, size);
|
|
} else {
|
|
JS_ReportError(
|
|
aCx, "Incorrect argument. Expecting a string or a typed array");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
if (!raw) {
|
|
JS_ReportError(aCx, "Unable to post to RIL");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
mSocket->SendSocketData(raw.release());
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
RilConsumer::Receive(JSContext* aCx,
|
|
uint32_t aClientId,
|
|
const UnixSocketBuffer* aBuffer)
|
|
{
|
|
MOZ_ASSERT(aBuffer);
|
|
|
|
Rooted<JSObject*> obj(aCx, CurrentGlobalOrNull(aCx));
|
|
|
|
Rooted<JSObject*> array(aCx, JS_NewUint8Array(aCx, aBuffer->GetSize()));
|
|
if (NS_WARN_IF(!array)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
{
|
|
AutoCheckCannotGC nogc;
|
|
bool isShared;
|
|
memcpy(JS_GetArrayBufferViewData(array, &isShared, nogc),
|
|
aBuffer->GetData(), aBuffer->GetSize());
|
|
MOZ_ASSERT(!isShared); // Array was constructed above.
|
|
}
|
|
|
|
AutoValueArray<2> args(aCx);
|
|
args[0].setNumber(aClientId);
|
|
args[1].setObject(*array);
|
|
|
|
Rooted<Value> rval(aCx);
|
|
JS_CallFunctionName(aCx, obj, "onRILMessage", args, &rval);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
RilConsumer::Close()
|
|
{
|
|
if (mSocket) {
|
|
mSocket->Close();
|
|
mSocket = nullptr;
|
|
}
|
|
}
|
|
|
|
// |RilSocketConnector|
|
|
|
|
void
|
|
RilConsumer::ReceiveSocketData(JSContext* aCx,
|
|
int aIndex,
|
|
UniquePtr<UnixSocketBuffer>& aBuffer)
|
|
{
|
|
Receive(aCx, (uint32_t)aIndex, aBuffer.get());
|
|
}
|
|
|
|
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());
|
|
}
|
|
|
|
//
|
|
// RilWorker
|
|
//
|
|
|
|
nsTArray<UniquePtr<RilWorker>> RilWorker::sRilWorkers;
|
|
|
|
nsresult
|
|
RilWorker::Register(unsigned int aClientId,
|
|
WorkerCrossThreadDispatcher* aDispatcher)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
sRilWorkers.EnsureLengthAtLeast(aClientId + 1);
|
|
|
|
if (sRilWorkers[aClientId]) {
|
|
NS_WARNING("RilWorkers already registered");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
// Now that we're set up, connect ourselves to the RIL thread.
|
|
sRilWorkers[aClientId] = MakeUnique<RilWorker>(aDispatcher);
|
|
|
|
nsresult rv = sRilWorkers[aClientId]->RegisterConsumer(aClientId);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
RilWorker::Shutdown()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
for (size_t i = 0; i < sRilWorkers.Length(); ++i) {
|
|
if (!sRilWorkers[i]) {
|
|
continue;
|
|
}
|
|
sRilWorkers[i]->UnregisterConsumer(i);
|
|
sRilWorkers[i] = nullptr;
|
|
}
|
|
}
|
|
|
|
RilWorker::RilWorker(WorkerCrossThreadDispatcher* aDispatcher)
|
|
: mDispatcher(aDispatcher)
|
|
{
|
|
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]);
|
|
|
|
auto rilConsumer = MakeUnique<RilConsumer>();
|
|
|
|
nsresult rv = rilConsumer->ConnectWorkerToRIL(aCx);
|
|
if (NS_FAILED(rv)) {
|
|
return false;
|
|
}
|
|
|
|
rv = rilConsumer->Register(mClientId, mDispatcher);
|
|
if (NS_FAILED(rv)) {
|
|
return false;
|
|
}
|
|
sRilConsumers[mClientId] = Move(rilConsumer);
|
|
|
|
return true;
|
|
}
|
|
|
|
private:
|
|
unsigned int mClientId;
|
|
RefPtr<WorkerCrossThreadDispatcher> mDispatcher;
|
|
};
|
|
|
|
nsresult
|
|
RilWorker::RegisterConsumer(unsigned int aClientId)
|
|
{
|
|
RefPtr<RegisterConsumerTask> task = new RegisterConsumerTask(aClientId,
|
|
mDispatcher);
|
|
if (!mDispatcher->PostTask(task)) {
|
|
NS_WARNING("Failed to post register-consumer task.");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
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
|
|
RilWorker::UnregisterConsumer(unsigned int aClientId)
|
|
{
|
|
RefPtr<UnregisterConsumerTask> task =
|
|
new UnregisterConsumerTask(aClientId);
|
|
|
|
if (!mDispatcher->PostTask(task)) {
|
|
NS_WARNING("Failed to post unregister-consumer task.");
|
|
return;
|
|
}
|
|
}
|
|
|
|
} // namespace ipc
|
|
} // namespace mozilla
|