Mirror of roytam1's Pale Moon fork just in case Moonchild and Tobin decide to go after him
https://github.com/roytam1/palemoon27
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1016 lines
29 KiB
1016 lines
29 KiB
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ |
|
/* vim: set ts=8 sts=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 "nsPerformance.h" |
|
#include "nsCOMPtr.h" |
|
#include "nsIHttpChannel.h" |
|
#include "nsITimedChannel.h" |
|
#include "nsDOMNavigationTiming.h" |
|
#include "nsContentUtils.h" |
|
#include "nsIScriptSecurityManager.h" |
|
#include "nsIDOMWindow.h" |
|
#include "nsILoadInfo.h" |
|
#include "nsIURI.h" |
|
#include "nsThreadUtils.h" |
|
#include "PerformanceEntry.h" |
|
#include "PerformanceMark.h" |
|
#include "PerformanceMeasure.h" |
|
#include "PerformanceResourceTiming.h" |
|
#include "mozilla/ErrorResult.h" |
|
#include "mozilla/dom/PerformanceBinding.h" |
|
#include "mozilla/dom/PerformanceTimingBinding.h" |
|
#include "mozilla/dom/PerformanceNavigationBinding.h" |
|
#include "mozilla/Preferences.h" |
|
#include "mozilla/IntegerPrintfMacros.h" |
|
#include "mozilla/TimeStamp.h" |
|
#include "js/HeapAPI.h" |
|
#include "WorkerPrivate.h" |
|
#include "WorkerRunnable.h" |
|
|
|
#ifdef MOZ_WIDGET_GONK |
|
#define PERFLOG(msg, ...) __android_log_print(ANDROID_LOG_INFO, "PerformanceTiming", msg, ##__VA_ARGS__) |
|
#else |
|
#define PERFLOG(msg, ...) printf_stderr(msg, ##__VA_ARGS__) |
|
#endif |
|
|
|
using namespace mozilla; |
|
using namespace mozilla::dom; |
|
using namespace mozilla::dom::workers; |
|
|
|
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE(nsPerformanceTiming, mPerformance) |
|
|
|
NS_IMPL_CYCLE_COLLECTION_ROOT_NATIVE(nsPerformanceTiming, AddRef) |
|
NS_IMPL_CYCLE_COLLECTION_UNROOT_NATIVE(nsPerformanceTiming, Release) |
|
|
|
nsPerformanceTiming::nsPerformanceTiming(nsPerformance* aPerformance, |
|
nsITimedChannel* aChannel, |
|
nsIHttpChannel* aHttpChannel, |
|
DOMHighResTimeStamp aZeroTime) |
|
: mPerformance(aPerformance), |
|
mFetchStart(0.0), |
|
mZeroTime(aZeroTime), |
|
mRedirectCount(0), |
|
mTimingAllowed(true), |
|
mAllRedirectsSameOrigin(true), |
|
mInitialized(!!aChannel), |
|
mReportCrossOriginRedirect(true) |
|
{ |
|
MOZ_ASSERT(aPerformance, "Parent performance object should be provided"); |
|
|
|
if (!nsContentUtils::IsPerformanceTimingEnabled()) { |
|
mZeroTime = 0; |
|
} |
|
|
|
// The aHttpChannel argument is null if this nsPerformanceTiming object |
|
// is being used for the navigation timing (document) and has a non-null |
|
// value for the resource timing (any resources within the page). |
|
if (aHttpChannel) { |
|
mTimingAllowed = CheckAllowedOrigin(aHttpChannel, aChannel); |
|
bool redirectsPassCheck = false; |
|
aChannel->GetAllRedirectsPassTimingAllowCheck(&redirectsPassCheck); |
|
mReportCrossOriginRedirect = mTimingAllowed && redirectsPassCheck; |
|
} |
|
|
|
InitializeTimingInfo(aChannel); |
|
} |
|
|
|
// Copy the timing info from the channel so we don't need to keep the channel |
|
// alive just to get the timestamps. |
|
void |
|
nsPerformanceTiming::InitializeTimingInfo(nsITimedChannel* aChannel) |
|
{ |
|
if (aChannel) { |
|
aChannel->GetAsyncOpen(&mAsyncOpen); |
|
aChannel->GetAllRedirectsSameOrigin(&mAllRedirectsSameOrigin); |
|
aChannel->GetRedirectCount(&mRedirectCount); |
|
aChannel->GetRedirectStart(&mRedirectStart); |
|
aChannel->GetRedirectEnd(&mRedirectEnd); |
|
aChannel->GetDomainLookupStart(&mDomainLookupStart); |
|
aChannel->GetDomainLookupEnd(&mDomainLookupEnd); |
|
aChannel->GetConnectStart(&mConnectStart); |
|
aChannel->GetConnectEnd(&mConnectEnd); |
|
aChannel->GetRequestStart(&mRequestStart); |
|
aChannel->GetResponseStart(&mResponseStart); |
|
aChannel->GetCacheReadStart(&mCacheReadStart); |
|
aChannel->GetResponseEnd(&mResponseEnd); |
|
aChannel->GetCacheReadEnd(&mCacheReadEnd); |
|
} |
|
} |
|
|
|
nsPerformanceTiming::~nsPerformanceTiming() |
|
{ |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformanceTiming::FetchStartHighRes() |
|
{ |
|
if (!mFetchStart) { |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return mZeroTime; |
|
} |
|
MOZ_ASSERT(!mAsyncOpen.IsNull(), "The fetch start time stamp should always be " |
|
"valid if the performance timing is enabled"); |
|
mFetchStart = (!mAsyncOpen.IsNull()) |
|
? TimeStampToDOMHighRes(mAsyncOpen) |
|
: 0.0; |
|
} |
|
return mFetchStart; |
|
} |
|
|
|
DOMTimeMilliSec |
|
nsPerformanceTiming::FetchStart() |
|
{ |
|
return static_cast<int64_t>(FetchStartHighRes()); |
|
} |
|
|
|
bool |
|
nsPerformanceTiming::CheckAllowedOrigin(nsIHttpChannel* aResourceChannel, |
|
nsITimedChannel* aChannel) |
|
{ |
|
if (!IsInitialized()) { |
|
return false; |
|
} |
|
|
|
// Check that the current document passes the ckeck. |
|
nsCOMPtr<nsILoadInfo> loadInfo; |
|
aResourceChannel->GetLoadInfo(getter_AddRefs(loadInfo)); |
|
if (!loadInfo) { |
|
return false; |
|
} |
|
nsCOMPtr<nsIPrincipal> principal = loadInfo->LoadingPrincipal(); |
|
|
|
// Check if the resource is either same origin as the page that started |
|
// the load, or if the response contains the proper Timing-Allow-Origin |
|
// header with the domain of the page that started the load. |
|
return aChannel->TimingAllowCheck(principal); |
|
} |
|
|
|
bool |
|
nsPerformanceTiming::TimingAllowed() const |
|
{ |
|
return mTimingAllowed; |
|
} |
|
|
|
uint16_t |
|
nsPerformanceTiming::GetRedirectCount() const |
|
{ |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return 0; |
|
} |
|
if (!mAllRedirectsSameOrigin) { |
|
return 0; |
|
} |
|
return mRedirectCount; |
|
} |
|
|
|
bool |
|
nsPerformanceTiming::ShouldReportCrossOriginRedirect() const |
|
{ |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return false; |
|
} |
|
|
|
// If the redirect count is 0, or if one of the cross-origin |
|
// redirects doesn't have the proper Timing-Allow-Origin header, |
|
// then RedirectStart and RedirectEnd will be set to zero |
|
return (mRedirectCount != 0) && mReportCrossOriginRedirect; |
|
} |
|
|
|
/** |
|
* RedirectStartHighRes() is used by both the navigation timing and the |
|
* resource timing. Since, navigation timing and resource timing check and |
|
* interpret cross-domain redirects in a different manner, |
|
* RedirectStartHighRes() will make no checks for cross-domain redirect. |
|
* It's up to the consumers of this method (nsPerformanceTiming::RedirectStart() |
|
* and PerformanceResourceTiming::RedirectStart() to make such verifications. |
|
* |
|
* @return a valid timing if the Performance Timing is enabled |
|
*/ |
|
DOMHighResTimeStamp |
|
nsPerformanceTiming::RedirectStartHighRes() |
|
{ |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return mZeroTime; |
|
} |
|
return TimeStampToDOMHighResOrFetchStart(mRedirectStart); |
|
} |
|
|
|
DOMTimeMilliSec |
|
nsPerformanceTiming::RedirectStart() |
|
{ |
|
if (!IsInitialized()) { |
|
return 0; |
|
} |
|
// We have to check if all the redirect URIs had the same origin (since there |
|
// is no check in RedirectStartHighRes()) |
|
if (mAllRedirectsSameOrigin && mRedirectCount) { |
|
return static_cast<int64_t>(RedirectStartHighRes()); |
|
} |
|
return 0; |
|
} |
|
|
|
/** |
|
* RedirectEndHighRes() is used by both the navigation timing and the resource |
|
* timing. Since, navigation timing and resource timing check and interpret |
|
* cross-domain redirects in a different manner, RedirectEndHighRes() will make |
|
* no checks for cross-domain redirect. It's up to the consumers of this method |
|
* (nsPerformanceTiming::RedirectEnd() and |
|
* PerformanceResourceTiming::RedirectEnd() to make such verifications. |
|
* |
|
* @return a valid timing if the Performance Timing is enabled |
|
*/ |
|
DOMHighResTimeStamp |
|
nsPerformanceTiming::RedirectEndHighRes() |
|
{ |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return mZeroTime; |
|
} |
|
return TimeStampToDOMHighResOrFetchStart(mRedirectEnd); |
|
} |
|
|
|
DOMTimeMilliSec |
|
nsPerformanceTiming::RedirectEnd() |
|
{ |
|
if (!IsInitialized()) { |
|
return 0; |
|
} |
|
// We have to check if all the redirect URIs had the same origin (since there |
|
// is no check in RedirectEndHighRes()) |
|
if (mAllRedirectsSameOrigin && mRedirectCount) { |
|
return static_cast<int64_t>(RedirectEndHighRes()); |
|
} |
|
return 0; |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformanceTiming::DomainLookupStartHighRes() |
|
{ |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return mZeroTime; |
|
} |
|
return TimeStampToDOMHighResOrFetchStart(mDomainLookupStart); |
|
} |
|
|
|
DOMTimeMilliSec |
|
nsPerformanceTiming::DomainLookupStart() |
|
{ |
|
return static_cast<int64_t>(DomainLookupStartHighRes()); |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformanceTiming::DomainLookupEndHighRes() |
|
{ |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return mZeroTime; |
|
} |
|
// Bug 1155008 - nsHttpTransaction is racy. Return DomainLookupStart when null |
|
return mDomainLookupEnd.IsNull() ? DomainLookupStartHighRes() |
|
: TimeStampToDOMHighRes(mDomainLookupEnd); |
|
} |
|
|
|
DOMTimeMilliSec |
|
nsPerformanceTiming::DomainLookupEnd() |
|
{ |
|
return static_cast<int64_t>(DomainLookupEndHighRes()); |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformanceTiming::ConnectStartHighRes() |
|
{ |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return mZeroTime; |
|
} |
|
return mConnectStart.IsNull() ? DomainLookupEndHighRes() |
|
: TimeStampToDOMHighRes(mConnectStart); |
|
} |
|
|
|
DOMTimeMilliSec |
|
nsPerformanceTiming::ConnectStart() |
|
{ |
|
return static_cast<int64_t>(ConnectStartHighRes()); |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformanceTiming::ConnectEndHighRes() |
|
{ |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return mZeroTime; |
|
} |
|
// Bug 1155008 - nsHttpTransaction is racy. Return ConnectStart when null |
|
return mConnectEnd.IsNull() ? ConnectStartHighRes() |
|
: TimeStampToDOMHighRes(mConnectEnd); |
|
} |
|
|
|
DOMTimeMilliSec |
|
nsPerformanceTiming::ConnectEnd() |
|
{ |
|
return static_cast<int64_t>(ConnectEndHighRes()); |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformanceTiming::RequestStartHighRes() |
|
{ |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return mZeroTime; |
|
} |
|
return TimeStampToDOMHighResOrFetchStart(mRequestStart); |
|
} |
|
|
|
DOMTimeMilliSec |
|
nsPerformanceTiming::RequestStart() |
|
{ |
|
return static_cast<int64_t>(RequestStartHighRes()); |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformanceTiming::ResponseStartHighRes() |
|
{ |
|
if (!nsContentUtils::IsPerformanceTimingEnabled() || !IsInitialized()) { |
|
return mZeroTime; |
|
} |
|
if (mResponseStart.IsNull() || |
|
(!mCacheReadStart.IsNull() && mCacheReadStart < mResponseStart)) { |
|
mResponseStart = mCacheReadStart; |
|
} |
|
return TimeStampToDOMHighResOrFetchStart(mResponseStart); |
|
} |
|
|
|
DOMTimeMilliSec |
|
nsPerformanceTiming::ResponseStart() |
|
{ |
|
return static_cast<int64_t>(ResponseStartHighRes()); |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformanceTiming::ResponseEndHighRes() |
|
{ |
|
if (!IsInitialized()) { |
|
return mZeroTime; |
|
} |
|
if (mResponseEnd.IsNull() || |
|
(!mCacheReadEnd.IsNull() && mCacheReadEnd < mResponseEnd)) { |
|
mResponseEnd = mCacheReadEnd; |
|
} |
|
// Bug 1155008 - nsHttpTransaction is racy. Return ResponseStart when null |
|
return mResponseEnd.IsNull() ? ResponseStartHighRes() |
|
: TimeStampToDOMHighRes(mResponseEnd); |
|
} |
|
|
|
DOMTimeMilliSec |
|
nsPerformanceTiming::ResponseEnd() |
|
{ |
|
return static_cast<int64_t>(ResponseEndHighRes()); |
|
} |
|
|
|
bool |
|
nsPerformanceTiming::IsInitialized() const |
|
{ |
|
return mInitialized; |
|
} |
|
|
|
JSObject* |
|
nsPerformanceTiming::WrapObject(JSContext *cx, JS::Handle<JSObject*> aGivenProto) |
|
{ |
|
return PerformanceTimingBinding::Wrap(cx, this, aGivenProto); |
|
} |
|
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE(nsPerformanceNavigation, mPerformance) |
|
|
|
NS_IMPL_CYCLE_COLLECTION_ROOT_NATIVE(nsPerformanceNavigation, AddRef) |
|
NS_IMPL_CYCLE_COLLECTION_UNROOT_NATIVE(nsPerformanceNavigation, Release) |
|
|
|
nsPerformanceNavigation::nsPerformanceNavigation(nsPerformance* aPerformance) |
|
: mPerformance(aPerformance) |
|
{ |
|
MOZ_ASSERT(aPerformance, "Parent performance object should be provided"); |
|
} |
|
|
|
nsPerformanceNavigation::~nsPerformanceNavigation() |
|
{ |
|
} |
|
|
|
JSObject* |
|
nsPerformanceNavigation::WrapObject(JSContext *cx, JS::Handle<JSObject*> aGivenProto) |
|
{ |
|
return PerformanceNavigationBinding::Wrap(cx, this, aGivenProto); |
|
} |
|
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(nsPerformance) |
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(nsPerformance, PerformanceBase) |
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mTiming, |
|
mNavigation, |
|
mParentPerformance) |
|
tmp->mMozMemory = nullptr; |
|
mozilla::DropJSObjects(this); |
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END |
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(nsPerformance, PerformanceBase) |
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTiming, |
|
mNavigation, |
|
mParentPerformance) |
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_SCRIPT_OBJECTS |
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END |
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN_INHERITED(nsPerformance, PerformanceBase) |
|
NS_IMPL_CYCLE_COLLECTION_TRACE_JS_MEMBER_CALLBACK(mMozMemory) |
|
NS_IMPL_CYCLE_COLLECTION_TRACE_END |
|
|
|
NS_IMPL_ADDREF_INHERITED(nsPerformance, PerformanceBase) |
|
NS_IMPL_RELEASE_INHERITED(nsPerformance, PerformanceBase) |
|
|
|
nsPerformance::nsPerformance(nsPIDOMWindow* aWindow, |
|
nsDOMNavigationTiming* aDOMTiming, |
|
nsITimedChannel* aChannel, |
|
nsPerformance* aParentPerformance) |
|
: PerformanceBase(aWindow), |
|
mDOMTiming(aDOMTiming), |
|
mChannel(aChannel), |
|
mParentPerformance(aParentPerformance) |
|
{ |
|
MOZ_ASSERT(aWindow, "Parent window object should be provided"); |
|
} |
|
|
|
nsPerformance::~nsPerformance() |
|
{ |
|
mozilla::DropJSObjects(this); |
|
} |
|
|
|
// QueryInterface implementation for nsPerformance |
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(nsPerformance) |
|
NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY |
|
NS_INTERFACE_MAP_ENTRY(nsISupports) |
|
NS_INTERFACE_MAP_END_INHERITING(DOMEventTargetHelper) |
|
|
|
void |
|
nsPerformance::GetMozMemory(JSContext *aCx, JS::MutableHandle<JSObject*> aObj) |
|
{ |
|
if (!mMozMemory) { |
|
mMozMemory = js::gc::NewMemoryInfoObject(aCx); |
|
if (mMozMemory) { |
|
mozilla::HoldJSObjects(this); |
|
} |
|
} |
|
|
|
aObj.set(mMozMemory); |
|
} |
|
|
|
nsPerformanceTiming* |
|
nsPerformance::Timing() |
|
{ |
|
if (!mTiming) { |
|
// For navigation timing, the third argument (an nsIHtttpChannel) is null |
|
// since the cross-domain redirect were already checked. |
|
// The last argument (zero time) for performance.timing is the navigation |
|
// start value. |
|
mTiming = new nsPerformanceTiming(this, mChannel, nullptr, |
|
mDOMTiming->GetNavigationStart()); |
|
} |
|
return mTiming; |
|
} |
|
|
|
void |
|
nsPerformance::DispatchBufferFullEvent() |
|
{ |
|
nsRefPtr<Event> event = NS_NewDOMEvent(this, nullptr, nullptr); |
|
// it bubbles, and it isn't cancelable |
|
nsresult rv = event->InitEvent(NS_LITERAL_STRING("resourcetimingbufferfull"), |
|
true, false); |
|
if (NS_FAILED(rv)) { |
|
return; |
|
} |
|
event->SetTrusted(true); |
|
DispatchDOMEvent(nullptr, event, nullptr, nullptr); |
|
} |
|
|
|
nsPerformanceNavigation* |
|
nsPerformance::Navigation() |
|
{ |
|
if (!mNavigation) { |
|
mNavigation = new nsPerformanceNavigation(this); |
|
} |
|
return mNavigation; |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformance::Now() const |
|
{ |
|
double nowTimeMs = GetDOMTiming()->TimeStampToDOMHighRes(TimeStamp::Now()); |
|
// Round down to the nearest 20us, because if the timer is too accurate people |
|
// can do nasty timing attacks with it. See similar code in the worker |
|
// Performance implementation. |
|
const double maxResolutionMs = 0.020; |
|
return floor(nowTimeMs / maxResolutionMs) * maxResolutionMs; |
|
} |
|
|
|
JSObject* |
|
nsPerformance::WrapObject(JSContext *cx, JS::Handle<JSObject*> aGivenProto) |
|
{ |
|
return PerformanceBinding::Wrap(cx, this, aGivenProto); |
|
} |
|
|
|
/** |
|
* An entry should be added only after the resource is loaded. |
|
* This method is not thread safe and can only be called on the main thread. |
|
*/ |
|
void |
|
nsPerformance::AddEntry(nsIHttpChannel* channel, |
|
nsITimedChannel* timedChannel) |
|
{ |
|
MOZ_ASSERT(NS_IsMainThread()); |
|
// Check if resource timing is prefed off. |
|
if (!nsContentUtils::IsResourceTimingEnabled()) { |
|
return; |
|
} |
|
|
|
// Don't add the entry if the buffer is full |
|
if (IsResourceEntryLimitReached()) { |
|
return; |
|
} |
|
|
|
if (channel && timedChannel) { |
|
nsAutoCString name; |
|
nsAutoString initiatorType; |
|
nsCOMPtr<nsIURI> originalURI; |
|
|
|
timedChannel->GetInitiatorType(initiatorType); |
|
|
|
// According to the spec, "The name attribute must return the resolved URL |
|
// of the requested resource. This attribute must not change even if the |
|
// fetch redirected to a different URL." |
|
channel->GetOriginalURI(getter_AddRefs(originalURI)); |
|
originalURI->GetSpec(name); |
|
NS_ConvertUTF8toUTF16 entryName(name); |
|
|
|
// The nsITimedChannel argument will be used to gather all the timings. |
|
// The nsIHttpChannel argument will be used to check if any cross-origin |
|
// redirects occurred. |
|
// The last argument is the "zero time" (offset). Since we don't want |
|
// any offset for the resource timing, this will be set to "0" - the |
|
// resource timing returns a relative timing (no offset). |
|
nsRefPtr<nsPerformanceTiming> performanceTiming = |
|
new nsPerformanceTiming(this, timedChannel, channel, |
|
0); |
|
|
|
// The PerformanceResourceTiming object will use the nsPerformanceTiming |
|
// object to get all the required timings. |
|
nsRefPtr<PerformanceResourceTiming> performanceEntry = |
|
new PerformanceResourceTiming(performanceTiming, this, entryName); |
|
|
|
// If the initiator type had no valid value, then set it to the default |
|
// ("other") value. |
|
if (initiatorType.IsEmpty()) { |
|
initiatorType = NS_LITERAL_STRING("other"); |
|
} |
|
performanceEntry->SetInitiatorType(initiatorType); |
|
InsertResourceEntry(performanceEntry); |
|
} |
|
} |
|
|
|
// To be removed once bug 1124165 lands |
|
bool |
|
nsPerformance::IsPerformanceTimingAttribute(const nsAString& aName) |
|
{ |
|
// Note that toJSON is added to this list due to bug 1047848 |
|
static const char* attributes[] = |
|
{"navigationStart", "unloadEventStart", "unloadEventEnd", "redirectStart", |
|
"redirectEnd", "fetchStart", "domainLookupStart", "domainLookupEnd", |
|
"connectStart", "connectEnd", "requestStart", "responseStart", |
|
"responseEnd", "domLoading", "domInteractive", "domContentLoadedEventStart", |
|
"domContentLoadedEventEnd", "domComplete", "loadEventStart", |
|
"loadEventEnd", nullptr}; |
|
|
|
for (uint32_t i = 0; attributes[i]; ++i) { |
|
if (aName.EqualsASCII(attributes[i])) { |
|
return true; |
|
} |
|
} |
|
return false; |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformance::GetPerformanceTimingFromString(const nsAString& aProperty) |
|
{ |
|
if (!IsPerformanceTimingAttribute(aProperty)) { |
|
return 0; |
|
} |
|
if (aProperty.EqualsLiteral("navigationStart")) { |
|
// DOMHighResTimeStamp is in relation to navigationStart, so this will be |
|
// zero. |
|
return GetDOMTiming()->GetNavigationStart(); |
|
} |
|
if (aProperty.EqualsLiteral("unloadEventStart")) { |
|
return GetDOMTiming()->GetUnloadEventStart(); |
|
} |
|
if (aProperty.EqualsLiteral("unloadEventEnd")) { |
|
return GetDOMTiming()->GetUnloadEventEnd(); |
|
} |
|
if (aProperty.EqualsLiteral("redirectStart")) { |
|
return Timing()->RedirectStart(); |
|
} |
|
if (aProperty.EqualsLiteral("redirectEnd")) { |
|
return Timing()->RedirectEnd(); |
|
} |
|
if (aProperty.EqualsLiteral("fetchStart")) { |
|
return Timing()->FetchStart(); |
|
} |
|
if (aProperty.EqualsLiteral("domainLookupStart")) { |
|
return Timing()->DomainLookupStart(); |
|
} |
|
if (aProperty.EqualsLiteral("domainLookupEnd")) { |
|
return Timing()->DomainLookupEnd(); |
|
} |
|
if (aProperty.EqualsLiteral("connectStart")) { |
|
return Timing()->ConnectStart(); |
|
} |
|
if (aProperty.EqualsLiteral("connectEnd")) { |
|
return Timing()->ConnectEnd(); |
|
} |
|
if (aProperty.EqualsLiteral("requestStart")) { |
|
return Timing()->RequestStart(); |
|
} |
|
if (aProperty.EqualsLiteral("responseStart")) { |
|
return Timing()->ResponseStart(); |
|
} |
|
if (aProperty.EqualsLiteral("responseEnd")) { |
|
return Timing()->ResponseEnd(); |
|
} |
|
if (aProperty.EqualsLiteral("domLoading")) { |
|
return GetDOMTiming()->GetDomLoading(); |
|
} |
|
if (aProperty.EqualsLiteral("domInteractive")) { |
|
return GetDOMTiming()->GetDomInteractive(); |
|
} |
|
if (aProperty.EqualsLiteral("domContentLoadedEventStart")) { |
|
return GetDOMTiming()->GetDomContentLoadedEventStart(); |
|
} |
|
if (aProperty.EqualsLiteral("domContentLoadedEventEnd")) { |
|
return GetDOMTiming()->GetDomContentLoadedEventEnd(); |
|
} |
|
if (aProperty.EqualsLiteral("domComplete")) { |
|
return GetDOMTiming()->GetDomComplete(); |
|
} |
|
if (aProperty.EqualsLiteral("loadEventStart")) { |
|
return GetDOMTiming()->GetLoadEventStart(); |
|
} |
|
if (aProperty.EqualsLiteral("loadEventEnd")) { |
|
return GetDOMTiming()->GetLoadEventEnd(); |
|
} |
|
MOZ_CRASH("IsPerformanceTimingAttribute and GetPerformanceTimingFromString are out of sync"); |
|
return 0; |
|
} |
|
|
|
namespace { |
|
|
|
// Helper classes |
|
class MOZ_STACK_CLASS PerformanceEntryComparator final |
|
{ |
|
public: |
|
bool Equals(const PerformanceEntry* aElem1, |
|
const PerformanceEntry* aElem2) const |
|
{ |
|
MOZ_ASSERT(aElem1 && aElem2, |
|
"Trying to compare null performance entries"); |
|
return aElem1->StartTime() == aElem2->StartTime(); |
|
} |
|
|
|
bool LessThan(const PerformanceEntry* aElem1, |
|
const PerformanceEntry* aElem2) const |
|
{ |
|
MOZ_ASSERT(aElem1 && aElem2, |
|
"Trying to compare null performance entries"); |
|
return aElem1->StartTime() < aElem2->StartTime(); |
|
} |
|
}; |
|
|
|
class PrefEnabledRunnable final : public WorkerMainThreadRunnable |
|
{ |
|
public: |
|
explicit PrefEnabledRunnable(WorkerPrivate* aWorkerPrivate) |
|
: WorkerMainThreadRunnable(aWorkerPrivate) |
|
, mEnabled(false) |
|
{ } |
|
|
|
bool MainThreadRun() override |
|
{ |
|
MOZ_ASSERT(NS_IsMainThread()); |
|
mEnabled = Preferences::GetBool("dom.enable_user_timing", false); |
|
return true; |
|
} |
|
|
|
bool IsEnabled() const |
|
{ |
|
return mEnabled; |
|
} |
|
|
|
private: |
|
bool mEnabled; |
|
}; |
|
|
|
} // anonymous namespace |
|
|
|
/* static */ bool |
|
nsPerformance::IsEnabled(JSContext* aCx, JSObject* aGlobal) |
|
{ |
|
if (NS_IsMainThread()) { |
|
return Preferences::GetBool("dom.enable_user_timing", false); |
|
} |
|
|
|
WorkerPrivate* workerPrivate = GetCurrentThreadWorkerPrivate(); |
|
MOZ_ASSERT(workerPrivate); |
|
workerPrivate->AssertIsOnWorkerThread(); |
|
|
|
nsRefPtr<PrefEnabledRunnable> runnable = |
|
new PrefEnabledRunnable(workerPrivate); |
|
runnable->Dispatch(workerPrivate->GetJSContext()); |
|
|
|
return runnable->IsEnabled(); |
|
} |
|
|
|
void |
|
nsPerformance::InsertUserEntry(PerformanceEntry* aEntry) |
|
{ |
|
MOZ_ASSERT(NS_IsMainThread()); |
|
|
|
if (nsContentUtils::IsUserTimingLoggingEnabled()) { |
|
nsAutoCString uri; |
|
nsresult rv = GetOwner()->GetDocumentURI()->GetHost(uri); |
|
if(NS_FAILED(rv)) { |
|
// If we have no URI, just put in "none". |
|
uri.AssignLiteral("none"); |
|
} |
|
PerformanceBase::LogEntry(aEntry, uri); |
|
} |
|
|
|
PerformanceBase::InsertUserEntry(aEntry); |
|
} |
|
|
|
DOMHighResTimeStamp |
|
nsPerformance::DeltaFromNavigationStart(DOMHighResTimeStamp aTime) |
|
{ |
|
// If the time we're trying to convert is equal to zero, it hasn't been set |
|
// yet so just return 0. |
|
if (aTime == 0) { |
|
return 0; |
|
} |
|
return aTime - GetDOMTiming()->GetNavigationStart(); |
|
} |
|
|
|
// PerformanceBase |
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(PerformanceBase) |
|
NS_INTERFACE_MAP_END_INHERITING(DOMEventTargetHelper) |
|
|
|
NS_IMPL_CYCLE_COLLECTION_INHERITED(PerformanceBase, |
|
DOMEventTargetHelper, |
|
mUserEntries, |
|
mResourceEntries); |
|
|
|
NS_IMPL_ADDREF_INHERITED(PerformanceBase, DOMEventTargetHelper) |
|
NS_IMPL_RELEASE_INHERITED(PerformanceBase, DOMEventTargetHelper) |
|
|
|
PerformanceBase::PerformanceBase() |
|
: mResourceTimingBufferSize(kDefaultResourceTimingBufferSize) |
|
{ |
|
MOZ_ASSERT(!NS_IsMainThread()); |
|
} |
|
|
|
PerformanceBase::PerformanceBase(nsPIDOMWindow* aWindow) |
|
: DOMEventTargetHelper(aWindow) |
|
, mResourceTimingBufferSize(kDefaultResourceTimingBufferSize) |
|
{ |
|
MOZ_ASSERT(NS_IsMainThread()); |
|
} |
|
|
|
PerformanceBase::~PerformanceBase() |
|
{} |
|
|
|
void |
|
PerformanceBase::GetEntries(nsTArray<nsRefPtr<PerformanceEntry>>& aRetval) |
|
{ |
|
aRetval = mResourceEntries; |
|
aRetval.AppendElements(mUserEntries); |
|
aRetval.Sort(PerformanceEntryComparator()); |
|
} |
|
|
|
void |
|
PerformanceBase::GetEntriesByType(const nsAString& aEntryType, |
|
nsTArray<nsRefPtr<PerformanceEntry>>& aRetval) |
|
{ |
|
if (aEntryType.EqualsLiteral("resource")) { |
|
aRetval = mResourceEntries; |
|
return; |
|
} |
|
|
|
aRetval.Clear(); |
|
|
|
if (aEntryType.EqualsLiteral("mark") || |
|
aEntryType.EqualsLiteral("measure")) { |
|
for (PerformanceEntry* entry : mUserEntries) { |
|
if (entry->GetEntryType().Equals(aEntryType)) { |
|
aRetval.AppendElement(entry); |
|
} |
|
} |
|
} |
|
} |
|
|
|
void |
|
PerformanceBase::GetEntriesByName(const nsAString& aName, |
|
const Optional<nsAString>& aEntryType, |
|
nsTArray<nsRefPtr<PerformanceEntry>>& aRetval) |
|
{ |
|
aRetval.Clear(); |
|
|
|
for (PerformanceEntry* entry : mResourceEntries) { |
|
if (entry->GetName().Equals(aName) && |
|
(!aEntryType.WasPassed() || |
|
entry->GetEntryType().Equals(aEntryType.Value()))) { |
|
aRetval.AppendElement(entry); |
|
} |
|
} |
|
|
|
for (PerformanceEntry* entry : mUserEntries) { |
|
if (entry->GetName().Equals(aName) && |
|
(!aEntryType.WasPassed() || |
|
entry->GetEntryType().Equals(aEntryType.Value()))) { |
|
aRetval.AppendElement(entry); |
|
} |
|
} |
|
|
|
aRetval.Sort(PerformanceEntryComparator()); |
|
} |
|
|
|
void |
|
PerformanceBase::ClearUserEntries(const Optional<nsAString>& aEntryName, |
|
const nsAString& aEntryType) |
|
{ |
|
for (uint32_t i = 0; i < mUserEntries.Length();) { |
|
if ((!aEntryName.WasPassed() || |
|
mUserEntries[i]->GetName().Equals(aEntryName.Value())) && |
|
(aEntryType.IsEmpty() || |
|
mUserEntries[i]->GetEntryType().Equals(aEntryType))) { |
|
mUserEntries.RemoveElementAt(i); |
|
} else { |
|
++i; |
|
} |
|
} |
|
} |
|
|
|
void |
|
PerformanceBase::ClearResourceTimings() |
|
{ |
|
MOZ_ASSERT(NS_IsMainThread()); |
|
mResourceEntries.Clear(); |
|
} |
|
|
|
void |
|
PerformanceBase::Mark(const nsAString& aName, ErrorResult& aRv) |
|
{ |
|
// Don't add the entry if the buffer is full. XXX should be removed by bug 1159003. |
|
if (mUserEntries.Length() >= mResourceTimingBufferSize) { |
|
return; |
|
} |
|
|
|
if (IsPerformanceTimingAttribute(aName)) { |
|
aRv.Throw(NS_ERROR_DOM_SYNTAX_ERR); |
|
return; |
|
} |
|
|
|
nsRefPtr<PerformanceMark> performanceMark = |
|
new PerformanceMark(GetAsISupports(), aName, Now()); |
|
InsertUserEntry(performanceMark); |
|
} |
|
|
|
void |
|
PerformanceBase::ClearMarks(const Optional<nsAString>& aName) |
|
{ |
|
ClearUserEntries(aName, NS_LITERAL_STRING("mark")); |
|
} |
|
|
|
DOMHighResTimeStamp |
|
PerformanceBase::ResolveTimestampFromName(const nsAString& aName, |
|
ErrorResult& aRv) |
|
{ |
|
nsAutoTArray<nsRefPtr<PerformanceEntry>, 1> arr; |
|
DOMHighResTimeStamp ts; |
|
Optional<nsAString> typeParam; |
|
nsAutoString str; |
|
str.AssignLiteral("mark"); |
|
typeParam = &str; |
|
GetEntriesByName(aName, typeParam, arr); |
|
if (!arr.IsEmpty()) { |
|
return arr.LastElement()->StartTime(); |
|
} |
|
|
|
if (!IsPerformanceTimingAttribute(aName)) { |
|
aRv.Throw(NS_ERROR_DOM_SYNTAX_ERR); |
|
return 0; |
|
} |
|
|
|
ts = GetPerformanceTimingFromString(aName); |
|
if (!ts) { |
|
aRv.Throw(NS_ERROR_DOM_INVALID_ACCESS_ERR); |
|
return 0; |
|
} |
|
|
|
return DeltaFromNavigationStart(ts); |
|
} |
|
|
|
void |
|
PerformanceBase::Measure(const nsAString& aName, |
|
const Optional<nsAString>& aStartMark, |
|
const Optional<nsAString>& aEndMark, |
|
ErrorResult& aRv) |
|
{ |
|
// Don't add the entry if the buffer is full. XXX should be removed by bug |
|
// 1159003. |
|
if (mUserEntries.Length() >= mResourceTimingBufferSize) { |
|
return; |
|
} |
|
|
|
DOMHighResTimeStamp startTime; |
|
DOMHighResTimeStamp endTime; |
|
|
|
if (IsPerformanceTimingAttribute(aName)) { |
|
aRv.Throw(NS_ERROR_DOM_SYNTAX_ERR); |
|
return; |
|
} |
|
|
|
if (aStartMark.WasPassed()) { |
|
startTime = ResolveTimestampFromName(aStartMark.Value(), aRv); |
|
if (NS_WARN_IF(aRv.Failed())) { |
|
return; |
|
} |
|
} else { |
|
// Navigation start is used in this case, but since DOMHighResTimeStamp is |
|
// in relation to navigation start, this will be zero if a name is not |
|
// passed. |
|
startTime = 0; |
|
} |
|
|
|
if (aEndMark.WasPassed()) { |
|
endTime = ResolveTimestampFromName(aEndMark.Value(), aRv); |
|
if (NS_WARN_IF(aRv.Failed())) { |
|
return; |
|
} |
|
} else { |
|
endTime = Now(); |
|
} |
|
|
|
nsRefPtr<PerformanceMeasure> performanceMeasure = |
|
new PerformanceMeasure(GetAsISupports(), aName, startTime, endTime); |
|
InsertUserEntry(performanceMeasure); |
|
} |
|
|
|
void |
|
PerformanceBase::ClearMeasures(const Optional<nsAString>& aName) |
|
{ |
|
ClearUserEntries(aName, NS_LITERAL_STRING("measure")); |
|
} |
|
|
|
void |
|
PerformanceBase::LogEntry(PerformanceEntry* aEntry, const nsACString& aOwner) const |
|
{ |
|
PERFLOG("Performance Entry: %s|%s|%s|%f|%f|%" PRIu64 "\n", |
|
aOwner.BeginReading(), |
|
NS_ConvertUTF16toUTF8(aEntry->GetEntryType()).get(), |
|
NS_ConvertUTF16toUTF8(aEntry->GetName()).get(), |
|
aEntry->StartTime(), |
|
aEntry->Duration(), |
|
static_cast<uint64_t>(PR_Now() / PR_USEC_PER_MSEC)); |
|
} |
|
|
|
void |
|
PerformanceBase::InsertUserEntry(PerformanceEntry* aEntry) |
|
{ |
|
mUserEntries.InsertElementSorted(aEntry, |
|
PerformanceEntryComparator()); |
|
} |
|
|
|
void |
|
PerformanceBase::SetResourceTimingBufferSize(uint64_t aMaxSize) |
|
{ |
|
mResourceTimingBufferSize = aMaxSize; |
|
} |
|
|
|
void |
|
PerformanceBase::InsertResourceEntry(PerformanceEntry* aEntry) |
|
{ |
|
MOZ_ASSERT(aEntry); |
|
MOZ_ASSERT(mResourceEntries.Length() < mResourceTimingBufferSize); |
|
if (mResourceEntries.Length() >= mResourceTimingBufferSize) { |
|
return; |
|
} |
|
|
|
mResourceEntries.InsertElementSorted(aEntry, |
|
PerformanceEntryComparator()); |
|
if (mResourceEntries.Length() == mResourceTimingBufferSize) { |
|
// call onresourcetimingbufferfull |
|
DispatchBufferFullEvent(); |
|
} |
|
}
|
|
|