blob: 2e4a61b3f603f09f9568d3a9c424a6d0abbb1c56 [file] [log] [blame]
/*
* Copyright (C) 2010 Google Inc. All rights reserved.
* Copyright (C) 2012 Intel Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef THIRD_PARTY_BLINK_RENDERER_CORE_TIMING_PERFORMANCE_H_
#define THIRD_PARTY_BLINK_RENDERER_CORE_TIMING_PERFORMANCE_H_
#include "base/single_thread_task_runner.h"
#include "third_party/blink/public/platform/web_resource_timing_info.h"
#include "third_party/blink/renderer/bindings/core/v8/string_or_double.h"
#include "third_party/blink/renderer/core/core_export.h"
#include "third_party/blink/renderer/core/dom/dom_high_res_time_stamp.h"
#include "third_party/blink/renderer/core/dom/events/event_target.h"
#include "third_party/blink/renderer/core/loader/frame_loader_types.h"
#include "third_party/blink/renderer/core/timing/performance_entry.h"
#include "third_party/blink/renderer/core/timing/performance_navigation_timing.h"
#include "third_party/blink/renderer/core/timing/performance_paint_timing.h"
#include "third_party/blink/renderer/core/timing/sub_task_attribution.h"
#include "third_party/blink/renderer/platform/heap/handle.h"
#include "third_party/blink/renderer/platform/timer.h"
#include "third_party/blink/renderer/platform/wtf/forward.h"
#include "third_party/blink/renderer/platform/wtf/hash_set.h"
#include "third_party/blink/renderer/platform/wtf/linked_hash_set.h"
#include "third_party/blink/renderer/platform/wtf/text/atomic_string.h"
#include "third_party/blink/renderer/platform/wtf/vector.h"
namespace blink {
class DoubleOrPerformanceMarkOptions;
class ExceptionState;
class MemoryInfo;
class PerformanceElementTiming;
class PerformanceEventTiming;
class PerformanceMark;
class PerformanceMeasure;
class PerformanceNavigation;
class PerformanceObserver;
class PerformanceTiming;
class ResourceResponse;
class ResourceTimingInfo;
class ScriptState;
class ScriptValue;
class SecurityOrigin;
class StringOrDoubleOrPerformanceMeasureOptions;
class SubTaskAttribution;
class UserTiming;
class V8ObjectBuilder;
using PerformanceEntryVector = HeapVector<TraceWrapperMember<PerformanceEntry>>;
using PerformanceEntryDeque = HeapDeque<TraceWrapperMember<PerformanceEntry>>;
class CORE_EXPORT Performance : public EventTargetWithInlineData {
DEFINE_WRAPPERTYPEINFO();
public:
~Performance() override;
const AtomicString& InterfaceName() const override;
// Overriden by WindowPerformance but not by WorkerPerformance.
virtual PerformanceTiming* timing() const;
virtual PerformanceNavigation* navigation() const;
virtual MemoryInfo* memory() const;
virtual bool shouldYield() const;
virtual void UpdateLongTaskInstrumentation() {}
// Reduce the resolution to prevent timing attacks. See:
// http://www.w3.org/TR/hr-time-2/#privacy-security
static double ClampTimeResolution(double time_seconds);
static DOMHighResTimeStamp MonotonicTimeToDOMHighResTimeStamp(
TimeTicks time_origin,
TimeTicks monotonic_time,
bool allow_negative_value);
// Translate given platform monotonic time in seconds into a high resolution
// DOMHighResTimeStamp in milliseconds. The result timestamp is relative to
// document's time origin and has a time resolution that is safe for
// exposing to web.
DOMHighResTimeStamp MonotonicTimeToDOMHighResTimeStamp(TimeTicks) const;
DOMHighResTimeStamp now() const;
// High Resolution Time Level 3 timeOrigin.
// (https://www.w3.org/TR/hr-time-3/#dom-performance-timeorigin)
DOMHighResTimeStamp timeOrigin() const;
// Internal getter method for the time origin value.
double GetTimeOrigin() const { return TimeTicksInSeconds(time_origin_); }
PerformanceEntryVector getEntries();
PerformanceEntryVector getEntriesByType(const AtomicString& entry_type);
PerformanceEntryVector getEntriesByName(const AtomicString& name,
const AtomicString& entry_type);
void clearResourceTimings();
void setResourceTimingBufferSize(unsigned);
DEFINE_ATTRIBUTE_EVENT_LISTENER(resourcetimingbufferfull,
kResourcetimingbufferfull);
void AddLongTaskTiming(
TimeTicks start_time,
TimeTicks end_time,
const AtomicString& name,
const String& culprit_frame_src,
const String& culprit_frame_id,
const String& culprit_frame_name,
const SubTaskAttribution::EntriesVector& sub_task_attributions);
// Generates and add a performance entry for the given ResourceTimingInfo.
// |overridden_initiator_type| allows the initiator type to be overridden to
// the frame element name for the main resource.
void GenerateAndAddResourceTiming(
const ResourceTimingInfo&,
const AtomicString& overridden_initiator_type = g_null_atom);
// Generates timing info suitable for appending to the performance entries of
// a context with |origin|. This should be rarely used; most callsites should
// prefer the convenience method |GenerateAndAddResourceTiming()|.
static WebResourceTimingInfo GenerateResourceTiming(
const SecurityOrigin& destination_origin,
const ResourceTimingInfo&,
ExecutionContext& context_for_use_counter);
void AddResourceTiming(const WebResourceTimingInfo&,
const AtomicString& initiator_type);
void NotifyNavigationTimingToObservers();
void AddFirstPaintTiming(TimeTicks start_time);
void AddFirstContentfulPaintTiming(TimeTicks start_time);
bool IsElementTimingBufferFull() const;
void AddElementTimingBuffer(PerformanceElementTiming&);
unsigned ElementTimingBufferSize() const;
void clearElementTimings();
void setElementTimingBufferMaxSize(unsigned);
DEFINE_ATTRIBUTE_EVENT_LISTENER(elementtimingbufferfull,
kElementtimingbufferfull);
bool IsEventTimingBufferFull() const;
void AddEventTimingBuffer(PerformanceEventTiming&);
unsigned EventTimingBufferSize() const;
void clearEventTimings();
void setEventTimingBufferMaxSize(unsigned);
DEFINE_ATTRIBUTE_EVENT_LISTENER(eventtimingbufferfull,
kEventtimingbufferfull);
PerformanceMark* mark(ScriptState*,
const AtomicString& mark_name,
ExceptionState&);
PerformanceMark* mark(
ScriptState*,
const AtomicString& mark_name,
DoubleOrPerformanceMarkOptions& start_time_or_mark_options,
ExceptionState&);
void clearMarks(const AtomicString& mark_name);
// This enum is used to index different possible strings for for UMA enum
// histogram. New enum values can be added, but existing enums must never be
// renumbered or deleted and reused.
// This enum should be consistent with MeasureParameterType
// in tools/metrics/histograms/enums.xml.
enum class MeasureParameterType {
kObjectObject = 0,
// 1 to 8, 13 to 25 are navigation-timing types.
kUnloadEventStart = 1,
kUnloadEventEnd = 2,
kDomInteractive = 3,
kDomContentLoadedEventStart = 4,
kDomContentLoadedEventEnd = 5,
kDomComplete = 6,
kLoadEventStart = 7,
kLoadEventEnd = 8,
kOther = 9,
kUndefinedOrNull = 10,
kNumber = 11,
kUnprovided = 12,
kNavigationStart = 13,
kRedirectStart = 14,
kRedirectEnd = 15,
kFetchStart = 16,
kDomainLookupStart = 17,
kDomainLookupEnd = 18,
kConnectStart = 19,
kConnectEnd = 20,
kSecureConnectionStart = 21,
kRequestStart = 22,
kResponseStart = 23,
kResponseEnd = 24,
kDomLoading = 25,
kMaxValue = kDomLoading
};
PerformanceMeasure* measure(ScriptState*,
const AtomicString& measure_name,
ExceptionState&);
PerformanceMeasure* measure(
ScriptState*,
const AtomicString& measure_name,
const StringOrDoubleOrPerformanceMeasureOptions& start_or_options,
ExceptionState&);
PerformanceMeasure* measure(
ScriptState*,
const AtomicString& measure_name,
const StringOrDoubleOrPerformanceMeasureOptions& start_or_options,
const StringOrDouble& end,
ExceptionState&);
void clearMeasures(const AtomicString& measure_name);
void UnregisterPerformanceObserver(PerformanceObserver&);
void RegisterPerformanceObserver(PerformanceObserver&);
void UpdatePerformanceObserverFilterOptions();
void ActivateObserver(PerformanceObserver&);
void ResumeSuspendedObservers();
bool HasObserverFor(PerformanceEntry::EntryType) const;
static bool AllowsTimingRedirect(const Vector<ResourceResponse>&,
const ResourceResponse&,
const SecurityOrigin&,
ExecutionContext*);
ScriptValue toJSONForBinding(ScriptState*) const;
void Trace(blink::Visitor*) override;
private:
static bool PassesTimingAllowCheck(const ResourceResponse&,
const SecurityOrigin&,
const AtomicString&,
ExecutionContext*);
void AddPaintTiming(PerformancePaintTiming::PaintType, TimeTicks start_time);
PerformanceMeasure* MeasureInternal(
ScriptState*,
const AtomicString& measure_name,
const StringOrDoubleOrPerformanceMeasureOptions& start,
const StringOrDouble& end,
ExceptionState&);
PerformanceMeasure* MeasureWithDetail(ScriptState*,
const AtomicString& measure_name,
const StringOrDouble& start,
const StringOrDouble& end,
const ScriptValue& detail,
ExceptionState&);
void CopySecondaryBuffer();
protected:
Performance(TimeTicks time_origin,
scoped_refptr<base::SingleThreadTaskRunner>);
// Expect WindowPerformance to override this method,
// WorkerPerformance doesn't have to override this.
virtual PerformanceNavigationTiming* CreateNavigationTimingInstance() {
return nullptr;
}
bool CanAddResourceTimingEntry();
void FireResourceTimingBufferFull(TimerBase*);
void NotifyObserversOfEntry(PerformanceEntry&) const;
void NotifyObserversOfEntries(PerformanceEntryVector&);
void DeliverObservationsTimerFired(TimerBase*);
virtual void BuildJSONValue(V8ObjectBuilder&) const;
PerformanceEntryVector resource_timing_buffer_;
// The secondary RT buffer, used to store incoming entries after the main
// buffer is full, until the resourcetimingbufferfull event fires.
PerformanceEntryDeque resource_timing_secondary_buffer_;
unsigned resource_timing_buffer_size_limit_;
// A flag indicating that the buffer became full, the appropriate event was
// queued, but haven't yet fired.
bool resource_timing_buffer_full_event_pending_ = false;
PerformanceEntryVector event_timing_buffer_;
unsigned event_timing_buffer_max_size_;
PerformanceEntryVector element_timing_buffer_;
unsigned element_timing_buffer_max_size_;
Member<PerformanceEntry> navigation_timing_;
TraceWrapperMember<UserTiming> user_timing_;
Member<PerformanceEntry> first_paint_timing_;
Member<PerformanceEntry> first_contentful_paint_timing_;
Member<PerformanceEventTiming> first_input_timing_;
TimeTicks time_origin_;
PerformanceEntryTypeMask observer_filter_options_;
HeapLinkedHashSet<TraceWrapperMember<PerformanceObserver>> observers_;
HeapLinkedHashSet<Member<PerformanceObserver>> active_observers_;
HeapLinkedHashSet<Member<PerformanceObserver>> suspended_observers_;
scoped_refptr<base::SingleThreadTaskRunner> task_runner_;
TaskRunnerTimer<Performance> deliver_observations_timer_;
TaskRunnerTimer<Performance> resource_timing_buffer_full_timer_;
};
} // namespace blink
#endif // THIRD_PARTY_BLINK_RENDERER_CORE_TIMING_PERFORMANCE_H_