gecko-dev/netwerk/protocol/http/nsHttpActivityDistributor.cpp

199 строки
6.5 KiB
C++

/* 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/. */
// HttpLog.h should generally be included first
#include "HttpLog.h"
#include "mozilla/net/SocketProcessChild.h"
#include "mozilla/net/SocketProcessParent.h"
#include "nsHttpActivityDistributor.h"
#include "nsHttpHandler.h"
#include "nsCOMPtr.h"
#include "nsIOService.h"
#include "nsNetUtil.h"
#include "nsQueryObject.h"
#include "nsThreadUtils.h"
#include "NullHttpChannel.h"
namespace mozilla {
namespace net {
typedef nsMainThreadPtrHolder<nsIHttpActivityObserver> ObserverHolder;
typedef nsMainThreadPtrHandle<nsIHttpActivityObserver> ObserverHandle;
NS_IMPL_ISUPPORTS(nsHttpActivityDistributor, nsIHttpActivityDistributor,
nsIHttpActivityObserver)
nsHttpActivityDistributor::nsHttpActivityDistributor()
: mLock("nsHttpActivityDistributor.mLock"), mActivated(false) {}
NS_IMETHODIMP
nsHttpActivityDistributor::ObserveActivity(nsISupports* aHttpChannel,
uint32_t aActivityType,
uint32_t aActivitySubtype,
PRTime aTimestamp,
uint64_t aExtraSizeData,
const nsACString& aExtraStringData) {
MOZ_ASSERT(XRE_IsParentProcess() && NS_IsMainThread());
for (size_t i = 0; i < mObservers.Length(); i++) {
Unused << mObservers[i]->ObserveActivity(aHttpChannel, aActivityType,
aActivitySubtype, aTimestamp,
aExtraSizeData, aExtraStringData);
}
return NS_OK;
}
NS_IMETHODIMP
nsHttpActivityDistributor::ObserveActivityWithArgs(
const HttpActivityArgs& aArgs, uint32_t aActivityType,
uint32_t aActivitySubtype, PRTime aTimestamp, uint64_t aExtraSizeData,
const nsACString& aExtraStringData) {
HttpActivityArgs args(aArgs);
nsCString extraStringData(aExtraStringData);
if (XRE_IsSocketProcess()) {
auto task = [args{std::move(args)}, aActivityType, aActivitySubtype,
aTimestamp, aExtraSizeData,
extraStringData{std::move(extraStringData)}]() {
SocketProcessChild::GetSingleton()->SendObserveHttpActivity(
args, aActivityType, aActivitySubtype, aTimestamp, aExtraSizeData,
extraStringData);
};
if (!NS_IsMainThread()) {
return NS_DispatchToMainThread(NS_NewRunnableFunction(
"net::nsHttpActivityDistributor::ObserveActivityWithArgs", task));
}
task();
return NS_OK;
}
MOZ_ASSERT(XRE_IsParentProcess());
RefPtr<nsHttpActivityDistributor> self = this;
auto task = [args{std::move(args)}, aActivityType, aActivitySubtype,
aTimestamp, aExtraSizeData,
extraStringData{std::move(extraStringData)},
self{std::move(self)}]() {
if (args.type() == HttpActivityArgs::Tuint64_t) {
nsWeakPtr weakPtr = gHttpHandler->GetWeakHttpChannel(args.get_uint64_t());
if (nsCOMPtr<nsIHttpChannel> channel = do_QueryReferent(weakPtr)) {
Unused << self->ObserveActivity(channel, aActivityType,
aActivitySubtype, aTimestamp,
aExtraSizeData, extraStringData);
}
} else if (args.type() == HttpActivityArgs::THttpActivity) {
nsCOMPtr<nsIURI> uri;
nsAutoCString portStr(""_ns);
int32_t port = args.get_HttpActivity().port();
bool endToEndSSL = args.get_HttpActivity().endToEndSSL();
if (port != -1 &&
((endToEndSSL && port != 443) || (!endToEndSSL && port != 80))) {
portStr.AppendInt(port);
}
nsresult rv = NS_NewURI(getter_AddRefs(uri),
(endToEndSSL ? "https://"_ns : "http://"_ns) +
args.get_HttpActivity().host() + portStr);
if (NS_FAILED(rv)) {
return;
}
RefPtr<NullHttpChannel> channel = new NullHttpChannel();
rv = channel->Init(uri, 0, nullptr, 0, nullptr);
MOZ_ASSERT(NS_SUCCEEDED(rv));
Unused << self->ObserveActivity(
nsCOMPtr<nsISupports>(do_QueryObject(channel)), aActivityType,
aActivitySubtype, aTimestamp, aExtraSizeData, extraStringData);
}
};
if (!NS_IsMainThread()) {
return NS_DispatchToMainThread(NS_NewRunnableFunction(
"net::nsHttpActivityDistributor::ObserveActivityWithArgs", task));
}
task();
return NS_OK;
}
NS_IMETHODIMP
nsHttpActivityDistributor::GetIsActive(bool* isActive) {
NS_ENSURE_ARG_POINTER(isActive);
MutexAutoLock lock(mLock);
if (XRE_IsSocketProcess()) {
*isActive = mActivated;
return NS_OK;
}
*isActive = !!mObservers.Length();
return NS_OK;
}
NS_IMETHODIMP nsHttpActivityDistributor::SetIsActive(bool aActived) {
MOZ_RELEASE_ASSERT(XRE_IsSocketProcess());
mActivated = aActived;
return NS_OK;
}
NS_IMETHODIMP
nsHttpActivityDistributor::AddObserver(nsIHttpActivityObserver* aObserver) {
MOZ_ASSERT(XRE_IsParentProcess());
ObserverHandle observer(
new ObserverHolder("nsIHttpActivityObserver", aObserver));
bool wasEmpty = false;
{
MutexAutoLock lock(mLock);
wasEmpty = mObservers.IsEmpty();
// XXX(Bug 1631371) Check if this should use a fallible operation as it
// pretended earlier.
mObservers.AppendElement(observer);
}
if (nsIOService::UseSocketProcess() && wasEmpty) {
auto task = []() {
SocketProcessParent* parent = SocketProcessParent::GetSingleton();
if (parent && parent->CanSend()) {
Unused << parent->SendOnHttpActivityDistributorActivated(true);
}
};
gIOService->CallOrWaitForSocketProcess(task);
}
return NS_OK;
}
NS_IMETHODIMP
nsHttpActivityDistributor::RemoveObserver(nsIHttpActivityObserver* aObserver) {
MOZ_ASSERT(XRE_IsParentProcess());
ObserverHandle observer(
new ObserverHolder("nsIHttpActivityObserver", aObserver));
bool isEmpty = false;
{
MutexAutoLock lock(mLock);
if (!mObservers.RemoveElement(observer)) return NS_ERROR_FAILURE;
isEmpty = mObservers.IsEmpty();
}
if (nsIOService::UseSocketProcess() && isEmpty) {
auto task = []() {
SocketProcessParent* parent = SocketProcessParent::GetSingleton();
if (parent && parent->CanSend()) {
Unused << parent->SendOnHttpActivityDistributorActivated(false);
}
};
gIOService->CallOrWaitForSocketProcess(task);
}
return NS_OK;
}
} // namespace net
} // namespace mozilla