summaryrefslogtreecommitdiffstats
path: root/toolkit/components/telemetry/core
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-28 14:29:10 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-28 14:29:10 +0000
commit2aa4a82499d4becd2284cdb482213d541b8804dd (patch)
treeb80bf8bf13c3766139fbacc530efd0dd9d54394c /toolkit/components/telemetry/core
parentInitial commit. (diff)
downloadfirefox-2aa4a82499d4becd2284cdb482213d541b8804dd.tar.xz
firefox-2aa4a82499d4becd2284cdb482213d541b8804dd.zip
Adding upstream version 86.0.1.upstream/86.0.1upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'toolkit/components/telemetry/core')
-rw-r--r--toolkit/components/telemetry/core/EventInfo.h57
-rw-r--r--toolkit/components/telemetry/core/ScalarInfo.h94
-rw-r--r--toolkit/components/telemetry/core/Stopwatch.cpp749
-rw-r--r--toolkit/components/telemetry/core/Stopwatch.h84
-rw-r--r--toolkit/components/telemetry/core/Telemetry.cpp2076
-rw-r--r--toolkit/components/telemetry/core/Telemetry.h586
-rw-r--r--toolkit/components/telemetry/core/TelemetryCommon.cpp208
-rw-r--r--toolkit/components/telemetry/core/TelemetryCommon.h193
-rw-r--r--toolkit/components/telemetry/core/TelemetryEvent.cpp1390
-rw-r--r--toolkit/components/telemetry/core/TelemetryEvent.h71
-rw-r--r--toolkit/components/telemetry/core/TelemetryHistogram.cpp3658
-rw-r--r--toolkit/components/telemetry/core/TelemetryHistogram.h120
-rw-r--r--toolkit/components/telemetry/core/TelemetryOrigin.cpp625
-rw-r--r--toolkit/components/telemetry/core/TelemetryOrigin.h45
-rw-r--r--toolkit/components/telemetry/core/TelemetryOriginData.inc2488
-rw-r--r--toolkit/components/telemetry/core/TelemetryScalar.cpp4202
-rw-r--r--toolkit/components/telemetry/core/TelemetryScalar.h130
-rw-r--r--toolkit/components/telemetry/core/TelemetryUserInteraction.cpp103
-rw-r--r--toolkit/components/telemetry/core/TelemetryUserInteraction.h18
-rw-r--r--toolkit/components/telemetry/core/UserInteractionInfo.h30
-rw-r--r--toolkit/components/telemetry/core/components.conf21
-rw-r--r--toolkit/components/telemetry/core/ipc/TelemetryComms.h416
-rw-r--r--toolkit/components/telemetry/core/ipc/TelemetryIPC.cpp59
-rw-r--r--toolkit/components/telemetry/core/ipc/TelemetryIPC.h116
-rw-r--r--toolkit/components/telemetry/core/ipc/TelemetryIPCAccumulator.cpp345
-rw-r--r--toolkit/components/telemetry/core/ipc/TelemetryIPCAccumulator.h54
-rw-r--r--toolkit/components/telemetry/core/nsITelemetry.idl702
27 files changed, 18640 insertions, 0 deletions
diff --git a/toolkit/components/telemetry/core/EventInfo.h b/toolkit/components/telemetry/core/EventInfo.h
new file mode 100644
index 0000000000..b80f85af92
--- /dev/null
+++ b/toolkit/components/telemetry/core/EventInfo.h
@@ -0,0 +1,57 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryEventInfo_h__
+#define TelemetryEventInfo_h__
+
+#include "TelemetryCommon.h"
+
+// This module is internal to Telemetry. The structures here hold data that
+// describe events.
+// It should only be used by TelemetryEventData.h and TelemetryEvent.cpp.
+//
+// For the public interface to Telemetry functionality, see Telemetry.h.
+
+namespace {
+
+struct CommonEventInfo {
+ // Indices for the category and expiration strings.
+ uint32_t category_offset;
+ uint32_t expiration_version_offset;
+
+ // The index and count for the extra key offsets in the extra table.
+ uint32_t extra_index;
+ uint32_t extra_count;
+
+ // The dataset this event is recorded in.
+ uint32_t dataset;
+
+ // Which processes to record this event in.
+ mozilla::Telemetry::Common::RecordedProcessType record_in_processes;
+
+ // Which products to record this event on.
+ mozilla::Telemetry::Common::SupportedProduct products;
+
+ // Convenience functions for accessing event strings.
+ const nsDependentCString expiration_version() const;
+ const nsDependentCString category() const;
+ const nsDependentCString extra_key(uint32_t index) const;
+};
+
+struct EventInfo {
+ // The corresponding CommonEventInfo.
+ const CommonEventInfo& common_info;
+
+ // Indices for the method & object strings.
+ uint32_t method_offset;
+ uint32_t object_offset;
+
+ const nsDependentCString method() const;
+ const nsDependentCString object() const;
+};
+
+} // namespace
+
+#endif // TelemetryEventInfo_h__
diff --git a/toolkit/components/telemetry/core/ScalarInfo.h b/toolkit/components/telemetry/core/ScalarInfo.h
new file mode 100644
index 0000000000..125ea7e88e
--- /dev/null
+++ b/toolkit/components/telemetry/core/ScalarInfo.h
@@ -0,0 +1,94 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryScalarInfo_h__
+#define TelemetryScalarInfo_h__
+
+#include "TelemetryCommon.h"
+
+// This module is internal to Telemetry. It defines a structure that holds the
+// scalar info. It should only be used by TelemetryScalarData.h automatically
+// generated file and TelemetryScalar.cpp. This should not be used anywhere
+// else. For the public interface to Telemetry functionality, see Telemetry.h.
+
+namespace {
+
+/**
+ * Base scalar information, common to both "static" and dynamic scalars.
+ */
+struct BaseScalarInfo {
+ uint32_t kind;
+ uint32_t dataset;
+ mozilla::Telemetry::Common::RecordedProcessType record_in_processes;
+ bool keyed;
+ uint32_t key_count;
+ uint32_t key_offset;
+ mozilla::Telemetry::Common::SupportedProduct products;
+ bool builtin;
+
+ constexpr BaseScalarInfo(
+ uint32_t aKind, uint32_t aDataset,
+ mozilla::Telemetry::Common::RecordedProcessType aRecordInProcess,
+ bool aKeyed, uint32_t aKeyCount, uint32_t aKeyOffset,
+ mozilla::Telemetry::Common::SupportedProduct aProducts,
+ bool aBuiltin = true)
+ : kind(aKind),
+ dataset(aDataset),
+ record_in_processes(aRecordInProcess),
+ keyed(aKeyed),
+ key_count(aKeyCount),
+ key_offset(aKeyOffset),
+ products(aProducts),
+ builtin(aBuiltin) {}
+ virtual ~BaseScalarInfo() = default;
+
+ virtual const char* name() const = 0;
+ virtual const char* expiration() const = 0;
+
+ virtual uint32_t storeOffset() const = 0;
+ virtual uint32_t storeCount() const = 0;
+};
+
+/**
+ * "Static" scalar definition: these are the ones riding
+ * the trains.
+ */
+struct ScalarInfo : BaseScalarInfo {
+ uint32_t name_offset;
+ uint32_t expiration_offset;
+ uint32_t store_count;
+ uint16_t store_offset;
+
+ // In order to cleanly support dynamic scalars in TelemetryScalar.cpp, we need
+ // to use virtual functions for |name| and |expiration|, as they won't be
+ // looked up in the static tables in that case. However, using virtual
+ // functions makes |ScalarInfo| non-aggregate and prevents from using
+ // aggregate initialization (curly brackets) in the generated
+ // TelemetryScalarData.h. To work around this problem we define a constructor
+ // that takes the exact number of parameters we need.
+ constexpr ScalarInfo(
+ uint32_t aKind, uint32_t aNameOffset, uint32_t aExpirationOffset,
+ uint32_t aDataset,
+ mozilla::Telemetry::Common::RecordedProcessType aRecordInProcess,
+ bool aKeyed, uint32_t aKeyCount, uint32_t aKeyOffset,
+ mozilla::Telemetry::Common::SupportedProduct aProducts,
+ uint32_t aStoreCount, uint32_t aStoreOffset)
+ : BaseScalarInfo(aKind, aDataset, aRecordInProcess, aKeyed, aKeyCount,
+ aKeyOffset, aProducts),
+ name_offset(aNameOffset),
+ expiration_offset(aExpirationOffset),
+ store_count(aStoreCount),
+ store_offset(aStoreOffset) {}
+
+ const char* name() const override;
+ const char* expiration() const override;
+
+ uint32_t storeOffset() const override { return store_offset; };
+ uint32_t storeCount() const override { return store_count; };
+};
+
+} // namespace
+
+#endif // TelemetryScalarInfo_h__
diff --git a/toolkit/components/telemetry/core/Stopwatch.cpp b/toolkit/components/telemetry/core/Stopwatch.cpp
new file mode 100644
index 0000000000..186f70c213
--- /dev/null
+++ b/toolkit/components/telemetry/core/Stopwatch.cpp
@@ -0,0 +1,749 @@
+/* -*- 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 "mozilla/telemetry/Stopwatch.h"
+
+#include "TelemetryHistogram.h"
+#include "TelemetryUserInteraction.h"
+
+#include "mozilla/dom/ScriptSettings.h"
+#include "mozilla/BackgroundHangMonitor.h"
+#include "mozilla/ClearOnShutdown.h"
+#include "mozilla/HangAnnotations.h"
+#include "mozilla/DataMutex.h"
+#include "mozilla/TimeStamp.h"
+#include "GeckoProfiler.h"
+#include "nsHashKeys.h"
+#include "nsContentUtils.h"
+#include "nsPrintfCString.h"
+#include "nsQueryObject.h"
+#include "nsRefPtrHashtable.h"
+#include "nsString.h"
+#include "xpcpublic.h"
+
+using mozilla::DataMutex;
+using mozilla::dom::AutoJSAPI;
+
+#define USER_INTERACTION_VALUE_MAX_LENGTH 50 // bytes
+
+static inline nsQueryObject<nsISupports> do_QueryReflector(
+ JSObject* aReflector) {
+ // None of the types we query to are implemented by Window or Location.
+ nsCOMPtr<nsISupports> reflector = xpc::ReflectorToISupportsStatic(aReflector);
+ return do_QueryObject(reflector);
+}
+
+static inline nsQueryObject<nsISupports> do_QueryReflector(
+ const JS::Value& aReflector) {
+ return do_QueryReflector(&aReflector.toObject());
+}
+
+static void LogError(JSContext* aCx, const nsCString& aMessage) {
+ // This is a bit of a hack to report an error with the current JS caller's
+ // location. We create an AutoJSAPI object bound to the current caller
+ // global, report a JS error, and then let AutoJSAPI's destructor report the
+ // error.
+ //
+ // Unfortunately, there isn't currently a more straightforward way to do
+ // this from C++.
+ JS::Rooted<JSObject*> global(aCx, JS::CurrentGlobalOrNull(aCx));
+
+ AutoJSAPI jsapi;
+ if (jsapi.Init(global)) {
+ JS_ReportErrorUTF8(jsapi.cx(), "%s", aMessage.get());
+ }
+}
+
+namespace mozilla::telemetry {
+
+class Timer final : public mozilla::LinkedListElement<RefPtr<Timer>> {
+ public:
+ NS_INLINE_DECL_REFCOUNTING(Timer)
+
+ Timer() = default;
+
+ void Start(bool aInSeconds) {
+ mStartTime = TimeStamp::Now();
+ mInSeconds = aInSeconds;
+ }
+
+ bool Started() { return !mStartTime.IsNull(); }
+
+ uint32_t Elapsed() {
+ auto delta = TimeStamp::Now() - mStartTime;
+ return mInSeconds ? delta.ToSeconds() : delta.ToMilliseconds();
+ }
+
+ TimeStamp& StartTime() { return mStartTime; }
+
+ bool& InSeconds() { return mInSeconds; }
+
+ /**
+ * Note that these values will want to be read from the
+ * BackgroundHangAnnotator thread. Callers should take a lock
+ * on Timers::mBHRAnnotationTimers before calling this.
+ */
+ void SetBHRAnnotation(const nsAString& aBHRAnnotationKey,
+ const nsACString& aBHRAnnotationValue) {
+ mBHRAnnotationKey = aBHRAnnotationKey;
+ mBHRAnnotationValue = aBHRAnnotationValue;
+ }
+
+ const nsString& GetBHRAnnotationKey() const { return mBHRAnnotationKey; }
+ const nsCString& GetBHRAnnotationValue() const { return mBHRAnnotationValue; }
+
+ private:
+ ~Timer() = default;
+ TimeStamp mStartTime{};
+ nsString mBHRAnnotationKey;
+ nsCString mBHRAnnotationValue;
+ bool mInSeconds;
+};
+
+#define TIMER_KEYS_IID \
+ { \
+ 0xef707178, 0x1544, 0x46e2, { \
+ 0xa3, 0xf5, 0x98, 0x38, 0xba, 0x60, 0xfd, 0x8f \
+ } \
+ }
+
+class TimerKeys final : public nsISupports {
+ public:
+ NS_DECL_ISUPPORTS
+ NS_DECLARE_STATIC_IID_ACCESSOR(TIMER_KEYS_IID)
+
+ Timer* Get(const nsAString& aKey, bool aCreate = true);
+
+ already_AddRefed<Timer> GetAndDelete(const nsAString& aKey) {
+ RefPtr<Timer> timer;
+ mTimers.Remove(aKey, getter_AddRefs(timer));
+ return timer.forget();
+ }
+
+ bool Delete(const nsAString& aKey) { return mTimers.Remove(aKey); }
+
+ private:
+ ~TimerKeys() = default;
+
+ nsRefPtrHashtable<nsStringHashKey, Timer> mTimers;
+};
+
+NS_DEFINE_STATIC_IID_ACCESSOR(TimerKeys, TIMER_KEYS_IID)
+
+NS_IMPL_ISUPPORTS(TimerKeys, TimerKeys)
+
+Timer* TimerKeys::Get(const nsAString& aKey, bool aCreate) {
+ if (aCreate) {
+ RefPtr<Timer>& timer = mTimers.GetOrInsert(aKey);
+ if (!timer) {
+ timer = new Timer();
+ }
+ return timer;
+ }
+ return mTimers.GetWeak(aKey);
+}
+
+class Timers final : public BackgroundHangAnnotator {
+ public:
+ Timers();
+
+ static Timers& Singleton();
+
+ NS_INLINE_DECL_REFCOUNTING(Timers)
+
+ JSObject* Get(JSContext* aCx, const nsAString& aHistogram,
+ bool aCreate = true);
+
+ TimerKeys* Get(JSContext* aCx, const nsAString& aHistogram,
+ JS::HandleObject aObj, bool aCreate = true);
+
+ Timer* Get(JSContext* aCx, const nsAString& aHistogram, JS::HandleObject aObj,
+ const nsAString& aKey, bool aCreate = true);
+
+ already_AddRefed<Timer> GetAndDelete(JSContext* aCx,
+ const nsAString& aHistogram,
+ JS::HandleObject aObj,
+ const nsAString& aKey);
+
+ bool Delete(JSContext* aCx, const nsAString& aHistogram,
+ JS::HandleObject aObj, const nsAString& aKey);
+
+ int32_t TimeElapsed(JSContext* aCx, const nsAString& aHistogram,
+ JS::HandleObject aObj, const nsAString& aKey,
+ bool aCanceledOkay = false);
+
+ bool Start(JSContext* aCx, const nsAString& aHistogram, JS::HandleObject aObj,
+ const nsAString& aKey, bool aInSeconds = false);
+
+ int32_t Finish(JSContext* aCx, const nsAString& aHistogram,
+ JS::HandleObject aObj, const nsAString& aKey,
+ bool aCanceledOkay = false);
+
+ bool& SuppressErrors() { return mSuppressErrors; }
+
+ bool StartUserInteraction(JSContext* aCx, const nsAString& aUserInteraction,
+ const nsACString& aValue, JS::HandleObject aObj);
+ bool RunningUserInteraction(JSContext* aCx, const nsAString& aUserInteraction,
+ JS::HandleObject aObj);
+ bool UpdateUserInteraction(JSContext* aCx, const nsAString& aUserInteraction,
+ const nsACString& aValue, JS::HandleObject aObj);
+ bool FinishUserInteraction(JSContext* aCx, const nsAString& aUserInteraction,
+ JS::HandleObject aObj,
+ const dom::Optional<nsACString>& aAdditionalText);
+ bool CancelUserInteraction(JSContext* aCx, const nsAString& aUserInteraction,
+ JS::HandleObject aObj);
+
+ void AnnotateHang(BackgroundHangAnnotations& aAnnotations) final;
+
+ private:
+ ~Timers();
+
+ JS::PersistentRooted<JSObject*> mTimers;
+ DataMutex<mozilla::LinkedList<RefPtr<Timer>>> mBHRAnnotationTimers;
+ bool mSuppressErrors = false;
+
+ static StaticRefPtr<Timers> sSingleton;
+};
+
+StaticRefPtr<Timers> Timers::sSingleton;
+
+/* static */ Timers& Timers::Singleton() {
+ if (!sSingleton) {
+ sSingleton = new Timers();
+ ClearOnShutdown(&sSingleton);
+ }
+ return *sSingleton;
+}
+
+Timers::Timers()
+ : mTimers(dom::RootingCx()), mBHRAnnotationTimers("BHRAnnotationTimers") {
+ AutoJSAPI jsapi;
+ MOZ_ALWAYS_TRUE(jsapi.Init(xpc::PrivilegedJunkScope()));
+
+ mTimers = JS::NewMapObject(jsapi.cx());
+ MOZ_RELEASE_ASSERT(mTimers);
+
+ BackgroundHangMonitor::RegisterAnnotator(*this);
+}
+
+Timers::~Timers() {
+ // We use a scope here to prevent a deadlock with the mutex that locks
+ // inside of ::UnregisterAnnotator.
+ {
+ auto annotationTimers = mBHRAnnotationTimers.Lock();
+ annotationTimers->clear();
+ }
+ BackgroundHangMonitor::UnregisterAnnotator(*this);
+}
+
+JSObject* Timers::Get(JSContext* aCx, const nsAString& aHistogram,
+ bool aCreate) {
+ JSAutoRealm ar(aCx, mTimers);
+
+ JS::RootedValue histogram(aCx);
+ JS::RootedValue objs(aCx);
+
+ if (!xpc::NonVoidStringToJsval(aCx, aHistogram, &histogram) ||
+ !JS::MapGet(aCx, mTimers, histogram, &objs)) {
+ return nullptr;
+ }
+ if (!objs.isObject()) {
+ if (aCreate) {
+ objs = JS::ObjectOrNullValue(JS::NewWeakMapObject(aCx));
+ }
+ if (!objs.isObject() || !JS::MapSet(aCx, mTimers, histogram, objs)) {
+ return nullptr;
+ }
+ }
+
+ return &objs.toObject();
+}
+
+TimerKeys* Timers::Get(JSContext* aCx, const nsAString& aHistogram,
+ JS::HandleObject aObj, bool aCreate) {
+ JSAutoRealm ar(aCx, mTimers);
+
+ JS::RootedObject objs(aCx, Get(aCx, aHistogram, aCreate));
+ if (!objs) {
+ return nullptr;
+ }
+
+ // If no object is passed, use mTimers as a stand-in for a null object
+ // (which cannot be used as a weak map key).
+ JS::RootedObject obj(aCx, aObj ? aObj : mTimers);
+ if (!JS_WrapObject(aCx, &obj)) {
+ return nullptr;
+ }
+
+ RefPtr<TimerKeys> keys;
+ JS::RootedValue keysObj(aCx);
+ if (!JS::GetWeakMapEntry(aCx, objs, obj, &keysObj)) {
+ return nullptr;
+ }
+ if (!keysObj.isObject()) {
+ if (aCreate) {
+ keys = new TimerKeys();
+ Unused << nsContentUtils::WrapNative(aCx, keys, &keysObj);
+ }
+ if (!keysObj.isObject() || !JS::SetWeakMapEntry(aCx, objs, obj, keysObj)) {
+ return nullptr;
+ }
+ }
+
+ keys = do_QueryReflector(keysObj);
+ return keys;
+}
+
+Timer* Timers::Get(JSContext* aCx, const nsAString& aHistogram,
+ JS::HandleObject aObj, const nsAString& aKey, bool aCreate) {
+ if (RefPtr<TimerKeys> keys = Get(aCx, aHistogram, aObj, aCreate)) {
+ return keys->Get(aKey, aCreate);
+ }
+ return nullptr;
+}
+
+already_AddRefed<Timer> Timers::GetAndDelete(JSContext* aCx,
+ const nsAString& aHistogram,
+ JS::HandleObject aObj,
+ const nsAString& aKey) {
+ if (RefPtr<TimerKeys> keys = Get(aCx, aHistogram, aObj, false)) {
+ return keys->GetAndDelete(aKey);
+ }
+ return nullptr;
+}
+
+bool Timers::Delete(JSContext* aCx, const nsAString& aHistogram,
+ JS::HandleObject aObj, const nsAString& aKey) {
+ if (RefPtr<TimerKeys> keys = Get(aCx, aHistogram, aObj, false)) {
+ return keys->Delete(aKey);
+ }
+ return false;
+}
+
+int32_t Timers::TimeElapsed(JSContext* aCx, const nsAString& aHistogram,
+ JS::HandleObject aObj, const nsAString& aKey,
+ bool aCanceledOkay) {
+ RefPtr<Timer> timer = Get(aCx, aHistogram, aObj, aKey, false);
+ if (!timer) {
+ if (!aCanceledOkay && !mSuppressErrors) {
+ LogError(aCx, nsPrintfCString(
+ "TelemetryStopwatch: requesting elapsed time for "
+ "nonexisting stopwatch. Histogram: \"%s\", key: \"%s\"",
+ NS_ConvertUTF16toUTF8(aHistogram).get(),
+ NS_ConvertUTF16toUTF8(aKey).get()));
+ }
+ return -1;
+ }
+
+ return timer->Elapsed();
+}
+
+bool Timers::Start(JSContext* aCx, const nsAString& aHistogram,
+ JS::HandleObject aObj, const nsAString& aKey,
+ bool aInSeconds) {
+ if (RefPtr<Timer> timer = Get(aCx, aHistogram, aObj, aKey)) {
+ if (timer->Started()) {
+ if (!mSuppressErrors) {
+ LogError(aCx,
+ nsPrintfCString(
+ "TelemetryStopwatch: key \"%s\" was already initialized",
+ NS_ConvertUTF16toUTF8(aHistogram).get()));
+ }
+ Delete(aCx, aHistogram, aObj, aKey);
+ } else {
+ timer->Start(aInSeconds);
+ return true;
+ }
+ }
+ return false;
+}
+
+int32_t Timers::Finish(JSContext* aCx, const nsAString& aHistogram,
+ JS::HandleObject aObj, const nsAString& aKey,
+ bool aCanceledOkay) {
+ RefPtr<Timer> timer = GetAndDelete(aCx, aHistogram, aObj, aKey);
+ if (!timer) {
+ if (!aCanceledOkay && !mSuppressErrors) {
+ LogError(aCx, nsPrintfCString(
+ "TelemetryStopwatch: finishing nonexisting stopwatch. "
+ "Histogram: \"%s\", key: \"%s\"",
+ NS_ConvertUTF16toUTF8(aHistogram).get(),
+ NS_ConvertUTF16toUTF8(aKey).get()));
+ }
+ return -1;
+ }
+
+ int32_t delta = timer->Elapsed();
+ NS_ConvertUTF16toUTF8 histogram(aHistogram);
+ nsresult rv;
+ if (!aKey.IsVoid()) {
+ NS_ConvertUTF16toUTF8 key(aKey);
+ rv = TelemetryHistogram::Accumulate(histogram.get(), key, delta);
+ } else {
+ rv = TelemetryHistogram::Accumulate(histogram.get(), delta);
+ }
+#ifdef MOZ_GECKO_PROFILER
+ nsCString markerText = histogram;
+ if (!aKey.IsVoid()) {
+ markerText.AppendLiteral(":");
+ markerText.Append(NS_ConvertUTF16toUTF8(aKey));
+ }
+ PROFILER_MARKER_TEXT("TelemetryStopwatch", OTHER,
+ MarkerTiming::IntervalUntilNowFrom(timer->StartTime()),
+ markerText);
+#endif
+ if (NS_FAILED(rv) && rv != NS_ERROR_NOT_AVAILABLE && !mSuppressErrors) {
+ LogError(aCx, nsPrintfCString(
+ "TelemetryStopwatch: failed to update the Histogram "
+ "\"%s\", using key: \"%s\"",
+ NS_ConvertUTF16toUTF8(aHistogram).get(),
+ NS_ConvertUTF16toUTF8(aKey).get()));
+ }
+ return NS_SUCCEEDED(rv) ? delta : -1;
+}
+
+bool Timers::StartUserInteraction(JSContext* aCx,
+ const nsAString& aUserInteraction,
+ const nsACString& aValue,
+ JS::HandleObject aObj) {
+ MOZ_ASSERT(NS_IsMainThread());
+
+ // Ensure that this ID maps to a UserInteraction that can be recorded
+ // for this product.
+ if (!TelemetryUserInteraction::CanRecord(aUserInteraction)) {
+ if (!mSuppressErrors) {
+ LogError(aCx, nsPrintfCString(
+ "UserInteraction with name \"%s\" cannot be recorded.",
+ NS_ConvertUTF16toUTF8(aUserInteraction).get()));
+ }
+ return false;
+ }
+
+ if (aValue.Length() > USER_INTERACTION_VALUE_MAX_LENGTH) {
+ if (!mSuppressErrors) {
+ LogError(aCx,
+ nsPrintfCString(
+ "UserInteraction with name \"%s\" cannot be recorded with"
+ "a value of length greater than %d (%s)",
+ NS_ConvertUTF16toUTF8(aUserInteraction).get(),
+ USER_INTERACTION_VALUE_MAX_LENGTH,
+ PromiseFlatCString(aValue).get()));
+ }
+ return false;
+ }
+
+ if (RefPtr<Timer> timer = Get(aCx, aUserInteraction, aObj, VoidString())) {
+ auto annotationTimers = mBHRAnnotationTimers.Lock();
+
+ if (timer->Started()) {
+ if (!mSuppressErrors) {
+ LogError(aCx,
+ nsPrintfCString(
+ "UserInteraction with name \"%s\" was already initialized",
+ NS_ConvertUTF16toUTF8(aUserInteraction).get()));
+ }
+ timer->removeFrom(*annotationTimers);
+ Delete(aCx, aUserInteraction, aObj, VoidString());
+ timer = Get(aCx, aUserInteraction, aObj, VoidString());
+
+ nsAutoString clobberText(aUserInteraction);
+ clobberText.AppendLiteral(u" (clobbered)");
+ timer->SetBHRAnnotation(clobberText, aValue);
+ } else {
+ timer->SetBHRAnnotation(aUserInteraction, aValue);
+ }
+
+ annotationTimers->insertBack(timer);
+ timer->Start(false);
+ return true;
+ }
+ return false;
+}
+
+bool Timers::RunningUserInteraction(JSContext* aCx,
+ const nsAString& aUserInteraction,
+ JS::HandleObject aObj) {
+ if (RefPtr<Timer> timer =
+ Get(aCx, aUserInteraction, aObj, VoidString(), false /* aCreate */)) {
+ return timer->Started();
+ }
+ return false;
+}
+
+bool Timers::UpdateUserInteraction(JSContext* aCx,
+ const nsAString& aUserInteraction,
+ const nsACString& aValue,
+ JS::HandleObject aObj) {
+ MOZ_ASSERT(NS_IsMainThread());
+
+ // Ensure that this ID maps to a UserInteraction that can be recorded
+ // for this product.
+ if (!TelemetryUserInteraction::CanRecord(aUserInteraction)) {
+ if (!mSuppressErrors) {
+ LogError(aCx, nsPrintfCString(
+ "UserInteraction with name \"%s\" cannot be recorded.",
+ NS_ConvertUTF16toUTF8(aUserInteraction).get()));
+ }
+ return false;
+ }
+
+ auto lock = mBHRAnnotationTimers.Lock();
+ if (RefPtr<Timer> timer = Get(aCx, aUserInteraction, aObj, VoidString())) {
+ if (!timer->Started()) {
+ if (!mSuppressErrors) {
+ LogError(aCx, nsPrintfCString(
+ "UserInteraction with id \"%s\" was not initialized",
+ NS_ConvertUTF16toUTF8(aUserInteraction).get()));
+ }
+ return false;
+ }
+ timer->SetBHRAnnotation(aUserInteraction, aValue);
+ return true;
+ }
+ return false;
+}
+
+bool Timers::FinishUserInteraction(
+ JSContext* aCx, const nsAString& aUserInteraction, JS::HandleObject aObj,
+ const dom::Optional<nsACString>& aAdditionalText) {
+ MOZ_ASSERT(NS_IsMainThread());
+
+ // Ensure that this ID maps to a UserInteraction that can be recorded
+ // for this product.
+ if (!TelemetryUserInteraction::CanRecord(aUserInteraction)) {
+ if (!mSuppressErrors) {
+ LogError(aCx, nsPrintfCString(
+ "UserInteraction with id \"%s\" cannot be recorded.",
+ NS_ConvertUTF16toUTF8(aUserInteraction).get()));
+ }
+ return false;
+ }
+
+ RefPtr<Timer> timer = GetAndDelete(aCx, aUserInteraction, aObj, VoidString());
+ if (!timer) {
+ if (!mSuppressErrors) {
+ LogError(aCx, nsPrintfCString(
+ "UserInteraction: finishing nonexisting stopwatch. "
+ "name: \"%s\"",
+ NS_ConvertUTF16toUTF8(aUserInteraction).get()));
+ }
+ return false;
+ }
+
+#ifdef MOZ_GECKO_PROFILER
+ if (profiler_can_accept_markers()) {
+ nsAutoCString markerText(timer->GetBHRAnnotationValue());
+ if (aAdditionalText.WasPassed()) {
+ markerText.Append(",");
+ markerText.Append(aAdditionalText.Value());
+ }
+
+ PROFILER_MARKER_TEXT(NS_ConvertUTF16toUTF8(aUserInteraction), OTHER,
+ MarkerTiming::IntervalUntilNowFrom(timer->StartTime()),
+ markerText);
+ }
+#endif
+
+ // The Timer will be held alive by the RefPtr that's still in the LinkedList,
+ // so the automatic removal from the LinkedList from the LinkedListElement
+ // destructor will not occur. We must remove it manually from the LinkedList
+ // instead.
+ {
+ auto annotationTimers = mBHRAnnotationTimers.Lock();
+ timer->removeFrom(*annotationTimers);
+ }
+
+ return true;
+}
+
+bool Timers::CancelUserInteraction(JSContext* aCx,
+ const nsAString& aUserInteraction,
+ JS::HandleObject aObj) {
+ MOZ_ASSERT(NS_IsMainThread());
+
+ // Ensure that this ID maps to a UserInteraction that can be recorded
+ // for this product.
+ if (!TelemetryUserInteraction::CanRecord(aUserInteraction)) {
+ if (!mSuppressErrors) {
+ LogError(aCx, nsPrintfCString(
+ "UserInteraction with id \"%s\" cannot be recorded.",
+ NS_ConvertUTF16toUTF8(aUserInteraction).get()));
+ }
+ return false;
+ }
+
+ RefPtr<Timer> timer = GetAndDelete(aCx, aUserInteraction, aObj, VoidString());
+ if (!timer) {
+ if (!mSuppressErrors) {
+ LogError(aCx, nsPrintfCString(
+ "UserInteraction: cancelling nonexisting stopwatch. "
+ "name: \"%s\"",
+ NS_ConvertUTF16toUTF8(aUserInteraction).get()));
+ }
+ return false;
+ }
+
+ // The Timer will be held alive by the RefPtr that's still in the LinkedList,
+ // so the automatic removal from the LinkedList from the LinkedListElement
+ // destructor will not occur. We must remove it manually from the LinkedList
+ // instead.
+ {
+ auto annotationTimers = mBHRAnnotationTimers.Lock();
+ timer->removeFrom(*annotationTimers);
+ }
+
+ return true;
+}
+
+void Timers::AnnotateHang(mozilla::BackgroundHangAnnotations& aAnnotations) {
+ auto annotationTimers = mBHRAnnotationTimers.Lock();
+ for (Timer* bhrAnnotationTimer : *annotationTimers) {
+ aAnnotations.AddAnnotation(bhrAnnotationTimer->GetBHRAnnotationKey(),
+ bhrAnnotationTimer->GetBHRAnnotationValue());
+ }
+}
+
+/* static */
+bool Stopwatch::Start(const dom::GlobalObject& aGlobal,
+ const nsAString& aHistogram, JS::Handle<JSObject*> aObj,
+ const dom::TelemetryStopwatchOptions& aOptions) {
+ return StartKeyed(aGlobal, aHistogram, VoidString(), aObj, aOptions);
+}
+/* static */
+bool Stopwatch::StartKeyed(const dom::GlobalObject& aGlobal,
+ const nsAString& aHistogram, const nsAString& aKey,
+ JS::Handle<JSObject*> aObj,
+ const dom::TelemetryStopwatchOptions& aOptions) {
+ return Timers::Singleton().Start(aGlobal.Context(), aHistogram, aObj, aKey,
+ aOptions.mInSeconds);
+}
+
+/* static */
+bool Stopwatch::Running(const dom::GlobalObject& aGlobal,
+ const nsAString& aHistogram,
+ JS::Handle<JSObject*> aObj) {
+ return RunningKeyed(aGlobal, aHistogram, VoidString(), aObj);
+}
+
+/* static */
+bool Stopwatch::RunningKeyed(const dom::GlobalObject& aGlobal,
+ const nsAString& aHistogram, const nsAString& aKey,
+ JS::Handle<JSObject*> aObj) {
+ return TimeElapsedKeyed(aGlobal, aHistogram, aKey, aObj, true) != -1;
+}
+
+/* static */
+int32_t Stopwatch::TimeElapsed(const dom::GlobalObject& aGlobal,
+ const nsAString& aHistogram,
+ JS::Handle<JSObject*> aObj, bool aCanceledOkay) {
+ return TimeElapsedKeyed(aGlobal, aHistogram, VoidString(), aObj,
+ aCanceledOkay);
+}
+
+/* static */
+int32_t Stopwatch::TimeElapsedKeyed(const dom::GlobalObject& aGlobal,
+ const nsAString& aHistogram,
+ const nsAString& aKey,
+ JS::Handle<JSObject*> aObj,
+ bool aCanceledOkay) {
+ return Timers::Singleton().TimeElapsed(aGlobal.Context(), aHistogram, aObj,
+ aKey, aCanceledOkay);
+}
+
+/* static */
+bool Stopwatch::Finish(const dom::GlobalObject& aGlobal,
+ const nsAString& aHistogram, JS::Handle<JSObject*> aObj,
+ bool aCanceledOkay) {
+ return FinishKeyed(aGlobal, aHistogram, VoidString(), aObj, aCanceledOkay);
+}
+
+/* static */
+bool Stopwatch::FinishKeyed(const dom::GlobalObject& aGlobal,
+ const nsAString& aHistogram, const nsAString& aKey,
+ JS::Handle<JSObject*> aObj, bool aCanceledOkay) {
+ return Timers::Singleton().Finish(aGlobal.Context(), aHistogram, aObj, aKey,
+ aCanceledOkay) != -1;
+}
+
+/* static */
+bool Stopwatch::Cancel(const dom::GlobalObject& aGlobal,
+ const nsAString& aHistogram,
+ JS::Handle<JSObject*> aObj) {
+ return CancelKeyed(aGlobal, aHistogram, VoidString(), aObj);
+}
+
+/* static */
+bool Stopwatch::CancelKeyed(const dom::GlobalObject& aGlobal,
+ const nsAString& aHistogram, const nsAString& aKey,
+ JS::Handle<JSObject*> aObj) {
+ return Timers::Singleton().Delete(aGlobal.Context(), aHistogram, aObj, aKey);
+}
+
+/* static */
+void Stopwatch::SetTestModeEnabled(const dom::GlobalObject& aGlobal,
+ bool aTesting) {
+ Timers::Singleton().SuppressErrors() = aTesting;
+}
+
+/* static */
+bool UserInteractionStopwatch::Start(const dom::GlobalObject& aGlobal,
+ const nsAString& aUserInteraction,
+ const nsACString& aValue,
+ JS::Handle<JSObject*> aObj) {
+ if (!NS_IsMainThread()) {
+ return false;
+ }
+ return Timers::Singleton().StartUserInteraction(
+ aGlobal.Context(), aUserInteraction, aValue, aObj);
+}
+
+/* static */
+bool UserInteractionStopwatch::Running(const dom::GlobalObject& aGlobal,
+ const nsAString& aUserInteraction,
+ JS::Handle<JSObject*> aObj) {
+ if (!NS_IsMainThread()) {
+ return false;
+ }
+ return Timers::Singleton().RunningUserInteraction(aGlobal.Context(),
+ aUserInteraction, aObj);
+}
+
+/* static */
+bool UserInteractionStopwatch::Update(const dom::GlobalObject& aGlobal,
+ const nsAString& aUserInteraction,
+ const nsACString& aValue,
+ JS::Handle<JSObject*> aObj) {
+ if (!NS_IsMainThread()) {
+ return false;
+ }
+ return Timers::Singleton().UpdateUserInteraction(
+ aGlobal.Context(), aUserInteraction, aValue, aObj);
+}
+
+/* static */
+bool UserInteractionStopwatch::Cancel(const dom::GlobalObject& aGlobal,
+ const nsAString& aUserInteraction,
+ JS::Handle<JSObject*> aObj) {
+ if (!NS_IsMainThread()) {
+ return false;
+ }
+ return Timers::Singleton().CancelUserInteraction(aGlobal.Context(),
+ aUserInteraction, aObj);
+}
+
+/* static */
+bool UserInteractionStopwatch::Finish(
+ const dom::GlobalObject& aGlobal, const nsAString& aUserInteraction,
+ JS::Handle<JSObject*> aObj,
+ const dom::Optional<nsACString>& aAdditionalText) {
+ if (!NS_IsMainThread()) {
+ return false;
+ }
+ return Timers::Singleton().FinishUserInteraction(
+ aGlobal.Context(), aUserInteraction, aObj, aAdditionalText);
+}
+
+} // namespace mozilla::telemetry
diff --git a/toolkit/components/telemetry/core/Stopwatch.h b/toolkit/components/telemetry/core/Stopwatch.h
new file mode 100644
index 0000000000..070872e250
--- /dev/null
+++ b/toolkit/components/telemetry/core/Stopwatch.h
@@ -0,0 +1,84 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef Stopwatch_h__
+#define Stopwatch_h__
+
+#include "mozilla/dom/TelemetryStopwatchBinding.h"
+
+namespace mozilla {
+namespace telemetry {
+
+class Stopwatch {
+ using GlobalObject = mozilla::dom::GlobalObject;
+
+ public:
+ static bool Start(const GlobalObject& global, const nsAString& histogram,
+ JS::Handle<JSObject*> obj,
+ const dom::TelemetryStopwatchOptions& options);
+
+ static bool Running(const GlobalObject& global, const nsAString& histogram,
+ JS::Handle<JSObject*> obj);
+
+ static bool Cancel(const GlobalObject& global, const nsAString& histogram,
+ JS::Handle<JSObject*> obj);
+
+ static int32_t TimeElapsed(const GlobalObject& global,
+ const nsAString& histogram,
+ JS::Handle<JSObject*> obj, bool canceledOkay);
+
+ static bool Finish(const GlobalObject& global, const nsAString& histogram,
+ JS::Handle<JSObject*> obj, bool canceledOkay);
+
+ static bool StartKeyed(const GlobalObject& global, const nsAString& histogram,
+ const nsAString& key, JS::Handle<JSObject*> obj,
+ const dom::TelemetryStopwatchOptions& options);
+
+ static bool RunningKeyed(const GlobalObject& global,
+ const nsAString& histogram, const nsAString& key,
+ JS::Handle<JSObject*> obj);
+
+ static bool CancelKeyed(const GlobalObject& global,
+ const nsAString& histogram, const nsAString& key,
+ JS::Handle<JSObject*> obj);
+
+ static int32_t TimeElapsedKeyed(const GlobalObject& global,
+ const nsAString& histogram,
+ const nsAString& key,
+ JS::Handle<JSObject*> obj, bool canceledOkay);
+
+ static bool FinishKeyed(const GlobalObject& global,
+ const nsAString& histogram, const nsAString& key,
+ JS::Handle<JSObject*> obj, bool canceledOkay);
+
+ static void SetTestModeEnabled(const GlobalObject& global, bool testing);
+};
+
+class UserInteractionStopwatch {
+ using GlobalObject = mozilla::dom::GlobalObject;
+
+ public:
+ static bool Start(const GlobalObject& aGlobal,
+ const nsAString& aUserInteraction, const nsACString& aValue,
+ JS::Handle<JSObject*> aObj);
+ static bool Running(const GlobalObject& aGlobal,
+ const nsAString& aUserInteraction,
+ JS::Handle<JSObject*> aObj);
+ static bool Update(const GlobalObject& aGlobal,
+ const nsAString& aUserInteraction,
+ const nsACString& aValue, JS::Handle<JSObject*> aObj);
+ static bool Cancel(const GlobalObject& aGlobal,
+ const nsAString& aUserInteraction,
+ JS::Handle<JSObject*> aObj);
+ static bool Finish(const GlobalObject& aGlobal,
+ const nsAString& aUserInteraction,
+ JS::Handle<JSObject*> aObj,
+ const dom::Optional<nsACString>& aAdditionalText);
+};
+
+} // namespace telemetry
+} // namespace mozilla
+
+#endif // Stopwatch_h__
diff --git a/toolkit/components/telemetry/core/Telemetry.cpp b/toolkit/components/telemetry/core/Telemetry.cpp
new file mode 100644
index 0000000000..a77745844d
--- /dev/null
+++ b/toolkit/components/telemetry/core/Telemetry.cpp
@@ -0,0 +1,2076 @@
+/* -*- 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 "Telemetry.h"
+
+#include <algorithm>
+#include <prio.h>
+#include <prproces.h>
+#if defined(XP_UNIX) && !defined(XP_DARWIN)
+# include <time.h>
+#else
+# include <chrono>
+#endif
+#include "base/pickle.h"
+#include "base/process_util.h"
+#if defined(MOZ_TELEMETRY_GECKOVIEW)
+# include "geckoview/TelemetryGeckoViewPersistence.h"
+#endif
+#include "ipc/TelemetryIPCAccumulator.h"
+#include "jsapi.h"
+#include "jsfriendapi.h"
+#include "js/Array.h" // JS::NewArrayObject
+#include "js/GCAPI.h"
+#include "mozilla/dom/ToJSValue.h"
+#include "mozilla/dom/Promise.h"
+#include "mozilla/Atomics.h"
+#include "mozilla/Attributes.h"
+#include "mozilla/BackgroundHangMonitor.h"
+#include "mozilla/Components.h"
+#include "mozilla/DataMutex.h"
+#include "mozilla/DebugOnly.h"
+#include "mozilla/FStream.h"
+#include "mozilla/IOInterposer.h"
+#include "mozilla/Likely.h"
+#include "mozilla/MathAlgorithms.h"
+#include "mozilla/MemoryReporting.h"
+#include "mozilla/MemoryTelemetry.h"
+#include "mozilla/ModuleUtils.h"
+#include "mozilla/Mutex.h"
+#include "mozilla/PoisonIOInterposer.h"
+#include "mozilla/Preferences.h"
+#include "mozilla/StartupTimeline.h"
+#include "mozilla/StaticPtr.h"
+#include "mozilla/Unused.h"
+#if defined(XP_WIN)
+# include "mozilla/WinDllServices.h"
+#endif
+#include "nsAppDirectoryServiceDefs.h"
+#include "nsBaseHashtable.h"
+#include "nsClassHashtable.h"
+#include "nsCOMArray.h"
+#include "nsCOMPtr.h"
+#include "nsDataHashtable.h"
+#include "nsHashKeys.h"
+#include "nsIDirectoryEnumerator.h"
+#include "nsDirectoryServiceDefs.h"
+#include "nsIFileStreams.h"
+#include "nsIMemoryReporter.h"
+#include "nsISeekableStream.h"
+#include "nsITelemetry.h"
+#if defined(XP_WIN)
+# include "other/UntrustedModules.h"
+#endif
+#include "nsJSUtils.h"
+#include "nsLocalFile.h"
+#include "nsNativeCharsetUtils.h"
+#include "nsNetCID.h"
+#include "nsNetUtil.h"
+#include "nsProxyRelease.h"
+#include "nsReadableUtils.h"
+#include "nsString.h"
+#include "nsTHashtable.h"
+#include "nsThreadManager.h"
+#include "nsThreadUtils.h"
+#if defined(XP_WIN)
+# include "nsUnicharUtils.h"
+#endif
+#include "nsVersionComparator.h"
+#include "nsXPCOMCIDInternal.h"
+#include "other/CombinedStacks.h"
+#include "other/TelemetryIOInterposeObserver.h"
+#include "plstr.h"
+#include "TelemetryCommon.h"
+#include "TelemetryEvent.h"
+#include "TelemetryHistogram.h"
+#include "TelemetryOrigin.h"
+#include "TelemetryScalar.h"
+#include "TelemetryUserInteraction.h"
+
+namespace {
+
+using namespace mozilla;
+using mozilla::dom::AutoJSAPI;
+using mozilla::dom::Promise;
+using mozilla::Telemetry::CombinedStacks;
+using mozilla::Telemetry::EventExtraEntry;
+using mozilla::Telemetry::TelemetryIOInterposeObserver;
+using Telemetry::Common::AutoHashtable;
+using Telemetry::Common::GetCurrentProduct;
+using Telemetry::Common::StringHashSet;
+using Telemetry::Common::SupportedProduct;
+using Telemetry::Common::ToJSString;
+
+// This is not a member of TelemetryImpl because we want to record I/O during
+// startup.
+StaticAutoPtr<TelemetryIOInterposeObserver> sTelemetryIOObserver;
+
+void ClearIOReporting() {
+ if (!sTelemetryIOObserver) {
+ return;
+ }
+ IOInterposer::Unregister(IOInterposeObserver::OpAllWithStaging,
+ sTelemetryIOObserver);
+ sTelemetryIOObserver = nullptr;
+}
+
+class TelemetryImpl final : public nsITelemetry, public nsIMemoryReporter {
+ NS_DECL_THREADSAFE_ISUPPORTS
+ NS_DECL_NSITELEMETRY
+ NS_DECL_NSIMEMORYREPORTER
+
+ public:
+ void InitMemoryReporter();
+
+ static already_AddRefed<nsITelemetry> CreateTelemetryInstance();
+ static void ShutdownTelemetry();
+ static void RecordSlowStatement(const nsACString& sql,
+ const nsACString& dbName, uint32_t delay);
+ struct Stat {
+ uint32_t hitCount;
+ uint32_t totalTime;
+ };
+ struct StmtStats {
+ struct Stat mainThread;
+ struct Stat otherThreads;
+ };
+ typedef nsBaseHashtableET<nsCStringHashKey, StmtStats> SlowSQLEntryType;
+
+ static void RecordIceCandidates(const uint32_t iceCandidateBitmask,
+ const bool success);
+ static bool CanRecordBase();
+ static bool CanRecordExtended();
+ static bool CanRecordReleaseData();
+ static bool CanRecordPrereleaseData();
+
+ private:
+ TelemetryImpl();
+ ~TelemetryImpl();
+
+ static nsCString SanitizeSQL(const nsACString& sql);
+
+ enum SanitizedState { Sanitized, Unsanitized };
+
+ static void StoreSlowSQL(const nsACString& offender, uint32_t delay,
+ SanitizedState state);
+
+ static bool ReflectMainThreadSQL(SlowSQLEntryType* entry, JSContext* cx,
+ JS::Handle<JSObject*> obj);
+ static bool ReflectOtherThreadsSQL(SlowSQLEntryType* entry, JSContext* cx,
+ JS::Handle<JSObject*> obj);
+ static bool ReflectSQL(const SlowSQLEntryType* entry, const Stat* stat,
+ JSContext* cx, JS::Handle<JSObject*> obj);
+
+ bool AddSQLInfo(JSContext* cx, JS::Handle<JSObject*> rootObj, bool mainThread,
+ bool privateSQL);
+ bool GetSQLStats(JSContext* cx, JS::MutableHandle<JS::Value> ret,
+ bool includePrivateSql);
+
+ void ReadLateWritesStacks(nsIFile* aProfileDir);
+
+ static StaticDataMutex<TelemetryImpl*> sTelemetry;
+ AutoHashtable<SlowSQLEntryType> mPrivateSQL;
+ AutoHashtable<SlowSQLEntryType> mSanitizedSQL;
+ Mutex mHashMutex;
+ Atomic<bool, SequentiallyConsistent> mCanRecordBase;
+ Atomic<bool, SequentiallyConsistent> mCanRecordExtended;
+
+ CombinedStacks
+ mLateWritesStacks; // This is collected out of the main thread.
+ bool mCachedTelemetryData;
+ uint32_t mLastShutdownTime;
+ uint32_t mFailedLockCount;
+ nsCOMArray<nsIFetchTelemetryDataCallback> mCallbacks;
+ friend class nsFetchTelemetryData;
+};
+
+StaticDataMutex<TelemetryImpl*> TelemetryImpl::sTelemetry(nullptr, nullptr);
+
+MOZ_DEFINE_MALLOC_SIZE_OF(TelemetryMallocSizeOf)
+
+NS_IMETHODIMP
+TelemetryImpl::CollectReports(nsIHandleReportCallback* aHandleReport,
+ nsISupports* aData, bool aAnonymize) {
+ mozilla::MallocSizeOf aMallocSizeOf = TelemetryMallocSizeOf;
+
+#define COLLECT_REPORT(name, size, desc) \
+ MOZ_COLLECT_REPORT(name, KIND_HEAP, UNITS_BYTES, size, desc)
+
+ COLLECT_REPORT("explicit/telemetry/impl", aMallocSizeOf(this),
+ "Memory used by the Telemetry core implemenation");
+
+ COLLECT_REPORT(
+ "explicit/telemetry/scalar/shallow",
+ TelemetryScalar::GetMapShallowSizesOfExcludingThis(aMallocSizeOf),
+ "Memory used by the Telemetry Scalar implemenation");
+
+ { // Scope for mHashMutex lock
+ MutexAutoLock lock(mHashMutex);
+ COLLECT_REPORT("explicit/telemetry/PrivateSQL",
+ mPrivateSQL.SizeOfExcludingThis(aMallocSizeOf),
+ "Memory used by the PrivateSQL Telemetry");
+
+ COLLECT_REPORT("explicit/telemetry/SanitizedSQL",
+ mSanitizedSQL.SizeOfExcludingThis(aMallocSizeOf),
+ "Memory used by the SanitizedSQL Telemetry");
+ }
+
+ if (sTelemetryIOObserver) {
+ COLLECT_REPORT("explicit/telemetry/IOObserver",
+ sTelemetryIOObserver->SizeOfIncludingThis(aMallocSizeOf),
+ "Memory used by the Telemetry IO Observer");
+ }
+
+ COLLECT_REPORT("explicit/telemetry/LateWritesStacks",
+ mLateWritesStacks.SizeOfExcludingThis(),
+ "Memory used by the Telemetry LateWrites Stack capturer");
+
+ COLLECT_REPORT("explicit/telemetry/Callbacks",
+ mCallbacks.ShallowSizeOfExcludingThis(aMallocSizeOf),
+ "Memory used by the Telemetry Callbacks array (shallow)");
+
+ COLLECT_REPORT(
+ "explicit/telemetry/histogram/data",
+ TelemetryHistogram::GetHistogramSizesOfIncludingThis(aMallocSizeOf),
+ "Memory used by Telemetry Histogram data");
+
+ COLLECT_REPORT("explicit/telemetry/scalar/data",
+ TelemetryScalar::GetScalarSizesOfIncludingThis(aMallocSizeOf),
+ "Memory used by Telemetry Scalar data");
+
+ COLLECT_REPORT("explicit/telemetry/event/data",
+ TelemetryEvent::SizeOfIncludingThis(aMallocSizeOf),
+ "Memory used by Telemetry Event data");
+
+ COLLECT_REPORT("explicit/telemetry/origin/data",
+ TelemetryOrigin::SizeOfIncludingThis(aMallocSizeOf),
+ "Memory used by Telemetry Origin data");
+
+#undef COLLECT_REPORT
+
+ return NS_OK;
+}
+
+void InitHistogramRecordingEnabled() {
+ TelemetryHistogram::InitHistogramRecordingEnabled();
+}
+
+using PathChar = filesystem::Path::value_type;
+using PathCharPtr = const PathChar*;
+
+static uint32_t ReadLastShutdownDuration(PathCharPtr filename) {
+ RefPtr<nsLocalFile> file =
+ new nsLocalFile(nsTDependentString<PathChar>(filename));
+ FILE* f;
+ if (NS_FAILED(file->OpenANSIFileDesc("r", &f)) || !f) {
+ return 0;
+ }
+
+ int shutdownTime;
+ int r = fscanf(f, "%d\n", &shutdownTime);
+ fclose(f);
+ if (r != 1) {
+ return 0;
+ }
+
+ return shutdownTime;
+}
+
+const int32_t kMaxFailedProfileLockFileSize = 10;
+
+bool GetFailedLockCount(nsIInputStream* inStream, uint32_t aCount,
+ unsigned int& result) {
+ nsAutoCString bufStr;
+ nsresult rv;
+ rv = NS_ReadInputStreamToString(inStream, bufStr, aCount);
+ NS_ENSURE_SUCCESS(rv, false);
+ result = bufStr.ToInteger(&rv);
+ return NS_SUCCEEDED(rv) && result > 0;
+}
+
+nsresult GetFailedProfileLockFile(nsIFile** aFile, nsIFile* aProfileDir) {
+ NS_ENSURE_ARG_POINTER(aProfileDir);
+
+ nsresult rv = aProfileDir->Clone(aFile);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ (*aFile)->AppendNative("Telemetry.FailedProfileLocks.txt"_ns);
+ return NS_OK;
+}
+
+class nsFetchTelemetryData : public Runnable {
+ public:
+ nsFetchTelemetryData(PathCharPtr aShutdownTimeFilename,
+ nsIFile* aFailedProfileLockFile, nsIFile* aProfileDir)
+ : mozilla::Runnable("nsFetchTelemetryData"),
+ mShutdownTimeFilename(aShutdownTimeFilename),
+ mFailedProfileLockFile(aFailedProfileLockFile),
+ mProfileDir(aProfileDir) {}
+
+ private:
+ PathCharPtr mShutdownTimeFilename;
+ nsCOMPtr<nsIFile> mFailedProfileLockFile;
+ nsCOMPtr<nsIFile> mProfileDir;
+
+ public:
+ void MainThread() {
+ auto lock = TelemetryImpl::sTelemetry.Lock();
+ auto telemetry = lock.ref();
+ telemetry->mCachedTelemetryData = true;
+ for (unsigned int i = 0, n = telemetry->mCallbacks.Count(); i < n; ++i) {
+ telemetry->mCallbacks[i]->Complete();
+ }
+ telemetry->mCallbacks.Clear();
+ }
+
+ NS_IMETHOD Run() override {
+ uint32_t failedLockCount = 0;
+ uint32_t lastShutdownDuration = 0;
+ LoadFailedLockCount(failedLockCount);
+ lastShutdownDuration = ReadLastShutdownDuration(mShutdownTimeFilename);
+ {
+ auto lock = TelemetryImpl::sTelemetry.Lock();
+ auto telemetry = lock.ref();
+ telemetry->mFailedLockCount = failedLockCount;
+ telemetry->mLastShutdownTime = lastShutdownDuration;
+ telemetry->ReadLateWritesStacks(mProfileDir);
+ }
+
+ TelemetryScalar::Set(Telemetry::ScalarID::BROWSER_TIMINGS_LAST_SHUTDOWN,
+ lastShutdownDuration);
+
+ nsCOMPtr<nsIRunnable> e =
+ NewRunnableMethod("nsFetchTelemetryData::MainThread", this,
+ &nsFetchTelemetryData::MainThread);
+ NS_ENSURE_STATE(e);
+ NS_DispatchToMainThread(e);
+ return NS_OK;
+ }
+
+ private:
+ nsresult LoadFailedLockCount(uint32_t& failedLockCount) {
+ failedLockCount = 0;
+ int64_t fileSize = 0;
+ nsresult rv = mFailedProfileLockFile->GetFileSize(&fileSize);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ NS_ENSURE_TRUE(fileSize <= kMaxFailedProfileLockFileSize,
+ NS_ERROR_UNEXPECTED);
+ nsCOMPtr<nsIInputStream> inStream;
+ rv = NS_NewLocalFileInputStream(getter_AddRefs(inStream),
+ mFailedProfileLockFile, PR_RDONLY);
+ NS_ENSURE_SUCCESS(rv, rv);
+ NS_ENSURE_TRUE(GetFailedLockCount(inStream, fileSize, failedLockCount),
+ NS_ERROR_UNEXPECTED);
+ inStream->Close();
+
+ mFailedProfileLockFile->Remove(false);
+ return NS_OK;
+ }
+};
+
+static TimeStamp gRecordedShutdownStartTime;
+static bool gAlreadyFreedShutdownTimeFileName = false;
+static PathCharPtr gRecordedShutdownTimeFileName = nullptr;
+
+static PathCharPtr GetShutdownTimeFileName() {
+ if (gAlreadyFreedShutdownTimeFileName) {
+ return nullptr;
+ }
+
+ if (!gRecordedShutdownTimeFileName) {
+ nsCOMPtr<nsIFile> mozFile;
+ NS_GetSpecialDirectory(NS_APP_USER_PROFILE_50_DIR, getter_AddRefs(mozFile));
+ if (!mozFile) return nullptr;
+
+ mozFile->AppendNative("Telemetry.ShutdownTime.txt"_ns);
+
+ gRecordedShutdownTimeFileName = NS_xstrdup(mozFile->NativePath().get());
+ }
+
+ return gRecordedShutdownTimeFileName;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetLastShutdownDuration(uint32_t* aResult) {
+ // The user must call AsyncFetchTelemetryData first. We return zero instead of
+ // reporting a failure so that the rest of telemetry can uniformly handle
+ // the read not being available yet.
+ if (!mCachedTelemetryData) {
+ *aResult = 0;
+ return NS_OK;
+ }
+
+ *aResult = mLastShutdownTime;
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetFailedProfileLockCount(uint32_t* aResult) {
+ // The user must call AsyncFetchTelemetryData first. We return zero instead of
+ // reporting a failure so that the rest of telemetry can uniformly handle
+ // the read not being available yet.
+ if (!mCachedTelemetryData) {
+ *aResult = 0;
+ return NS_OK;
+ }
+
+ *aResult = mFailedLockCount;
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::AsyncFetchTelemetryData(
+ nsIFetchTelemetryDataCallback* aCallback) {
+ // We have finished reading the data already, just call the callback.
+ if (mCachedTelemetryData) {
+ aCallback->Complete();
+ return NS_OK;
+ }
+
+ // We already have a read request running, just remember the callback.
+ if (mCallbacks.Count() != 0) {
+ mCallbacks.AppendObject(aCallback);
+ return NS_OK;
+ }
+
+ // We make this check so that GetShutdownTimeFileName() doesn't get
+ // called; calling that function without telemetry enabled violates
+ // assumptions that the write-the-shutdown-timestamp machinery makes.
+ if (!Telemetry::CanRecordExtended()) {
+ mCachedTelemetryData = true;
+ aCallback->Complete();
+ return NS_OK;
+ }
+
+ // Send the read to a background thread provided by the stream transport
+ // service to avoid a read in the main thread.
+ nsCOMPtr<nsIEventTarget> targetThread =
+ do_GetService(NS_STREAMTRANSPORTSERVICE_CONTRACTID);
+ if (!targetThread) {
+ mCachedTelemetryData = true;
+ aCallback->Complete();
+ return NS_OK;
+ }
+
+ // We have to get the filename from the main thread.
+ PathCharPtr shutdownTimeFilename = GetShutdownTimeFileName();
+ if (!shutdownTimeFilename) {
+ mCachedTelemetryData = true;
+ aCallback->Complete();
+ return NS_OK;
+ }
+
+ nsCOMPtr<nsIFile> profileDir;
+ nsresult rv = NS_GetSpecialDirectory(NS_APP_USER_PROFILE_50_DIR,
+ getter_AddRefs(profileDir));
+ if (NS_FAILED(rv)) {
+ mCachedTelemetryData = true;
+ aCallback->Complete();
+ return NS_OK;
+ }
+
+ nsCOMPtr<nsIFile> failedProfileLockFile;
+ rv = GetFailedProfileLockFile(getter_AddRefs(failedProfileLockFile),
+ profileDir);
+ if (NS_FAILED(rv)) {
+ mCachedTelemetryData = true;
+ aCallback->Complete();
+ return NS_OK;
+ }
+
+ mCallbacks.AppendObject(aCallback);
+
+ nsCOMPtr<nsIRunnable> event = new nsFetchTelemetryData(
+ shutdownTimeFilename, failedProfileLockFile, profileDir);
+
+ targetThread->Dispatch(event, NS_DISPATCH_NORMAL);
+ return NS_OK;
+}
+
+TelemetryImpl::TelemetryImpl()
+ : mHashMutex("Telemetry::mHashMutex"),
+ mCanRecordBase(false),
+ mCanRecordExtended(false),
+ mCachedTelemetryData(false),
+ mLastShutdownTime(0),
+ mFailedLockCount(0) {
+ // We expect TelemetryHistogram::InitializeGlobalState() to have been
+ // called before we get to this point.
+ MOZ_ASSERT(TelemetryHistogram::GlobalStateHasBeenInitialized());
+}
+
+TelemetryImpl::~TelemetryImpl() {
+ UnregisterWeakMemoryReporter(this);
+
+ // This is still racey as access to these collections is guarded using
+ // sTelemetry. We will fix this in bug 1367344.
+ MutexAutoLock hashLock(mHashMutex);
+}
+
+void TelemetryImpl::InitMemoryReporter() { RegisterWeakMemoryReporter(this); }
+
+bool TelemetryImpl::ReflectSQL(const SlowSQLEntryType* entry, const Stat* stat,
+ JSContext* cx, JS::Handle<JSObject*> obj) {
+ if (stat->hitCount == 0) return true;
+
+ const nsACString& sql = entry->GetKey();
+
+ JS::Rooted<JSObject*> arrayObj(cx, JS::NewArrayObject(cx, 0));
+ if (!arrayObj) {
+ return false;
+ }
+ return (
+ JS_DefineElement(cx, arrayObj, 0, stat->hitCount, JSPROP_ENUMERATE) &&
+ JS_DefineElement(cx, arrayObj, 1, stat->totalTime, JSPROP_ENUMERATE) &&
+ JS_DefineProperty(cx, obj, sql.BeginReading(), arrayObj,
+ JSPROP_ENUMERATE));
+}
+
+bool TelemetryImpl::ReflectMainThreadSQL(SlowSQLEntryType* entry, JSContext* cx,
+ JS::Handle<JSObject*> obj) {
+ return ReflectSQL(entry, &entry->GetModifiableData()->mainThread, cx, obj);
+}
+
+bool TelemetryImpl::ReflectOtherThreadsSQL(SlowSQLEntryType* entry,
+ JSContext* cx,
+ JS::Handle<JSObject*> obj) {
+ return ReflectSQL(entry, &entry->GetModifiableData()->otherThreads, cx, obj);
+}
+
+bool TelemetryImpl::AddSQLInfo(JSContext* cx, JS::Handle<JSObject*> rootObj,
+ bool mainThread, bool privateSQL) {
+ JS::Rooted<JSObject*> statsObj(cx, JS_NewPlainObject(cx));
+ if (!statsObj) return false;
+
+ AutoHashtable<SlowSQLEntryType>& sqlMap =
+ (privateSQL ? mPrivateSQL : mSanitizedSQL);
+ AutoHashtable<SlowSQLEntryType>::ReflectEntryFunc reflectFunction =
+ (mainThread ? ReflectMainThreadSQL : ReflectOtherThreadsSQL);
+ if (!sqlMap.ReflectIntoJS(reflectFunction, cx, statsObj)) {
+ return false;
+ }
+
+ return JS_DefineProperty(cx, rootObj,
+ mainThread ? "mainThread" : "otherThreads", statsObj,
+ JSPROP_ENUMERATE);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::SetHistogramRecordingEnabled(const nsACString& id,
+ bool aEnabled) {
+ return TelemetryHistogram::SetHistogramRecordingEnabled(id, aEnabled);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetSnapshotForHistograms(const nsACString& aStoreName,
+ bool aClearStore, bool aFilterTest,
+ JSContext* aCx,
+ JS::MutableHandleValue aResult) {
+ constexpr auto defaultStore = "main"_ns;
+ unsigned int dataset = mCanRecordExtended
+ ? nsITelemetry::DATASET_PRERELEASE_CHANNELS
+ : nsITelemetry::DATASET_ALL_CHANNELS;
+ return TelemetryHistogram::CreateHistogramSnapshots(
+ aCx, aResult, aStoreName.IsVoid() ? defaultStore : aStoreName, dataset,
+ aClearStore, aFilterTest);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetSnapshotForKeyedHistograms(const nsACString& aStoreName,
+ bool aClearStore, bool aFilterTest,
+ JSContext* aCx,
+ JS::MutableHandleValue aResult) {
+ constexpr auto defaultStore = "main"_ns;
+ unsigned int dataset = mCanRecordExtended
+ ? nsITelemetry::DATASET_PRERELEASE_CHANNELS
+ : nsITelemetry::DATASET_ALL_CHANNELS;
+ return TelemetryHistogram::GetKeyedHistogramSnapshots(
+ aCx, aResult, aStoreName.IsVoid() ? defaultStore : aStoreName, dataset,
+ aClearStore, aFilterTest);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetCategoricalLabels(JSContext* aCx,
+ JS::MutableHandleValue aResult) {
+ return TelemetryHistogram::GetCategoricalHistogramLabels(aCx, aResult);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetSnapshotForScalars(const nsACString& aStoreName,
+ bool aClearStore, bool aFilterTest,
+ JSContext* aCx,
+ JS::MutableHandleValue aResult) {
+ constexpr auto defaultStore = "main"_ns;
+ unsigned int dataset = mCanRecordExtended
+ ? nsITelemetry::DATASET_PRERELEASE_CHANNELS
+ : nsITelemetry::DATASET_ALL_CHANNELS;
+ return TelemetryScalar::CreateSnapshots(
+ dataset, aClearStore, aCx, 1, aResult, aFilterTest,
+ aStoreName.IsVoid() ? defaultStore : aStoreName);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetSnapshotForKeyedScalars(const nsACString& aStoreName,
+ bool aClearStore, bool aFilterTest,
+ JSContext* aCx,
+ JS::MutableHandleValue aResult) {
+ constexpr auto defaultStore = "main"_ns;
+ unsigned int dataset = mCanRecordExtended
+ ? nsITelemetry::DATASET_PRERELEASE_CHANNELS
+ : nsITelemetry::DATASET_ALL_CHANNELS;
+ return TelemetryScalar::CreateKeyedSnapshots(
+ dataset, aClearStore, aCx, 1, aResult, aFilterTest,
+ aStoreName.IsVoid() ? defaultStore : aStoreName);
+}
+
+bool TelemetryImpl::GetSQLStats(JSContext* cx, JS::MutableHandle<JS::Value> ret,
+ bool includePrivateSql) {
+ JS::Rooted<JSObject*> root_obj(cx, JS_NewPlainObject(cx));
+ if (!root_obj) return false;
+ ret.setObject(*root_obj);
+
+ MutexAutoLock hashMutex(mHashMutex);
+ // Add info about slow SQL queries on the main thread
+ if (!AddSQLInfo(cx, root_obj, true, includePrivateSql)) return false;
+ // Add info about slow SQL queries on other threads
+ if (!AddSQLInfo(cx, root_obj, false, includePrivateSql)) return false;
+
+ return true;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetSlowSQL(JSContext* cx, JS::MutableHandle<JS::Value> ret) {
+ if (GetSQLStats(cx, ret, false)) return NS_OK;
+ return NS_ERROR_FAILURE;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetDebugSlowSQL(JSContext* cx,
+ JS::MutableHandle<JS::Value> ret) {
+ bool revealPrivateSql =
+ Preferences::GetBool("toolkit.telemetry.debugSlowSql", false);
+ if (GetSQLStats(cx, ret, revealPrivateSql)) return NS_OK;
+ return NS_ERROR_FAILURE;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetMaximalNumberOfConcurrentThreads(uint32_t* ret) {
+ *ret = nsThreadManager::get().GetHighestNumberOfThreads();
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetUntrustedModuleLoadEvents(uint32_t aFlags, JSContext* cx,
+ Promise** aPromise) {
+#if defined(XP_WIN)
+ return Telemetry::GetUntrustedModuleLoadEvents(aFlags, cx, aPromise);
+#else
+ return NS_ERROR_NOT_IMPLEMENTED;
+#endif
+}
+
+#if defined(MOZ_GECKO_PROFILER)
+class GetLoadedModulesResultRunnable final : public Runnable {
+ nsMainThreadPtrHandle<Promise> mPromise;
+ SharedLibraryInfo mRawModules;
+ nsCOMPtr<nsIThread> mWorkerThread;
+# if defined(XP_WIN)
+ nsDataHashtable<nsStringHashKey, nsString> mCertSubjects;
+# endif // defined(XP_WIN)
+
+ public:
+ GetLoadedModulesResultRunnable(const nsMainThreadPtrHandle<Promise>& aPromise,
+ const SharedLibraryInfo& rawModules)
+ : mozilla::Runnable("GetLoadedModulesResultRunnable"),
+ mPromise(aPromise),
+ mRawModules(rawModules),
+ mWorkerThread(do_GetCurrentThread()) {
+ MOZ_ASSERT(!NS_IsMainThread());
+# if defined(XP_WIN)
+ ObtainCertSubjects();
+# endif // defined(XP_WIN)
+ }
+
+ NS_IMETHOD
+ Run() override {
+ MOZ_ASSERT(NS_IsMainThread());
+
+ mWorkerThread->Shutdown();
+
+ AutoJSAPI jsapi;
+ if (NS_WARN_IF(!jsapi.Init(mPromise->GetGlobalObject()))) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+
+ JSContext* cx = jsapi.cx();
+
+ JS::RootedObject moduleArray(cx, JS::NewArrayObject(cx, 0));
+ if (!moduleArray) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+
+ for (unsigned int i = 0, n = mRawModules.GetSize(); i != n; i++) {
+ const SharedLibrary& info = mRawModules.GetEntry(i);
+
+ JS::RootedObject moduleObj(cx, JS_NewPlainObject(cx));
+ if (!moduleObj) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+
+ // Module name.
+ JS::RootedString moduleName(
+ cx, JS_NewUCStringCopyZ(cx, info.GetModuleName().get()));
+ if (!moduleName || !JS_DefineProperty(cx, moduleObj, "name", moduleName,
+ JSPROP_ENUMERATE)) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+
+ // Module debug name.
+ JS::RootedValue moduleDebugName(cx);
+
+ if (!info.GetDebugName().IsEmpty()) {
+ JS::RootedString str_moduleDebugName(
+ cx, JS_NewUCStringCopyZ(cx, info.GetDebugName().get()));
+ if (!str_moduleDebugName) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+ moduleDebugName.setString(str_moduleDebugName);
+ } else {
+ moduleDebugName.setNull();
+ }
+
+ if (!JS_DefineProperty(cx, moduleObj, "debugName", moduleDebugName,
+ JSPROP_ENUMERATE)) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+
+ // Module Breakpad identifier.
+ JS::RootedValue id(cx);
+
+ if (!info.GetBreakpadId().IsEmpty()) {
+ JS::RootedString str_id(
+ cx, JS_NewStringCopyZ(cx, info.GetBreakpadId().get()));
+ if (!str_id) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+ id.setString(str_id);
+ } else {
+ id.setNull();
+ }
+
+ if (!JS_DefineProperty(cx, moduleObj, "debugID", id, JSPROP_ENUMERATE)) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+
+ // Module version.
+ JS::RootedValue version(cx);
+
+ if (!info.GetVersion().IsEmpty()) {
+ JS::RootedString v(
+ cx, JS_NewStringCopyZ(cx, info.GetVersion().BeginReading()));
+ if (!v) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+ version.setString(v);
+ } else {
+ version.setNull();
+ }
+
+ if (!JS_DefineProperty(cx, moduleObj, "version", version,
+ JSPROP_ENUMERATE)) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+
+# if defined(XP_WIN)
+ // Cert Subject.
+ nsString* subject = mCertSubjects.GetValue(info.GetModulePath());
+ if (subject) {
+ JS::RootedString jsOrg(cx, ToJSString(cx, *subject));
+ if (!jsOrg) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+
+ JS::RootedValue certSubject(cx);
+ certSubject.setString(jsOrg);
+
+ if (!JS_DefineProperty(cx, moduleObj, "certSubject", certSubject,
+ JSPROP_ENUMERATE)) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+ }
+# endif // defined(XP_WIN)
+
+ if (!JS_DefineElement(cx, moduleArray, i, moduleObj, JSPROP_ENUMERATE)) {
+ mPromise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+ }
+
+ mPromise->MaybeResolve(moduleArray);
+ return NS_OK;
+ }
+
+ private:
+# if defined(XP_WIN)
+ void ObtainCertSubjects() {
+ MOZ_ASSERT(!NS_IsMainThread());
+
+ // NB: Currently we cannot lower this down to the profiler layer due to
+ // differing startup dependencies between the profiler and DllServices.
+ RefPtr<DllServices> dllSvc(DllServices::Get());
+
+ for (unsigned int i = 0, n = mRawModules.GetSize(); i != n; i++) {
+ const SharedLibrary& info = mRawModules.GetEntry(i);
+
+ auto orgName = dllSvc->GetBinaryOrgName(info.GetModulePath().get());
+ if (orgName) {
+ mCertSubjects.Put(info.GetModulePath(),
+ nsDependentString(orgName.get()));
+ }
+ }
+ }
+# endif // defined(XP_WIN)
+};
+
+class GetLoadedModulesRunnable final : public Runnable {
+ nsMainThreadPtrHandle<Promise> mPromise;
+
+ public:
+ explicit GetLoadedModulesRunnable(
+ const nsMainThreadPtrHandle<Promise>& aPromise)
+ : mozilla::Runnable("GetLoadedModulesRunnable"), mPromise(aPromise) {}
+
+ NS_IMETHOD
+ Run() override {
+ nsCOMPtr<nsIRunnable> resultRunnable = new GetLoadedModulesResultRunnable(
+ mPromise, SharedLibraryInfo::GetInfoForSelf());
+ return NS_DispatchToMainThread(resultRunnable);
+ }
+};
+#endif // MOZ_GECKO_PROFILER
+
+NS_IMETHODIMP
+TelemetryImpl::GetLoadedModules(JSContext* cx, Promise** aPromise) {
+#if defined(MOZ_GECKO_PROFILER)
+ nsIGlobalObject* global = xpc::CurrentNativeGlobal(cx);
+ if (NS_WARN_IF(!global)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ ErrorResult result;
+ RefPtr<Promise> promise = Promise::Create(global, result);
+ if (NS_WARN_IF(result.Failed())) {
+ return result.StealNSResult();
+ }
+
+ nsCOMPtr<nsIThreadManager> tm = do_GetService(NS_THREADMANAGER_CONTRACTID);
+ nsCOMPtr<nsIThread> getModulesThread;
+ nsresult rv = tm->NewThread(0, 0, getter_AddRefs(getModulesThread));
+ if (NS_WARN_IF(NS_FAILED(rv))) {
+ promise->MaybeReject(NS_ERROR_FAILURE);
+ return NS_OK;
+ }
+
+ nsMainThreadPtrHandle<Promise> mainThreadPromise(
+ new nsMainThreadPtrHolder<Promise>(
+ "TelemetryImpl::GetLoadedModules::Promise", promise));
+ nsCOMPtr<nsIRunnable> runnable =
+ new GetLoadedModulesRunnable(mainThreadPromise);
+ promise.forget(aPromise);
+
+ return getModulesThread->Dispatch(runnable, nsIEventTarget::DISPATCH_NORMAL);
+#else // MOZ_GECKO_PROFILER
+ return NS_ERROR_NOT_IMPLEMENTED;
+#endif // MOZ_GECKO_PROFILER
+}
+
+static bool IsValidBreakpadId(const std::string& breakpadId) {
+ if (breakpadId.size() < 33) {
+ return false;
+ }
+ for (char c : breakpadId) {
+ if ((c < '0' || c > '9') && (c < 'A' || c > 'F')) {
+ return false;
+ }
+ }
+ return true;
+}
+
+// Read a stack from the given file name. In case of any error, aStack is
+// unchanged.
+static void ReadStack(PathCharPtr aFileName,
+ Telemetry::ProcessedStack& aStack) {
+ IFStream file(aFileName);
+
+ size_t numModules;
+ file >> numModules;
+ if (file.fail()) {
+ return;
+ }
+
+ char newline = file.get();
+ if (file.fail() || newline != '\n') {
+ return;
+ }
+
+ Telemetry::ProcessedStack stack;
+ for (size_t i = 0; i < numModules; ++i) {
+ std::string breakpadId;
+ file >> breakpadId;
+ if (file.fail() || !IsValidBreakpadId(breakpadId)) {
+ return;
+ }
+
+ char space = file.get();
+ if (file.fail() || space != ' ') {
+ return;
+ }
+
+ std::string moduleName;
+ getline(file, moduleName);
+ if (file.fail() || moduleName[0] == ' ') {
+ return;
+ }
+
+ Telemetry::ProcessedStack::Module module = {
+ NS_ConvertUTF8toUTF16(moduleName.c_str()),
+ nsCString(breakpadId.c_str(), breakpadId.size()),
+ };
+ stack.AddModule(module);
+ }
+
+ size_t numFrames;
+ file >> numFrames;
+ if (file.fail()) {
+ return;
+ }
+
+ newline = file.get();
+ if (file.fail() || newline != '\n') {
+ return;
+ }
+
+ for (size_t i = 0; i < numFrames; ++i) {
+ uint16_t index;
+ file >> index;
+ uintptr_t offset;
+ file >> std::hex >> offset >> std::dec;
+ if (file.fail()) {
+ return;
+ }
+
+ Telemetry::ProcessedStack::Frame frame = {offset, index};
+ stack.AddFrame(frame);
+ }
+
+ aStack = stack;
+}
+
+void TelemetryImpl::ReadLateWritesStacks(nsIFile* aProfileDir) {
+ nsCOMPtr<nsIDirectoryEnumerator> files;
+ if (NS_FAILED(aProfileDir->GetDirectoryEntries(getter_AddRefs(files)))) {
+ return;
+ }
+
+ constexpr auto prefix = u"Telemetry.LateWriteFinal-"_ns;
+ nsCOMPtr<nsIFile> file;
+ while (NS_SUCCEEDED(files->GetNextFile(getter_AddRefs(file))) && file) {
+ nsAutoString leafName;
+ if (NS_FAILED(file->GetLeafName(leafName)) ||
+ !StringBeginsWith(leafName, prefix)) {
+ continue;
+ }
+
+ Telemetry::ProcessedStack stack;
+ ReadStack(file->NativePath().get(), stack);
+ if (stack.GetStackSize() != 0) {
+ mLateWritesStacks.AddStack(stack);
+ }
+ // Delete the file so that we don't report it again on the next run.
+ file->Remove(false);
+ }
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetLateWrites(JSContext* cx, JS::MutableHandle<JS::Value> ret) {
+ // The user must call AsyncReadTelemetryData first. We return an empty list
+ // instead of reporting a failure so that the rest of telemetry can uniformly
+ // handle the read not being available yet.
+
+ // FIXME: we allocate the js object again and again in the getter. We should
+ // figure out a way to cache it. In order to do that we have to call
+ // JS_AddNamedObjectRoot. A natural place to do so is in the TelemetryImpl
+ // constructor, but it is not clear how to get a JSContext in there.
+ // Another option would be to call it in here when we first call
+ // CreateJSStackObject, but we would still need to figure out where to call
+ // JS_RemoveObjectRoot. Would it be ok to never call JS_RemoveObjectRoot
+ // and just set the pointer to nullptr is the telemetry destructor?
+
+ JSObject* report;
+ if (!mCachedTelemetryData) {
+ CombinedStacks empty;
+ report = CreateJSStackObject(cx, empty);
+ } else {
+ report = CreateJSStackObject(cx, mLateWritesStacks);
+ }
+
+ if (report == nullptr) {
+ return NS_ERROR_FAILURE;
+ }
+
+ ret.setObject(*report);
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetHistogramById(const nsACString& name, JSContext* cx,
+ JS::MutableHandle<JS::Value> ret) {
+ return TelemetryHistogram::GetHistogramById(name, cx, ret);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetKeyedHistogramById(const nsACString& name, JSContext* cx,
+ JS::MutableHandle<JS::Value> ret) {
+ return TelemetryHistogram::GetKeyedHistogramById(name, cx, ret);
+}
+
+/**
+ * Indicates if Telemetry can record base data (FHR data). This is true if the
+ * FHR data reporting service or self-support are enabled.
+ *
+ * In the unlikely event that adding a new base probe is needed, please check
+ * the data collection wiki at https://wiki.mozilla.org/Firefox/Data_Collection
+ * and talk to the Telemetry team.
+ */
+NS_IMETHODIMP
+TelemetryImpl::GetCanRecordBase(bool* ret) {
+ *ret = mCanRecordBase;
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::SetCanRecordBase(bool canRecord) {
+#ifndef FUZZING
+ if (canRecord != mCanRecordBase) {
+ TelemetryHistogram::SetCanRecordBase(canRecord);
+ TelemetryScalar::SetCanRecordBase(canRecord);
+ TelemetryEvent::SetCanRecordBase(canRecord);
+ mCanRecordBase = canRecord;
+ }
+#endif
+ return NS_OK;
+}
+
+/**
+ * Indicates if Telemetry is allowed to record extended data. Returns false if
+ * the user hasn't opted into "extended Telemetry" on the Release channel, when
+ * the user has explicitly opted out of Telemetry on Nightly/Aurora/Beta or if
+ * manually set to false during tests. If the returned value is false, gathering
+ * of extended telemetry statistics is disabled.
+ */
+NS_IMETHODIMP
+TelemetryImpl::GetCanRecordExtended(bool* ret) {
+ *ret = mCanRecordExtended;
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::SetCanRecordExtended(bool canRecord) {
+#ifndef FUZZING
+ if (canRecord != mCanRecordExtended) {
+ TelemetryHistogram::SetCanRecordExtended(canRecord);
+ TelemetryScalar::SetCanRecordExtended(canRecord);
+ TelemetryEvent::SetCanRecordExtended(canRecord);
+ mCanRecordExtended = canRecord;
+ }
+#endif
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetCanRecordReleaseData(bool* ret) {
+ *ret = mCanRecordBase;
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetCanRecordPrereleaseData(bool* ret) {
+ *ret = mCanRecordExtended;
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetIsOfficialTelemetry(bool* ret) {
+#if defined(MOZILLA_OFFICIAL) && defined(MOZ_TELEMETRY_REPORTING) && \
+ !defined(DEBUG)
+ *ret = true;
+#else
+ *ret = false;
+#endif
+ return NS_OK;
+}
+
+already_AddRefed<nsITelemetry> TelemetryImpl::CreateTelemetryInstance() {
+ {
+ auto lock = sTelemetry.Lock();
+ MOZ_ASSERT(
+ *lock == nullptr,
+ "CreateTelemetryInstance may only be called once, via GetService()");
+ }
+
+ bool useTelemetry = false;
+#ifndef FUZZING
+ if (XRE_IsParentProcess() || XRE_IsContentProcess() || XRE_IsGPUProcess() ||
+ XRE_IsSocketProcess()) {
+ useTelemetry = true;
+ }
+#endif
+
+ // First, initialize the TelemetryHistogram and TelemetryScalar global states.
+ TelemetryHistogram::InitializeGlobalState(useTelemetry, useTelemetry);
+ TelemetryScalar::InitializeGlobalState(useTelemetry, useTelemetry);
+
+ // Only record events from the parent process.
+ TelemetryEvent::InitializeGlobalState(XRE_IsParentProcess(),
+ XRE_IsParentProcess());
+ TelemetryOrigin::InitializeGlobalState();
+ // Currently, only UserInteractions from the parent process are recorded.
+ TelemetryUserInteraction::InitializeGlobalState(useTelemetry, useTelemetry);
+
+ // Now, create and initialize the Telemetry global state.
+ TelemetryImpl* telemetry = new TelemetryImpl();
+ {
+ auto lock = sTelemetry.Lock();
+ *lock = telemetry;
+ // AddRef for the local reference before releasing the lock.
+ NS_ADDREF(telemetry);
+ }
+
+ // AddRef for the caller
+ nsCOMPtr<nsITelemetry> ret = telemetry;
+
+ telemetry->mCanRecordBase = useTelemetry;
+ telemetry->mCanRecordExtended = useTelemetry;
+
+ telemetry->InitMemoryReporter();
+ InitHistogramRecordingEnabled(); // requires sTelemetry to exist
+
+ return ret.forget();
+}
+
+void TelemetryImpl::ShutdownTelemetry() {
+ // No point in collecting IO beyond this point
+ ClearIOReporting();
+ {
+ auto lock = sTelemetry.Lock();
+ NS_IF_RELEASE(lock.ref());
+ }
+
+ // Lastly, de-initialise the TelemetryHistogram and TelemetryScalar global
+ // states, so as to release any heap storage that would otherwise be kept
+ // alive by it.
+ TelemetryHistogram::DeInitializeGlobalState();
+ TelemetryScalar::DeInitializeGlobalState();
+ TelemetryEvent::DeInitializeGlobalState();
+ TelemetryOrigin::DeInitializeGlobalState();
+ TelemetryUserInteraction::DeInitializeGlobalState();
+ TelemetryIPCAccumulator::DeInitializeGlobalState();
+}
+
+void TelemetryImpl::StoreSlowSQL(const nsACString& sql, uint32_t delay,
+ SanitizedState state) {
+ auto lock = sTelemetry.Lock();
+ auto telemetry = lock.ref();
+ AutoHashtable<SlowSQLEntryType>* slowSQLMap = nullptr;
+ if (state == Sanitized)
+ slowSQLMap = &(telemetry->mSanitizedSQL);
+ else
+ slowSQLMap = &(telemetry->mPrivateSQL);
+
+ MutexAutoLock hashMutex(telemetry->mHashMutex);
+
+ SlowSQLEntryType* entry = slowSQLMap->GetEntry(sql);
+ if (!entry) {
+ entry = slowSQLMap->PutEntry(sql);
+ if (MOZ_UNLIKELY(!entry)) return;
+ entry->GetModifiableData()->mainThread.hitCount = 0;
+ entry->GetModifiableData()->mainThread.totalTime = 0;
+ entry->GetModifiableData()->otherThreads.hitCount = 0;
+ entry->GetModifiableData()->otherThreads.totalTime = 0;
+ }
+
+ if (NS_IsMainThread()) {
+ entry->GetModifiableData()->mainThread.hitCount++;
+ entry->GetModifiableData()->mainThread.totalTime += delay;
+ } else {
+ entry->GetModifiableData()->otherThreads.hitCount++;
+ entry->GetModifiableData()->otherThreads.totalTime += delay;
+ }
+}
+
+/**
+ * This method replaces string literals in SQL strings with the word :private
+ *
+ * States used in this state machine:
+ *
+ * NORMAL:
+ * - This is the active state when not iterating over a string literal or
+ * comment
+ *
+ * SINGLE_QUOTE:
+ * - Defined here: http://www.sqlite.org/lang_expr.html
+ * - This state represents iterating over a string literal opened with
+ * a single quote.
+ * - A single quote within the string can be encoded by putting 2 single quotes
+ * in a row, e.g. 'This literal contains an escaped quote '''
+ * - Any double quotes found within a single-quoted literal are ignored
+ * - This state covers BLOB literals, e.g. X'ABC123'
+ * - The string literal and the enclosing quotes will be replaced with
+ * the text :private
+ *
+ * DOUBLE_QUOTE:
+ * - Same rules as the SINGLE_QUOTE state.
+ * - According to http://www.sqlite.org/lang_keywords.html,
+ * SQLite interprets text in double quotes as an identifier unless it's used in
+ * a context where it cannot be resolved to an identifier and a string literal
+ * is allowed. This method removes text in double-quotes for safety.
+ *
+ * DASH_COMMENT:
+ * - http://www.sqlite.org/lang_comment.html
+ * - A dash comment starts with two dashes in a row,
+ * e.g. DROP TABLE foo -- a comment
+ * - Any text following two dashes in a row is interpreted as a comment until
+ * end of input or a newline character
+ * - Any quotes found within the comment are ignored and no replacements made
+ *
+ * C_STYLE_COMMENT:
+ * - http://www.sqlite.org/lang_comment.html
+ * - A C-style comment starts with a forward slash and an asterisk, and ends
+ * with an asterisk and a forward slash
+ * - Any text following comment start is interpreted as a comment up to end of
+ * input or comment end
+ * - Any quotes found within the comment are ignored and no replacements made
+ */
+nsCString TelemetryImpl::SanitizeSQL(const nsACString& sql) {
+ nsCString output;
+ int length = sql.Length();
+
+ typedef enum {
+ NORMAL,
+ SINGLE_QUOTE,
+ DOUBLE_QUOTE,
+ DASH_COMMENT,
+ C_STYLE_COMMENT,
+ } State;
+
+ State state = NORMAL;
+ int fragmentStart = 0;
+ for (int i = 0; i < length; i++) {
+ char character = sql[i];
+ char nextCharacter = (i + 1 < length) ? sql[i + 1] : '\0';
+
+ switch (character) {
+ case '\'':
+ case '"':
+ if (state == NORMAL) {
+ state = (character == '\'') ? SINGLE_QUOTE : DOUBLE_QUOTE;
+ output +=
+ nsDependentCSubstring(sql, fragmentStart, i - fragmentStart);
+ output += ":private";
+ fragmentStart = -1;
+ } else if ((state == SINGLE_QUOTE && character == '\'') ||
+ (state == DOUBLE_QUOTE && character == '"')) {
+ if (nextCharacter == character) {
+ // Two consecutive quotes within a string literal are a single
+ // escaped quote
+ i++;
+ } else {
+ state = NORMAL;
+ fragmentStart = i + 1;
+ }
+ }
+ break;
+ case '-':
+ if (state == NORMAL) {
+ if (nextCharacter == '-') {
+ state = DASH_COMMENT;
+ i++;
+ }
+ }
+ break;
+ case '\n':
+ if (state == DASH_COMMENT) {
+ state = NORMAL;
+ }
+ break;
+ case '/':
+ if (state == NORMAL) {
+ if (nextCharacter == '*') {
+ state = C_STYLE_COMMENT;
+ i++;
+ }
+ }
+ break;
+ case '*':
+ if (state == C_STYLE_COMMENT) {
+ if (nextCharacter == '/') {
+ state = NORMAL;
+ }
+ }
+ break;
+ default:
+ continue;
+ }
+ }
+
+ if ((fragmentStart >= 0) && fragmentStart < length)
+ output += nsDependentCSubstring(sql, fragmentStart, length - fragmentStart);
+
+ return output;
+}
+
+// An allowlist mechanism to prevent Telemetry reporting on Addon & Thunderbird
+// DBs.
+struct TrackedDBEntry {
+ const char* mName;
+ const uint32_t mNameLength;
+
+ // This struct isn't meant to be used beyond the static arrays below.
+ constexpr TrackedDBEntry(const char* aName, uint32_t aNameLength)
+ : mName(aName), mNameLength(aNameLength) {}
+
+ TrackedDBEntry() = delete;
+ TrackedDBEntry(TrackedDBEntry&) = delete;
+};
+
+#define TRACKEDDB_ENTRY(_name) \
+ { _name, (sizeof(_name) - 1) }
+
+// An allowlist of database names. If the database name exactly matches one of
+// these then its SQL statements will always be recorded.
+static constexpr TrackedDBEntry kTrackedDBs[] = {
+ // IndexedDB for about:home, see aboutHome.js
+ TRACKEDDB_ENTRY("818200132aebmoouht.sqlite"),
+ TRACKEDDB_ENTRY("addons.sqlite"),
+ TRACKEDDB_ENTRY("content-prefs.sqlite"),
+ TRACKEDDB_ENTRY("cookies.sqlite"),
+ TRACKEDDB_ENTRY("extensions.sqlite"),
+ TRACKEDDB_ENTRY("favicons.sqlite"),
+ TRACKEDDB_ENTRY("formhistory.sqlite"),
+ TRACKEDDB_ENTRY("index.sqlite"),
+ TRACKEDDB_ENTRY("netpredictions.sqlite"),
+ TRACKEDDB_ENTRY("permissions.sqlite"),
+ TRACKEDDB_ENTRY("places.sqlite"),
+ TRACKEDDB_ENTRY("reading-list.sqlite"),
+ TRACKEDDB_ENTRY("search.sqlite"),
+ TRACKEDDB_ENTRY("signons.sqlite"),
+ TRACKEDDB_ENTRY("urlclassifier3.sqlite"),
+ TRACKEDDB_ENTRY("webappsstore.sqlite")};
+
+// An allowlist of database name prefixes. If the database name begins with
+// one of these prefixes then its SQL statements will always be recorded.
+static const TrackedDBEntry kTrackedDBPrefixes[] = {
+ TRACKEDDB_ENTRY("indexedDB-")};
+
+#undef TRACKEDDB_ENTRY
+
+// Slow SQL statements will be automatically
+// trimmed to kMaxSlowStatementLength characters.
+// This limit doesn't include the ellipsis and DB name,
+// that are appended at the end of the stored statement.
+const uint32_t kMaxSlowStatementLength = 1000;
+
+void TelemetryImpl::RecordSlowStatement(const nsACString& sql,
+ const nsACString& dbName,
+ uint32_t delay) {
+ MOZ_ASSERT(!sql.IsEmpty());
+ MOZ_ASSERT(!dbName.IsEmpty());
+
+ {
+ auto lock = sTelemetry.Lock();
+ if (!lock.ref() || !TelemetryHistogram::CanRecordExtended()) {
+ return;
+ }
+ }
+
+ bool recordStatement = false;
+
+ for (const TrackedDBEntry& nameEntry : kTrackedDBs) {
+ MOZ_ASSERT(nameEntry.mNameLength);
+ const nsDependentCString name(nameEntry.mName, nameEntry.mNameLength);
+ if (dbName == name) {
+ recordStatement = true;
+ break;
+ }
+ }
+
+ if (!recordStatement) {
+ for (const TrackedDBEntry& prefixEntry : kTrackedDBPrefixes) {
+ MOZ_ASSERT(prefixEntry.mNameLength);
+ const nsDependentCString prefix(prefixEntry.mName,
+ prefixEntry.mNameLength);
+ if (StringBeginsWith(dbName, prefix)) {
+ recordStatement = true;
+ break;
+ }
+ }
+ }
+
+ if (recordStatement) {
+ nsAutoCString sanitizedSQL(SanitizeSQL(sql));
+ if (sanitizedSQL.Length() > kMaxSlowStatementLength) {
+ sanitizedSQL.SetLength(kMaxSlowStatementLength);
+ sanitizedSQL += "...";
+ }
+ sanitizedSQL.AppendPrintf(" /* %s */", nsPromiseFlatCString(dbName).get());
+ StoreSlowSQL(sanitizedSQL, delay, Sanitized);
+ } else {
+ // Report aggregate DB-level statistics for addon DBs
+ nsAutoCString aggregate;
+ aggregate.AppendPrintf("Untracked SQL for %s",
+ nsPromiseFlatCString(dbName).get());
+ StoreSlowSQL(aggregate, delay, Sanitized);
+ }
+
+ nsAutoCString fullSQL;
+ fullSQL.AppendPrintf("%s /* %s */", nsPromiseFlatCString(sql).get(),
+ nsPromiseFlatCString(dbName).get());
+ StoreSlowSQL(fullSQL, delay, Unsanitized);
+}
+
+bool TelemetryImpl::CanRecordBase() {
+ auto lock = sTelemetry.Lock();
+ auto telemetry = lock.ref();
+ if (!telemetry) {
+ return false;
+ }
+ bool canRecordBase;
+ nsresult rv = telemetry->GetCanRecordBase(&canRecordBase);
+ return NS_SUCCEEDED(rv) && canRecordBase;
+}
+
+bool TelemetryImpl::CanRecordExtended() {
+ auto lock = sTelemetry.Lock();
+ auto telemetry = lock.ref();
+ if (!telemetry) {
+ return false;
+ }
+ bool canRecordExtended;
+ nsresult rv = telemetry->GetCanRecordExtended(&canRecordExtended);
+ return NS_SUCCEEDED(rv) && canRecordExtended;
+}
+
+bool TelemetryImpl::CanRecordReleaseData() { return CanRecordBase(); }
+
+bool TelemetryImpl::CanRecordPrereleaseData() { return CanRecordExtended(); }
+
+NS_IMPL_ISUPPORTS(TelemetryImpl, nsITelemetry, nsIMemoryReporter)
+
+NS_IMETHODIMP
+TelemetryImpl::GetFileIOReports(JSContext* cx, JS::MutableHandleValue ret) {
+ if (sTelemetryIOObserver) {
+ JS::Rooted<JSObject*> obj(cx, JS_NewPlainObject(cx));
+ if (!obj) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!sTelemetryIOObserver->ReflectIntoJS(cx, obj)) {
+ return NS_ERROR_FAILURE;
+ }
+ ret.setObject(*obj);
+ return NS_OK;
+ }
+ ret.setNull();
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::MsSinceProcessStart(double* aResult) {
+ return Telemetry::Common::MsSinceProcessStart(aResult);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::MsSinceProcessStartIncludingSuspend(double* aResult) {
+ return Telemetry::Common::MsSinceProcessStartIncludingSuspend(aResult);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::MsSinceProcessStartExcludingSuspend(double* aResult) {
+ return Telemetry::Common::MsSinceProcessStartExcludingSuspend(aResult);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::MsSystemNow(double* aResult) {
+#if defined(XP_UNIX) && !defined(XP_DARWIN)
+ timespec ts;
+ clock_gettime(CLOCK_REALTIME, &ts);
+ *aResult = ts.tv_sec * 1000 + ts.tv_nsec / 1000000;
+#else
+ using namespace std::chrono;
+ milliseconds ms =
+ duration_cast<milliseconds>(system_clock::now().time_since_epoch());
+ *aResult = static_cast<double>(ms.count());
+#endif // XP_UNIX && !XP_DARWIN
+
+ return NS_OK;
+}
+
+// Telemetry Scalars IDL Implementation
+
+NS_IMETHODIMP
+TelemetryImpl::ScalarAdd(const nsACString& aName, JS::HandleValue aVal,
+ JSContext* aCx) {
+ return TelemetryScalar::Add(aName, aVal, aCx);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::ScalarSet(const nsACString& aName, JS::HandleValue aVal,
+ JSContext* aCx) {
+ return TelemetryScalar::Set(aName, aVal, aCx);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::ScalarSetMaximum(const nsACString& aName, JS::HandleValue aVal,
+ JSContext* aCx) {
+ return TelemetryScalar::SetMaximum(aName, aVal, aCx);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::KeyedScalarAdd(const nsACString& aName, const nsAString& aKey,
+ JS::HandleValue aVal, JSContext* aCx) {
+ return TelemetryScalar::Add(aName, aKey, aVal, aCx);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::KeyedScalarSet(const nsACString& aName, const nsAString& aKey,
+ JS::HandleValue aVal, JSContext* aCx) {
+ return TelemetryScalar::Set(aName, aKey, aVal, aCx);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::KeyedScalarSetMaximum(const nsACString& aName,
+ const nsAString& aKey,
+ JS::HandleValue aVal, JSContext* aCx) {
+ return TelemetryScalar::SetMaximum(aName, aKey, aVal, aCx);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::RegisterScalars(const nsACString& aCategoryName,
+ JS::Handle<JS::Value> aScalarData,
+ JSContext* cx) {
+ return TelemetryScalar::RegisterScalars(aCategoryName, aScalarData, false,
+ cx);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::RegisterBuiltinScalars(const nsACString& aCategoryName,
+ JS::Handle<JS::Value> aScalarData,
+ JSContext* cx) {
+ return TelemetryScalar::RegisterScalars(aCategoryName, aScalarData, true, cx);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::ClearScalars() {
+ TelemetryScalar::ClearScalars();
+ return NS_OK;
+}
+
+// Telemetry Event IDL implementation.
+
+NS_IMETHODIMP
+TelemetryImpl::RecordEvent(const nsACString& aCategory,
+ const nsACString& aMethod, const nsACString& aObject,
+ JS::HandleValue aValue, JS::HandleValue aExtra,
+ JSContext* aCx, uint8_t optional_argc) {
+ return TelemetryEvent::RecordEvent(aCategory, aMethod, aObject, aValue,
+ aExtra, aCx, optional_argc);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::SnapshotEvents(uint32_t aDataset, bool aClear,
+ uint32_t aEventLimit, JSContext* aCx,
+ uint8_t optional_argc,
+ JS::MutableHandleValue aResult) {
+ return TelemetryEvent::CreateSnapshots(aDataset, aClear, aEventLimit, aCx,
+ optional_argc, aResult);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::RegisterEvents(const nsACString& aCategory,
+ JS::Handle<JS::Value> aEventData, JSContext* cx) {
+ return TelemetryEvent::RegisterEvents(aCategory, aEventData, false, cx);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::RegisterBuiltinEvents(const nsACString& aCategory,
+ JS::Handle<JS::Value> aEventData,
+ JSContext* cx) {
+ return TelemetryEvent::RegisterEvents(aCategory, aEventData, true, cx);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::ClearEvents() {
+ TelemetryEvent::ClearEvents();
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::SetEventRecordingEnabled(const nsACString& aCategory,
+ bool aEnabled) {
+ TelemetryEvent::SetEventRecordingEnabled(aCategory, aEnabled);
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetOriginSnapshot(bool aClear, JSContext* aCx,
+ JS::MutableHandleValue aResult) {
+ return TelemetryOrigin::GetOriginSnapshot(aClear, aCx, aResult);
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetEncodedOriginSnapshot(bool aClear, JSContext* aCx,
+ Promise** aResult) {
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+ NS_ENSURE_ARG_POINTER(aResult);
+ nsIGlobalObject* global = xpc::CurrentNativeGlobal(aCx);
+ if (NS_WARN_IF(!global)) {
+ return NS_ERROR_FAILURE;
+ }
+ ErrorResult erv;
+ RefPtr<Promise> promise = Promise::Create(global, erv);
+ if (NS_WARN_IF(erv.Failed())) {
+ return erv.StealNSResult();
+ }
+
+ // TODO: Put this all on a Worker Thread
+
+ JS::RootedValue snapshot(aCx);
+ nsresult rv;
+ rv = TelemetryOrigin::GetEncodedOriginSnapshot(aClear, aCx, &snapshot);
+ if (NS_WARN_IF(NS_FAILED(rv))) {
+ return rv;
+ }
+ promise->MaybeResolve(snapshot);
+ promise.forget(aResult);
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::ClearOrigins() {
+ TelemetryOrigin::ClearOrigins();
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::FlushBatchedChildTelemetry() {
+ TelemetryIPCAccumulator::IPCTimerFired(nullptr, nullptr);
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::EarlyInit() {
+ Unused << MemoryTelemetry::Get();
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::DelayedInit() {
+ MemoryTelemetry::Get().DelayedInit();
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::Shutdown() {
+ MemoryTelemetry::Get().Shutdown();
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GatherMemory(JSContext* aCx, Promise** aResult) {
+ ErrorResult rv;
+ RefPtr<Promise> promise = Promise::Create(xpc::CurrentNativeGlobal(aCx), rv);
+ if (rv.Failed()) {
+ return rv.StealNSResult();
+ }
+
+ MemoryTelemetry::Get().GatherReports(
+ [promise]() { promise->MaybeResolve(JS::UndefinedHandleValue); });
+
+ promise.forget(aResult);
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+TelemetryImpl::GetAllStores(JSContext* aCx, JS::MutableHandleValue aResult) {
+ StringHashSet stores;
+ nsresult rv;
+
+ rv = TelemetryHistogram::GetAllStores(stores);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ rv = TelemetryScalar::GetAllStores(stores);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ JS::RootedVector<JS::Value> allStores(aCx);
+ if (!allStores.reserve(stores.Count())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ for (auto iter = stores.Iter(); !iter.Done(); iter.Next()) {
+ auto& value = iter.Get()->GetKey();
+ JS::RootedValue store(aCx);
+
+ store.setString(ToJSString(aCx, value));
+ if (!allStores.append(store)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ JS::Rooted<JSObject*> rarray(aCx, JS::NewArrayObject(aCx, allStores));
+ if (rarray == nullptr) {
+ return NS_ERROR_FAILURE;
+ }
+ aResult.setObject(*rarray);
+
+ return NS_OK;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// EXTERNALLY VISIBLE FUNCTIONS in no name space
+// These are NOT listed in Telemetry.h
+
+/**
+ * The XRE_TelemetryAdd function is to be used by embedding applications
+ * that can't use mozilla::Telemetry::Accumulate() directly.
+ */
+void XRE_TelemetryAccumulate(int aID, uint32_t aSample) {
+ mozilla::Telemetry::Accumulate((mozilla::Telemetry::HistogramID)aID, aSample);
+}
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// EXTERNALLY VISIBLE FUNCTIONS in mozilla::
+// These are NOT listed in Telemetry.h
+
+namespace mozilla {
+
+void RecordShutdownStartTimeStamp() {
+#ifdef DEBUG
+ // FIXME: this function should only be called once, since it should be called
+ // at the earliest point we *know* we are shutting down. Unfortunately
+ // this assert has been firing. Given that if we are called multiple times
+ // we just keep the last timestamp, the assert is commented for now.
+ static bool recorded = false;
+ // MOZ_ASSERT(!recorded);
+ (void)
+ recorded; // Silence unused-var warnings (remove when assert re-enabled)
+ recorded = true;
+#endif
+
+ if (!Telemetry::CanRecordExtended()) return;
+
+ gRecordedShutdownStartTime = TimeStamp::Now();
+
+ GetShutdownTimeFileName();
+}
+
+void RecordShutdownEndTimeStamp() {
+ if (!gRecordedShutdownTimeFileName || gAlreadyFreedShutdownTimeFileName)
+ return;
+
+ PathString name(gRecordedShutdownTimeFileName);
+ free(const_cast<PathChar*>(gRecordedShutdownTimeFileName));
+ gRecordedShutdownTimeFileName = nullptr;
+ gAlreadyFreedShutdownTimeFileName = true;
+
+ if (gRecordedShutdownStartTime.IsNull()) {
+ // If |CanRecordExtended()| is true before |AsyncFetchTelemetryData| is
+ // called and then disabled before shutdown, |RecordShutdownStartTimeStamp|
+ // will bail out and we will end up with a null |gRecordedShutdownStartTime|
+ // here. This can happen during tests.
+ return;
+ }
+
+ nsTAutoString<PathChar> tmpName(name);
+ tmpName.AppendLiteral(".tmp");
+ RefPtr<nsLocalFile> tmpFile = new nsLocalFile(tmpName);
+ FILE* f;
+ if (NS_FAILED(tmpFile->OpenANSIFileDesc("w", &f)) || !f) return;
+ // On a normal release build this should be called just before
+ // calling _exit, but on a debug build or when the user forces a full
+ // shutdown this is called as late as possible, so we have to
+ // allow this write as write poisoning will be enabled.
+ MozillaRegisterDebugFILE(f);
+
+ TimeStamp now = TimeStamp::Now();
+ MOZ_ASSERT(now >= gRecordedShutdownStartTime);
+ TimeDuration diff = now - gRecordedShutdownStartTime;
+ uint32_t diff2 = diff.ToMilliseconds();
+ int written = fprintf(f, "%d\n", diff2);
+ MozillaUnRegisterDebugFILE(f);
+ int rv = fclose(f);
+ if (written < 0 || rv != 0) {
+ tmpFile->Remove(false);
+ return;
+ }
+ RefPtr<nsLocalFile> file = new nsLocalFile(name);
+ nsAutoString leafName;
+ file->GetLeafName(leafName);
+ tmpFile->RenameTo(nullptr, leafName);
+}
+
+} // namespace mozilla
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// EXTERNALLY VISIBLE FUNCTIONS in mozilla::Telemetry::
+// These are listed in Telemetry.h
+
+namespace mozilla::Telemetry {
+
+// The external API for controlling recording state
+void SetHistogramRecordingEnabled(HistogramID aID, bool aEnabled) {
+ TelemetryHistogram::SetHistogramRecordingEnabled(aID, aEnabled);
+}
+
+void Accumulate(HistogramID aHistogram, uint32_t aSample) {
+ TelemetryHistogram::Accumulate(aHistogram, aSample);
+}
+
+void Accumulate(HistogramID aHistogram, const nsTArray<uint32_t>& aSamples) {
+ TelemetryHistogram::Accumulate(aHistogram, aSamples);
+}
+
+void Accumulate(HistogramID aID, const nsCString& aKey, uint32_t aSample) {
+ TelemetryHistogram::Accumulate(aID, aKey, aSample);
+}
+
+void Accumulate(HistogramID aID, const nsCString& aKey,
+ const nsTArray<uint32_t>& aSamples) {
+ TelemetryHistogram::Accumulate(aID, aKey, aSamples);
+}
+
+void Accumulate(const char* name, uint32_t sample) {
+ TelemetryHistogram::Accumulate(name, sample);
+}
+
+void Accumulate(const char* name, const nsCString& key, uint32_t sample) {
+ TelemetryHistogram::Accumulate(name, key, sample);
+}
+
+void AccumulateCategorical(HistogramID id, const nsCString& label) {
+ TelemetryHistogram::AccumulateCategorical(id, label);
+}
+
+void AccumulateCategorical(HistogramID id, const nsTArray<nsCString>& labels) {
+ TelemetryHistogram::AccumulateCategorical(id, labels);
+}
+
+void AccumulateTimeDelta(HistogramID aHistogram, TimeStamp start,
+ TimeStamp end) {
+ if (start > end) {
+ Accumulate(aHistogram, 0);
+ return;
+ }
+ Accumulate(aHistogram, static_cast<uint32_t>((end - start).ToMilliseconds()));
+}
+
+void AccumulateTimeDelta(HistogramID aHistogram, const nsCString& key,
+ TimeStamp start, TimeStamp end) {
+ if (start > end) {
+ Accumulate(aHistogram, key, 0);
+ return;
+ }
+ Accumulate(aHistogram, key,
+ static_cast<uint32_t>((end - start).ToMilliseconds()));
+}
+const char* GetHistogramName(HistogramID id) {
+ return TelemetryHistogram::GetHistogramName(id);
+}
+
+bool CanRecordBase() { return TelemetryImpl::CanRecordBase(); }
+
+bool CanRecordExtended() { return TelemetryImpl::CanRecordExtended(); }
+
+bool CanRecordReleaseData() { return TelemetryImpl::CanRecordReleaseData(); }
+
+bool CanRecordPrereleaseData() {
+ return TelemetryImpl::CanRecordPrereleaseData();
+}
+
+void RecordSlowSQLStatement(const nsACString& statement,
+ const nsACString& dbName, uint32_t delay) {
+ TelemetryImpl::RecordSlowStatement(statement, dbName, delay);
+}
+
+void Init() {
+ // Make the service manager hold a long-lived reference to the service
+ nsCOMPtr<nsITelemetry> telemetryService =
+ do_GetService("@mozilla.org/base/telemetry;1");
+ MOZ_ASSERT(telemetryService);
+}
+
+void WriteFailedProfileLock(nsIFile* aProfileDir) {
+ nsCOMPtr<nsIFile> file;
+ nsresult rv = GetFailedProfileLockFile(getter_AddRefs(file), aProfileDir);
+ NS_ENSURE_SUCCESS_VOID(rv);
+ int64_t fileSize = 0;
+ rv = file->GetFileSize(&fileSize);
+ // It's expected that the file might not exist yet
+ if (NS_FAILED(rv) && rv != NS_ERROR_FILE_NOT_FOUND) {
+ return;
+ }
+ nsCOMPtr<nsIFileStream> fileStream;
+ rv = NS_NewLocalFileStream(getter_AddRefs(fileStream), file,
+ PR_RDWR | PR_CREATE_FILE, 0640);
+ NS_ENSURE_SUCCESS_VOID(rv);
+ NS_ENSURE_TRUE_VOID(fileSize <= kMaxFailedProfileLockFileSize);
+ unsigned int failedLockCount = 0;
+ if (fileSize > 0) {
+ nsCOMPtr<nsIInputStream> inStream = do_QueryInterface(fileStream);
+ NS_ENSURE_TRUE_VOID(inStream);
+ if (!GetFailedLockCount(inStream, fileSize, failedLockCount)) {
+ failedLockCount = 0;
+ }
+ }
+ ++failedLockCount;
+ nsAutoCString bufStr;
+ bufStr.AppendInt(static_cast<int>(failedLockCount));
+ nsCOMPtr<nsISeekableStream> seekStream = do_QueryInterface(fileStream);
+ NS_ENSURE_TRUE_VOID(seekStream);
+ // If we read in an existing failed lock count, we need to reset the file ptr
+ if (fileSize > 0) {
+ rv = seekStream->Seek(nsISeekableStream::NS_SEEK_SET, 0);
+ NS_ENSURE_SUCCESS_VOID(rv);
+ }
+ nsCOMPtr<nsIOutputStream> outStream = do_QueryInterface(fileStream);
+ uint32_t bytesLeft = bufStr.Length();
+ const char* bytes = bufStr.get();
+ do {
+ uint32_t written = 0;
+ rv = outStream->Write(bytes, bytesLeft, &written);
+ if (NS_FAILED(rv)) {
+ break;
+ }
+ bytes += written;
+ bytesLeft -= written;
+ } while (bytesLeft > 0);
+ seekStream->SetEOF();
+}
+
+void InitIOReporting(nsIFile* aXreDir) {
+ // Never initialize twice
+ if (sTelemetryIOObserver) {
+ return;
+ }
+
+ sTelemetryIOObserver = new TelemetryIOInterposeObserver(aXreDir);
+ IOInterposer::Register(IOInterposeObserver::OpAllWithStaging,
+ sTelemetryIOObserver);
+}
+
+void SetProfileDir(nsIFile* aProfD) {
+ if (!sTelemetryIOObserver || !aProfD) {
+ return;
+ }
+ nsAutoString profDirPath;
+ nsresult rv = aProfD->GetPath(profDirPath);
+ if (NS_FAILED(rv)) {
+ return;
+ }
+ sTelemetryIOObserver->AddPath(profDirPath, u"{profile}"_ns);
+}
+
+// Scalar API C++ Endpoints
+
+void ScalarAdd(mozilla::Telemetry::ScalarID aId, uint32_t aVal) {
+ TelemetryScalar::Add(aId, aVal);
+}
+
+void ScalarSet(mozilla::Telemetry::ScalarID aId, uint32_t aVal) {
+ TelemetryScalar::Set(aId, aVal);
+}
+
+void ScalarSet(mozilla::Telemetry::ScalarID aId, bool aVal) {
+ TelemetryScalar::Set(aId, aVal);
+}
+
+void ScalarSet(mozilla::Telemetry::ScalarID aId, const nsAString& aVal) {
+ TelemetryScalar::Set(aId, aVal);
+}
+
+void ScalarSetMaximum(mozilla::Telemetry::ScalarID aId, uint32_t aVal) {
+ TelemetryScalar::SetMaximum(aId, aVal);
+}
+
+void ScalarAdd(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ uint32_t aVal) {
+ TelemetryScalar::Add(aId, aKey, aVal);
+}
+
+void ScalarSet(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ uint32_t aVal) {
+ TelemetryScalar::Set(aId, aKey, aVal);
+}
+
+void ScalarSet(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ bool aVal) {
+ TelemetryScalar::Set(aId, aKey, aVal);
+}
+
+void ScalarSetMaximum(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ uint32_t aVal) {
+ TelemetryScalar::SetMaximum(aId, aKey, aVal);
+}
+
+void RecordEvent(
+ mozilla::Telemetry::EventID aId, const mozilla::Maybe<nsCString>& aValue,
+ const mozilla::Maybe<CopyableTArray<EventExtraEntry>>& aExtra) {
+ TelemetryEvent::RecordEventNative(aId, aValue, aExtra);
+}
+
+void SetEventRecordingEnabled(const nsACString& aCategory, bool aEnabled) {
+ TelemetryEvent::SetEventRecordingEnabled(aCategory, aEnabled);
+}
+
+void RecordOrigin(mozilla::Telemetry::OriginMetricID aId,
+ const nsACString& aOrigin) {
+ TelemetryOrigin::RecordOrigin(aId, aOrigin);
+}
+
+void ShutdownTelemetry() { TelemetryImpl::ShutdownTelemetry(); }
+
+} // namespace mozilla::Telemetry
+
+NS_IMPL_COMPONENT_FACTORY(nsITelemetry) {
+ return TelemetryImpl::CreateTelemetryInstance().downcast<nsISupports>();
+}
diff --git a/toolkit/components/telemetry/core/Telemetry.h b/toolkit/components/telemetry/core/Telemetry.h
new file mode 100644
index 0000000000..b50836372c
--- /dev/null
+++ b/toolkit/components/telemetry/core/Telemetry.h
@@ -0,0 +1,586 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef Telemetry_h__
+#define Telemetry_h__
+
+#include "mozilla/Maybe.h"
+#include "mozilla/TelemetryEventEnums.h"
+#include "mozilla/TelemetryHistogramEnums.h"
+#include "mozilla/TelemetryOriginEnums.h"
+#include "mozilla/TelemetryScalarEnums.h"
+#include "mozilla/TimeStamp.h"
+#include "nsString.h"
+#include "nsTArray.h"
+#include "nsXULAppAPI.h"
+
+/******************************************************************************
+ * This implements the Telemetry system.
+ * It allows recording into histograms as well some more specialized data
+ * points and gives access to the data.
+ *
+ * For documentation on how to add and use new Telemetry probes, see:
+ * https://developer.mozilla.org/en-US/docs/Mozilla/Performance/Adding_a_new_Telemetry_probe
+ *
+ * For more general information on Telemetry see:
+ * https://wiki.mozilla.org/Telemetry
+ *****************************************************************************/
+
+namespace mozilla {
+namespace Telemetry {
+
+struct HistogramAccumulation;
+struct KeyedHistogramAccumulation;
+struct ScalarAction;
+struct KeyedScalarAction;
+struct ChildEventData;
+struct EventExtraEntry;
+
+/**
+ * Initialize the Telemetry service on the main thread at startup.
+ */
+void Init();
+
+/**
+ * Shutdown the Telemetry service.
+ */
+void ShutdownTelemetry();
+
+/**
+ * Adds sample to a histogram defined in TelemetryHistogramEnums.h
+ *
+ * @param id - histogram id
+ * @param sample - value to record.
+ */
+void Accumulate(HistogramID id, uint32_t sample);
+
+/**
+ * Adds an array of samples to a histogram defined in TelemetryHistograms.h
+ * @param id - histogram id
+ * @param samples - values to record.
+ */
+void Accumulate(HistogramID id, const nsTArray<uint32_t>& samples);
+
+/**
+ * Adds sample to a keyed histogram defined in TelemetryHistogramEnums.h
+ *
+ * @param id - keyed histogram id
+ * @param key - the string key
+ * @param sample - (optional) value to record, defaults to 1.
+ */
+void Accumulate(HistogramID id, const nsCString& key, uint32_t sample = 1);
+
+/**
+ * Adds an array of samples to a histogram defined in TelemetryHistograms.h
+ * @param id - histogram id
+ * @param samples - values to record.
+ * @param key - the string key
+ */
+void Accumulate(HistogramID id, const nsCString& key,
+ const nsTArray<uint32_t>& samples);
+
+/**
+ * Adds a sample to a histogram defined in TelemetryHistogramEnums.h.
+ * This function is here to support telemetry measurements from Java,
+ * where we have only names and not numeric IDs. You should almost
+ * certainly be using the by-enum-id version instead of this one.
+ *
+ * @param name - histogram name
+ * @param sample - value to record
+ */
+void Accumulate(const char* name, uint32_t sample);
+
+/**
+ * Adds a sample to a histogram defined in TelemetryHistogramEnums.h.
+ * This function is here to support telemetry measurements from Java,
+ * where we have only names and not numeric IDs. You should almost
+ * certainly be using the by-enum-id version instead of this one.
+ *
+ * @param name - histogram name
+ * @param key - the string key
+ * @param sample - sample - (optional) value to record, defaults to 1.
+ */
+void Accumulate(const char* name, const nsCString& key, uint32_t sample = 1);
+
+/**
+ * Adds sample to a categorical histogram defined in TelemetryHistogramEnums.h
+ * This is the typesafe - and preferred - way to use the categorical histograms
+ * by passing values from the corresponding Telemetry::LABELS_* enum.
+ *
+ * @param enumValue - Label value from one of the Telemetry::LABELS_* enums.
+ */
+template <class E>
+void AccumulateCategorical(E enumValue) {
+ static_assert(IsCategoricalLabelEnum<E>::value,
+ "Only categorical label enum types are supported.");
+ Accumulate(static_cast<HistogramID>(CategoricalLabelId<E>::value),
+ static_cast<uint32_t>(enumValue));
+};
+
+/**
+ * Adds an array of samples to categorical histograms defined in
+ * TelemetryHistogramEnums.h This is the typesafe - and preferred - way to use
+ * the categorical histograms by passing values from the corresponding
+ * Telemetry::LABELS_* enums.
+ *
+ * @param enumValues - Array of labels from Telemetry::LABELS_* enums.
+ */
+template <class E>
+void AccumulateCategorical(const nsTArray<E>& enumValues) {
+ static_assert(IsCategoricalLabelEnum<E>::value,
+ "Only categorical label enum types are supported.");
+ nsTArray<uint32_t> intSamples(enumValues.Length());
+
+ for (E aValue : enumValues) {
+ intSamples.AppendElement(static_cast<uint32_t>(aValue));
+ }
+
+ HistogramID categoricalId =
+ static_cast<HistogramID>(CategoricalLabelId<E>::value);
+
+ Accumulate(categoricalId, intSamples);
+}
+
+/**
+ * Adds sample to a keyed categorical histogram defined in
+ * TelemetryHistogramEnums.h This is the typesafe - and preferred - way to use
+ * the keyed categorical histograms by passing values from the corresponding
+ * Telemetry::LABELS_* enum.
+ *
+ * @param key - the string key
+ * @param enumValue - Label value from one of the Telemetry::LABELS_* enums.
+ */
+template <class E>
+void AccumulateCategoricalKeyed(const nsCString& key, E enumValue) {
+ static_assert(IsCategoricalLabelEnum<E>::value,
+ "Only categorical label enum types are supported.");
+ Accumulate(static_cast<HistogramID>(CategoricalLabelId<E>::value), key,
+ static_cast<uint32_t>(enumValue));
+};
+
+/**
+ * Adds an array of samples to a keyed categorical histogram defined in
+ * TelemetryHistogramEnums.h. This is the typesafe - and preferred - way to use
+ * the keyed categorical histograms by passing values from the corresponding
+ * Telemetry::LABELS_*enum.
+ *
+ * @param key - the string key
+ * @param enumValue - Label value from one of the Telemetry::LABELS_* enums.
+ */
+template <class E>
+void AccumulateCategoricalKeyed(const nsCString& key,
+ const nsTArray<E>& enumValues) {
+ static_assert(IsCategoricalLabelEnum<E>::value,
+ "Only categorical label enum types are supported.");
+ nsTArray<uint32_t> intSamples(enumValues.Length());
+
+ for (E aValue : enumValues) {
+ intSamples.AppendElement(static_cast<uint32_t>(aValue));
+ }
+
+ Accumulate(static_cast<HistogramID>(CategoricalLabelId<E>::value), key,
+ intSamples);
+};
+
+/**
+ * Adds sample to a categorical histogram defined in TelemetryHistogramEnums.h
+ * This string will be matched against the labels defined in Histograms.json.
+ * If the string does not match a label defined for the histogram, nothing will
+ * be recorded.
+ *
+ * @param id - The histogram id.
+ * @param label - A string label value that is defined in Histograms.json for
+ * this histogram.
+ */
+void AccumulateCategorical(HistogramID id, const nsCString& label);
+
+/**
+ * Adds an array of samples to a categorical histogram defined in
+ * Histograms.json
+ *
+ * @param id - The histogram id
+ * @param labels - The array of labels to accumulate
+ */
+void AccumulateCategorical(HistogramID id, const nsTArray<nsCString>& labels);
+
+/**
+ * Adds time delta in milliseconds to a histogram defined in
+ * TelemetryHistogramEnums.h
+ *
+ * @param id - histogram id
+ * @param start - start time
+ * @param end - end time
+ */
+void AccumulateTimeDelta(HistogramID id, TimeStamp start,
+ TimeStamp end = TimeStamp::Now());
+
+/**
+ * Adds time delta in milliseconds to a keyed histogram defined in
+ * TelemetryHistogramEnums.h
+ *
+ * @param id - histogram id
+ * @param key - the string key
+ * @param start - start time
+ * @param end - end time
+ */
+void AccumulateTimeDelta(HistogramID id, const nsCString& key, TimeStamp start,
+ TimeStamp end = TimeStamp::Now());
+
+/**
+ * Enable/disable recording for this histogram in this process at runtime.
+ * Recording is enabled by default, unless listed at
+ * kRecordingInitiallyDisabledIDs[]. id must be a valid telemetry enum,
+ *
+ * @param id - histogram id
+ * @param enabled - whether or not to enable recording from now on.
+ */
+void SetHistogramRecordingEnabled(HistogramID id, bool enabled);
+
+const char* GetHistogramName(HistogramID id);
+
+class MOZ_RAII RuntimeAutoTimer {
+ public:
+ explicit RuntimeAutoTimer(Telemetry::HistogramID aId,
+ TimeStamp aStart = TimeStamp::Now())
+ : id(aId), start(aStart) {}
+ explicit RuntimeAutoTimer(Telemetry::HistogramID aId, const nsCString& aKey,
+ TimeStamp aStart = TimeStamp::Now())
+ : id(aId), key(aKey), start(aStart) {
+ MOZ_ASSERT(!aKey.IsEmpty(), "The key must not be empty.");
+ }
+
+ ~RuntimeAutoTimer() {
+ if (key.IsEmpty()) {
+ AccumulateTimeDelta(id, start);
+ } else {
+ AccumulateTimeDelta(id, key, start);
+ }
+ }
+
+ private:
+ Telemetry::HistogramID id;
+ const nsCString key;
+ const TimeStamp start;
+};
+
+template <HistogramID id>
+class MOZ_RAII AutoTimer {
+ public:
+ explicit AutoTimer(TimeStamp aStart = TimeStamp::Now()) : start(aStart) {}
+
+ explicit AutoTimer(const nsCString& aKey, TimeStamp aStart = TimeStamp::Now())
+ : start(aStart), key(aKey) {
+ MOZ_ASSERT(!aKey.IsEmpty(), "The key must not be empty.");
+ }
+
+ ~AutoTimer() {
+ if (key.IsEmpty()) {
+ AccumulateTimeDelta(id, start);
+ } else {
+ AccumulateTimeDelta(id, key, start);
+ }
+ }
+
+ private:
+ const TimeStamp start;
+ const nsCString key;
+};
+
+class MOZ_RAII RuntimeAutoCounter {
+ public:
+ explicit RuntimeAutoCounter(HistogramID aId, uint32_t counterStart = 0)
+ : id(aId), counter(counterStart) {}
+
+ ~RuntimeAutoCounter() { Accumulate(id, counter); }
+
+ // Prefix increment only, to encourage good habits.
+ void operator++() {
+ if (NS_WARN_IF(counter == std::numeric_limits<uint32_t>::max())) {
+ return;
+ }
+ ++counter;
+ }
+
+ // Chaining doesn't make any sense, don't return anything.
+ void operator+=(int increment) {
+ if (NS_WARN_IF(increment > 0 &&
+ static_cast<uint32_t>(increment) >
+ (std::numeric_limits<uint32_t>::max() - counter))) {
+ counter = std::numeric_limits<uint32_t>::max();
+ return;
+ }
+ if (NS_WARN_IF(increment < 0 &&
+ static_cast<uint32_t>(-increment) > counter)) {
+ counter = std::numeric_limits<uint32_t>::min();
+ return;
+ }
+ counter += increment;
+ }
+
+ private:
+ HistogramID id;
+ uint32_t counter;
+};
+
+template <HistogramID id>
+class MOZ_RAII AutoCounter {
+ public:
+ explicit AutoCounter(uint32_t counterStart = 0) : counter(counterStart) {}
+
+ ~AutoCounter() { Accumulate(id, counter); }
+
+ // Prefix increment only, to encourage good habits.
+ void operator++() {
+ if (NS_WARN_IF(counter == std::numeric_limits<uint32_t>::max())) {
+ return;
+ }
+ ++counter;
+ }
+
+ // Chaining doesn't make any sense, don't return anything.
+ void operator+=(int increment) {
+ if (NS_WARN_IF(increment > 0 &&
+ static_cast<uint32_t>(increment) >
+ (std::numeric_limits<uint32_t>::max() - counter))) {
+ counter = std::numeric_limits<uint32_t>::max();
+ return;
+ }
+ if (NS_WARN_IF(increment < 0 &&
+ static_cast<uint32_t>(-increment) > counter)) {
+ counter = std::numeric_limits<uint32_t>::min();
+ return;
+ }
+ counter += increment;
+ }
+
+ private:
+ uint32_t counter;
+};
+
+/**
+ * Indicates whether Telemetry base data recording is turned on. Added for
+ * future uses.
+ */
+bool CanRecordBase();
+
+/**
+ * Indicates whether Telemetry extended data recording is turned on. This is
+ * intended to guard calls to Accumulate when the statistic being recorded is
+ * expensive to compute.
+ */
+bool CanRecordExtended();
+
+/**
+ * Indicates whether Telemetry release data recording is turned on. Usually
+ * true.
+ *
+ * @see nsITelemetry.canRecordReleaseData
+ */
+bool CanRecordReleaseData();
+
+/**
+ * Indicates whether Telemetry pre-release data recording is turned on. Tends
+ * to be true on pre-release channels.
+ *
+ * @see nsITelemetry.canRecordPrereleaseData
+ */
+bool CanRecordPrereleaseData();
+
+/**
+ * Records slow SQL statements for Telemetry reporting.
+ *
+ * @param statement - offending SQL statement to record
+ * @param dbName - DB filename
+ * @param delay - execution time in milliseconds
+ */
+void RecordSlowSQLStatement(const nsACString& statement,
+ const nsACString& dbName, uint32_t delay);
+
+/**
+ * Initialize I/O Reporting
+ * Initially this only records I/O for files in the binary directory.
+ *
+ * @param aXreDir - XRE directory
+ */
+void InitIOReporting(nsIFile* aXreDir);
+
+/**
+ * Set the profile directory. Once called, files in the profile directory will
+ * be included in I/O reporting. We can't use the directory
+ * service to obtain this information because it isn't running yet.
+ */
+void SetProfileDir(nsIFile* aProfD);
+
+/**
+ * Called to inform Telemetry that startup has completed.
+ */
+void LeavingStartupStage();
+
+/**
+ * Called to inform Telemetry that shutdown is commencing.
+ */
+void EnteringShutdownStage();
+
+/**
+ * Thresholds for a statement to be considered slow, in milliseconds
+ */
+const uint32_t kSlowSQLThresholdForMainThread = 50;
+const uint32_t kSlowSQLThresholdForHelperThreads = 100;
+
+/**
+ * Record a failed attempt at locking the user's profile.
+ *
+ * @param aProfileDir The profile directory whose lock attempt failed
+ */
+void WriteFailedProfileLock(nsIFile* aProfileDir);
+
+/**
+ * Adds the value to the given scalar.
+ *
+ * @param aId The scalar enum id.
+ * @param aValue The value to add to the scalar.
+ */
+void ScalarAdd(mozilla::Telemetry::ScalarID aId, uint32_t aValue);
+
+/**
+ * Sets the scalar to the given value.
+ *
+ * @param aId The scalar enum id.
+ * @param aValue The value to set the scalar to.
+ */
+void ScalarSet(mozilla::Telemetry::ScalarID aId, uint32_t aValue);
+
+/**
+ * Sets the scalar to the given value.
+ *
+ * @param aId The scalar enum id.
+ * @param aValue The value to set the scalar to.
+ */
+void ScalarSet(mozilla::Telemetry::ScalarID aId, bool aValue);
+
+/**
+ * Sets the scalar to the given value.
+ *
+ * @param aId The scalar enum id.
+ * @param aValue The value to set the scalar to, truncated to
+ * 50 characters if exceeding that length.
+ */
+void ScalarSet(mozilla::Telemetry::ScalarID aId, const nsAString& aValue);
+
+/**
+ * Sets the scalar to the maximum of the current and the passed value.
+ *
+ * @param aId The scalar enum id.
+ * @param aValue The value the scalar is set to if its greater
+ * than the current value.
+ */
+void ScalarSetMaximum(mozilla::Telemetry::ScalarID aId, uint32_t aValue);
+
+/**
+ * Adds the value to the given scalar.
+ *
+ * @param aId The scalar enum id.
+ * @param aKey The scalar key.
+ * @param aValue The value to add to the scalar.
+ */
+void ScalarAdd(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ uint32_t aValue);
+
+/**
+ * Sets the scalar to the given value.
+ *
+ * @param aId The scalar enum id.
+ * @param aKey The scalar key.
+ * @param aValue The value to set the scalar to.
+ */
+void ScalarSet(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ uint32_t aValue);
+
+/**
+ * Sets the scalar to the given value.
+ *
+ * @param aId The scalar enum id.
+ * @param aKey The scalar key.
+ * @param aValue The value to set the scalar to.
+ */
+void ScalarSet(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ bool aValue);
+
+/**
+ * Sets the scalar to the maximum of the current and the passed value.
+ *
+ * @param aId The scalar enum id.
+ * @param aKey The scalar key.
+ * @param aValue The value the scalar is set to if its greater
+ * than the current value.
+ */
+void ScalarSetMaximum(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ uint32_t aValue);
+
+template <ScalarID id>
+class MOZ_RAII AutoScalarTimer {
+ public:
+ explicit AutoScalarTimer(TimeStamp aStart = TimeStamp::Now())
+ : start(aStart) {}
+
+ explicit AutoScalarTimer(const nsAString& aKey,
+ TimeStamp aStart = TimeStamp::Now())
+ : start(aStart), key(aKey) {
+ MOZ_ASSERT(!aKey.IsEmpty(), "The key must not be empty.");
+ }
+
+ ~AutoScalarTimer() {
+ TimeStamp end = TimeStamp::Now();
+ uint32_t delta = static_cast<uint32_t>((end - start).ToMilliseconds());
+ if (key.IsEmpty()) {
+ mozilla::Telemetry::ScalarSet(id, delta);
+ } else {
+ mozilla::Telemetry::ScalarSet(id, key, delta);
+ }
+ }
+
+ private:
+ const TimeStamp start;
+ const nsString key;
+};
+
+/**
+ * Records an event. See the Event documentation for more information:
+ * https://firefox-source-docs.mozilla.org/toolkit/components/telemetry/telemetry/collection/events.html
+ *
+ * @param aId The event enum id.
+ * @param aValue Optional. The event value.
+ * @param aExtra Optional. The event's extra key/value pairs.
+ */
+void RecordEvent(mozilla::Telemetry::EventID aId,
+ const mozilla::Maybe<nsCString>& aValue,
+ const mozilla::Maybe<CopyableTArray<EventExtraEntry>>& aExtra);
+
+/**
+ * Enables recording of events in a category.
+ * Events default to recording disabled.
+ * This toggles recording for all events in the specified category.
+ *
+ * @param aCategory The category name.
+ * @param aEnabled Whether recording should be enabled or disabled.
+ */
+void SetEventRecordingEnabled(const nsACString& aCategory, bool aEnabled);
+
+/**
+ * YOU PROBABLY SHOULDN'T USE THIS.
+ * THIS IS AN EXPERIMENTAL API NOT YET READY FOR GENERAL USE.
+ *
+ * Records that the metric is true for the stated origin.
+ *
+ * @param aId the metric.
+ * @param aOrigin the origin on which to record the metric as true.
+ */
+void RecordOrigin(mozilla::Telemetry::OriginMetricID aId,
+ const nsACString& aOrigin);
+
+} // namespace Telemetry
+} // namespace mozilla
+
+#endif // Telemetry_h__
diff --git a/toolkit/components/telemetry/core/TelemetryCommon.cpp b/toolkit/components/telemetry/core/TelemetryCommon.cpp
new file mode 100644
index 0000000000..833f6b843b
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryCommon.cpp
@@ -0,0 +1,208 @@
+/* -*- 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 "TelemetryCommon.h"
+
+#include <cstring>
+#include "mozilla/TimeStamp.h"
+#include "mozilla/StaticPrefs_toolkit.h"
+#include "nsComponentManagerUtils.h"
+#include "nsIConsoleService.h"
+#include "nsITelemetry.h"
+#include "nsServiceManagerUtils.h"
+#include "nsThreadUtils.h"
+#include "nsVersionComparator.h"
+#include "TelemetryProcessData.h"
+#include "Telemetry.h"
+#include "mozilla/Uptime.h"
+
+namespace mozilla::Telemetry::Common {
+
+bool IsExpiredVersion(const char* aExpiration) {
+ MOZ_ASSERT(aExpiration);
+ // Note: We intentionally don't construct a static Version object here as we
+ // saw odd crashes around this (see bug 1334105).
+ return strcmp(aExpiration, "never") && strcmp(aExpiration, "default") &&
+ (mozilla::Version(aExpiration) <= MOZ_APP_VERSION);
+}
+
+bool IsInDataset(uint32_t aDataset, uint32_t aContainingDataset) {
+ if (aDataset == aContainingDataset) {
+ return true;
+ }
+
+ // The "optin on release channel" dataset is a superset of the
+ // "optout on release channel one".
+ if (aContainingDataset == nsITelemetry::DATASET_PRERELEASE_CHANNELS &&
+ aDataset == nsITelemetry::DATASET_ALL_CHANNELS) {
+ return true;
+ }
+
+ return false;
+}
+
+bool CanRecordDataset(uint32_t aDataset, bool aCanRecordBase,
+ bool aCanRecordExtended) {
+ // If we are extended telemetry is enabled, we are allowed to record
+ // regardless of the dataset.
+ if (aCanRecordExtended) {
+ return true;
+ }
+
+ // If base telemetry data is enabled and we're trying to record base
+ // telemetry, allow it.
+ if (aCanRecordBase &&
+ IsInDataset(aDataset, nsITelemetry::DATASET_ALL_CHANNELS)) {
+ return true;
+ }
+
+ // We're not recording extended telemetry or this is not the base
+ // dataset. Bail out.
+ return false;
+}
+
+bool CanRecordInProcess(RecordedProcessType processes,
+ GeckoProcessType processType) {
+ // We can use (1 << ProcessType) due to the way RecordedProcessType is
+ // defined.
+ bool canRecordProcess =
+ !!(processes & static_cast<RecordedProcessType>(1 << processType));
+
+ return canRecordProcess;
+}
+
+bool CanRecordInProcess(RecordedProcessType processes, ProcessID processId) {
+ return CanRecordInProcess(processes, GetGeckoProcessType(processId));
+}
+
+bool CanRecordProduct(SupportedProduct aProducts) {
+ return mozilla::StaticPrefs::
+ toolkit_telemetry_testing_overrideProductsCheck() ||
+ !!(aProducts & GetCurrentProduct());
+}
+
+nsresult MsSinceProcessStart(double* aResult) {
+ *aResult =
+ (TimeStamp::NowLoRes() - TimeStamp::ProcessCreation()).ToMilliseconds();
+ return NS_OK;
+}
+
+nsresult MsSinceProcessStartIncludingSuspend(double* aResult) {
+ auto rv = mozilla::ProcessUptimeMs();
+ if (rv) {
+ *aResult = rv.value();
+ return NS_OK;
+ }
+ return NS_ERROR_NOT_AVAILABLE;
+}
+
+nsresult MsSinceProcessStartExcludingSuspend(double* aResult) {
+ auto rv = mozilla::ProcessUptimeExcludingSuspendMs();
+ if (rv) {
+ *aResult = rv.value();
+ return NS_OK;
+ }
+ return NS_ERROR_NOT_AVAILABLE;
+}
+
+void LogToBrowserConsole(uint32_t aLogLevel, const nsAString& aMsg) {
+ if (!NS_IsMainThread()) {
+ nsString msg(aMsg);
+ nsCOMPtr<nsIRunnable> task = NS_NewRunnableFunction(
+ "Telemetry::Common::LogToBrowserConsole",
+ [aLogLevel, msg]() { LogToBrowserConsole(aLogLevel, msg); });
+ NS_DispatchToMainThread(task.forget(), NS_DISPATCH_NORMAL);
+ return;
+ }
+
+ nsCOMPtr<nsIConsoleService> console(
+ do_GetService("@mozilla.org/consoleservice;1"));
+ if (!console) {
+ NS_WARNING("Failed to log message to console.");
+ return;
+ }
+
+ nsCOMPtr<nsIScriptError> error(do_CreateInstance(NS_SCRIPTERROR_CONTRACTID));
+ error->Init(aMsg, u""_ns, u""_ns, 0, 0, aLogLevel, "chrome javascript",
+ false /* from private window */, true /* from chrome context */);
+ console->LogMessage(error);
+}
+
+const char* GetNameForProcessID(ProcessID process) {
+ MOZ_ASSERT(process < ProcessID::Count);
+ return ProcessIDToString[static_cast<uint32_t>(process)];
+}
+
+ProcessID GetIDForProcessName(const char* aProcessName) {
+ for (uint32_t id = 0; id < static_cast<uint32_t>(ProcessID::Count); id++) {
+ if (!strcmp(GetNameForProcessID(ProcessID(id)), aProcessName)) {
+ return ProcessID(id);
+ }
+ }
+
+ return ProcessID::Count;
+}
+
+GeckoProcessType GetGeckoProcessType(ProcessID process) {
+ MOZ_ASSERT(process < ProcessID::Count);
+ return ProcessIDToGeckoProcessType[static_cast<uint32_t>(process)];
+}
+
+bool IsStringCharValid(const char aChar, const bool aAllowInfixPeriod,
+ const bool aAllowInfixUnderscore) {
+ return (aChar >= 'A' && aChar <= 'Z') || (aChar >= 'a' && aChar <= 'z') ||
+ (aChar >= '0' && aChar <= '9') ||
+ (aAllowInfixPeriod && (aChar == '.')) ||
+ (aAllowInfixUnderscore && (aChar == '_'));
+}
+
+bool IsValidIdentifierString(const nsACString& aStr, const size_t aMaxLength,
+ const bool aAllowInfixPeriod,
+ const bool aAllowInfixUnderscore) {
+ // Check string length.
+ if (aStr.Length() > aMaxLength) {
+ return false;
+ }
+
+ // Check string characters.
+ const char* first = aStr.BeginReading();
+ const char* end = aStr.EndReading();
+
+ for (const char* cur = first; cur < end; ++cur) {
+ const bool infix = (cur != first) && (cur != (end - 1));
+ if (!IsStringCharValid(*cur, aAllowInfixPeriod && infix,
+ aAllowInfixUnderscore && infix)) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+JSString* ToJSString(JSContext* cx, const nsACString& aStr) {
+ const NS_ConvertUTF8toUTF16 wide(aStr);
+ return JS_NewUCStringCopyN(cx, wide.Data(), wide.Length());
+}
+
+JSString* ToJSString(JSContext* cx, const nsAString& aStr) {
+ return JS_NewUCStringCopyN(cx, aStr.Data(), aStr.Length());
+}
+
+SupportedProduct GetCurrentProduct() {
+#if defined(MOZ_WIDGET_ANDROID)
+ if (mozilla::StaticPrefs::toolkit_telemetry_geckoview_streaming()) {
+ return SupportedProduct::GeckoviewStreaming;
+ } else {
+ return SupportedProduct::Fennec;
+ }
+#elif defined(MOZ_THUNDERBIRD)
+ return SupportedProduct::Thunderbird;
+#else
+ return SupportedProduct::Firefox;
+#endif
+}
+
+} // namespace mozilla::Telemetry::Common
diff --git a/toolkit/components/telemetry/core/TelemetryCommon.h b/toolkit/components/telemetry/core/TelemetryCommon.h
new file mode 100644
index 0000000000..5af8f7f0cd
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryCommon.h
@@ -0,0 +1,193 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryCommon_h__
+#define TelemetryCommon_h__
+
+#include "jsapi.h"
+#include "mozilla/TypedEnumBits.h"
+#include "mozilla/TelemetryProcessEnums.h"
+#include "nsIScriptError.h"
+#include "nsTHashtable.h"
+#include "nsHashKeys.h"
+#include "nsXULAppAPI.h"
+
+namespace mozilla {
+namespace Telemetry {
+namespace Common {
+
+typedef nsTHashtable<nsCStringHashKey> StringHashSet;
+
+enum class RecordedProcessType : uint16_t {
+ Main = (1 << GeckoProcessType_Default), // Also known as "parent process"
+ Content = (1 << GeckoProcessType_Content),
+ Gpu = (1 << GeckoProcessType_GPU),
+ Socket = (1 << GeckoProcessType_Socket),
+ AllChildren = 0xFFFF - 1, // All the child processes (i.e. content, gpu, ...)
+ // Always `All-Main` to allow easy matching.
+ All = 0xFFFF // All the processes
+};
+MOZ_MAKE_ENUM_CLASS_BITWISE_OPERATORS(RecordedProcessType);
+static_assert(static_cast<uint16_t>(RecordedProcessType::Main) == 1,
+ "Main process type must be equal to 1 to allow easy matching in "
+ "CanRecordInProcess");
+
+enum class SupportedProduct : uint8_t {
+ Firefox = (1 << 0),
+ Fennec = (1 << 1),
+ // Note that `1 << 2` (former GeckoView) is missing in the representation
+ // but isn't necessary to be maintained, but we see no point in filling it
+ // at this time.
+ GeckoviewStreaming = (1 << 3),
+ Thunderbird = (1 << 4),
+};
+MOZ_MAKE_ENUM_CLASS_BITWISE_OPERATORS(SupportedProduct);
+
+template <class EntryType>
+class AutoHashtable : public nsTHashtable<EntryType> {
+ public:
+ explicit AutoHashtable(
+ uint32_t initLength = PLDHashTable::kDefaultInitialLength);
+ typedef bool (*ReflectEntryFunc)(EntryType* entry, JSContext* cx,
+ JS::Handle<JSObject*> obj);
+ bool ReflectIntoJS(ReflectEntryFunc entryFunc, JSContext* cx,
+ JS::Handle<JSObject*> obj);
+};
+
+template <class EntryType>
+AutoHashtable<EntryType>::AutoHashtable(uint32_t initLength)
+ : nsTHashtable<EntryType>(initLength) {}
+
+/**
+ * Reflect the individual entries of table into JS, usually by defining
+ * some property and value of obj. entryFunc is called for each entry.
+ */
+template <typename EntryType>
+bool AutoHashtable<EntryType>::ReflectIntoJS(ReflectEntryFunc entryFunc,
+ JSContext* cx,
+ JS::Handle<JSObject*> obj) {
+ for (auto iter = this->Iter(); !iter.Done(); iter.Next()) {
+ if (!entryFunc(iter.Get(), cx, obj)) {
+ return false;
+ }
+ }
+ return true;
+}
+
+bool IsExpiredVersion(const char* aExpiration);
+bool IsInDataset(uint32_t aDataset, uint32_t aContainingDataset);
+bool CanRecordDataset(uint32_t aDataset, bool aCanRecordBase,
+ bool aCanRecordExtended);
+bool CanRecordInProcess(RecordedProcessType aProcesses,
+ GeckoProcessType aProcess);
+bool CanRecordInProcess(RecordedProcessType aProcesses, ProcessID aProcess);
+bool CanRecordProduct(SupportedProduct aProducts);
+
+/**
+ * Return the number of milliseconds since process start using monotonic
+ * timestamps (unaffected by system clock changes). Depending on the platform,
+ * this can include the time the device was suspended (Windows) or not (Linux,
+ * macOS).
+ *
+ * @return NS_OK on success.
+ */
+nsresult MsSinceProcessStart(double* aResult);
+
+/**
+ * Return the number of milliseconds since process start using monotonic
+ * timestamps (unaffected by system clock changes), including the time the
+ * system was suspended.
+ *
+ * @return NS_OK on success, NS_ERROR_NOT_AVAILABLE if the data is unavailable
+ * (this can happen on old operating systems).
+ */
+nsresult MsSinceProcessStartIncludingSuspend(double* aResult);
+
+/**
+ * Return the number of milliseconds since process start using monotonic
+ * timestamps (unaffected by system clock changes), excluding the time the
+ * system was suspended.
+ *
+ * @return NS_OK on success, NS_ERROR_NOT_AVAILABLE if the data is unavailable
+ * (this can happen on old operating systems).
+ */
+nsresult MsSinceProcessStartExcludingSuspend(double* aResult);
+
+/**
+ * Dumps a log message to the Browser Console using the provided level.
+ *
+ * @param aLogLevel The level to use when displaying the message in the browser
+ * console (e.g. nsIScriptError::warningFlag, ...).
+ * @param aMsg The text message to print to the console.
+ */
+void LogToBrowserConsole(uint32_t aLogLevel, const nsAString& aMsg);
+
+/**
+ * Get the name string for a ProcessID.
+ * This is the name we use for the Telemetry payloads.
+ */
+const char* GetNameForProcessID(ProcessID process);
+
+/**
+ * Get the process id give a process name.
+ *
+ * @param aProcessName - the name of the process.
+ * @returns {ProcessID} one value from ProcessID::* or ProcessID::Count if the
+ * name of the process was not found.
+ */
+ProcessID GetIDForProcessName(const char* aProcessName);
+
+/**
+ * Get the GeckoProcessType for a ProcessID.
+ * Telemetry distinguishes between more process types than the GeckoProcessType,
+ * so the mapping is not direct.
+ */
+GeckoProcessType GetGeckoProcessType(ProcessID process);
+
+/**
+ * Check if the passed telemetry identifier is valid.
+ *
+ * @param aStr The string identifier.
+ * @param aMaxLength The maximum length of the identifier.
+ * @param aAllowInfixPeriod Whether or not to allow infix dots.
+ * @param aAllowInfixUnderscore Whether or not to allow infix underscores.
+ * @returns true if the string validates correctly, false otherwise.
+ */
+bool IsValidIdentifierString(const nsACString& aStr, const size_t aMaxLength,
+ const bool aAllowInfixPeriod,
+ const bool aAllowInfixUnderscore);
+
+/**
+ * Convert the given UTF8 string to a JavaScript string. The returned
+ * string's contents will be the UTF16 conversion of the given string.
+ *
+ * @param cx The JS context.
+ * @param aStr The UTF8 string.
+ * @returns a JavaScript string.
+ */
+JSString* ToJSString(JSContext* cx, const nsACString& aStr);
+
+/**
+ * Convert the given UTF16 string to a JavaScript string.
+ *
+ * @param cx The JS context.
+ * @param aStr The UTF16 string.
+ * @returns a JavaScript string.
+ */
+JSString* ToJSString(JSContext* cx, const nsAString& aStr);
+
+/**
+ * Get an identifier for the currently-running product.
+ * This is not stable over time and may change.
+ *
+ * @returns the product identifier
+ */
+SupportedProduct GetCurrentProduct();
+
+} // namespace Common
+} // namespace Telemetry
+} // namespace mozilla
+
+#endif // TelemetryCommon_h__
diff --git a/toolkit/components/telemetry/core/TelemetryEvent.cpp b/toolkit/components/telemetry/core/TelemetryEvent.cpp
new file mode 100644
index 0000000000..2a8ece107e
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryEvent.cpp
@@ -0,0 +1,1390 @@
+/* -*- 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 "Telemetry.h"
+#include "TelemetryEvent.h"
+#include <prtime.h>
+#include <limits>
+#include "ipc/TelemetryIPCAccumulator.h"
+#include "jsapi.h"
+#include "js/Array.h" // JS::GetArrayLength, JS::IsArrayObject, JS::NewArrayObject
+#include "mozilla/Maybe.h"
+#include "mozilla/Services.h"
+#include "mozilla/StaticMutex.h"
+#include "mozilla/StaticPtr.h"
+#include "mozilla/Unused.h"
+#include "nsClassHashtable.h"
+#include "nsDataHashtable.h"
+#include "nsHashKeys.h"
+#include "nsIObserverService.h"
+#include "nsITelemetry.h"
+#include "nsJSUtils.h"
+#include "nsPrintfCString.h"
+#include "nsTArray.h"
+#include "nsUTF8Utils.h"
+#include "nsXULAppAPI.h"
+#include "TelemetryCommon.h"
+#include "TelemetryEventData.h"
+#include "TelemetryScalar.h"
+
+using mozilla::Maybe;
+using mozilla::StaticAutoPtr;
+using mozilla::StaticMutex;
+using mozilla::StaticMutexAutoLock;
+using mozilla::TimeStamp;
+using mozilla::Telemetry::ChildEventData;
+using mozilla::Telemetry::EventExtraEntry;
+using mozilla::Telemetry::LABELS_TELEMETRY_EVENT_RECORDING_ERROR;
+using mozilla::Telemetry::LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR;
+using mozilla::Telemetry::ProcessID;
+using mozilla::Telemetry::Common::CanRecordDataset;
+using mozilla::Telemetry::Common::CanRecordInProcess;
+using mozilla::Telemetry::Common::CanRecordProduct;
+using mozilla::Telemetry::Common::GetNameForProcessID;
+using mozilla::Telemetry::Common::IsExpiredVersion;
+using mozilla::Telemetry::Common::IsInDataset;
+using mozilla::Telemetry::Common::IsValidIdentifierString;
+using mozilla::Telemetry::Common::LogToBrowserConsole;
+using mozilla::Telemetry::Common::MsSinceProcessStart;
+using mozilla::Telemetry::Common::ToJSString;
+
+namespace TelemetryIPCAccumulator = mozilla::TelemetryIPCAccumulator;
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// Naming: there are two kinds of functions in this file:
+//
+// * Functions taking a StaticMutexAutoLock: these can only be reached via
+// an interface function (TelemetryEvent::*). They expect the interface
+// function to have acquired |gTelemetryEventsMutex|, so they do not
+// have to be thread-safe.
+//
+// * Functions named TelemetryEvent::*. This is the external interface.
+// Entries and exits to these functions are serialised using
+// |gTelemetryEventsMutex|.
+//
+// Avoiding races and deadlocks:
+//
+// All functions in the external interface (TelemetryEvent::*) are
+// serialised using the mutex |gTelemetryEventsMutex|. This means
+// that the external interface is thread-safe, and the internal
+// functions can ignore thread safety. But it also brings a danger
+// of deadlock if any function in the external interface can get back
+// to that interface. That is, we will deadlock on any call chain like
+// this:
+//
+// TelemetryEvent::* -> .. any functions .. -> TelemetryEvent::*
+//
+// To reduce the danger of that happening, observe the following rules:
+//
+// * No function in TelemetryEvent::* may directly call, nor take the
+// address of, any other function in TelemetryEvent::*.
+//
+// * No internal function may call, nor take the address
+// of, any function in TelemetryEvent::*.
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE TYPES
+
+namespace {
+
+const uint32_t kEventCount =
+ static_cast<uint32_t>(mozilla::Telemetry::EventID::EventCount);
+// This is a special event id used to mark expired events, to make expiry checks
+// cheap at runtime.
+const uint32_t kExpiredEventId = std::numeric_limits<uint32_t>::max();
+static_assert(kExpiredEventId > kEventCount,
+ "Built-in event count should be less than the expired event id.");
+
+// Maximum length of any passed value string, in UTF8 byte sequence length.
+const uint32_t kMaxValueByteLength = 80;
+// Maximum length of any string value in the extra dictionary, in UTF8 byte
+// sequence length.
+const uint32_t kMaxExtraValueByteLength = 80;
+// Maximum length of dynamic method names, in UTF8 byte sequence length.
+const uint32_t kMaxMethodNameByteLength = 20;
+// Maximum length of dynamic object names, in UTF8 byte sequence length.
+const uint32_t kMaxObjectNameByteLength = 20;
+// Maximum length of extra key names, in UTF8 byte sequence length.
+const uint32_t kMaxExtraKeyNameByteLength = 15;
+// The maximum number of valid extra keys for an event.
+const uint32_t kMaxExtraKeyCount = 10;
+// The number of event records allowed in an event ping.
+const uint32_t kEventPingLimit = 1000;
+
+struct EventKey {
+ uint32_t id;
+ bool dynamic;
+};
+
+struct DynamicEventInfo {
+ DynamicEventInfo(const nsACString& category, const nsACString& method,
+ const nsACString& object, nsTArray<nsCString>& extra_keys,
+ bool recordOnRelease, bool builtin)
+ : category(category),
+ method(method),
+ object(object),
+ extra_keys(extra_keys.Clone()),
+ recordOnRelease(recordOnRelease),
+ builtin(builtin) {}
+
+ DynamicEventInfo(const DynamicEventInfo&) = default;
+ DynamicEventInfo& operator=(const DynamicEventInfo&) = delete;
+
+ const nsCString category;
+ const nsCString method;
+ const nsCString object;
+ const CopyableTArray<nsCString> extra_keys;
+ const bool recordOnRelease;
+ const bool builtin;
+
+ size_t SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const {
+ size_t n = 0;
+
+ n += category.SizeOfExcludingThisIfUnshared(aMallocSizeOf);
+ n += method.SizeOfExcludingThisIfUnshared(aMallocSizeOf);
+ n += object.SizeOfExcludingThisIfUnshared(aMallocSizeOf);
+ n += extra_keys.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ for (auto& key : extra_keys) {
+ n += key.SizeOfExcludingThisIfUnshared(aMallocSizeOf);
+ }
+
+ return n;
+ }
+};
+
+enum class RecordEventResult {
+ Ok,
+ UnknownEvent,
+ InvalidExtraKey,
+ StorageLimitReached,
+ ExpiredEvent,
+ WrongProcess,
+ CannotRecord,
+};
+
+typedef CopyableTArray<EventExtraEntry> ExtraArray;
+
+class EventRecord {
+ public:
+ EventRecord(double timestamp, const EventKey& key,
+ const Maybe<nsCString>& value, const ExtraArray& extra)
+ : mTimestamp(timestamp),
+ mEventKey(key),
+ mValue(value),
+ mExtra(extra.Clone()) {}
+
+ EventRecord(const EventRecord& other) = default;
+
+ EventRecord& operator=(const EventRecord& other) = delete;
+
+ double Timestamp() const { return mTimestamp; }
+ const EventKey& GetEventKey() const { return mEventKey; }
+ const Maybe<nsCString>& Value() const { return mValue; }
+ const ExtraArray& Extra() const { return mExtra; }
+
+ size_t SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const;
+
+ private:
+ const double mTimestamp;
+ const EventKey mEventKey;
+ const Maybe<nsCString> mValue;
+ const ExtraArray mExtra;
+};
+
+// Implements the methods for EventInfo.
+const nsDependentCString EventInfo::method() const {
+ return nsDependentCString(&gEventsStringTable[this->method_offset]);
+}
+
+const nsDependentCString EventInfo::object() const {
+ return nsDependentCString(&gEventsStringTable[this->object_offset]);
+}
+
+// Implements the methods for CommonEventInfo.
+const nsDependentCString CommonEventInfo::category() const {
+ return nsDependentCString(&gEventsStringTable[this->category_offset]);
+}
+
+const nsDependentCString CommonEventInfo::expiration_version() const {
+ return nsDependentCString(
+ &gEventsStringTable[this->expiration_version_offset]);
+}
+
+const nsDependentCString CommonEventInfo::extra_key(uint32_t index) const {
+ MOZ_ASSERT(index < this->extra_count);
+ uint32_t key_index = gExtraKeysTable[this->extra_index + index];
+ return nsDependentCString(&gEventsStringTable[key_index]);
+}
+
+// Implementation for the EventRecord class.
+size_t EventRecord::SizeOfExcludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) const {
+ size_t n = 0;
+
+ if (mValue) {
+ n += mValue.value().SizeOfExcludingThisIfUnshared(aMallocSizeOf);
+ }
+
+ n += mExtra.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ for (uint32_t i = 0; i < mExtra.Length(); ++i) {
+ n += mExtra[i].key.SizeOfExcludingThisIfUnshared(aMallocSizeOf);
+ n += mExtra[i].value.SizeOfExcludingThisIfUnshared(aMallocSizeOf);
+ }
+
+ return n;
+}
+
+nsCString UniqueEventName(const nsACString& category, const nsACString& method,
+ const nsACString& object) {
+ nsCString name;
+ name.Append(category);
+ name.AppendLiteral("#");
+ name.Append(method);
+ name.AppendLiteral("#");
+ name.Append(object);
+ return name;
+}
+
+nsCString UniqueEventName(const EventInfo& info) {
+ return UniqueEventName(info.common_info.category(), info.method(),
+ info.object());
+}
+
+nsCString UniqueEventName(const DynamicEventInfo& info) {
+ return UniqueEventName(info.category, info.method, info.object);
+}
+
+void TruncateToByteLength(nsCString& str, uint32_t length) {
+ // last will be the index of the first byte of the current multi-byte
+ // sequence.
+ uint32_t last = RewindToPriorUTF8Codepoint(str.get(), length);
+ str.Truncate(last);
+}
+
+} // anonymous namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE STATE, SHARED BY ALL THREADS
+
+namespace {
+
+// Set to true once this global state has been initialized.
+bool gInitDone = false;
+
+bool gCanRecordBase;
+bool gCanRecordExtended;
+
+// The EventName -> EventKey cache map.
+nsClassHashtable<nsCStringHashKey, EventKey> gEventNameIDMap(kEventCount);
+
+// The CategoryName set.
+nsTHashtable<nsCStringHashKey> gCategoryNames;
+
+// This tracks the IDs of the categories for which recording is enabled.
+nsTHashtable<nsCStringHashKey> gEnabledCategories;
+
+// The main event storage. Events are inserted here, keyed by process id and
+// in recording order.
+typedef nsUint32HashKey ProcessIDHashKey;
+typedef nsTArray<EventRecord> EventRecordArray;
+typedef nsClassHashtable<ProcessIDHashKey, EventRecordArray>
+ EventRecordsMapType;
+
+EventRecordsMapType gEventRecords;
+
+// The details on dynamic events that are recorded from addons are registered
+// here.
+StaticAutoPtr<nsTArray<DynamicEventInfo>> gDynamicEventInfo;
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: thread-safe helpers for event recording.
+
+namespace {
+
+unsigned int GetDataset(const StaticMutexAutoLock& lock,
+ const EventKey& eventKey) {
+ if (!eventKey.dynamic) {
+ return gEventInfo[eventKey.id].common_info.dataset;
+ }
+
+ if (!gDynamicEventInfo) {
+ return nsITelemetry::DATASET_PRERELEASE_CHANNELS;
+ }
+
+ return (*gDynamicEventInfo)[eventKey.id].recordOnRelease
+ ? nsITelemetry::DATASET_ALL_CHANNELS
+ : nsITelemetry::DATASET_PRERELEASE_CHANNELS;
+}
+
+nsCString GetCategory(const StaticMutexAutoLock& lock,
+ const EventKey& eventKey) {
+ if (!eventKey.dynamic) {
+ return gEventInfo[eventKey.id].common_info.category();
+ }
+
+ if (!gDynamicEventInfo) {
+ return ""_ns;
+ }
+
+ return (*gDynamicEventInfo)[eventKey.id].category;
+}
+
+bool CanRecordEvent(const StaticMutexAutoLock& lock, const EventKey& eventKey,
+ ProcessID process) {
+ if (!gCanRecordBase) {
+ return false;
+ }
+
+ if (!CanRecordDataset(GetDataset(lock, eventKey), gCanRecordBase,
+ gCanRecordExtended)) {
+ return false;
+ }
+
+ // We don't allow specifying a process to record in for dynamic events.
+ if (!eventKey.dynamic) {
+ const CommonEventInfo& info = gEventInfo[eventKey.id].common_info;
+
+ if (!CanRecordProduct(info.products)) {
+ return false;
+ }
+
+ if (!CanRecordInProcess(info.record_in_processes, process)) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+bool IsExpired(const EventKey& key) { return key.id == kExpiredEventId; }
+
+EventRecordArray* GetEventRecordsForProcess(const StaticMutexAutoLock& lock,
+ ProcessID processType) {
+ EventRecordArray* eventRecords = nullptr;
+ if (!gEventRecords.Get(uint32_t(processType), &eventRecords)) {
+ eventRecords = new EventRecordArray();
+ gEventRecords.Put(uint32_t(processType), eventRecords);
+ }
+ return eventRecords;
+}
+
+EventKey* GetEventKey(const StaticMutexAutoLock& lock,
+ const nsACString& category, const nsACString& method,
+ const nsACString& object) {
+ EventKey* event;
+ const nsCString& name = UniqueEventName(category, method, object);
+ if (!gEventNameIDMap.Get(name, &event)) {
+ return nullptr;
+ }
+ return event;
+}
+
+static bool CheckExtraKeysValid(const EventKey& eventKey,
+ const ExtraArray& extra) {
+ nsTHashtable<nsCStringHashKey> validExtraKeys;
+ if (!eventKey.dynamic) {
+ const CommonEventInfo& common = gEventInfo[eventKey.id].common_info;
+ for (uint32_t i = 0; i < common.extra_count; ++i) {
+ validExtraKeys.PutEntry(common.extra_key(i));
+ }
+ } else if (gDynamicEventInfo) {
+ const DynamicEventInfo& info = (*gDynamicEventInfo)[eventKey.id];
+ for (uint32_t i = 0, len = info.extra_keys.Length(); i < len; ++i) {
+ validExtraKeys.PutEntry(info.extra_keys[i]);
+ }
+ }
+
+ for (uint32_t i = 0; i < extra.Length(); ++i) {
+ if (!validExtraKeys.GetEntry(extra[i].key)) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+RecordEventResult RecordEvent(const StaticMutexAutoLock& lock,
+ ProcessID processType, double timestamp,
+ const nsACString& category,
+ const nsACString& method,
+ const nsACString& object,
+ const Maybe<nsCString>& value,
+ const ExtraArray& extra) {
+ // Look up the event id.
+ EventKey* eventKey = GetEventKey(lock, category, method, object);
+ if (!eventKey) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_RECORDING_ERROR::UnknownEvent);
+ return RecordEventResult::UnknownEvent;
+ }
+
+ // If the event is expired or not enabled for this process, we silently drop
+ // this call. We don't want recording for expired probes to be an error so
+ // code doesn't have to be removed at a specific time or version. Even logging
+ // warnings would become very noisy.
+ if (IsExpired(*eventKey)) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_RECORDING_ERROR::Expired);
+ return RecordEventResult::ExpiredEvent;
+ }
+
+ // Fixup the process id only for non-builtin (e.g. supporting build faster)
+ // dynamic events.
+ auto dynamicNonBuiltin =
+ eventKey->dynamic && !(*gDynamicEventInfo)[eventKey->id].builtin;
+ if (dynamicNonBuiltin) {
+ processType = ProcessID::Dynamic;
+ }
+
+ // Check whether the extra keys passed are valid.
+ if (!CheckExtraKeysValid(*eventKey, extra)) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_RECORDING_ERROR::ExtraKey);
+ return RecordEventResult::InvalidExtraKey;
+ }
+
+ // Check whether we can record this event.
+ if (!CanRecordEvent(lock, *eventKey, processType)) {
+ return RecordEventResult::CannotRecord;
+ }
+
+ // Count the number of times this event has been recorded, even if its
+ // category does not have recording enabled.
+ TelemetryScalar::SummarizeEvent(UniqueEventName(category, method, object),
+ processType, dynamicNonBuiltin);
+
+ // Check whether this event's category has recording enabled
+ if (!gEnabledCategories.GetEntry(GetCategory(lock, *eventKey))) {
+ return RecordEventResult::Ok;
+ }
+
+ EventRecordArray* eventRecords = GetEventRecordsForProcess(lock, processType);
+ eventRecords->AppendElement(EventRecord(timestamp, *eventKey, value, extra));
+
+ // Notify observers when we hit the "event" ping event record limit.
+ if (eventRecords->Length() == kEventPingLimit) {
+ return RecordEventResult::StorageLimitReached;
+ }
+
+ return RecordEventResult::Ok;
+}
+
+RecordEventResult ShouldRecordChildEvent(const StaticMutexAutoLock& lock,
+ const nsACString& category,
+ const nsACString& method,
+ const nsACString& object) {
+ EventKey* eventKey = GetEventKey(lock, category, method, object);
+ if (!eventKey) {
+ // This event is unknown in this process, but it might be a dynamic event
+ // that was registered in the parent process.
+ return RecordEventResult::Ok;
+ }
+
+ if (IsExpired(*eventKey)) {
+ return RecordEventResult::ExpiredEvent;
+ }
+
+ const auto processes =
+ gEventInfo[eventKey->id].common_info.record_in_processes;
+ if (!CanRecordInProcess(processes, XRE_GetProcessType())) {
+ return RecordEventResult::WrongProcess;
+ }
+
+ return RecordEventResult::Ok;
+}
+
+void RegisterEvents(const StaticMutexAutoLock& lock, const nsACString& category,
+ const nsTArray<DynamicEventInfo>& eventInfos,
+ const nsTArray<bool>& eventExpired, bool aBuiltin) {
+ MOZ_ASSERT(eventInfos.Length() == eventExpired.Length(),
+ "Event data array sizes should match.");
+
+ // Register the new events.
+ if (!gDynamicEventInfo) {
+ gDynamicEventInfo = new nsTArray<DynamicEventInfo>();
+ }
+
+ for (uint32_t i = 0, len = eventInfos.Length(); i < len; ++i) {
+ const nsCString& eventName = UniqueEventName(eventInfos[i]);
+
+ // Re-registering events can happen for two reasons and we don't print
+ // warnings:
+ //
+ // * When add-ons update.
+ // We don't support changing their definition, but the expiry might have
+ // changed.
+ // * When dynamic builtins ("build faster") events are registered.
+ // The dynamic definition takes precedence then.
+ EventKey* existing = nullptr;
+ if (!aBuiltin && gEventNameIDMap.Get(eventName, &existing)) {
+ if (eventExpired[i]) {
+ existing->id = kExpiredEventId;
+ }
+ continue;
+ }
+
+ gDynamicEventInfo->AppendElement(eventInfos[i]);
+ uint32_t eventId =
+ eventExpired[i] ? kExpiredEventId : gDynamicEventInfo->Length() - 1;
+ gEventNameIDMap.Put(eventName, new EventKey{eventId, true});
+ }
+
+ // If it is a builtin, add the category name in order to enable it later.
+ if (aBuiltin) {
+ gCategoryNames.PutEntry(category);
+ }
+
+ if (!aBuiltin) {
+ // Now after successful registration enable recording for this category
+ // (if not a dynamic builtin).
+ gEnabledCategories.PutEntry(category);
+ }
+}
+
+} // anonymous namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: thread-unsafe helpers for event handling.
+
+namespace {
+
+nsresult SerializeEventsArray(const EventRecordArray& events, JSContext* cx,
+ JS::MutableHandleObject result,
+ unsigned int dataset) {
+ // We serialize the events to a JS array.
+ JS::RootedObject eventsArray(cx, JS::NewArrayObject(cx, events.Length()));
+ if (!eventsArray) {
+ return NS_ERROR_FAILURE;
+ }
+
+ for (uint32_t i = 0; i < events.Length(); ++i) {
+ const EventRecord& record = events[i];
+
+ // Each entry is an array of one of the forms:
+ // [timestamp, category, method, object, value]
+ // [timestamp, category, method, object, null, extra]
+ // [timestamp, category, method, object, value, extra]
+ JS::RootedVector<JS::Value> items(cx);
+
+ // Add timestamp.
+ JS::Rooted<JS::Value> val(cx);
+ if (!items.append(JS::NumberValue(floor(record.Timestamp())))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Add category, method, object.
+ auto addCategoryMethodObjectValues = [&](const nsACString& category,
+ const nsACString& method,
+ const nsACString& object) -> bool {
+ return items.append(JS::StringValue(ToJSString(cx, category))) &&
+ items.append(JS::StringValue(ToJSString(cx, method))) &&
+ items.append(JS::StringValue(ToJSString(cx, object)));
+ };
+
+ const EventKey& eventKey = record.GetEventKey();
+ if (!eventKey.dynamic) {
+ const EventInfo& info = gEventInfo[eventKey.id];
+ if (!addCategoryMethodObjectValues(info.common_info.category(),
+ info.method(), info.object())) {
+ return NS_ERROR_FAILURE;
+ }
+ } else if (gDynamicEventInfo) {
+ const DynamicEventInfo& info = (*gDynamicEventInfo)[eventKey.id];
+ if (!addCategoryMethodObjectValues(info.category, info.method,
+ info.object)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ // Add the optional string value only when needed.
+ // When the value field is empty and extra is not set, we can save a little
+ // space that way. We still need to submit a null value if extra is set, to
+ // match the form: [ts, category, method, object, null, extra]
+ if (record.Value()) {
+ if (!items.append(
+ JS::StringValue(ToJSString(cx, record.Value().value())))) {
+ return NS_ERROR_FAILURE;
+ }
+ } else if (!record.Extra().IsEmpty()) {
+ if (!items.append(JS::NullValue())) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ // Add the optional extra dictionary.
+ // To save a little space, only add it when it is not empty.
+ if (!record.Extra().IsEmpty()) {
+ JS::RootedObject obj(cx, JS_NewPlainObject(cx));
+ if (!obj) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Add extra key & value entries.
+ const ExtraArray& extra = record.Extra();
+ for (uint32_t i = 0; i < extra.Length(); ++i) {
+ JS::Rooted<JS::Value> value(cx);
+ value.setString(ToJSString(cx, extra[i].value));
+
+ if (!JS_DefineProperty(cx, obj, extra[i].key.get(), value,
+ JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+ val.setObject(*obj);
+
+ if (!items.append(val)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ // Add the record to the events array.
+ JS::RootedObject itemsArray(cx, JS::NewArrayObject(cx, items));
+ if (!JS_DefineElement(cx, eventsArray, i, itemsArray, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ result.set(eventsArray);
+ return NS_OK;
+}
+
+} // anonymous namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// EXTERNALLY VISIBLE FUNCTIONS in namespace TelemetryEvents::
+
+// This is a StaticMutex rather than a plain Mutex (1) so that
+// it gets initialised in a thread-safe manner the first time
+// it is used, and (2) because it is never de-initialised, and
+// a normal Mutex would show up as a leak in BloatView. StaticMutex
+// also has the "OffTheBooks" property, so it won't show as a leak
+// in BloatView.
+// Another reason to use a StaticMutex instead of a plain Mutex is
+// that, due to the nature of Telemetry, we cannot rely on having a
+// mutex initialized in InitializeGlobalState. Unfortunately, we
+// cannot make sure that no other function is called before this point.
+static StaticMutex gTelemetryEventsMutex;
+
+void TelemetryEvent::InitializeGlobalState(bool aCanRecordBase,
+ bool aCanRecordExtended) {
+ StaticMutexAutoLock locker(gTelemetryEventsMutex);
+ MOZ_ASSERT(!gInitDone,
+ "TelemetryEvent::InitializeGlobalState "
+ "may only be called once");
+
+ gCanRecordBase = aCanRecordBase;
+ gCanRecordExtended = aCanRecordExtended;
+
+ // Populate the static event name->id cache. Note that the event names are
+ // statically allocated and come from the automatically generated
+ // TelemetryEventData.h.
+ const uint32_t eventCount =
+ static_cast<uint32_t>(mozilla::Telemetry::EventID::EventCount);
+ for (uint32_t i = 0; i < eventCount; ++i) {
+ const EventInfo& info = gEventInfo[i];
+ uint32_t eventId = i;
+
+ // If this event is expired or not recorded in this process, mark it with
+ // a special event id.
+ // This avoids doing repeated checks at runtime.
+ if (IsExpiredVersion(info.common_info.expiration_version().get())) {
+ eventId = kExpiredEventId;
+ }
+
+ gEventNameIDMap.Put(UniqueEventName(info), new EventKey{eventId, false});
+ gCategoryNames.PutEntry(info.common_info.category());
+ }
+
+ gInitDone = true;
+}
+
+void TelemetryEvent::DeInitializeGlobalState() {
+ StaticMutexAutoLock locker(gTelemetryEventsMutex);
+ MOZ_ASSERT(gInitDone);
+
+ gCanRecordBase = false;
+ gCanRecordExtended = false;
+
+ gEventNameIDMap.Clear();
+ gCategoryNames.Clear();
+ gEnabledCategories.Clear();
+ gEventRecords.Clear();
+
+ gDynamicEventInfo = nullptr;
+
+ gInitDone = false;
+}
+
+void TelemetryEvent::SetCanRecordBase(bool b) {
+ StaticMutexAutoLock locker(gTelemetryEventsMutex);
+ gCanRecordBase = b;
+}
+
+void TelemetryEvent::SetCanRecordExtended(bool b) {
+ StaticMutexAutoLock locker(gTelemetryEventsMutex);
+ gCanRecordExtended = b;
+}
+
+nsresult TelemetryEvent::RecordChildEvents(
+ ProcessID aProcessType,
+ const nsTArray<mozilla::Telemetry::ChildEventData>& aEvents) {
+ MOZ_ASSERT(XRE_IsParentProcess());
+ StaticMutexAutoLock locker(gTelemetryEventsMutex);
+ for (uint32_t i = 0; i < aEvents.Length(); ++i) {
+ const mozilla::Telemetry::ChildEventData& e = aEvents[i];
+
+ // Timestamps from child processes are absolute. We fix them up here to be
+ // relative to the main process start time.
+ // This allows us to put events from all processes on the same timeline.
+ double relativeTimestamp =
+ (e.timestamp - TimeStamp::ProcessCreation()).ToMilliseconds();
+
+ ::RecordEvent(locker, aProcessType, relativeTimestamp, e.category, e.method,
+ e.object, e.value, e.extra);
+ }
+ return NS_OK;
+}
+
+nsresult TelemetryEvent::RecordEvent(const nsACString& aCategory,
+ const nsACString& aMethod,
+ const nsACString& aObject,
+ JS::HandleValue aValue,
+ JS::HandleValue aExtra, JSContext* cx,
+ uint8_t optional_argc) {
+ // Check value argument.
+ if ((optional_argc > 0) && !aValue.isNull() && !aValue.isString()) {
+ LogToBrowserConsole(nsIScriptError::warningFlag,
+ u"Invalid type for value parameter."_ns);
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_RECORDING_ERROR::Value);
+ return NS_OK;
+ }
+
+ // Extract value parameter.
+ Maybe<nsCString> value;
+ if (aValue.isString()) {
+ nsAutoJSString jsStr;
+ if (!jsStr.init(cx, aValue)) {
+ LogToBrowserConsole(nsIScriptError::warningFlag,
+ u"Invalid string value for value parameter."_ns);
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_RECORDING_ERROR::Value);
+ return NS_OK;
+ }
+
+ nsCString str = NS_ConvertUTF16toUTF8(jsStr);
+ if (str.Length() > kMaxValueByteLength) {
+ LogToBrowserConsole(
+ nsIScriptError::warningFlag,
+ nsLiteralString(
+ u"Value parameter exceeds maximum string length, truncating."));
+ TruncateToByteLength(str, kMaxValueByteLength);
+ }
+ value = mozilla::Some(str);
+ }
+
+ // Check extra argument.
+ if ((optional_argc > 1) && !aExtra.isNull() && !aExtra.isObject()) {
+ LogToBrowserConsole(nsIScriptError::warningFlag,
+ u"Invalid type for extra parameter."_ns);
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_RECORDING_ERROR::Extra);
+ return NS_OK;
+ }
+
+ // Extract extra dictionary.
+ ExtraArray extra;
+ if (aExtra.isObject()) {
+ JS::RootedObject obj(cx, &aExtra.toObject());
+ JS::Rooted<JS::IdVector> ids(cx, JS::IdVector(cx));
+ if (!JS_Enumerate(cx, obj, &ids)) {
+ LogToBrowserConsole(nsIScriptError::warningFlag,
+ u"Failed to enumerate object."_ns);
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_RECORDING_ERROR::Extra);
+ return NS_OK;
+ }
+
+ for (size_t i = 0, n = ids.length(); i < n; i++) {
+ nsAutoJSString key;
+ if (!key.init(cx, ids[i])) {
+ LogToBrowserConsole(
+ nsIScriptError::warningFlag,
+ nsLiteralString(
+ u"Extra dictionary should only contain string keys."));
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_RECORDING_ERROR::Extra);
+ return NS_OK;
+ }
+
+ JS::Rooted<JS::Value> value(cx);
+ if (!JS_GetPropertyById(cx, obj, ids[i], &value)) {
+ LogToBrowserConsole(nsIScriptError::warningFlag,
+ u"Failed to get extra property."_ns);
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_RECORDING_ERROR::Extra);
+ return NS_OK;
+ }
+
+ nsAutoJSString jsStr;
+ if (!value.isString() || !jsStr.init(cx, value)) {
+ LogToBrowserConsole(nsIScriptError::warningFlag,
+ u"Extra properties should have string values."_ns);
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_RECORDING_ERROR::Extra);
+ return NS_OK;
+ }
+
+ nsCString str = NS_ConvertUTF16toUTF8(jsStr);
+ if (str.Length() > kMaxExtraValueByteLength) {
+ LogToBrowserConsole(
+ nsIScriptError::warningFlag,
+ nsLiteralString(
+ u"Extra value exceeds maximum string length, truncating."));
+ TruncateToByteLength(str, kMaxExtraValueByteLength);
+ }
+
+ extra.AppendElement(EventExtraEntry{NS_ConvertUTF16toUTF8(key), str});
+ }
+ }
+
+ // Lock for accessing internal data.
+ // While the lock is being held, no complex calls like JS calls can be made,
+ // as all of these could record Telemetry, which would result in deadlock.
+ RecordEventResult res;
+ if (!XRE_IsParentProcess()) {
+ {
+ StaticMutexAutoLock lock(gTelemetryEventsMutex);
+ res = ::ShouldRecordChildEvent(lock, aCategory, aMethod, aObject);
+ }
+
+ if (res == RecordEventResult::Ok) {
+ TelemetryIPCAccumulator::RecordChildEvent(
+ TimeStamp::NowLoRes(), aCategory, aMethod, aObject, value, extra);
+ }
+ } else {
+ StaticMutexAutoLock lock(gTelemetryEventsMutex);
+
+ if (!gInitDone) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Get the current time.
+ double timestamp = -1;
+ if (NS_WARN_IF(NS_FAILED(MsSinceProcessStart(&timestamp)))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ res = ::RecordEvent(lock, ProcessID::Parent, timestamp, aCategory, aMethod,
+ aObject, value, extra);
+ }
+
+ // Trigger warnings or errors where needed.
+ switch (res) {
+ case RecordEventResult::UnknownEvent: {
+ nsPrintfCString msg(R"(Unknown event: ["%s", "%s", "%s"])",
+ PromiseFlatCString(aCategory).get(),
+ PromiseFlatCString(aMethod).get(),
+ PromiseFlatCString(aObject).get());
+ LogToBrowserConsole(nsIScriptError::errorFlag,
+ NS_ConvertUTF8toUTF16(msg));
+ return NS_OK;
+ }
+ case RecordEventResult::InvalidExtraKey: {
+ nsPrintfCString msg(R"(Invalid extra key for event ["%s", "%s", "%s"].)",
+ PromiseFlatCString(aCategory).get(),
+ PromiseFlatCString(aMethod).get(),
+ PromiseFlatCString(aObject).get());
+ LogToBrowserConsole(nsIScriptError::warningFlag,
+ NS_ConvertUTF8toUTF16(msg));
+ return NS_OK;
+ }
+ case RecordEventResult::StorageLimitReached: {
+ LogToBrowserConsole(nsIScriptError::warningFlag,
+ u"Event storage limit reached."_ns);
+ nsCOMPtr<nsIObserverService> serv =
+ mozilla::services::GetObserverService();
+ if (serv) {
+ serv->NotifyObservers(nullptr, "event-telemetry-storage-limit-reached",
+ nullptr);
+ }
+ return NS_OK;
+ }
+ default:
+ return NS_OK;
+ }
+}
+
+void TelemetryEvent::RecordEventNative(
+ mozilla::Telemetry::EventID aId, const mozilla::Maybe<nsCString>& aValue,
+ const mozilla::Maybe<ExtraArray>& aExtra) {
+ // Truncate aValue if present and necessary.
+ mozilla::Maybe<nsCString> value;
+ if (aValue) {
+ nsCString valueStr = aValue.ref();
+ if (valueStr.Length() > kMaxValueByteLength) {
+ TruncateToByteLength(valueStr, kMaxValueByteLength);
+ }
+ value = mozilla::Some(valueStr);
+ }
+
+ // Truncate any over-long extra values.
+ ExtraArray extra;
+ if (aExtra) {
+ extra = aExtra.value();
+ for (auto& item : extra) {
+ if (item.value.Length() > kMaxExtraValueByteLength) {
+ TruncateToByteLength(item.value, kMaxExtraValueByteLength);
+ }
+ }
+ }
+
+ const EventInfo& info = gEventInfo[static_cast<uint32_t>(aId)];
+ const nsCString category(info.common_info.category());
+ const nsCString method(info.method());
+ const nsCString object(info.object());
+ if (!XRE_IsParentProcess()) {
+ RecordEventResult res;
+ {
+ StaticMutexAutoLock lock(gTelemetryEventsMutex);
+ res = ::ShouldRecordChildEvent(lock, category, method, object);
+ }
+
+ if (res == RecordEventResult::Ok) {
+ TelemetryIPCAccumulator::RecordChildEvent(TimeStamp::NowLoRes(), category,
+ method, object, value, extra);
+ }
+ } else {
+ StaticMutexAutoLock lock(gTelemetryEventsMutex);
+
+ if (!gInitDone) {
+ return;
+ }
+
+ // Get the current time.
+ double timestamp = -1;
+ if (NS_WARN_IF(NS_FAILED(MsSinceProcessStart(&timestamp)))) {
+ return;
+ }
+
+ ::RecordEvent(lock, ProcessID::Parent, timestamp, category, method, object,
+ value, extra);
+ }
+}
+
+static bool GetArrayPropertyValues(JSContext* cx, JS::HandleObject obj,
+ const char* property,
+ nsTArray<nsCString>* results) {
+ JS::RootedValue value(cx);
+ if (!JS_GetProperty(cx, obj, property, &value)) {
+ JS_ReportErrorASCII(cx, R"(Missing required property "%s" for event)",
+ property);
+ return false;
+ }
+
+ bool isArray = false;
+ if (!JS::IsArrayObject(cx, value, &isArray) || !isArray) {
+ JS_ReportErrorASCII(cx, R"(Property "%s" for event should be an array)",
+ property);
+ return false;
+ }
+
+ JS::RootedObject arrayObj(cx, &value.toObject());
+ uint32_t arrayLength;
+ if (!JS::GetArrayLength(cx, arrayObj, &arrayLength)) {
+ return false;
+ }
+
+ for (uint32_t arrayIdx = 0; arrayIdx < arrayLength; ++arrayIdx) {
+ JS::Rooted<JS::Value> element(cx);
+ if (!JS_GetElement(cx, arrayObj, arrayIdx, &element)) {
+ return false;
+ }
+
+ if (!element.isString()) {
+ JS_ReportErrorASCII(
+ cx, R"(Array entries for event property "%s" should be strings)",
+ property);
+ return false;
+ }
+
+ nsAutoJSString jsStr;
+ if (!jsStr.init(cx, element)) {
+ return false;
+ }
+
+ results->AppendElement(NS_ConvertUTF16toUTF8(jsStr));
+ }
+
+ return true;
+}
+
+nsresult TelemetryEvent::RegisterEvents(const nsACString& aCategory,
+ JS::Handle<JS::Value> aEventData,
+ bool aBuiltin, JSContext* cx) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Events can only be registered in the parent process");
+
+ if (!IsValidIdentifierString(aCategory, 30, true, true)) {
+ JS_ReportErrorASCII(
+ cx, "Category parameter should match the identifier pattern.");
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Category);
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ if (!aEventData.isObject()) {
+ JS_ReportErrorASCII(cx, "Event data parameter should be an object");
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Other);
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ JS::RootedObject obj(cx, &aEventData.toObject());
+ JS::Rooted<JS::IdVector> eventPropertyIds(cx, JS::IdVector(cx));
+ if (!JS_Enumerate(cx, obj, &eventPropertyIds)) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Other);
+ return NS_ERROR_FAILURE;
+ }
+
+ // Collect the event data into local storage first.
+ // Only after successfully validating all contained events will we register
+ // them into global storage.
+ nsTArray<DynamicEventInfo> newEventInfos;
+ nsTArray<bool> newEventExpired;
+
+ for (size_t i = 0, n = eventPropertyIds.length(); i < n; i++) {
+ nsAutoJSString eventName;
+ if (!eventName.init(cx, eventPropertyIds[i])) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Other);
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!IsValidIdentifierString(NS_ConvertUTF16toUTF8(eventName),
+ kMaxMethodNameByteLength, false, true)) {
+ JS_ReportErrorASCII(cx,
+ "Event names should match the identifier pattern.");
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Name);
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ JS::RootedValue value(cx);
+ if (!JS_GetPropertyById(cx, obj, eventPropertyIds[i], &value) ||
+ !value.isObject()) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Other);
+ return NS_ERROR_FAILURE;
+ }
+ JS::RootedObject eventObj(cx, &value.toObject());
+
+ // Extract the event registration data.
+ nsTArray<nsCString> methods;
+ nsTArray<nsCString> objects;
+ nsTArray<nsCString> extra_keys;
+ bool expired = false;
+ bool recordOnRelease = false;
+
+ // The methods & objects properties are required.
+ if (!GetArrayPropertyValues(cx, eventObj, "methods", &methods)) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Other);
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!GetArrayPropertyValues(cx, eventObj, "objects", &objects)) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Other);
+ return NS_ERROR_FAILURE;
+ }
+
+ // extra_keys is optional.
+ bool hasProperty = false;
+ if (JS_HasProperty(cx, eventObj, "extra_keys", &hasProperty) &&
+ hasProperty) {
+ if (!GetArrayPropertyValues(cx, eventObj, "extra_keys", &extra_keys)) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Other);
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ // expired is optional.
+ if (JS_HasProperty(cx, eventObj, "expired", &hasProperty) && hasProperty) {
+ JS::RootedValue temp(cx);
+ if (!JS_GetProperty(cx, eventObj, "expired", &temp) ||
+ !temp.isBoolean()) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Other);
+ return NS_ERROR_FAILURE;
+ }
+
+ expired = temp.toBoolean();
+ }
+
+ // record_on_release is optional.
+ if (JS_HasProperty(cx, eventObj, "record_on_release", &hasProperty) &&
+ hasProperty) {
+ JS::RootedValue temp(cx);
+ if (!JS_GetProperty(cx, eventObj, "record_on_release", &temp) ||
+ !temp.isBoolean()) {
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Other);
+ return NS_ERROR_FAILURE;
+ }
+
+ recordOnRelease = temp.toBoolean();
+ }
+
+ // Validate methods.
+ for (auto& method : methods) {
+ if (!IsValidIdentifierString(method, kMaxMethodNameByteLength, false,
+ true)) {
+ JS_ReportErrorASCII(
+ cx, "Method names should match the identifier pattern.");
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Method);
+ return NS_ERROR_INVALID_ARG;
+ }
+ }
+
+ // Validate objects.
+ for (auto& object : objects) {
+ if (!IsValidIdentifierString(object, kMaxObjectNameByteLength, false,
+ true)) {
+ JS_ReportErrorASCII(
+ cx, "Object names should match the identifier pattern.");
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::Object);
+ return NS_ERROR_INVALID_ARG;
+ }
+ }
+
+ // Validate extra keys.
+ if (extra_keys.Length() > kMaxExtraKeyCount) {
+ JS_ReportErrorASCII(cx, "No more than 10 extra keys can be registered.");
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::ExtraKeys);
+ return NS_ERROR_INVALID_ARG;
+ }
+ for (auto& key : extra_keys) {
+ if (!IsValidIdentifierString(key, kMaxExtraKeyNameByteLength, false,
+ true)) {
+ JS_ReportErrorASCII(
+ cx, "Extra key names should match the identifier pattern.");
+ mozilla::Telemetry::AccumulateCategorical(
+ LABELS_TELEMETRY_EVENT_REGISTRATION_ERROR::ExtraKeys);
+ return NS_ERROR_INVALID_ARG;
+ }
+ }
+
+ // Append event infos to be registered.
+ for (auto& method : methods) {
+ for (auto& object : objects) {
+ // We defer the actual registration here in case any other event
+ // description is invalid. In that case we don't need to roll back any
+ // partial registration.
+ DynamicEventInfo info{aCategory, method, object,
+ extra_keys, recordOnRelease, aBuiltin};
+ newEventInfos.AppendElement(info);
+ newEventExpired.AppendElement(expired);
+ }
+ }
+ }
+
+ {
+ StaticMutexAutoLock locker(gTelemetryEventsMutex);
+ RegisterEvents(locker, aCategory, newEventInfos, newEventExpired, aBuiltin);
+ }
+
+ return NS_OK;
+}
+
+nsresult TelemetryEvent::CreateSnapshots(uint32_t aDataset, bool aClear,
+ uint32_t aEventLimit, JSContext* cx,
+ uint8_t optional_argc,
+ JS::MutableHandleValue aResult) {
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Creating a JS snapshot of the events is a two-step process:
+ // (1) Lock the storage and copy the events into function-local storage.
+ // (2) Serialize the events into JS.
+ // We can't hold a lock for (2) because we will run into deadlocks otherwise
+ // from JS recording Telemetry.
+
+ // (1) Extract the events from storage with a lock held.
+ nsTArray<std::pair<const char*, EventRecordArray>> processEvents;
+ nsTArray<std::pair<uint32_t, EventRecordArray>> leftovers;
+ {
+ StaticMutexAutoLock locker(gTelemetryEventsMutex);
+
+ if (!gInitDone) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // The snapshotting function is the same for both static and dynamic builtin
+ // events. We can use the same function and store the events in the same
+ // output storage.
+ auto snapshotter = [aDataset, &locker, &processEvents, &leftovers, aClear,
+ optional_argc,
+ aEventLimit](EventRecordsMapType& aProcessStorage) {
+ for (auto iter = aProcessStorage.Iter(); !iter.Done(); iter.Next()) {
+ const EventRecordArray* eventStorage = iter.UserData();
+ EventRecordArray events;
+ EventRecordArray leftoverEvents;
+
+ const uint32_t len = eventStorage->Length();
+ for (uint32_t i = 0; i < len; ++i) {
+ const EventRecord& record = (*eventStorage)[i];
+ if (IsInDataset(GetDataset(locker, record.GetEventKey()), aDataset)) {
+ // If we have a limit, adhere to it. If we have a limit and are
+ // going to clear, save the leftovers for later.
+ if (optional_argc < 2 || events.Length() < aEventLimit) {
+ events.AppendElement(record);
+ } else if (aClear) {
+ leftoverEvents.AppendElement(record);
+ }
+ }
+ }
+
+ if (events.Length()) {
+ const char* processName = GetNameForProcessID(ProcessID(iter.Key()));
+ processEvents.AppendElement(
+ std::make_pair(processName, std::move(events)));
+ if (leftoverEvents.Length()) {
+ leftovers.AppendElement(
+ std::make_pair(iter.Key(), std::move(leftoverEvents)));
+ }
+ }
+ }
+ };
+
+ // Take a snapshot of the plain and dynamic builtin events.
+ snapshotter(gEventRecords);
+ if (aClear) {
+ gEventRecords.Clear();
+ for (auto& pair : leftovers) {
+ gEventRecords.Put(pair.first,
+ new EventRecordArray(std::move(pair.second)));
+ }
+ leftovers.Clear();
+ }
+ }
+
+ // (2) Serialize the events to a JS object.
+ JS::RootedObject rootObj(cx, JS_NewPlainObject(cx));
+ if (!rootObj) {
+ return NS_ERROR_FAILURE;
+ }
+
+ const uint32_t processLength = processEvents.Length();
+ for (uint32_t i = 0; i < processLength; ++i) {
+ JS::RootedObject eventsArray(cx);
+ if (NS_FAILED(SerializeEventsArray(processEvents[i].second, cx,
+ &eventsArray, aDataset))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!JS_DefineProperty(cx, rootObj, processEvents[i].first, eventsArray,
+ JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ aResult.setObject(*rootObj);
+ return NS_OK;
+}
+
+/**
+ * Resets all the stored events. This is intended to be only used in tests.
+ */
+void TelemetryEvent::ClearEvents() {
+ StaticMutexAutoLock lock(gTelemetryEventsMutex);
+
+ if (!gInitDone) {
+ return;
+ }
+
+ gEventRecords.Clear();
+}
+
+void TelemetryEvent::SetEventRecordingEnabled(const nsACString& category,
+ bool enabled) {
+ StaticMutexAutoLock locker(gTelemetryEventsMutex);
+
+ if (!gCategoryNames.Contains(category)) {
+ LogToBrowserConsole(
+ nsIScriptError::warningFlag,
+ NS_ConvertUTF8toUTF16(
+ nsLiteralCString(
+ "Unknown category for SetEventRecordingEnabled: ") +
+ category));
+ return;
+ }
+
+ if (enabled) {
+ gEnabledCategories.PutEntry(category);
+ } else {
+ gEnabledCategories.RemoveEntry(category);
+ }
+}
+
+size_t TelemetryEvent::SizeOfIncludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) {
+ StaticMutexAutoLock locker(gTelemetryEventsMutex);
+ size_t n = 0;
+
+ auto getSizeOfRecords = [aMallocSizeOf](auto& storageMap) {
+ size_t partial = storageMap.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ for (auto iter = storageMap.Iter(); !iter.Done(); iter.Next()) {
+ EventRecordArray* eventRecords = iter.UserData();
+ partial += eventRecords->ShallowSizeOfIncludingThis(aMallocSizeOf);
+
+ const uint32_t len = eventRecords->Length();
+ for (uint32_t i = 0; i < len; ++i) {
+ partial += (*eventRecords)[i].SizeOfExcludingThis(aMallocSizeOf);
+ }
+ }
+ return partial;
+ };
+
+ n += getSizeOfRecords(gEventRecords);
+
+ n += gEventNameIDMap.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ for (auto iter = gEventNameIDMap.ConstIter(); !iter.Done(); iter.Next()) {
+ n += iter.Key().SizeOfExcludingThisIfUnshared(aMallocSizeOf);
+ }
+
+ n += gCategoryNames.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ n += gEnabledCategories.ShallowSizeOfExcludingThis(aMallocSizeOf);
+
+ if (gDynamicEventInfo) {
+ n += gDynamicEventInfo->ShallowSizeOfIncludingThis(aMallocSizeOf);
+ for (auto& info : *gDynamicEventInfo) {
+ n += info.SizeOfExcludingThis(aMallocSizeOf);
+ }
+ }
+
+ return n;
+}
diff --git a/toolkit/components/telemetry/core/TelemetryEvent.h b/toolkit/components/telemetry/core/TelemetryEvent.h
new file mode 100644
index 0000000000..8c3ecd4f65
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryEvent.h
@@ -0,0 +1,71 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryEvent_h__
+#define TelemetryEvent_h__
+
+#include <stdint.h>
+#include "js/TypeDecls.h"
+#include "mozilla/Maybe.h"
+#include "mozilla/TelemetryEventEnums.h"
+#include "mozilla/TelemetryProcessEnums.h"
+#include "nsTArray.h"
+#include "nsString.h"
+
+namespace mozilla {
+namespace Telemetry {
+struct ChildEventData;
+struct EventExtraEntry;
+} // namespace Telemetry
+} // namespace mozilla
+
+using mozilla::Telemetry::EventExtraEntry;
+
+// This module is internal to Telemetry. It encapsulates Telemetry's
+// event recording and storage logic. It should only be used by
+// Telemetry.cpp. These functions should not be used anywhere else.
+// For the public interface to Telemetry functionality, see Telemetry.h.
+
+namespace TelemetryEvent {
+
+void InitializeGlobalState(bool canRecordBase, bool canRecordExtended);
+void DeInitializeGlobalState();
+
+void SetCanRecordBase(bool b);
+void SetCanRecordExtended(bool b);
+
+// C++ API Endpoint.
+void RecordEventNative(
+ mozilla::Telemetry::EventID aId, const mozilla::Maybe<nsCString>& aValue,
+ const mozilla::Maybe<CopyableTArray<EventExtraEntry>>& aExtra);
+
+// JS API Endpoints.
+nsresult RecordEvent(const nsACString& aCategory, const nsACString& aMethod,
+ const nsACString& aObject, JS::HandleValue aValue,
+ JS::HandleValue aExtra, JSContext* aCx,
+ uint8_t optional_argc);
+
+void SetEventRecordingEnabled(const nsACString& aCategory, bool aEnabled);
+nsresult RegisterEvents(const nsACString& aCategory,
+ JS::Handle<JS::Value> aEventData, bool aBuiltin,
+ JSContext* cx);
+
+nsresult CreateSnapshots(uint32_t aDataset, bool aClear, uint32_t aEventLimit,
+ JSContext* aCx, uint8_t optional_argc,
+ JS::MutableHandleValue aResult);
+
+// Record events from child processes.
+nsresult RecordChildEvents(
+ mozilla::Telemetry::ProcessID aProcessType,
+ const nsTArray<mozilla::Telemetry::ChildEventData>& aEvents);
+
+// Only to be used for testing.
+void ClearEvents();
+
+size_t SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf);
+
+} // namespace TelemetryEvent
+
+#endif // TelemetryEvent_h__
diff --git a/toolkit/components/telemetry/core/TelemetryHistogram.cpp b/toolkit/components/telemetry/core/TelemetryHistogram.cpp
new file mode 100644
index 0000000000..047682285c
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryHistogram.cpp
@@ -0,0 +1,3658 @@
+/* -*- 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 "TelemetryHistogram.h"
+
+#include <limits>
+#include "base/histogram.h"
+#include "geckoview/streaming/GeckoViewStreamingTelemetry.h"
+#include "ipc/TelemetryIPCAccumulator.h"
+#include "jsapi.h"
+#include "jsfriendapi.h"
+#include "js/Array.h" // JS::GetArrayLength, JS::IsArrayObject, JS::NewArrayObject
+#include "js/GCAPI.h"
+#include "js/Object.h" // JS::GetClass, JS::GetPrivate, JS::SetPrivate
+#include "mozilla/dom/ToJSValue.h"
+#include "mozilla/gfx/GPUProcessManager.h"
+#include "mozilla/Atomics.h"
+#include "mozilla/JSONWriter.h"
+#include "mozilla/StartupTimeline.h"
+#include "mozilla/StaticMutex.h"
+#include "mozilla/Unused.h"
+#include "nsClassHashtable.h"
+#include "nsString.h"
+#include "nsHashKeys.h"
+#include "nsITelemetry.h"
+#include "nsPrintfCString.h"
+#include "TelemetryHistogramNameMap.h"
+#include "TelemetryScalar.h"
+
+using base::BooleanHistogram;
+using base::CountHistogram;
+using base::FlagHistogram;
+using base::LinearHistogram;
+using mozilla::MakeTuple;
+using mozilla::StaticMutex;
+using mozilla::StaticMutexAutoLock;
+using mozilla::Telemetry::HistogramAccumulation;
+using mozilla::Telemetry::HistogramCount;
+using mozilla::Telemetry::HistogramID;
+using mozilla::Telemetry::HistogramIDByNameLookup;
+using mozilla::Telemetry::KeyedHistogramAccumulation;
+using mozilla::Telemetry::ProcessID;
+using mozilla::Telemetry::Common::CanRecordDataset;
+using mozilla::Telemetry::Common::CanRecordProduct;
+using mozilla::Telemetry::Common::GetCurrentProduct;
+using mozilla::Telemetry::Common::GetIDForProcessName;
+using mozilla::Telemetry::Common::GetNameForProcessID;
+using mozilla::Telemetry::Common::IsExpiredVersion;
+using mozilla::Telemetry::Common::IsInDataset;
+using mozilla::Telemetry::Common::LogToBrowserConsole;
+using mozilla::Telemetry::Common::RecordedProcessType;
+using mozilla::Telemetry::Common::StringHashSet;
+using mozilla::Telemetry::Common::SupportedProduct;
+using mozilla::Telemetry::Common::ToJSString;
+
+namespace TelemetryIPCAccumulator = mozilla::TelemetryIPCAccumulator;
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// Naming: there are two kinds of functions in this file:
+//
+// * Functions named internal_*: these can only be reached via an
+// interface function (TelemetryHistogram::*). They mostly expect
+// the interface function to have acquired
+// |gTelemetryHistogramMutex|, so they do not have to be
+// thread-safe. However, those internal_* functions that are
+// reachable from internal_WrapAndReturnHistogram and
+// internal_WrapAndReturnKeyedHistogram can sometimes be called
+// without |gTelemetryHistogramMutex|, and so might be racey.
+//
+// * Functions named TelemetryHistogram::*. This is the external interface.
+// Entries and exits to these functions are serialised using
+// |gTelemetryHistogramMutex|, except for GetKeyedHistogramSnapshots and
+// CreateHistogramSnapshots.
+//
+// Avoiding races and deadlocks:
+//
+// All functions in the external interface (TelemetryHistogram::*) are
+// serialised using the mutex |gTelemetryHistogramMutex|. This means
+// that the external interface is thread-safe, and many of the
+// internal_* functions can ignore thread safety. But it also brings
+// a danger of deadlock if any function in the external interface can
+// get back to that interface. That is, we will deadlock on any call
+// chain like this
+//
+// TelemetryHistogram::* -> .. any functions .. -> TelemetryHistogram::*
+//
+// To reduce the danger of that happening, observe the following rules:
+//
+// * No function in TelemetryHistogram::* may directly call, nor take the
+// address of, any other function in TelemetryHistogram::*.
+//
+// * No internal function internal_* may call, nor take the address
+// of, any function in TelemetryHistogram::*.
+//
+// internal_WrapAndReturnHistogram and
+// internal_WrapAndReturnKeyedHistogram are not protected by
+// |gTelemetryHistogramMutex| because they make calls to the JS
+// engine, but that can in turn call back to Telemetry and hence back
+// to a TelemetryHistogram:: function, in order to report GC and other
+// statistics. This would lead to deadlock due to attempted double
+// acquisition of |gTelemetryHistogramMutex|, if the internal_* functions
+// were required to be protected by |gTelemetryHistogramMutex|. To
+// break that cycle, we relax that requirement. Unfortunately this
+// means that this file is not guaranteed race-free.
+
+// This is a StaticMutex rather than a plain Mutex (1) so that
+// it gets initialised in a thread-safe manner the first time
+// it is used, and (2) because it is never de-initialised, and
+// a normal Mutex would show up as a leak in BloatView. StaticMutex
+// also has the "OffTheBooks" property, so it won't show as a leak
+// in BloatView.
+static StaticMutex gTelemetryHistogramMutex;
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE TYPES
+
+namespace {
+
+// Hardcoded probes
+//
+// The order of elements here is important to minimize the memory footprint of a
+// HistogramInfo instance.
+//
+// Any adjustements need to be reflected in gen_histogram_data.py
+struct HistogramInfo {
+ uint32_t min;
+ uint32_t max;
+ uint32_t bucketCount;
+ uint32_t name_offset;
+ uint32_t expiration_offset;
+ uint32_t label_count;
+ uint32_t key_count;
+ uint32_t store_count;
+ uint16_t label_index;
+ uint16_t key_index;
+ uint16_t store_index;
+ RecordedProcessType record_in_processes;
+ bool keyed;
+ uint8_t histogramType;
+ uint8_t dataset;
+ SupportedProduct products;
+
+ const char* name() const;
+ const char* expiration() const;
+ nsresult label_id(const char* label, uint32_t* labelId) const;
+ bool allows_key(const nsACString& key) const;
+ bool is_single_store() const;
+};
+
+// Structs used to keep information about the histograms for which a
+// snapshot should be created.
+struct HistogramSnapshotData {
+ CopyableTArray<base::Histogram::Sample> mBucketRanges;
+ CopyableTArray<base::Histogram::Count> mBucketCounts;
+ int64_t mSampleSum; // Same type as base::Histogram::SampleSet::sum_
+};
+
+struct HistogramSnapshotInfo {
+ HistogramSnapshotData data;
+ HistogramID histogramID;
+};
+
+typedef mozilla::Vector<HistogramSnapshotInfo> HistogramSnapshotsArray;
+typedef mozilla::Vector<HistogramSnapshotsArray> HistogramProcessSnapshotsArray;
+
+// The following is used to handle snapshot information for keyed histograms.
+typedef nsDataHashtable<nsCStringHashKey, HistogramSnapshotData>
+ KeyedHistogramSnapshotData;
+
+struct KeyedHistogramSnapshotInfo {
+ KeyedHistogramSnapshotData data;
+ HistogramID histogramId;
+};
+
+typedef mozilla::Vector<KeyedHistogramSnapshotInfo>
+ KeyedHistogramSnapshotsArray;
+typedef mozilla::Vector<KeyedHistogramSnapshotsArray>
+ KeyedHistogramProcessSnapshotsArray;
+
+/**
+ * A Histogram storage engine.
+ *
+ * Takes care of recording data into multiple stores if necessary.
+ */
+class Histogram {
+ public:
+ /*
+ * Create a new histogram instance from the given info.
+ *
+ * If the histogram is already expired, this does not allocate.
+ */
+ Histogram(HistogramID histogramId, const HistogramInfo& info, bool expired);
+ ~Histogram();
+
+ /**
+ * Add a sample to this histogram in all registered stores.
+ */
+ void Add(uint32_t sample);
+
+ /**
+ * Clear the named store for this histogram.
+ */
+ void Clear(const nsACString& store);
+
+ /**
+ * Get the histogram instance from the named store.
+ */
+ bool GetHistogram(const nsACString& store, base::Histogram** h);
+
+ bool IsExpired() const { return mIsExpired; }
+
+ size_t SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf);
+
+ private:
+ // String -> Histogram*
+ typedef nsClassHashtable<nsCStringHashKey, base::Histogram> HistogramStore;
+ HistogramStore mStorage;
+
+ // A valid pointer if this histogram belongs to only the main store
+ base::Histogram* mSingleStore;
+
+ // We don't track stores for expired histograms.
+ // We just store a single flag and all other operations become a no-op.
+ bool mIsExpired;
+};
+
+class KeyedHistogram {
+ public:
+ KeyedHistogram(HistogramID id, const HistogramInfo& info, bool expired);
+ ~KeyedHistogram();
+ nsresult GetHistogram(const nsCString& aStore, const nsCString& key,
+ base::Histogram** histogram);
+ base::Histogram* GetHistogram(const nsCString& aStore, const nsCString& key);
+ uint32_t GetHistogramType() const { return mHistogramInfo.histogramType; }
+ nsresult GetKeys(const StaticMutexAutoLock& aLock, const nsCString& store,
+ nsTArray<nsCString>& aKeys);
+ // Note: unlike other methods, GetJSSnapshot is thread safe.
+ nsresult GetJSSnapshot(JSContext* cx, JS::Handle<JSObject*> obj,
+ const nsACString& aStore, bool clearSubsession);
+ nsresult GetSnapshot(const StaticMutexAutoLock& aLock,
+ const nsACString& aStore,
+ KeyedHistogramSnapshotData& aSnapshot,
+ bool aClearSubsession);
+
+ nsresult Add(const nsCString& key, uint32_t aSample, ProcessID aProcessType);
+ void Clear(const nsACString& aStore);
+
+ HistogramID GetHistogramID() const { return mId; }
+
+ bool IsEmpty(const nsACString& aStore) const;
+
+ bool IsExpired() const { return mIsExpired; }
+
+ size_t SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf);
+
+ private:
+ typedef nsClassHashtable<nsCStringHashKey, base::Histogram>
+ KeyedHistogramMapType;
+ typedef nsClassHashtable<nsCStringHashKey, KeyedHistogramMapType>
+ StoreMapType;
+
+ StoreMapType mStorage;
+ // A valid map if this histogram belongs to only the main store
+ KeyedHistogramMapType* mSingleStore;
+
+ const HistogramID mId;
+ const HistogramInfo& mHistogramInfo;
+ bool mIsExpired;
+};
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE STATE, SHARED BY ALL THREADS
+
+namespace {
+
+// Set to true once this global state has been initialized
+bool gInitDone = false;
+
+// Whether we are collecting the base, opt-out, Histogram data.
+bool gCanRecordBase = false;
+// Whether we are collecting the extended, opt-in, Histogram data.
+bool gCanRecordExtended = false;
+
+// The storage for actual Histogram instances.
+// We use separate ones for plain and keyed histograms.
+Histogram** gHistogramStorage;
+// Keyed histograms internally map string keys to individual Histogram
+// instances.
+KeyedHistogram** gKeyedHistogramStorage;
+
+// To simplify logic below we use a single histogram instance for all expired
+// histograms.
+Histogram* gExpiredHistogram = nullptr;
+
+// The single placeholder for expired keyed histograms.
+KeyedHistogram* gExpiredKeyedHistogram = nullptr;
+
+// This tracks whether recording is enabled for specific histograms.
+// To utilize C++ initialization rules, we invert the meaning to "disabled".
+bool gHistogramRecordingDisabled[HistogramCount] = {};
+
+// This is for gHistogramInfos, gHistogramStringTable
+#include "TelemetryHistogramData.inc"
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE CONSTANTS
+
+namespace {
+
+// List of histogram IDs which should have recording disabled initially.
+const HistogramID kRecordingInitiallyDisabledIDs[] = {
+ mozilla::Telemetry::FX_REFRESH_DRIVER_SYNC_SCROLL_FRAME_DELAY_MS,
+
+ // The array must not be empty. Leave these item here.
+ mozilla::Telemetry::TELEMETRY_TEST_COUNT_INIT_NO_RECORD,
+ mozilla::Telemetry::TELEMETRY_TEST_KEYED_COUNT_INIT_NO_RECORD};
+
+const char* TEST_HISTOGRAM_PREFIX = "TELEMETRY_TEST_";
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// The core storage access functions.
+// They wrap access to the histogram storage and lookup caches.
+
+namespace {
+
+size_t internal_KeyedHistogramStorageIndex(HistogramID aHistogramId,
+ ProcessID aProcessId) {
+ return aHistogramId * size_t(ProcessID::Count) + size_t(aProcessId);
+}
+
+size_t internal_HistogramStorageIndex(const StaticMutexAutoLock& aLock,
+ HistogramID aHistogramId,
+ ProcessID aProcessId) {
+ static_assert(HistogramCount < std::numeric_limits<size_t>::max() /
+ size_t(ProcessID::Count),
+ "Too many histograms and processes to store in a 1D array.");
+
+ return aHistogramId * size_t(ProcessID::Count) + size_t(aProcessId);
+}
+
+Histogram* internal_GetHistogramFromStorage(const StaticMutexAutoLock& aLock,
+ HistogramID aHistogramId,
+ ProcessID aProcessId) {
+ size_t index =
+ internal_HistogramStorageIndex(aLock, aHistogramId, aProcessId);
+ return gHistogramStorage[index];
+}
+
+void internal_SetHistogramInStorage(const StaticMutexAutoLock& aLock,
+ HistogramID aHistogramId,
+ ProcessID aProcessId,
+ Histogram* aHistogram) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Histograms are stored only in the parent process.");
+
+ size_t index =
+ internal_HistogramStorageIndex(aLock, aHistogramId, aProcessId);
+ MOZ_ASSERT(!gHistogramStorage[index],
+ "Mustn't overwrite storage without clearing it first.");
+ gHistogramStorage[index] = aHistogram;
+}
+
+KeyedHistogram* internal_GetKeyedHistogramFromStorage(HistogramID aHistogramId,
+ ProcessID aProcessId) {
+ size_t index = internal_KeyedHistogramStorageIndex(aHistogramId, aProcessId);
+ return gKeyedHistogramStorage[index];
+}
+
+void internal_SetKeyedHistogramInStorage(HistogramID aHistogramId,
+ ProcessID aProcessId,
+ KeyedHistogram* aKeyedHistogram) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Keyed Histograms are stored only in the parent process.");
+
+ size_t index = internal_KeyedHistogramStorageIndex(aHistogramId, aProcessId);
+ MOZ_ASSERT(!gKeyedHistogramStorage[index],
+ "Mustn't overwrite storage without clearing it first");
+ gKeyedHistogramStorage[index] = aKeyedHistogram;
+}
+
+// Factory function for base::Histogram instances.
+base::Histogram* internal_CreateBaseHistogramInstance(const HistogramInfo& info,
+ int bucketsOffset);
+
+// Factory function for histogram instances.
+Histogram* internal_CreateHistogramInstance(HistogramID histogramId);
+
+bool internal_IsHistogramEnumId(HistogramID aID) {
+ static_assert(((HistogramID)-1 > 0), "ID should be unsigned.");
+ return aID < HistogramCount;
+}
+
+// Look up a plain histogram by id.
+Histogram* internal_GetHistogramById(const StaticMutexAutoLock& aLock,
+ HistogramID histogramId,
+ ProcessID processId,
+ bool instantiate = true) {
+ MOZ_ASSERT(internal_IsHistogramEnumId(histogramId));
+ MOZ_ASSERT(!gHistogramInfos[histogramId].keyed);
+ MOZ_ASSERT(processId < ProcessID::Count);
+
+ Histogram* h =
+ internal_GetHistogramFromStorage(aLock, histogramId, processId);
+ if (h || !instantiate) {
+ return h;
+ }
+
+ h = internal_CreateHistogramInstance(histogramId);
+ MOZ_ASSERT(h);
+ internal_SetHistogramInStorage(aLock, histogramId, processId, h);
+
+ return h;
+}
+
+// Look up a keyed histogram by id.
+KeyedHistogram* internal_GetKeyedHistogramById(HistogramID histogramId,
+ ProcessID processId,
+ bool instantiate = true) {
+ MOZ_ASSERT(internal_IsHistogramEnumId(histogramId));
+ MOZ_ASSERT(gHistogramInfos[histogramId].keyed);
+ MOZ_ASSERT(processId < ProcessID::Count);
+
+ KeyedHistogram* kh =
+ internal_GetKeyedHistogramFromStorage(histogramId, processId);
+ if (kh || !instantiate) {
+ return kh;
+ }
+
+ const HistogramInfo& info = gHistogramInfos[histogramId];
+ const bool isExpired = IsExpiredVersion(info.expiration());
+
+ // If the keyed histogram is expired, set its storage to the expired
+ // keyed histogram.
+ if (isExpired) {
+ if (!gExpiredKeyedHistogram) {
+ // If we don't have an expired keyed histogram, create one.
+ gExpiredKeyedHistogram =
+ new KeyedHistogram(histogramId, info, true /* expired */);
+ MOZ_ASSERT(gExpiredKeyedHistogram);
+ }
+ kh = gExpiredKeyedHistogram;
+ } else {
+ kh = new KeyedHistogram(histogramId, info, false /* expired */);
+ }
+
+ internal_SetKeyedHistogramInStorage(histogramId, processId, kh);
+
+ return kh;
+}
+
+// Look up a histogram id from a histogram name.
+nsresult internal_GetHistogramIdByName(const StaticMutexAutoLock& aLock,
+ const nsACString& name,
+ HistogramID* id) {
+ const uint32_t idx = HistogramIDByNameLookup(name);
+ MOZ_ASSERT(idx < HistogramCount,
+ "Intermediate lookup should always give a valid index.");
+
+ // The lookup hashes the input and uses it as an index into the value array.
+ // Hash collisions can still happen for unknown values,
+ // therefore we check that the name matches.
+ if (name.Equals(gHistogramInfos[idx].name())) {
+ *id = HistogramID(idx);
+ return NS_OK;
+ }
+
+ return NS_ERROR_ILLEGAL_VALUE;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: Misc small helpers
+
+namespace {
+
+bool internal_CanRecordBase() { return gCanRecordBase; }
+
+bool internal_CanRecordExtended() { return gCanRecordExtended; }
+
+bool internal_AttemptedGPUProcess() {
+ // Check if it was tried to launch a process.
+ bool attemptedGPUProcess = false;
+ if (auto gpm = mozilla::gfx::GPUProcessManager::Get()) {
+ attemptedGPUProcess = gpm->AttemptedGPUProcess();
+ }
+ return attemptedGPUProcess;
+}
+
+// Note: this is completely unrelated to mozilla::IsEmpty.
+bool internal_IsEmpty(const StaticMutexAutoLock& aLock,
+ const base::Histogram* h) {
+ return h->is_empty();
+}
+
+void internal_SetHistogramRecordingEnabled(const StaticMutexAutoLock& aLock,
+ HistogramID id, bool aEnabled) {
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+ gHistogramRecordingDisabled[id] = !aEnabled;
+}
+
+bool internal_IsRecordingEnabled(HistogramID id) {
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+ return !gHistogramRecordingDisabled[id];
+}
+
+const char* HistogramInfo::name() const {
+ return &gHistogramStringTable[this->name_offset];
+}
+
+const char* HistogramInfo::expiration() const {
+ return &gHistogramStringTable[this->expiration_offset];
+}
+
+nsresult HistogramInfo::label_id(const char* label, uint32_t* labelId) const {
+ MOZ_ASSERT(label);
+ MOZ_ASSERT(this->histogramType == nsITelemetry::HISTOGRAM_CATEGORICAL);
+ if (this->histogramType != nsITelemetry::HISTOGRAM_CATEGORICAL) {
+ return NS_ERROR_FAILURE;
+ }
+
+ for (uint32_t i = 0; i < this->label_count; ++i) {
+ // gHistogramLabelTable contains the indices of the label strings in the
+ // gHistogramStringTable.
+ // They are stored in-order and consecutively, from the offset label_index
+ // to (label_index + label_count).
+ uint32_t string_offset = gHistogramLabelTable[this->label_index + i];
+ const char* const str = &gHistogramStringTable[string_offset];
+ if (::strcmp(label, str) == 0) {
+ *labelId = i;
+ return NS_OK;
+ }
+ }
+
+ return NS_ERROR_FAILURE;
+}
+
+bool HistogramInfo::allows_key(const nsACString& key) const {
+ MOZ_ASSERT(this->keyed);
+
+ // If we didn't specify a list of allowed keys, just return true.
+ if (this->key_count == 0) {
+ return true;
+ }
+
+ // Otherwise, check if |key| is in the list of allowed keys.
+ for (uint32_t i = 0; i < this->key_count; ++i) {
+ // gHistogramKeyTable contains the indices of the key strings in the
+ // gHistogramStringTable. They are stored in-order and consecutively,
+ // from the offset key_index to (key_index + key_count).
+ uint32_t string_offset = gHistogramKeyTable[this->key_index + i];
+ const char* const str = &gHistogramStringTable[string_offset];
+ if (key.EqualsASCII(str)) {
+ return true;
+ }
+ }
+
+ // |key| was not found.
+ return false;
+}
+
+bool HistogramInfo::is_single_store() const {
+ return store_count == 1 && store_index == UINT16_MAX;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: Histogram Get, Add, Clone, Clear functions
+
+namespace {
+
+nsresult internal_CheckHistogramArguments(const HistogramInfo& info) {
+ if (info.histogramType != nsITelemetry::HISTOGRAM_BOOLEAN &&
+ info.histogramType != nsITelemetry::HISTOGRAM_FLAG &&
+ info.histogramType != nsITelemetry::HISTOGRAM_COUNT) {
+ // Sanity checks for histogram parameters.
+ if (info.min >= info.max) {
+ return NS_ERROR_ILLEGAL_VALUE;
+ }
+
+ if (info.bucketCount <= 2) {
+ return NS_ERROR_ILLEGAL_VALUE;
+ }
+
+ if (info.min < 1) {
+ return NS_ERROR_ILLEGAL_VALUE;
+ }
+ }
+
+ return NS_OK;
+}
+
+Histogram* internal_CreateHistogramInstance(HistogramID histogramId) {
+ const HistogramInfo& info = gHistogramInfos[histogramId];
+
+ if (NS_FAILED(internal_CheckHistogramArguments(info))) {
+ MOZ_ASSERT(false, "Failed histogram argument checks.");
+ return nullptr;
+ }
+
+ const bool isExpired = IsExpiredVersion(info.expiration());
+
+ if (isExpired) {
+ if (!gExpiredHistogram) {
+ gExpiredHistogram = new Histogram(histogramId, info, /* expired */ true);
+ }
+
+ return gExpiredHistogram;
+ }
+
+ Histogram* wrapper = new Histogram(histogramId, info, /* expired */ false);
+
+ return wrapper;
+}
+
+base::Histogram* internal_CreateBaseHistogramInstance(
+ const HistogramInfo& passedInfo, int bucketsOffset) {
+ if (NS_FAILED(internal_CheckHistogramArguments(passedInfo))) {
+ MOZ_ASSERT(false, "Failed histogram argument checks.");
+ return nullptr;
+ }
+
+ // We don't actually store data for expired histograms at all.
+ MOZ_ASSERT(!IsExpiredVersion(passedInfo.expiration()));
+
+ HistogramInfo info = passedInfo;
+ const int* buckets = &gHistogramBucketLowerBounds[bucketsOffset];
+
+ base::Histogram::Flags flags = base::Histogram::kNoFlags;
+ base::Histogram* h = nullptr;
+ switch (info.histogramType) {
+ case nsITelemetry::HISTOGRAM_EXPONENTIAL:
+ h = base::Histogram::FactoryGet(info.min, info.max, info.bucketCount,
+ flags, buckets);
+ break;
+ case nsITelemetry::HISTOGRAM_LINEAR:
+ case nsITelemetry::HISTOGRAM_CATEGORICAL:
+ h = LinearHistogram::FactoryGet(info.min, info.max, info.bucketCount,
+ flags, buckets);
+ break;
+ case nsITelemetry::HISTOGRAM_BOOLEAN:
+ h = BooleanHistogram::FactoryGet(flags, buckets);
+ break;
+ case nsITelemetry::HISTOGRAM_FLAG:
+ h = FlagHistogram::FactoryGet(flags, buckets);
+ break;
+ case nsITelemetry::HISTOGRAM_COUNT:
+ h = CountHistogram::FactoryGet(flags, buckets);
+ break;
+ default:
+ MOZ_ASSERT(false, "Invalid histogram type");
+ return nullptr;
+ }
+
+ return h;
+}
+
+nsresult internal_HistogramAdd(const StaticMutexAutoLock& aLock,
+ Histogram& histogram, const HistogramID id,
+ uint32_t value, ProcessID aProcessType) {
+ // Check if we are allowed to record the data.
+ bool canRecordDataset =
+ CanRecordDataset(gHistogramInfos[id].dataset, internal_CanRecordBase(),
+ internal_CanRecordExtended());
+ // If `histogram` is a non-parent-process histogram, then recording-enabled
+ // has been checked in its owner process.
+ if (!canRecordDataset ||
+ (aProcessType == ProcessID::Parent && !internal_IsRecordingEnabled(id))) {
+ return NS_OK;
+ }
+
+ // Don't record if the current platform is not enabled
+ if (!CanRecordProduct(gHistogramInfos[id].products)) {
+ return NS_OK;
+ }
+
+ if (&histogram != gExpiredHistogram &&
+ GetCurrentProduct() == SupportedProduct::GeckoviewStreaming) {
+ const HistogramInfo& info = gHistogramInfos[id];
+ GeckoViewStreamingTelemetry::HistogramAccumulate(
+ nsDependentCString(info.name()),
+ info.histogramType == nsITelemetry::HISTOGRAM_CATEGORICAL, value);
+ return NS_OK;
+ }
+
+ // The internal representation of a base::Histogram's buckets uses `int`.
+ // Clamp large values of `value` to be INT_MAX so they continue to be treated
+ // as large values (instead of negative ones).
+ if (value > INT_MAX) {
+ TelemetryScalar::Add(
+ mozilla::Telemetry::ScalarID::TELEMETRY_ACCUMULATE_CLAMPED_VALUES,
+ NS_ConvertASCIItoUTF16(gHistogramInfos[id].name()), 1);
+ value = INT_MAX;
+ }
+
+ histogram.Add(value);
+
+ return NS_OK;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: Histogram reflection helpers
+
+namespace {
+
+/**
+ * Copy histograms and samples to Mozilla-friendly structures.
+ * Please note that this version does not make use of JS contexts.
+ *
+ * @param {StaticMutexAutoLock} the proof we hold the mutex.
+ * @param {Histogram} the histogram to reflect.
+ * @return {nsresult} NS_ERROR_FAILURE if we fail to allocate memory for the
+ * snapshot.
+ */
+nsresult internal_GetHistogramAndSamples(const StaticMutexAutoLock& aLock,
+ const base::Histogram* h,
+ HistogramSnapshotData& aSnapshot) {
+ MOZ_ASSERT(h);
+
+ // Convert the ranges of the buckets to a nsTArray.
+ const size_t bucketCount = h->bucket_count();
+ for (size_t i = 0; i < bucketCount; i++) {
+ // XXX(Bug 1631371) Check if this should use a fallible operation as it
+ // pretended earlier, or change the return type to void.
+ aSnapshot.mBucketRanges.AppendElement(h->ranges(i));
+ }
+
+ // Get a snapshot of the samples.
+ base::Histogram::SampleSet ss = h->SnapshotSample();
+
+ // Get the number of samples in each bucket.
+ for (size_t i = 0; i < bucketCount; i++) {
+ // XXX(Bug 1631371) Check if this should use a fallible operation as it
+ // pretended earlier, or change the return type to void.
+ aSnapshot.mBucketCounts.AppendElement(ss.counts(i));
+ }
+
+ // Finally, save the |sum|. We don't need to reflect declared_min,
+ // declared_max and histogram_type as they are in gHistogramInfo.
+ aSnapshot.mSampleSum = ss.sum();
+ return NS_OK;
+}
+
+/**
+ * Reflect a histogram snapshot into a JavaScript object.
+ * The returned histogram object will have the following properties:
+ *
+ * bucket_count - Number of buckets of this histogram
+ * histogram_type - HISTOGRAM_EXPONENTIAL, HISTOGRAM_LINEAR,
+ * HISTOGRAM_BOOLEAN, HISTOGRAM_FLAG, HISTOGRAM_COUNT, or HISTOGRAM_CATEGORICAL
+ * sum - sum of the bucket contents
+ * range - A 2-item array of minimum and maximum bucket size
+ * values - Map from bucket start to the bucket's count
+ */
+nsresult internal_ReflectHistogramAndSamples(
+ JSContext* cx, JS::Handle<JSObject*> obj,
+ const HistogramInfo& aHistogramInfo,
+ const HistogramSnapshotData& aSnapshot) {
+ if (!(JS_DefineProperty(cx, obj, "bucket_count", aHistogramInfo.bucketCount,
+ JSPROP_ENUMERATE) &&
+ JS_DefineProperty(cx, obj, "histogram_type",
+ aHistogramInfo.histogramType, JSPROP_ENUMERATE) &&
+ JS_DefineProperty(cx, obj, "sum", double(aSnapshot.mSampleSum),
+ JSPROP_ENUMERATE))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Don't rely on the bucket counts from "aHistogramInfo": it may
+ // differ from the length of aSnapshot.mBucketCounts due to expired
+ // histograms.
+ const size_t count = aSnapshot.mBucketCounts.Length();
+ MOZ_ASSERT(count == aSnapshot.mBucketRanges.Length(),
+ "The number of buckets and the number of counts must match.");
+
+ // Create the "range" property and add it to the final object.
+ JS::Rooted<JSObject*> rarray(cx, JS::NewArrayObject(cx, 2));
+ if (rarray == nullptr ||
+ !JS_DefineProperty(cx, obj, "range", rarray, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ // Add [min, max] into the range array
+ if (!JS_DefineElement(cx, rarray, 0, aHistogramInfo.min, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ if (!JS_DefineElement(cx, rarray, 1, aHistogramInfo.max, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ JS::Rooted<JSObject*> values(cx, JS_NewPlainObject(cx));
+ if (values == nullptr ||
+ !JS_DefineProperty(cx, obj, "values", values, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ bool first = true;
+ size_t last = 0;
+
+ for (size_t i = 0; i < count; i++) {
+ auto value = aSnapshot.mBucketCounts[i];
+ if (value == 0) {
+ continue;
+ }
+
+ if (i > 0 && first) {
+ auto range = aSnapshot.mBucketRanges[i - 1];
+ if (!JS_DefineProperty(cx, values, nsPrintfCString("%d", range).get(), 0,
+ JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ first = false;
+ last = i + 1;
+
+ auto range = aSnapshot.mBucketRanges[i];
+ if (!JS_DefineProperty(cx, values, nsPrintfCString("%d", range).get(),
+ value, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ if (last > 0 && last < count) {
+ auto range = aSnapshot.mBucketRanges[last];
+ if (!JS_DefineProperty(cx, values, nsPrintfCString("%d", range).get(), 0,
+ JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ return NS_OK;
+}
+
+bool internal_ShouldReflectHistogram(const StaticMutexAutoLock& aLock,
+ base::Histogram* h, HistogramID id) {
+ // Only flag histograms are serialized when they are empty.
+ // This has historical reasons, changing this will require downstream changes.
+ // The cheaper path here is to just deprecate flag histograms in favor
+ // of scalars.
+ uint32_t type = gHistogramInfos[id].histogramType;
+ if (internal_IsEmpty(aLock, h) && type != nsITelemetry::HISTOGRAM_FLAG) {
+ return false;
+ }
+
+ // Don't reflect the histogram if it's not allowed in this product.
+ if (!CanRecordProduct(gHistogramInfos[id].products)) {
+ return false;
+ }
+
+ return true;
+}
+
+/**
+ * Helper function to get a snapshot of the histograms.
+ *
+ * @param {aLock} the lock proof.
+ * @param {aStore} the name of the store to snapshot.
+ * @param {aDataset} the dataset for which the snapshot is being requested.
+ * @param {aClearSubsession} whether or not to clear the data after
+ * taking the snapshot.
+ * @param {aIncludeGPU} whether or not to include data for the GPU.
+ * @param {aOutSnapshot} the container in which the snapshot data will be
+ * stored.
+ * @return {nsresult} NS_OK if the snapshot was successfully taken or
+ * NS_ERROR_OUT_OF_MEMORY if it failed to allocate memory.
+ */
+nsresult internal_GetHistogramsSnapshot(
+ const StaticMutexAutoLock& aLock, const nsACString& aStore,
+ unsigned int aDataset, bool aClearSubsession, bool aIncludeGPU,
+ bool aFilterTest, HistogramProcessSnapshotsArray& aOutSnapshot) {
+ if (!aOutSnapshot.resize(static_cast<uint32_t>(ProcessID::Count))) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+
+ for (uint32_t process = 0; process < static_cast<uint32_t>(ProcessID::Count);
+ ++process) {
+ HistogramSnapshotsArray& hArray = aOutSnapshot[process];
+
+ for (size_t i = 0; i < HistogramCount; ++i) {
+ const HistogramInfo& info = gHistogramInfos[i];
+ if (info.keyed) {
+ continue;
+ }
+
+ HistogramID id = HistogramID(i);
+
+ if (!CanRecordInProcess(info.record_in_processes, ProcessID(process)) ||
+ ((ProcessID(process) == ProcessID::Gpu) && !aIncludeGPU)) {
+ continue;
+ }
+
+ if (!IsInDataset(info.dataset, aDataset)) {
+ continue;
+ }
+
+ bool shouldInstantiate =
+ info.histogramType == nsITelemetry::HISTOGRAM_FLAG;
+ Histogram* w = internal_GetHistogramById(aLock, id, ProcessID(process),
+ shouldInstantiate);
+ if (!w || w->IsExpired()) {
+ continue;
+ }
+
+ base::Histogram* h = nullptr;
+ if (!w->GetHistogram(aStore, &h)) {
+ continue;
+ }
+
+ if (!internal_ShouldReflectHistogram(aLock, h, id)) {
+ continue;
+ }
+
+ const char* name = info.name();
+ if (aFilterTest && strncmp(TEST_HISTOGRAM_PREFIX, name,
+ strlen(TEST_HISTOGRAM_PREFIX)) == 0) {
+ if (aClearSubsession) {
+ h->Clear();
+ }
+ continue;
+ }
+
+ HistogramSnapshotData snapshotData;
+ if (NS_FAILED(internal_GetHistogramAndSamples(aLock, h, snapshotData))) {
+ continue;
+ }
+
+ if (!hArray.emplaceBack(HistogramSnapshotInfo{snapshotData, id})) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+
+ if (aClearSubsession) {
+ h->Clear();
+ }
+ }
+ }
+ return NS_OK;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: class Histogram
+
+namespace {
+
+Histogram::Histogram(HistogramID histogramId, const HistogramInfo& info,
+ bool expired)
+ : mStorage(), mSingleStore(nullptr), mIsExpired(expired) {
+ if (IsExpired()) {
+ return;
+ }
+
+ base::Histogram* h;
+ const int bucketsOffset = gHistogramBucketLowerBoundIndex[histogramId];
+
+ if (info.is_single_store()) {
+ mSingleStore = internal_CreateBaseHistogramInstance(info, bucketsOffset);
+ } else {
+ for (uint32_t i = 0; i < info.store_count; i++) {
+ auto store = nsDependentCString(
+ &gHistogramStringTable[gHistogramStoresTable[info.store_index + i]]);
+ h = internal_CreateBaseHistogramInstance(info, bucketsOffset);
+ mStorage.Put(store, h);
+ }
+ }
+}
+
+Histogram::~Histogram() { delete mSingleStore; }
+
+void Histogram::Add(uint32_t sample) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Only add to histograms in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return;
+ }
+
+ if (IsExpired()) {
+ return;
+ }
+
+ if (mSingleStore != nullptr) {
+ mSingleStore->Add(sample);
+ } else {
+ for (auto iter = mStorage.Iter(); !iter.Done(); iter.Next()) {
+ auto& h = iter.Data();
+ h->Add(sample);
+ }
+ }
+}
+
+void Histogram::Clear(const nsACString& store) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Only clear histograms in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return;
+ }
+
+ if (mSingleStore != nullptr) {
+ if (store.EqualsASCII("main")) {
+ mSingleStore->Clear();
+ }
+ } else {
+ base::Histogram* h = nullptr;
+ bool found = GetHistogram(store, &h);
+ if (!found) {
+ return;
+ }
+ MOZ_ASSERT(h, "Should have found a valid histogram in the named store");
+
+ h->Clear();
+ }
+}
+
+bool Histogram::GetHistogram(const nsACString& store, base::Histogram** h) {
+ MOZ_ASSERT(!IsExpired());
+ if (IsExpired()) {
+ return false;
+ }
+
+ if (mSingleStore != nullptr) {
+ if (store.EqualsASCII("main")) {
+ *h = mSingleStore;
+ return true;
+ }
+
+ return false;
+ }
+
+ return mStorage.Get(store, h);
+}
+
+size_t Histogram::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) {
+ size_t n = 0;
+ n += aMallocSizeOf(this);
+ /*
+ * In theory mStorage.SizeOfExcludingThis should included the data part of the
+ * map, but the numbers seemed low, so we are only taking the shallow size and
+ * do the iteration here.
+ */
+ n += mStorage.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ for (auto iter = mStorage.Iter(); !iter.Done(); iter.Next()) {
+ auto& h = iter.Data();
+ n += h->SizeOfIncludingThis(aMallocSizeOf);
+ }
+ if (mSingleStore != nullptr) {
+ // base::Histogram doesn't have SizeOfExcludingThis, so we are overcounting
+ // the pointer here.
+ n += mSingleStore->SizeOfIncludingThis(aMallocSizeOf);
+ }
+ return n;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: class KeyedHistogram and internal_ReflectKeyedHistogram
+
+namespace {
+
+nsresult internal_ReflectKeyedHistogram(
+ const KeyedHistogramSnapshotData& aSnapshot, const HistogramInfo& info,
+ JSContext* aCx, JS::Handle<JSObject*> aObj) {
+ for (auto iter = aSnapshot.ConstIter(); !iter.Done(); iter.Next()) {
+ HistogramSnapshotData& keyData = iter.Data();
+
+ JS::RootedObject histogramSnapshot(aCx, JS_NewPlainObject(aCx));
+ if (!histogramSnapshot) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (NS_FAILED(internal_ReflectHistogramAndSamples(aCx, histogramSnapshot,
+ info, keyData))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ const NS_ConvertUTF8toUTF16 key(iter.Key());
+ if (!JS_DefineUCProperty(aCx, aObj, key.Data(), key.Length(),
+ histogramSnapshot, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ return NS_OK;
+}
+
+KeyedHistogram::KeyedHistogram(HistogramID id, const HistogramInfo& info,
+ bool expired)
+ : mStorage(),
+ mSingleStore(nullptr),
+ mId(id),
+ mHistogramInfo(info),
+ mIsExpired(expired) {
+ if (IsExpired()) {
+ return;
+ }
+
+ if (info.is_single_store()) {
+ mSingleStore = new KeyedHistogramMapType;
+ } else {
+ for (uint32_t i = 0; i < info.store_count; i++) {
+ auto store = nsDependentCString(
+ &gHistogramStringTable[gHistogramStoresTable[info.store_index + i]]);
+ mStorage.Put(store, new KeyedHistogramMapType);
+ }
+ }
+}
+
+KeyedHistogram::~KeyedHistogram() { delete mSingleStore; }
+
+nsresult KeyedHistogram::GetHistogram(const nsCString& aStore,
+ const nsCString& key,
+ base::Histogram** histogram) {
+ if (IsExpired()) {
+ MOZ_ASSERT(false,
+ "KeyedHistogram::GetHistogram called on an expired histogram.");
+ return NS_ERROR_FAILURE;
+ }
+
+ KeyedHistogramMapType* histogramMap;
+ bool found;
+
+ if (mSingleStore != nullptr) {
+ histogramMap = mSingleStore;
+ } else {
+ found = mStorage.Get(aStore, &histogramMap);
+ if (!found) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ found = histogramMap->Get(key, histogram);
+ if (found) {
+ return NS_OK;
+ }
+
+ int bucketsOffset = gHistogramBucketLowerBoundIndex[mId];
+ base::Histogram* h =
+ internal_CreateBaseHistogramInstance(mHistogramInfo, bucketsOffset);
+ if (!h) {
+ return NS_ERROR_FAILURE;
+ }
+
+ h->ClearFlags(base::Histogram::kUmaTargetedHistogramFlag);
+ *histogram = h;
+
+ bool inserted = histogramMap->Put(key, h, mozilla::fallible);
+ if (MOZ_UNLIKELY(!inserted)) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+ return NS_OK;
+}
+
+base::Histogram* KeyedHistogram::GetHistogram(const nsCString& aStore,
+ const nsCString& key) {
+ base::Histogram* h = nullptr;
+ if (NS_FAILED(GetHistogram(aStore, key, &h))) {
+ return nullptr;
+ }
+ return h;
+}
+
+nsresult KeyedHistogram::Add(const nsCString& key, uint32_t sample,
+ ProcessID aProcessType) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Only add to keyed histograms in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ bool canRecordDataset =
+ CanRecordDataset(mHistogramInfo.dataset, internal_CanRecordBase(),
+ internal_CanRecordExtended());
+ // If `histogram` is a non-parent-process histogram, then recording-enabled
+ // has been checked in its owner process.
+ if (!canRecordDataset || (aProcessType == ProcessID::Parent &&
+ !internal_IsRecordingEnabled(mId))) {
+ return NS_OK;
+ }
+
+ // Don't record if expired.
+ if (IsExpired()) {
+ return NS_OK;
+ }
+
+ // Don't record if the current platform is not enabled
+ if (!CanRecordProduct(gHistogramInfos[mId].products)) {
+ return NS_OK;
+ }
+
+ // The internal representation of a base::Histogram's buckets uses `int`.
+ // Clamp large values of `sample` to be INT_MAX so they continue to be treated
+ // as large values (instead of negative ones).
+ if (sample > INT_MAX) {
+ TelemetryScalar::Add(
+ mozilla::Telemetry::ScalarID::TELEMETRY_ACCUMULATE_CLAMPED_VALUES,
+ NS_ConvertASCIItoUTF16(mHistogramInfo.name()), 1);
+ sample = INT_MAX;
+ }
+
+ base::Histogram* histogram;
+ if (mSingleStore != nullptr) {
+ histogram = GetHistogram("main"_ns, key);
+ if (!histogram) {
+ MOZ_ASSERT(false, "Missing histogram in single store.");
+ return NS_ERROR_FAILURE;
+ }
+
+ histogram->Add(sample);
+ } else {
+ for (uint32_t i = 0; i < mHistogramInfo.store_count; i++) {
+ auto store = nsDependentCString(
+ &gHistogramStringTable
+ [gHistogramStoresTable[mHistogramInfo.store_index + i]]);
+ base::Histogram* histogram = GetHistogram(store, key);
+ MOZ_ASSERT(histogram);
+ if (histogram) {
+ histogram->Add(sample);
+ } else {
+ return NS_ERROR_FAILURE;
+ }
+ }
+ }
+
+ return NS_OK;
+}
+
+void KeyedHistogram::Clear(const nsACString& aStore) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Only clear keyed histograms in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return;
+ }
+
+ if (IsExpired()) {
+ return;
+ }
+
+ if (mSingleStore) {
+ if (aStore.EqualsASCII("main")) {
+ mSingleStore->Clear();
+ }
+ return;
+ }
+
+ KeyedHistogramMapType* histogramMap;
+ bool found = mStorage.Get(aStore, &histogramMap);
+ if (!found) {
+ return;
+ }
+
+ histogramMap->Clear();
+}
+
+bool KeyedHistogram::IsEmpty(const nsACString& aStore) const {
+ if (mSingleStore != nullptr) {
+ if (aStore.EqualsASCII("main")) {
+ return mSingleStore->IsEmpty();
+ }
+
+ return true;
+ }
+
+ KeyedHistogramMapType* histogramMap;
+ bool found = mStorage.Get(aStore, &histogramMap);
+ if (!found) {
+ return true;
+ }
+ return histogramMap->IsEmpty();
+}
+
+size_t KeyedHistogram::SizeOfIncludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) {
+ size_t n = 0;
+ n += aMallocSizeOf(this);
+ /*
+ * In theory mStorage.SizeOfExcludingThis should included the data part of the
+ * map, but the numbers seemed low, so we are only taking the shallow size and
+ * do the iteration here.
+ */
+ n += mStorage.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ for (auto iter = mStorage.Iter(); !iter.Done(); iter.Next()) {
+ auto& h = iter.Data();
+ n += h->SizeOfIncludingThis(aMallocSizeOf);
+ }
+ if (mSingleStore != nullptr) {
+ // base::Histogram doesn't have SizeOfExcludingThis, so we are overcounting
+ // the pointer here.
+ n += mSingleStore->SizeOfExcludingThis(aMallocSizeOf);
+ }
+ return n;
+}
+
+nsresult KeyedHistogram::GetKeys(const StaticMutexAutoLock& aLock,
+ const nsCString& store,
+ nsTArray<nsCString>& aKeys) {
+ KeyedHistogramMapType* histogramMap;
+ if (mSingleStore != nullptr) {
+ histogramMap = mSingleStore;
+ } else {
+ bool found = mStorage.Get(store, &histogramMap);
+ if (!found) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ if (!aKeys.SetCapacity(histogramMap->Count(), mozilla::fallible)) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+
+ for (auto iter = histogramMap->Iter(); !iter.Done(); iter.Next()) {
+ if (!aKeys.AppendElement(iter.Key(), mozilla::fallible)) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult KeyedHistogram::GetJSSnapshot(JSContext* cx, JS::Handle<JSObject*> obj,
+ const nsACString& aStore,
+ bool clearSubsession) {
+ // Get a snapshot of the data.
+ KeyedHistogramSnapshotData dataSnapshot;
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ MOZ_ASSERT(internal_IsHistogramEnumId(mId));
+
+ // Take a snapshot of the data here, protected by the lock, and then,
+ // outside of the lock protection, mirror it to a JS structure.
+ nsresult rv = GetSnapshot(locker, aStore, dataSnapshot, clearSubsession);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ }
+
+ // Now that we have a copy of the data, mirror it to JS.
+ return internal_ReflectKeyedHistogram(dataSnapshot, gHistogramInfos[mId], cx,
+ obj);
+}
+
+/**
+ * Return a histogram snapshot for the named store.
+ *
+ * If getting the snapshot succeeds, NS_OK is returned and `aSnapshot` contains
+ * the snapshot data. If the histogram is not available in the named store,
+ * NS_ERROR_NO_CONTENT is returned. For other errors, NS_ERROR_FAILURE is
+ * returned.
+ */
+nsresult KeyedHistogram::GetSnapshot(const StaticMutexAutoLock& aLock,
+ const nsACString& aStore,
+ KeyedHistogramSnapshotData& aSnapshot,
+ bool aClearSubsession) {
+ KeyedHistogramMapType* histogramMap;
+ if (mSingleStore != nullptr) {
+ if (!aStore.EqualsASCII("main")) {
+ return NS_ERROR_NO_CONTENT;
+ }
+
+ histogramMap = mSingleStore;
+ } else {
+ bool found = mStorage.Get(aStore, &histogramMap);
+ if (!found) {
+ // Nothing in the main store is fine, it's just handled as empty
+ return NS_ERROR_NO_CONTENT;
+ }
+ }
+
+ // Snapshot every key.
+ for (auto iter = histogramMap->ConstIter(); !iter.Done(); iter.Next()) {
+ base::Histogram* keyData = iter.UserData();
+ if (!keyData) {
+ return NS_ERROR_FAILURE;
+ }
+
+ HistogramSnapshotData keySnapshot;
+ if (NS_FAILED(
+ internal_GetHistogramAndSamples(aLock, keyData, keySnapshot))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Append to the final snapshot.
+ aSnapshot.Put(iter.Key(), std::move(keySnapshot));
+ }
+
+ if (aClearSubsession) {
+ Clear(aStore);
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Helper function to get a snapshot of the keyed histograms.
+ *
+ * @param {aLock} the lock proof.
+ * @param {aDataset} the dataset for which the snapshot is being requested.
+ * @param {aClearSubsession} whether or not to clear the data after
+ * taking the snapshot.
+ * @param {aIncludeGPU} whether or not to include data for the GPU.
+ * @param {aOutSnapshot} the container in which the snapshot data will be
+ * stored.
+ * @return {nsresult} NS_OK if the snapshot was successfully taken or
+ * NS_ERROR_OUT_OF_MEMORY if it failed to allocate memory.
+ */
+nsresult internal_GetKeyedHistogramsSnapshot(
+ const StaticMutexAutoLock& aLock, const nsACString& aStore,
+ unsigned int aDataset, bool aClearSubsession, bool aIncludeGPU,
+ bool aFilterTest, KeyedHistogramProcessSnapshotsArray& aOutSnapshot) {
+ if (!aOutSnapshot.resize(static_cast<uint32_t>(ProcessID::Count))) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+
+ for (uint32_t process = 0; process < static_cast<uint32_t>(ProcessID::Count);
+ ++process) {
+ KeyedHistogramSnapshotsArray& hArray = aOutSnapshot[process];
+
+ for (size_t i = 0; i < HistogramCount; ++i) {
+ HistogramID id = HistogramID(i);
+ const HistogramInfo& info = gHistogramInfos[id];
+ if (!info.keyed) {
+ continue;
+ }
+
+ if (!CanRecordInProcess(info.record_in_processes, ProcessID(process)) ||
+ ((ProcessID(process) == ProcessID::Gpu) && !aIncludeGPU)) {
+ continue;
+ }
+
+ if (!IsInDataset(info.dataset, aDataset)) {
+ continue;
+ }
+
+ KeyedHistogram* keyed =
+ internal_GetKeyedHistogramById(id, ProcessID(process),
+ /* instantiate = */ false);
+ if (!keyed || keyed->IsEmpty(aStore) || keyed->IsExpired()) {
+ continue;
+ }
+
+ const char* name = info.name();
+ if (aFilterTest && strncmp(TEST_HISTOGRAM_PREFIX, name,
+ strlen(TEST_HISTOGRAM_PREFIX)) == 0) {
+ if (aClearSubsession) {
+ keyed->Clear(aStore);
+ }
+ continue;
+ }
+
+ // Take a snapshot of the keyed histogram data!
+ KeyedHistogramSnapshotData snapshot;
+ if (!NS_SUCCEEDED(
+ keyed->GetSnapshot(aLock, aStore, snapshot, aClearSubsession))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!hArray.emplaceBack(
+ KeyedHistogramSnapshotInfo{std::move(snapshot), id})) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+ }
+ }
+ return NS_OK;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: thread-unsafe helpers for the external interface
+
+namespace {
+
+bool internal_RemoteAccumulate(const StaticMutexAutoLock& aLock,
+ HistogramID aId, uint32_t aSample) {
+ if (XRE_IsParentProcess()) {
+ return false;
+ }
+
+ if (!internal_IsRecordingEnabled(aId)) {
+ return true;
+ }
+
+ TelemetryIPCAccumulator::AccumulateChildHistogram(aId, aSample);
+ return true;
+}
+
+bool internal_RemoteAccumulate(const StaticMutexAutoLock& aLock,
+ HistogramID aId, const nsCString& aKey,
+ uint32_t aSample) {
+ if (XRE_IsParentProcess()) {
+ return false;
+ }
+
+ if (!internal_IsRecordingEnabled(aId)) {
+ return true;
+ }
+
+ TelemetryIPCAccumulator::AccumulateChildKeyedHistogram(aId, aKey, aSample);
+ return true;
+}
+
+void internal_Accumulate(const StaticMutexAutoLock& aLock, HistogramID aId,
+ uint32_t aSample) {
+ if (!internal_CanRecordBase() ||
+ internal_RemoteAccumulate(aLock, aId, aSample)) {
+ return;
+ }
+
+ Histogram* w = internal_GetHistogramById(aLock, aId, ProcessID::Parent);
+ MOZ_ASSERT(w);
+ internal_HistogramAdd(aLock, *w, aId, aSample, ProcessID::Parent);
+}
+
+void internal_Accumulate(const StaticMutexAutoLock& aLock, HistogramID aId,
+ const nsCString& aKey, uint32_t aSample) {
+ if (!gInitDone || !internal_CanRecordBase() ||
+ internal_RemoteAccumulate(aLock, aId, aKey, aSample)) {
+ return;
+ }
+
+ KeyedHistogram* keyed =
+ internal_GetKeyedHistogramById(aId, ProcessID::Parent);
+ MOZ_ASSERT(keyed);
+ keyed->Add(aKey, aSample, ProcessID::Parent);
+}
+
+void internal_AccumulateChild(const StaticMutexAutoLock& aLock,
+ ProcessID aProcessType, HistogramID aId,
+ uint32_t aSample) {
+ if (!internal_CanRecordBase()) {
+ return;
+ }
+
+ Histogram* w = internal_GetHistogramById(aLock, aId, aProcessType);
+ if (w == nullptr) {
+ NS_WARNING("Failed GetHistogramById for CHILD");
+ } else {
+ internal_HistogramAdd(aLock, *w, aId, aSample, aProcessType);
+ }
+}
+
+void internal_AccumulateChildKeyed(const StaticMutexAutoLock& aLock,
+ ProcessID aProcessType, HistogramID aId,
+ const nsCString& aKey, uint32_t aSample) {
+ if (!gInitDone || !internal_CanRecordBase()) {
+ return;
+ }
+
+ KeyedHistogram* keyed = internal_GetKeyedHistogramById(aId, aProcessType);
+ MOZ_ASSERT(keyed);
+ keyed->Add(aKey, aSample, aProcessType);
+}
+
+void internal_ClearHistogram(const StaticMutexAutoLock& aLock, HistogramID id,
+ const nsACString& aStore) {
+ MOZ_ASSERT(XRE_IsParentProcess());
+ if (!XRE_IsParentProcess()) {
+ return;
+ }
+
+ // Handle keyed histograms.
+ if (gHistogramInfos[id].keyed) {
+ for (uint32_t process = 0;
+ process < static_cast<uint32_t>(ProcessID::Count); ++process) {
+ KeyedHistogram* kh = internal_GetKeyedHistogramById(
+ id, static_cast<ProcessID>(process), /* instantiate = */ false);
+ if (kh) {
+ kh->Clear(aStore);
+ }
+ }
+ } else {
+ // Reset the histograms instances for all processes.
+ for (uint32_t process = 0;
+ process < static_cast<uint32_t>(ProcessID::Count); ++process) {
+ Histogram* h =
+ internal_GetHistogramById(aLock, id, static_cast<ProcessID>(process),
+ /* instantiate = */ false);
+ if (h) {
+ h->Clear(aStore);
+ }
+ }
+ }
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: JSHistogram_* functions
+
+// NOTE: the functions in this section:
+//
+// internal_JSHistogram_Add
+// internal_JSHistogram_Name
+// internal_JSHistogram_Snapshot
+// internal_JSHistogram_Clear
+// internal_WrapAndReturnHistogram
+//
+// all run without protection from |gTelemetryHistogramMutex|. If they
+// held |gTelemetryHistogramMutex|, there would be the possibility of
+// deadlock because the JS_ calls that they make may call back into the
+// TelemetryHistogram interface, hence trying to re-acquire the mutex.
+//
+// This means that these functions potentially race against threads, but
+// that seems preferable to risking deadlock.
+
+namespace {
+
+void internal_JSHistogram_finalize(JSFreeOp*, JSObject*);
+
+static const JSClassOps sJSHistogramClassOps = {nullptr, /* addProperty */
+ nullptr, /* delProperty */
+ nullptr, /* enumerate */
+ nullptr, /* newEnumerate */
+ nullptr, /* resolve */
+ nullptr, /* mayResolve */
+ internal_JSHistogram_finalize};
+
+static const JSClass sJSHistogramClass = {
+ "JSHistogram", /* name */
+ JSCLASS_HAS_PRIVATE | JSCLASS_FOREGROUND_FINALIZE, /* flags */
+ &sJSHistogramClassOps};
+
+struct JSHistogramData {
+ HistogramID histogramId;
+};
+
+bool internal_JSHistogram_CoerceValue(JSContext* aCx,
+ JS::Handle<JS::Value> aElement,
+ HistogramID aId, uint32_t aHistogramType,
+ uint32_t& aValue) {
+ if (aElement.isString()) {
+ // Strings only allowed for categorical histograms
+ if (aHistogramType != nsITelemetry::HISTOGRAM_CATEGORICAL) {
+ LogToBrowserConsole(
+ nsIScriptError::errorFlag,
+ nsLiteralString(
+ u"String argument only allowed for categorical histogram"));
+ return false;
+ }
+
+ // Label is given by the string argument
+ nsAutoJSString label;
+ if (!label.init(aCx, aElement)) {
+ LogToBrowserConsole(nsIScriptError::errorFlag,
+ u"Invalid string parameter"_ns);
+ return false;
+ }
+
+ // Get the label id for accumulation
+ nsresult rv = gHistogramInfos[aId].label_id(
+ NS_ConvertUTF16toUTF8(label).get(), &aValue);
+ if (NS_FAILED(rv)) {
+ nsPrintfCString msg("'%s' is an invalid string label",
+ NS_ConvertUTF16toUTF8(label).get());
+ LogToBrowserConsole(nsIScriptError::errorFlag,
+ NS_ConvertUTF8toUTF16(msg));
+ return false;
+ }
+ } else if (!(aElement.isNumber() || aElement.isBoolean())) {
+ LogToBrowserConsole(nsIScriptError::errorFlag, u"Argument not a number"_ns);
+ return false;
+ } else if (aElement.isNumber() && aElement.toNumber() > UINT32_MAX) {
+ // Clamp large numerical arguments to aValue's acceptable values.
+ // JS::ToUint32 will take aElement modulo 2^32 before returning it, which
+ // may result in a smaller final value.
+ aValue = UINT32_MAX;
+#ifdef DEBUG
+ LogToBrowserConsole(nsIScriptError::errorFlag,
+ u"Clamped large numeric value"_ns);
+#endif
+ } else if (!JS::ToUint32(aCx, aElement, &aValue)) {
+ LogToBrowserConsole(nsIScriptError::errorFlag,
+ u"Failed to convert element to UInt32"_ns);
+ return false;
+ }
+
+ // If we're here then all type checks have passed and aValue contains the
+ // coerced integer
+ return true;
+}
+
+bool internal_JSHistogram_GetValueArray(JSContext* aCx, JS::CallArgs& args,
+ uint32_t aHistogramType,
+ HistogramID aId, bool isKeyed,
+ nsTArray<uint32_t>& aArray) {
+ // This function populates aArray with the values extracted from args. Handles
+ // keyed and non-keyed histograms, and single and array of values. Also
+ // performs sanity checks on the arguments. Returns true upon successful
+ // population, false otherwise.
+
+ uint32_t firstArgIndex = 0;
+ if (isKeyed) {
+ firstArgIndex = 1;
+ }
+
+ // Special case of no argument (or only key) and count histogram
+ if (args.length() == firstArgIndex) {
+ if (!(aHistogramType == nsITelemetry::HISTOGRAM_COUNT)) {
+ LogToBrowserConsole(
+ nsIScriptError::errorFlag,
+ nsLiteralString(
+ u"Need at least one argument for non count type histogram"));
+ return false;
+ }
+
+ aArray.AppendElement(1);
+ return true;
+ }
+
+ if (args[firstArgIndex].isObject() && !args[firstArgIndex].isString()) {
+ JS::Rooted<JSObject*> arrayObj(aCx, &args[firstArgIndex].toObject());
+
+ bool isArray = false;
+ JS::IsArrayObject(aCx, arrayObj, &isArray);
+
+ if (!isArray) {
+ LogToBrowserConsole(
+ nsIScriptError::errorFlag,
+ nsLiteralString(
+ u"The argument to accumulate can't be a non-array object"));
+ return false;
+ }
+
+ uint32_t arrayLength = 0;
+ if (!JS::GetArrayLength(aCx, arrayObj, &arrayLength)) {
+ LogToBrowserConsole(nsIScriptError::errorFlag,
+ u"Failed while trying to get array length"_ns);
+ return false;
+ }
+
+ for (uint32_t arrayIdx = 0; arrayIdx < arrayLength; arrayIdx++) {
+ JS::Rooted<JS::Value> element(aCx);
+
+ if (!JS_GetElement(aCx, arrayObj, arrayIdx, &element)) {
+ nsPrintfCString msg("Failed while trying to get element at index %d",
+ arrayIdx);
+ LogToBrowserConsole(nsIScriptError::errorFlag,
+ NS_ConvertUTF8toUTF16(msg));
+ return false;
+ }
+
+ uint32_t value = 0;
+ if (!internal_JSHistogram_CoerceValue(aCx, element, aId, aHistogramType,
+ value)) {
+ nsPrintfCString msg("Element at index %d failed type checks", arrayIdx);
+ LogToBrowserConsole(nsIScriptError::errorFlag,
+ NS_ConvertUTF8toUTF16(msg));
+ return false;
+ }
+ aArray.AppendElement(value);
+ }
+
+ return true;
+ }
+
+ uint32_t value = 0;
+ if (!internal_JSHistogram_CoerceValue(aCx, args[firstArgIndex], aId,
+ aHistogramType, value)) {
+ return false;
+ }
+ aArray.AppendElement(value);
+ return true;
+}
+
+bool internal_JSHistogram_Add(JSContext* cx, unsigned argc, JS::Value* vp) {
+ JS::CallArgs args = CallArgsFromVp(argc, vp);
+
+ if (!args.thisv().isObject() ||
+ JS::GetClass(&args.thisv().toObject()) != &sJSHistogramClass) {
+ JS_ReportErrorASCII(cx, "Wrong JS class, expected JSHistogram class");
+ return false;
+ }
+
+ JSObject* obj = &args.thisv().toObject();
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+ HistogramID id = data->histogramId;
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+ uint32_t type = gHistogramInfos[id].histogramType;
+
+ // This function should always return |undefined| and never fail but
+ // rather report failures using the console.
+ args.rval().setUndefined();
+
+ nsTArray<uint32_t> values;
+ if (!internal_JSHistogram_GetValueArray(cx, args, type, id, false, values)) {
+ // Either GetValueArray or CoerceValue utility function will have printed a
+ // meaningful error message, so we simply return true
+ return true;
+ }
+
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ for (uint32_t aValue : values) {
+ internal_Accumulate(locker, id, aValue);
+ }
+ }
+ return true;
+}
+
+bool internal_JSHistogram_Name(JSContext* cx, unsigned argc, JS::Value* vp) {
+ JS::CallArgs args = CallArgsFromVp(argc, vp);
+
+ if (!args.thisv().isObject() ||
+ JS::GetClass(&args.thisv().toObject()) != &sJSHistogramClass) {
+ JS_ReportErrorASCII(cx, "Wrong JS class, expected JSHistogram class");
+ return false;
+ }
+
+ JSObject* obj = &args.thisv().toObject();
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+ HistogramID id = data->histogramId;
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+ const char* name = gHistogramInfos[id].name();
+
+ auto cname = NS_ConvertASCIItoUTF16(name);
+ args.rval().setString(ToJSString(cx, cname));
+
+ return true;
+}
+
+/**
+ * Extract the store name from JavaScript function arguments.
+ * The first and only argument needs to be an object with a "store" property.
+ * If no arguments are given it defaults to "main".
+ */
+nsresult internal_JS_StoreFromObjectArgument(JSContext* cx,
+ const JS::CallArgs& args,
+ nsAutoString& aStoreName) {
+ if (args.length() == 0) {
+ aStoreName.AssignLiteral("main");
+ } else if (args.length() == 1) {
+ if (!args[0].isObject()) {
+ JS_ReportErrorASCII(cx, "Expected object argument.");
+ return NS_ERROR_FAILURE;
+ }
+
+ JS::RootedValue storeValue(cx);
+ JS::RootedObject argsObject(cx, &args[0].toObject());
+ if (!JS_GetProperty(cx, argsObject, "store", &storeValue)) {
+ JS_ReportErrorASCII(cx,
+ "Expected object argument to have property 'store'.");
+ return NS_ERROR_FAILURE;
+ }
+
+ nsAutoJSString store;
+ if (!storeValue.isString() || !store.init(cx, storeValue)) {
+ JS_ReportErrorASCII(
+ cx, "Expected object argument's 'store' property to be a string.");
+ return NS_ERROR_FAILURE;
+ }
+
+ aStoreName.Assign(store);
+ } else {
+ JS_ReportErrorASCII(cx, "Expected at most one argument.");
+ return NS_ERROR_FAILURE;
+ }
+
+ return NS_OK;
+}
+
+bool internal_JSHistogram_Snapshot(JSContext* cx, unsigned argc,
+ JS::Value* vp) {
+ JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
+
+ if (!XRE_IsParentProcess()) {
+ JS_ReportErrorASCII(
+ cx, "Histograms can only be snapshotted in the parent process");
+ return false;
+ }
+
+ if (!args.thisv().isObject() ||
+ JS::GetClass(&args.thisv().toObject()) != &sJSHistogramClass) {
+ JS_ReportErrorASCII(cx, "Wrong JS class, expected JSHistogram class");
+ return false;
+ }
+
+ JSObject* obj = &args.thisv().toObject();
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+ HistogramID id = data->histogramId;
+
+ nsAutoString storeName;
+ nsresult rv = internal_JS_StoreFromObjectArgument(cx, args, storeName);
+ if (NS_FAILED(rv)) {
+ return false;
+ }
+
+ HistogramSnapshotData dataSnapshot;
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+
+ // This is not good standard behavior given that we have histogram instances
+ // covering multiple processes.
+ // However, changing this requires some broader changes to callers.
+ Histogram* w = internal_GetHistogramById(locker, id, ProcessID::Parent);
+ base::Histogram* h = nullptr;
+ if (!w->GetHistogram(NS_ConvertUTF16toUTF8(storeName), &h)) {
+ // When it's not in the named store, let's skip the snapshot completely,
+ // but don't fail
+ args.rval().setUndefined();
+ return true;
+ }
+ // Take a snapshot of the data here, protected by the lock, and then,
+ // outside of the lock protection, mirror it to a JS structure
+ if (NS_FAILED(internal_GetHistogramAndSamples(locker, h, dataSnapshot))) {
+ return false;
+ }
+ }
+
+ JS::Rooted<JSObject*> snapshot(cx, JS_NewPlainObject(cx));
+ if (!snapshot) {
+ return false;
+ }
+
+ if (NS_FAILED(internal_ReflectHistogramAndSamples(
+ cx, snapshot, gHistogramInfos[id], dataSnapshot))) {
+ return false;
+ }
+
+ args.rval().setObject(*snapshot);
+ return true;
+}
+
+bool internal_JSHistogram_Clear(JSContext* cx, unsigned argc, JS::Value* vp) {
+ if (!XRE_IsParentProcess()) {
+ JS_ReportErrorASCII(cx,
+ "Histograms can only be cleared in the parent process");
+ return false;
+ }
+
+ JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
+
+ if (!args.thisv().isObject() ||
+ JS::GetClass(&args.thisv().toObject()) != &sJSHistogramClass) {
+ JS_ReportErrorASCII(cx, "Wrong JS class, expected JSHistogram class");
+ return false;
+ }
+
+ JSObject* obj = &args.thisv().toObject();
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+
+ nsAutoString storeName;
+ nsresult rv = internal_JS_StoreFromObjectArgument(cx, args, storeName);
+ if (NS_FAILED(rv)) {
+ return false;
+ }
+
+ // This function should always return |undefined| and never fail but
+ // rather report failures using the console.
+ args.rval().setUndefined();
+
+ HistogramID id = data->histogramId;
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+ internal_ClearHistogram(locker, id, NS_ConvertUTF16toUTF8(storeName));
+ }
+
+ return true;
+}
+
+// NOTE: Runs without protection from |gTelemetryHistogramMutex|.
+// See comment at the top of this section.
+nsresult internal_WrapAndReturnHistogram(HistogramID id, JSContext* cx,
+ JS::MutableHandle<JS::Value> ret) {
+ JS::Rooted<JSObject*> obj(cx, JS_NewObject(cx, &sJSHistogramClass));
+ if (!obj) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // The 3 functions that are wrapped up here are eventually called
+ // by the same thread that runs this function.
+ if (!(JS_DefineFunction(cx, obj, "add", internal_JSHistogram_Add, 1, 0) &&
+ JS_DefineFunction(cx, obj, "name", internal_JSHistogram_Name, 1, 0) &&
+ JS_DefineFunction(cx, obj, "snapshot", internal_JSHistogram_Snapshot, 1,
+ 0) &&
+ JS_DefineFunction(cx, obj, "clear", internal_JSHistogram_Clear, 1,
+ 0))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ JSHistogramData* data = new JSHistogramData{id};
+ JS::SetPrivate(obj, data);
+ ret.setObject(*obj);
+
+ return NS_OK;
+}
+
+void internal_JSHistogram_finalize(JSFreeOp*, JSObject* obj) {
+ if (!obj || JS::GetClass(obj) != &sJSHistogramClass) {
+ MOZ_ASSERT_UNREACHABLE("Should have the right JS class.");
+ return;
+ }
+
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+ delete data;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: JSKeyedHistogram_* functions
+
+// NOTE: the functions in this section:
+//
+// internal_JSKeyedHistogram_Add
+// internal_JSKeyedHistogram_Name
+// internal_JSKeyedHistogram_Keys
+// internal_JSKeyedHistogram_Snapshot
+// internal_JSKeyedHistogram_Clear
+// internal_WrapAndReturnKeyedHistogram
+//
+// Same comments as above, at the JSHistogram_* section, regarding
+// deadlock avoidance, apply.
+
+namespace {
+
+void internal_JSKeyedHistogram_finalize(JSFreeOp*, JSObject*);
+
+static const JSClassOps sJSKeyedHistogramClassOps = {
+ nullptr, /* addProperty */
+ nullptr, /* delProperty */
+ nullptr, /* enumerate */
+ nullptr, /* newEnumerate */
+ nullptr, /* resolve */
+ nullptr, /* mayResolve */
+ internal_JSKeyedHistogram_finalize};
+
+static const JSClass sJSKeyedHistogramClass = {
+ "JSKeyedHistogram", /* name */
+ JSCLASS_HAS_PRIVATE | JSCLASS_FOREGROUND_FINALIZE, /* flags */
+ &sJSKeyedHistogramClassOps};
+
+bool internal_JSKeyedHistogram_Snapshot(JSContext* cx, unsigned argc,
+ JS::Value* vp) {
+ if (!XRE_IsParentProcess()) {
+ JS_ReportErrorASCII(
+ cx, "Keyed histograms can only be snapshotted in the parent process");
+ return false;
+ }
+
+ JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
+
+ if (!args.thisv().isObject() ||
+ JS::GetClass(&args.thisv().toObject()) != &sJSKeyedHistogramClass) {
+ JS_ReportErrorASCII(cx, "Wrong JS class, expected JSKeyedHistogram class");
+ return false;
+ }
+
+ JSObject* obj = &args.thisv().toObject();
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+ HistogramID id = data->histogramId;
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+
+ // This function should always return |undefined| and never fail but
+ // rather report failures using the console.
+ args.rval().setUndefined();
+
+ // This is not good standard behavior given that we have histogram instances
+ // covering multiple processes.
+ // However, changing this requires some broader changes to callers.
+ KeyedHistogram* keyed = internal_GetKeyedHistogramById(
+ id, ProcessID::Parent, /* instantiate = */ true);
+ if (!keyed) {
+ JS_ReportErrorASCII(cx, "Failed to look up keyed histogram");
+ return false;
+ }
+
+ nsAutoString storeName;
+ nsresult rv;
+ rv = internal_JS_StoreFromObjectArgument(cx, args, storeName);
+ if (NS_FAILED(rv)) {
+ return false;
+ }
+
+ JS::RootedObject snapshot(cx, JS_NewPlainObject(cx));
+ if (!snapshot) {
+ JS_ReportErrorASCII(cx, "Failed to create object");
+ return false;
+ }
+
+ rv = keyed->GetJSSnapshot(cx, snapshot, NS_ConvertUTF16toUTF8(storeName),
+ false);
+
+ // If the store is not available, we return nothing and don't fail
+ if (rv == NS_ERROR_NO_CONTENT) {
+ args.rval().setUndefined();
+ return true;
+ }
+
+ if (!NS_SUCCEEDED(rv)) {
+ JS_ReportErrorASCII(cx, "Failed to reflect keyed histograms");
+ return false;
+ }
+
+ args.rval().setObject(*snapshot);
+ return true;
+}
+
+bool internal_JSKeyedHistogram_Add(JSContext* cx, unsigned argc,
+ JS::Value* vp) {
+ JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
+
+ if (!args.thisv().isObject() ||
+ JS::GetClass(&args.thisv().toObject()) != &sJSKeyedHistogramClass) {
+ JS_ReportErrorASCII(cx, "Wrong JS class, expected JSKeyedHistogram class");
+ return false;
+ }
+
+ JSObject* obj = &args.thisv().toObject();
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+ HistogramID id = data->histogramId;
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+
+ // This function should always return |undefined| and never fail but
+ // rather report failures using the console.
+ args.rval().setUndefined();
+ if (args.length() < 1) {
+ LogToBrowserConsole(nsIScriptError::errorFlag, u"Expected one argument"_ns);
+ return true;
+ }
+
+ nsAutoJSString key;
+ if (!args[0].isString() || !key.init(cx, args[0])) {
+ LogToBrowserConsole(nsIScriptError::errorFlag, u"Not a string"_ns);
+ return true;
+ }
+
+ // Check if we're allowed to record in the provided key, for this histogram.
+ if (!gHistogramInfos[id].allows_key(NS_ConvertUTF16toUTF8(key))) {
+ nsPrintfCString msg("%s - key '%s' not allowed for this keyed histogram",
+ gHistogramInfos[id].name(),
+ NS_ConvertUTF16toUTF8(key).get());
+ LogToBrowserConsole(nsIScriptError::errorFlag, NS_ConvertUTF8toUTF16(msg));
+ TelemetryScalar::Add(mozilla::Telemetry::ScalarID::
+ TELEMETRY_ACCUMULATE_UNKNOWN_HISTOGRAM_KEYS,
+ NS_ConvertASCIItoUTF16(gHistogramInfos[id].name()), 1);
+ return true;
+ }
+
+ const uint32_t type = gHistogramInfos[id].histogramType;
+
+ nsTArray<uint32_t> values;
+ if (!internal_JSHistogram_GetValueArray(cx, args, type, id, true, values)) {
+ // Either GetValueArray or CoerceValue utility function will have printed a
+ // meaningful error message so we simple return true
+ return true;
+ }
+
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ for (uint32_t aValue : values) {
+ internal_Accumulate(locker, id, NS_ConvertUTF16toUTF8(key), aValue);
+ }
+ }
+ return true;
+}
+
+bool internal_JSKeyedHistogram_Name(JSContext* cx, unsigned argc,
+ JS::Value* vp) {
+ JS::CallArgs args = CallArgsFromVp(argc, vp);
+
+ if (!args.thisv().isObject() ||
+ JS::GetClass(&args.thisv().toObject()) != &sJSKeyedHistogramClass) {
+ JS_ReportErrorASCII(cx, "Wrong JS class, expected JSKeyedHistogram class");
+ return false;
+ }
+
+ JSObject* obj = &args.thisv().toObject();
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+ HistogramID id = data->histogramId;
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+ const char* name = gHistogramInfos[id].name();
+
+ auto cname = NS_ConvertASCIItoUTF16(name);
+ args.rval().setString(ToJSString(cx, cname));
+
+ return true;
+}
+
+bool internal_JSKeyedHistogram_Keys(JSContext* cx, unsigned argc,
+ JS::Value* vp) {
+ JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
+
+ if (!args.thisv().isObject() ||
+ JS::GetClass(&args.thisv().toObject()) != &sJSKeyedHistogramClass) {
+ JS_ReportErrorASCII(cx, "Wrong JS class, expected JSKeyedHistogram class");
+ return false;
+ }
+
+ JSObject* obj = &args.thisv().toObject();
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+ HistogramID id = data->histogramId;
+
+ nsAutoString storeName;
+ nsresult rv = internal_JS_StoreFromObjectArgument(cx, args, storeName);
+ if (NS_FAILED(rv)) {
+ return false;
+ }
+
+ nsTArray<nsCString> keys;
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+
+ // This is not good standard behavior given that we have histogram instances
+ // covering multiple processes.
+ // However, changing this requires some broader changes to callers.
+ KeyedHistogram* keyed =
+ internal_GetKeyedHistogramById(id, ProcessID::Parent);
+
+ MOZ_ASSERT(keyed);
+ if (!keyed) {
+ return false;
+ }
+
+ if (NS_FAILED(
+ keyed->GetKeys(locker, NS_ConvertUTF16toUTF8(storeName), keys))) {
+ return false;
+ }
+ }
+
+ // Convert keys from nsTArray<nsCString> to JS array.
+ JS::RootedVector<JS::Value> autoKeys(cx);
+ if (!autoKeys.reserve(keys.Length())) {
+ return false;
+ }
+
+ for (const auto& key : keys) {
+ JS::RootedValue jsKey(cx);
+ jsKey.setString(ToJSString(cx, key));
+ if (!autoKeys.append(jsKey)) {
+ return false;
+ }
+ }
+
+ JS::RootedObject jsKeys(cx, JS::NewArrayObject(cx, autoKeys));
+ if (!jsKeys) {
+ return false;
+ }
+
+ args.rval().setObject(*jsKeys);
+ return true;
+}
+
+bool internal_JSKeyedHistogram_Clear(JSContext* cx, unsigned argc,
+ JS::Value* vp) {
+ if (!XRE_IsParentProcess()) {
+ JS_ReportErrorASCII(
+ cx, "Keyed histograms can only be cleared in the parent process");
+ return false;
+ }
+
+ JS::CallArgs args = JS::CallArgsFromVp(argc, vp);
+
+ if (!args.thisv().isObject() ||
+ JS::GetClass(&args.thisv().toObject()) != &sJSKeyedHistogramClass) {
+ JS_ReportErrorASCII(cx, "Wrong JS class, expected JSKeyedHistogram class");
+ return false;
+ }
+
+ JSObject* obj = &args.thisv().toObject();
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+ HistogramID id = data->histogramId;
+
+ // This function should always return |undefined| and never fail but
+ // rather report failures using the console.
+ args.rval().setUndefined();
+
+ nsAutoString storeName;
+ nsresult rv = internal_JS_StoreFromObjectArgument(cx, args, storeName);
+ if (NS_FAILED(rv)) {
+ return false;
+ }
+
+ KeyedHistogram* keyed = nullptr;
+ {
+ MOZ_ASSERT(internal_IsHistogramEnumId(id));
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+
+ // This is not good standard behavior given that we have histogram instances
+ // covering multiple processes.
+ // However, changing this requires some broader changes to callers.
+ keyed = internal_GetKeyedHistogramById(id, ProcessID::Parent,
+ /* instantiate = */ false);
+
+ if (!keyed) {
+ return true;
+ }
+
+ keyed->Clear(NS_ConvertUTF16toUTF8(storeName));
+ }
+
+ return true;
+}
+
+// NOTE: Runs without protection from |gTelemetryHistogramMutex|.
+// See comment at the top of this section.
+nsresult internal_WrapAndReturnKeyedHistogram(
+ HistogramID id, JSContext* cx, JS::MutableHandle<JS::Value> ret) {
+ JS::Rooted<JSObject*> obj(cx, JS_NewObject(cx, &sJSKeyedHistogramClass));
+ if (!obj) return NS_ERROR_FAILURE;
+ // The 6 functions that are wrapped up here are eventually called
+ // by the same thread that runs this function.
+ if (!(JS_DefineFunction(cx, obj, "add", internal_JSKeyedHistogram_Add, 2,
+ 0) &&
+ JS_DefineFunction(cx, obj, "name", internal_JSKeyedHistogram_Name, 1,
+ 0) &&
+ JS_DefineFunction(cx, obj, "snapshot",
+ internal_JSKeyedHistogram_Snapshot, 1, 0) &&
+ JS_DefineFunction(cx, obj, "keys", internal_JSKeyedHistogram_Keys, 1,
+ 0) &&
+ JS_DefineFunction(cx, obj, "clear", internal_JSKeyedHistogram_Clear, 1,
+ 0))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ JSHistogramData* data = new JSHistogramData{id};
+ JS::SetPrivate(obj, data);
+ ret.setObject(*obj);
+
+ return NS_OK;
+}
+
+void internal_JSKeyedHistogram_finalize(JSFreeOp*, JSObject* obj) {
+ if (!obj || JS::GetClass(obj) != &sJSKeyedHistogramClass) {
+ MOZ_ASSERT_UNREACHABLE("Should have the right JS class.");
+ return;
+ }
+
+ JSHistogramData* data = static_cast<JSHistogramData*>(JS::GetPrivate(obj));
+ MOZ_ASSERT(data);
+ delete data;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// EXTERNALLY VISIBLE FUNCTIONS in namespace TelemetryHistogram::
+
+// All of these functions are actually in namespace TelemetryHistogram::,
+// but the ::TelemetryHistogram prefix is given explicitly. This is
+// because it is critical to see which calls from these functions are
+// to another function in this interface. Mis-identifying "inwards
+// calls" from "calls to another function in this interface" will lead
+// to deadlocking and/or races. See comments at the top of the file
+// for further (important!) details.
+
+void TelemetryHistogram::InitializeGlobalState(bool canRecordBase,
+ bool canRecordExtended) {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ MOZ_ASSERT(!gInitDone,
+ "TelemetryHistogram::InitializeGlobalState "
+ "may only be called once");
+
+ gCanRecordBase = canRecordBase;
+ gCanRecordExtended = canRecordExtended;
+
+ if (XRE_IsParentProcess()) {
+ gHistogramStorage =
+ new Histogram* [HistogramCount * size_t(ProcessID::Count)] {};
+ gKeyedHistogramStorage =
+ new KeyedHistogram* [HistogramCount * size_t(ProcessID::Count)] {};
+ }
+
+ // Some Telemetry histograms depend on the value of C++ constants and hardcode
+ // their values in Histograms.json.
+ // We add static asserts here for those values to match so that future changes
+ // don't go unnoticed.
+ // clang-format off
+ static_assert((uint32_t(JS::GCReason::NUM_TELEMETRY_REASONS) + 1) ==
+ gHistogramInfos[mozilla::Telemetry::GC_MINOR_REASON].bucketCount &&
+ (uint32_t(JS::GCReason::NUM_TELEMETRY_REASONS) + 1) ==
+ gHistogramInfos[mozilla::Telemetry::GC_MINOR_REASON_LONG].bucketCount &&
+ (uint32_t(JS::GCReason::NUM_TELEMETRY_REASONS) + 1) ==
+ gHistogramInfos[mozilla::Telemetry::GC_REASON_2].bucketCount,
+ "NUM_TELEMETRY_REASONS is assumed to be a fixed value in Histograms.json."
+ " If this was an intentional change, update the n_values for the "
+ "following in Histograms.json: GC_MINOR_REASON, GC_MINOR_REASON_LONG, "
+ "GC_REASON_2");
+
+ // clang-format on
+
+ gInitDone = true;
+}
+
+void TelemetryHistogram::DeInitializeGlobalState() {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ gCanRecordBase = false;
+ gCanRecordExtended = false;
+ gInitDone = false;
+
+ // FactoryGet `new`s Histograms for us, but requires us to manually delete.
+ if (XRE_IsParentProcess()) {
+ for (size_t i = 0; i < HistogramCount * size_t(ProcessID::Count); ++i) {
+ if (gKeyedHistogramStorage[i] != gExpiredKeyedHistogram) {
+ delete gKeyedHistogramStorage[i];
+ }
+ if (gHistogramStorage[i] != gExpiredHistogram) {
+ delete gHistogramStorage[i];
+ }
+ }
+ delete[] gHistogramStorage;
+ delete[] gKeyedHistogramStorage;
+ }
+ delete gExpiredHistogram;
+ gExpiredHistogram = nullptr;
+ delete gExpiredKeyedHistogram;
+ gExpiredKeyedHistogram = nullptr;
+}
+
+#ifdef DEBUG
+bool TelemetryHistogram::GlobalStateHasBeenInitialized() {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ return gInitDone;
+}
+#endif
+
+bool TelemetryHistogram::CanRecordBase() {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ return internal_CanRecordBase();
+}
+
+void TelemetryHistogram::SetCanRecordBase(bool b) {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ gCanRecordBase = b;
+}
+
+bool TelemetryHistogram::CanRecordExtended() {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ return internal_CanRecordExtended();
+}
+
+void TelemetryHistogram::SetCanRecordExtended(bool b) {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ gCanRecordExtended = b;
+}
+
+void TelemetryHistogram::InitHistogramRecordingEnabled() {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ auto processType = XRE_GetProcessType();
+ for (size_t i = 0; i < HistogramCount; ++i) {
+ const HistogramInfo& h = gHistogramInfos[i];
+ mozilla::Telemetry::HistogramID id = mozilla::Telemetry::HistogramID(i);
+ bool canRecordInProcess =
+ CanRecordInProcess(h.record_in_processes, processType);
+ internal_SetHistogramRecordingEnabled(locker, id, canRecordInProcess);
+ }
+
+ for (auto recordingInitiallyDisabledID : kRecordingInitiallyDisabledIDs) {
+ internal_SetHistogramRecordingEnabled(locker, recordingInitiallyDisabledID,
+ false);
+ }
+}
+
+void TelemetryHistogram::SetHistogramRecordingEnabled(HistogramID aID,
+ bool aEnabled) {
+ if (NS_WARN_IF(!internal_IsHistogramEnumId(aID))) {
+ MOZ_ASSERT_UNREACHABLE("Histogram usage requires valid ids.");
+ return;
+ }
+
+ const HistogramInfo& h = gHistogramInfos[aID];
+ if (!CanRecordInProcess(h.record_in_processes, XRE_GetProcessType())) {
+ // Don't permit record_in_process-disabled recording to be re-enabled.
+ return;
+ }
+
+ if (!CanRecordProduct(h.products)) {
+ // Don't permit products-disabled recording to be re-enabled.
+ return;
+ }
+
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ internal_SetHistogramRecordingEnabled(locker, aID, aEnabled);
+}
+
+nsresult TelemetryHistogram::SetHistogramRecordingEnabled(
+ const nsACString& name, bool aEnabled) {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ HistogramID id;
+ if (NS_FAILED(internal_GetHistogramIdByName(locker, name, &id))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ const HistogramInfo& hi = gHistogramInfos[id];
+ if (CanRecordInProcess(hi.record_in_processes, XRE_GetProcessType())) {
+ internal_SetHistogramRecordingEnabled(locker, id, aEnabled);
+ }
+ return NS_OK;
+}
+
+void TelemetryHistogram::Accumulate(HistogramID aID, uint32_t aSample) {
+ if (NS_WARN_IF(!internal_IsHistogramEnumId(aID))) {
+ MOZ_ASSERT_UNREACHABLE("Histogram usage requires valid ids.");
+ return;
+ }
+
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ internal_Accumulate(locker, aID, aSample);
+}
+
+void TelemetryHistogram::Accumulate(HistogramID aID,
+ const nsTArray<uint32_t>& aSamples) {
+ if (NS_WARN_IF(!internal_IsHistogramEnumId(aID))) {
+ MOZ_ASSERT_UNREACHABLE("Histogram usage requires valid ids.");
+ return;
+ }
+
+ MOZ_ASSERT(!gHistogramInfos[aID].keyed,
+ "Cannot accumulate into a keyed histogram. No key given.");
+
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ for (uint32_t sample : aSamples) {
+ internal_Accumulate(locker, aID, sample);
+ }
+}
+
+void TelemetryHistogram::Accumulate(HistogramID aID, const nsCString& aKey,
+ uint32_t aSample) {
+ if (NS_WARN_IF(!internal_IsHistogramEnumId(aID))) {
+ MOZ_ASSERT_UNREACHABLE("Histogram usage requires valid ids.");
+ return;
+ }
+
+ // Check if we're allowed to record in the provided key, for this histogram.
+ if (!gHistogramInfos[aID].allows_key(aKey)) {
+ nsPrintfCString msg("%s - key '%s' not allowed for this keyed histogram",
+ gHistogramInfos[aID].name(), aKey.get());
+ LogToBrowserConsole(nsIScriptError::errorFlag, NS_ConvertUTF8toUTF16(msg));
+ TelemetryScalar::Add(mozilla::Telemetry::ScalarID::
+ TELEMETRY_ACCUMULATE_UNKNOWN_HISTOGRAM_KEYS,
+ NS_ConvertASCIItoUTF16(gHistogramInfos[aID].name()),
+ 1);
+ return;
+ }
+
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ internal_Accumulate(locker, aID, aKey, aSample);
+}
+
+void TelemetryHistogram::Accumulate(HistogramID aID, const nsCString& aKey,
+ const nsTArray<uint32_t>& aSamples) {
+ if (NS_WARN_IF(!internal_IsHistogramEnumId(aID))) {
+ MOZ_ASSERT_UNREACHABLE("Histogram usage requires valid ids");
+ return;
+ }
+
+ // Check that this histogram is keyed
+ MOZ_ASSERT(gHistogramInfos[aID].keyed,
+ "Cannot accumulate into a non-keyed histogram using a key.");
+
+ // Check if we're allowed to record in the provided key, for this histogram.
+ if (!gHistogramInfos[aID].allows_key(aKey)) {
+ nsPrintfCString msg("%s - key '%s' not allowed for this keyed histogram",
+ gHistogramInfos[aID].name(), aKey.get());
+ LogToBrowserConsole(nsIScriptError::errorFlag, NS_ConvertUTF8toUTF16(msg));
+ TelemetryScalar::Add(mozilla::Telemetry::ScalarID::
+ TELEMETRY_ACCUMULATE_UNKNOWN_HISTOGRAM_KEYS,
+ NS_ConvertASCIItoUTF16(gHistogramInfos[aID].name()),
+ 1);
+ return;
+ }
+
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ for (uint32_t sample : aSamples) {
+ internal_Accumulate(locker, aID, aKey, sample);
+ }
+}
+
+nsresult TelemetryHistogram::Accumulate(const char* name, uint32_t sample) {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ if (!internal_CanRecordBase()) {
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+ HistogramID id;
+ nsresult rv =
+ internal_GetHistogramIdByName(locker, nsDependentCString(name), &id);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ internal_Accumulate(locker, id, sample);
+ return NS_OK;
+}
+
+nsresult TelemetryHistogram::Accumulate(const char* name, const nsCString& key,
+ uint32_t sample) {
+ bool keyNotAllowed = false;
+
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ if (!internal_CanRecordBase()) {
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+ HistogramID id;
+ nsresult rv =
+ internal_GetHistogramIdByName(locker, nsDependentCString(name), &id);
+ if (NS_SUCCEEDED(rv)) {
+ // Check if we're allowed to record in the provided key, for this
+ // histogram.
+ if (gHistogramInfos[id].allows_key(key)) {
+ internal_Accumulate(locker, id, key, sample);
+ return NS_OK;
+ }
+ // We're holding |gTelemetryHistogramMutex|, so we can't print a message
+ // here.
+ keyNotAllowed = true;
+ }
+ }
+
+ if (keyNotAllowed) {
+ LogToBrowserConsole(nsIScriptError::errorFlag,
+ u"Key not allowed for this keyed histogram"_ns);
+ TelemetryScalar::Add(mozilla::Telemetry::ScalarID::
+ TELEMETRY_ACCUMULATE_UNKNOWN_HISTOGRAM_KEYS,
+ NS_ConvertASCIItoUTF16(name), 1);
+ }
+ return NS_ERROR_FAILURE;
+}
+
+void TelemetryHistogram::AccumulateCategorical(HistogramID aId,
+ const nsCString& label) {
+ if (NS_WARN_IF(!internal_IsHistogramEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Histogram usage requires valid ids.");
+ return;
+ }
+
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ if (!internal_CanRecordBase()) {
+ return;
+ }
+ uint32_t labelId = 0;
+ if (NS_FAILED(gHistogramInfos[aId].label_id(label.get(), &labelId))) {
+ return;
+ }
+ internal_Accumulate(locker, aId, labelId);
+}
+
+void TelemetryHistogram::AccumulateCategorical(
+ HistogramID aId, const nsTArray<nsCString>& aLabels) {
+ if (NS_WARN_IF(!internal_IsHistogramEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Histogram usage requires valid ids.");
+ return;
+ }
+
+ if (!internal_CanRecordBase()) {
+ return;
+ }
+
+ // We use two loops, one for getting label_ids and another one for actually
+ // accumulating the values. This ensures that in the case of an invalid label
+ // in the array, no values are accumulated. In any call to this API, either
+ // all or (in case of error) none of the values will be accumulated.
+
+ nsTArray<uint32_t> intSamples(aLabels.Length());
+ for (const nsCString& label : aLabels) {
+ uint32_t labelId = 0;
+ if (NS_FAILED(gHistogramInfos[aId].label_id(label.get(), &labelId))) {
+ return;
+ }
+ intSamples.AppendElement(labelId);
+ }
+
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+
+ for (uint32_t sample : intSamples) {
+ internal_Accumulate(locker, aId, sample);
+ }
+}
+
+void TelemetryHistogram::AccumulateChild(
+ ProcessID aProcessType,
+ const nsTArray<HistogramAccumulation>& aAccumulations) {
+ MOZ_ASSERT(XRE_IsParentProcess());
+
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ if (!internal_CanRecordBase()) {
+ return;
+ }
+ for (uint32_t i = 0; i < aAccumulations.Length(); ++i) {
+ if (NS_WARN_IF(!internal_IsHistogramEnumId(aAccumulations[i].mId))) {
+ MOZ_ASSERT_UNREACHABLE("Histogram usage requires valid ids.");
+ continue;
+ }
+ internal_AccumulateChild(locker, aProcessType, aAccumulations[i].mId,
+ aAccumulations[i].mSample);
+ }
+}
+
+void TelemetryHistogram::AccumulateChildKeyed(
+ ProcessID aProcessType,
+ const nsTArray<KeyedHistogramAccumulation>& aAccumulations) {
+ MOZ_ASSERT(XRE_IsParentProcess());
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ if (!internal_CanRecordBase()) {
+ return;
+ }
+ for (uint32_t i = 0; i < aAccumulations.Length(); ++i) {
+ if (NS_WARN_IF(!internal_IsHistogramEnumId(aAccumulations[i].mId))) {
+ MOZ_ASSERT_UNREACHABLE("Histogram usage requires valid ids.");
+ continue;
+ }
+ internal_AccumulateChildKeyed(locker, aProcessType, aAccumulations[i].mId,
+ aAccumulations[i].mKey,
+ aAccumulations[i].mSample);
+ }
+}
+
+nsresult TelemetryHistogram::GetAllStores(StringHashSet& set) {
+ for (uint32_t storeIdx : gHistogramStoresTable) {
+ const char* name = &gHistogramStringTable[storeIdx];
+ nsAutoCString store;
+ store.AssignASCII(name);
+ if (!set.PutEntry(store)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+ return NS_OK;
+}
+
+nsresult TelemetryHistogram::GetCategoricalHistogramLabels(
+ JSContext* aCx, JS::MutableHandle<JS::Value> aResult) {
+ JS::Rooted<JSObject*> root_obj(aCx, JS_NewPlainObject(aCx));
+ if (!root_obj) {
+ return NS_ERROR_FAILURE;
+ }
+ aResult.setObject(*root_obj);
+
+ for (const HistogramInfo& info : gHistogramInfos) {
+ if (info.histogramType != nsITelemetry::HISTOGRAM_CATEGORICAL) {
+ continue;
+ }
+
+ const char* name = info.name();
+ JS::RootedObject labels(aCx, JS::NewArrayObject(aCx, info.label_count));
+ if (!labels) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!JS_DefineProperty(aCx, root_obj, name, labels, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ for (uint32_t i = 0; i < info.label_count; ++i) {
+ uint32_t string_offset = gHistogramLabelTable[info.label_index + i];
+ const char* const label = &gHistogramStringTable[string_offset];
+ auto clabel = NS_ConvertASCIItoUTF16(label);
+ JS::Rooted<JS::Value> value(aCx);
+ value.setString(ToJSString(aCx, clabel));
+ if (!JS_DefineElement(aCx, labels, i, value, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult TelemetryHistogram::GetHistogramById(
+ const nsACString& name, JSContext* cx, JS::MutableHandle<JS::Value> ret) {
+ HistogramID id;
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ nsresult rv = internal_GetHistogramIdByName(locker, name, &id);
+ if (NS_FAILED(rv)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (gHistogramInfos[id].keyed) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+ // Runs without protection from |gTelemetryHistogramMutex|
+ return internal_WrapAndReturnHistogram(id, cx, ret);
+}
+
+nsresult TelemetryHistogram::GetKeyedHistogramById(
+ const nsACString& name, JSContext* cx, JS::MutableHandle<JS::Value> ret) {
+ HistogramID id;
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ nsresult rv = internal_GetHistogramIdByName(locker, name, &id);
+ if (NS_FAILED(rv)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!gHistogramInfos[id].keyed) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+ // Runs without protection from |gTelemetryHistogramMutex|
+ return internal_WrapAndReturnKeyedHistogram(id, cx, ret);
+}
+
+const char* TelemetryHistogram::GetHistogramName(HistogramID id) {
+ if (NS_WARN_IF(!internal_IsHistogramEnumId(id))) {
+ MOZ_ASSERT_UNREACHABLE("Histogram usage requires valid ids.");
+ return nullptr;
+ }
+
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ const HistogramInfo& h = gHistogramInfos[id];
+ return h.name();
+}
+
+nsresult TelemetryHistogram::CreateHistogramSnapshots(
+ JSContext* aCx, JS::MutableHandleValue aResult, const nsACString& aStore,
+ unsigned int aDataset, bool aClearSubsession, bool aFilterTest) {
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Runs without protection from |gTelemetryHistogramMutex|
+ JS::Rooted<JSObject*> root_obj(aCx, JS_NewPlainObject(aCx));
+ if (!root_obj) {
+ return NS_ERROR_FAILURE;
+ }
+ aResult.setObject(*root_obj);
+
+ // Include the GPU process in histogram snapshots only if we actually tried
+ // to launch a process for it.
+ bool includeGPUProcess = internal_AttemptedGPUProcess();
+
+ HistogramProcessSnapshotsArray processHistArray;
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ nsresult rv = internal_GetHistogramsSnapshot(
+ locker, aStore, aDataset, aClearSubsession, includeGPUProcess,
+ aFilterTest, processHistArray);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ }
+
+ // Make the JS calls on the stashed histograms for every process
+ for (uint32_t process = 0; process < processHistArray.length(); ++process) {
+ JS::Rooted<JSObject*> processObject(aCx, JS_NewPlainObject(aCx));
+ if (!processObject) {
+ return NS_ERROR_FAILURE;
+ }
+ if (!JS_DefineProperty(aCx, root_obj,
+ GetNameForProcessID(ProcessID(process)),
+ processObject, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ for (const HistogramSnapshotInfo& hData : processHistArray[process]) {
+ HistogramID id = hData.histogramID;
+
+ JS::Rooted<JSObject*> hobj(aCx, JS_NewPlainObject(aCx));
+ if (!hobj) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (NS_FAILED(internal_ReflectHistogramAndSamples(
+ aCx, hobj, gHistogramInfos[id], hData.data))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!JS_DefineProperty(aCx, processObject, gHistogramInfos[id].name(),
+ hobj, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+ }
+ return NS_OK;
+}
+
+nsresult TelemetryHistogram::GetKeyedHistogramSnapshots(
+ JSContext* aCx, JS::MutableHandleValue aResult, const nsACString& aStore,
+ unsigned int aDataset, bool aClearSubsession, bool aFilterTest) {
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Runs without protection from |gTelemetryHistogramMutex|
+ JS::Rooted<JSObject*> obj(aCx, JS_NewPlainObject(aCx));
+ if (!obj) {
+ return NS_ERROR_FAILURE;
+ }
+ aResult.setObject(*obj);
+
+ // Include the GPU process in histogram snapshots only if we actually tried
+ // to launch a process for it.
+ bool includeGPUProcess = internal_AttemptedGPUProcess();
+
+ // Get a snapshot of all the data while holding the mutex.
+ KeyedHistogramProcessSnapshotsArray processHistArray;
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ nsresult rv = internal_GetKeyedHistogramsSnapshot(
+ locker, aStore, aDataset, aClearSubsession, includeGPUProcess,
+ aFilterTest, processHistArray);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ }
+
+ // Mirror the snapshot data to JS, now that we released the mutex.
+ for (uint32_t process = 0; process < processHistArray.length(); ++process) {
+ JS::Rooted<JSObject*> processObject(aCx, JS_NewPlainObject(aCx));
+ if (!processObject) {
+ return NS_ERROR_FAILURE;
+ }
+ if (!JS_DefineProperty(aCx, obj, GetNameForProcessID(ProcessID(process)),
+ processObject, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ for (const KeyedHistogramSnapshotInfo& hData : processHistArray[process]) {
+ const HistogramInfo& info = gHistogramInfos[hData.histogramId];
+
+ JS::RootedObject snapshot(aCx, JS_NewPlainObject(aCx));
+ if (!snapshot) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!NS_SUCCEEDED(internal_ReflectKeyedHistogram(hData.data, info, aCx,
+ snapshot))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!JS_DefineProperty(aCx, processObject, info.name(), snapshot,
+ JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+ }
+ return NS_OK;
+}
+
+size_t TelemetryHistogram::GetHistogramSizesOfIncludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+
+ size_t n = 0;
+
+ // If we allocated the array, let's count the number of pointers in there and
+ // each entry's size.
+ if (gKeyedHistogramStorage) {
+ n += HistogramCount * size_t(ProcessID::Count) * sizeof(KeyedHistogram*);
+ for (size_t i = 0; i < HistogramCount * size_t(ProcessID::Count); ++i) {
+ if (gKeyedHistogramStorage[i] &&
+ gKeyedHistogramStorage[i] != gExpiredKeyedHistogram) {
+ n += gKeyedHistogramStorage[i]->SizeOfIncludingThis(aMallocSizeOf);
+ }
+ }
+ }
+
+ // If we allocated the array, let's count the number of pointers in there.
+ if (gHistogramStorage) {
+ n += HistogramCount * size_t(ProcessID::Count) * sizeof(Histogram*);
+ for (size_t i = 0; i < HistogramCount * size_t(ProcessID::Count); ++i) {
+ if (gHistogramStorage[i] && gHistogramStorage[i] != gExpiredHistogram) {
+ n += gHistogramStorage[i]->SizeOfIncludingThis(aMallocSizeOf);
+ }
+ }
+ }
+
+ // We only allocate the expired (keyed) histogram once.
+ if (gExpiredKeyedHistogram) {
+ n += gExpiredKeyedHistogram->SizeOfIncludingThis(aMallocSizeOf);
+ }
+
+ if (gExpiredHistogram) {
+ n += gExpiredHistogram->SizeOfIncludingThis(aMallocSizeOf);
+ }
+
+ return n;
+}
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: GeckoView specific helpers
+
+namespace base {
+class PersistedSampleSet : public base::Histogram::SampleSet {
+ public:
+ explicit PersistedSampleSet(const nsTArray<base::Histogram::Count>& aCounts,
+ int64_t aSampleSum);
+};
+
+PersistedSampleSet::PersistedSampleSet(
+ const nsTArray<base::Histogram::Count>& aCounts, int64_t aSampleSum) {
+ // Initialize the data in the base class. See Histogram::SampleSet
+ // for the fields documentation.
+ const size_t numCounts = aCounts.Length();
+ counts_.SetLength(numCounts);
+
+ for (size_t i = 0; i < numCounts; i++) {
+ counts_[i] = aCounts[i];
+ redundant_count_ += aCounts[i];
+ }
+ sum_ = aSampleSum;
+};
+} // namespace base
+
+namespace {
+/**
+ * Helper function to write histogram properties to JSON.
+ * Please note that this needs to be called between
+ * StartObjectProperty/EndObject calls that mark the histogram's
+ * JSON creation.
+ */
+void internal_ReflectHistogramToJSON(const HistogramSnapshotData& aSnapshot,
+ mozilla::JSONWriter& aWriter) {
+ aWriter.IntProperty("sum", aSnapshot.mSampleSum);
+
+ // Fill the "counts" property.
+ aWriter.StartArrayProperty("counts");
+ for (size_t i = 0; i < aSnapshot.mBucketCounts.Length(); i++) {
+ aWriter.IntElement(aSnapshot.mBucketCounts[i]);
+ }
+ aWriter.EndArray();
+}
+
+bool internal_CanRecordHistogram(const HistogramID id, ProcessID aProcessType) {
+ // Check if we are allowed to record the data.
+ if (!CanRecordDataset(gHistogramInfos[id].dataset, internal_CanRecordBase(),
+ internal_CanRecordExtended())) {
+ return false;
+ }
+
+ // Check if we're allowed to record in the given process.
+ if (aProcessType == ProcessID::Parent && !internal_IsRecordingEnabled(id)) {
+ return false;
+ }
+
+ if (aProcessType != ProcessID::Parent &&
+ !CanRecordInProcess(gHistogramInfos[id].record_in_processes,
+ aProcessType)) {
+ return false;
+ }
+
+ // Don't record if the current platform is not enabled
+ if (!CanRecordProduct(gHistogramInfos[id].products)) {
+ return false;
+ }
+
+ return true;
+}
+
+nsresult internal_ParseHistogramData(
+ JSContext* aCx, JS::HandleId aEntryId, JS::HandleObject aContainerObj,
+ nsACString& aOutName, nsTArray<base::Histogram::Count>& aOutCountArray,
+ int64_t& aOutSum) {
+ // Get the histogram name.
+ nsAutoJSString histogramName;
+ if (!histogramName.init(aCx, aEntryId)) {
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+
+ CopyUTF16toUTF8(histogramName, aOutName);
+
+ // Get the data for this histogram.
+ JS::RootedValue histogramData(aCx);
+ if (!JS_GetPropertyById(aCx, aContainerObj, aEntryId, &histogramData)) {
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!histogramData.isObject()) {
+ // base::Histogram data need to be an object. If that's not the case, skip
+ // it and try to load the rest of the data.
+ return NS_ERROR_FAILURE;
+ }
+
+ // Get the "sum" property.
+ JS::RootedValue sumValue(aCx);
+ JS::RootedObject histogramObj(aCx, &histogramData.toObject());
+ if (!JS_GetProperty(aCx, histogramObj, "sum", &sumValue)) {
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!JS::ToInt64(aCx, sumValue, &aOutSum)) {
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+
+ // Get the "counts" array.
+ JS::RootedValue countsArray(aCx);
+ bool countsIsArray = false;
+ if (!JS_GetProperty(aCx, histogramObj, "counts", &countsArray) ||
+ !JS::IsArrayObject(aCx, countsArray, &countsIsArray)) {
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!countsIsArray) {
+ // The "counts" property needs to be an array. If this is not the case,
+ // skip this histogram.
+ return NS_ERROR_FAILURE;
+ }
+
+ // Get the length of the array.
+ uint32_t countsLen = 0;
+ JS::RootedObject countsArrayObj(aCx, &countsArray.toObject());
+ if (!JS::GetArrayLength(aCx, countsArrayObj, &countsLen)) {
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+
+ // Parse the "counts" in the array.
+ for (uint32_t arrayIdx = 0; arrayIdx < countsLen; arrayIdx++) {
+ JS::RootedValue elementValue(aCx);
+ int countAsInt = 0;
+ if (!JS_GetElement(aCx, countsArrayObj, arrayIdx, &elementValue) ||
+ !JS::ToInt32(aCx, elementValue, &countAsInt)) {
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+ aOutCountArray.AppendElement(countAsInt);
+ }
+
+ return NS_OK;
+}
+
+} // Anonymous namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PUBLIC: GeckoView serialization/deserialization functions.
+
+nsresult TelemetryHistogram::SerializeHistograms(mozilla::JSONWriter& aWriter) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Only save histograms in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Include the GPU process in histogram snapshots only if we actually tried
+ // to launch a process for it.
+ bool includeGPUProcess = internal_AttemptedGPUProcess();
+
+ // Take a snapshot of the histograms.
+ HistogramProcessSnapshotsArray processHistArray;
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ // We always request the "opt-in"/"prerelease" dataset: we internally
+ // record the right subset, so this will only return "prerelease" if
+ // it was recorded.
+ if (NS_FAILED(internal_GetHistogramsSnapshot(
+ locker, "main"_ns, nsITelemetry::DATASET_PRERELEASE_CHANNELS,
+ false /* aClearSubsession */, includeGPUProcess,
+ false /* aFilterTest */, processHistArray))) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ // Make the JSON calls on the stashed histograms for every process
+ for (uint32_t process = 0; process < processHistArray.length(); ++process) {
+ aWriter.StartObjectProperty(
+ mozilla::MakeStringSpan(GetNameForProcessID(ProcessID(process))));
+
+ for (const HistogramSnapshotInfo& hData : processHistArray[process]) {
+ HistogramID id = hData.histogramID;
+
+ aWriter.StartObjectProperty(
+ mozilla::MakeStringSpan(gHistogramInfos[id].name()));
+ internal_ReflectHistogramToJSON(hData.data, aWriter);
+ aWriter.EndObject();
+ }
+ aWriter.EndObject();
+ }
+
+ return NS_OK;
+}
+
+nsresult TelemetryHistogram::SerializeKeyedHistograms(
+ mozilla::JSONWriter& aWriter) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Only save keyed histograms in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Include the GPU process in histogram snapshots only if we actually tried
+ // to launch a process for it.
+ bool includeGPUProcess = internal_AttemptedGPUProcess();
+
+ // Take a snapshot of the keyed histograms.
+ KeyedHistogramProcessSnapshotsArray processHistArray;
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+ // We always request the "opt-in"/"prerelease" dataset: we internally
+ // record the right subset, so this will only return "prerelease" if
+ // it was recorded.
+ if (NS_FAILED(internal_GetKeyedHistogramsSnapshot(
+ locker, "main"_ns, nsITelemetry::DATASET_PRERELEASE_CHANNELS,
+ false /* aClearSubsession */, includeGPUProcess,
+ false /* aFilterTest */, processHistArray))) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ // Serialize the keyed histograms for every process.
+ for (uint32_t process = 0; process < processHistArray.length(); ++process) {
+ aWriter.StartObjectProperty(
+ mozilla::MakeStringSpan(GetNameForProcessID(ProcessID(process))));
+
+ const KeyedHistogramSnapshotsArray& hArray = processHistArray[process];
+ for (size_t i = 0; i < hArray.length(); ++i) {
+ const KeyedHistogramSnapshotInfo& hData = hArray[i];
+ HistogramID id = hData.histogramId;
+ const HistogramInfo& info = gHistogramInfos[id];
+
+ aWriter.StartObjectProperty(mozilla::MakeStringSpan(info.name()));
+
+ // Each key is a new object with a "sum" and a "counts" property.
+ for (auto iter = hData.data.ConstIter(); !iter.Done(); iter.Next()) {
+ HistogramSnapshotData& keyData = iter.Data();
+ aWriter.StartObjectProperty(PromiseFlatCString(iter.Key()));
+ internal_ReflectHistogramToJSON(keyData, aWriter);
+ aWriter.EndObject();
+ }
+
+ aWriter.EndObject();
+ }
+ aWriter.EndObject();
+ }
+
+ return NS_OK;
+}
+
+nsresult TelemetryHistogram::DeserializeHistograms(JSContext* aCx,
+ JS::HandleValue aData) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Only load histograms in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Telemetry is disabled. This should never happen, but let's leave this check
+ // for consistency with other histogram updates routines.
+ if (!internal_CanRecordBase()) {
+ return NS_OK;
+ }
+
+ typedef mozilla::Tuple<nsCString, nsTArray<base::Histogram::Count>, int64_t>
+ PersistedHistogramTuple;
+ typedef mozilla::Vector<PersistedHistogramTuple> PersistedHistogramArray;
+ typedef mozilla::Vector<PersistedHistogramArray> PersistedHistogramStorage;
+
+ // Before updating the histograms, we need to get the data out of the JS
+ // wrappers. We can't hold the histogram mutex while handling JS stuff.
+ // Build a <histogram name, value> map.
+ JS::RootedObject histogramDataObj(aCx, &aData.toObject());
+ JS::Rooted<JS::IdVector> processes(aCx, JS::IdVector(aCx));
+ if (!JS_Enumerate(aCx, histogramDataObj, &processes)) {
+ // We can't even enumerate the processes in the loaded data, so
+ // there is nothing we could recover from the persistence file. Bail out.
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+
+ // Make sure we have enough storage for all the processes.
+ PersistedHistogramStorage histogramsToUpdate;
+ if (!histogramsToUpdate.resize(static_cast<uint32_t>(ProcessID::Count))) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+
+ // The following block of code attempts to extract as much data as possible
+ // from the serialized JSON, even in case of light data corruptions: if, for
+ // example, the data for a single process is corrupted or is in an unexpected
+ // form, we press on and attempt to load the data for the other processes.
+ JS::RootedId process(aCx);
+ for (auto& processVal : processes) {
+ // This is required as JS API calls require an Handle<jsid> and not a
+ // plain jsid.
+ process = processVal;
+ // Get the process name.
+ nsAutoJSString processNameJS;
+ if (!processNameJS.init(aCx, process)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Make sure it's valid. Note that this is safe to call outside
+ // of a locked section.
+ NS_ConvertUTF16toUTF8 processName(processNameJS);
+ ProcessID processID = GetIDForProcessName(processName.get());
+ if (processID == ProcessID::Count) {
+ NS_WARNING(
+ nsPrintfCString("Failed to get process ID for %s", processName.get())
+ .get());
+ continue;
+ }
+
+ // And its probes.
+ JS::RootedValue processData(aCx);
+ if (!JS_GetPropertyById(aCx, histogramDataObj, process, &processData)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ if (!processData.isObject()) {
+ // |processData| should be an object containing histograms. If this is
+ // not the case, silently skip and try to load the data for the other
+ // processes.
+ continue;
+ }
+
+ // Iterate through each histogram.
+ JS::RootedObject processDataObj(aCx, &processData.toObject());
+ JS::Rooted<JS::IdVector> histograms(aCx, JS::IdVector(aCx));
+ if (!JS_Enumerate(aCx, processDataObj, &histograms)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Get a reference to the deserialized data for this process.
+ PersistedHistogramArray& deserializedProcessData =
+ histogramsToUpdate[static_cast<uint32_t>(processID)];
+
+ JS::RootedId histogram(aCx);
+ for (auto& histogramVal : histograms) {
+ histogram = histogramVal;
+
+ int64_t sum = 0;
+ nsTArray<base::Histogram::Count> deserializedCounts;
+ nsCString histogramName;
+ if (NS_FAILED(internal_ParseHistogramData(aCx, histogram, processDataObj,
+ histogramName,
+ deserializedCounts, sum))) {
+ continue;
+ }
+
+ // Finally append the deserialized data to the storage.
+ if (!deserializedProcessData.emplaceBack(MakeTuple(
+ std::move(histogramName), std::move(deserializedCounts), sum))) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+ }
+ }
+
+ // Update the histogram storage.
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+
+ for (uint32_t process = 0; process < histogramsToUpdate.length();
+ ++process) {
+ PersistedHistogramArray& processArray = histogramsToUpdate[process];
+
+ for (auto& histogramData : processArray) {
+ // Attempt to get the corresponding ID for the deserialized histogram
+ // name.
+ HistogramID id;
+ if (NS_FAILED(internal_GetHistogramIdByName(
+ locker, mozilla::Get<0>(histogramData), &id))) {
+ continue;
+ }
+
+ ProcessID procID = static_cast<ProcessID>(process);
+ if (!internal_CanRecordHistogram(id, procID)) {
+ // We're not allowed to record this, so don't try to restore it.
+ continue;
+ }
+
+ // Get the Histogram instance: this will instantiate it if it doesn't
+ // exist.
+ Histogram* w = internal_GetHistogramById(locker, id, procID);
+ MOZ_ASSERT(w);
+
+ if (!w || w->IsExpired()) {
+ continue;
+ }
+
+ base::Histogram* h = nullptr;
+ constexpr auto store = "main"_ns;
+ if (!w->GetHistogram(store, &h)) {
+ continue;
+ }
+ MOZ_ASSERT(h);
+
+ if (!h) {
+ // Don't restore expired histograms.
+ continue;
+ }
+
+ // Make sure that histogram counts have matching sizes. If not,
+ // |AddSampleSet| will fail and crash.
+ size_t numCounts = mozilla::Get<1>(histogramData).Length();
+ if (h->bucket_count() != numCounts) {
+ MOZ_ASSERT(false,
+ "The number of restored buckets does not match with the "
+ "on in the definition");
+ continue;
+ }
+
+ // Update the data for the histogram.
+ h->AddSampleSet(
+ base::PersistedSampleSet(std::move(mozilla::Get<1>(histogramData)),
+ mozilla::Get<2>(histogramData)));
+ }
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult TelemetryHistogram::DeserializeKeyedHistograms(JSContext* aCx,
+ JS::HandleValue aData) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Only load keyed histograms in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Telemetry is disabled. This should never happen, but let's leave this check
+ // for consistency with other histogram updates routines.
+ if (!internal_CanRecordBase()) {
+ return NS_OK;
+ }
+
+ typedef mozilla::Tuple<nsCString, nsCString, nsTArray<base::Histogram::Count>,
+ int64_t>
+ PersistedKeyedHistogramTuple;
+ typedef mozilla::Vector<PersistedKeyedHistogramTuple>
+ PersistedKeyedHistogramArray;
+ typedef mozilla::Vector<PersistedKeyedHistogramArray>
+ PersistedKeyedHistogramStorage;
+
+ // Before updating the histograms, we need to get the data out of the JS
+ // wrappers. We can't hold the histogram mutex while handling JS stuff.
+ // Build a <histogram name, value> map.
+ JS::RootedObject histogramDataObj(aCx, &aData.toObject());
+ JS::Rooted<JS::IdVector> processes(aCx, JS::IdVector(aCx));
+ if (!JS_Enumerate(aCx, histogramDataObj, &processes)) {
+ // We can't even enumerate the processes in the loaded data, so
+ // there is nothing we could recover from the persistence file. Bail out.
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+
+ // Make sure we have enough storage for all the processes.
+ PersistedKeyedHistogramStorage histogramsToUpdate;
+ if (!histogramsToUpdate.resize(static_cast<uint32_t>(ProcessID::Count))) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+
+ // The following block of code attempts to extract as much data as possible
+ // from the serialized JSON, even in case of light data corruptions: if, for
+ // example, the data for a single process is corrupted or is in an unexpected
+ // form, we press on and attempt to load the data for the other processes.
+ JS::RootedId process(aCx);
+ for (auto& processVal : processes) {
+ // This is required as JS API calls require an Handle<jsid> and not a
+ // plain jsid.
+ process = processVal;
+ // Get the process name.
+ nsAutoJSString processNameJS;
+ if (!processNameJS.init(aCx, process)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Make sure it's valid. Note that this is safe to call outside
+ // of a locked section.
+ NS_ConvertUTF16toUTF8 processName(processNameJS);
+ ProcessID processID = GetIDForProcessName(processName.get());
+ if (processID == ProcessID::Count) {
+ NS_WARNING(
+ nsPrintfCString("Failed to get process ID for %s", processName.get())
+ .get());
+ continue;
+ }
+
+ // And its probes.
+ JS::RootedValue processData(aCx);
+ if (!JS_GetPropertyById(aCx, histogramDataObj, process, &processData)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ if (!processData.isObject()) {
+ // |processData| should be an object containing histograms. If this is
+ // not the case, silently skip and try to load the data for the other
+ // processes.
+ continue;
+ }
+
+ // Iterate through each keyed histogram.
+ JS::RootedObject processDataObj(aCx, &processData.toObject());
+ JS::Rooted<JS::IdVector> histograms(aCx, JS::IdVector(aCx));
+ if (!JS_Enumerate(aCx, processDataObj, &histograms)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Get a reference to the deserialized data for this process.
+ PersistedKeyedHistogramArray& deserializedProcessData =
+ histogramsToUpdate[static_cast<uint32_t>(processID)];
+
+ JS::RootedId histogram(aCx);
+ for (auto& histogramVal : histograms) {
+ histogram = histogramVal;
+ // Get the histogram name.
+ nsAutoJSString histogramName;
+ if (!histogramName.init(aCx, histogram)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Get the data for this histogram.
+ JS::RootedValue histogramData(aCx);
+ if (!JS_GetPropertyById(aCx, processDataObj, histogram, &histogramData)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Iterate through each key in the histogram.
+ JS::RootedObject keysDataObj(aCx, &histogramData.toObject());
+ JS::Rooted<JS::IdVector> keys(aCx, JS::IdVector(aCx));
+ if (!JS_Enumerate(aCx, keysDataObj, &keys)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ JS::RootedId key(aCx);
+ for (auto& keyVal : keys) {
+ key = keyVal;
+
+ int64_t sum = 0;
+ nsTArray<base::Histogram::Count> deserializedCounts;
+ nsCString keyName;
+ if (NS_FAILED(internal_ParseHistogramData(
+ aCx, key, keysDataObj, keyName, deserializedCounts, sum))) {
+ continue;
+ }
+
+ // Finally append the deserialized data to the storage.
+ if (!deserializedProcessData.emplaceBack(MakeTuple(
+ nsCString(NS_ConvertUTF16toUTF8(histogramName)),
+ std::move(keyName), std::move(deserializedCounts), sum))) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+ }
+ }
+ }
+
+ // Update the keyed histogram storage.
+ {
+ StaticMutexAutoLock locker(gTelemetryHistogramMutex);
+
+ for (uint32_t process = 0; process < histogramsToUpdate.length();
+ ++process) {
+ PersistedKeyedHistogramArray& processArray = histogramsToUpdate[process];
+
+ for (auto& histogramData : processArray) {
+ // Attempt to get the corresponding ID for the deserialized histogram
+ // name.
+ HistogramID id;
+ if (NS_FAILED(internal_GetHistogramIdByName(
+ locker, mozilla::Get<0>(histogramData), &id))) {
+ continue;
+ }
+
+ ProcessID procID = static_cast<ProcessID>(process);
+ if (!internal_CanRecordHistogram(id, procID)) {
+ // We're not allowed to record this, so don't try to restore it.
+ continue;
+ }
+
+ KeyedHistogram* keyed = internal_GetKeyedHistogramById(id, procID);
+ MOZ_ASSERT(keyed);
+
+ if (!keyed || keyed->IsExpired()) {
+ // Don't restore if we don't have a destination storage or the
+ // histogram is expired.
+ continue;
+ }
+
+ // Get data for the key we're looking for.
+ base::Histogram* h = nullptr;
+ if (NS_FAILED(keyed->GetHistogram(
+ "main"_ns, mozilla::Get<1>(histogramData), &h))) {
+ continue;
+ }
+ MOZ_ASSERT(h);
+
+ if (!h) {
+ // Don't restore if we don't have a destination storage.
+ continue;
+ }
+
+ // Make sure that histogram counts have matching sizes. If not,
+ // |AddSampleSet| will fail and crash.
+ size_t numCounts = mozilla::Get<2>(histogramData).Length();
+ if (h->bucket_count() != numCounts) {
+ MOZ_ASSERT(false,
+ "The number of restored buckets does not match with the "
+ "on in the definition");
+ continue;
+ }
+
+ // Update the data for the histogram.
+ h->AddSampleSet(
+ base::PersistedSampleSet(std::move(mozilla::Get<2>(histogramData)),
+ mozilla::Get<3>(histogramData)));
+ }
+ }
+ }
+
+ return NS_OK;
+}
diff --git a/toolkit/components/telemetry/core/TelemetryHistogram.h b/toolkit/components/telemetry/core/TelemetryHistogram.h
new file mode 100644
index 0000000000..a9b7490b41
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryHistogram.h
@@ -0,0 +1,120 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryHistogram_h__
+#define TelemetryHistogram_h__
+
+#include "mozilla/TelemetryComms.h"
+#include "mozilla/TelemetryHistogramEnums.h"
+#include "mozilla/TelemetryProcessEnums.h"
+#include "nsXULAppAPI.h"
+#include "TelemetryCommon.h"
+
+namespace mozilla {
+// This is only used for the GeckoView persistence.
+class JSONWriter;
+} // namespace mozilla
+
+// This module is internal to Telemetry. It encapsulates Telemetry's
+// histogram accumulation and storage logic. It should only be used by
+// Telemetry.cpp. These functions should not be used anywhere else.
+// For the public interface to Telemetry functionality, see Telemetry.h.
+
+namespace TelemetryHistogram {
+
+void InitializeGlobalState(bool canRecordBase, bool canRecordExtended);
+void DeInitializeGlobalState();
+#ifdef DEBUG
+bool GlobalStateHasBeenInitialized();
+#endif
+
+bool CanRecordBase();
+void SetCanRecordBase(bool b);
+bool CanRecordExtended();
+void SetCanRecordExtended(bool b);
+
+void InitHistogramRecordingEnabled();
+void SetHistogramRecordingEnabled(mozilla::Telemetry::HistogramID aID,
+ bool aEnabled);
+
+nsresult SetHistogramRecordingEnabled(const nsACString& id, bool aEnabled);
+
+void Accumulate(mozilla::Telemetry::HistogramID aHistogram, uint32_t aSample);
+void Accumulate(mozilla::Telemetry::HistogramID aHistogram,
+ const nsTArray<uint32_t>& aSamples);
+void Accumulate(mozilla::Telemetry::HistogramID aID, const nsCString& aKey,
+ uint32_t aSample);
+void Accumulate(mozilla::Telemetry::HistogramID aID, const nsCString& aKey,
+ const nsTArray<uint32_t>& aSamples);
+/*
+ * Accumulate a sample into the named histogram.
+ *
+ * Returns NS_OK on success.
+ * Returns NS_ERROR_NOT_AVAILABLE if recording Telemetry is disabled.
+ * Returns NS_ERROR_FAILURE on other errors.
+ */
+nsresult Accumulate(const char* name, uint32_t sample);
+
+/*
+ * Accumulate a sample into the named keyed histogram by key.
+ *
+ * Returns NS_OK on success.
+ * Returns NS_ERROR_NOT_AVAILABLE if recording Telemetry is disabled.
+ * Returns NS_ERROR_FAILURE on other errors.
+ */
+nsresult Accumulate(const char* name, const nsCString& key, uint32_t sample);
+
+void AccumulateCategorical(mozilla::Telemetry::HistogramID aId,
+ const nsCString& aLabel);
+void AccumulateCategorical(mozilla::Telemetry::HistogramID aId,
+ const nsTArray<nsCString>& aLabels);
+
+void AccumulateChild(
+ mozilla::Telemetry::ProcessID aProcessType,
+ const nsTArray<mozilla::Telemetry::HistogramAccumulation>& aAccumulations);
+void AccumulateChildKeyed(
+ mozilla::Telemetry::ProcessID aProcessType,
+ const nsTArray<mozilla::Telemetry::KeyedHistogramAccumulation>&
+ aAccumulations);
+
+/**
+ * Append the list of registered stores to the given set.
+ */
+nsresult GetAllStores(mozilla::Telemetry::Common::StringHashSet& set);
+
+nsresult GetCategoricalHistogramLabels(JSContext* aCx,
+ JS::MutableHandle<JS::Value> aResult);
+
+nsresult GetHistogramById(const nsACString& name, JSContext* cx,
+ JS::MutableHandle<JS::Value> ret);
+
+nsresult GetKeyedHistogramById(const nsACString& name, JSContext* cx,
+ JS::MutableHandle<JS::Value> ret);
+
+const char* GetHistogramName(mozilla::Telemetry::HistogramID id);
+
+nsresult CreateHistogramSnapshots(JSContext* aCx,
+ JS::MutableHandleValue aResult,
+ const nsACString& aStore,
+ unsigned int aDataset, bool aClearSubsession,
+ bool aFilterTest = false);
+
+nsresult GetKeyedHistogramSnapshots(
+ JSContext* aCx, JS::MutableHandleValue aResult, const nsACString& aStore,
+ unsigned int aDataset, bool aClearSubsession, bool aFilterTest = false);
+
+size_t GetHistogramSizesOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf);
+
+// These functions are only meant to be used for GeckoView persistence.
+// They are responsible for updating in-memory probes with the data persisted
+// on the disk and vice-versa.
+nsresult SerializeHistograms(mozilla::JSONWriter& aWriter);
+nsresult SerializeKeyedHistograms(mozilla::JSONWriter& aWriter);
+nsresult DeserializeHistograms(JSContext* aCx, JS::HandleValue aData);
+nsresult DeserializeKeyedHistograms(JSContext* aCx, JS::HandleValue aData);
+
+} // namespace TelemetryHistogram
+
+#endif // TelemetryHistogram_h__
diff --git a/toolkit/components/telemetry/core/TelemetryOrigin.cpp b/toolkit/components/telemetry/core/TelemetryOrigin.cpp
new file mode 100644
index 0000000000..6c32796d98
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryOrigin.cpp
@@ -0,0 +1,625 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=2 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 "Telemetry.h"
+#include "TelemetryOrigin.h"
+
+#include "nsDataHashtable.h"
+#include "nsIObserverService.h"
+#include "nsPrintfCString.h"
+#include "nsTArray.h"
+#include "TelemetryCommon.h"
+#include "TelemetryOriginEnums.h"
+
+#include "js/Array.h" // JS::NewArrayObject
+#include "mozilla/Atomics.h"
+#include "mozilla/Base64.h"
+#include "mozilla/dom/PrioEncoder.h"
+#include "mozilla/Preferences.h"
+#include "mozilla/Services.h"
+#include "mozilla/StaticMutex.h"
+#include "mozilla/Tuple.h"
+#include "mozilla/UniquePtr.h"
+
+#include <cmath>
+#include <type_traits>
+
+using mozilla::Get;
+using mozilla::MakeTuple;
+using mozilla::MakeUnique;
+using mozilla::MallocSizeOf;
+using mozilla::StaticMutex;
+using mozilla::StaticMutexAutoLock;
+using mozilla::Tuple;
+using mozilla::UniquePtr;
+using mozilla::dom::PrioEncoder;
+using mozilla::Telemetry::OriginMetricID;
+using mozilla::Telemetry::Common::ToJSString;
+
+/***********************************************************************
+ *
+ * Firefox Origin Telemetry
+ * Docs:
+ * https://firefox-source-docs.mozilla.org/toolkit/components/telemetry/telemetry/collection/origin.html
+ *
+ * Origin Telemetry stores pairs of information (metric, origin) which boils
+ * down to "$metric happened on $origin".
+ *
+ * Prio can only encode up-to-2046-length bit vectors. The process of
+ * transforming these pairs of information into bit vectors is called "App
+ * Encoding". The bit vectors are then "Prio Encoded" into binary goop. The
+ * binary goop is then "Base64 Encoded" into strings.
+ *
+ */
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE TYPES
+
+namespace {
+
+class OriginMetricIDHashKey : public PLDHashEntryHdr {
+ public:
+ typedef const OriginMetricID& KeyType;
+ typedef const OriginMetricID* KeyTypePointer;
+
+ explicit OriginMetricIDHashKey(KeyTypePointer aKey) : mValue(*aKey) {}
+ OriginMetricIDHashKey(OriginMetricIDHashKey&& aOther)
+ : PLDHashEntryHdr(std::move(aOther)), mValue(std::move(aOther.mValue)) {}
+ ~OriginMetricIDHashKey() = default;
+
+ KeyType GetKey() const { return mValue; }
+ bool KeyEquals(KeyTypePointer aKey) const { return *aKey == mValue; }
+
+ static KeyTypePointer KeyToPointer(KeyType aKey) { return &aKey; }
+ static PLDHashNumber HashKey(KeyTypePointer aKey) {
+ return static_cast<std::underlying_type<OriginMetricID>::type>(*aKey);
+ }
+ enum { ALLOW_MEMMOVE = true };
+
+ private:
+ const OriginMetricID mValue;
+};
+
+} // anonymous namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE STATE, SHARED BY ALL THREADS
+//
+// Access for all of this state (except gInitDone) must be guarded by
+// gTelemetryOriginMutex.
+
+namespace {
+
+// This is a StaticMutex rather than a plain Mutex (1) so that
+// it gets initialised in a thread-safe manner the first time
+// it is used, and (2) because it is never de-initialised, and
+// a normal Mutex would show up as a leak in BloatView. StaticMutex
+// also has the "OffTheBooks" property, so it won't show as a leak
+// in BloatView.
+// Another reason to use a StaticMutex instead of a plain Mutex is
+// that, due to the nature of Telemetry, we cannot rely on having a
+// mutex initialized in InitializeGlobalState. Unfortunately, we
+// cannot make sure that no other function is called before this point.
+static StaticMutex gTelemetryOriginMutex;
+
+typedef nsTArray<Tuple<const char*, const char*>> OriginHashesList;
+UniquePtr<OriginHashesList> gOriginHashesList;
+
+typedef nsDataHashtable<nsCStringHashKey, size_t> OriginToIndexMap;
+UniquePtr<OriginToIndexMap> gOriginToIndexMap;
+
+typedef nsDataHashtable<nsCStringHashKey, size_t> HashToIndexMap;
+UniquePtr<HashToIndexMap> gHashToIndexMap;
+
+typedef nsDataHashtable<nsCStringHashKey, uint32_t> OriginBag;
+typedef nsDataHashtable<OriginMetricIDHashKey, OriginBag> IdToOriginBag;
+
+UniquePtr<IdToOriginBag> gMetricToOriginBag;
+
+mozilla::Atomic<bool, mozilla::Relaxed> gInitDone(false);
+
+// Useful for app-encoded data
+typedef nsTArray<std::pair<OriginMetricID, nsTArray<nsTArray<bool>>>>
+ IdBoolsPairArray;
+
+// Prio has a maximum supported number of bools it can encode at a time.
+// This means a single metric may result in several encoded payloads if the
+// number of origins exceeds the number of bools.
+// Each encoded payload corresponds to an element in the `prioData` array in the
+// "prio" ping.
+// This number is the number of encoded payloads needed per metric, and is
+// equivalent to "how many bitvectors do we need to encode this whole list of
+// origins?"
+static uint32_t gPrioDatasPerMetric;
+
+// The number of "meta-origins": in-band metadata about origin telemetry.
+// Currently 1: the "unknown origin recorded" meta-origin.
+static uint32_t kNumMetaOrigins = 1;
+
+constexpr auto kUnknownOrigin = "__UNKNOWN__"_ns;
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: thread-safe helpers
+
+namespace {
+
+const char* GetNameForMetricID(OriginMetricID aId) {
+ MOZ_ASSERT(aId < OriginMetricID::Count);
+ return mozilla::Telemetry::MetricIDToString[static_cast<uint32_t>(aId)];
+}
+
+// Calculates the number of `prioData` elements we'd need if we were asked for
+// an encoded snapshot right now.
+uint32_t PrioDataCount(const StaticMutexAutoLock& lock) {
+ uint32_t count = 0;
+ auto iter = gMetricToOriginBag->ConstIter();
+ for (; !iter.Done(); iter.Next()) {
+ auto originIt = iter.Data().ConstIter();
+ uint32_t maxOriginCount = 0;
+ for (; !originIt.Done(); originIt.Next()) {
+ maxOriginCount = std::max(maxOriginCount, originIt.Data());
+ }
+ count += gPrioDatasPerMetric * maxOriginCount;
+ }
+ return count;
+}
+
+// Takes the storage and turns it into bool arrays for Prio to encode, turning
+// { metric1: [origin1, origin2, ...], ...}
+// into
+// [(metric1, [[shard1], [shard2], ...]), ...]
+// Note: if an origin is present multiple times for a given metric, we must
+// generate multiple (id, boolvectors) pairs so that they are all reported.
+// Meaning
+// { metric1: [origin1, origin2, origin2] }
+// must turn into (with a pretend gNumBooleans of 1)
+// [(metric1, [[1], [1]]), (metric1, [[0], [1]])]
+nsresult AppEncodeTo(const StaticMutexAutoLock& lock,
+ IdBoolsPairArray& aResult) {
+ auto iter = gMetricToOriginBag->ConstIter();
+ for (; !iter.Done(); iter.Next()) {
+ OriginMetricID id = iter.Key();
+ const OriginBag& bag = iter.Data();
+
+ uint32_t generation = 1;
+ uint32_t maxGeneration = 1;
+ do {
+ // Fill in the result bool vectors with `false`s.
+ nsTArray<nsTArray<bool>> metricData(gPrioDatasPerMetric);
+ metricData.SetLength(gPrioDatasPerMetric);
+ for (size_t i = 0; i < metricData.Length() - 1; ++i) {
+ metricData[i].SetLength(PrioEncoder::gNumBooleans);
+ for (auto& metricDatum : metricData[i]) {
+ metricDatum = false;
+ }
+ }
+ auto& lastArray = metricData[metricData.Length() - 1];
+ lastArray.SetLength((gOriginHashesList->Length() + kNumMetaOrigins) %
+ PrioEncoder::gNumBooleans);
+ for (auto& metricDatum : lastArray) {
+ metricDatum = false;
+ }
+
+ auto originIt = bag.ConstIter();
+ for (; !originIt.Done(); originIt.Next()) {
+ uint32_t originCount = originIt.Data();
+ if (originCount >= generation) {
+ maxGeneration = std::max(maxGeneration, originCount);
+
+ const nsACString& origin = originIt.Key();
+ size_t index;
+ if (!gOriginToIndexMap->Get(origin, &index)) {
+ return NS_ERROR_FAILURE;
+ }
+ MOZ_ASSERT(index < (gOriginHashesList->Length() + kNumMetaOrigins));
+ size_t shardIndex = index / PrioEncoder::gNumBooleans;
+ MOZ_ASSERT(shardIndex < metricData.Length());
+ MOZ_ASSERT(index % PrioEncoder::gNumBooleans <
+ metricData[shardIndex].Length());
+ metricData[shardIndex][index % PrioEncoder::gNumBooleans] = true;
+ }
+ }
+ aResult.EmplaceBack(id, std::move(metricData));
+ } while (generation++ < maxGeneration);
+ }
+ return NS_OK;
+}
+
+} // anonymous namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// EXTERNALLY VISIBLE FUNCTIONS in namespace TelemetryOrigin::
+
+void TelemetryOrigin::InitializeGlobalState() {
+ if (!XRE_IsParentProcess()) {
+ return;
+ }
+
+ StaticMutexAutoLock locker(gTelemetryOriginMutex);
+
+ MOZ_ASSERT(!gInitDone,
+ "TelemetryOrigin::InitializeGlobalState "
+ "may only be called once");
+
+ // The contents and order of the arrays that follow matter.
+ // Both ensure a consistent app-encoding.
+ static const char sOriginStrings[] = {
+#define ORIGIN(origin, hash) origin "\0"
+#include "TelemetryOriginData.inc"
+#undef ORIGIN
+ };
+ static const char sHashStrings[] = {
+#define ORIGIN(origin, hash) hash "\0"
+#include "TelemetryOriginData.inc"
+#undef ORIGIN
+ };
+
+ struct OriginHashLengths {
+ uint8_t originLength;
+ uint8_t hashLength;
+ };
+ static const OriginHashLengths sOriginHashLengths[] = {
+#define ORIGIN(origin, hash) {sizeof(origin), sizeof(hash)},
+#include "TelemetryOriginData.inc"
+#undef ORIGIN
+ };
+
+ static const size_t kNumOrigins = MOZ_ARRAY_LENGTH(sOriginHashLengths);
+
+ gOriginHashesList = MakeUnique<OriginHashesList>(kNumOrigins);
+
+ gPrioDatasPerMetric =
+ ceil(static_cast<double>(kNumOrigins + kNumMetaOrigins) /
+ PrioEncoder::gNumBooleans);
+
+ gOriginToIndexMap =
+ MakeUnique<OriginToIndexMap>(kNumOrigins + kNumMetaOrigins);
+ gHashToIndexMap = MakeUnique<HashToIndexMap>(kNumOrigins);
+ size_t originOffset = 0;
+ size_t hashOffset = 0;
+ for (size_t i = 0; i < kNumOrigins; ++i) {
+ const char* origin = &sOriginStrings[originOffset];
+ const char* hash = &sHashStrings[hashOffset];
+ MOZ_ASSERT(!kUnknownOrigin.Equals(origin),
+ "Unknown origin literal is reserved in Origin Telemetry");
+
+ gOriginHashesList->AppendElement(MakeTuple(origin, hash));
+
+ const size_t originLength = sOriginHashLengths[i].originLength;
+ const size_t hashLength = sOriginHashLengths[i].hashLength;
+
+ originOffset += originLength;
+ hashOffset += hashLength;
+
+ // -1 to leave off the null terminators.
+ gOriginToIndexMap->Put(nsDependentCString(origin, originLength - 1), i);
+ gHashToIndexMap->Put(nsDependentCString(hash, hashLength - 1), i);
+ }
+
+ // Add the meta-origin for tracking recordings to untracked origins.
+ gOriginToIndexMap->Put(kUnknownOrigin, gOriginHashesList->Length());
+
+ gMetricToOriginBag = MakeUnique<IdToOriginBag>();
+
+ // This map shouldn't change at runtime, so make debug builds complain
+ // if it tries.
+ gOriginToIndexMap->MarkImmutable();
+ gHashToIndexMap->MarkImmutable();
+
+ gInitDone = true;
+}
+
+void TelemetryOrigin::DeInitializeGlobalState() {
+ if (!XRE_IsParentProcess()) {
+ return;
+ }
+
+ StaticMutexAutoLock locker(gTelemetryOriginMutex);
+ MOZ_ASSERT(gInitDone);
+
+ if (!gInitDone) {
+ return;
+ }
+
+ gOriginHashesList = nullptr;
+
+ gOriginToIndexMap = nullptr;
+
+ gHashToIndexMap = nullptr;
+
+ gMetricToOriginBag = nullptr;
+
+ gInitDone = false;
+}
+
+nsresult TelemetryOrigin::RecordOrigin(OriginMetricID aId,
+ const nsACString& aOrigin) {
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ uint32_t prioDataCount;
+ {
+ StaticMutexAutoLock locker(gTelemetryOriginMutex);
+
+ // Common Telemetry error-handling practices for recording functions:
+ // only illegal calls return errors whereas merely incorrect ones are mutely
+ // ignored.
+ if (!gInitDone) {
+ return NS_OK;
+ }
+
+ size_t index;
+ nsCString origin(aOrigin);
+ if (gHashToIndexMap->Get(aOrigin, &index)) {
+ MOZ_ASSERT(aOrigin.Equals(Get<1>((*gOriginHashesList)[index])));
+ origin = Get<0>((*gOriginHashesList)[index]);
+ }
+
+ if (!gOriginToIndexMap->Contains(origin)) {
+ // Only record one unknown origin per metric per snapshot.
+ // (otherwise we may get swamped and blow our data budget.)
+ if (gMetricToOriginBag->Contains(aId) &&
+ gMetricToOriginBag->GetOrInsert(aId).Contains(kUnknownOrigin)) {
+ return NS_OK;
+ }
+ origin = kUnknownOrigin;
+ }
+
+ auto& originBag = gMetricToOriginBag->GetOrInsert(aId);
+ originBag.GetOrInsert(origin)++;
+
+ prioDataCount = PrioDataCount(locker);
+ }
+
+ static uint32_t sPrioPingLimit =
+ mozilla::Preferences::GetUint("toolkit.telemetry.prioping.dataLimit", 10);
+ if (prioDataCount >= sPrioPingLimit) {
+ nsCOMPtr<nsIObserverService> os = mozilla::services::GetObserverService();
+ if (os) {
+ // Ensure we don't notify while holding the lock in case of synchronous
+ // dispatch. May deadlock ourselves if we then trigger a snapshot.
+ os->NotifyObservers(nullptr, "origin-telemetry-storage-limit-reached",
+ nullptr);
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult TelemetryOrigin::GetOriginSnapshot(bool aClear, JSContext* aCx,
+ JS::MutableHandleValue aResult) {
+ if (NS_WARN_IF(!XRE_IsParentProcess())) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!gInitDone) {
+ return NS_OK;
+ }
+
+ // Step 1: Grab the lock, copy into stack-local storage, optionally clear.
+ IdToOriginBag copy;
+ {
+ StaticMutexAutoLock locker(gTelemetryOriginMutex);
+
+ if (aClear) {
+ // I'd really prefer to clear after we're sure the snapshot didn't go
+ // awry, but we can't hold a lock preventing recording while using JS
+ // APIs. And replaying any interleaving recording sounds like too much
+ // squeeze for not enough juice.
+
+ gMetricToOriginBag->SwapElements(copy);
+ } else {
+ auto iter = gMetricToOriginBag->ConstIter();
+ for (; !iter.Done(); iter.Next()) {
+ OriginBag& bag = copy.GetOrInsert(iter.Key());
+ auto originIt = iter.Data().ConstIter();
+ for (; !originIt.Done(); originIt.Next()) {
+ bag.Put(originIt.Key(), originIt.Data());
+ }
+ }
+ }
+ }
+
+ // Step 2: Without the lock, generate JS datastructure for snapshotting
+ JS::Rooted<JSObject*> rootObj(aCx, JS_NewPlainObject(aCx));
+ if (NS_WARN_IF(!rootObj)) {
+ return NS_ERROR_FAILURE;
+ }
+ aResult.setObject(*rootObj);
+ for (auto iter = copy.ConstIter(); !iter.Done(); iter.Next()) {
+ JS::RootedObject originsObj(aCx, JS_NewPlainObject(aCx));
+ if (NS_WARN_IF(!originsObj)) {
+ return NS_ERROR_FAILURE;
+ }
+ if (!JS_DefineProperty(aCx, rootObj, GetNameForMetricID(iter.Key()),
+ originsObj, JSPROP_ENUMERATE)) {
+ NS_WARNING("Failed to define property in origin snapshot.");
+ return NS_ERROR_FAILURE;
+ }
+
+ auto originIt = iter.Data().ConstIter();
+ for (; !originIt.Done(); originIt.Next()) {
+ if (!JS_DefineProperty(aCx, originsObj,
+ nsPromiseFlatCString(originIt.Key()).get(),
+ originIt.Data(), JSPROP_ENUMERATE)) {
+ NS_WARNING("Failed to define origin and count in snapshot.");
+ return NS_ERROR_FAILURE;
+ }
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult TelemetryOrigin::GetEncodedOriginSnapshot(
+ bool aClear, JSContext* aCx, JS::MutableHandleValue aSnapshot) {
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!gInitDone) {
+ return NS_OK;
+ }
+
+ // Step 1: Take the lock and app-encode. Optionally clear.
+ nsresult rv;
+ IdBoolsPairArray appEncodedMetricData;
+ {
+ StaticMutexAutoLock lock(gTelemetryOriginMutex);
+
+ rv = AppEncodeTo(lock, appEncodedMetricData);
+ if (NS_WARN_IF(NS_FAILED(rv))) {
+ return rv;
+ }
+
+ if (aClear) {
+ // I'd really prefer to clear after we're sure the snapshot didn't go
+ // awry, but we can't hold a lock preventing recording while using JS
+ // APIs. And replaying any interleaving recording sounds like too much
+ // squeeze for not enough juice.
+
+ gMetricToOriginBag->Clear();
+ }
+ }
+
+ // Step 2: Don't need the lock to prio-encode and base64-encode
+ nsTArray<std::pair<nsCString, std::pair<nsCString, nsCString>>> prioData;
+ for (auto& metricData : appEncodedMetricData) {
+ auto& boolVectors = metricData.second;
+ for (uint32_t i = 0; i < boolVectors.Length(); ++i) {
+ // "encoding" is of the form `metricName-X` where X is the shard index.
+ nsCString encodingName =
+ nsPrintfCString("%s-%u", GetNameForMetricID(metricData.first), i);
+ nsCString aResult;
+ nsCString bResult;
+ rv = PrioEncoder::EncodeNative(encodingName, boolVectors[i], aResult,
+ bResult);
+ if (NS_WARN_IF(NS_FAILED(rv))) {
+ return rv;
+ }
+ nsCString aBase64;
+ rv = mozilla::Base64Encode(aResult, aBase64);
+ if (NS_WARN_IF(NS_FAILED(rv))) {
+ return rv;
+ }
+ nsCString bBase64;
+ rv = mozilla::Base64Encode(bResult, bBase64);
+ if (NS_WARN_IF(NS_FAILED(rv))) {
+ return rv;
+ }
+
+ prioData.EmplaceBack(std::move(encodingName),
+ std::pair(std::move(aBase64), std::move(bBase64)));
+ }
+ }
+
+ // Step 3: Still don't need the lock to translate to JS
+ // The resulting data structure is:
+ // [{
+ // encoding: <encoding name>,
+ // prio: {
+ // a: <base64 string>,
+ // b: <base64 string>,
+ // },
+ // }, ...]
+
+ JS::RootedObject prioDataArray(aCx,
+ JS::NewArrayObject(aCx, prioData.Length()));
+ if (NS_WARN_IF(!prioDataArray)) {
+ return NS_ERROR_FAILURE;
+ }
+ uint32_t i = 0;
+ for (auto& prioDatum : prioData) {
+ JS::RootedObject prioDatumObj(aCx, JS_NewPlainObject(aCx));
+ if (NS_WARN_IF(!prioDatumObj)) {
+ return NS_ERROR_FAILURE;
+ }
+ JSString* encoding = ToJSString(aCx, prioDatum.first);
+ JS::RootedString rootedEncoding(aCx, encoding);
+ if (NS_WARN_IF(!JS_DefineProperty(aCx, prioDatumObj, "encoding",
+ rootedEncoding, JSPROP_ENUMERATE))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ JS::RootedObject prioObj(aCx, JS_NewPlainObject(aCx));
+ if (NS_WARN_IF(!prioObj)) {
+ return NS_ERROR_FAILURE;
+ }
+ if (NS_WARN_IF(!JS_DefineProperty(aCx, prioDatumObj, "prio", prioObj,
+ JSPROP_ENUMERATE))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ JS::RootedString aRootStr(aCx, ToJSString(aCx, prioDatum.second.first));
+ if (NS_WARN_IF(!JS_DefineProperty(aCx, prioObj, "a", aRootStr,
+ JSPROP_ENUMERATE))) {
+ return NS_ERROR_FAILURE;
+ }
+ JS::RootedString bRootStr(aCx, ToJSString(aCx, prioDatum.second.second));
+ if (NS_WARN_IF(!JS_DefineProperty(aCx, prioObj, "b", bRootStr,
+ JSPROP_ENUMERATE))) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (NS_WARN_IF(!JS_DefineElement(aCx, prioDataArray, i++, prioDatumObj,
+ JSPROP_ENUMERATE))) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ aSnapshot.setObject(*prioDataArray);
+
+ return NS_OK;
+}
+
+/**
+ * Resets all the stored events. This is intended to be only used in tests.
+ */
+void TelemetryOrigin::ClearOrigins() {
+ StaticMutexAutoLock lock(gTelemetryOriginMutex);
+
+ if (!gInitDone) {
+ return;
+ }
+
+ gMetricToOriginBag->Clear();
+}
+
+size_t TelemetryOrigin::SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) {
+ StaticMutexAutoLock locker(gTelemetryOriginMutex);
+ size_t n = 0;
+
+ if (!gInitDone) {
+ return 0;
+ }
+
+ n += gMetricToOriginBag->ShallowSizeOfIncludingThis(aMallocSizeOf);
+ auto iter = gMetricToOriginBag->ConstIter();
+ for (; !iter.Done(); iter.Next()) {
+ // The string hashkey and count should both be contained by the hashtable.
+ n += iter.Data().ShallowSizeOfIncludingThis(aMallocSizeOf);
+ }
+
+ // The string hashkey and ID should both be contained within the hashtable.
+ n += gOriginToIndexMap->ShallowSizeOfIncludingThis(aMallocSizeOf);
+
+ return n;
+}
+
+size_t TelemetryOrigin::SizeOfPrioDatasPerMetric() {
+ return gPrioDatasPerMetric;
+}
diff --git a/toolkit/components/telemetry/core/TelemetryOrigin.h b/toolkit/components/telemetry/core/TelemetryOrigin.h
new file mode 100644
index 0000000000..b0d3721200
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryOrigin.h
@@ -0,0 +1,45 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryOrigin_h__
+#define TelemetryOrigin_h__
+
+#include "TelemetryOriginEnums.h"
+#include "jsapi.h"
+#include "nsError.h"
+#include "nsString.h"
+
+// This module is internal to Telemetry. It encapsulates Telemetry's
+// origin recording and storage logic. It should only be used by
+// Telemetry.cpp. These functions should not be used anywhere else.
+// For the public interface to Telemetry functionality, see Telemetry.h.
+
+namespace TelemetryOrigin {
+
+void InitializeGlobalState();
+void DeInitializeGlobalState();
+
+// C++ Recording Endpoint
+nsresult RecordOrigin(mozilla::Telemetry::OriginMetricID aId,
+ const nsACString& aOrigin);
+
+// JS API Endpoints.
+nsresult GetOriginSnapshot(bool aClear, JSContext* aCx,
+ JS::MutableHandleValue aResult);
+
+nsresult GetEncodedOriginSnapshot(bool aClear, JSContext* aCx,
+ JS::MutableHandleValue aSnapshot);
+
+// Only to be used for testing.
+void ClearOrigins();
+
+size_t SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf);
+
+// Only to be used for testing.
+size_t SizeOfPrioDatasPerMetric();
+
+} // namespace TelemetryOrigin
+
+#endif // TelemetryOrigin_h__
diff --git a/toolkit/components/telemetry/core/TelemetryOriginData.inc b/toolkit/components/telemetry/core/TelemetryOriginData.inc
new file mode 100644
index 0000000000..7587e2664e
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryOriginData.inc
@@ -0,0 +1,2488 @@
+// dummy origin, this is used to be a counter of page loads.
+ORIGIN("PAGELOAD", "PAGELOAD")
+
+ORIGIN("advertstream.com", "lzPiT1FuoHNMKQ1Hw8AaTi68TokOB24ciFBqmCk62ek=")
+ORIGIN("kitaramedia.com", "r+U9PL3uMrjCKe8/T8goY9MHPA+6JckC3R+/1R9TQKA=")
+ORIGIN("questionmarket.com", "3KCO/qN+KmApmfH3RaXAmdR65Z/TRfrr6pds7aDKn1c=")
+ORIGIN("3lift.com", "33Xrix7c41Jc9q3InjMWHq+yKVoa/u2IB511kr4X+Ro=")
+ORIGIN("affine.tv", "wKqJRMTORbe8f4TO5LTGaTlt4yZVKlyomX2Exvfcv6A=")
+ORIGIN("longtailvideo.com", "B0tCrpMgOedsm2p2cLJcHrITf/amX5cxB+90PrEpJ8c=")
+ORIGIN("undertonenetworks.com", "uaBhUaQhVEMpY9MctAg2QNOszcD6VUq2Gkf44GOp3tA=")
+ORIGIN("rhythmnewmedia.com", "PmyV0h3/zQb41TFz6eMHhmHA7KoTtMCSiFWjhQhu22A=")
+ORIGIN("admob.com", "LDMime9kpePYvq6Pq12mURD/D8Z2KMKzs09NjrzyPLw=")
+ORIGIN("traffichouse.com", "DhgYTwLBhShT2/9n10Uf9eVPSsSK0Zg06CHADRkjDVM=")
+ORIGIN("factortg.com", "zS+1e249KvImm0daAdpW1C1HpRvWpbqPPEEI6AWX10Q=")
+ORIGIN("quantcount.com", "8Bca3PU4reqBFEDTN15xsGcRQatcIW6hGEJPsH8daRk=")
+ORIGIN("appnexus.com", "ZlSg6m2m27lIX92VkVZFHXs1aayLjp8tBoyfwH/11bI=")
+ORIGIN("v12data.com", "ZpbW94B+SOgv5mRkNXIbY2MnG5TaK80sScRRIWeMsp4=")
+ORIGIN("bid-tag.com", "P4ulhbe1IeRuMjlcZluAXHFYtSz0PaWk76ZJ/pzGjFQ=")
+ORIGIN("clickinc.com", "nJ+L3BudR50yb6HuaNUrJoK17P0bvWB+sqZT5KroZdQ=")
+ORIGIN("afdads.com", "+vp5Yzs7IM1XOC1VdP1xraGeHfyq7JCoBsMS/gV8CTk=")
+ORIGIN("btbuckets.com", "x5JtU/50sZIVMIKmMorAtTdoT6yE1HDfiFUwnKQhhPU=")
+ORIGIN("cognitivematch.com", "ziKo8oeJCeyyVU+xVoXjf2jX27VvvlKgadQqeT8WPE4=")
+ORIGIN("avalanchers.com", "eGOLSecVm5ksVhQjdJpvUrkQr5Ed0sSkvprXJ7AbRD0=")
+ORIGIN("dapper.net", "6CxVmpKEdWaYeBU4qvfbyYLqCRMDwbNcRh10TMsq19U=")
+ORIGIN("networldmedia.net", "KRlqDNv+uaCXsDmahrWHQcMadOP6/tMJrT4wzxd7NrY=")
+ORIGIN("convertro.com", "PLFRTpSjCz8R1H+0ng6T/qlAA+IrlT1L+sGUfNWlriE=")
+ORIGIN("ibsys.com", "y8DIdXX1kcixnIf1T7hXF2d/inOBu4ZGajTJ0K7S0Dw=")
+ORIGIN("iaded.com", "5uVkUjmYmPw1UgCxRVliOru7zUT+pe6yTtjkgf55faA=")
+ORIGIN("aim.com", "r9KkX7rpREZfKvp+XEqfa42rFIBW8TpkIaXE8C9LI4o=")
+ORIGIN("uniqlick.com", "VVLYCKI6MjaEDANBZ5Kbe8C+3LDtijCZ+hEyvFLErOA=")
+ORIGIN("sensic.net", "r6wMZ6WDj/xIqRT7yHUGHPnCPOhNG7+YbGCl2Vn/1vQ=")
+ORIGIN("valuedopinions.co.uk", "576FreIj3ZYNxiI2UgBZtu332vqG836ytq9hn0060H0=")
+ORIGIN("google.com.om", "9b0yq8T61pWJfi+7cmceD8FYfu7QT73RY6Rrorz20xk=")
+ORIGIN("adpepper.us", "mGdGac9aZMkM6HGAskyj0feWsNnPC0XqYM+bKawUvjU=")
+ORIGIN("intentmedia.com", "aryeFrblEh0unR89MI9Pofexmjw991lN87NKA4qmN4o=")
+ORIGIN("zincx.com", "mywmqQRodivUxwsamYCOQCngDiObIwLqMe5JJaKmya8=")
+ORIGIN("63squares.com", "3Ac0sF3gOLvJDCRA4/eKxO3usxdXfxi4+K00rdq419E=")
+ORIGIN("polarmobile.com", "DzKM4PViM8GcD+JoOP8md/uos5+sHD2/TDpLDCtuHt0=")
+ORIGIN("a2dfp.net", "dWzIJxdxMHhHi6nHPRPJgsLdaG95OxpZ7BqVbPOCQ9I=")
+ORIGIN("tapit.com", "nZE8+1LCJ5ntd4jnwyuTY7yuINpQBqANH0AWcZtn04Y=")
+ORIGIN("cubics.com", "FQeoRzACI6A9sMXqjgpylqHbBdSHVT0QPEPNhU0iE3E=")
+ORIGIN("google.co.tz", "OsMBp3zD6/0p0Q/4cyIHTsYEeos9eiWuoewgQ+46C4I=")
+ORIGIN("adservice.google.ca", "/sRtEM3sGLNc2uK8+f3ZLTNGC3eJbfW2fq6SFk8LuzQ=")
+ORIGIN("buysellads.com", "Fdj9N80yBFsl2a3uWCSMt04TzljANzoEkwWEVJaFmfQ=")
+ORIGIN("google.co.th", "PjNhWOpPN9BNVy7Om6huml6EIlOnxT3cbgndLutTWjE=")
+ORIGIN("csi-tracking.com", "4ngLaqk4s9nfIlOWCOq7YV76SH2fhernmqCE0rPLT2I=")
+ORIGIN("seevolution.com", "0eZ/0CvRNPtGcP10am2XeNOczE7q/YMPaK/r6tRkIMM=")
+ORIGIN("adroitinteractive.com", "EEiybYKuHMJ8OHmau6WrImdSKl9o+ZmSGhB+aYBjEas=")
+ORIGIN("google.ws", "nqZM17wkSqlm3jP/vkrVFmYkAQLRyTZ+fz8jGOLGXUE=")
+ORIGIN("vkontakte.ru", "yYjB8fQKsd2LqcluIbH6qQanpm/yJhypZ2EEhL19xD4=")
+ORIGIN("pantherssl.com", "kKbWJnZcLARAXXr+Rw8qfjtvv8qRpYCGXwQ27UfedrU=")
+ORIGIN("creative-serving.com", "8gCMYvAF/tSzGEiRQR9sheK/FZtwLrKU8RdOWy/XYig=")
+ORIGIN("attributionmodel.com", "KEiccRGg1zPPuN5lX7pAegNhTQTJjnX+sBDk6DdssF0=")
+ORIGIN("ero-advertising.com", "C1S/RJBN7+QxjfdM6+2EK2B1iT3iFtIs3UJ/93hbr7I=")
+ORIGIN("integralads.com", "jfzyERbKS9ONoU+jeNV9ponS1N8GvUVgpknOiJ4Sw4k=")
+ORIGIN("netmng.com", "ejWPm9YwLa2j+3gnQeyFNiLAx+7jMUV75q+M7bo9R6M=")
+ORIGIN("rutarget.ru", "6A99vGepWBIGZMkvDL5elkaLY2t1p4irEKf3uyFQZgU=")
+ORIGIN("enecto.com", "3EG0vjyyZw87xjR2+sP3PHNBpjU6eg+dI33dBksC2II=")
+ORIGIN("mookie1.com", "IOW6Ekz4YgwwrNbHMPoDBtwX13/KI3i6G4/Mofw4E9U=")
+ORIGIN("flite.com", "3H1hKHEsZkL7bXrmM4sOaga/6mtazZq+lj4v0g1Z6iA=")
+ORIGIN("buzzlogic.com", "uldFf7XvwRvyhbrXAcZwa0DXce/rscSXpmrO2N82iCQ=")
+ORIGIN("c3tag.com", "2C9W9JqKPfEJxtxzZs69VFP2q2T6QowSnybqnC/vL5I=")
+ORIGIN("campaign-archive1.com", "VIiFN/Y2Hjkvg7eVEdXQjv3/HHu93VJSWDDtWts5sbk=")
+ORIGIN("brandcrumb.com", "6UcuY9uzDrOUcj7MjpPsITvcYfyeKiCAsnBM7EU37uk=")
+ORIGIN("oberon-media.com", "ZBsO0hpsU7SjPAjbx2fBm4JTnMqHcNRwaRbmqqmgG4g=")
+ORIGIN("futureads.com", "FB36V57yJbJRA0XX6tJ1C2Mnr+GI/wvLBIqxuwl7rRI=")
+ORIGIN("ratevoice.com", "m1QdeUYXqSYlDLNB0K35pSQUGHBeyVLRl6wLrB0E0Io=")
+ORIGIN("jobthread.com", "lt2kIRh5mKuDx8EmkaJQwTi98KQsb9sQ62aqMJb0X84=")
+ORIGIN("contextuads.com", "+psLidb/2EmAl76vOjG6hHI/WDwIKpOqE76JXJ4GJJc=")
+ORIGIN("exoclick.com", "oqDgdw6g3nqAAXbmtqYFMgBdmzsrvE5VKnCxR5WPkb8=")
+ORIGIN("pjatr.com", "KZMDNouN9anyyVSviOzQHpLb8uE6JcMO80hYIiYkNfU=")
+ORIGIN("sparkstudios.com", "UzZaGWHSDW9ypbyp5Nki1nxCU6igL02ixm07I3CoaAs=")
+ORIGIN("coxdigitalsolutions.com", "HjiDaCtdOqaNcFQZBU2EHiKv6JKSI7W8iMNU1AhmSNE=")
+ORIGIN("chango.com", "AB/CeisT06/NbY71sI5TpA01wcY2tRZI4WXSgd9C6vY=")
+ORIGIN("sagemetrics.com", "afKUC52VQeoCCqsqLxviIuzPDg68tyG1xQEs37z5MSI=")
+ORIGIN("glam.com", "V890DZ6azY2uH46yDmW5pkOfzIqEQ7MD1rbmLz/ZNPE=")
+ORIGIN("cnzz.com", "gqvJZLE7HvtBduKYJDVL0aIQjsclVvFHF+3WMPy4IcM=")
+ORIGIN("xtendmedia.com", "o1/BlBZfg77yvucPscObiNFiPvyWEX3IZMrsOm0572o=")
+ORIGIN("clicktale.com", "ZMHO54G+D7dJky0hTtFzSWTCIdSoqYKHcrNqZiNrn5k=")
+ORIGIN("bouncex.com", "jW59n05tCRUMldpoJD2HTElcx4Ass7aSwdtua6amVDU=")
+ORIGIN("sitemeter.com", "v9yPP1Aetre4daHcGiu0moYGLGLWod5l3FK7w/0yGaY=")
+ORIGIN("rightmedia.com", "IdNcB9TeoTwAKPo1rM/z5YCMBv588CQ3poGh1j57KDM=")
+ORIGIN("excitad.com", "+i4qSDCs7C6dnWCH7AQB+RjZa974EmULJVyWQ4yGqZg=")
+ORIGIN("keymetric.net", "krK/aZhO4oGdmIG6eHQXddsgoftGO6HMWGpF9w6UKMg=")
+ORIGIN("4info.com", "ReB3bdUueaplUpDtdy0EPG+EQtTu2vakjMDhhM0TfWc=")
+ORIGIN("yieldmo.com", "hlHpyexEgydpFvFtL6drZa5//r+1Fr5bGIgVA1SbVGo=")
+ORIGIN("omnicomgroup.com", "VGCgjSICH4Zn28kGfiGfFQRLAjNJm9iclR3z8D/4tMA=")
+ORIGIN("clickfuse.com", "TvqeQ0JokyvASz8f8oXJvIQi7HDMZcAx0h/kGphEqCE=")
+ORIGIN("tensquare.com", "NkRkcabAtWIwQbJZFIpV2C0PedHrOCe7hF6C5jLNM6c=")
+ORIGIN("syncapse.com", "CaW9comwyn7az8C6Ea56pYuBMeqI+b9LFsrCrDQJjD8=")
+ORIGIN("monster.com", "tgCqaFrYaLaxkVSpBEPY/aoVrnFjDb99X8zzHdy/MOE=")
+ORIGIN("adf.ly", "2vijsLchTBjilW7zkxIt0J/dFJuy+H4ClGNkVNcdIAk=")
+ORIGIN("observerapp.com", "r6qaWoGn1fMVvXSUrgPmejC5SAlazITfy0dH4kWWQpc=")
+ORIGIN("precisionclick.com", "vAD+oMzJ41OR5731JmpBL56EzQmhJrZLmKe2R6MCPao=")
+ORIGIN("google.es", "vtJn5KltjScFTJ99FPIgf9eTZHWF6KgULdhVMm4xMK8=")
+ORIGIN("themig.com", "B+fIakNg0qxnR1T8MNQOg2+POyRDoEYHajXco/59fnw=")
+ORIGIN("maps.google.com", "C1mkefBpsxsfN65HxyBO5h3ccSqu9NVHHwXupY++CJU=")
+ORIGIN("tellapt.com", "OR8bdbSWvenf92OCJ7UAHEUhm7tXMBAHJxLTDkXp7JM=")
+ORIGIN("fluct.jp", "UQk5cT1zgIey7FamHqkO/bSsw8+hujTNbJumjN2HlYc=")
+ORIGIN("google.ee", "BIjZakI3ChrLy6S7OTUampyXO+WUJFp4dYEfvXDQTtc=")
+ORIGIN("branica.com", "FToe9BNJgjRJtRbW90WedqLog4lTu2nBzArWyL3QlJI=")
+ORIGIN("mediashakers.com", "COlBtDziXmW11S8sXXP/4lc5nHpd9KRGpDPvw8dWASQ=")
+ORIGIN("targetingmarketplace.com", "65SSNzFz4JI9NXzRlTHSvHqYrHH0iNISX9OeM2SrWOQ=")
+ORIGIN("vemba.com", "BpIyozDkb8KupVJVWm4KijPYelKqlRAIdJ0f8Fb9+SM=")
+ORIGIN("whiteops.com", "gZ5i5o3SPjOPSa81ICS4is2PksHCemz1Wmey4bOcKNQ=")
+ORIGIN("akamai.com", "6T1MRyXgMODdmg2HLpWNQ9AOGS5W+0GKYlFpPx7lmDM=")
+ORIGIN("isocket.com", "mE9TvU6T+Erh+89tD93xCNeResD6eDUgahW2LJWl5pI=")
+ORIGIN("cetrk.com", "/Gkfd+5K41Wug5HPrDkXaHulsW0F7TwgpXx40IUOcA0=")
+ORIGIN("clustrmaps.com", "N1vrG+vOj/28ik/2cXMnYaSSLRs8LHBE+ZeUOUQC9Xg=")
+ORIGIN("yume.com", "SmoI3OtvYtvm2JTJGN/Bu4cQSuL9fC5/AFUs4VvC0O4=")
+ORIGIN("anormal-media.de", "uqMvjGmAIyWQ4VNDv0BruRgUyMJC+/E3Nd9ZxQrLfB0=")
+ORIGIN("gocampaignlive.com", "I1iOtQMGOjWh88D8YIYbdUEQtxmlT1e/gipFqt1iSIw=")
+ORIGIN("tlvmedia.com", "FyIc54NMi40Y1qPduUURK0oVR95jZsB9fH6CTOleqHA=")
+ORIGIN("clearsightinteractive.com", "p8yKdWMurgxes7oEVd9rsqJYJM+8qKNRqcrGzROnjVM=")
+ORIGIN("eyewonder.com", "GLt5nHEaCPNA+UKNZJl0QlcGytkQBxguWjw51VgD1Nw=")
+ORIGIN("zaful.com", "6SIh7TMO5PxtODq2nWR3FljrJu1x4PAQGxbVcGVPjAo=")
+ORIGIN("health.google.com", "YMXGsyp800IvpySasfnGKuxPJ9Z1ZDLoc2VBayNakcs=")
+ORIGIN("zaparena.com", "0G4gryDrvXxJ7BhebTFpOLvqoLhYyW5Zp7TzHqNaIlM=")
+ORIGIN("adtiger.de", "2ZOmahXP4x/soUCHBVapGRE5TfMSib0XyKv81PVU/yQ=")
+ORIGIN("homesessive.com", "xZu4N7c8jTy+2+m5P6jzxWIlntRk3QHMg748CE2gS7Y=")
+ORIGIN("adsummos.com", "ucbwCCsY6D/DWb2CQm8O22+jP1uqc8SMkXJc3qFkXpU=")
+ORIGIN("apnewsregistry.com", "HTR3YiuYgtJ+vjUyH/6gLSy+3DfH0dXqHlNGupq0Wuw=")
+ORIGIN("collective.com", "+NMdHgQwS+wfHHeo8gZQmTphUPJuj2vAxwilL8Uc/6o=")
+ORIGIN("adnxs.com", "7nX5mLXDcidEaboQwV1VJyYzNCBTtkA81iuECMXY5Fo=")
+ORIGIN("google.co.cr", "RKmmbs4vJGvqaXg6S3aF3fjvtOfh0I8c8NuqmtM4r5k=")
+ORIGIN("mashlogic.com", "bT6rtBUhcgNQzE28jxPiKLxQYD5nJ7KcSxY+n2LMqag=")
+ORIGIN("optim.al", "NwOUOqcy0tsYdaCPGumud/1R17u7vgGdMroneAyOLkw=")
+ORIGIN("visitorville.com", "MQ6A5Y8R6v4tH4rbMUp740c+fll/CqW7XHesKHNAKHo=")
+ORIGIN("adaptiveads.com", "kZxDtr9Iu9860NIni3vwiTSWCrx9O+Q1l6JvNzaGQmk=")
+ORIGIN("resolutionmedia.com", "8TSL9+24iYsHqJWYpakAMZnduMlZcYSDbItU8vTi4TA=")
+ORIGIN("adgrx.com", "qJKRlvyzSSIjYGsF+C/gWZ9mg2qPT6kp5EHx1X6JrSw=")
+ORIGIN("netflame.cc", "+MOYz0h/b0UMcFybuDwhyfIuZUYRCPnM5L38/UpS+zE=")
+ORIGIN("amung.us", "hFwKFPFDX2/9FJX1uDlbHm4Fe1D3yZmbWY1xmli4C3c=")
+ORIGIN("adclickmedia.com", "nbOsjSMTpJGDwIe/4/ECERHCj9YOfV6JSNqcoXy/sUs=")
+ORIGIN("merchenta.com", "g+NZrL8wMAL9aqoxz+fMREJmbKAHSmuTflB8ePdQmlE=")
+ORIGIN("techsolutions.com.tw", "ADdun7GzDX48lmc5NIiLgmbM6+Kj9lyGfgXgc1vvc4k=")
+ORIGIN("spacechimpmedia.com", "6HaL2hYUffSktL9oV+3ivRmRNMqtX4LOkoPxlg7jnrU=")
+ORIGIN("adleave.com", "FIwRZheNEmwHiPX9o2W3z9sicsmOWoa9Dvm5TmnKTUA=")
+ORIGIN("adloox.com", "YwePt91Q8kvuUGlIl3OftlE/PdZnQ8KWF2ioAKeVRp0=")
+ORIGIN("wsod.com", "Wbds8Pry37Rzid6n/Ae9sQSA3NJ/2QYmhJeYA5IuQTk=")
+ORIGIN("exitjunction.com", "cdHutr9A+uhif+V3a4zU7LodKQ7iJqyFP1zAZChCcjs=")
+ORIGIN("burstmedia.com", "xCXgwrmHIA24wa2uImPfFo03XtjeI6BSm77UM7LbQtA=")
+ORIGIN("qksz.com", "nv1dgCC+C7XXzOSfrUh0L/F7vO3DqAImxAY8qcWr4iI=")
+ORIGIN("medianet.com", "mFJR5mBMKGXEFe7VT4gpyBGMlSfGemSkj5ZPSknxFi0=")
+ORIGIN("meetic-partners.com", "ThgIEpaFJSbHTTc6BiY+Ql5R/wy8pQo4prKs8FCS4IE=")
+ORIGIN("adservice.google.com", "ORbzeOvsHcz/DEuwyZXenmorqa79AoIdfjEXXKIxKIY=")
+ORIGIN("aweber.com", "yS05ryjdIMp62z//tNjawBngAYVXAzukDN+dgqUdGsI=")
+ORIGIN("plusone.google.com", "wuwBibFAMty+38JOtk6FSMny3KVthMu4qneskVIlHsg=")
+ORIGIN("appenda.com", "aL1Yt4LK9rAL7jYQi30tSQjJBKYs6C1zZXuAV5J/c8U=")
+ORIGIN("dotomi.com", "QI/FeeR3ClhaV5p9a7t+N1FwXNY4XU19AKSvO4CsSnA=")
+ORIGIN("google.be", "ninu4S6jmuJLUzNcVcaQNegSM3CcHnyEVWrJP/2l+Ts=")
+ORIGIN("mythingsmedia.com", "KUuS6yDDFrmUb76MzPb4aXO8vlFkyFrxkmyBnpAmTIA=")
+ORIGIN("convertglobal.com", "2lDkaFtMu4slWg82xvdmqrZ7Kde8vB4VR/NhgDQ7NPk=")
+ORIGIN("thesearchagency.com", "6u/xtg4AUhAUjfs2Y55TLePN8ML0GtSgEgxqrO/QWKs=")
+ORIGIN("inadco.com", "DWrLv7cj04P7JgFPALFLDeJEYQNRrh9fXOcCAhSwkEw=")
+ORIGIN("rekko.com", "B3enJR58B0tg0RnROi02A2zPTt++gETMUHQrunZ4LP8=")
+ORIGIN("info.yahoo.com", "Bo0Ips4yD7Mo9paKI9ZmjM1QNZfZXJu1IiCdJZjr3vo=")
+ORIGIN("cptgt.com", "gayCCzQq9VTbfda3XNTVKvsTtZ70aMxJjRjmz2WIkcA=")
+ORIGIN("durasite.net", "jhwXAQ9zduqoyBWZUQ97VzQz7kdqdosQOSINMJ3OSlA=")
+ORIGIN("acceleratorusa.com", "s818ajKTF+HJkSUa+A+9NiiNUFC1nh2yoX5b1y9VuKA=")
+ORIGIN("instantservice.com", "iR6sIYVjMYoMl0LrX1craxzR5kwXAeXCG5qnrqm5w5s=")
+ORIGIN("google.com.lb", "QRqWnfgnKv7q1BHLbASqAmV+GYAbO1a2IkR5GIy8yV4=")
+ORIGIN("sociomantic.com", "yiruXvD5/RdaeA4L4qiVye5KK+dS/sJdCsArvnc2ek0=")
+ORIGIN("oridian.com", "gucHgioh6QQn0YvWjcsAqjz2wcPGqJOwlE6HqY4eTqs=")
+ORIGIN("delivr.com", "gMMld57FUKXhJFE7aOBDJuXu1FeE7+Ol3710ErsSgQY=")
+ORIGIN("video.google.com", "W5Hu3W8FRu+TXUGv3+JGZixp+VYu+Mc88jxi1zlSioM=")
+ORIGIN("targetix.net", "II0h7dl15K4NjrYoYtwIystgZoCtiYnDKLanFv9NK/o=")
+ORIGIN("edit.yahoo.com", "SFsNdsCUL2ddjCOFIsqZgO0AEemrig4oLZYQ4b60qQ4=")
+ORIGIN("abmr.net", "vyzHmI2ZXlyORjLWyWe5Nu5A18ps8WAHua/k9orgxSk=")
+ORIGIN("getglue.com", "XrKgzCMU+SIsxn559X4/oXzm29RtdXKUjaJmF3oA87k=")
+ORIGIN("w3roi.com", "slv3XYfhEYMozv2NjRwqQttYj/704l8ZOWEy1ea/ZfY=")
+ORIGIN("google.co.ck", "wtAUUILgVGy7bNn4bBuRS+Sl2aQ8TkfaQAbDtuco89s=")
+ORIGIN("adforgeinc.com", "LC90sUPTTfQsEZeYknzPpaGZbFm+W/fM7VZub3nOmaA=")
+ORIGIN("jink.de", "NkYIIACtTFZyM7SYh+KixyVRi9yUaw/XS3eaX92OwwU=")
+ORIGIN("newstogram.com", "wuvbG2khEdIyTMTlY8o0CeMJwTVRGHPknszbyIPViAE=")
+ORIGIN("cedexis.net", "JUrkmXu81hrfXEPf/+Xo6iQuZYCGrUy5ai1WDVbwqhE=")
+ORIGIN("yceml.net", "HkTU8NGjJrTniwO73t2KkVXP/tU/Xn8zAPxEmohPx14=")
+ORIGIN("dianomi.com", "8XmsGH9qbQlCnsBAA9qnuQ/SYugR/0eOoUWc5UTNzc0=")
+ORIGIN("dynadmic.com", "jZ3Vbcb0eV8vIFo6vhCJpMEG8spAvAEQGECEhGK+HOE=")
+ORIGIN("traffichaus.com", "rnw1MC2i4AkbvQ2/cy5Kk6jR0rUX01ADVmmRyDpKWik=")
+ORIGIN("gtopstats.com", "9ktW+Ikq9Z8KQ5EH83jX7GRim9/JS03IYHiBM1Y77Bo=")
+ORIGIN("dwin1.com", "IibMkEU+23MPkOzcsystn0F63nIaDyv8PFOlLwz8Lqs=")
+ORIGIN("oggifinogi.com", "mBAJHuVCnpQSrcwwVK5iqtHVO0wkEEEnPbWHl3XotsA=")
+ORIGIN("adviva.co.uk", "WGv/0dK/pV5I6eaUqtwHv3P+uhiKuLtzQPphA1J3qJs=")
+ORIGIN("silverpop.com", "+OotVnE7ElacHZxRR05KRcWvrCJ8q4t7jqph1qUK0T8=")
+ORIGIN("google.vg", "ftG7d3FYjsBSdhGyNBFrziNEVr6uAyXbuWesfDnHNgQ=")
+ORIGIN("yieldlab.de", "mrIdzNV/L8m7QVYzwBqClrkNrr+lXPfWizMoL4i7tcU=")
+ORIGIN("exponential.com", "V8s4EPGzy33Nkcx1LOr2qg+XMDNrp/bTEmPFzYX/iDY=")
+ORIGIN("tweetdeck.com", "dTioGUQ9RAZ5oJTeK2YoezH28GBqslprT398ADJsteQ=")
+ORIGIN("interpolls.com", "vAFbOgHR7xQDDz1p+O38bc11ecge/fEwA1Ew1B+U/Zk=")
+ORIGIN("eproof.com", "DfB6Llyile5CZNjpWqog84WzrhnWpQs9aGfnh1knJDk=")
+ORIGIN("powerlinks.com", "1f2+BYGoCr8D7Gk3ZDhxuGpAxqZFc+3gWTNE7P2QQGM=")
+ORIGIN("tidaltv.com", "1H1Au8eOuQS6Rmxse51Uyn9WIosmoTkCIoj07vpwXAg=")
+ORIGIN("investor.google.com", "kUUYwhw+1LLCGgW2YkZcaSNrkNp07rF549uTxVWZTeA=")
+ORIGIN("propellerads.com", "jItVlgf2s4ZvcRg6g+wXvRPvmrgi+P+C9K0pVy84fzM=")
+ORIGIN("exosrv.com", "z74xyytEZKRMU+v2qgfOBP2lvJVAfeI9dTDZwjKB62k=")
+ORIGIN("networkedblogs.com", "exGHPpgFgmutZ6vZK3GMn7yI4mjXMfqHzvBG2OHDSm0=")
+ORIGIN("mojiva.com", "Sr9+fmbj35lZpWx5vxhB7mls5UhuBk6GjvcAS1Q9rpo=")
+ORIGIN("orkut.com", "fJBdx7C+mncegbGgoX03EMXrKSbFgBlPoho2LajAnW4=")
+ORIGIN("wordstream.com", "Z74VG2rzUNqnYEKru/0KdVwxAyZy5wkajsdhdXgQ748=")
+ORIGIN("adimg.net", "s/tzyGdSeKO0uAfeLVS3gMi/ssyXMBUtCG9fRB7y8Pk=")
+ORIGIN("kantarmedia.com", "oPcp2qlgSxdI07s0UtjKTvpNUTcxih3hI4wv5wjlqi0=")
+ORIGIN("ib-ibi.com", "NyQ50hZGemE9pNd0mXHTItjHe6/MaqaDoJv27wFYRE0=")
+ORIGIN("specificmedia.com", "ZDTdQnS2i5Gc0Vs9epWe3h0/nS+ues0Vb92vQDA1aS4=")
+ORIGIN("eyeota.net", "Qx+Le3QORoglsURbdYVKBKwsQJatAZUXmKmS6wMp/OA=")
+ORIGIN("compete.com", "Uy2ILN0FIX96fyZVQy2yoSxAITZQBEs508g9ZpDt5w8=")
+ORIGIN("company-target.com", "KjV2twYHLQMs+YfZzTPDpZG2L00z+Sq9/54PhwHJBek=")
+ORIGIN("buzzfed.com", "SkWhkKsjZmgxk8TXa/saoVg5Y+M+jiRXwrhZ6tLSX+o=")
+ORIGIN("directresponsegroup.com", "cmQaWu2FqCnmDOOlQ0FxACJ+tCoxHQ1VoZCzT1wgyBM=")
+ORIGIN("matomy.com", "QJnmIkFNiWDJ0rUeYHWyNtzeSEQHIb5vTkvQR6HgZZY=")
+ORIGIN("wahoha.com", "0S97413t9NSABQJyiAMQuEIujWeJtrVMVazqPTttTsc=")
+ORIGIN("syndigonetworks.com", "AFlvi8VWr4x9QjHPMWvRENn8uYbynhERb1Omi895JZI=")
+ORIGIN("nxtck.com", "t9ee8RI0Zxx43hY7ANMSBKuhxv6itI3efl6K8J3KUl4=")
+ORIGIN("cardlytics.com", "KrVgrlR76dF1CT3WUiPrMNdahbbd7/LHl2JtGr/wAjw=")
+ORIGIN("addgloo.com", "7lPH/t6gzZWmDJhjvE3IfkS2LxdzOLcZswNq9rtkh24=")
+ORIGIN("travoramedia.com", "6ITWPFThdCQ03ZyrVCg7Ssx2jJWlsw55fLppDROxk3I=")
+ORIGIN("qnsr.com", "6sgOt0L6+8i5QUa37sMGv5PwIgs/jya7ecw3ekYny9E=")
+ORIGIN("baynote.net", "E88yOcJMaDDrW5NDvsPi23zvKxXiWEDkrbIvqrdBCWM=")
+ORIGIN("inflectionpointmedia.com", "+hsR2GJaVU8MvB8zEwopimEvhsk5OH2CX4mNxYIcYxU=")
+ORIGIN("assoc-amazon.com", "aWX8MjAVx8cfGADC4OL59TOxMNUEQfOXMkFR3quDBu0=")
+ORIGIN("msn.com", "AAykEU507YSZKkhPNYjPKqbsFiCmHFVNteU3PLal2Ac=")
+ORIGIN("videos.google.com", "nxXfqLB7LP1zVYA9J4I2xxSr4tQ3jZJTBKUSn/k5V4w=")
+ORIGIN("responsys.com", "/6MhwRDCWiqA7+YsO0EdFEZj90o+oZKXRYGafYoWNjE=")
+ORIGIN("google.no", "8xD4clQg8QKeANlArT7b7VeGj9ZeN47PoOjMwoTwGPc=")
+ORIGIN("aggregateknowledge.com", "YoU8LOeWsCyDIuymFhN4SfPyWqNZMLooB4e0avXX39c=")
+ORIGIN("vibrantmedia.com", "xE/aFEPfW6nY7HWw5s61xuMc89heSgqK6KexaLsI9zU=")
+ORIGIN("moolah-media.com", "GDvnfDahH1vKo03GrYeyTlmZpW4JteHCx7wq1yHXYzU=")
+ORIGIN("bidsystem.com", "Zg2f7+Jpj1kL7mkBWUtEoDnM/OoyjiBLfTR/JnpUsz0=")
+ORIGIN("2mdn.net", "Xm63aZSDF1ukZYkajq/x3lUgU5C1xt4RyVhOt7cnvDo=")
+ORIGIN("netmining.com", "b5iS89MEV6o6OMSmkhkj3L9dbBzljosVl/hWXWnESoc=")
+ORIGIN("fmpub.net", "DUS9whCJEJOacADOoeklwJF2ROAcnjem9IikEyxENbs=")
+ORIGIN("behavioralengine.com", "EeJ9yRv28ofJwAyEs6Z3U/eraZkKMGiHDkPiCgivLv0=")
+ORIGIN("certifica.com", "Ivfe7hBpzf1CjTPMqc1N2Rt5qex8JdugReZ+2bLVIKA=")
+ORIGIN("monoloop.com", "WgXQelAlaDRilUq8Ktz5BviYmxtPoiSQ/JWB8dJ7Uc4=")
+ORIGIN("google.dz", "EAGO3NhlV4jHzTEVauNpdFp/+HsGuRlyughsh/vZr50=")
+ORIGIN("free-pagerank.com", "hBe8ExTpOAQsH5KG2sW7WgVdIWMJRnSIGUiV/v/h7dM=")
+ORIGIN("cj.com", "qAxk1Az4tihti7K4v79PUklG6u4/rG1kNZQ0KBMnwuY=")
+ORIGIN("buzzparadise.com", "L7qE98Fc3M50Tg6pVA2COkGf5QHIYgd3cleALOuNwcs=")
+ORIGIN("googleartproject.com", "yw3kZEgeAJBX9gqaQ1fac+cRWJApJ9pBAuhAARyoevc=")
+ORIGIN("google.dm", "cjlWsTc4OZ84VGdTfTUhpT0+0KmtInDgI82DsABBhnw=")
+ORIGIN("wiredminds.de", "PO/EhzizxyioCXnZBs/TWAWhHfkiECldcEG+2vbz4TA=")
+ORIGIN("acuityplatform.com", "avZn0aQTb0iugZL7/gE6RtUhZn013qCcANRW2PPq6/k=")
+ORIGIN("google.dj", "o+6bIEWA35eIfxii6xWr5nc4BI8T8xvMRQR743o75rI=")
+ORIGIN("prismapp.io", "7s0BLFOeKLJx0CpCJXujTjzTHOHORwIuEQbX8s937aY=")
+ORIGIN("google.de", "el+7VCp9VgG2y446iq39UVIh2HdbH2A5X7Kn4RF34FE=")
+ORIGIN("halogennetwork.com", "DkHBwJFE7sVymwJXKi2H+2irLAzqq44TGCL/Js5RYkw=")
+ORIGIN("eyeconomy.com", "z903DSOw0OaUHh/ui85Rktxqgv27lUdquSV8BSgaZOk=")
+ORIGIN("yieldoptimizer.com", "e8GWz2rRGg2M9eqyo5L69djEraOg0nCaxzMYEi8CYgo=")
+ORIGIN("cmmeglobal.com", "R902VTDO+CRI08Q5JtGDvk/7Ap/4HLwVV/BZMR45N6s=")
+ORIGIN("eyeconomy.co.uk", "iN2kM+YSYmT5kLq3lubVlyUjueLpp8r0LKbH3zuZdn8=")
+ORIGIN("adbrite.com", "Dv79wiOaEEoCyci4y9DlIKtqdKP//LS0pjgWYPz5uAE=")
+ORIGIN("spinner.com", "y5uXjYshCekA8SeyRtaFY1NJ+KbmQPiYh0cHa+m7PMY=")
+ORIGIN("mailchimp.com", "MevRnR2fKlEzfoAUdYC7Bxq55gNheFNpPfu55s4UnE8=")
+ORIGIN("beaconads.com", "5n06TfOg55vixWVdtp+J9VSevdh/idDPwk3IoC1AOMg=")
+ORIGIN("adspirit.net", "66dCq7cjFR+91+ufbJ74Vcmr8KdhSquAA9P32QxxVqk=")
+ORIGIN("relevad.com", "XDJVLdrzlAKLE5rfA1UGJeWJKnJo8/Tu15XUreUQmYg=")
+ORIGIN("goldspotmedia.com", "ZiroH7RovDl3+pG+qtlOJ6n6pVIqBsl9wKd2Tbv3Arc=")
+ORIGIN("cyberplex.com", "N69Pbmc4mAXqaqyIszUqYAshGBgBHoRxeDmiU2SzzN4=")
+ORIGIN("moikrug.ru", "og7wP8NQyhdf4sDARDrMKj5uVHXxbdofbjxVoKYQcHE=")
+ORIGIN("communicatorcorp.com", "+TD3trr49nslzAi+I1g0MSRYkVJDBJvG13qTCQxWMPs=")
+ORIGIN("vk.com", "eabQm5mUr2rpvbyT5RIdK1uOTPvowVajgQ1P7YsZWiw=")
+ORIGIN("adinterax.com", "xVL30k9nbAb/vpCK58PcU9uFUKEvvP2lv8tesu3Tcmw=")
+ORIGIN("inuvo.com", "+3lUokO7m3bIwK27shMFHRSkYtO+bVniQBaBVkGcI1k=")
+ORIGIN("horyzon-media.com", "gciBQnT+9UwUuU0ulv4s8QPmRgpSsU9GcGtb66nXlCQ=")
+ORIGIN("unrulymedia.com", "nhVjDUXKrsSB0kYHdlqhbCYLXa9NU7yHjgc5EqErJTQ=")
+ORIGIN("optinmonster.com", "kKv7PNjOfP5x8LsPgejqbc8VOUY0/foEf++cxx3Oub0=")
+ORIGIN("webads.co.uk", "xwQQRiuCyl+vnN9JSsVL+/PmLlvT4Yw7x1udTC44j1Q=")
+ORIGIN("pswec.com", "7ai7FE6WnxFcMN7FBOSChq2L8UcvRO1OHeEXpXJhwXc=")
+ORIGIN("pontiflex.com", "hpJheNxFkI7SEMpl3J3KO5m5jAVJTf7fOhC7uRurxPc=")
+ORIGIN("pntra.com", "T3nk2SfwvX7V+rdlpmZ7/nRP/AAR/KydLdMZ7tfJ4rI=")
+ORIGIN("struq.com", "yvPLa2Z8nHyS6XoNHDWtUqkAER/M1gY1i4DjPDWH1p0=")
+ORIGIN("begun.ru", "ZnjupIH4YpDpokN0YElocIRDE9g741ydO4kXGPhitOI=")
+ORIGIN("unbounce.com", "xbu1EDLTG3nusOA85Y94fpKvPU8HBjLN7ZvqoCeeVf8=")
+ORIGIN("insightexpressai.com", "Z95DPWMqBITUVD1WeAFDcUjWdLRrUqXudOtJ2pKKAcw=")
+ORIGIN("admanager-xertive.com", "x95d5OHmhJdNzg1mgFfSwzmkgLcLSbO9HfGqkAiHOdo=")
+ORIGIN("facebook.fr", "UsBY6V/I0OUbud1LcvE2SqRxFXR1qENdqnHo4clTNhU=")
+ORIGIN("google.nu", "Hds0/Ouo5rzSVcBhyjsGrDtUXO7UOfyuzfkoJhYWpy0=")
+ORIGIN("valuead.com", "stAQXq+jfAZzfKFwUl5buXFLG8/A6mqPKsGHzaeHN+4=")
+ORIGIN("rundsp.com", "+GmB4+M4A7eAYiJYhKmaqBOwREu6GUfNHMJY0qyxi9Y=")
+ORIGIN("pictela.com", "Ki4//aXH7ZpjYC8Sw8fn8Ylu2mWgqlDgkiHVHIyFrCQ=")
+ORIGIN("google.co.ke", "3tutpCCz7658D7vFuXCN0+2usW/BVBj7CyF83PdbJBk=")
+ORIGIN("intergi.com", "ltPbxjny1bWPIR+DQicULsv2itmzBGj2dnK16qNKfGs=")
+ORIGIN("adgainersolutions.com", "Mm93fXFyrBh7wNtPYPV+kPCVrg9eppbByTKau95sA50=")
+ORIGIN("atoomic.com", "lDAUC8EjBglx/2Wnz4PFW4FbM1BumUj8VVf7l1yclPM=")
+ORIGIN("netaffiliation.com", "zdZBW5aRNh+Y32U0DArUeUopXCOjKnwfFX+Tq5PYsTY=")
+ORIGIN("google.co.kr", "QPYvbD3Gl58ABVG11qbzBIuL5HvKe26KBOg1dbZL6j0=")
+ORIGIN("tradetracker.net", "W8T4XUVer+7ItOZKRfZR3n5wE4P8jdaRO3n6egZfWsQ=")
+ORIGIN("pop6.com", "EG6wPtvfCk9sAg7Sm1K+8mihKPwFuJ01FFzReqmR/4A=")
+ORIGIN("blogads.com", "ZTZghw5egl++jCUHL3u6zE/kUti26YyLAo9DtdIEeEg=")
+ORIGIN("extend.tv", "cVjW2sDgOVZ5aX8Pj0E2JZpetO6ewTaKqBvhaRHsyEI=")
+ORIGIN("multiplestreammktg.com", "0mkF1IPDx98sNSitZxc0mnSFxlFfKush1tknM09pcmk=")
+ORIGIN("adglare.net", "4bNqF9W9vvGSx9pbm9MCdxFfNoMWDkBNZhIEE0SOHL4=")
+ORIGIN("investingchannel.com", "JlMfJ3cBmwIxFLDxJ5wzD/oKqOGqCO6fQEvIKLdF9sQ=")
+ORIGIN("adsafemedia.com", "nvKTY1o7j9ul1XcE4WHV5X/tIkDEv1LozrOIZR2fziQ=")
+ORIGIN("dialogmgr.com", "f9Crde46FORcW096gWHgUK38PqcMMhwRKOHXfsEsitM=")
+ORIGIN("flattr.com", "2iy4wCeDgx98ElRMy0z4+jviMfVQbRd+ozNlDuasV3I=")
+ORIGIN("blogcounter.de", "CFVHm6gYiqt1pZ/t5lKBHrLh3ZbLfv3hEq4tjZUdTW0=")
+ORIGIN("reinvigorate.net", "wobQXpJBUZZIUVZ46+mnQpN2IF/cL0lt2oIym2UrO0g=")
+ORIGIN("rim.com", "qRd7xU3/Anp4mFlQ60ov7q4UJDw6LC3mkZ32KWBBeHQ=")
+ORIGIN("mcafee.com", "CuD3bBd0aQwXcGbYFMAD0hYuKDs6wlQM4N3C2kIScWg=")
+ORIGIN("provers.pro", "V9TfRjTyEdRmxTH9MY1bO2tF6peO6GkWqAq5aWDL2/g=")
+ORIGIN("enginenetwork.com", "byKK4qD0uHBh4qBOTQK2MIY4fOsg2tML1A5BT2BpMQo=")
+ORIGIN("blvdstatus.com", "1xo1IDqEoVsV3mx2fBsG2wcY0u9syb4VH/3ok/Pe4/A=")
+ORIGIN("viewablemedia.net", "koXY7NosiOoUAx5Z/+tn5/Fp/8M90/9ZzaBTnUVG+cs=")
+ORIGIN("admailtiser.com", "s3TK5N5rafzoU3ZI7pYvF/VyyjxPvmQRXiFdgKA2Iqs=")
+ORIGIN("microsoftstore.com", "AVah/TypPj6DtouZcrgFSW4rvq3tj9x2BCA6TNh++b4=")
+ORIGIN("ohana-media.com", "LS1Z2FSrt7tZlCpcutN4r9IYc4XoMerz3x1ClMuEK0w=")
+ORIGIN("moatads.com", "+Iry112ma4fPUyBnC9m5VRHJDm7/CLl+iZPxazw1hpQ=")
+ORIGIN("perimeterx.net", "+sWqJO2JjmRTblobsTYvLKE0e8CYOoiLx83vnzHnlyo=")
+ORIGIN("optify.net", "D7rlergC8Eq+7C/J4Or2JbYSVaHTUdIjncUIyh1+coM=")
+ORIGIN("adyoulike.com", "KbKDmYoOFvjcVOAClaC403siqt7N5faogZzi9UJcfgE=")
+ORIGIN("dsmmadvantage.com", "Sjw1F37vTEevqCoDntjYojavYr8Q3Ka/uN2qFpdcbSE=")
+ORIGIN("accounts.google.com", "R5E6a1Ye1oBysmYUL4oSh4XF508Epehc5ERLpA1LXoY=")
+ORIGIN("4dsply.com", "SVAL/V7uBm7x9aBvuZ7nnho6a+bYeI4Krr3JcyibgaY=")
+ORIGIN("adv-adserver.com", "1yy7SGQ0VQEdrVeYBvKq5DH7OSakzAPYfoFPl7czDzU=")
+ORIGIN("buzzster.com", "WyNdsSbzsCgAkegzDubqtwsEXMs7Ww0S6ukaftaQwic=")
+ORIGIN("xgraph.net", "Wsb7ERuF7SYHos2li9GUKcKJFW9JqVAr6q0K1JKJLA8=")
+ORIGIN("destinationurl.com", "uAosIftniKbWBC7pBqxuMSd+Q6dimVCwsZVOJ4f009s=")
+ORIGIN("buy.at", "4x0LqPhjvnxYz+UJ0EJcl9gVAK7QCfso1Wf3ybuE1Rs=")
+ORIGIN("engago.com", "kt4mrpbteg8SsGoxtPnDo5YE26UKOZcEx4iOmdZzAQo=")
+ORIGIN("atgsvcs.com", "9WzcUrv3jGBodbxGmBmMox+gRcmkgrm3B+2zw0e6FBg=")
+ORIGIN("proclivitymedia.com", "IlzxSST84DmUrSRzFfylznj7Zy+eQDQ5ZrkfX2zaypI=")
+ORIGIN("brainient.com", "0lt5A7Arcy0SDog3V5njA34wm/9LSP5rgVBPCQQTt3M=")
+ORIGIN("augme.com", "v4Zp+LUJ9n086cejquiujxpaBlXBWZzr0lTWwnrCaEM=")
+ORIGIN("foxnetworks.com", "IsyCxhA8Z09c5WUVFmj+RZyLEO6fbg91ls6g3ovbB1w=")
+ORIGIN("adnium.com", "uMm+kgNfG5714SAuQSXb75KjmgHbNfOXYvHaePhDs9w=")
+ORIGIN("adcentriconline.com", "LRSXo7qFBuUbdfvFl3JMJSxjCuRe0DPMXLCGRKpClwo=")
+ORIGIN("netelixir.com", "i/9qDtIY0LamfIEaqFMlFurdRIxDMphsf0DX5+m29PM=")
+ORIGIN("gleam.io", "vRTtGfK8nMsnLE55cGrevtSIXHlCw7NIF6ZTebtzDQc=")
+ORIGIN("onm.de", "BOzSdx8RrZlfqbQ5GewdnPD13kqracvKJRi780xCCWU=")
+ORIGIN("web-stat.com", "u7g7cP/v1DsAK9gu0KIoh3clDyKTRiZQV13CyQNaAH0=")
+ORIGIN("emediate.dk", "218QtJHWdJTXg5S2dJr14DbLCBpwLypOoIcDDc+Vkts=")
+ORIGIN("skype.com", "aSGYlwD3gGX9Scr7j6zHN4J0EFLytHWwTE/OKxI4KN4=")
+ORIGIN("perfectmarket.com", "oDwujqJLNDAFlm2ljeDJlCq+i5NZvRI3Ivt7LJezSJ8=")
+ORIGIN("sophus3.co.uk", "3W1GzUfGlXJ1UqTJK60ebEOCpOggGOmanq2SQDDxAKQ=")
+ORIGIN("gstatic.com", "KRWIs5dvl2QBQEMKrOW3v6bUoxS/D18rdMhSyVYrV5I=")
+ORIGIN("yieldmanager.net", "v3LILto7TnCshaNBmZe8nEUNZbvqMyvg1SOLax4pJao=")
+ORIGIN("innity.com", "s9Y6xPilUh4YNNX00mvpt1uY1I533Lj6Bhyb8VPi8NU=")
+ORIGIN("yhmg.com", "o9lC6a5JrzCYUydLL17OUqh/Own8I1LHIzjLMjC2LCs=")
+ORIGIN("afy11.net", "XApDUUARayrOd61+kq0X3kcHouV+iVULEvyyEwym6Rk=")
+ORIGIN("medley.com", "bszpZRxlyxRErBqIVuet2fOZm2WRMEEhO1Ci24uEgu0=")
+ORIGIN("applovin.com", "tmol56cuQy/BaV2sWpJ3KSYnU82mTorX2IYSZrcJDOg=")
+ORIGIN("encrypted.google.com", "vRyMkoWxQgeRIeTey/H2OLV21i0W+AlkNvX7mfWW+ZQ=")
+ORIGIN("sesamestats.com", "bzLMMh/MQKm/RwEmhJci1Qg6qiGkHbja7KtoaztPT6A=")
+ORIGIN("adinsight.com", "7U1PAwhZ3Ks6OBKtynT9nZ8d/P6T6i42Gt70nsixhpI=")
+ORIGIN("yieldlab.net", "9Xs/zaEYwC0BhPETkE3IiBP++l2T+AHezBdXxiGNG/A=")
+ORIGIN("platform-one.co.jp", "H0nUTRgQ1aeCE4i3EyfYRU9bxc2pcq3thOsQjMs/U4k=")
+ORIGIN("jivox.com", "ihv8/C+ro46hACQRxlhogzecn8A5ZnxD10SfdQZ6P28=")
+ORIGIN("collective-media.net", "PdDk+nVjBg37Wt4T8b0r4Luk+ub4OwZ5QCuOfub/KG8=")
+ORIGIN("safecount.net", "0XTKAlW6TBmGCrvXK8rcD6QPjNAn4h3iRFsbDKuzXN4=")
+ORIGIN("adsupply.com", "zH3NKvQQeHJL5MtbEsXFfxdlJi1oLyQGVJUCOGHyXDA=")
+ORIGIN("pagefair.net", "5AeCEU+Co96Llf7+xmmedcpfe/pRqsCSkn7LdGjjmNc=")
+ORIGIN("openxenterprise.com", "c13tdcJCGXkZm7YV/VJPr3pLLhI7M/C6jmS46swPRTU=")
+ORIGIN("btrll.com", "cQBzwWnqZCbRgTyHaTNWz9yDRfwxEThzMdWfU7dtoVQ=")
+ORIGIN("zune.net", "BZQgFDVowgfnP1MmrjVDmBT4JHvUdvi58vV/PUscQhA=")
+ORIGIN("yimg.com", "xMPFEjIzL1JQiE5FrDWs0uSrp4z/DXTrLpTynGMwvIo=")
+ORIGIN("dp-dhl.com", "afhcrI/WpNE9+mHQ/v+OV0VpxHoXHNkF5VUE2mj71xs=")
+ORIGIN("answers.yahoo.com", "QXruU2Eh/6Gb0y03/P+aAepO6eYjyn7IE+DMdqg49+w=")
+ORIGIN("amazon.fr", "ruBUwyHltWjWNyuFfbVCSd+aI/XbE3gxsJMnnOhhe9E=")
+ORIGIN("infonline.de", "doqVyFcV9wLR3uXQYIg1GdOGw/rInZCC/VTcz8Ekseo=")
+ORIGIN("liveramp.com", "rX9jerNrKvfR6LJhTeysutH20pMC4IQHO9EnJ4NY4gY=")
+ORIGIN("amadesa.com", "ru/+YLMxRaBvakngaPUpMfsuPdT9V6gLFx8pwDHGZuE=")
+ORIGIN("fetchback.com", "tK3wa+YIqZZKsygE+rpSuwvZj/KyLEIk4skGdjpL0PQ=")
+ORIGIN("adrolays.com", "IdQCFJw2xpAfJGrjXD0xa7+/z9e3TH6r2D+aNS8pvBo=")
+ORIGIN("channelintelligence.com", "sesNUlwEicObBzYFtfkfEkeB490Ia5kY3+213Xfcicg=")
+ORIGIN("stumbleupon.com", "vvfVTaMmyANMlqFVU6/PMoLGMI9wrMNIfkFRW8tWLPU=")
+ORIGIN("google.com.my", "+wxyAihDwyyPXlhDKr88YAf2VONDIRMj/Gs0k96EB4Y=")
+ORIGIN("leadlander.com", "aLWv58CW4hBeQln3tXU6k2A8K3JhNjRpWMBL5nscTrQ=")
+ORIGIN("atlassolutions.com", "7AOy0TLqLK84D8jY+GFp0FmwmGp4mvIHO6nNQMIBu8M=")
+ORIGIN("storygize.com", "1XduF+J6z8wYEnNayqUi8Euv+LH2QWPOQqsHgKBrU5I=")
+ORIGIN("peer39.com", "BTOWmT2DdU661Ws0g9/hmrVX3hYSumQ0B0rGqvYhJXE=")
+ORIGIN("trueffect.com", "kHIL3WX3PIRcWLAf5oBrSXbc+VqdQIXEC2NoevMk0OA=")
+ORIGIN("smartadserver.com", "J0zhHOxsrVKOwU0TSx2bUoDUeVpJf/5AMOHfjJOqv0Y=")
+ORIGIN("ekolay.net", "Vm6b6JTgFyDadxQ7CHwPDHnFZkgA9Tms/a6u7q/MzWA=")
+ORIGIN("google.com.mm", "iZ9B45w1iY7mDNHxKrocb2BLqh8AV3TXhxpF9AFoSCQ=")
+ORIGIN("binlayer.com", "0hqyi/mIHvvOF9rMNHRFG4Oi0yBCrIDA8s8OQcgGobc=")
+ORIGIN("iesnare.com", "yVO0dAyNJbnsTkD/IxQ/YXo71eTFrs6T/lfR2ZZFpew=")
+ORIGIN("advg.jp", "jVuFYPhyzIjysw2lS0XpRRPXI8UMQ9XY1zYiuIh5H2E=")
+ORIGIN("halogenmediagroup.com", "kQ0GmMyOzQPhmyW8P5VPTFfdAtUG8H7VhL1fvpb9y5g=")
+ORIGIN("disqusads.com", "iDAzUpzo0KLzpcu32hp50Bh+3NVWg3N44eH65Pp8IAY=")
+ORIGIN("fulltango.com", "9mrImGBJG6G6BZK3syy2E/oV9YcR2MNr0en476qK9Dg=")
+ORIGIN("efrontier.com", "mXduzowBX7q6BqpB6FdEvhdpAkASNXoqt+zaCxV9H14=")
+ORIGIN("searchforce.com", "PXRA20FtmExbUEEQTgYvvE88hOLbo/DvsqG5IRBCOmI=")
+ORIGIN("clixpy.com", "hvFmUWHoufSc+aaJlAzPlo5qaZ7WsjNIFL+Kc1cIJBo=")
+ORIGIN("augur.io", "h5vSG1HfB9LqVUpbB0wFg2uKsi/rOmvxtcqgAcEZGXo=")
+ORIGIN("marketgid.com", "fzB8JaoZsp5lrs5aTDUafz8plBIlRvFE1FNAG3osKbU=")
+ORIGIN("madvertise.com", "K1MX/rF3itrruxJJ33XtXHSX+yCzdXt5mEjMIBYgZMg=")
+ORIGIN("drive.google.com", "EqDwJoP7wN+qFew9xtSbEwE2b9eiDT9P8uY803o2jJw=")
+ORIGIN("google.co.ve", "2oB/es8AoKyn5ASwPrn07QuW/oEr2zmzJ7TYvPzX8OY=")
+ORIGIN("mandatory.com", "PF+EiLz43zbXtvJhkMzQoZiRYuphJE7nlDiZA6KunWU=")
+ORIGIN("google.co.vi", "X36Z9nLIVdnGv383CNv7vyCoqMLhR4WY1r0tYd+CI7A=")
+ORIGIN("licdn.com", "ZUA9jakhHmDNeBVI1jEVY6nKnfbb2dx1p/Zieta+jYg=")
+ORIGIN("activemeter.com", "WCyGFOlG6qPoL5Tvi5RFCCiO3zqYPm8vBrN5Uzj3etE=")
+ORIGIN("adcloud.com", "QTZSl0v4uHS9zNU61nne9IWMoB9Ca+WOiBIWOuQpxXI=")
+ORIGIN("gomez.com", "S7W7K6Nynq4VPae2/7D7M6bTEUaiEUTuPPlaKA+TzMw=")
+ORIGIN("addthis.com", "pJ7Mq7EiTnbpoX6J4WPsKT0qyaONExhOTgfJQkJuYTA=")
+ORIGIN("adfusion.com", "jmVCMZhdFGcqD3ixU+G4SHkxazwTZN8twjkP9f7Z/9M=")
+ORIGIN("51network.com", "8sxM6Yw0sT7KM5CW7WPFMMWY8tknqANTOCzmavf2m9I=")
+ORIGIN("spylog.com", "LwvQJMO+yr67nlNqHM29omxi5TYS48wkAFq5hhTm0XI=")
+ORIGIN("smallbusiness.yahoo.com", "6V/NUvhEa0ZVJuFSTV4qPzUXnKcowJJsX4/UFjNPbeU=")
+ORIGIN("connectedads.net", "Far8npXw9cntNLXE9GI5ZFO86DHYUPnn2cV/UXr+h/8=")
+ORIGIN("typepad.com", "KhAuvxEqMELaNaQIly+DosVlvlkA6oKiyc1vJS29AFk=")
+ORIGIN("adventori.com", "VJQz0TQjp6AgQlst3U3Tq7zpeoNrjB+vdEZe2JLhJao=")
+ORIGIN("actonsoftware.com", "RqTHP4FvOTT4KwyJ2KMqeEq9k6KapaoU7K/3DBV/O1E=")
+ORIGIN("mediabrix.com", "jHEDIc7OVfVr2b5KrS3uzT8IpIQpU3Aa83ScImaQOZE=")
+ORIGIN("clickintext.net", "K/pf0PqwusuqULQXFx+xJ/hgixh/Vi1xCEWzS/6om3M=")
+ORIGIN("adsvelocity.com", "mNOaA7HnxV+efMBkHkLcSRlfq0Th8B3dNWZMDFOm8MI=")
+ORIGIN("internetbrands.com", "Dja1WVyFShBFMEk54QdLLHDvjHaG7DhrVQzkwpxL37E=")
+ORIGIN("accuenmedia.com", "918mhOe6udg8mvzVekt9afZo6mz3jD+wgBmW5oPv5vk=")
+ORIGIN("rollick.io", "RPacef0jy6Z0+HcNJIijFPFolEAlLzZM4cTDJgX9S6g=")
+ORIGIN("iprospect.com", "1m6rxh/Xlv5GKLttg12sSfCn/UHZl6KuKMO1hyuBCCQ=")
+ORIGIN("gemius.com", "aA0v5H9OmpjQ0ZfRYZKHl0I0yVxOBFo0J+1C0w7nHDI=")
+ORIGIN("travel.yahoo.com", "uHiHGUBvTCptS8A1S7nZEQZrs/ogNm4G168MoQrml2c=")
+ORIGIN("betgenius.com", "X9PO7XKh4Nks+dlgfqerx9UVDm30BHIPoYQh8GBU+p0=")
+ORIGIN("lotlinx.com", "Oy8wzpn2JWM20o45mkatkSa6/IUxR3bcJfVZN+9/H8s=")
+ORIGIN("effectivemeasure.net", "u2WpE/EhOzny6W4cncKEMoJVxKwTZYlpcvBpc+TZ9ow=")
+ORIGIN("adecn.com", "7tQwoGFwkeuw4q2HE5DtapdJzftPaecm2DIvn2WKPCg=")
+ORIGIN("3dstats.com", "rpK7RbtyTxvoF1SGUAVTSCyB+Jo8B8+4tt0Mth/N99k=")
+ORIGIN("bing.com", "oMydI77SYFyiFJSEY0NLuMq1p+p+idtplVxDpOwUXLA=")
+ORIGIN("steelhouse.com", "nNhmd9DY8w+aZ/lpcomjpd1aJjY5jIJq7eweZMdeTtI=")
+ORIGIN("webmecanik.com", "A/mHgdfVaun45MOFrNtyF0+DIT61lGIjXP+ZcBGO+S8=")
+ORIGIN("llnwd.net", "V9tZPRP8KuZsSQ3xt+jUV6MdNX3dR94NEI2Rl2nxDvE=")
+ORIGIN("radarurl.com", "eEqsqHPgSHWCG64gX/r4oyeucFp9zz7YFdE/mSPEgg4=")
+ORIGIN("webtrends.com", "Cf0hmPujx20NKX/BUoFSSBBi2FGBWwoR4nzpB8VQtI4=")
+ORIGIN("auditude.com", "Ldpc7BwfJ3IqXw7oS35TbeQHDyeOctynFVDV0YEHGZg=")
+ORIGIN("nspmotion.com", "c1O1C1TG3AsNL9vKd+VLG/iGH65nfHfdj0VLZD4BSug=")
+ORIGIN("adotmob.com", "NreKOaYNa7ApLRNLKcXqy4dAR5OdHPgGxvb/f2d08f4=")
+ORIGIN("zumobi.com", "W7D+CCrjbmoGMKxG8HzLNjGWbLw9wQXd4T3Jfxe9vos=")
+ORIGIN("lucidmedia.com", "cTk+2sQJDlCedyiFyRYyCY5yQz5NyG6Q9FFkmNuC+4M=")
+ORIGIN("conversionruler.com", "zsnHMFGYvGI3/wkFGFfY98Lzc7ZAU82D8wKyvsA/ML0=")
+ORIGIN("webtrackingservices.com", "j9XdHI7XRu2GlrafLZ2yuY83XiEB/AUJNBTST30hejM=")
+ORIGIN("bazaarvoice.com", "K6Cr8SE57mbGFO2Y+ysdsReGPv+u4x/qrTEtH3quvuw=")
+ORIGIN("anormal-tracker.de", "DyXUYAeY8tMDqmC1CKzsRcm9lYtyL3+bLo7RcBcCZvY=")
+ORIGIN("insightgrit.com", "jK3ykpEGpYvNV+/KyLZM6r3Q5RBU9sLEMH/rd7F0jvo=")
+ORIGIN("bitcoinplus.com", "nGdtUMCQodWo4rQxpMNTh9m3eW5wYBA5cop7jzrV0Jg=")
+ORIGIN("pronunciator.com", "fl79pDqRdq03IpmfIu+PJpS1U4Vn7jWOYMoLB5/0Vj8=")
+ORIGIN("google.ml", "W4N7D0aeX03dJrPHq9p4LrvxW3HZG2cn5ymIwYWNDAM=")
+ORIGIN("pm14.com", "APC3BqXXq8xBU/SQRpAAiGIfyyD4VrSdm/Fvh1Jz+jM=")
+ORIGIN("adplan-ds.com", "ZouJaXGKEjghdQEb/+8iJ2fxe7BpVz+KrN4uJ9Qx6ZI=")
+ORIGIN("dt07.net", "SV+3y1aZA/ff7K8Q+tD8V+GbYl3EJD7n6XS+yYC/yfE=")
+ORIGIN("google.kz", "/e4ku4fk5Jqza+nc+QrjxDUz5j3wEbPuKw1EaxQkM9U=")
+ORIGIN("navegg.com", "GPwJLWULL7Flmo0Hve3lEEqBYI/IYm5RntWdoBYnEI4=")
+ORIGIN("intercom.io", "dLS8gNpGei3Mv3WHCaFWKNLKxT/ag3IHV5RRB5q3HCk=")
+ORIGIN("paid-to-promote.net", "kOCQLSZzrGHqzj2o02pK8rTuOGh0EjeY2CwTke/tNlQ=")
+ORIGIN("cobalt.com", "3TJHs3Ez5sobhBRQ6k2oTCZTTBl1hxQ0KtY4xU7f720=")
+ORIGIN("google.ki", "vfYWYKvnlaEyB2o+w0rk+GbDLyYyE6BV1o01YGmqsCQ=")
+ORIGIN("theblogfrog.com", "AtZILLNbvgrQ0osM1hgLTYsnNK4+vtUaUv6IscowzAw=")
+ORIGIN("adxpose.com", "xy5Wnbvgi/X9LW1lGDkUlzqxxZN3/bXNUM4wbMsT5GQ=")
+ORIGIN("vizury.com", "Xl9nK/mymX+2diAeUXAr5qGeiLsw6GSIxBPDCzsmN/o=")
+ORIGIN("samurai-factory.jp", "klCSl5zGHkBQdlo3aWYyN1SztpDXPie5lo5L7PEbZzk=")
+ORIGIN("yandex.ru/set/s/rsya-tag-users/data", "emw7UmKVbwPiPHjqSskuOG75U6hd2/ObNjtHzDWo1jU=")
+ORIGIN("digitize.ie", "2zZouShoPV5ffYUboDehpj0odIzgJHDXXVTjXSb/hqs=")
+ORIGIN("activeconversion.com", "fI92e+EYIjax6P/ddnXqvexjTploDX3RqwXFaZF33ac=")
+ORIGIN("xplusone.com", "7vhz+dMUtziue59+c5qP5riTVRPRuiq0pZ1qsUtPL2k=")
+ORIGIN("sptag1.com", "fDhxXyLRj8D1AnveNbV8X3vSDN9c9H8EYbBkPkkVnDE=")
+ORIGIN("nuffnang.com", "RYzJsJXbSGk7ypKV26Jf8F8rNN4YxhTKGFIggh5EUfI=")
+ORIGIN("adzly.com", "ehp0mFkLZmI5LTcJgvwx9wgNIhYKemUXiB/Y2C/ViEE=")
+ORIGIN("domodomain.com", "0AK//U5zoK5XT9uYvgi34r8yePj37QqWFeUke4O/C/A=")
+ORIGIN("cedexis.com", "ixiy07RvivO7wye8LQrW4RBEpWRiaXhy75U29P4Tx0M=")
+ORIGIN("consiliummedia.com", "5sGXJQRcF19qF8jgjlNDFt/uV0mgONVask1aO1m+I/U=")
+ORIGIN("supersonicads.com", "DsfuuDOQ32vSgMmEJqcblp0kZ7+9we6/d6Pe7YVrRuI=")
+ORIGIN("goldbach.com", "DzYb5p3yI6lYkqeSAE3Ooyr33DxDmjXjDiuNe+Ta3Ic=")
+ORIGIN("tmogul.com", "Mbla0LRlGaMavYjD+RZXzIhfQS4hmEiLMUh5K8XABGk=")
+ORIGIN("blogherads.com", "WxyjK1Q1vgxiEKlAAapla1ryXUMMAX2aKgidf5PmjMw=")
+ORIGIN("huffingtonpost.com", "l+pTBZ0bbldkglrtSv+CEYoXHBB3vqr+1kEnOvkpbJk=")
+ORIGIN("googletagmanager.com", "htTEl//KYksZTKdJ+KNDK+Jp5eGQk1QWfu66o1+oEyI=")
+ORIGIN("emediate.eu", "0ODQ8QFAI3x/YmCYO3zfmAvkCfRQTTt4DrrywrCW90w=")
+ORIGIN("adinsight.eu", "ti8O8FRd5hJ+0tjPlL2RiejfxsH4tRz/NpY1EiQzrqQ=")
+ORIGIN("ucoz.br", "Vz6OMjOuXbYEH1TRaPMQ0avft0/iM1mtUD7Bu9M4Eic=")
+ORIGIN("webtraffic.no", "y5GojvnaOVlQYtTWDbC5faHwnxfMu1yqj/BvFUnfnC4=")
+ORIGIN("chemistry.com", "fc9XfHbE1F0QDmoUwF19St7cv7u3u7HEYtsvO1Y3BMs=")
+ORIGIN("avsads.com", "1rXwqgoASkHJ4WldS6iH871me7eINomahJyP/IYB1W0=")
+ORIGIN("salesforce.com", "/mqEVQJxEhYbN6akygr6Ht5ozpWMwVKVPd0VnMB14+M=")
+ORIGIN("adlantic.nl", "YiWZErVw8ucMrTwCoCRH80adN4Grtt+enMdzYvVdZcM=")
+ORIGIN("2leep.com", "374XwW5oUoTNREtXgoUo1XZ6MOlz6iTTA6xHsAfGzOE=")
+ORIGIN("messenger.com", "G9hyol24e/338S2UlqIvxBegSLg2Z4GWAXROoINT+j4=")
+ORIGIN("affinesystems.com", "6E7DmkZzDdp8WWTRdGTGH8udd3iMTpFJuRT1yuPUluU=")
+ORIGIN("adbutler.com", "w4uhdjO1mD31IglbLIIaPT4hOBp0OLU1kkNWblrsD8M=")
+ORIGIN("flytxt.com", "0PAoVdmrGNUDiCmncBSzNCGHzAZ6Cq/SpBraDeliZq0=")
+ORIGIN("etargetnet.com", "WSZSEaigHH2yjszSb6yOQ2IIUoHBiQdHhkQjnGBuGDY=")
+ORIGIN("connexity.net", "ocvRYa3FOLyhBs4fqNYnYdTF18fS5ca906VBmVzRBK0=")
+ORIGIN("ml314.com", "mryRX1QjRaQUFM7H09/pT/vsnqrdngBZquMIjAqJBKI=")
+ORIGIN("pulpix.com", "t5DTechWhM2KFmUw9YB4nO6QnW5Wu1gcSHJDaziUXCU=")
+ORIGIN("google.co.jp", "aDCaocCSpMzdhohHIpeTianvwftx1c6ZFE6a2lx3R8o=")
+ORIGIN("liadm.com", "jQQ/xvUeNzHUK/v7ygtlnnU8cXwJcNd38rTDPb0BAak=")
+ORIGIN("piximedia.com", "sX9G4Ha88ZpN/zs3ErL7+48b3AixnUTLVOdmHcyThvo=")
+ORIGIN("epicadvertising.com", "ibht+XW0ZlwKL/YhK2hc6Rn49xSbQszlknaXcnyTZgI=")
+ORIGIN("clovenetwork.com", "NX2B6f7+skyQBlXl07n4s/WQJdi1J6bIgnjgTeFv4TI=")
+ORIGIN("metrixlab.com", "cMVk95F3+YcfLe/jAbIx2Fg8vneiO8lgKAfjsZG11X0=")
+ORIGIN("gogrid.com", "tMSG7CD2VOyr/fpXR0y6occpuQOO+j8b+dz7Cuchkl4=")
+ORIGIN("sendpulse.com", "RGjG+8ecPvEb+7ugNRMnS8iFU6MQwKSUTD0amF+kf/8=")
+ORIGIN("bridgetrack.com", "p8I/6ybAPOhv/uNomd7RreHKgZ2k4uJibXSfIbDkJkg=")
+ORIGIN("industrybrains.com", "NK+W1LTjLJ6x4gah7aoeTgLBWxmte9lFG5TaaiXWpXQ=")
+ORIGIN("rnmd.net", "d9AzbtAjxrs7/tt1vT6H0Hl9y50o09SKVld2OAEIPSA=")
+ORIGIN("semantiqo.com", "CDbjLVtn4Zvgr/6j6Vl3ZB5BAPqZRrWZDroHrS87oiA=")
+ORIGIN("mundomedia.com", "PvGX03JCq2cMHEa7zMoQzzYqqvEvk5vV7HZzCtYnf0E=")
+ORIGIN("adometry.com", "QXZUOahlsV347h/Cdw6GFOs/I1kZEj2Ypd2j3GUJ96Q=")
+ORIGIN("improvedigital.com", "pGlavDB71LKjq/IVa/qCiSwfK7+qMco6StWmR5MghgQ=")
+ORIGIN("adreadypixels.com", "Lz3aXGGGlwSfDZNwt0F3J3plQXvQuUsn0Fc5IcU0exY=")
+ORIGIN("adoftheyear.com", "RgFHHu52TUk38Ac+QfApVoqROs9pioMWEZgvfLXieXU=")
+ORIGIN("libertystmedia.com", "OTuSLaS+BQVy0iat9sVIFODIyIC1cGAZh4b+mQTIW8M=")
+ORIGIN("todacell.com", "LkGXxyM48XMw1Bz659Q42lo6s1U3ZkN++VSErVRPsFM=")
+ORIGIN("google.com.bd", "FJf1hnUiOLUE2U9Pwn+TmVRWrRtie7VXFOdPL9kQ2tE=")
+ORIGIN("lijit.com", "liNCnulffOLAkMdQZ9+DwWvlPE42M7WRlo0tRCmRBms=")
+ORIGIN("adbot.tw", "GyAZpuXlitPI/Ui+mO1UZjBv+TAMyt4e6gIQTzo/ItA=")
+ORIGIN("tubemogul.com", "aNGm4P7SYIkpfj530e0Lkz1/W1AUEqawT5pJrGljb0Q=")
+ORIGIN("bounceexchange.com", "cxT55v4jmBBHspjXWhTsCjb0YHamNj7H2IbBEt8X++0=")
+ORIGIN("google.com.bo", "3sSwKJ5tYkovp44Nl2HXY1g72ucXhiVbZWrdPtSQq9M=")
+ORIGIN("radiatemedia.com", "zjb4s3HYD53s/1DsMuT5oFNqtWIneSx5HdMBlewDnOs=")
+ORIGIN("scribefire.com", "eaynEw4jO2+1pm/vzsuDcdpmb6znZDbK90FnULRSa20=")
+ORIGIN("dailyme.com", "WUTJT6nisdZWKspOVotoEnY5vzDFqKk2UirHg5Eu6zw=")
+ORIGIN("survata.com", "DIrAwMLx+fDoBLfJjRvtm8kd7oXkc8pA1qgal1a+URQ=")
+ORIGIN("ad6media.fr", "Con2Uft5y7Nff1Pyb0MsBmyFZlav7TiLHm60OkElImg=")
+ORIGIN("conversantmedia.com", "G0hfghWgz2h1MjRNUBw6iVCP1BkWtlfTAkwEYJ+vw9Q=")
+ORIGIN("deepintent.com", "i32KOzhwTYxHkxmvAUX7m2XLzldQ36s/NUSbuU29lL0=")
+ORIGIN("v12group.com", "drr4ieWmeQYD13fwYQP+zIC2XKexTHjtq2anvVzveOE=")
+ORIGIN("fyre.co", "d5n79A2vVIkX/5xwRkUu2TDQTXnV/r+POffir+xc/QA=")
+ORIGIN("blutrumpet.com", "C48UkNLmz81TyGK9bpgAb0pNAdApRO1BP6qb9T+4jFg=")
+ORIGIN("amazon.it", "iOLn61YiC254tNUc0kTGKzRTQykDIfDPcESX41dw/5I=")
+ORIGIN("quantcast.com", "tqJniNfw/V2Qc0r6zrcVU72V2cYB/nu7MqFk+NMVUCM=")
+ORIGIN("hearst.com", "EWwQZoAI6PIVqLJMJ+maAjqi0G9Ctdo0O1v4/BCWa8w=")
+ORIGIN("footprintlive.com", "d5i5SH9CEaqCUs5hEk6xNcyyWTzD6zDe46UJHYCZ8Gw=")
+ORIGIN("vizisense.net", "C/bXnkdHjb9cvs9WNC5FDA+2gysRR41cxRK/sMDI3gA=")
+ORIGIN("wrethicap.info", "QDlWaCT4s8iMjgidRmyN/h4uu+mFWbTA+YR1aBNiWWA=")
+ORIGIN("zemanta.com", "8VarLa8DQjkWCa+w0Ujslc8Cy5UKWdRQO9k3CzscYbc=")
+ORIGIN("optimumresponse.com", "n88o6ZhgM9w9jPZU5SZrvY4D8qQRoxpD+Ww0AuJG214=")
+ORIGIN("tumblr.com", "GLl/AZLOor/CIUqAEGdbT2GR01uPwiDEULC+DCYCDEU=")
+ORIGIN("semasio.com", "ttyd1vqHYqIAFog9h0miMnjsjpwV9MJs7XVepWnTJoE=")
+ORIGIN("mdadx.com", "NVFX6OX1ST67fcYvnkAjXSteP036HGMFQJH0XGIc4vU=")
+ORIGIN("iprom.net", "MgY5pTtQyJ2o8FHGA+qmTn9+NF/ArtWWf1X2NwumuTw=")
+ORIGIN("boo-box.com", "yBudFZR41Wl7LklsYDRcXNW4pwTlU0JNjxafKUog//Y=")
+ORIGIN("sixapart.com", "eQj5qV0mQJ4vQobZIjoc1bD61+a/YENAtUhLfD28QXk=")
+ORIGIN("adworx.be", "WGqoo0Jd2/PD47kU9jI24VC34/OXZk5V6I6kVKo0aRA=")
+ORIGIN("news.yahoo.com", "zKgDsS4g1aznXovUDs+eirHklypN/8cJWT3kn9+jRts=")
+ORIGIN("neustar.biz", "sLSEFq8pL3rrfX5S17+T3jgeZk5zqL1Rbw+vtOqjWZk=")
+ORIGIN("nurago.de", "f+ZbVChRL7h7oCmc7LMFjyX8FVRkvrYUL7JdRLqIcXs=")
+ORIGIN("saymedia.com", "IyRnEDkzvdpNpZhUnwo+vfaVFSpnXEI0siM3lDmwMqU=")
+ORIGIN("yieldify.com", "htpGkU5otclT7l5BK4oiRLPpXY/UbbmXEPOzvj6V7CE=")
+ORIGIN("didit.com", "EwlPtGRYq83JGukcOQuMPMm/3tBqzmn7DwS5qhY0SMM=")
+ORIGIN("shareaholic.com", "a4kKpV3Ht0ca6IKyM4uE8BK/njoLX8Xit8c3dPMEQ6s=")
+ORIGIN("crispmedia.com", "kN6hEY6AEt8iTmuMYPWh6XQwzwq2r8EFBfZKqKkf7WE=")
+ORIGIN("games.yahoo.com", "IWPd3cU1Zuw+KfvaaDtEdIFADUBUP12gekKGqF5EH3I=")
+ORIGIN("statcounter.com", "B+7oZVCK0bBuETEc8rMsvOl+yDDmpjiRso87bn91IGY=")
+ORIGIN("pulsemgr.com", "DQzDFdrMUYKuJZINKKEzx8DjJwCy2HQxW54jIVRXnVs=")
+ORIGIN("mxpnl.com", "59YXD9qxauWHpbyZItdTgmwKVndKB9LYLO6Y5/8toHw=")
+ORIGIN("nanigans.com", "Hexm5igS7LLytN4IwyQ2hE5fofljE+1A3h3Vs6RAIQU=")
+ORIGIN("mashero.com", "5Xwz1OaivsSqTAZuXSMn/JhmQM50tMlAWHOS/qvQgpc=")
+ORIGIN("pagefair.com", "RQBqHCxreS0un5KEHVMBQ7qfW3FBYqo+pb5DZJcnx8Q=")
+ORIGIN("clickbooth.com", "DBW/ovv9rAJ6cwBzUFHjOTCuhIbifhk2O50G0uf55e0=")
+ORIGIN("inskinmedia.com", "5UyB5mtBvxPI9qBfT8lEtA1UZAZOBlTLj2ZOsn0cySg=")
+ORIGIN("webtrekk.net", "miAUqSOn+xzopTH66lTExAhiGcczrmnSc0y6L5ZiCuc=")
+ORIGIN("appssavvy.com", "62bO0+PVr5GdHgxT4Y2Mbka2YnV0kxvKQrq3IFP3y9g=")
+ORIGIN("online-metrix.net", "ttjM9NgnZKkSuDdRM7kgItoxcyfCHgQxmnNoGbepNuc=")
+ORIGIN("aolanswers.com", "Ro//odqJUdd/nZ19maVt3QRHxbzg0Al26ylQ7PP8eLU=")
+ORIGIN("amazon-adsystem.com", "tT128d9e1yVpGo5xhdhCOcSOeOfbq57FHkSzQzV52jc=")
+ORIGIN("technorati.com", "1PNVCjM+3w32NG286R7wtBB2dzDqof5El0m0c8tqYLQ=")
+ORIGIN("bidtellect.com", "klJ73numjGb8neMkR3R0JynCuF7BUjGZPszxKTbFkes=")
+ORIGIN("admized.com", "L1lKxO4MVP7K9TZ5pPm5uM0kNGH8AfE4qpRYX7RBH74=")
+ORIGIN("blogrollr.com", "SUJzNTTG5q5oLt6fW9VTkK90D85kMm3gLo0WuXGaZUY=")
+ORIGIN("adverline.com", "wPxFTS6dyblLB9Y9Sq0Ouy8WtxBp73lt7g8irb9wpCg=")
+ORIGIN("myads.com", "QfBY4+LoOSNxvUncaALYJ7hYjWDK9sIDX8ZPa8rPdUM=")
+ORIGIN("mobsmith.com", "BNTrv8gVBVzFdbfKkBUnP9tyMixxQ+hnHbbh/NqRoXI=")
+ORIGIN("triggit.com", "EsYoAZ88ow3iGV98odSF6L78BC59RFREw2ovpLi/Kjo=")
+ORIGIN("roxr.net", "FUuBKnmXS/qF/9SaGZlSAXAmZYXCAuAOv0GqPvlvQIk=")
+ORIGIN("protected.media", "QAQVzBN/orU55Vb0OaOCPzIYs+HyPsE1LY3tr6Xdsx8=")
+ORIGIN("friendfeed.com", "YZJ5B8ip13exLu0uJdcXYxgKJWjMIT0Tuqn9LIQHJ3M=")
+ORIGIN("sdfje.com", "gHP0jLLBEimhJ8dtAS7IOw5aoPiIjyIbzDVUk5mv8zY=")
+ORIGIN("iperceptions.com", "Fo+KxyOLqOfLYym0kTa8cHKjc718JEPo1zyokbMqUA0=")
+ORIGIN("9c9media.ca", "+KCCx4sGnahonL5SfUAcbtdg3iEhfwzwhLqAVPRDfVI=")
+ORIGIN("websitealive9.com", "WIu7JinjyCT9yqWoh0HoSfHlmWCIllh1VBDkKiSc1ng=")
+ORIGIN("zestad.com", "R6FCagxZz6HVUgQwqlXt1L0WvrGgih7LC7cVi5bVe8A=")
+ORIGIN("semasio.net", "chj56FhqSIC7gUS0OTZTpnplWWJtGQyClMXYIhFkwFw=")
+ORIGIN("activengage.com", "o4T8gCkPzRrz1FX9oiHoE1+NHd5xq8qXWY53qUpStNI=")
+ORIGIN("googletagservices.com", "2ImL5DSoDWPg3cjouHn8exn+jYt2MvwPegDEGRtBynw=")
+ORIGIN("papayamobile.com", "VqN1NraNtLjP7RHiKUTDzeQicdwarH9wHLQ1PX36chM=")
+ORIGIN("kcdwa.com", "WReRQk/heSPxbyBHP9mbgTa/xiZreWnP7EOCDuevuvA=")
+ORIGIN("simply.com", "oYsAqU/08Awcw0vRgicisV4e3kEpoUaQxD79rpPe9ao=")
+ORIGIN("opera.com", "VjSMqUcyiPpR95r1AAQNoVWNYOc8h0KDJVSe2Zvf/3c=")
+ORIGIN("mydas.mobi", "c4V7tNERF4OgfrSt8ZEJqWHxKclXTJaFxPUnrwEFrLQ=")
+ORIGIN("mycounter.com.ua", "uzlvdTFQnUFxzdSQ802rHfy8NT59YClYys20iQSXtBg=")
+ORIGIN("mobilemeteor.com", "r+ITMaY0m8235GzrI9gvRb9Y9i6C5+1214ynBwFt5CY=")
+ORIGIN("chrome.google.com", "Xbq8tZLhAuUbGkEvCGg6iizZyoJYWqr5wpFdp/fC1mE=")
+ORIGIN("showmeinn.com", "fNpRdarLrozVrYbD5VoFy1w7b8j38YaYQZzTmAlOMT8=")
+ORIGIN("geoads.com", "X2sbtx24wIamHbywdYkvfwH/l0aVk3Nhe3PymW2gOH8=")
+ORIGIN("opinmind.com", "51iVHWCxLYLIhOzJ1Ihec58gkR6d7hwtvoybCUjNsh8=")
+ORIGIN("vindicogroup.com", "kTrinToZ78g1l1Eu7lbiU06XhvamyoYiBVo8Sd0hdfg=")
+ORIGIN("selectablemedia.com", "wLNh/8U+dw0qQWo3aOto/Myp9fdk3tWh8hti4P6+FKw=")
+ORIGIN("appmetrx.com", "mZTyXVdF7LPUV/evI78il/lNDu1NM/hQD05RJj7EU88=")
+ORIGIN("reklamz.com", "/eAIbYUO9nkoaS7faBbm2JSqND9tmGC3km+SUc65bX0=")
+ORIGIN("adbureau.net", "cBKKhYyALrSnihAAOB3zGUNzPAHn4TJL++WyRlLTTVY=")
+ORIGIN("google.je", "JatsoZB53rUR/9UAFz8xBKG6NK7EMBIu4J0/FrJREdo=")
+ORIGIN("mopub.com", "ApdsE5MKLZCe/D6g1Vi2XrW5OBL6YP7086MlrVkjGH8=")
+ORIGIN("checkm8.com", "298kF+8AJBTI2YCI5XWVVuDloZlMcdBaicdxMmZV+Bg=")
+ORIGIN("clixtell.com", "joDvd5B0X7mHAhBrxgynShMMvDeU2zW47R6iaqd2cMg=")
+ORIGIN("sptag3.com", "keMzBzhQ9SbPCTGsBH9J+MzRUvUfBdqbWDmG1XbCi9A=")
+ORIGIN("mdotlabs.com", "lYGYbpUNIiU+Fet34ehDePpmcnflNfm12nHTI49K8lM=")
+ORIGIN("help.yahoo.com", "r37O2A4HLeQda8NJqtcywLwRSugi0HjHiMwRcpA2W6s=")
+ORIGIN("adpepper.com", "uIhRnLGTDgdSyi6R/oMpqDYsWGz2DQecdYwG3QiSFgw=")
+ORIGIN("adjug.com", "jrCEecZ7wq0InUqdLeMCDBAgtaDZcBbTbttjNDzylts=")
+ORIGIN("adtechjp.com", "RBiZC24E9+v5Ipwo7ic7iMZ2J3joeJT3P5PNRBbsFo8=")
+ORIGIN("conversiondashboard.com", "OOU4+JUThV7b/hIKAowYyF+hv8tElVAMD3UAoDDT+hQ=")
+ORIGIN("adbuyer.com", "AVW5//O8yvDn76JW3nlikfHDboyoyQLx6EUBPvjeDZY=")
+ORIGIN("adcirrus.com", "PnrDXW30VUX/MfinzqJxMJ2wL7hLNWTDMrA95gPjwro=")
+ORIGIN("datranmedia.com", "sK7ie612ekm5DMw9um54IMfzAHm7Jq+0qh6gSLYTaiw=")
+ORIGIN("ds-iq.com", "lAEidJkHhIojlLnGgJHO1iCUkZRIn9V/oCcKEK/RG5g=")
+ORIGIN("picasaweb.google.com", "ZfzTURTHW/Bb94ekMqRUVL7fympJ8JghAI5XtiH6hYc=")
+ORIGIN("successfultogether.co.uk", "QEs/StErvXBntaIwqJfC0dQNvWU9qJitwD0bUjjYYoI=")
+ORIGIN("google.tl", "SYu/vNHOupVtKAf75A2OyUqZQ38oStoAk/rcaku0Hzw=")
+ORIGIN("google.tm", "6Sjo9tUBF7LxlhaEe+TJWnJeHmbG5dAN19uI73A6i1U=")
+ORIGIN("adpdealerservices.com", "vYxPzJHxLPWuVlOBY5iyvvOlTouuVROZFYRpAaP2fIg=")
+ORIGIN("google.to", "1nH7ErY6HMzUJlm2bY8gBMvLpLc9WEeNnoCbyXS1tRo=")
+ORIGIN("google.tk", "sybdVxB1dIAu/4qfTQxFDfdFGCC2L4u+pdYBTKgcrYw=")
+ORIGIN("google.td", "syfp7MFAs1fHHNIozj5KHGnrfQNMDTwliAXX85iwZjI=")
+ORIGIN("google.com.br", "6kfVZPGsPTN0xZTVGHn3Lt6R7N4q65OxhCyWPDJ1VK0=")
+ORIGIN("clickaider.com", "kL9BgXIcJFCq3w+ndcuQ3eRg3yqvqH2zQQOzOwkeH1M=")
+ORIGIN("adnologies.com", "CDh0cp58WWwrnU62eRL9SCYHZfB4dgUyETiERWCIQcs=")
+ORIGIN("islay.tech", "U4FCxs4fHPGR6k9rMmfkZ+eBlMaQ5YH3rTHgbH9Eelg=")
+ORIGIN("trends.google.com", "jV6YKBczZgSViR9rad76EFDeifMQZTw3k18mBjpOrUU=")
+ORIGIN("cpxadroit.com", "TN2q1q0zW54VKnLEe1kgExJIIldHESIOXvbsQntJK4c=")
+ORIGIN("google.tt", "d/WkVwe6I/4aHWf1sMKxnVRgKMU7DM/M7OxuxS1MQ9o=")
+ORIGIN("eleavers.com", "4UfP9a6HP0ZkeCuWSD0o34bavirKWGQ/ZVXUcMl/mFg=")
+ORIGIN("mixpo.com", "SmC0UOVBEj8vP3JbaMni/O8iO6v2bhSscPXgCoRi6h0=")
+ORIGIN("ybrantdigital.com", "91CvRjHVBO33qCPmJ7HZP0gGZaouB9+a0q9qIB75bE0=")
+ORIGIN("convergetrack.com", "DXCSO9vK2VjV/Gid2eKrrBd+JL16O4H+sc6f4uo/KYw=")
+ORIGIN("adsonar.com", "jkkRP78rZLO88Q2qHg9kKeO7NeWj5Z3sH+8MdldckSo=")
+ORIGIN("pepperjam.com", "RCn2jIY/bio9Lfn0kmS/eDImuQj0ExOWD9PvGsC2KAY=")
+ORIGIN("adelixir.com", "tuhy2eIt9dDToVjuGywlYB3RNFFJ2vRVCQTCfLZXsPg=")
+ORIGIN("bmmetrix.com", "05E939UEjPc8TXPXZBGjQWCUjeWUxC143mwUMAOPI+s=")
+ORIGIN("google.com.bz", "sNGtnPBxrq+GQkTS3nRfzgu7fEQHS2DSeMKLFt9dzSY=")
+ORIGIN("google.co.ma", "ShjAECiCzz750nf/GUc4dq8rKBJkCuV9i8oytHO2Qhc=")
+ORIGIN("performancing.com", "CETLq75b/Z1eSZ5cAPvrKuIy4VdWe/3T/4YiQo8rWZc=")
+ORIGIN("legalredirect.yahoo.com", "0rnpcvELKPfhygkqucWC7awSvO6L1RyLZDvYAKaZbtA=")
+ORIGIN("play.google.com", "j3TbCCaFHBTOUtUkEk+5tCI6jmveHzovWeoIUY26ceY=")
+ORIGIN("retargeter.com", "gwmltgCFeaUs8eHrDkBrDcsyXbqN8dF1xbxqPtcxTQM=")
+ORIGIN("facebook.de", "UyI5vMnt92gQIwcHmL7l7F5Ka8fAu2jh6OkJnkX9/5Q=")
+ORIGIN("crsspxl.com", "/3e8PFKLOpN/AJtTs/8chkm5vJ0NysM3rGCrYd00iz8=")
+ORIGIN("5min.com", "zY2tj0G+tL0D+K0O9oYjMcEcDJX5DabI2dB54U3Jb28=")
+ORIGIN("hotjobs.yahoo.com", "5zWRNgLTj6gg2PasX7vklMP2xSZ8HYPef+AQr8XwfR0=")
+ORIGIN("histats.com", "Ec09xsPXrAwk5ge17DLayMyaZvmv5w31MWXdGR3Apdc=")
+ORIGIN("google.co.mz", "dzDVQtwyBvkXDFTXG26ZH1XQ48360QIXnvMQHSchFX4=")
+ORIGIN("digitalriver.com", "oqHggIDtFPLtcHOGFjL7QmQHaGDVBz91lhCPQH5fw6c=")
+ORIGIN("sphere.com", "XBRq4KRFUGNUKROrH7MMloKBsA1ASRJ+If0wnwfgq8c=")
+ORIGIN("gfk.com", "a4Pd/FO+ApJ4hDcTFrz8+Wt822YNVFGZuSVTXj+Qi7Q=")
+ORIGIN("codesearch.google.com", "BWaynFWySl3kyxjJMly3IEreP5/tFRBLKST7j+gr8mE=")
+ORIGIN("ad-maven.com", "NQcQ/bsnyuB4A+g4iBNv1jwVQT/+8dlShbzx3pSeEv8=")
+ORIGIN("adpredictive.com", "vpVhxUwzYt2WX/Q7xXCfc+l74LIVlmM6XtZboDxgv5w=")
+ORIGIN("uptrends.com", "NK8B32W6lV3g6vK2mFbLRfXSCCzPz+Kp4O3eqmdt6js=")
+ORIGIN("operasoftware.com", "vvknWHMmxR8dXcuPul5t5XiCyPYmvkfS3l0DqvXI1aM=")
+ORIGIN("facebook.net", "+P10RWqWUlKqc7QgFK1QtZR70eC/KOv0zDgGFhoO8AQ=")
+ORIGIN("adultadworld.com", "3SvcSRWgZBrrL9CW8Aue11YMV7M4BFX7yS+Qxs0z72Q=")
+ORIGIN("cambio.com", "eIonAX8ACNj0numGfpmVbg3lxD2KAz/BHHwLPGtjl/U=")
+ORIGIN("hitsniffer.com", "dpxIvtQm1+aenv6dkNcs7Zp3clf/RMzeaDRZnHBLYGc=")
+ORIGIN("ecsanalytics.com", "xTEedVDnFy5NsydMT63HJanqb/X8WOe0lkWktJiJrL0=")
+ORIGIN("adzcentral.com", "yh2xJZv9Q0GkdPsXi5ofQZ3c4WoAbmwxbD/7PnRwpR0=")
+ORIGIN("google.com.bh", "zxCVLzVSfSjiMHin972TNjrLG4eXaIOA/GVzeOFXfaQ=")
+ORIGIN("contextweb.com", "mACymJ5S9fgbKmG0eFydHOAoXN+pevZUqVP8XRlGG2U=")
+ORIGIN("google.com.bn", "7HZo4ud2VB+QOqp7kcPGarqT0t8XtmwWpfL6xjKjvOw=")
+ORIGIN("inspectlet.com", "3qiiBbrR01Ezhj0uWq8po5X6oQU2lC3/7bHUH/pof6o=")
+ORIGIN("belstat.be", "Nz70OfEoAl9pI18Vmv7R7wApjuev4o3ePpphg5gv610=")
+ORIGIN("trafficmp.com", "A/hAVx+2aUMa9V+85J8fw0499VSG68yw7VHr4/3qjns=")
+ORIGIN("lyris.com", "63ow+MVunQ46Z4HmAPFKTGYNAJuGKPJW4+y4TGOIauc=")
+ORIGIN("avocet.io", "Kj5WtrxLxQMLsUQn2442MBZw6DRxnbrubRrQjIOYtnc=")
+ORIGIN("xaxis.com", "UJiTht0/3FcJsFxoVJdrUxKcs228KnIq8DglTXBnY3U=")
+ORIGIN("vresp.com", "WEQYxN1Ks63xWNpLq3qPcxBC1JuJuwdZi8XiLtQqB8U=")
+ORIGIN("lzjl.com", "Gy8OsQkqGISxYgl7xJSRljaIVFO5RtdjPOpRs31SLyM=")
+ORIGIN("idgtechnetwork.com", "tapYb80WRoUV8WBKKWlYNSTMtslJHklzpPi+2rkig0I=")
+ORIGIN("buzzcity.com", "5Y71NsBoY23x9pChQX5LYhjM99plPt5dTqr5x10VIs4=")
+ORIGIN("globe7.com", "b9MxXfl8hgWaRnRtl0A6x2/ZehErKsvca8wKrum6mmY=")
+ORIGIN("admaximizer.com", "yQdBdTh+2TY88vM3uFhOCRuJnHgT7hmrNBdcygyjKfI=")
+ORIGIN("webtraffic.se", "x15jWGw9jQnkyHdVY2PeGjYN2sx1/IFlZJA2rJsogWk=")
+ORIGIN("loomia.com", "yng8ClckuBys/tgoyQEc7NzCQJDKARIf0snPK6rppls=")
+ORIGIN("searchforce.net", "1y4+rcwtzBLOWF5faiZQLe/qdIM0xssCY6lbdMtfkak=")
+ORIGIN("everestjs.net", "BL8mx96zTfnc646A6K6dZ+PuLbITPmi+Yonh+y51mFc=")
+ORIGIN("webtrendslive.com", "SO/L154nrRfwQIKfflRw8uNS60sDnBpqYNtnf/3ZtJA=")
+ORIGIN("fanplayr.com", "XR2GTJz6hQyuov6O8ibdbf2XE+bcWzW0Jy0ART3ioso=")
+ORIGIN("extreme-dm.com", "Tz6gauTEZnsR/LsqvTH7oV0RPQ6jmnEBGxnOu4mS1Ig=")
+ORIGIN("edgesuite.net", "PHbUopy4jkciQfBTV03WGEe4uaRIbJ6B1VklU4AYing=")
+ORIGIN("weborama.fr", "UWSUJDlJfAKZNFUaJ8zT5XxlKonYZEYZM/FXwhszdIo=")
+ORIGIN("resonatenetworks.com", "7mR5fWMBMnFmgYoMOdWkwagv/2b/vjM6i2Ry/I9KFRY=")
+ORIGIN("rhythmone.com", "vUKY/qmCgyCFeDvKhZZz98TI4FIwga+q0CmSaIjdaRc=")
+ORIGIN("localyokelmedia.com", "fa3VrSV1DeA2NIL7qyg9k9HPNyhHb2grie3ItXqn+zU=")
+ORIGIN("convergedirect.com", "rFFl61m+cyUBs5zcB0AeypWwJj4zkbH0lQIVGiOIMEc=")
+ORIGIN("postrank.com", "diuWkFWmauSQshcUpHOfJGVz+dvfbyr1SbyOVWFDdXA=")
+ORIGIN("bkrtx.com", "EchQwgMoeVAIsXKC92PGhrIWw8zLGNgDClLKKvO5YQk=")
+ORIGIN("nextstat.com", "AWh9sMGdV0BoQx2yNotCfHhvVS9WWc9w4InClA8fNIA=")
+ORIGIN("socialchorus.com", "Ie/7IWfwNZs+TC11sdD3JouyaSaUhIE0tpl48VpEGFg=")
+ORIGIN("clickhype.com", "6xGz0UVzhJ+fXMT6vvTrs/I9PaAhgIiVI3J/U2Rupos=")
+ORIGIN("psonstrentie.info", "cGAM+JRHsInNdT1F0Zpbr57ZJIa7+ymq/X4V104gO68=")
+ORIGIN("oneiota.co.uk", "dfpAIeVW1wEaeeLGpeOu5MG+g2xa5XifFQMeZjIeLP0=")
+ORIGIN("sputnik.ru", "/gMvsTnrUcIrJcVQ3i3R0W7U6sBIkIcBZMXfnAEI+OI=")
+ORIGIN("etracker.com", "8qUQYkcXPawly7pN3QS/ub5wHzRhT+TH+AZ6xqGQZyE=")
+ORIGIN("alenty.com", "LZHUMUXlUAli372+imcTv3sAQbSlz6//Z8W7Pqk+5vM=")
+ORIGIN("shopzilla.com", "fXcJcKZerEh+0/edq4cHpAt1a2Sj/7dWVaUDxCRa4l0=")
+ORIGIN("admatrix.jp", "0F/8/dbSmdRjpCJB9svNyoCO1s6/k3wCKL/4V3wErfs=")
+ORIGIN("ffn.com", "mvfKH3PSvEr5livAMr4yERj3FmI9O3DF0RzWH6jr7FM=")
+ORIGIN("hotwords.com", "M0kBNa3qL2Fzp4HK6zMgelojPhAeRW4pr26IfRx0uos=")
+ORIGIN("adobe.com", "FlCbaZR01PhErY63f0goCj04k4OLGgE9zAeTinsLYPo=")
+ORIGIN("google-melange.com", "OuaG4QrFEKPPW3o3tBxhGO7wgqnkOjLTNDLOfWiymfY=")
+ORIGIN("addvantagemedia.com", "+XorU2mTHixNiDCk59NF+l05vVAB11pUV7b2rRQunyk=")
+ORIGIN("marinsm.com", "OCzQpKu/qeb/9hZFeuIp3ht9sPctqcGKHwi+aPAbB5s=")
+ORIGIN("qjex.net", "/D1XToBeK8A5aIxeONMps1CppJpdYqXIHmfmntqW7aI=")
+ORIGIN("tonefuse.com", "BJoaCN2K9e3VHfztUD/QVi6cuhjKxl87txMqiDdcQEM=")
+ORIGIN("quismatch.com", "NJuZ9pOxS8IxQf7laBtVnmq+JqrTrN5qgHZNi0fTaCk=")
+ORIGIN("do-not-tracker.org", "sQ8o1NENpHrlnitJQoKgHBYP2f0rYhLbD7IKR3bHTJg=")
+ORIGIN("reklamport.com", "DTaTz/YB/akSuliuI0lljnQNWs5Q4Jb1AZyWbAASbTU=")
+ORIGIN("vimeocdn.com", "Bk8jL5uM4wPXwywJkXLsNrFDOUbfDOIDrZiNNr87RHE=")
+ORIGIN("vizisense.com", "DnX9MSgetgYlXavkU9pi4wKf4aP7h3BeQwr6sIop8mY=")
+ORIGIN("gamesforwindows.com", "XwEqDJxPIdcePX0oHBMYHNn+bOdiANlg+HUmmx1G77o=")
+ORIGIN("yellowhammermg.com", "QQbIonUO7R+209TURFtSolDsN6ExCdVxsDcnjsNzN4w=")
+ORIGIN("netseer.com", "lD/hzZ+yYQCzoEbcwzhC3C/UNwzLW6+xtT57ckPUmYE=")
+ORIGIN("crwdcntrl.net", "c8dUNdbMfrQtbwiNBUhhCO2Nk9Q8RVx2ib3IC93MGUU=")
+ORIGIN("inviziads.com", "EKfVWxx0iM9NNCh3AXmrHqqO2fBO55NkJ4rNl4wFE1Q=")
+ORIGIN("mail.google.com", "EVqNwgvUlPp21Cg1gTu9o/67iQoOeoQWLjY4DwSpML4=")
+ORIGIN("silver-path.com", "aiU6P4feg6jggb8og47s2UsTR92fKC8akU2OtJIqnls=")
+ORIGIN("usocial.pro", "UEuRz89KapQjCap07Hj8V1cyW8KPR1NmU63IukffD04=")
+ORIGIN("feedproxy.google.com", "soyMPUqdz1DP0kl6fTYcqJ0MUNszZ1RgHJ6ka7X1Tr4=")
+ORIGIN("blogcatalog.com", "V9kEIey8v/0XyJ9hwmHSqcJWMHpn4d7htg6ybbl4HsM=")
+ORIGIN("zetaemailsolutions.com", "dzR7bu2HkyRnRgPfzr16L8m1BUJjXrKvYOYMMqed3bk=")
+ORIGIN("brightroll.com", "YsqaMb9rZU+tyl/EbYmlut7kLZemJmCCsMC9NrEJDW4=")
+ORIGIN("iprom.si", "LtQnXJ9kKL+zyK8ZRvJKJgd9pIGbRFuqeyy5wLE4lR4=")
+ORIGIN("blaze.com", "yips+kva6VqnMXKgdR7/r++cQmgr6GAJaCMTxx+vK6o=")
+ORIGIN("adside.com", "iZrrygOv2b5qaULjuzSQ5kmY00JTs7aPLf3ejmomgXc=")
+ORIGIN("sophus3.com", "pd/5CqTD2Q0/+jq0dGQb/amR+eC2jajzgExTYkaOdGo=")
+ORIGIN("aboutecho.com", "FVHRLoqdLo2y4StPhazkXYVaYUriaVTjw0NoMRIhw2U=")
+ORIGIN("nugg.ad", "eFYLiPXueqloEBQEmDk+esAbr5Sb/CCzD7liNES3FUg=")
+ORIGIN("gfkdaphne.com", "IY36+E9JReWD4ZCd0UkBf+jF9w0sKHVUyTCcsMZK72o=")
+ORIGIN("office.com", "u8ucZsw2sDq966c98tMjhs8QkZUhrv4PGnYDn3dPwYc=")
+ORIGIN("ooyala.com", "jyFfTg+Nf0o66V3L67XdwZGJ/W1ZyWPy4LC05pArm/g=")
+ORIGIN("drawbrid.ge", "uhaiiprZAToXZPRBilSISOTlwjjfFXogPrmn2hF9fE8=")
+ORIGIN("eyereturnmarketing.com", "cYl5I/nQ+m2S4DxLMugkD8+sgF8t6llrynYWbyxVIKk=")
+ORIGIN("publicidees.com", "0+DkU+xYxynLOD2ccScpe7Ahd4s1yLs7QBLtdJq83hM=")
+ORIGIN("vimeo.com", "Lrd6afAXxiv07BkHlLf5am3Bn6HRPnE75kDjM3HiYv8=")
+ORIGIN("cbproads.com", "kwjC63twLD0e9zHNg7lLFg/uzi3j+hH9oDv3YyBY384=")
+ORIGIN("adglare.com", "xv9t5ZlWdr7mIrl48AO14N2nNzDb9VZt/3LQf5PPUWk=")
+ORIGIN("leadformix.com", "bax5UUC4FHZ+YO4yLKBiTMLPkC/f4Y/Oez8CLYtH6RE=")
+ORIGIN("tellapart.com", "W7G2XWe65Yyd3oio/bLxS7XfQw8Nu421O1atYzu2dBY=")
+ORIGIN("mail.yahoo.com", "1rLzD1viMjF7CLEIc59EWufKqDeoYiGL0Pfkr20L4rU=")
+ORIGIN("amazon.de", "wEAf0lX5nZLBCM2/nHrScJUFJznvqICFtfp76bqAjXs=")
+ORIGIN("clicksor.net", "dIP9Qc5zu1cYmzdTfPgWTvifeTOOqO1tXBQKJjC40+s=")
+ORIGIN("effectivemeasure.com", "8UU71AwYL1Q8lF+XFkbMyDzMwFM6xPp/wOkLnO4wTQY=")
+ORIGIN("clickability.com", "Pfw/wsSl79rWe9WwJbKxKXmGYIPhF0utlJrrgGPPEV0=")
+ORIGIN("p-td.com", "6Ne0rjsv45sNv+LxGsX07nya+4T7MVOdOscHqdMewYk=")
+ORIGIN("roia.biz", "hvQfW9uyCefnJdVFo1oJknhl3GuMm3S0Dp7VrTlDGXQ=")
+ORIGIN("google.com.cu", "gbnixk4hNDbtRUhytxv3WaGFoJ0VF8i31fuE9a+ChmI=")
+ORIGIN("staticstuff.net", "wLWeVLBJCDUYQSi5rbYl8MI32OW5kmWFNRfGI97bIYg=")
+ORIGIN("google.com.cy", "Z3jeUlpBpWP/jcVQXKvjDpNPmtF6dLDycgc51ndeGpI=")
+ORIGIN("intelligencefocus.com", "spmalYlfF4nprunctTwRiT+US5YqexulnMxjbzFLlhQ=")
+ORIGIN("decideinteractive.com", "vBB05juGLFuOBy93fk9s3lazmiMJ9lcEFDzxfEofv04=")
+ORIGIN("google.com.co", "S3GWHUYtC8Y6dALdA+P/wxh4UkUseHpzNAZmpMJY838=")
+ORIGIN("ixs1.net", "N49ymCbtISe+a5Vsdt0ow88jzdOmFqQx8CQUZhYZD4g=")
+ORIGIN("leadforce1.com", "l0SspEq8ho3jJAzLlbO2PDTk11THouHgCLglktl+zgg=")
+ORIGIN("truste.com", "MFUVNxulamuMOYe838yN37dzuuSU0fjAg7ohvhSXIVM=")
+ORIGIN("adition.com", "fSoWNG7T3lI/VEH/0EU57ZB23ssv94Ioso/INXX7im4=")
+ORIGIN("globaltakeoff.com", "NRB0Dmx4syRviERwsgnrAH4l7mMAFxoj1A7J5q4X/d8=")
+ORIGIN("bunchball.com", "Pj0Z62t4sAM2F4c3KcMz4BS0Dj4j0G/+u8ZDjtsgBFY=")
+ORIGIN("actisens.com", "VBbN6c8FMyixSaUTb/Bs1kT8ZK8vM45Yf3n1m6yU4Sk=")
+ORIGIN("ypolicyblog.com", "57/thf5QHzgo9NkP0ho4dkUwDGMx4ZsxCldPFvhglNA=")
+ORIGIN("affinity.com", "7QciEuTqHy6enlGKo6qzr9GzQuQhfL/oMemk/v1tyEI=")
+ORIGIN("103092804.com", "5jDWdRlk9lHC1EjLjXq8cU/kBb9borIyccsOMAX6ing=")
+ORIGIN("nedstatbasic.net", "npHeEKB4+gsnGDB2VUuERmJV1u3xSEy8BDpnmNYLFZw=")
+ORIGIN("z5x.com", "7SQip9j8IxSLiLj4iDbH3jyaL9JPNg+NgIBP0G5SDa4=")
+ORIGIN("sublime.xyz", "/+44zHotyS0zm5TdLubaZUL00qcq2364H8dRVxFg91A=")
+ORIGIN("gismads.jp", "n1TcDkfYJSfN1Eup//o98resGYCeRMsOmQdMVnW5+84=")
+ORIGIN("channeladvisor.com", "+WEd2sGf8fUer9CXE2LFhpeWUEax0NExEXFLlvp9cus=")
+ORIGIN("plus.google.com", "dCUT61Ynh8StdDL7hMjrqtyRLLXjuljGr18zQCl4bOg=")
+ORIGIN("aimatch.com", "sN1pFYc56pIswm/GeOi/W4jcyU5mu6R88XKERqaUfgQ=")
+ORIGIN("jumptap.com", "La49XRZT9svfrN5KYnO6Lg2/BTlkPwYisVsxj4YQXEw=")
+ORIGIN("hubspot.com", "vfFsqOH73q75yzwzKFRiEEz502OYvp4qLssbjIGh1/E=")
+ORIGIN("getgamesmart.com", "m3i06IY1MZucJMEnQg7zkftCoTgJJUIKTTgl9n2nw3A=")
+ORIGIN("adnetwork.vn", "knfS9opi1rk7ftQmTHoSeTm1WvIM9qvaUeO69FJYC4g=")
+ORIGIN("inq.com", "KQtmEPSnnaZUbrVbbcVl6BtU8Vfygpi2ga/Ugt3YkI4=")
+ORIGIN("feedburner.com", "E/+J+AF889i5QV1aC0+K2FLlMDHA/X7abW/BPggPHaA=")
+ORIGIN("automattic.com", "3Mb1qQDfHtA6xWj14acAk6/EEd2/SR7wPq+izh14Hco=")
+ORIGIN("adeurope.com", "z+mMgB7Zk0DBevXr5fO7lVms7N5/nUP7Hxsu3HeA/OQ=")
+ORIGIN("genius.com", "25+RnbpWO6HrEEWUktAmddFv1MzQjATjq9gR8HOmix0=")
+ORIGIN("upsellit.com", "p7R2p7I9x5EuhgNQJJNmM0e8Lpjh4JuUHzCT7HXJH1o=")
+ORIGIN("google.ci", "8RlUYPxUENZZTm0sTCIJmV+ltoOEStBZCn5GR7Y0/Vs=")
+ORIGIN("kontera.com", "Yq7kqnlKnbwt6qqsiNg5jqfb08preP53hraDKWEbbgI=")
+ORIGIN("nr7.us", "ks8rs8ADHg1Dvwz0yAGdlc73ZttUFO/bDLZyH2xtrjs=")
+ORIGIN("smaato.com", "4cOlKyV3/1dxo8agsrJlUuo0Ytouk8X4U5HTFiLW7lM=")
+ORIGIN("tremormedia.com", "sjgsqsBtQ4nVvhliZ0uRSHZe1EGPbxi/44hwIAflqMg=")
+ORIGIN("adlegend.com", "0fqv3+xIncxSkv2aN+L/607lzP9Lhq8V1QLj9a0qF2Y=")
+ORIGIN("pocketcents.com", "rX0C6QuqxkfJzbBrgMLfMb2+12LIY5KeFUB9V1Ot1Y8=")
+ORIGIN("webclicktracker.com", "5VVuv0OKtUVLmWOY9clA4BgcQ7fd1hkuYhvthxX/UJ8=")
+ORIGIN("appier.com", "8I7PyxxPoH4m7yQtMcHGq6ltS0m217mk30NCdqes9oI=")
+ORIGIN("double-check.com", "960rmB7psH3FkaufvjfvAhgRnKRfDKKgB86m503xbGQ=")
+ORIGIN("stat-track.com", "muhb6wAtUneh+r7cnW2s8GpzwfHl7BED2IWA6QQx7LI=")
+ORIGIN("sabavision.com", "Fh6A2K1b6XluE3o6sjJc9B1cJNqDnXuRWRZ8Dpj2nyg=")
+ORIGIN("sabre.com", "P1kJVps8e6DmOcVHltseqvuE7Ahb7wfJ6Gi2+1jl3qU=")
+ORIGIN("searchmarketing.com", "FeXRBc9S08/s3LIZ/KKq+MoC3+AubExgl1xzk7j2F24=")
+ORIGIN("ucoz.ru", "sxFwFMsGASa4vaBE07mSKS2uhEySbDzPaIDWBF4n95k=")
+ORIGIN("adsbyisocket.com", "QqMQ5kyir/hVU/BE4/uSkUBlZXw2bX6gKub92fuI2dk=")
+ORIGIN("steelhousemedia.com", "Qe9ElOgZne2VBtBTrjCyfGW7jOzbwbE6IAJV9Y6gcOs=")
+ORIGIN("adverticum.net", "5s+ae9gmMF7ztxFE8DKMBYP3PoNg4xOHCR8AVylMrdE=")
+ORIGIN("c3metrics.com", "rBQfTIt689pu2anCTNotawqoL4YAdaFX1DkCmCBKYKY=")
+ORIGIN("qualaroo.com", "gUulZ1vu/vFVAWpuA2I3hOT4/2SN9oQEkJOeRb8Cd8Y=")
+ORIGIN("lowermybills.com", "24wNKxq9y7uDmXqWNi84qQJXVbiZnBgAkWLu8MlBdKo=")
+ORIGIN("openstat.ru", "dN0fQ/Er9DXtu2j9WYLL5iEsQPLG18FZcnWkCZfTN6I=")
+ORIGIN("sproutinc.com", "bQgehomchXASb/VR4mcFgx834U9ETZ3HSeKpGK+FTlM=")
+ORIGIN("zune.com", "ez2LMRE5J/Lvgwy8JG7KgDzKuZLEiwotOQ7XVYFWSFk=")
+ORIGIN("moviefone.com", "3n2H0YVCxjJ+b/ErKc74mjL1jxNyZgklAi4CORZT3pQ=")
+ORIGIN("nabbr.com", "N3WE/M4EOORm5B0CTjIwW0/iGfjQ2GEfAP5HiF6wLuc=")
+ORIGIN("adworx.nl", "UpElD664AaoORabGTZqggrD6WjoGTQgvgzxFt22Ofkc=")
+ORIGIN("pinimg.com", "KCy4/gS/4jgdrcCIqXcU8GJsFXsjSVwEUBsX3cDCiog=")
+ORIGIN("tradetracker.com", "29kKXg5PaWFK+NsNVDkEJus8p/BDdR3IuXB1Sj9feOg=")
+ORIGIN("juicyads.com", "PtO6NodUOA97J1bU2TwwomZ8X3D0P7d/n9di5a5wPA8=")
+ORIGIN("amigos.com", "76odtdlW+oTxhtN5V/UPvNAoeEW9FUDMYeWgntmmWTo=")
+ORIGIN("google.co.zw", "/ggwpITSE8Y37+Qj0N48DrdI5XFnG+6qvKCcIPwvgFs=")
+ORIGIN("intensedebate.com", "z4R0je8lWHZ3++m/1mopmr8cCF049PghhfMDOSai1ck=")
+ORIGIN("webtraxs.com", "AvT8nsGZ1lCGxaRvs8wnoQqSM60IENz6u2lBzgGjo7M=")
+ORIGIN("com.com", "c6KKDWpLwNcMXBsAa9/V4K7JkodTz7cCNEsw6Db8CUA=")
+ORIGIN("wiredminds.com", "0sGzWwxDUUplV+S72n/wAPq2uVFkvjk/RzFeFRF54Og=")
+ORIGIN("netapplications.com", "eyPSfI+fQm7piiYhHQ2Mxw3N1ug0tko0b60aicP0mqo=")
+ORIGIN("google.co.za", "wUDP0/Y+TIB3Zr66HZkuF6DDudKL441t6B7vNdibIMc=")
+ORIGIN("google.co.zm", "/HoIxMUq44R4bknVHzQ9WxWQXi2sjJch55y6qlBeHLY=")
+ORIGIN("addynamo.com", "YCvtDJHG4f6kfTizDwHaVsYq1a93Oqmoe11OzAcsFdo=")
+ORIGIN("phonalytics.com", "4rz2mY/zrdDHKQh3/UwHCSa9vzplmoul3LiBdNjD7EI=")
+ORIGIN("tailsweep.com", "peZ+47Qru7eBiN0OqpZ6HffZRsb7qWRCunN63q+ni2U=")
+ORIGIN("google.it", "U2qNN9DGbXcbTqBk/6KoRqG4PlyBohs1JMKEJ/oakkU=")
+ORIGIN("google.is", "gBvknSp9eKN/v8WqmFgeiICGqUQy+HRuNa4A5I9Bcpk=")
+ORIGIN("google.iq", "+0y8kLJ5eY7cxZcrE4wq6nigj0BrumUSFptBFoFtIOM=")
+ORIGIN("ydworld.com", "iAlX5sCM/AsPeKF//yFhH2WbSq9bp3nGm/FWRSq/HIk=")
+ORIGIN("theboombox.com", "5hsvqyy1+oNrZ9tUlw6oATjDJWvtbeIARueyI6SDwuM=")
+ORIGIN("demandmedia.com", "F8iRhSEEJ7aC0xrTiUu7YylQl4FcReFLov9rgj/tMlg=")
+ORIGIN("monetizemore.com", "g5lPAFPBanYsk0yVUMNsSVQHuPMiStXf1NFiYbfBeOE=")
+ORIGIN("adxvalue.com", "aQT2wZQTZpRpHtitSXIfXVzOc/ieHGjPAWIxkcnXFWk=")
+ORIGIN("google.ie", "PvBSwtAZHLbYSwwZCcYakq9i8BmJwyXor/ZGpZZPzYo=")
+ORIGIN("instinctiveads.com", "6YMpF37RjCxldZXeZRRahxsvw5r1CSC1bpSwOYSkz4E=")
+ORIGIN("velti.com", "0efpZvDV39K6IBsG/SVz7cej//zQx3/wS2/7QFOW2MQ=")
+ORIGIN("researchnow.com", "TY1QEneMQukHkyzkKwQc+l7s114hOP1XqMzwRivp1RI=")
+ORIGIN("google.im", "580B/6jd5Cglnig9kX/kryURbUpTPFzfFNRL3XB65aw=")
+ORIGIN("socital.com", "qjdv7UoPpnMp13E/KcgQ9dWwnTYmKEgUAMy0bAZSF48=")
+ORIGIN("suggestions.yahoo.com", "sUtFAUwZwm/8k0p59gqPHLLFB3KMhhVKnXqoVXfJzhA=")
+ORIGIN("adnetwork.net", "lj+8DCwDS7UE9vNzHoqsQMbatyGJBkYkbbvSDtvt1P0=")
+ORIGIN("eyeblaster.com", "8WdyxvvUvALmzEYrJpsgB+EvzJGlFoqsf2uxVDMcLD8=")
+ORIGIN("xg4ken.com", "W4pCIdSPaCGIhNCuLN0Zq3lDKBLQ9Lrl28XNdXjvB8c=")
+ORIGIN("percentmobile.com", "Q84jmAJLkDQq2xRLsvP0/QLbkts1dbTt+OWDts9a2q0=")
+ORIGIN("33across.com", "FZgvrkh1i/+6HQpPo8zqGofoNXWrtVJJDnW+/wwJtOs=")
+ORIGIN("rocketfuel.com", "jAtAu1hVKvQSPT9e2xX7f977M8A0w9JGilkbX6ipl9I=")
+ORIGIN("coremetrics.com", "xIiREq8oi4GDVbAsIgFNJGkPsNaY1glpp2hcG0bXQjI=")
+ORIGIN("mobials.com", "tDxvo8/UcF2KlNFOty8yeibBQrbc4kbW0PjPRwGKoj0=")
+ORIGIN("awio.com", "wJetY9lUx1vvHOHeuZZd79efwS+notBd9yrucMfOBc8=")
+ORIGIN("thenumagroup.com", "AVmY1yjDeFqj+30fGuyIYUZmdWqNeKIFzBnXjhLpQLI=")
+ORIGIN("keewurd.com", "C40AcvyGrDtbYLN8rLRDTO5MbInH+zglTq//1C3ebs8=")
+ORIGIN("adstours.com", "mHbZ1D3Uiaihjb9YMkhXkw7DE3GoEvDx0LoQP9/OZeY=")
+ORIGIN("qoof.com", "pEDQapu3XbOxR9XVpPoPFUrI8x+la79O6XzQmp3A71Q=")
+ORIGIN("smrtlnks.com", "f+93xuCJd0s/3aJHhWmgncEhxgPqlrOBNeXR14GzGwY=")
+ORIGIN("owneriq.net", "2mdyMRhajhkU8MEKxkaJ+EaM1cWQKmEP0eif4DZPIho=")
+ORIGIN("webtrekk.com", "TdE5eGpDmxlOcye5v80DxXDVNjT/WJ0A9I9l922diFA=")
+ORIGIN("yieldivision.com", "fmitR4w0SxpNLAeuqtLL1a4Kjp/oSIqxoTDbPgeNx4A=")
+ORIGIN("jaroop.com", "KPzOF0B6s3xqN2Z8I1IA2VyZKHWnT+oNjnO79JUP9Z4=")
+ORIGIN("sitecompass.com", "c/uxDUITNfQUeJHDdSrBUccb6LkngROypU1EoAZ/7OQ=")
+ORIGIN("addynamo.net", "i0eWqs1MKpVP3L9hLrZROu6SPoorCg8ZmA4qSajssK4=")
+ORIGIN("am.ua", "AX/SI6SF349nsFSTCjxCd0izdKWO2DSqACPOak466A8=")
+ORIGIN("maxmind.com", "nZEEvaXcQqXVknmE88xqI+Yl2VI1RNQJv3Cp8WUfjoM=")
+ORIGIN("adsrevenue.net", "GqrX7nlxQn7OEqjmEDBVrypCwjyWXj3yDIbvQDY1/TE=")
+ORIGIN("acuity.com", "nOb7beXII3P/2pDnBsKAAYcwvLGsO8OTxhqDiuWizbM=")
+ORIGIN("adready.com", "1JSMn+0P/BayAovarC+Ti+iBsNDtsmhzPsfT0sqRHhc=")
+ORIGIN("mobfox.com", "JnLE/J0LV0YyGeQKC4DoFwxtRrYilBm6EIkXakWLowo=")
+ORIGIN("optnmstr.com", "mixnLRFrXb9/j4IYUpF6VmFURJS/KYCJRt376PiORSU=")
+ORIGIN("veeseo.com", "z+xZWvGNxMV/ATSYr9T0W8l1ctiYp5r0MrT5mVSG4FQ=")
+ORIGIN("google.co.ls", "eKmbkkxrPGi9K7U+ZX51nxj5LQF12aM0gxQuvCMY84Y=")
+ORIGIN("mywebgrocer.com", "yJW7xT48YNTutq6b2VZoJ9fxp17Y0nC+nBaRHlTMlbY=")
+ORIGIN("andbeyond.media", "zKMRvR3bZbYfeMOozv3iwI3kKgFh1qX6GaxZFThsSWs=")
+ORIGIN("amazon.es", "D98NwOvz5uUufPccYWxC4T1TGdCALCLT5QkhIHkstwQ=")
+ORIGIN("bluemetrix.com", "BiMas+YlzPfwZvOl0c9vwo0gulvQDfpq+8XXUC3waSc=")
+ORIGIN("officelive.com", "55VZH4JJ8GJrbhD8PKLMKIvGR+f09k82DJZZwy9X9Jg=")
+ORIGIN("banner-rotation.com", "R7Sg/bMr+7cnw4+glviPsWSFIhBykGTKJS1c8BKC3mE=")
+ORIGIN("tribalfusion.com", "pjIyqddRcHdfD+9mPWvwsOSFRsDzpvEKzON7w7xWa6w=")
+ORIGIN("quadrantone.com", "O8w1rWBMV7Xwh9RksBNPoPhemP2bKxuVPkrcbrmWZq4=")
+ORIGIN("pch.com", "x3eO8nRYA5dWIjEnwhP5RNcm0uWrgpNl+EdOeo8PxaM=")
+ORIGIN("adlantis.jp", "CEwGmCG4AyXFu6yI2Gp7EdYTlh7fdMvZuo+DfoFT708=")
+ORIGIN("adyard.de", "1JiCSZQq3r1o6oabYWXuGGXTTMahxHW74EdJk2NDOio=")
+ORIGIN("admagnet.com", "vU9/CGC+4RwUtjeChfiyU0YalMJsU120D5cvCiAia8E=")
+ORIGIN("krux.com", "lkk/Hs4FNlenTotUn0f9ruKbTh0bz5zB41q9fJ+JuVI=")
+ORIGIN("owneriq.com", "otZi6l139sg0LsCgBgW60L1IJP8G5meG5sQIP5YwGfI=")
+ORIGIN("newtention.net", "N8dpoC5YSuTqKH7DT3lWoX8JFA1Ad8m9YjXpo/XaQgo=")
+ORIGIN("fbcdn.net", "86fFUL2qrG1elBfJWnuYu3+Lk3FxGZSjRRjnMQqiUu8=")
+ORIGIN("skribit.com", "I0igNFi9QFB0smx93FA7eub18bV/MkC9JMvuBTDb8sU=")
+ORIGIN("specificmedia.co.uk", "lItXPpboiYMiGrYTehpJvwZcbr7y33bHlnymB+nntzc=")
+ORIGIN("livefyre.com", "ySVutV1Nn+gn8Y3YIjUEHBtf+iS8bKzsd/4jAWQUdjM=")
+ORIGIN("piwik.org", "5NNw0g60XfWQgCWA64Ubjse6eXrQJIZ0AoZJetYwgeI=")
+ORIGIN("scribol.com", "RGvLXspeGH/KddJ87QMVIluTWMMBm3v1EP78VuBVyEw=")
+ORIGIN("visiblemeasures.com", "MpWiwtyDxXsPwRrPkKuPdxWq8EKo57B10aTS4SWa0CI=")
+ORIGIN("marktest.com", "imUuVTwemnFG1GTrrHrkFOv82Rjy0BE5pinTSdRRURQ=")
+ORIGIN("onetruefan.com", "2CsQMBOQrcDthef0okcjbBP79mi0Gq3zDJtHi4Tdh90=")
+ORIGIN("verticalhealth.net", "kuJUrZobOA99Krqt9UhQ78ZW6CKcffzIFpitMIwjs8A=")
+ORIGIN("ivdopia.com", "wywT70v4SapDgoGc6fUctMKGf8tkGHUWEKsuTEwffSY=")
+ORIGIN("stormiq.com", "BxbNDSzQkqVl9PyaMXdHgDY7RaF++vY3vW07TA1cztE=")
+ORIGIN("wave.google.com", "4Y4GpMvNlIZIBarPZ9VtxKBAueBLCI+HWVGza2tkV10=")
+ORIGIN("advisormedia.cz", "CDh8MVo1zXBBdjth85/zNcN2SsZeatBxyfqdqT9/AUs=")
+ORIGIN("batanganetwork.com", "qp4NHmx9dw+MJUu+tcGWBZEMpWFMRiSbNqfcFYKH6ho=")
+ORIGIN("winamp.com", "SYJvfcgrNZPLVvi0x0vugmbwVO0gqJLIRp7yf8a5SEg=")
+ORIGIN("getsatisfaction.com", "FSlj1x6AJACPA1pxIoVbO5tMQ4a/dSMP4sWcW1D/VB8=")
+ORIGIN("istrack.com", "QRMTEIyHFLjFul4ODfuK1fGCTwwvxNZId0bKeh81QLE=")
+ORIGIN("brighttag.com", "urZb0cLf5pbUCXg1XhZylMlvg2uXP+GPQRArX9CYGbY=")
+ORIGIN("twitter.jp", "2v9NbReRqIwbkOE9UsV/HERW8ZxrY0MwcbUvtQ8n7c0=")
+ORIGIN("afterdownload.com", "+IdFGnA1bGVoI0na+F0YzRCbqEClkJIWlAnZQq5l45E=")
+ORIGIN("rsz.sk", "ZLtFnCx/zJepMh1hkzpd5ga0Z1JNoSJOgCVs6exWSgc=")
+ORIGIN("calendar.yahoo.com", "0vYyFXrGpg/smZlVrOrimBG6HnJP6k/cQZbhpTmNsj0=")
+ORIGIN("ltassrv.com", "ILbJ3P+o+EZ2optE9n0+wDkKVYUmHXaN/jNJKtNn7WU=")
+ORIGIN("realestate.yahoo.com", "gzo1NwI+VZZg/fkbcBie6bMPMuP9iE3hMtvoM1lkErA=")
+ORIGIN("heyzap.com", "N6H0ammEK8G+3Kddn/1d9dDOH3zSyQoJAV6zMc0U0pE=")
+ORIGIN("translate.google.com", "2ykQOuwa4AMt+DYKc3CM+rngwxuCnHPEqx+F+pSFKrM=")
+ORIGIN("youtube.com", "LvOZqM9U3cK9V1r05/4lr38ecDvgztKSGdyzL4bvE8c=")
+ORIGIN("targ.ad", "0EqYvNu6YwkUDKmjaTFucMum04oJAhrcBIn4tW4bAPQ=")
+ORIGIN("apps.yahoo.com", "BviAZINJG/oh+zy1nIBxCzWO+1aWJKu6pf7KMjgKMLE=")
+ORIGIN("icrossing.com", "tdxmgtSf2aALNbqlM3M0Ewda1Zfu8mNa7W5YgCsFqQs=")
+ORIGIN("dc-storm.com", "B6+pB7Q6SU4nDBoFAVu2DBX5/SxBXmLAPXddVt5NwEg=")
+ORIGIN("tmnetads.com", "FKPWUWo9692lAMYcGAqYKNP7jfyvHJFvUZRfEgqW2kU=")
+ORIGIN("tynt.com", "5/z7F96vqBrvqBnCnptQjrJGvaEHJCLj4TD32K76Iz8=")
+ORIGIN("sports.yahoo.com", "rTLgjxadYV44w+liZqrSdCJRtYrxGRnoEWjF3pEgsKs=")
+ORIGIN("tumri.com", "WLhr/ZNDswg3uc8Y/aUuXPdwol2KpVRcdNYE4gbLMzU=")
+ORIGIN("bidvertiser.com", "u9S5dwNx3jFYgJz7PBNqvUhQhrrwbmw4RvUHDHIA2d0=")
+ORIGIN("betssonpalantir.com", "6/uYP15SiPCXO+rVv85pG1OsPGwEZe0WDImaQ3mtGZU=")
+ORIGIN("socialinterface.com", "kjsu2xCXea+7E38D/7qjCUtg80VL5e8mAWAHP6yLN3k=")
+ORIGIN("adfrontiers.com", "FnB9mI4KSs3vQcB5eav+vWwFRs5KV7wXqkmW3AWVqok=")
+ORIGIN("autocentre.ua", "5EbXtoNqrqBh5HJCcrZ6r5Vba5Zbxv1EUCk5aeB9FN8=")
+ORIGIN("adotube.com", "TWYwR0mf1lpDrhn2abI8HsIg1PdUMcfgpsad9Yow0OI=")
+ORIGIN("yt1187.net", "E8gRuIMxYEzrTiJ6xZ/i6ylESlMA9XzkImqMmSa3Qi0=")
+ORIGIN("pixlee.com", "XAY80sQSgBSgkb1hoXGDv7UfBYfRW7Yo8j4RnjA76KQ=")
+ORIGIN("bvmedia.ca", "0PNf2vqDlRXjhW++No8q0AqN8FHat6Zbx1TVw5lB2Pg=")
+ORIGIN("mediaplex.com", "02wN+pw3B3dlh7pw22uUuFPceDQ7WwoAQPAUbwXyyXE=")
+ORIGIN("webiqonline.com", "sVIqa2FqMvDxJFccwQyUQXAbvTL4JLM1gEjET8gWb04=")
+ORIGIN("complexmedianetwork.com", "UOZIHG4Y6OtcMPgVKytky5aydIQkW9y2t2ukSkuBkdY=")
+ORIGIN("esm1.net", "WpsYp2gkeAoeHRgtlnNxe7cbteSrYtX6cv02zPHy7UQ=")
+ORIGIN("lqcdn.com", "dr0qbKP1X4iNTa+7W1uDw+rP4mW7qwcekm/kJvpJf3k=")
+ORIGIN("theboot.com", "fc5cngUUBDBsxuXT+YWQbOpdo9kIrAfKdi/81g888Pc=")
+ORIGIN("typekit.com", "srgLh3TQTSNcLcGTpQvT51iybygpShrTeut3D36hFZM=")
+ORIGIN("adblade.com", "uVm6qGD4wvPVxLRKN2LSx/2x+VqODYTpRWkHBMzlwfI=")
+ORIGIN("lypn.com", "Hmc/SWPalBc7Et/5JtU+kIsuZl1IPaCHGNFM0gaeuH8=")
+ORIGIN("hit-parade.com", "qaREBOakRdNtXF7ckaRnYSyfUjw5gVb7hgKmdkVbilA=")
+ORIGIN("lptracker.io", "zesFc4ajX9O4qLQxSITwKgOZcL/nZrTUzyDLZ/EKenI=")
+ORIGIN("datonics.com", "112FmpjR9+hTH4FHFH1mzjWZj2BL8vPx7UlwVuMu6Eo=")
+ORIGIN("teads.tv", "UIDqIe6gwzqXsUxA3qWD/1C/LbVp/cvZI1L4L4bkEUQ=")
+ORIGIN("grapeshot.co.uk", "OzwfbmF7qS2tJd4YX1DSLrvVwnBV6iyA0Y4w2CtrJFo=")
+ORIGIN("nrelate.com", "y/muysFZKKKWWUL0binVGYR6azZ5WJ7qSzniDTQUGUE=")
+ORIGIN("infra-ad.com", "Q2R9TTtKb/h9xUeQ9+AdymVgld7sXfL95stOESKjlrM=")
+ORIGIN("developers.google.com", "+UDLhMuL1h3C/GXUlim2bopVQNpVlTklpJATRbAnbhQ=")
+ORIGIN("genesismedia.com", "etQ+E2hQPtUwj0i/dmxIlqqHipJHQudPYHv7GIQCp9g=")
+ORIGIN("adyield.com", "ganOVedK9uhPcjDITZKDiElpdJrQcZYoNgLwmB/n8IU=")
+ORIGIN("mythings.com", "I99tEOtT0u1peCXT7LalhPYNj1644CLrvpVgozcLuP4=")
+ORIGIN("bittads.com", "xdwxNPtiRsb1+Qe2jgZh9M36LvkCKGJzALmU/v8PIj0=")
+ORIGIN("google.hn", "Dy/jjJXGs8b+ORf91GvlXItWT2n6MVgAgiXw940/J8g=")
+ORIGIN("shoporielder.pro", "5Rde3/5aqgsjR35gDTKnrgF3HntiokioKm0SADcLoG0=")
+ORIGIN("list-manage.com", "B9REVtHoWG7N1VaMhnWVGn0SGt+JNFcJfbCYEQxDdLI=")
+ORIGIN("epicmobileads.com", "JANJ/nOy7PgWfFzLt3prSZaiKrY9/UYQEvEPstcNOEY=")
+ORIGIN("mm7.net", "E3QODwOWnBhIM0RbfCEzjkG23Rv/izRBUzN9EosVTxo=")
+ORIGIN("amazingcounters.com", "ZMsVmhgYge3gYoJnA065Tq8q/t8fezpRhSvgsyVSxZE=")
+ORIGIN("adextent.com", "b4eY4qs0DguhACYg5wVSvS6BiuTCf0xKcrk1o73zyDQ=")
+ORIGIN("distiltag.com", "ulda9DZudAuoclaAcB4gNNm7KfYRuhPj2EnvF0DAw0Q=")
+ORIGIN("google.hr", "BaJjjtBMjjRuhVYYMG6xq6fTiRMB1LuWFnLHj0+TaLg=")
+ORIGIN("google.ht", "bHo9kzkzIYFQkiv+59zwDcA8IRPymm91V3TvtluTqc4=")
+ORIGIN("google.hu", "4kTSC0vRHAFGzf15jq4CJ+bmKR5r9JtVY7Kww7vukK0=")
+ORIGIN("websitealive5.com", "5fy4vAsSnpIckqqZyPZXHFsix+ec2a821/QcBCG7BqI=")
+ORIGIN("moat.com", "7mbfW33kpLnnNe+DDRRnv4m+Xj6A5+d69Ile0LL1+hc=")
+ORIGIN("bufferapp.com", "QRIEpR8vsyFZcagYbHPfJsbDthHfvBaAIFoxBpX2ccs=")
+ORIGIN("bigmir.net", "wgUkPrzStnl5xaiujN+UlWtGra36z8CkW/UJd9b+9Io=")
+ORIGIN("adap.tv", "MRLV3etU+HaP8l7UE01TklDKh8B8ZVVHuRs6irHmN2o=")
+ORIGIN("buysight.com", "H6OceeSCljDkVHy6CUOYQJfqvyesA+bdEedWLBTzI+k=")
+ORIGIN("shorttailmedia.com", "xTrdgMvfveEEUo7/3afimH/IqkkajB6mKiolsdBykKo=")
+ORIGIN("xbox.com", "6vyeJig5j8x2i711keRiw6nye81ajp+QJVUFCNyalW8=")
+ORIGIN("overture.com", "jdgYtaYuFkOYs5oGUBWzrHJ/pXuF3wrpbrh0LsG/V0o=")
+ORIGIN("widgetserver.com", "RTYSQib4wS+K4w8ABGPBpBALRQfm9L3aHxUCV8J3fFU=")
+ORIGIN("groovinads.com", "JQtBIItBVTugA8RoPuKKlEP5OJ+Sh0F9zfDdPvC6CR0=")
+ORIGIN("adx1.com", "GJ6VCQIK5VALgVJOr5uEPCiz+z9BOZSbozm9x/baU3c=")
+ORIGIN("burstbeacon.com", "pgYZpPX7EnAgNgNGJ4kTcQF0EWz/fHEgtQqJbVvRHIA=")
+ORIGIN("mediametrie-estat.com", "V+sNc+pC3f19LL8mN1KKlkVCH/YJUwUzCL9PUcRdpUY=")
+ORIGIN("storygize.net", "ftVQvd+KftlaUth8xl/hjTqllEp+CQ9O3qCpvj9jj4s=")
+ORIGIN("popads.net", "/fkgEt5vedNSd14DJbtGxJg1JieOlnulAqDzP5V2bXc=")
+ORIGIN("getclicky.com", "eV4l3pf4xW8yHUL2tij1pz/wdKuEcNp+qwPqesUaUlM=")
+ORIGIN("federatedmedia.net", "aH8PfwCWdLyf9rlMIbcRKOU/oyHYs8ba4jFQvcP6018=")
+ORIGIN("yandex.st", "hFt7YrxUOaykrcRcS3TekcbprTizjD4wq0jnH5U6kvo=")
+ORIGIN("linkshare.com", "moSMU3yBQcWqQL+S0bJ112nrSDPW3eFcOooyHs8DoGg=")
+ORIGIN("music.google.com", "rvOjI6Ji36k+lOJ/YUC7mZEuAWd+aS7ZYkQBSSMFSS8=")
+ORIGIN("markit.com", "pLlnPagDzZ1PfK83dhF07KM6eHDMTxBmlUNyxAJJCPw=")
+ORIGIN("certona.com", "3YbOk/LouPoBgnvoSgO6uym7uJd/B2sRq53zGq7Z+tw=")
+ORIGIN("facilitatedigital.com", "KpgFYYRxi/FuH0/cKg6V/Zlc2GTFdSoAUSmmkfXSbrc=")
+ORIGIN("ewaydirect.com", "6RyqcJUeRhls4Ozk3C6hdjO4RyAyOfQn3GeCz80r2do=")
+ORIGIN("thecounter.com", "MZ9ixaYzmSFsfLZyyfNMTbYqyy5l8IRGqnu8ZinillA=")
+ORIGIN("sabrehospitality.com", "Pi3jrJ+YHetrwN2T8SCSpudS/tEseuUcLiAKNhLI8Qg=")
+ORIGIN("evisionsmarketing.com", "79S6ftm8UdgigApAHlKmd8hISSsmCFVwiTEOt9En37g=")
+ORIGIN("suite66.com", "Omr8womR5HZuPptH3c6R1QgSy8DzG08evsinBig7grc=")
+ORIGIN("xa.net", "F6RTdH9u4GGI/bgloB8TZ9rU/pJaoRiRnw5mixuRG6A=")
+ORIGIN("popunder.ru", "o77kAiKL2SaKxWTTI5MoU0cZbf8dwl+j9WWz0nZcD94=")
+ORIGIN("websitealive1.com", "sGXv5NYqQILO8O01kfaSYKAuB6YBjNObPtDo7jDx42w=")
+ORIGIN("ucoz.net", "93Rb2uYd0aujdBSZiRQc41eB3wUxYl8q0LBTzwBDuBQ=")
+ORIGIN("mediaforge.com", "zI0JGQYn/g67QdsaZI0bgqSlFLO9AHDWq2Uyk75l97A=")
+ORIGIN("support.google.com", "639r5sthwF+y5H2OmZQlVNaxk+LAZmDjSQCmMf3bpFU=")
+ORIGIN("adcloud.net", "sWETIIclIQpafU+HzjzSByMniEl8siVKxP46udDglPE=")
+ORIGIN("adobedtm.com", "cu09xEOu6TPET/wqV1gaJ10DWhzyHAfLnNilWVdO7vY=")
+ORIGIN("technoratimedia.com", "EvPJLc1scD/g22ExyC8xZqVRl1dDKpga74ApSx9OpnM=")
+ORIGIN("clicmanager.fr", "Kw/7HtyR4anW/seQsw2PzjHHoEL53FJinDwjc3/bR4k=")
+ORIGIN("redaril.com", "lt2muU5Yk8FjpfiD563qpkd8jDCV68tfn+RDnlIGnEo=")
+ORIGIN("lfov.net", "cmWghawvYlkDklu2b+k85mEqU7c/XCHZrWQN38btbr0=")
+ORIGIN("adaptly.com", "k/b3JTXeDn17ayvqow/mGo5yvPATRtFkPTS7OorQPLc=")
+ORIGIN("idg.com", "S42GOZr4VHeRApEQgQ4vpo82pYynNb3+f7jUm3fGuao=")
+ORIGIN("viral-loops.com", "muKTm6IMX2nkAgso7epGhlHGFtZNdsZ6mqOFn5y9mN4=")
+ORIGIN("rensovetors.info", "4l8JksdA9j3WvaCwA7s0izjKDieqvwTy9egHjKdDvxQ=")
+ORIGIN("impressiondesk.com", "RI+fxZdxVuo6nvSN00jDmGNrPkSJ91P7+dX5OJX4Y98=")
+ORIGIN("linkconnector.com", "CytQHsuX6uW8qbCNz7j//V+ozpB2PhiqBwRC/gxV1Wo=")
+ORIGIN("engineseeker.com", "KlQ1q0Fd9401eHnwvHlQDB6fM5bSiPYhN+Ghp/V5Wac=")
+ORIGIN("monetate.net", "QD7UJLpIjqejHo4Z8PbABYIv5KyHaTQf2LYzc+E1Uu0=")
+ORIGIN("luckyorange.com", "nzwRTu95vNO2eAHPikhWjXEgmgItIUIIvx0iEBFFAOg=")
+ORIGIN("adperium.com", "NYqcavywSEeEpXoHTPTaBoYgBYxNOuPIGy23EX0UQE8=")
+ORIGIN("belstat.de", "U85QGR768/RpjbgP23zpx3z2eHtMBR3mYkCOaKBmNj8=")
+ORIGIN("aprecision.net", "vwBqk5A8GVNM3ZqdIDdAYeuhiOvau1Piz4aDEppnlpc=")
+ORIGIN("proxilinks.com", "lMEeSNZ1ZkRWIHd5dqVmn4AXnvOzgg2PdP3H/O9cSQA=")
+ORIGIN("google.com.ly", "ixvCgu9B0REEoEM2z7txf8F37bpq8GPdMvgXgUgQFz0=")
+ORIGIN("freeonlineusers.com", "Lt4scl+lzdsRGvhMUxtZ90GjNoY6H4+RJfeZU58/oE4=")
+ORIGIN("rapleaf.com", "YXrn84yx5ne29xPEnhJbPDr4qlM13ukGq5+6WlVjwTI=")
+ORIGIN("layer-ads.net", "QeLBcs0MrSBx1YS9sQ7rAm/u5lvdisOhUixqMK3bemQ=")
+ORIGIN("revenuemax.de", "eFmeTnSdLgQ8uzcs52FiJSdbdOAVL+oYvpk+2MSLYvY=")
+ORIGIN("smartlook.com", "o2T+rQMW059B1CmofhGBOkAb4UqAkWNlrH5pcufN6UI=")
+ORIGIN("retailautomata.com", "Pd3njCGoqhB8zOuKJNliJWRkHgymcd4OXiS+Yphvn8c=")
+ORIGIN("adhigh.net", "qSO1yHzUYEhBq6DIjDxpi4vmLzcFwrXbSkEI35ZG6hM=")
+ORIGIN("doubleverify.com", "W74F2qsLcFTSfrlMHfI9LfxGcby28ojsJTyS0MfSBbM=")
+ORIGIN("instinctive.io", "+3BMm0M53oUykDFFjq86MoLGJM0JXBelwrwJKXlQTw8=")
+ORIGIN("avatars.yahoo.com", "scZnkdTVGzYoW2oyp7Sh8feDBn4HgAJCTYj4SC/7W7g=")
+ORIGIN("rmbn.net", "N8MALaG6slMpsuwQkdL+nLO2XNkzCIjVhhSRPxILyOY=")
+ORIGIN("rimmkaufman.com", "eA2BHbGwt1OqFrPlHJlaNZtEzxmq9NqlXShX0K9bSyg=")
+ORIGIN("cpalead.com", "hgbjJCddD6gjVn7EskJr3638USW0wPvTiAceR2WHJ2I=")
+ORIGIN("statistik-gallup.net", "RFqyMMnhCQ9YsLFYOL6Uibs4jZ1ugr5jDmZex61gr48=")
+ORIGIN("plista.com", "lsTii4yqdmUO3tCitWZWM/TuFLQTzCHcn5976GTzqjA=")
+ORIGIN("rkdms.com", "Ot7LqVfkyvO6X6zYd0ovJi+YYRUgLzLxURNxSn19+Rk=")
+ORIGIN("getpolymorph.com", "uIrHZ8cMpKmNuEKLWuwxw2UqFSB/P+K833uG0rQ91Zw=")
+ORIGIN("guj.de", "VOJZUJtHndXxauMsGVU/SRFtIqvRYlCveQR6N05tbIM=")
+ORIGIN("google.com.sg", "0y7cTz5fk89VB0GSi97XnubOIgPt+bvtElSf1b8Ugq4=")
+ORIGIN("google.com.sb", "ofYwW4O6dyFptaNlIVJEpetu9wIh1D6eHDODk6/HA64=")
+ORIGIN("google.com.sa", "5zpMZmET2yKNY+Sk7p/gTgwAsx+HqS0tiSuC32ttLww=")
+ORIGIN("triplelift.com", "fvTKUs1XCsmuuqzjSjNiFPfSlT9CsqoxWTcBev54rgA=")
+ORIGIN("nielsen.com", "n2CeFmwF6QsgAqDbaAD1fQxvRBltagp68jNbOuHHEWY=")
+ORIGIN("google.com.sl", "ZDEB/DtHg6kofLlhRF9AuyPY8tII8tGS9WwvRcprjJA=")
+ORIGIN("talk.google.com", "b5KfS4pVDceMUMJSj37hEOSmXgLhxIwR+AmPA6RYtYE=")
+ORIGIN("google.com.sv", "/hzOpQR36lDRjFqpmKc5YlcE62ss7VV9lmPCIFVZHiU=")
+ORIGIN("jemmgroup.com", "LyYB5lA7hw3RYSGmnqkiR7oBQmirvVxI8txJO6LGeRA=")
+ORIGIN("adfunkyserver.com", "8WPqOuNIPCQ9BF5xrw4+2S8jmOZw5pOcHdhP/A99iik=")
+ORIGIN("finance.google.com", "5V9GIwZAheN8qG8Bt3yWYjlNbD+aaCV84EROekwumMY=")
+ORIGIN("engagebdr.com", "wWzsEqm/Wgto4hSlH5EWC7U/LIvZOQD1QYGtKrxbGHg=")
+ORIGIN("imrworldwide.com", "bAUNKY+TxAWdHV9w2u6DmjoZR8jSdEFDvhq07c5GdK4=")
+ORIGIN("motigo.com", "IR1R5Ufm9lpZzTcAceL9sjIuBsUL3y65hnK/oE4MP0A=")
+ORIGIN("hotelchamp.com", "vAVyJgsFyPNQsrLvj5l1VOpKi0Lt4OM4ukgwUVdWUPw=")
+ORIGIN("getintent.com", "vQH1p9iwJgxuXLYr/dCARV5CV0iXDYC6v2a5ECB+Acw=")
+ORIGIN("etrigue.com", "1tgiK0tnFjSLDdHDTuYcnVPtG8QSkfcnEh5d12HxPU4=")
+ORIGIN("ipredictive.com", "Ldc3H1bvXeD97IBtmwEXa49RYSeSc2xdqChcUtWuusE=")
+ORIGIN("fastclick.com", "cprcmOh6mfPl1+SpomeZ6uWsKP4C6eiKqZm56+3/wp8=")
+ORIGIN("gannett.com", "FoWMjIi4KnWz4wsE4EVEaf9eqBM6FwzYFpQXAvj8/aY=")
+ORIGIN("luminate.com", "1WSp/mFP9jnN/8D5MX5yt99E7GnG3hR2bpIXcbJgvbQ=")
+ORIGIN("keyade.com", "F8/ccXm0Ac/ggV6f3/jV4RxAclBQsakRxGQ0YiGJE4w=")
+ORIGIN("srtk.net", "EAVwtxTsVJXhDT9e7+Ks6pD5s6Z6pFReYIbe98+yUEE=")
+ORIGIN("belstat.com", "HITWA4OXFMlSDt1HWwzb9Xh7EpciaCz2Kg53GoNuhwU=")
+ORIGIN("localytics.com", "mP0oTjyQ2NDaDXjY2MFqCJpNx6DA5SrXwDDH8GelW/o=")
+ORIGIN("js-kit.com", "9eRG2SRmZTWJHVsEqFWx9QUcgcH1R1slOrdYqk+y8i0=")
+ORIGIN("ipcounter.de", "bSjbacEgw2Z5gnVIIHoOcFKE4FGqIUK6F4R13HtGhM4=")
+ORIGIN("aol.com", "VpPrtVw+1hvXI7rOU7K/i+ovPOsjq6CyghUal+4E3dc=")
+ORIGIN("tumri.net", "xPY0IR/3iuymK3Cw7oGD465viz2ZuqS7yZ6aaRQPce0=")
+ORIGIN("contaxe.com", "juVQPkEM9J/6JEe1KrSUBKlXixX2BlDKCjCpYILDZSM=")
+ORIGIN("marimedia.net", "y0ZIr+pLi786Yrz8tICKUeX7qCDQgFSUbfbJBTR0iYQ=")
+ORIGIN("adswizz.com", "or8h6Gw8g+gT3e3KPQu1hEtgsMmKAb39WNUVLK4NtBs=")
+ORIGIN("noisecreep.com", "A16Ikz2q8874FI2HZvOUqJeel2kEkNsHeKdJveF8lgI=")
+ORIGIN("600z.com", "mKj5fw+rktatT6SQODZjOoUD1/uZvOOn81FCsR3fhj8=")
+ORIGIN("applifier.com", "PeWNW2NvmkL+VfaTyLUfNWa1pY1JojAwzH7X5TXqoqA=")
+ORIGIN("contentabc.com", "+hwvGbypUdpuRyjKkPeqq0C9868o5m9JpQm4UUU4TaE=")
+ORIGIN("adsfac.us", "5lUcDe6XA8ZT7a7xhcfQZN5hb/XSpflr1h4ifcHwwsg=")
+ORIGIN("scandinavianadnetworks.com", "X/cxusJ4St++KvMOCljt4ygypFxZnb2G3DOjFlDMW4E=")
+ORIGIN("books.google.com", "9fleVvXABvxVKjhXpqqQuzY/xAGZ3HdMIEJXxfknhmY=")
+ORIGIN("admarketplace.net", "BB3+ucHXA4SA2tvbM7C43A3CCYTiUL7C4G/rgZk3gic=")
+ORIGIN("viglink.com", "O/X7R+xeDrOu0Vg5ZpFW33YTb/bCMCKddIOu2y8kl/A=")
+ORIGIN("tiqiq.com", "6LKPaKJ/ff/cIIZ0469COVmWmkDeU8C4jxsjwibVpH4=")
+ORIGIN("adspeed.net", "TVloZEa9TPnFXwjX5njRIGYytHnJUP+WGlv0wHSv/go=")
+ORIGIN("adsrvr.org", "1SdIroP4iTri+eWN1aTcVINc1aa1bwza07xa4r4lKXA=")
+ORIGIN("xcvgdf.party", "vUouwesxMaDcFi0y2s3kLwXDVmamAA1upoYqY9WMEuo=")
+ORIGIN("adserver.yahoo.com", "6b+JsMVq5gMP1kvPLCIZEy2N5yNdmfdQtWi/JpDriGs=")
+ORIGIN("synacor.com", "KeKiBduor85rvnPiqkhf5s3NkVmSPbxQzIzavljKNWs=")
+ORIGIN("websitealive2.com", "afAci3ruzA7FdXqinGCllTPuPQg5o67IH5efwbOooeQ=")
+ORIGIN("sessioncam.com", "YXQwPWRAHpYwm8yKIK4bI5XLOUO+e1MI6dczGPd4hik=")
+ORIGIN("google.com.ai", "HLNqJnsQ9WoyKrfdzCS3G58XcGqry0qj7iRSU+RA3SE=")
+ORIGIN("google.com.af", "gqCv5xCojj2GELft0wIpCYBQu9FnLqDrsAR0EWNxKTg=")
+ORIGIN("google.com.ag", "WLv5omNpSbHsNrQX/a+Lt+Clk4az5sxM9U/ypkF0xC4=")
+ORIGIN("proximic.com", "R+DH78rBJ6AYN746EPfU3dvoI+Z1Lwa/NMnNGVvXxbg=")
+ORIGIN("act-on.com", "NSdALIRE4WKYrhuKcS0mC07s7f/W1YTEQWNaEZz6fj4=")
+ORIGIN("google.com.au", "wj++R3urDO6J5UF0dOqhRiriBsbxcN+yLcZgp85FVLQ=")
+ORIGIN("reklamstore.com", "MwA3OqJVe/8HB4CoZdUCwo6tLdnlH9TFPcI68Pofozs=")
+ORIGIN("google.com.ar", "2ylMfrX1xah7lWeAoO4Xx+YBzXdSgmUGiEgsJFVxqHg=")
+ORIGIN("clearlink.com", "UM4V5KJ1t99QaNxglgbynlMsFaqCTaOO31yH4TzmF9w=")
+ORIGIN("clickwinks.com", "cRxbmtCdAXUxmlg9sM2amI6S0IzoEOJ5uDApQDqMdto=")
+ORIGIN("tradedoubler.com", "4Q89ZfqA1XJB5eNLzHk6IsQlPqNHc/uG/R+qCn9sPfk=")
+ORIGIN("adotsolution.com", "fQEF51g72Y/qIHN96sDO1doj+yTAh20xK5aGKMOaMJ0=")
+ORIGIN("shortcuts.com", "JN3REEdW94Qs5q9WK8AglE0QanOLzUg3+q29qiAhDJI=")
+ORIGIN("adingo.jp", "vcBM3JA2SF5WEcbCodBvFoOnf1oCwb8VdakTApnEgG0=")
+ORIGIN("trafficscore.com", "3MiGABVx7fl8rkAiYVuWD2VBafl4030yuW1E6MLlKt8=")
+ORIGIN("onad.eu", "VWw+tdYT073mEYgV67BaSvhEqheSbJdI1QVtNgimNPQ=")
+ORIGIN("aerifymedia.com", "SwWXYgM1tNya3zi0brL/SJN3Eva0tgynB23TSbACyr4=")
+ORIGIN("adscience.nl", "Rt4Gqw3gW3UhTEFzkanXJMsM+iY/3jBFUxXVCnihYCM=")
+ORIGIN("secure-adserver.com", "fmwQUC/tP0PhpNhJQ4HINYI5XcJQvyfxGpsuzx1yQYI=")
+ORIGIN("aolcdn.com", "cOBb1eCzy1D5jIuJ3mMj1Wc+txQnaVMnj5307KeROzI=")
+ORIGIN("visbrands.com", "OuZvmBHc8vLu0digEOGz8ci2TfsQewyagVeaud0PkyA=")
+ORIGIN("vgwort.de", "LEkG+eOrEZM8B2qeH/nHjD7pEUNqbLJsALOBSKMSxZ8=")
+ORIGIN("blogher.com", "rHJBkZ5si3s/Jw7vKqIuwrXp4LsIrFqzzniRI7nNJis=")
+ORIGIN("mercent.com", "Q6AaAeLdcLtJbF6WFLheiYYBZQKVq5jiBHRuC2JdwSw=")
+ORIGIN("yandex.ru/cycounter", "m7Xg30ieKBLCyA/TmJzAqDa8lOBZR5CqWswTuzVDUFA=")
+ORIGIN("adsnative.com", "YpKgfQ0hB5Pca+9CwLWrlGOn3pCZJHTgvVz3cExNJKo=")
+ORIGIN("brightcove.com", "olHyu01w9/aX07WuYU1qCor8TxYC//avFaV96b5SOP4=")
+ORIGIN("richrelevance.com", "sy3oBW3g6fkwdd4vmGVfeRSqASIdQ9e+6mJnJE6pJ1Q=")
+ORIGIN("yandex.ru", "pCZT2iEKVLaHTzfw1KEtpeibtDbyxqAfgyRucc21ROU=")
+ORIGIN("dmtry.com", "ygk1S4+c+eQCzznpTmZrp/cyA7o32B23jVE2meGIMSI=")
+ORIGIN("ybx.io", "jX/PoyrR7rYqLNNuj5wdQOcDgiXtUZv0+jIe0OIPPlA=")
+ORIGIN("ibpxl.com", "kLvX/lZeEw77QXrCB6z26wZbBo78f1XF5cz2ei7wZkE=")
+ORIGIN("yesads.com", "r2L+dzB179bpqNOfoDpV4lZnJ/9QXDVzwYOuHd1G4kw=")
+ORIGIN("everestads.net", "PtpGWlz98somavzJEfUcaHyTJpM7Lw84N31xOAceKf4=")
+ORIGIN("vserv.mobi", "ixhNYW4Opo+b7/3r2vHsm1VmDYNHLzGeXmwbsVlgaVc=")
+ORIGIN("baronsoffers.com", "Zb6J0QC/MSZQ4OqSAfmErFTg87kfHbmUqqntt69d3F8=")
+ORIGIN("oracle.com", "rzXeKt+U81NGy+4wIbrNTKXFi3ugf62yEa2YKvtUlO0=")
+ORIGIN("zypmedia.com", "obNUX5cIQyTu6ahk0yriQFmjIa6mPfHl0QaXCE2Ltio=")
+ORIGIN("cmadsasia.com", "tf61+ecR8iOkxc3UCD0+c1NOdqgP9xPxrBvCTvXOE6M=")
+ORIGIN("agkn.com", "KdByr4N8/MdNcPJzOwLN5nc1eKfMvTEkez95JzsdohI=")
+ORIGIN("eulerian.com", "YFIK6sbINhUrKWz55EhyvfdP7Y7C8y9OeIWKzZ5ycKg=")
+ORIGIN("adaramedia.com", "QygYtOQzJZOIZaLkU2+CiwvPYn6qNEX8THz+X5eLLhM=")
+ORIGIN("healthpricer.com", "sUXitjYwfMMBpd1wEEXhfmzVn+rihQ4N3oT6S1qSQdA=")
+ORIGIN("engadget.com", "cw83oeY3rKJjcdnKqZlamAGLlq7Trm5VrVr+2oSkXbo=")
+ORIGIN("bloom-hq.com", "nAoKwmSE9bz03DGzNI5F+6QN1RMeGsjn5uhaG7r9mE8=")
+ORIGIN("autos.yahoo.com", "IQ6RptYgUAhmqlvpc8Z1rHsgX3fWnkpGJ2NLkK0aFlI=")
+ORIGIN("mybuys.com", "l48Ii/o6b1AWTaeesPeqla+8lOqLa5dW6SqQqpoV2fA=")
+ORIGIN("everything.yahoo.com", "BVVykvQDuYIjBfpGvzLjD4keh5yBPLH+lDMk0ELgG0U=")
+ORIGIN("nprove.com", "fwj8VG8tebk+DLLuBCAr+4J3oc2QgpBHpyV7xPnY+Sg=")
+ORIGIN("admoda.com", "aoi9vZTRk0hmx9krdN/5vM4NcTyrV716cOuWT2SmgHo=")
+ORIGIN("microsoftalumni.com", "QcUoOaSE/yop5EFnLmpzvMggUQxAubtYb3nU7P6tFFM=")
+ORIGIN("congoo.com", "+N3qJ5CD/ur0O3guxumfX9JbluXO0Or5jZzqcMsGbUY=")
+ORIGIN("adsbwm.com", "J4uVlJedJbSPZdwzjLuARm75ijTUX/TLLDY2o9jgIeY=")
+ORIGIN("uservoice.com", "fZC56+ANvKTW6lW4YsZvBtuAIqEJouux1/9j1+YdSxk=")
+ORIGIN("gw-ec.com", "fBNHG/eL6yPly1IZ5FNyeuvhREfHTR/drFjtFVaY6kU=")
+ORIGIN("buzz.yahoo.com", "BIZPfvuKP/Xv6yTLpvs3gAOpob6FYf78GrnqGnO3xP0=")
+ORIGIN("ubermedia.com", "alKytBgn2TxamPHToLSeeVJe/Z9eJriz+3SgLXkLMnw=")
+ORIGIN("goldbachgroup.com", "YPgqx82u5+p4FqdU0V+NuOyZwhHFf17iN9nadeMpolo=")
+ORIGIN("valueclick.com", "Qgd0wwfMgtxg5wbd1yJI/tLjsXi5ciD7GtrkyK54wIM=")
+ORIGIN("waterfrontmedia.com", "1w9EIOvvo30BltzkE8bMt7kJrfPbxbJIadNV8ggYmpk=")
+ORIGIN("operamediaworks.com", "2ijoyxGV+uJqrymQ+WragchadMo3aX1B5iRb8CMxJAA=")
+ORIGIN("google.jo", "8nDeN6i9YVGcnWZZ2FmdazsxreM0fHaH5lHx8M7dHGw=")
+ORIGIN("microsoft.com", "D5O8juISn1bvLlrkTgBrP1j+ORAyXgxOVCLgPM24Mes=")
+ORIGIN("toolbar.google.com", "4a4xIyqheVr/dpWBga9xbWSHn5QkRrRvDMgtYxM9zmQ=")
+ORIGIN("up-value.de", "0QYe3hmMTtJstkK8hYzo4wGf7I3Aft6/BQBjkCfgHdg=")
+ORIGIN("tap.me", "zXTFlqy3BKMtLoLbgSaRo63dsmH4IC/HFCrr7Cp6Yz0=")
+ORIGIN("adxvalue.de", "Om2y+VYmYPSWG9L8rdgVYyn8KnhhFhK/0odKj9WZkcY=")
+ORIGIN("trumba.com", "ZFmoRqH16nCPMGMQaJejir0KwvcryPrAeBdJ4PwcuVk=")
+ORIGIN("xmladed.com", "Zuo8N9dPaNBz5Q/Eh4u1u11ucDaG5vYFWG69ya+EzpI=")
+ORIGIN("apmebf.com", "lPQDfHCdCFSYZ7GpU270vw1QxsfWRMLcBpu8uhn2u+I=")
+ORIGIN("omg.yahoo.com", "g3B8kA/T0ZK9ge9LnFa3W8daI1DtrKmTGiv9T7Tp/Ho=")
+ORIGIN("buzzfeed.com", "FAwJbzbbpBW1t6u3TvXw7eJ/Akzwot2pcpF17lLUTx8=")
+ORIGIN("affbuzzads.com", "PUWu9ytGWhs7jQ+4aAqJ3EzzjD02HwBymkRuiP5MCQg=")
+ORIGIN("taboola.com", "CWmk83tOcbqI08iNZ5h73n5aF/Sh8imA6D9JeoFHnqc=")
+ORIGIN("admagnet.net", "0afcIw919TlAwpyj3FpfI48ApyXWxJnXUdBE0NYvS94=")
+ORIGIN("geniee.co.jp", "TEpKU96+OP2XbvR3a81eI8GXnOIuz1oxCbmvHEv3Qk0=")
+ORIGIN("pointroll.com", "ayRHArscm6+1+iL20Qyw0WIfVoAiXNWofHwg/qc4ULQ=")
+ORIGIN("deltaprojects.se", "ljGRTE7TX6jB5mZEDtfjUcO20ysU/oU9i9pRo+vniN0=")
+ORIGIN("chartbeat.net", "1EB04+Rp2FK4oe6UI9zLfyQoZDP64eY9GlvPqfssfaU=")
+ORIGIN("microsoftalumni.org", "9VlyWuE8sptiQH93/OGR8+Fmu0KDWJwtKpcgRbm8qVM=")
+ORIGIN("tracksimple.com", "uiwv8mTkAW++SR1pmgPlfUh0z3pbSmC/O6IKAvNiUhQ=")
+ORIGIN("backbeatmedia.com", "ehZSahwKBU2TrmA/gvrkKomSE1WeqKxH3F2Y9TwfC0w=")
+ORIGIN("ayads.co", "hMroLyhXyOdSjuT371gca+H4AmjKgY9XneSxUQoWwu4=")
+ORIGIN("doubleclick.net", "uXNT1PzjAVau8b402OMAIGDejKbiXfQX5iXvPASfO/s=")
+ORIGIN("adscale.de", "QwnUBW1yJNoFkRHBOh5ZWsKlIVnM3nfriOUs5mx4/oU=")
+ORIGIN("etracker.de", "qgtbwdFAwa8R4cmoa0jCji61vlH9rFWePa1EzZugYa0=")
+ORIGIN("stylelist.com", "FqbOc/Al0L4TmyJIvvH+EEt2bjbvRJTdchmg+YdZ8Jo=")
+ORIGIN("google.com.na", "TM4wcN6lJQaEaqMXeNJGY51ZrFAJAXAVd72ZjPESnnM=")
+ORIGIN("resultlinks.com", "I0mKEgQ4rTSDFmO2C6LWTExSbDeMLiZZevzqL2Iuexk=")
+ORIGIN("lockerdome.com", "bkQMLPiXrqn/FHwAtq8Ki9k5viP+ykWDdOiXb8wA968=")
+ORIGIN("liveintent.com", "eNZmdXD8iqrdLwepzhNtd17vzFzJXMboL39Ztp7XYrA=")
+ORIGIN("voicefive.com", "dNi1ZOfeWFm2G9nKB3pgFZzn1+c9Gjg64HEFeZNSykk=")
+ORIGIN("revinet.com", "XZGE37npYLGp/x2GtNlo68fGTgt54EZc9gKLpAhbcoE=")
+ORIGIN("clicksor.com", "tL052g9CBFu4weeERJzpc43poifMFN7Fb+D5kFBwLSs=")
+ORIGIN("i-stats.com", "hPIRK1EDJ4G4bMRvQ+o/IrhPPULot7i/Gumc8+vq2ps=")
+ORIGIN("marketo.net", "2WSth6AnvC8Jqq9ZgjNU6mmKfwijv363Jtlv7TFaolc=")
+ORIGIN("usemax.de", "69uhn8coMdxPkV6KxtbhU5l3LEeGjPPoiwZkCYexJQ0=")
+ORIGIN("ru4.com", "bpWQoUiC1aBmwV51ZoZLM7XCw9FxDaaRHopKP48fmRg=")
+ORIGIN("m6d.com", "8iCZ4/JoGeSYYCmEyJOMjWNaQHIzkrjtUM7hXq163/M=")
+ORIGIN("spot200.com", "iEpA0zbrcgH7or9Nz5Yjsyk2NQLrxIC41513MDmgHlo=")
+ORIGIN("gopjn.com", "MkJgZMBi3vpX9KtJUR/g9w1sSrI6V7NnAGHGsNxFslM=")
+ORIGIN("nextperformance.com", "bqlq81KyI4RumM6JkU0VLI2N5LUHDoleYEg4K1UoM6k=")
+ORIGIN("hurra.com", "NTKrKIv81eCuygXJwSmEPbACcQfmd0PPBDI/KokWzcA=")
+ORIGIN("pntrs.com", "L5Qt7WrqUpAFUCflldH1XJ+fVn1he60o3/NyWUwOmrk=")
+ORIGIN("contextin.com", "N7bxxUMaVG/neSxFQeTPq2O9Z/k4Wpb4InEC/orZi9Y=")
+ORIGIN("notepad.yahoo.com", "1kpI0RuYWWuRY0Q7LTUzU43Za0TSd9/CGnQeRUqeP4U=")
+ORIGIN("gb-world.net", "qTqp8vesQIs0uuWkJ6fxQRRtRMKxHfQjDhy7SWF0B8I=")
+ORIGIN("google.si", "IEqFlBcWCndVZ5FsT4EhTDWw26zmAPd+tArgNvkJ1do=")
+ORIGIN("opentracker.net", "G75vxthdefFtl3t4nXAIirpvOY20316LAmqE81flVTA=")
+ORIGIN("google.sh", "AGIBSyPMxA/WINa8hSFE2G6yo719GBTxI2ubkU4Z65Y=")
+ORIGIN("aim4media.com", "/+xA3D+1uh/3xJAoX+3ss+Vfihk9Mw/oQyrSV3mDIVg=")
+ORIGIN("kikin.com", "jjQzTVb3UfMx5MP7x7hG7gRQba+AiAu/4jukFpD8aNo=")
+ORIGIN("dsnrgroup.com", "dErd95Vex2x7zyK/IWg79AZVmkwsZSu8+v898tdNTMw=")
+ORIGIN("dataium.com", "3TQSSAEIE/QWjiUc2aTzA8KzXi71LkEwRvKE+ZBlNOU=")
+ORIGIN("imrworldwide.net", "yLh2FHJpMJaeMQXZXrfxh8XamjzTpfDQw0KiNc1ARN0=")
+ORIGIN("yoggrt.com", "s0WDzz7YyBVhwwpNwDCE/xK9HN63+htUhq1qEQcao2k=")
+ORIGIN("streamray.com", "RZ8I3NraxF12bUiV8uPEV3F84jpWgF0gkbAZTQDITIg=")
+ORIGIN("bttrack.com", "MXgbFANmi/dgruw5ETsxpP1pviPkuCnbx9CE5oDjpuU=")
+ORIGIN("albacross.com", "m88uBlbwdI766uJEDXSVcSMKeKhuESuOr54UIQWWp24=")
+ORIGIN("clearsaleing.com", "Po2h8458JGJu6rHSsbjZsV3+pzkZbW9xbpZPbyJA7bM=")
+ORIGIN("ebay.com", "FwXBfevpOhVrZOt0bKl61/o4+6lQNlL8r4r5VMCVMSE=")
+ORIGIN("google.sn", "FFrDoTk659V/Gmo8DGu+xSBtpYruejl9tue+4f7GAh8=")
+ORIGIN("bnmla.com", "tXMEBHGi5d0rZS0Kv6ede8CMBAxutcjYYH8gB+BmDx0=")
+ORIGIN("affasi.com", "xnRXdzqWjPu/ctB2HU2gAva5NdqEjVOli8Rp1tSlQ2g=")
+ORIGIN("google.vu", "1+qfA9IUF+XSb6++j0oVZlhbqGVB4squHa/dz340+MQ=")
+ORIGIN("trackalyzer.com", "T9LC8UHSzoKs8igkstVv5x0JIzlF2xOvRVzC7NE/DNw=")
+ORIGIN("google.sc", "J4nmbKbRb/gRmtWPVS4zaUHRjZ/RiUxaaJy+7iXF7Js=")
+ORIGIN("valueclick.net", "Lv0AwEsJUt1e+zqiJQJ1e3CZqMvaKp1sp2SZFq12LiQ=")
+ORIGIN("msndirect.com", "jJ9r3LzOQ+ZrqUU4W++g5aBM3KYeyAPNUoT2Y1LKZRw=")
+ORIGIN("boudja.com", "y/3GAp/MRrIucrYpKVP2+V1aTFhO8ItlD/MbDtGq/YE=")
+ORIGIN("admedia.com", "eaxrT3b1kvcTElYzc7yYtPdN97bW73BhfM/CMlx3Dxg=")
+ORIGIN("longboardmedia.com", "peFCJ/SYamGXeTQpRmMzbVa6M1Fx/DXJiMDpAkIIrG0=")
+ORIGIN("google.dk", "9EYG9ykW2FFg4Llod9SOLiP7z2zl3LhUgD6fttJaNhY=")
+ORIGIN("fizzbuzzmedia.net", "awHz8dMekp0dYxinuWez+Ewwsbe8KCSvNbxA9cuoY7Q=")
+ORIGIN("layer-ad.org", "iS4LydySWdVKmKPJSRhc/ST55i/Q1D0DFzTB16O1DW0=")
+ORIGIN("cbox.ws", "61ygRDbHnQYr/fQI70V++SJuFuAanPwgZPd2aslQZo4=")
+ORIGIN("audienceinsights.net", "c+Dw9cS1zE+X/uE+vn9CDPg/DQdJHFQri+OQ0BQ9QNE=")
+ORIGIN("contentwidgets.net", "HAknI9fjEAKm6AQjRjLRcr5r1K/0IUGQwfFSF4e1Ih4=")
+ORIGIN("google.tn", "gDwFWIvusifAIsEXQ3J331VIwmUJzWGb+3loKX4VG2U=")
+ORIGIN("checkout.google.com", "+bTzy0fQhaHzF9lNq3G4v12VA4h0MG7yulCXi4eMNFo=")
+ORIGIN("linezing.com", "MO0rPgeBEVkpu6BK7fwjnnPdC7iY2NyfIJd5voG81w0=")
+ORIGIN("go-mpulse.net", "xcr0C2skvWzTyI3WFjTvjlie22zQ/+j8ncT/W96VMpE=")
+ORIGIN("google.rw", "HY8iRcXNiDuStNJfuVHkMvUT1k+77SAX435vFvYGyhg=")
+ORIGIN("quisma.com", "ZBA3/bP7Ur8/lFBR/t5awczOJ5BGEN7FhDcbdF4OkdQ=")
+ORIGIN("seewhy.com", "LuOQlloZdjVHNc/JuyUSQpjrW4j1zjTHvmTn5L9zmhA=")
+ORIGIN("google.co.nz", "P271UfEoHANkzIFS8/lMGBQUc8yW4yn9Sqd3UF7z7Vc=")
+ORIGIN("adwitserver.com", "X5jl2yQu5vwNiCAa49XRMx1cQZt5d+jUrkvNqtusdTQ=")
+ORIGIN("xiti.com", "idMS6Whn748VlENvC3kZAymuJNH/PNiepUBUYAYybWk=")
+ORIGIN("proximic.net", "srLW2Eyjq1huea4wW8KEGgDXjbAMTMk13Uojtus0Pik=")
+ORIGIN("upcoming.yahoo.com", "9TNmDbbszDoenc/LwWSxgtAW1a2hBh3ALKVQcoa3l8A=")
+ORIGIN("google.tg", "0+mAvK/M+N/VsdYixaHsttDWxLp9hFLHami4CkgLNJY=")
+ORIGIN("wishabi.com", "NF8FH9ztMQzWAyb/eOQl68bGjUfD8hk7+zoJYTrjCww=")
+ORIGIN("conversive.nl", "4Q4PPuynMdHeozElLmTEsFH4Z3yrukNCH8Kdokc3/84=")
+ORIGIN("ientry.com", "4O+6JS503kZemPKMBy1rbdxagqnvmbU3QX8b1OAlEVM=")
+ORIGIN("zapunited.com", "VE6p4SOJtvhVR7FhQS2bDUlxmmtOTqBIkCh2yC9eCcY=")
+ORIGIN("stylemepretty.com", "wRhC3jghSpLhlJKi6MlbFaAH9UBgamtfggfaFQNqRSQ=")
+ORIGIN("graphenedigitalanalytics.in", "AcqId4/F9k1FTPl7Lr1JhD5CbYbtT4idkKLMRexfQfk=")
+ORIGIN("amazon.ca", "NL7R6K6xhzLJTgrW44t2EMZj0gatZRcmi+0JgeKblTA=")
+ORIGIN("ambientdigital.com.vn", "+mmcQW0PVduNqFKZagtzgScMqi5NTsFvFwXT+XtqwkM=")
+ORIGIN("sodoit.com", "BHphQU/B7hVy1qjP6YrXk+KFZo/8BBSJizQYQV2Lr1k=")
+ORIGIN("adtotal.pl", "SvG6guIMmnpay6wn2tFa2vtT9ySaaA8l1Gww4aEey/M=")
+ORIGIN("proclivitysystems.com", "dtFJZwIN5uANZENlz8JgS0QxyR8jBrd4NEfTe4ldEaY=")
+ORIGIN("bizo.com", "AspWJJ5EfsASPKHXxvsln2Zg93GPoU2j/GpSteMGpY4=")
+ORIGIN("narrative.io", "8GfUtvRoqEMYAGP0R//J8+wq/20B9/xa7BHd1t/zdI4=")
+ORIGIN("wowanalytics.co.uk", "4RDC0dZ+zOe3TH8EyxoNu0orjgp/Bm6wHyAs/iJIvgI=")
+ORIGIN("dashboardad.net", "R2FM70uN28hDA0TRFHpENcCRs0MP6FMC3m5+Tkv0+X8=")
+ORIGIN("adversal.com", "eL3UX06vBSgjZeueDtV4D1MX6a0rDgWYdhXq2Qyupq0=")
+ORIGIN("makers.com", "RBwr/5MRgo1+oD+Oh9TOupRQcKwg0+oaET22fxs8I8Q=")
+ORIGIN("facebook.com", "NeAyZg7bkhwMDOWb+iidxahMcbmVhLNZ101rA9AN5m8=")
+ORIGIN("brightedge.com", "e6nDthl1YIJe5BHnGN4cifPZw3PjOnJE09kY2YCJhug=")
+ORIGIN("itisatracker.com", "coqG/WdwSAEI41XJBN7yU0IqFPCp7UnLAv2FBp+2b0Q=")
+ORIGIN("cams.com", "77ZZsofss08/vROdukNv+B8WCimQN1/xjQAzDCs9KgI=")
+ORIGIN("bucksense.com", "nezQc+rw3XAGZQNi0Ic6ajhXBMZ6R5B0sKZCiA1P8ZE=")
+ORIGIN("adhood.com", "bLDRESPjRIWAAof6Y6mZRW2pJEdR0niKZxJDnYjHj8g=")
+ORIGIN("viewbix.com", "qaYP0aIWZ6r7XOZfiNzAqyK6XXsFkD8XdEe5qf/mp+U=")
+ORIGIN("ymail.com", "f8mD6lUvfI0VP8MI1iHrT1LoSqY+zM86c1aYoRoqSo0=")
+ORIGIN("adaction.se", "4LoHXWU/UN9/osQ5XYAyF4JjGZE/sLb2+ksakP1Sf2Q=")
+ORIGIN("oo4.com", "yYa7aZFgsC8RgeMu6rO2/MvrF787kvQKaV4UXE2m8N8=")
+ORIGIN("google.com.pa", "ORFI9lZ6yaBOdN3+NaTEsir6ReMfRL/ILAAe9MilvLQ=")
+ORIGIN("google.com.pg", "aK5AGBiIbwPo3buGHHq4pBEoQ3/u4sweLFrdgEyXCgs=")
+ORIGIN("google.com.pe", "CCLnONoACptZdOr+R3CF6t9jcqDblUzBGkFclioqP+0=")
+ORIGIN("google.com.pk", "v4UnBOb3wxodpEIkutS7dvb7bvvkJBoRI3Z35Ns7tHQ=")
+ORIGIN("sites.google.com", "ABYLUrTeQ2zalyYvWk2K5f8ZTBXsBm6n9znJa4I3PMY=")
+ORIGIN("google.com.ph", "oDLQhXLSR4EO1HyTrXrMH6wgU2gKeO3AePE6ZVilFU8=")
+ORIGIN("metricsdirect.com", "IQB01jja+rbxpaRV3I5jnSSSICwGGRz5dcasAg4L2zE=")
+ORIGIN("google.com.pr", "4aVNUPXOsZZMdahD+ALmNbTtaUsVhE1T2fnJx4Z3PQw=")
+ORIGIN("affili.net", "UlI3sh5+MtqgOSVezGjCQqcqix4qePTOsB1j8JGwn+8=")
+ORIGIN("adtoll.com", "RU/IUnRZICyFnElLiDqW8W80kRXIx6zlL2k8SfRSKJ8=")
+ORIGIN("google.com.py", "CliUZUGvF0Mf9GfeFL+YEPZlKpEEb0invpBwD5aLUbk=")
+ORIGIN("ignitad.com", "mtlOcHyjRc+69p6QgRq+auU1mw17BnGFmFepgJmkIVw=")
+ORIGIN("gostats.com", "LVmEcBR8dJmqPC/GD4jHL6Z2LbI3TJWRtMm0j0ldfmk=")
+ORIGIN("persianstat.com", "79JNMyI059WWglwA26d7TA5jAqrKx8/h5R0ouWldYwc=")
+ORIGIN("outbrain.com", "+vM0aIwc1Q/u1KS0hPMpcb35i4qXCLapgFhrj0cCLs0=")
+ORIGIN("ucoz.com", "P9xVFUiE8nPLnoqpCJ1ofFm+grMnC58q9kKEVvQ9HAM=")
+ORIGIN("meebo.com", "t8/Ek9Ak7ZvgBar3a5RZOW1gCS8EAlBkJLxqIOmTVJ8=")
+ORIGIN("burstly.com", "R7qUl4SkcZDX6YUOZIOf68/U1IG/jp9ZpaLhPQdUKKc=")
+ORIGIN("crimtan.com", "K2MQpx6gZbOebQYnzOepr+1MMQ+njWHEqwIozT7ROvg=")
+ORIGIN("comclick.com", "N+/mJoV1QMivMcUU42ad1BHQpnTybjY7U4h0LkoeafU=")
+ORIGIN("liveinternet.ru", "FrvRXv8fg2soJz37uN/i6Z/1IbOld60KNrdJnNt2o8U=")
+ORIGIN("advombat.ru", "7GZzyazYzkBCOgfZgjMGL1vq0MAaTC1CPrflWgQwHRc=")
+ORIGIN("editions.com", "61QBYUiXxnNr8pZwkA3Gnkdb9oBe7u3zw/dDR4IwP8A=")
+ORIGIN("ebuzzing.com", "KYvPd7HDRZ53AAflAaejYJPlHLO7fQ5Kjkz3IlFh1V0=")
+ORIGIN("adzerk.com", "bcyGKZYeqV0YHkLr9yj9AMLvR2V8o8LljLbBjyxEFe8=")
+ORIGIN("adxyield.com", "2daTXt0j897apgPeS+hVk5qgJuUVpKCj/8guK7M0NmQ=")
+ORIGIN("videologygroup.com", "PsrodQzSbfw1HqKYvnL9wNcC80kNqlQudaj3yPlV+No=")
+ORIGIN("googlesyndication.com", "Jbk4yGpJrjwETi3FAfhujm8mfe0akiIF9H50O2+SW2Q=")
+ORIGIN("onlinewebstats.com", "iuEv6XDyJrqqKpKcgbtDPiXqcWhtj6xKs+UabHCuVe8=")
+ORIGIN("formalyzer.com", "To5Xdml2Ym6aOIlT5sZ92OtkhXVDZNzvJgk2ZKDymiE=")
+ORIGIN("digitalwindow.com", "wZLo2pDEqcvIu6Ci5O+Zn+ua1hjziuFul+aNYJpX31E=")
+ORIGIN("linkz.net", "LUKffYtyPqJ2E/SVcvB945G2BbDakeCC39F6m1t2VTs=")
+ORIGIN("px-eu.dynamicyield.com", "oVUIxA2jK0YtCcXU8kkUv+5ZTzWTlNuLmQgW47iNllA=")
+ORIGIN("woopra.com", "tbiUsvexk7twV35xsoAkJ+dgPwJSLsfMbK8t+PjdFVw=")
+ORIGIN("exelator.com", "lYhicFys4dYZauta5SlCvmpy9myqcXyFweDRgmAgzfc=")
+ORIGIN("yabuka.com", "b4SA4RUrCu/Mhcj6+flLyMX8cb1Vfzoqm+wAg4c9FSg=")
+ORIGIN("www.yahoo.com", "0cdbrLs9UADi21bvCuuY51ZOMVf26Fa5puKcytic38k=")
+ORIGIN("4mads.com", "vBoOy3mHZIEdVx0E9iYKo7CzOjHgksTvc5R7OaosdjM=")
+ORIGIN("shine.yahoo.com", "I8cNu4QF7Ec1Slwd8n9WqQem1n+8DPkHPFDdrse+sXk=")
+ORIGIN("adxpansion.com", "ZTCnlYHjGXPIrKyTeZAhluqt6QyrmP1zBHDG0q2KW6I=")
+ORIGIN("vindicosuite.com", "n605Qx6S2PN/Erno0XJxZT2kXRDDqzWyt7yiUJDyp6c=")
+ORIGIN("spotxchange.com", "PWe2f4nohZCOvdhJOVRSIyi/9DFZJuH7srY5xM/IsuM=")
+ORIGIN("akqa.com", "wQd7SYO/cOvnZM+zOTMFOh+uoZGgMgp7cfEbPbkjS/o=")
+ORIGIN("aoltechguru.com", "y7vZ5YVX5v7E+HJEAH7tl6+DkklEsy1CqcBrTjn/IYY=")
+ORIGIN("leadbolt.com", "xqUYGzyZtSODss1FwMAvsBxw0x8IrB7QLtFS+F3lPSU=")
+ORIGIN("eyeviewdigital.com", "cKVphxihaBpEAUTJzJF5So73Eq7coXlMO9ud+Um0XTo=")
+ORIGIN("agencytradingdesk.net", "78WIxAPOjIzrlIVxZpBxDOMpKP+qkZfDw2jlscfaVZk=")
+ORIGIN("pawnation.com", "0Vhtca5kM5V9Ry4QdLpCAmO5KuCKXUO7jzLPmVG6Or8=")
+ORIGIN("widgets.yahoo.com", "UCNojlsdBycxP3fSeIAa5xLQ8at9gCjgbxy52NzZcpQ=")
+ORIGIN("casalemedia.com", "IEY3iayqxVJU8OtikaeCXeAYoLX1OJ5BsoULAnqGA8g=")
+ORIGIN("peerfly.com", "sPyPhgQ+u10FEGQD9s13fhGzP3qvUUUkZxkCc06DeZo=")
+ORIGIN("groupm.com", "ALfktirfcZG8t+IHoMa6QS1AvRNSjyzpbQAKiOqw3zQ=")
+ORIGIN("google.nl", "kRaXgkx+slRP+Pf4wgOAWOPzVeUJyv1GlffGrUDeDJE=")
+ORIGIN("pulse.yahoo.com", "ilZdJHwI/3/QlQ2KHze/LaKerkoN1lEm2HoNt8q0tAA=")
+ORIGIN("openx.com", "xv6V9x43xDspP8sTXQNlX2xzm54ttdBUrZt6Y4wJuv4=")
+ORIGIN("smowtion.com", "PlbexQJ6+j9VMmppHitvRh9dO96gqeiPjYfhGTTfbSw=")
+ORIGIN("google.ne", "V2QG2e3j6D5yDm5GqdTzwh0JoFCC3ql9wudnlxU/Alw=")
+ORIGIN("media.net", "g5zNFXMV8UAp+npY3chZ+LNxjnPioBUrrmYFpb8DlH0=")
+ORIGIN("userplane.com", "OrVLdF54VcDTxuoWx8iXyqPzfJw+k8a+2RU9WwFkE5M=")
+ORIGIN("ftjcfx.com", "Nj/4+bPOtR+lZTesgNe7uLNGLHWnZZvjR/BGNuqUOOc=")
+ORIGIN("scanscout.com", "QMlmpaSGyMxT5tZqTvnQJ0Falnwr56BgU5a1ao5zKsc=")
+ORIGIN("audience2media.com", "9G7M7GgTOHdcEPFAftGE3xhm6X8uMcTkiQaJSUNMiGo=")
+ORIGIN("google.nr", "sDgk7t88uiwkv8+kl6GIZK43D3FwQsONGSbuEI1ozSY=")
+ORIGIN("batanga.com", "hfDXHzfUbbIWNoLm1TC3x/atSiy+ocmH6gNd1H6gpd4=")
+ORIGIN("2o7.net", "9z3OfrFf/VjxUDdTtlxd4cb4XSjPMTjdZA+a6u1r3Vw=")
+ORIGIN("perfiliate.com", "Ty7IUf+U/nPZ6sR7WcSJgIJ/GLgXg3V8eAEn8V0X9BE=")
+ORIGIN("entireweb.com", "ftH2yKaUUCiP6761X5PKXIK/Q+bkWB+bj6HMHcnACoU=")
+ORIGIN("openx.org", "cATl5c0LSEoGf0SbKKuDeBDEhC3/TnqEHqbL6Vk+Q4I=")
+ORIGIN("mexad.com", "xMCVoa6Jd9Ji9qR2ljLLuakr59RRvNWnv8Gy7GckLn8=")
+ORIGIN("trafmag.com", "ttgwdKvZwPmsmSnlvbM0RY8DLWJF/Jec1RNeulQjcdM=")
+ORIGIN("adfox.yandex.ru", "7vvhGofXQpX68eIbx9B7RCX4SPmwn0KmPsUTw35P9C4=")
+ORIGIN("adocean.pl", "HtiJTQVO+KamTUt8ZWy1XpAZ/mDiFcx6LKvWhpXijO4=")
+ORIGIN("ignitionone.net", "wzuWjS7sTZnBtGoAuJgn57ofR4boiHmtLzd1sBzFa5g=")
+ORIGIN("justuno.com", "2dHm3T2UczuFiQBlLc7aI2sqh+DJcGdFdCeId/1kh2s=")
+ORIGIN("demandbase.com", "AYNaN4gxNZlKQGgWl8ZRr754/JPu+Gu+P5t76am7E0I=")
+ORIGIN("adreadytractions.com", "7Fq9bxW87utO0UHVviHnAfvx90+96zvGB6bMliuSI20=")
+ORIGIN("clickfrog.ru", "Yp4LmbX91PAXkOdatXTK2tQ0IAFKCHOYXrGwQrK0IAE=")
+ORIGIN("kissinsights.com", "H3t0k5PCLcKzw73X1PwdH1mqrI8xVkrx6qrDOXEXGrk=")
+ORIGIN("conduit.com", "VH/THuOqMk5IeP2IZZ3vcu8MuDVn0oXwRXSuQItgkps=")
+ORIGIN("tracemyip.org", "czX6UKa0/sSKKslXhntSTx0RGfMkQ+SXSNevtERz74w=")
+ORIGIN("crashlytics.com", "7559mwZrt1MRUQmX9X1/ZPqaiXa6UWUZSsjN5nhuCb8=")
+ORIGIN("storeland.ru", "YK63ZSM8bYyRq0LVGF5MMgKzEwNBcxo8zq644PnqPSM=")
+ORIGIN("advertising.yahoo.com", "IYZ88DG/iss/2nzsN7R0RYXACvPCBjxn0KwaF8mlXhI=")
+ORIGIN("inadcoads.com", "w2lhwH00E5WXo4yiFOS5SRcL7i1IOzCZg3JFeVrLnPc=")
+ORIGIN("fullstory.com", "Axh1PCCCPDSZCC9IYuNjUJRtwhFBb/vQNlylFyTAEQ4=")
+ORIGIN("skimresources.com", "7/uIGTt09r/tRlg3/K0XNb3UdyvBZ2nTzi6Q6OD80j8=")
+ORIGIN("gsicommerce.com", "Tfy3V5qKimoR4VwLn8AehmsgPa42Mrx4i5Elkz2SJE4=")
+ORIGIN("360yield.com", "meAQwlKCJR368GV3NOcDfiABpFNNC2AWzCS3zOGLTf8=")
+ORIGIN("pinpoll.com", "+WH4BFPKmzBNAtV0IOVCLMb/T/bBY9nbA1io7xJmAbc=")
+ORIGIN("quicknoodles.com", "A3p9ONjoEWoEqHnWJKT1F73aguUT9//GOqYXy0zO3U0=")
+ORIGIN("appcast.io", "HmbM8V6GWRNsPeM6kftlf5KaVHWSkkANosMwVC0pZcY=")
+ORIGIN("popcde.com", "UWt1l9jrvcqZeugg/75+dgwTvLZrML2lfKFit+Gi5i0=")
+ORIGIN("xtify.com", "rcWHYtIClsqteZiyvoYNTXO+DV3NUcynD6vMXU2MVIo=")
+ORIGIN("bigdoor.com", "cGmGubExcI9RDkaF5rr+ASDIsBR4mbTXxeUqTX6JkHM=")
+ORIGIN("acxiomapac.com", "T47YG7ZxgBTjXYEdlI0Ra6hZzgzFqREjTi+lSyESELo=")
+ORIGIN("dl-rms.com", "RF5mvUJT71liRbX3cO2YF9wm39mnD4MldbM6jKQy5rA=")
+ORIGIN("nexage.com", "ituALroEfLlNnvt2InRsxFRVBY+MKmkuaTUQkAAQcNE=")
+ORIGIN("ggpht.com", "p4OSwrRYVwAm1/4Medcd8B5+oryhM+1e7wQVDg3UjoY=")
+ORIGIN("teracent.net", "d1NRwA2WuXkFYACVNA4jqpTMjrrnGvwPTak4p8EbiR8=")
+ORIGIN("cxense.com", "ZOHNlLaui+/22uq79KlEwnlBOkSyov9AayYoj1Airbc=")
+ORIGIN("decdna.net", "ZiLv4hUdkDCsXPgRSbiu7O1j494zssDuCiywupvk/Xg=")
+ORIGIN("adgentdigital.com", "SpDjiJjfxowZ+8DlhN2qDzGEsDGnfzA9D31aGPk5F/Y=")
+ORIGIN("popularmedia.com", "vBMWVXURF+YQxz+7UxfJVsfU57vUmfw8pGIqey8RQ24=")
+ORIGIN("conduit-services.com", "vKydEg9OqboklKxp2HgeiPUL4EIqm1hJoQDbFwET4/M=")
+ORIGIN("appengine.google.com", "tkHDTcesxXVRfA6fhplL3gfbJTlT5W3kA8jmBptXt1A=")
+ORIGIN("tyroo.com", "HIyJZKz/pxyt7Ji3VXTgoLlz0Xvy/h+3EP+xZBZnAh4=")
+ORIGIN("groceryshopping.net", "mn9tiKQ3blDTZUGehOqZtVjk/4ie17vIJa8TI6fCcNI=")
+ORIGIN("simpli.fi", "W/P2mAz76bisjTcaKvCNlMdmmGJGCnkvgjP6TYijt2Q=")
+ORIGIN("google.co.ao", "rcdMp2UkiGgCzs1nmlgOnLW8ea8ChsdCeZNyq8rxR88=")
+ORIGIN("tisoomi.com", "FRArVYWAc9PhDi/W7fRPrlJC2UQc8hCN1/LXL6i0dsQ=")
+ORIGIN("avmws.com", "0FRcGYEhuimiq7Cx8p2kBPTcEX8H/svnbsFoXVJdr/E=")
+ORIGIN("tinymce.com", "8IqiBSZkPrFxfX6b9si08URJivscg481PlK2x7sdq6c=")
+ORIGIN("rmxads.com", "KEOjKJnSkuaU+1x5UP9SQg/nMoUzhYIbnjV8Yj9Kqkg=")
+ORIGIN("adsafeprotected.com", "If5mXVjpwUSZFC4Dnk0/QEBX4bVAsophWLUrjFHYjgk=")
+ORIGIN("adatus.com", "ukqWBm01VIXL75F0MK0JgHj934AJKHV1dXObH6iozmM=")
+ORIGIN("adcolony.com", "zcQFANtv2riPlFTWeqdKpZJRIbtOOZ2JTPqMypZoVV4=")
+ORIGIN("mediamath.com", "fJUFPSkVoshAqFxGHVKS0174sUoWJzB/syU1CE9Y29o=")
+ORIGIN("madhouse.cn", "QxPp9FBKk9MawRvYblqDD4QONZ++920oEj7YGlv8gY4=")
+ORIGIN("dedicatednetworks.com", "5SWr64Xn+K/JNeczjCfFbYdRJuP+prQgHiTVStAlDSM=")
+ORIGIN("getsitecontrol.com", "Ic76Vl1Ox+wPUaJQYEP28szLj4vd7oPa9xoSZprGc1s=")
+ORIGIN("dt00.net", "2CtICrGLWt5Cs1bjSMn8rpaWfRb3oD91Qc85eH7daA0=")
+ORIGIN("markmonitor.com", "QAeQpGPy17396Myck1UG1GQym8Acb87LPzfucx1LG9o=")
+ORIGIN("revsci.net", "3BWdA95qp9f1l3So/PZs5DfT9QKbCP22KKAtIy+zhG0=")
+ORIGIN("belstat.fr", "ukGada/IvOjIbWfQLmynZpweKKdrvNp2lBnRVXAtIvs=")
+ORIGIN("instagram.com", "rkQFJ+B17LzNvF2pGNkLMzawz4ECE4o5EyvztNKCcnk=")
+ORIGIN("googleadservices.com", "3P+l5YQcA80Guqe8hwrVTcC3KAa8jfMZ8gTTjYt4r0U=")
+ORIGIN("harrenmedia.com", "Lsqnr5B+qiT4UKeKx4+y/cBxj91ylj8JcpEy1pbgDUY=")
+ORIGIN("addlvr.com", "fIy93Ud3/bKTSPYxJwM8tmlvmYsvijk1IZop+BU5YgY=")
+ORIGIN("s-msn.com", "imIZD4Em24jWxA9UneBx/g2LmdX0ecgnGQuZtnKXIug=")
+ORIGIN("politads.com", "OQU8bSsQ+lOse1ftSk6B2uuOFXcZj8ZrRKfoAUSFjXU=")
+ORIGIN("underdogmedia.com", "ijXDMM0rIO0xLNjyefyQVPgvCSJXQC3AzeGKmSG9uhc=")
+ORIGIN("voice2page.com", "s3rnKg3mi/Wt9cMsp4LOW3zjDQH61ygv4a6Uipdn3ts=")
+ORIGIN("777seo.com", "eldqMjZZgtfVqOeE6PX4CRuIIWVC7gAmKjjmHttO8ng=")
+ORIGIN("xaded.com", "EDi2Ob4fnXXwsDvIiDjbgEWQLpDygeGiwynR/0NDYwI=")
+ORIGIN("sageanalyst.net", "nfWZtyLDddOq4/zjeTel/YT1H64F5BGehZPn6Rpc0kU=")
+ORIGIN("poprule.com", "Wz/RomQnOe0TipPZq6rTJw76SX50Hkf0pQJMUgihRjY=")
+ORIGIN("sedotracker.de", "xGyI0MuOo0vBII1AQXpWwLSvWV00uHtLKRvAz2w9Uy0=")
+ORIGIN("eyenewton.ru", "iR1lyJEPm3fHBaeuHWmz2rp7Nzlyr7n/QGoMBFZckPg=")
+ORIGIN("mmismm.com", "dpjF8/t6AJ2guj3EAWK8/nodwGmI0QsruutmlgEhTTQ=")
+ORIGIN("realmedia.com", "WzAhBlKC62qdpLYcNJAxUBvyWv12ABJcYVc098LqWfk=")
+ORIGIN("linkedin.com", "6yXS0BsWjrjik69K3yJdhnmFTRswoxMOf6nyPTE5WUY=")
+ORIGIN("extensionfactory.com", "37edxbg+AemA1e6QaI/V3XzNzHevccMDmF3+4PLFKZM=")
+ORIGIN("admotion.com", "RyXRTdjlY9ojpaIMN8/KD0ERzXZcQVbtsw/5d4e4Oac=")
+ORIGIN("tweetmeme.com", "/f+mrFB9Bk1ApNhbqHwq3fUXbmr51iOcotbfWSTAPls=")
+ORIGIN("local.yahoo.com", "2PXZc5QZj7xIIPAmONlv+NtO32FFO8vKeGUX0oXLne0=")
+ORIGIN("eulerian.net", "QvypiataekoLsPGhqXsLISMpJOC8RiAyj38jx3D5l5k=")
+ORIGIN("umbel.com", "b0RAkBf+1pMdpQ+YWX4khKT9XI35x0sxYnLjq3DRJcU=")
+ORIGIN("admeta.com", "s3u6RdtinOExQDCaPiUqxpn0MR9gurHdDp4HOJb2gYw=")
+ORIGIN("complex.com", "pK2SUkq/ag7hoSdp+3TqqvwdangQoxz+xGmLL3ZYflA=")
+ORIGIN("luckyorange.net", "H7caEDN3jCZpX2i8+weECMIgq5bmi3xmD5FxHZM/rTA=")
+ORIGIN("google.com.qa", "i0Ih/3nRXVPFwY0M0KguiDgYgyAv+Pf+X0OwETlJLiQ=")
+ORIGIN("intermundomedia.com", "crjUZIUbTGXtxa2dZWLqH26fYAC5+f0u0hCCfm2fLrQ=")
+ORIGIN("matomymarket.com", "ItPQmeoiLpkEHpEiY5zt4PeMx3+xe7peD6qp4rTlsS8=")
+ORIGIN("smileymedia.com", "FcOYiv+ZER+ZpRUYx/2LffBLT07aeV8qSTPRaCG1HbU=")
+ORIGIN("adtruth.com", "aZRc+j4CfmEYOhG75YRPp2FLKKRKJStZcXO+FlbFweA=")
+ORIGIN("adsmarket.com", "X07gMyQTDL9T5iL/EI9cGrlGqKGBIBWXi+R7xV0pOnU=")
+ORIGIN("panoramio.com", "AL6GWrD5o7RdsdV2Bh1NlCbhdPn8qkMHRA5QQe5osNs=")
+ORIGIN("hlserve.com", "F/hXmd5pK70vGuvMoEocBpLsQ2Cr0VHtVuDnhpZVFKs=")
+ORIGIN("clickequations.net", "CUN9GpCYRuoCkrCiyCHa1je2cyrTBw/z9VD4o8CtTM8=")
+ORIGIN("mdotm.com", "cN6MwsLxj5JG0fpT8zLoED9YQAGy/8fEbYbrMo7Jw4w=")
+ORIGIN("scoreloop.com", "eZXkKFvgbF1JzWTvh8ZevIavN5z/zb+4BSMoXGjuIZ0=")
+ORIGIN("causes.com", "qHdUKXgCRzGmiSdUCeBZt32GJfioRHURD+FBu/YPy5k=")
+ORIGIN("revcontent.com", "W3JdvHglaPmv6OwJnYjH1sqHt/PxZtn2TakEQ/exh3g=")
+ORIGIN("vdopia.com", "6U/SnyBk7+1zxD/c4q8+LPCD37Lxvg+OYC7UY75mhoY=")
+ORIGIN("mongoosemetrics.com", "5hC4XCZe3E8jj/M3Ed6ASCqGk8vkm+VfSmvQZtlzWKM=")
+ORIGIN("adsfac.sg", "f8fm6lzg8UT1teEHoZMgomXgGzFFKRYWTe/9PLK42to=")
+ORIGIN("evolvemediametrics.com", "iuBKNh0Xs74ukYztt5EZeP4IZ4/wxuYoNfNgYaG7oCk=")
+ORIGIN("webgozar.com", "JWqrquZ+f0GxZIQ0hBPrBcyEHx14XAlArW+dvfb+oW8=")
+ORIGIN("adbrn.com", "WngE0QhzxQUHCuH+zsNsSRjiYj34m9bZeqXK6BCEeC4=")
+ORIGIN("e-kolay.net", "raPEZSsoZ2AeTWM9gPgnWbei0MleSYXMw2EWSa/BGHM=")
+ORIGIN("gravatar.com", "KXWTNxfibEiN7rJGrabCTjrZmZUCoiNs0mw4Ofjipa4=")
+ORIGIN("acquisio.com", "Qjcwx4iaaVPGX0+NcKKjctemrQRAbwUfrEwA+stDI1o=")
+ORIGIN("predictad.com", "0I3CgWsiIyI93tLL6wYxT6k71LW3jxwbZBUvxQiSv58=")
+ORIGIN("wp.pl", "h3Qd/uSKV7aL4JAkGvt4pzkNDeAeicO8RPsceJ3xJXM=")
+ORIGIN("cc-dt.com", "WQm7SYEn7sipfJGJrt1/+J11Gd6JjJWv67AfUnmxqSw=")
+ORIGIN("adkeeper.com", "VSTOz95HX5XXOSfmPS8LvjTrnohRrI2kCLC5SZHJfNU=")
+ORIGIN("nuffnang.com.my", "T/+nVrMIEhb8xcNm4y6I33ovr9R0qLECBnPoorcmg0Q=")
+ORIGIN("w3counter.com", "bX8lxAzCmAWTMqgncueKHcQp1n/eMk5fhg9kl3YgKaQ=")
+ORIGIN("orangesoda.com", "Z9xZPMzt9salBCLkgH6XyacNSZ2yV5nsWvwsgbV360g=")
+ORIGIN("audiencescience.com", "ZL8xWNeRjP+vOOSWv57Zcx8y27giPnY0QJMp5Hsi0XU=")
+ORIGIN("sharethis.com", "tYyTYgHsnXvnTO9BXWPXjlgNKuM+tEPLaNoWaeNzhMo=")
+ORIGIN("fimserve.com", "ypQoagCJoEK5gooafj0rXIFnTtKWc/pTnG3zmlipwqk=")
+ORIGIN("google.com.gh", "NQgzQz54D2er7eJ0j2I/wJYnHtthxU94HRWcvmiilOk=")
+ORIGIN("google.com.gi", "cB6tv+IJt2k4Hbd1JwgAyzw7T7hgSRMUJQEcdNMSAHY=")
+ORIGIN("businessol.com", "P442IHfbcnhPJ4eDXCtwO7uqodGW8IKnZMLhx/TvR+Q=")
+ORIGIN("invitemedia.com", "Yl2FDsWvndsTS2wPkGtcndPVLOu9j1Za76sEoF3nIgk=")
+ORIGIN("adality.de", "o1i+s9ym/w6wh4CxNC671QBf0RwsMIPgRhisSwL0h1M=")
+ORIGIN("flickr.com", "9YeOmEW6nlEpFRegpUOZ8bEh4P+fCZzJZ3q4R2bvr+4=")
+ORIGIN("adsperity.com", "aUFQw2Uw/fY+IEeZKCcY4x1codNeAPf7zijxfwKmFew=")
+ORIGIN("directtrack.com", "5Sp2WflMcleqkpkKAyR4Vp2KAXjCCpPBjTJANCOqISg=")
+ORIGIN("google.com.gt", "XH3m3MFUIT5JiS5uywFRBQdpf8ywOBdJCZnHswykaoA=")
+ORIGIN("optmd.com", "GafAj6TcTcfTWeVuDyUIon5oBqPLKe9+/3jJuUtUKTw=")
+ORIGIN("adsfac.eu", "zm+ZjXeRYcWhvkB1c52CdSyHia3Zakx2t4e5OfNh20c=")
+ORIGIN("oewabox.at", "tQR+jez42pX8on1UiDJVRBLz4wURmQQkExWAj9uynfk=")
+ORIGIN("mixpanel.com", "8lcK8TCZyq8VHyQ6Go8EL7MAp13lLr3FqcLVbbLNoxk=")
+ORIGIN("customerconversio.com", "RyxbbiZ7fRmXHDE12ddwITsio3Hl0BrmpX20fKVf+6U=")
+ORIGIN("bannerbank.ru", "RCxq1a15ggvbye6EAJGErCZB5iJkKAqjHdqwxBwSET8=")
+ORIGIN("cdnma.com", "D2diPk5F0sPT36HpSTo7/iXqb69oZjpvXVvPAZA1Vy0=")
+ORIGIN("ignitionone.com", "RdkWDRVFqvbz4xmF8jLQp+qkm2jmhSwZxxNAUvontCU=")
+ORIGIN("dlqm.net", "Uoo/sjc/qA1b3UnhZPhs8iK3h5M83ewfPnGFQQr/ywM=")
+ORIGIN("zanox.com", "leeFZsNdnblFWsUutZ9LE49kmMZ0nM2Q4AhsAwiEPB0=")
+ORIGIN("mirando.de", "qeuThB9BdP9g2/Ki2GA72ZOW03Gs0Vz0cUi6wgI6YUM=")
+ORIGIN("godatafeed.com", "d7SKc9V7dafDQQaMrkK4zkTO5Rsgir0icVKPnNcgrTk=")
+ORIGIN("sponsorads.de", "Pwef80uIEOegBMQBGD6kLDICV6LGFc2Ql65GiqNfSA8=")
+ORIGIN("lakana.com", "yJ/HzEco8mueGN11C3bn2mk+2FxLoYyGqZ/fr/wi2bM=")
+ORIGIN("websitealive.com", "p5kMnVTKcxHmxr6xaUUtuk02WGmZevB2JGc+JozAEmQ=")
+ORIGIN("blacklabelads.com", "azo1Ed81Lzjpn9FlC3X2/6vnQYxE+cZGtxk+EJ9MvGo=")
+ORIGIN("cpmstar.com", "UuJEHjAcH64fTHMBtuIvIJnH9uBHhQoSfPcEghfiyZk=")
+ORIGIN("relestar.com", "QmSHGIwWkcAzk3XielsLaL725/t91bdIn9FU0n8W5ew=")
+ORIGIN("eyereturn.com", "tFDOM8+/eREe7cxaqKjq1MIDHRQ6B5I1jHSGL26ztTk=")
+ORIGIN("searchignite.com", "5wkJCdRs/gzxjcAi/tz6j2E1XzrJogUn7EJBAUfG3bU=")
+ORIGIN("criteo.net", "D7unWJiTohnOIEF5lOZqh9RABdcYSwXKAV5dDkD+YCY=")
+ORIGIN("skupenet.com", "xAYsImGOJVlo28LFiFP1bKcXxfrYZ+/mfX/ctfj9G2Q=")
+ORIGIN("megaindex.ru", "E/ezYJlbwJSoIyVTv16EPBNV0Zxsh0UveWDFKgcgfUY=")
+ORIGIN("sas.com", "7KuXE6nQisq5P/Rs9/tW72ZNOb1PL8xxY8E3BfalbDc=")
+ORIGIN("microad.jp", "aYKXropR9a8noZ/msW6HQnEkBTQmJUYv3yY2nsdYtqM=")
+ORIGIN("ctasnet.com", "0sFpvoLFdg+VLBeiXgulnHZ4HgEAnW0FsmSRVUQ2K+k=")
+ORIGIN("google.az", "8e9bs1hRKFS/HoVW4AWE/8SAQhMN+6WT/qxqCnSaKzU=")
+ORIGIN("adperfect.com", "fqmR935JgS5jMmWIqLQgBktvkZXJx1k0dtG5+ZjY+94=")
+ORIGIN("marchex.com", "NBNKhHNcd4OgulsquyUgsP1AFIhBVKN+mo4bYNUvgyI=")
+ORIGIN("admission.net", "ax8pkM1V8ny/PitTuzp4P/E0uNI2f/qCUfyrJSf+eSo=")
+ORIGIN("y-track.com", "g4X+09RVAgCmX41vL6c58SxyAqglkNeo3EAV2kxlmeA=")
+ORIGIN("worldwidetelescope.org", "7BwfWSVCnzIcA+XC/VmRK+FNWrLQwtPKCoAyzc2NAHY=")
+ORIGIN("foxonestop.com", "YZlYTAIPo9O/FRg7vsGx5IXm6+JKugjeMiRepmB4uwI=")
+ORIGIN("googleusercontent.com", "FKFwhGtIF5nkJDljk/cC6hvnswwBMNyQDKAlkIh7Yqk=")
+ORIGIN("theepicmediagroup.com", "i2hksGl6nwYlEBzGjvzVfzO8PoLfqphZCqY3FT6WqE4=")
+ORIGIN("resonateinsights.com", "nTop+2DVEMk1m4wvcD5Zq8JDktcWN8vVKnnSvF1l8jU=")
+ORIGIN("amazon.com", "DyfB7zRqDBriXdPkTNEjOc73l83FSEo2wiODXqaz31Q=")
+ORIGIN("usitechnologies.com", "fRhso3ragsg/5UTEoeB9XqySbyixUEo1NjKjJ8vjmjI=")
+ORIGIN("martinimedianetwork.com", "74T0ChjFz9mJDVbVj5TqGnfSLX2i6YbPZYowLkUpsSU=")
+ORIGIN("clickotmedia.com", "HUR7WBOjGrl27FOHHRuU3p+QOx6yTdq3xdLerUzayX4=")
+ORIGIN("pinterest.com", "Y4xrG2vLYYV+ZYU9ure/UB9ehQ28HgF/SXnLY1X4EvE=")
+ORIGIN("komli.net", "H9sNLgUnw+X7Uk+vCd0wRhnOUlT69Cnse/DeXt4Utjw=")
+ORIGIN("traffiliate.com", "mCSJ0CPnSsuGcqIhcur/o+SV9sAu4Ut6oLV1n1Yb9Ds=")
+ORIGIN("arkwrightshomebrew.com", "aI8ui56mW4IG6HQz4tpkH4Gu4EKYVGVHbpFQIyYRReA=")
+ORIGIN("onevision.com.tw", "jCDvBRzh31TPx9bJVIZm+iq++MYEgY96NCUFHgKAS6Y=")
+ORIGIN("jirbo.com", "PIsWES6BR7tWKB8zPhinVsk0YvK7hVygToGDOcgIjmQ=")
+ORIGIN("tremorhub.com", "ONG7Rscv0VqoBbv3ZNSJwnD97JjvGbCmPbCReQBtT7U=")
+ORIGIN("accelerator-media.com", "y7MOkcdXYUDEoyFZZXFYcvRJiqPdduqs0qUc9EDSvKY=")
+ORIGIN("rovion.com", "GclQAd/kg7cuZf2DZuH1PdM6/K2wNLS/2pZWuOhKJD4=")
+ORIGIN("clearstream.tv", "YEhi40ZxP6Lsh10i0kMddvMNy/IUPsHx1O/V3Um5PV8=")
+ORIGIN("tqlkg.com", "0qH8CXE4XrwGmhY9eIFrcqMaenAN0R3djBeZAK45PRU=")
+ORIGIN("experian.com", "3mvcWUSZUTTd04pQbdCb5Wez6pa7K0kPgnRPI23QaMc=")
+ORIGIN("marketo.com", "CsRxmMxwMnqWI58v1dRb05VYTVQVqJnK/+Ud1hDAYD0=")
+ORIGIN("marketingsolutions.yahoo.com", "1MU6vtBUp/eQgo44JiBn9NGIPqmHYgKMCCIsvEbtjoE=")
+ORIGIN("cya2.net", "6v26DRilXV+9eS2I+gDm6OgaqsbH8MEUuKY/3ycH2rc=")
+ORIGIN("adwords.google.com", "2iFSHddOnnMrskkYzy5JALgRZQyy7r0y8IWdSk+GAEU=")
+ORIGIN("optimizely.com", "T1/GWTBgFRkO+KZhAyMK585r+s4Nk4/VUNmyd+JHB9c=")
+ORIGIN("fuel451.com", "/6Qag2mwFb/26kgwrbzYtpzbgIEATRlwKu12nf4fXAA=")
+ORIGIN("touchcommerce.com", "YzgJvATJ3+AmtHbqO8yptQTZVgXHWgIN2q11hLOrVr0=")
+ORIGIN("digg.com", "MOiGhBLzJ2II85kGtVWP42VmM6Hcwm6YSo1wZCO6MDQ=")
+ORIGIN("switchconcepts.com", "kMCY8vCBDLGsr/7/PNIb7ZH9PD7q81iAA6GZ8BJe8TY=")
+ORIGIN("addfreestats.com", "TJELSIW9FSeND0bf6XyrjM0V3Shj2qk5ahxLHzrymV8=")
+ORIGIN("sevenads.net", "ARMngFn9xM3L6Lw8fh3uSIEo4ym0fQ2tBjfGBskjd5Q=")
+ORIGIN("z5x.net", "FyyO7HDastA3iQSL5TvOGti0F/RZec274CLxv/GdTdQ=")
+ORIGIN("boostbox.com.br", "Mz9EutrY8OQ6c+e/862uWemoiN2GgiuNdyedTteQlb0=")
+ORIGIN("quantserve.com", "mKjDx0szPJWpBfWPOoltvbiej5qg7Bi6b1x2eaTDBcs=")
+ORIGIN("shinobi.jp", "CkmLbxY/d2UGssX/hEiOWYE9HsJXvPAkaevUnANs6Nc=")
+ORIGIN("dynamicoxygen.com", "GNlfgH9VCNXejDRYBHbzhnqwdaVQyqHKzO4Lg5ky5mw=")
+ORIGIN("contactatonce.com", "q1OUEd5LMZpY3IXykmFgIDaHMQBXDWq7U0ncsU0UPuM=")
+ORIGIN("google.mg", "FbFum2k/tR7YrNt/l5AHPXzh3Fv58q2oRsC6TqIghzI=")
+ORIGIN("google.me", "UHyPtcO8Eov4/N4drByh+ZWNsm84cAEI3/DbzQ6rsYQ=")
+ORIGIN("google.md", "B0TEWh9tM9CXduQUrAIfms08C9wCu1bEJxivUQslJWM=")
+ORIGIN("google.mk", "2NZUkEAvhZ7HggIWZvjkfyDahrX+31//EnwzTgT0pmc=")
+ORIGIN("metanetwork.com", "nTGfDf1jorZ7+Z/knkvDrMqJno4hmEMfELhrHKgxtpo=")
+ORIGIN("yahoofs.com", "WRUwV4rsdBAmZZfdjZw+mnNh66UJGaOu88dwgbfAOF4=")
+ORIGIN("kokteyl.com", "CDD4GCgY53cKo0iiZN7SQ885MABBNRqLMqoGz9Y/E78=")
+ORIGIN("domdex.net", "PwG06EKyumOa+K3H4KqT3xPm3EHkin7Kb8Gwud4ofBA=")
+ORIGIN("joystiq.com", "CmAl1N22afQTvHF6+kVcOIbErvhNwk7/v/P0VHX2fF4=")
+ORIGIN("google.ms", "MxVr3CIK8Zt9cZE6IdtGcc2Np90oaDECj19fgc/sgCY=")
+ORIGIN("netshelter.com", "6Tvt4x3DEXXJj+5fivfXhmJE2RxXX5zEX9TSZ2XWTpc=")
+ORIGIN("google.mw", "qp7BFSsESR55x4c4yLrivgP7N5hX5/xPYJ4sv6+YQRE=")
+ORIGIN("google.mv", "wbOz7Zz4DpvPo2HcykwEcga6p+uviOSGv5EM591vB2c=")
+ORIGIN("google.mu", "1lxopNVrcvCjueLNhl26o84abJaYH8U8OZcGVMnuD9s=")
+ORIGIN("iasds01.com", "baa0LygbfFrLo2JnbieP02z19w0KRzZ1I03pBcIGpxo=")
+ORIGIN("adserverpub.com", "2MA48s4xSlY9Sg2bFugOnlbbi1olKIod1C+9M9RizsU=")
+ORIGIN("ewebse.com", "4xFWDoI9u+7p7Y1BBFekdkOBZagnywdeEPBFKEqW8Qc=")
+ORIGIN("olark.com", "EL9inOG9VCEVY6H5pBd8sf0pjbrCzRFjlmX8zHCa+LE=")
+ORIGIN("traveladvertising.com", "dUrIr4Xthw1OSXT1nimCtiFNNojeJrH13GYLK1JM4lY=")
+ORIGIN("mapquest.com", "0g/Aoe+Iuqq0ZFjSbhZVXAfMD4mAWzW61u1NBMlKrBw=")
+ORIGIN("groups.google.com", "xen3JDG+xaPAlJKPWzeZWIJbGmFwFnzOY3rBBEJ3oKo=")
+ORIGIN("emxdigital.com", "Te6/rJP3qE7KSJKlbgQSMd0YSWN8kWXVo9TgZhXFnXc=")
+ORIGIN("chango.ca", "AgXXQ8WGtQygCTh10PtIrAuUgs1h//hqCP9YYrzBbq4=")
+ORIGIN("ensighten.com", "0sycK5ZBgof+Mcs6gARfqC6uXbnRXVsjYazSmpKAbUw=")
+ORIGIN("hotmart.com", "C5GebeMcJF24rXh8dNLIAiNkQJILsurdkTdndQcZFlk=")
+ORIGIN("barilliance.com", "5YMp4HVN3ZQA8WRw4ZFwltiTOpJ7813GUGn41CYtXfQ=")
+ORIGIN("nr-data.net", "iikUgReFeRO7gfdDzHUKYYvAtYx5I5Xms0+SiGhJdj0=")
+ORIGIN("medialets.com", "gFk35BPQDX3Dc8lTVB3eg56lTXk9tUf0QinEqmtPC54=")
+ORIGIN("brandsideplatform.com", "2Aor4FtQWmwsGC5NqqnM6vn94lB246uMfCaPwx61E/8=")
+ORIGIN("userapi.com", "AHEjjwFRRIk6C9UiLD0olZCZmaRsmOwbmerJETcUT7U=")
+ORIGIN("lfstmedia.com", "Vj+zgZz/8dKWSwXd91R1TVhhu+jbUbYFqrl846AJ/Uc=")
+ORIGIN("kinopoisk.ru", "bKE7PsD9CMzOOtSESlnkhvYitTzHW142YuB9JDRa3wU=")
+ORIGIN("clickdistrict.com", "6hYyaDaVZFPmJUcUZ9h7BZ2Fhm1gNVtgFXpZcwgiO7w=")
+ORIGIN("adadvisor.net", "Me9kKp1u1GPFFOm9ooYMnoiGPCOeQWplu7dnDuTvzq4=")
+ORIGIN("adspirit.com", "dR5Nh9/etZmkGfRMDABX/vjsW38cXBVvcNP0wwsqOsw=")
+ORIGIN("kaltura.com", "VNW+5ymDw4UqsI0UkgvOoaBoSLMCQg03bjV0vUQik6c=")
+ORIGIN("btstatic.com", "m4AQ0RNYtEJK4kQNtr/x1u55ykpvJwv+LExPsEwq2xg=")
+ORIGIN("leadback.com", "xGAPZ6PE14YZwXuJroRrYPbH47PRS93Kq/HLv8vxN9s=")
+ORIGIN("ppcprotect.com", "CIx29J/hM1O5TiH2GHggAgwIqI0fpQ9obCvEqND2sso=")
+ORIGIN("vdna-assets.com", "n3nn3YXjHZGHmKTBmmkvYP4tMCIDhz3tX0l4rcHM3V0=")
+ORIGIN("adfox.ru", "sh4CXPgS1+lUfXsHKYBizQh/Rfo/V4EVO2rfNsp3H9c=")
+ORIGIN("interclick.com", "LvCfDU33F/Lr5hw+SZ0v1W9FM/QdDBi+Ym6TBQbVvT0=")
+ORIGIN("reddit.com", "Y7jPzch+J1UGqhw9qv8G0WdiYRz5jIGZQyB+rnv6r0w=")
+ORIGIN("ieaddons.com", "ZqaFzcCyLTtyhshT98D84mB+0w3pU8pY8nxrvg/WP00=")
+ORIGIN("google.mn", "MiuIHgZIcVmxCZ3Mpah7cKTZDThLVXUXPV6RrxeSPGE=")
+ORIGIN("echosearch.com", "C4YZQUG4xXTFcsP+HjDqUneOf3NLw+/fzXQSbtpFtC0=")
+ORIGIN("bitmedia.io", "frOM3axfVJ7n/IdkG2VlKXN3LV3SMgDtk10vfN6NdMA=")
+ORIGIN("udmserve.net", "E0/RXPWMjRdnlbHD4FfP4hfINOwcSy37CU2DTHZ3iNI=")
+ORIGIN("stackadapt.com", "rBYUhD1+sNOL4cnar0kLq0PjrSHVgXSgP8U4gjttuek=")
+ORIGIN("millennialmedia.com", "3IEO/EJFjA7cjb3uOwINASrXo1j+6Er50RNSZeK839U=")
+ORIGIN("picasa.google.com", "gysJeYOwYsgFwERqMUj2m9kQIZj998qVqm740/99a9s=")
+ORIGIN("google.ch", "x7Uf+U1ql8KFcHhwy1IZottMgpme5TNMdhYehUgmTLc=")
+ORIGIN("admarvel.com", "ocK/DTgNfQ2c67xIeOki9Oqfo5JmsD+cycIn2qJtdNk=")
+ORIGIN("mkt51.net", "OSLwZ5GsVZNEpWHKK1VwIfOfL5w5RjvtAXwUC6OQ0Cg=")
+ORIGIN("google.cl", "546HM8M0Wp1WjzK4HqVOCDMd0Ql9BcgryBjGuMyHqxs=")
+ORIGIN("thinkrealtime.com", "/2C5dz1Ss7WwpT8D6RkJuQ1CupFBHsaK/wyZ/1kWhiY=")
+ORIGIN("google.ca", "F8hpVR2ojJX4qREKApIXU0gToLsWZJHzKgs6dSCMlcw=")
+ORIGIN("adpersia.com", "GGmQT1YZY3VcXaNMBzFiOQ9lNq70XL2S2li9sMmxbfU=")
+ORIGIN("google.cd", "3DIt8oOAFgd7MeS7hAparpnjis1nbo4jylxdGKZ1rtw=")
+ORIGIN("google.cg", "EzdLGoCasg+aGVva0m9orb+X7s0nEMOYw7PUnN+si7s=")
+ORIGIN("google.cf", "p8idACgpwTndX540CjuX58RQyBGQEVFK9FkJAznnSZE=")
+ORIGIN("google.cz", "LN2NBW9VxUZ+KvllelMKdiR3yNpEgRg/JdFfnzhG7ZI=")
+ORIGIN("pheedo.com", "kuO9ZH5oslIPbMOhWB+7LUrDDPLS1+89g5IzKP0auSs=")
+ORIGIN("akncdn.com", "Q0/7DeaWa+cJcoRcl8oulj8n0no/JbaUXMckHfbnwec=")
+ORIGIN("trafficrevenue.net", "UGWp5MzHP7anX7B9TSlrJi7k7HbhZxA0d+F9mqUu/Zc=")
+ORIGIN("adreactor.com", "FXU94/aW9yB/jv0r64hUJ1jaRlh2n0NoxAr2NYXdpLs=")
+ORIGIN("analytics.yahoo.com", "fPfQTfHRDndHGAw1cPj1HhL1DtRPC7uj3gBJpYG/7ms=")
+ORIGIN("google.cv", "B6aki1/PFv5CWyJmaP1YE4ORA7MEluuPt9JPhlaGTBk=")
+ORIGIN("axf8.net", "SJzyJvCU7yNf2PipzWsW7di5pg+SzMpamyfG4Dfp5hM=")
+ORIGIN("beanstockmedia.com", "wYXXDkxW7N+KSq6YFZvIGK8QWqnJtxrMQIhI9mFMJZA=")
+ORIGIN("gtop.ro", "5j0TdwyQ7AvMsIVYiUnV68mNz1d+hgCg/AlRFbMy7Xs=")
+ORIGIN("google.com.do", "9gZmzicakB+lvDixIWedcqVCQKdmhorPxKSj0RAH4J4=")
+ORIGIN("xad.com", "+9ICFdzoZOHHK4G0JQjBgF9jaHiDj6kemb11hk8ajwg=")
+ORIGIN("pixel.sg", "79DOndo08/+xKci7pjuo+ck9ttX+3i0IiD4a8rBapFg=")
+ORIGIN("hotlog.ru", "uCtkNOBad6G6tJS2LBAYOFpFyODRzbTQi46Ovz4vsKo=")
+ORIGIN("adsfac.net", "eHNovZ3wJujh82BdEh0ljkS2UKrsnE2JN2LicMGSc90=")
+ORIGIN("sketchup.google.com", "4adbukGj0RzHQ8+3nfZU6rjcviAxJEwY22ceJPrfugM=")
+ORIGIN("conviva.com", "ZiHxoPvSfjdmbk9N8nAjkYzGCplKadfV9YQLWCHI9Us=")
+ORIGIN("mediaocean.com", "RsXNjY3HEcH56rrScEwstoaS2yb43WLIWLxeuDZPyok=")
+ORIGIN("shinystat.com", "xLK++t8SqIHwkwcYgIdiABZSICAZI5uMXNmo7Df9JWM=")
+ORIGIN("tweetup.com", "ZvWLG1eYq55FJfWPqWw+9Uhh4O9GYy4RJ4d6k3wyFCw=")
+ORIGIN("mindset-media.com", "vBziBg6GwM+y/R2x900xAjf3DA66156YSaho1Sv+wwM=")
+ORIGIN("veremedia.com", "JYgr9acFRsbbi7EmlL3UmWaf0d9QSZswZB5Vly/i7WE=")
+ORIGIN("infogroup.com", "pu64Hnf//pi0QZogP1vCbRoruoAYfMBAy04rcTb0ddQ=")
+ORIGIN("shoutcast.com", "gVMxBTqTp0lLCD2L0PT9Qu0hG250oROigk6Bm2/U7qU=")
+ORIGIN("twelvefold.com", "lEvdIp3tpx/IZ8YDG3wt0MnPZz18TmCn18UYztYOz5E=")
+ORIGIN("peerset.com", "FGuIyFWPjHIMPEnFP28uvbp1rlTn0sWNlDqaTfU8Wlk=")
+ORIGIN("wishabi.net", "wA6Xlk8hVcT5a4+AoELj3oLC2wCQCLA2uBM5wEj91Gc=")
+ORIGIN("switchadhub.com", "oF2qaDMtkBw7k+dTwnyU3t+QCUrcX00FGZ8pyGYyU4w=")
+ORIGIN("estara.com", "AbRwgc9pHMaK4lezSL+TSlHdeVx8M3ZEF37GKcVmr8s=")
+ORIGIN("dyntrk.com", "ZqbUfema6OjafVQv7zZhYghHUjaCF7lJW3OAkTAGshA=")
+ORIGIN("ad-score.com", "+tV3Oakjw63hvEkwU00IeybEZzqKAqj4GD+VrEvvNJc=")
+ORIGIN("adsummos.net", "sNuYs6deC3Oa49kf/6YgIEQzUCVuYBs++K8vhCCs5EQ=")
+ORIGIN("sptag2.com", "Qqf7sP0ytdD9vk6WbVCkDv8OjPkPC3++HJkrarY/TuQ=")
+ORIGIN("daphnecm.com", "S2eyId/mnNWYaT2IdyTm2RRvJJwUuCENKtiFoYh82cg=")
+ORIGIN("websitealive6.com", "Aur+NtRD3FHz44KD7b54Ls8+SBHmHiD7aYk4UB3StvI=")
+ORIGIN("dailyfinance.com", "C4P11jI/fL/+vw/28ZUszXJ0xHLjD8fYCs5l4U8+SWM=")
+ORIGIN("i.ua", "3SjuIK17KVP94ebIKRENSKeCcLT4iEqlpx+owW8H0pQ=")
+ORIGIN("adternal.com", "KalD1p1jS8XelXVybEaOX1w0/LQrlz/3pvLKF0bPo3c=")
+ORIGIN("www.google.com", "vJqPK2//1YVx4Yi7EQVF+Ps69RzfGmNpbVBamHCoW+U=")
+ORIGIN("dmpxs.com", "4NnbW4vzjQfTW37xV3t+hPeGhEPfuHIt84frlK9qBgo=")
+ORIGIN("permuto.com", "jy0X6AdIuXABbdInZw8m64twq1BWSgU9PGBjMHMMqnA=")
+ORIGIN("lytiks.com", "E+SvgO6FVVieXJXZa/jRfMlmZuRkmH2JWqLLZOdzFjo=")
+ORIGIN("yandex.ru/clck/click", "rbUJTeMVNsu+cCVgGv3wwaV/iTBYB+mmyLk0CIkeGOo=")
+ORIGIN("lifestreetmedia.com", "uNuXMGjMi6ucDE4EghXD2Gt6YONHizxtc/MjtgJZWuQ=")
+ORIGIN("iacadvertising.com", "7kiJsLKAeRLzEK71yW7c0TDHOy1JPpU4OVJQh48Wa2M=")
+ORIGIN("google.com.vn", "wieBI+zdpgrN0PXNzs3jMH/mFj9S0fu7Bwm9B1lwYHA=")
+ORIGIN("yoc.com", "E6KZV0OjEKfjDGOSdyxUeCMqyiz0WU9oVIjZmAhgbOs=")
+ORIGIN("google.com.vc", "8SLhnFwJliJl86DDpQ80yTOd03d1Ps3zHTfN5dfhBRc=")
+ORIGIN("heronpartners.com.au", "EKmGlXteGJVSI10btVZ9Hix7gHXF/pdDAVohNwYv92M=")
+ORIGIN("dedicatedmedia.com", "ZCr3eVd+Z3lI3FWX+8QmbGtMXepgURI5tJmDMrxewSM=")
+ORIGIN("eqads.com", "vZIQ66NeasMvaMaEUe6d9k6KagRch+eobsqSN5w+hoA=")
+ORIGIN("projectwonderful.com", "k5o3pKV2Z9jVm3Hl7tzsfdX+KmuqpiUtBavllKxNc1Y=")
+ORIGIN("evolvemediacorp.com", "zpEQTjQEdDCsstKBdEKca8jA+iM+zxiPMYrrnW528I4=")
+ORIGIN("webgozar.ir", "4j5dOare0AX2aVLo0zXwNedPfluJNp0TuHL7JVDcefU=")
+ORIGIN("liveperson.com", "0hOIheaBe2IrHt8tB7UJhzSw9t80WWXZpVwanQwyTr4=")
+ORIGIN("websitealive3.com", "BGQ9yzLZp96Iqm0t7iXrhD26kBK/IQ8AUfkXnqn02p0=")
+ORIGIN("dwstat.cn", "206q8jUyaCsYPX7yA+JXWvJyDmcbq9ahhmszX8CzGa4=")
+ORIGIN("adrevolution.com", "JeK60//TuczAoxBsmCZl6oLrcXX6BfVtkFUjBke3n2Q=")
+ORIGIN("omniture.com", "CsqQBRoN2C/bLi3Zj9zEkHYt7Jcpkook10fHi+w7jNM=")
+ORIGIN("veruta.com", "9I6ej0XsQEePvelgWRn9cmIbXkDc6Cv5CxulNJc6lIM=")
+ORIGIN("gravity.com", "gPqWmGtoz1Gwa99U6NLkZofxBaAHWL+mlxp+FzsxXpk=")
+ORIGIN("netshelter.net", "sH3PAKpgrk4KK0VQlg6mFeX5zmbiffCjrdeS5P/CG44=")
+ORIGIN("fathomdelivers.com", "8HXJjz/RNbP90JMQdOnLr85qO5vx4Z8jWO+fhPiB5jo=")
+ORIGIN("mc.yandex.ru", "PNZAwztdc7k+X5y9lgqCT4Zp6dGz3tHpfbWiKim1r/o=")
+ORIGIN("compuware.com", "/tiTZMjwmHxuxf+CE0OgYRsSAuMW16SQeH1chrgy5Dk=")
+ORIGIN("w55c.net", "QKoc1xcuERNaZhDo/NBYskByq818T0hwy/26kNaV1ic=")
+ORIGIN("anonymous-media.com", "Sxtqpe1QzdGniJHfrmkEgUKZRZuIdHR6PmJGdQb/Ibc=")
+ORIGIN("bizmey.com", "W8NT3a/x4lp7082fw2jWltEMKZodztG2OEzoxTC2/Kg=")
+ORIGIN("cpxinteractive.com", "8MEN1yx2q/3nUOUjsTfPgPpBP2K1WyzGTCNxLEfV9EI=")
+ORIGIN("mobiletheory.com", "yYFUxkUUfPnVfuTda5/CbAzKErv80Xv8Uqx/ybAViEE=")
+ORIGIN("picadmedia.com", "f55vWm6KWAVReZ1JIQGtp37Dt4/Ar2NxrLnF6deBxFY=")
+ORIGIN("springmetrics.com", "JqU2UDBBpezNhdWQ6rRmYmSzZGLR2DDHpTN8NfdCElo=")
+ORIGIN("wunderloop.net", "KpZ0PwuU4cZyyjAOXU/i8HGzZZ76Q+3xn35ASQFimpw=")
+ORIGIN("msg.yahoo.com", "nRFNUlJcJvSHd5daJPPjjK2XPZF//nEn3Hy3H7hDekc=")
+ORIGIN("csdata3.com", "S4eiVPvOScLfuKNuZFZ7qrLQ4ZU0Q2JW+ymQ2azdRU8=")
+ORIGIN("buysafe.com", "iEpTfJIzDt9ogMN2UPHkqZQewqXEXQn8sdUnu0bA1Xk=")
+ORIGIN("tns-counter.ru", "TVzPxqJx14XrxqoruMtxmEMJZjHlPX9LBUnmuOub8UI=")
+ORIGIN("dinclinx.com", "Tv+yNARYT/2y9gjarhxpiFlSFtdec/P9bmQxffsevQ4=")
+ORIGIN("amgdgt.com", "2ot5D3UINdSQ4JEET3fAcqNgPhCeJSPbbY3Y+lBImAo=")
+ORIGIN("glammedia.com", "9O0BlrL5QDgOY39bOlssCKGAWxbgNZqvWN/VS6XKetU=")
+ORIGIN("quakemarketing.com", "3/jVCVkwCibsOE5C9B1KulFGU3GhA+CqykJKgyuujSQ=")
+ORIGIN("advertise.com", "f5ik6QEVMkhZEV5/T+lsjtoeNvYS2QrCiH1YVHTKI8w=")
+ORIGIN("ppctracking.net", "ZLEts2V3492WdYabOx8f0X5UF47WTLAe6S4PYvWmXRA=")
+ORIGIN("ad4game.com", "8VrSZSkCjQnnFowBg7okluuLI7FGQlqtEyCtBJFgbic=")
+ORIGIN("websitealive8.com", "abwTyMuDEzql4/vegcAAH4kDK00eYwC+7T7Ek3Oc6zw=")
+ORIGIN("popadscdn.net", "8eFA2PKudEf8XSE02KHliQNyq+vQSDin/pD1KhflC4A=")
+ORIGIN("crm-metrix.com", "tFXiZ2T0b0kXkT8vdxMHxlEr7mbiBz5YgCzzhXkim8k=")
+ORIGIN("trackersimulator.org", "w5+97KYvqCfTZ31OARaAqvvPOCo1krYQuIfjEsyUWYg=")
+ORIGIN("spongegroup.com", "+c+DpR8GCDQxD8mL/W0doMRDucRD9QBI88VY32ftQ/4=")
+ORIGIN("mybloglog.com", "/4NeVNQ1M3654f6WhT6Rp9Rg/UC3CdwkOdjwgVB4K1k=")
+ORIGIN("google.li", "bSg66dRJ0Y7Xi68p0BK4KyUmgC0fUVOaMaCcDVpfv14=")
+ORIGIN("google.lk", "0M7cqIquw23mEG/hPKJtQJZJTd2n3LIz1hKNL5Po1Kg=")
+ORIGIN("lexosmedia.com", "nDPPUa6tm2v6SmNppuCY5R5TqTtqEpFCja24pwwvPFU=")
+ORIGIN("adlibrium.com", "HDqJ8S3hhwl6HJB/A2s5FxSQ1B+1o55XbQHtXC5jxJY=")
+ORIGIN("google.lv", "IGPTfCdZRCF1/Qp//4SnzWMBLbFETQAniEqGMd+QlDs=")
+ORIGIN("wpp.com", "dSetmBZ8rLLLCD1uFeyqdc6OIR6VBGHMI74WBlYf2eA=")
+ORIGIN("adknife.com", "QnSNJPpPLFqs7vYXGRo3AJkhJUwkCHMKS+OhNLreZBc=")
+ORIGIN("crosspixel.net", "zW4y8mD4u3XpHrnhBp8mwQ/9Bh7qPLt7BXUajfhhv2M=")
+ORIGIN("snapengage.com", "dOHq6VLSI+Z6mBCMepKXxPAE6HQWFkTpf8kcSdGNmTY=")
+ORIGIN("xplosion.de", "8+0JhpNNG8DXKa/lGsFDA1ux/MMgmFj/CCks0sEnqO8=")
+ORIGIN("adkernel.com", "aAIrahvZwEzvs6RN5f2GIQ2vG0HC7hhqt7FJ9QIrMEg=")
+ORIGIN("websitealive4.com", "jt5oZTrUUh5TDJkEMk74ZNErld5gQccPO1B55RSsQ7Y=")
+ORIGIN("carbonads.com", "mefqduGOSJItMB7R+Dj8XuiX4UVwb+ixQllUXWecE2c=")
+ORIGIN("otracking.com", "UhBmG37WdPq6vXsfqdOBbSuiLyobqXKQKlbH9cgQGW8=")
+ORIGIN("autoblog.com", "rPgd1xerke9Lzy49Zwrib9/jRmQAldnUUaskRhHv+5M=")
+ORIGIN("allstarmediagroup.com", "IgYlMENDsOWmSyI8gN54Cp4TRkJugq/5XcaDrekaVBM=")
+ORIGIN("bouncex.net", "qkmWxivaN0fcvwbEzkr8vZrZOOwAdAC3KP78Ex86Enc=")
+ORIGIN("lookery.com", "gY3To6JJa9KEkh3V36QnfIL6wnQ6KIOguizkpx89xYw=")
+ORIGIN("convert.com", "PISd8WcWNhXNCiuq+8m9OZ3TgWzltV3plCgbW3WQ2yk=")
+ORIGIN("adocean-global.com", "6CCVz27fKMJrRaDxKU2Scx+3RbKCjtC5LxPkZaXnzXU=")
+ORIGIN("att.com", "d0nCZX38Hk2lrLEsa57DjG5t8uQuxA+UPKHBAKbDf+Q=")
+ORIGIN("designbloxlive.com", "qZSx2/N4uZJk1ydmpFGnxc+dxQiuXub0DNgpbkv1g3M=")
+ORIGIN("vistrac.com", "LAfv+rV2isKtDJ+WmD00GpuY6nAtOeXNAOVgDZ2EZ24=")
+ORIGIN("intellitxt.com", "NVnY4Cd3jXQngn0PWWi2EAnS8bNPJsvszTGir0rulz8=")
+ORIGIN("crazyegg.com", "4+/K5s18gWYF1Ic2GgPEZrH1dAinX0jndtw1uiS/56k=")
+ORIGIN("mediatrust.com", "rfl4Ymc3xN7C6Dh5tB8qoCSPKDOOME6PjssAIIZlBhg=")
+ORIGIN("adviva.net", "HarhZIbpE4x14AJZmZta3OsAuxQRfVWCjcq2uC5dYME=")
+ORIGIN("campaigngrid.com", "zbZfKo8ZFCuicjSLlzX4Xxg3c5N9Q10JNFQl9hwBBgk=")
+ORIGIN("rocketmail.com", "02vfjTiqvuyLePEfyprfrmxD5X6dYrRZ9kQkfX9N1wE=")
+ORIGIN("superfish.com", "s3C5LrOHkv5LcVHXaNVWpVn9rMZnqvDGcOWqQopsYu4=")
+ORIGIN("adomik.com", "sYMWu5JDeAIqnMPnJqGOcN5rsbh93KPoHWItQ2pl7Bk=")
+ORIGIN("netconversions.com", "ejMElOuFS1cQ9JntjBwr2Ru/y40H8VXcr6gJdswNG0M=")
+ORIGIN("adonion.com", "xhbtiZVLFexGZbgGFjMjLhswc8SkNPN9RA33ToKW+Wc=")
+ORIGIN("my.yahoo.com", "FYnJcJFlhsFlqyyD1eZLo9cyE6zY3g3P8Y7ppAv63zY=")
+ORIGIN("adriver.ru", "in2sNGu/AEhp02fc7hy0lAUUK8mBIJ6AUxubpDOEdlw=")
+ORIGIN("adknowledge.com", "o0zse4Z7fM+b+ViG85zFEBuSdYdQpoKQkDrAnL0DajI=")
+ORIGIN("rhythmxchange.com", "8T0gblAQunnZ//eXTHBS6NZAfkuHGqzklgmh5KW1Dt0=")
+ORIGIN("thebrighttag.com", "YJXz/sKCs62da1tHuMpXfDtvXrBHx/rzPeTgyt9BKpg=")
+ORIGIN("oxamedia.com", "z4PxwXxQ3l/pLTO0y1I5a0ATT1sdM/JLNB8h37aj+6U=")
+ORIGIN("movies.yahoo.com", "Y10FiN5kWDV5LMSZWAZs6TpI4D7GcXbD2cBncMrcGpY=")
+ORIGIN("matchbin.com", "M8xBMW7EtS7cW0qJIWkXgKbHJAj4L3tpQirYpX9NFzQ=")
+ORIGIN("opt.ne.jp", "y9HMvDonlBduiZAre9DZ/ByK9+0W413xHnsc9mvqu04=")
+ORIGIN("browser-update.org", "D13YWLbPoZomvza7sraf++qFkIjdv7MtLMs1sAz4Nr8=")
+ORIGIN("websitealive7.com", "er0qGppCIDgii70Ssj03XNwQ7Hjb0oi+nG6voZJO/GU=")
+ORIGIN("apture.com", "NSTDqUtQrEtgnuEDnpRKEJi46ThS/xgpUdy22UJePhQ=")
+ORIGIN("infectiousmedia.com", "mnIoFQV/aoK0KnbqZ7N41YZI4f0R1Lbpte0mZftw9JA=")
+ORIGIN("undertonevideo.com", "u/PuW+sHCPtLjGkV0Vroel+Zi2BmbTYobZF4SM+vG+s=")
+ORIGIN("google.cat", "k4DmM08D2qiWXQPhGPrpyrXirM33cddjtW2xvV/yyHg=")
+ORIGIN("huntmads.com", "9H+w/cVuKsdQop2z/YlEM9ec/ozs9iychDUEX94CQSQ=")
+ORIGIN("google.bj", "iPeNPS7WuZQ/5sTeF2+NpMQdq0qrjHhHciAVSHNKfo4=")
+ORIGIN("google.bi", "3rtF23RFNUZaXyuAizDKRBDiKbWM6D0eUgrxcR5chIs=")
+ORIGIN("google.ba", "DUSYSZeg4k0u06L3xuP2ye0q4pIcGq1TgfDVa0U1AZs=")
+ORIGIN("servedbyopenx.com", "iznY7JdbbgWEvdjloOtxKGmqwzl8ELKjycl0Fn3UJJY=")
+ORIGIN("google.bg", "BnCLogJbzLdBYDbKPxEEm2etYQl6zq89pKhZ520alAk=")
+ORIGIN("clickmanage.com", "AU6E/BZQlPvADha2b7N9NLc8xQyvA7eyJ/xAcHm0qqo=")
+ORIGIN("google.by", "y9B0FoAFG/T5HmhtZ5hw3D9X2tbqOkogY0iPtJa9/as=")
+ORIGIN("surphace.com", "KHIZQy89ZTBaqpkTZSITwGnWjn+pZnnhlRwMjAFm84g=")
+ORIGIN("google.bs", "XljeXn3Rm7rYRvO+F+Nn05Jojg7+bkML09mfPY5bqEw=")
+ORIGIN("adspirit.de", "0lG0/truCjj5wk6gR247O+9gTIG3tSORfYcvydMC7Xs=")
+ORIGIN("google.bt", "4GaOzpg+S+HVv3nzULvz1PjvWhfPanbpRY+OKffaydI=")
+ORIGIN("gaug.es", "zu5jsFSLhxBLCEpRDYaNYN+JicD5S5lG1FePnwwvpPw=")
+ORIGIN("tapgage.com", "wjCNKyL58d264ItYPo8WrHpkepliC/4HHjcBe40zIns=")
+ORIGIN("oneupweb.com", "58Yq06aCdVYVbGZ9rFQUZHlBBWYkof2vQgutxfblkFY=")
+ORIGIN("affectv.co.uk", "93xddV9hnF2qfqLT1UEsaU0d96T2xa65Aw1Xh4xoYWI=")
+ORIGIN("paypopup.com", "Fp5K6TthfsQVDJixT0ZJ96tHIFr52NazB9wfK0J5UJ8=")
+ORIGIN("carts.guru", "aJvJm6hhvHjsx5vd7paUq+/6rxk/HAY8SlcX6ho8TKk=")
+ORIGIN("gamned.com", "ivaB91dO+U26QN0Frjg7MkFmgZDgLKiY25imJ1RC0rs=")
+ORIGIN("yieldbuild.com", "EFYzwzRBVKFwoe6UGvlJZLeedBaummRiR0qbUGCdqis=")
+ORIGIN("sptag.com", "sZ9cgdH8TtKZoPMCzqmhetNceaLRH/n4cLax8ju4bO0=")
+ORIGIN("reachlocal.com", "Jn8z4v6wrZaK3zEspvG+8xRvtqB25OAJ8y8bvfHMcD0=")
+ORIGIN("eviltracker.net", "K1cV9PtNwyBpd6plW9sROObHwRcU9s2xrhJEmLrnK88=")
+ORIGIN("orbengine.com", "PayIC4V/B2uTywlOv3dypCqiqvdaXf/x+6oFkj7LZwI=")
+ORIGIN("trackingsoft.com", "EiHP/Gac4SOOvyQhIKGYhjPFfNYKNNZeBRinn8+27Rw=")
+ORIGIN("nostringsattached.com", "QyWLuBbowDWoPhp6SerVulx5gmxBFjUKD0EndtYmaXA=")
+ORIGIN("zenfs.com", "8rUvUWwO/3lL+yV6WOlvPt6t3N5k4XqTUg2ngskf4/o=")
+ORIGIN("openx.net", "N2yxTBUwCIRdJzRyvTu1E9k6xDFv4Mivlvi/R/VBxmI=")
+ORIGIN("demdex.com", "7g6HOQuRswnx4IJ4j6ZqjHynFhLYODwRKQN4RlTPusU=")
+ORIGIN("crosspixelmedia.com", "qCHolpbrnqGYKTCcO+rC4WPPxqU1ypw7g6m4j5A8E/M=")
+ORIGIN("nowspots.com", "fqofyOV55PGcHGOUcYhE/73b6EQ107dpw5RnSIY0WDQ=")
+ORIGIN("scorecardresearch.com", "PBR77QQCAh1DGHwG3Oe3UfPlynlzqpKY0QhJ05b6Ryk=")
+ORIGIN("compasslabs.com", "6ax8UJMTPQgLvILYac8xsCUV8r/B9aTxq2E5myr3NHI=")
+ORIGIN("various.com", "tVT8ZfkfShu03gdLEMKSBVXJ/8sOswZJ3bXm3OeTs28=")
+ORIGIN("medicxmedia.com", "C9h3TE/GPOsEAKwiv6qUPcj5q5BHgGQkFvOc3Njg2/g=")
+ORIGIN("optimost.com", "nRMxFiGdLD/ogOBWZlqBaVDHD56enah1UxWYTKcRzKY=")
+ORIGIN("adlooxtracking.com", "BBq2eH+GoIFgUj0KKEi0ZR26lvlV/bIiwyfB8qiXMBc=")
+ORIGIN("adbutler.de", "PjzD1i+4C882oJ67vFls7pGn26tx+cIqPd2KVxwXe+Q=")
+ORIGIN("windowsphone.com", "ZX3pXTD6OTW1hsqEZbWF1Hg39emDx3U3ehELJPjj0fs=")
+ORIGIN("clickdensity.com", "zZ3w4VKWJiz68G2HVbvvff/CaEaCdiuGgufTUbkaIV4=")
+ORIGIN("gumgum.com", "RCVqKGRSE/eZBNqF7W+5uvHFdzRmG0j5IDDoOU/nXcE=")
+ORIGIN("adonnetwork.net", "IoE/tNI9x/474hl90K0Uu/dOf4k+hpSZYsNFK1w/PLw=")
+ORIGIN("ivwbox.de", "0Inl+SjZyOCQC2rOgTuAuMlqbDPnYUueTZ6Nfo+il3g=")
+ORIGIN("csm-secure.com", "GiYPYviUBALrdgx5MKC139ZzXfpGnAPXpgATx0Lc2uE=")
+ORIGIN("atwola.com", "FljpjwtfAW//yhmIzHzxbj08wiU8UL7xui+aHb+1A3w=")
+ORIGIN("accelia.net", "M+nOwkJf98I7XdGhVMk5QE3crfheAbIEvEujzu5aDsQ=")
+ORIGIN("demdex.net", "RiD7v7ztR5Sits5FKjWoww0gH0cijPsX3LVnc+tST8U=")
+ORIGIN("hipcricket.com", "O2+3XKxBXhO6ZH0wydEcdBHCqIp0tjnfljoNrEaXDWU=")
+ORIGIN("touchclarity.com", "/ngzbBBrrHgm02Kjup26lEBR9zf1qLtyCiPz0uIDVxI=")
+ORIGIN("trafficfacts.com", "VH7DF3Ey4kK2avLlZz2QPOyN0pHcgy7WeyIMB2EZg1s=")
+ORIGIN("aloodo.com", "bz+zvsA6+UzbWz6a1ZFPh36c1gTNPr0GbEtpAgK3JgM=")
+ORIGIN("yieldads.com", "C+hM4lGQA6qtZiPlld0Rf7iI1WpUZ0yJmnK5NFKP/KM=")
+ORIGIN("did-it.com", "yWJDUgjczrtKx9vvOwS7UVxdPCr0o0QTnUDgDUGqRRw=")
+ORIGIN("e-planning.net", "mwy5sQ8Vmo1DG/Ezap/zAgr13EYBy2yL4kQ6QjFT93s=")
+ORIGIN("creafi.com", "2WXK7KcdkNcPYdImSYTyN4CMb4TsfRcAfdUVzCx5bgs=")
+ORIGIN("manifest.ru", "vLViDR637oqxqGivTf7msK2QO94WG+nLdypu4kJ0/SE=")
+ORIGIN("out-there-media.com", "j7OTv6FCeL6fRKiiQ+W9JyZFae9hJcHNmGSK/ld+5yM=")
+ORIGIN("tinder.com", "PUnNFz/j+8i2kObF/IAFpVkVFLCCcjcYSPV3sxpNBVk=")
+ORIGIN("ligatus.com", "3kHilnp/OaIYgoFtAdTe8XWn+dnFmwFxp0MkX2d48s8=")
+ORIGIN("an.yandex.ru", "BSGPiITwntLN6Imf+ya24Kz1XRJFZaY1x2ugKbzBOKM=")
+ORIGIN("sublimemedia.net", "j7CnXCrtbgF9a42k4+AIyJ6A3X27L0e3jg+YhDX8fGU=")
+ORIGIN("newsinc.com", "kVDev/mub6TeeqCQF/XggdcBuE1otUpbA0MvFcdmcoc=")
+ORIGIN("listrak.com", "WKsBd0PdXhhkstYrFMKmVwf7DWpxsfLqRuloNTN3po4=")
+ORIGIN("outlook.com", "lHK41tv1djlJ4bMJ5dsVV+3VF4ts3uTDPUMDC43aTH4=")
+ORIGIN("google.com.mx", "mp+7jW8fCPkmMBXnWThyDmWDjKuEKCMUhb+l/FK8hlI=")
+ORIGIN("giantrealm.com", "2zRwu8vXxNs6wqrUsQLQq6PySn8xIOh+WrTybZgzbyE=")
+ORIGIN("comscore.com", "QzNSOPPrx5QDdPR/TOXRxk4o/0xSl6ITJRVbLOTzP5w=")
+ORIGIN("reson8.com", "4MsqsZZHxEiREFxa6yHNe5p/4QUXPmTkDay5zBIdH+Q=")
+ORIGIN("teracent.com", "VKZrSciAO5SQJqPMc4w/IFbt6V0c8dXdIS8PTVdY4rg=")
+ORIGIN("webhosting.yahoo.com", "e2KqMowGTAU1FrEdhsTsYVCAN0jTBs8TeAx3OQ9LAwo=")
+ORIGIN("vtrenz.net", "/m1WTKhyv1O0spA5d5leSUejOVmvUUtQf8lJLtm8Enc=")
+ORIGIN("sitescout.com", "RmtZAHK1IxnT1r7GF/FVJCYNIgvKd6zfSMZAeGlQZnc=")
+ORIGIN("ismatlab.com", "T2nfD3TX691ohE35O2qgIcWHUKyS4SEvpg63T2F0XJ4=")
+ORIGIN("caraytech.com.ar", "HZHSKqnq3PoINXnMhChHecRHLYCe5oAo1n7WBFUjp8I=")
+ORIGIN("moceanmobile.com", "A5woKPh+7ntb9nnO9Y4vxVmn/bAgzxFGjpudN/xXFHU=")
+ORIGIN("atinternet.com", "PsZouLzxqu1S0uyP/7CeckPZPPOs3WCrzo7+0j9IfSA=")
+ORIGIN("healthvault.com", "BxwKg9uXYVFk1umhzMh2D35cYLD4OIhig4T67hU6zKo=")
+ORIGIN("sensis.com.au", "pIUJU87sLv5xmrPhP6q8VeMhdPG51kbF6kctO246UQk=")
+ORIGIN("script.google.com", "xWsfr1aQ8TluuIuPrxRy4UV2o5eqlSPZ+NfKWeHcgkc=")
+ORIGIN("ze-fir.com", "YML1TMqiIlIQpTddycueE3SkQ08D3Wu9kTNWjC8iXUM=")
+ORIGIN("moolahmedia.com", "5VC2ZEN/joyP0n0n0BfKs+jc2GN5m2MvPw5ko26g7KU=")
+ORIGIN("feedburner.google.com", "vDUDMGkwkwTEqC5WB3rzaup0AK4tqNqECcOP1k0OK5A=")
+ORIGIN("adshuffle.com", "3Tr6TsxUbX8dHZ4WKWbX4CXl9k+vIlrYomVTYjYxfmI=")
+ORIGIN("gigcount.com", "cJ7qbI5IP0yLvn4ge+0imSCuEgGKRbqMbsTCjpYb9uc=")
+ORIGIN("adverticum.com", "jWuQC0k7bIOLxgM8mYycraGWYwg3HPnTrYJ8c+ptDAQ=")
+ORIGIN("madisonlogic.com", "Ri7LE+1T+iS26LbqmYrwpOg9A1j+XZHF7PsvvVhSPTQ=")
+ORIGIN("mobvision.com", "3saxMfd9HefZFbc5V91gMfrcrJTFhPIdLKnWj3ymeas=")
+ORIGIN("liverail.com", "//Qy2j7Jwzc/5Pz3R3QVAaepFCyFSqo8mXqBTF8rPj4=")
+ORIGIN("komli.com", "R2IrNUXWiZKyKtk5nzLAEO1ilbfRH7+UtOwoc6J+WMo=")
+ORIGIN("google.co.il", "Ftw0Pb6fsV4pTcMwM8UIw50EXhDw+G767tviD9Qd7Sc=")
+ORIGIN("google.com.ec", "rCfhO8ux4aFP3Ih6Ib+2QgxYBjPYu6QXHyepBkCfKQQ=")
+ORIGIN("socialtwist.com", "23djzeMdy0kNOwCE0vwEl86TK+N32kmBMSFbzuC1ddo=")
+ORIGIN("google.com.eg", "TAv5+M9tgeuQ9RkL2gfRtiAH0TAm+Oe19j9eh82uObo=")
+ORIGIN("etineria.com", "LQL+nEqkrSTiKg1wPmKSSOQcZys9+3TdfX4SYu3EyJ8=")
+ORIGIN("adengage.com", "50ECv2gmZ3tzApGp0WNfR3/Nc3SM5seemUaghJ4q5VA=")
+ORIGIN("google.com.et", "4s+N3uXFwoEoiaYAX4XxKQcFlex3JVQ+ZJ1jH1R9fZQ=")
+ORIGIN("adtech.com", "C1SqueVBZcjpwemYFwQSilNBxICk7GtzxqG93CXXk0k=")
+ORIGIN("rlcdn.com", "Bm/FeLM535Kb9STQLEnI5bHkzFe8Ag+3kgVI8CUSdc4=")
+ORIGIN("poool.fr", "9IrmY943dDkULFKwNxYpxdAI7H8IhTf6wSTwcSzT4hY=")
+ORIGIN("yandex.ru/portal/set/any", "AaKqS0qYo00XhtTSqlzDX9GE6m14J9LRzczXvs+NAww=")
+ORIGIN("recaptcha.net", "11gzf+26hSNqW+LV+Izbsnbpvxfyaer53CppNLpGUmI=")
+ORIGIN("game-advertising-online.com", "9Kzsd6CSJatgH0uJXrQ3xkIdLBJ2g/TaxhIVMwYbkKA=")
+ORIGIN("levexis.com", "Lsdv+LIa+QImFWTFM5KDT93UquTwL9KFLLisRK7Y1YQ=")
+ORIGIN("address.yahoo.com", "xHTMwNke92FedHAVmeZR6YUFVi+fnVKWdSLD59g+kqA=")
+ORIGIN("groups.yahoo.com", "I7LTRGZfOq2f9V6hjm/XFnx4EuuhebwJy3N+lDBeXgk=")
+ORIGIN("ucoz.ae", "XRqCnvaWLkVMGowra7rvJhviNh1pFkwUf5/fJGNuVVI=")
+ORIGIN("brilig.com", "8XJqA+nt6g1W7ywZDMPJylHeVeV8abJ5FqZHzBbPvCM=")
+ORIGIN("epsilon.com", "Eql9xUB9l904/ZAypi5c9OcmbK9W/JgH6fPVUBmPuwE=")
+ORIGIN("displaymarketplace.com", "qtV1mvMVjiihPUGLMUmrFhUl0qQJnnEAu1tKqCxbXPI=")
+ORIGIN("collarity.com", "SGS7eIbXhvBZRqPlO1xjTrCC1JaqNmpWYRuNDKKyIls=")
+ORIGIN("ohanaqb.com", "PyNCWhYnrpfW5nZMJ4+o74snB+2RNTZ0vOzD80vTG/s=")
+ORIGIN("magnetic.com", "AsYccxfK3rNSIUM+uFewJIXGgAHju8LevaWDuXb3vNg=")
+ORIGIN("bizographics.com", "8xr49DGl6c9oF7GWIoQRPAyF0io0eHtb8Xzn/FeOSCo=")
+ORIGIN("match.com", "gDmwCjAn+8lCyYVVzgWgfDNs4AI7je+dNLaZgHEcqnU=")
+ORIGIN("advertising.com", "Ia9UpUKFDWP3A2vcIBlUsQjg8Ae1V19DYitUzo2b8To=")
+ORIGIN("newtention.de", "aJQ8IWTWXVJOLAB0XZm+IlN4bZicJ2yRE+EKG+30kSM=")
+ORIGIN("adthink.com", "/fG65RvzRmLKDjELVR0NRqhbhLl4feZLk5936lNvcBY=")
+ORIGIN("vidible.tv", "x5/2kt46fhJHrovZhZGUNzc00UCAVhloziwkCG+Bihw=")
+ORIGIN("msads.net", "NUljKE7BPQdY4dNy6eJqzpu7usRwwHw7M8UO2kIeB8I=")
+ORIGIN("adroll.com", "cEuM2qqp4xTrKLo06MfuifUAbOSgd3hBLgXp4ItsANU=")
+ORIGIN("talkgadget.google.com", "3VYNNfzenpCbW5n4iTs8mkS8Ke75y1IQnWlfo9oqyC0=")
+ORIGIN("visualdna-stats.com", "wepRyC9srnVxtDYv1rsS3xTVCJi907qEXKTvrrbhogg=")
+ORIGIN("newrelic.com", "ShwHUYFNKihKSa5FtzNPEgPaTDism8EL7JT+O95AcqU=")
+ORIGIN("thismoment.com", "7mAAG/1hHg05J5MXN0atecBTBPx8fuNVxjEFbcLoFno=")
+ORIGIN("acuityads.com", "Hj5v8n3qLepEg+e+GS07ab7DONpP7lnbkxFWZPA9Y28=")
+ORIGIN("getiton.com", "79Np0/teEMvVrlP/w1PfHKci/e2ZVnurC+214v20EkY=")
+ORIGIN("adify.com", "WL5kzFRmmcWr944DgtVGHJA2Gviyl9MQh04tQYY3+zU=")
+ORIGIN("google.cm", "sxuQYg9orI/ijnwYUu9HXzYWab+n2VeTlZtZD44tfOk=")
+ORIGIN("twyn.com", "ilOVshUHCcfeCvwpmZHq0jXG7VQKo/FOdSBwzPum3lE=")
+ORIGIN("itsoneiota.com", "z8n7rZFaPOMPa54SOcSnE4XEaUs5/Pps5POZHYo+/C0=")
+ORIGIN("games.com", "5PI4PppyuED2TB8nRYOuctrh+Z8j9vGae+U6YYg58Dw=")
+ORIGIN("admarketplace.com", "DO/Ok/qhxwymKC0vTTyPlZxxPApCAaL7y0EsCjn791Q=")
+ORIGIN("videoegg.com", "Etz/ESoKy8lqFzjItOKrbQ1aCZXH2Qr5h+cK0EHHero=")
+ORIGIN("google.com.kh", "d6fuSU6sKTMSwfenlYW4sSY/zoA3AYvu+xo08datVO0=")
+ORIGIN("loopfuse.net", "txF3ukzddgM9tjA+HaMB+AOxcobw1MANfqe+Mj3Zdws=")
+ORIGIN("google.cn", "IoLfwbF8wQzSIDD2Zvvqpm5L3rAtpMarmlSbljDPTDA=")
+ORIGIN("adconion.com", "RYdqLhqxCSkWIFS5wx9TGDxSRlCnB1rwQIr5bCSU71M=")
+ORIGIN("onaudience.com", "6JoaqA3+7A7JvhiZ9JDxIc0xe7wywGEy9xrGFITIs0I=")
+ORIGIN("bluestreak.com", "QDQ1gUgBjj4CtRG/Ign5byffdtoDbvPq5uxNgaz6jN0=")
+ORIGIN("tattomedia.com", "yOLmNRXdqyBJKjkxgOKStPdgpqV02oGqocY63mtGvpU=")
+ORIGIN("fizzbuzzmedia.com", "k+z0EIrEi3FGFSmALOAKm7eK67qYu2YclyySRqWZ3Ag=")
+ORIGIN("hitsprocessor.com", "icxSh5FSj9QKqvu0WuIUg8gZicjSiQxr03Whgh6JT/I=")
+ORIGIN("google.com.kw", "A4bhr3ZyINQ96DBODYXICTG+ogW83xHBckN/NF9bKXw=")
+ORIGIN("pages05.net", "gV9gb5wuLJ+JheRYBY1Uz6CZhadM0cUoC+G4QAU72zw=")
+ORIGIN("atemda.com", "mwX7kDFYGZbyLM+zSCUfhdyyZDuunc+yi09hFAfcVoI=")
+ORIGIN("statistics.ro", "Ii1SErCCvrTERUdCb275X1gbvBtZ0Nh0qddzeGFV3rk=")
+ORIGIN("mecglobal.com", "BiREpY6huJtVtopoHJcUKexZtp4WDlFAytnVM1c/LRA=")
+ORIGIN("statisfy.net", "c1KOXVvvVmmwRTfJ7hPRsmaV4OdTGBGWTXU1PZ/XLRA=")
+ORIGIN("dgit.com", "+XddheJHUzZWgKL4oiHO5DLhg3400GL3HrYilzUUtT0=")
+ORIGIN("vertster.com", "n2Ow6goFxmaUId50MyIc/XSC2HQr0yc5ak9NPwLH/Nw=")
+ORIGIN("adnetik.com", "8ACXMYJn3b9Vuaj4iFnqsQYSZz7x2H/5c1k1Fv539ZU=")
+ORIGIN("centraliprom.com", "AQ+6dWH27tNgjv7SLbAjLBkue5b/Ec5EFuRSSY3W1rc=")
+ORIGIN("payhit.com", "nO+kiPhoiBGOmbQy+U5pn94s1ymMaWVcvikO3u45Q7U=")
+ORIGIN("traveladnetwork.com", "Pgwj806yX6wAFDv/ffWBJiAzIZntvAPRsNYjoNTXMsk=")
+ORIGIN("crowdscience.com", "nLVwgh9ntCblDoVeF6G9o0dgHF0eTc43PBDPbbVTEE8=")
+ORIGIN("frogsex.com", "Ypc9kvpw9bGz481+mxUONafK2oYFwhBoLiLi1PkUFw4=")
+ORIGIN("wibiya.com", "0D3UjZcxYxtkJvrMK8k+FQiHQKhTKeDR+0x4bbwFGcM=")
+ORIGIN("lkqd.com", "i83wIU1cph74qQIXOYAxQosTHNOSSap6LcK6ES/Whws=")
+ORIGIN("search.yahoo.com", "bJsgJLiLaShyCkb5RvEefZWctQIlGRBxwYDdTwlJnQ4=")
+ORIGIN("gssprt.jp", "kTB+xbjBP/VNtE5m9PITLCrJdkCj0oyFEeo9GoszxP4=")
+ORIGIN("cmadseu.com", "079cwiPM5k8zx0BYWD9pKqI3uHr+fcRxiAOnFK5fboo=")
+ORIGIN("advertisespace.com", "XEakfck3Ikxj4cKk0vV1u5qgp2qX6IZdQFywC62BEC4=")
+ORIGIN("advertserve.com", "t12SwusR2YeOfm70kaEdUbCFihrRHuNYDJHqDiu37C4=")
+ORIGIN("sonobi.com", "4qJpNzDeoaFE0agTIebmZiwaGp7RwGSvRajfqD203jA=")
+ORIGIN("emjcd.com", "usLPmYl5H+pctVqm2PDWERKH1AX1RG9G8Ika/oQzGWc=")
+ORIGIN("fiksu.com", "DBYvXSEwIZyj/m5KrTXX+KkCqcKjQNpr2wm5gCGeTE4=")
+ORIGIN("faithadnet.com", "uy4yJkVgMFanjlsOgntwLVDF4XNMGQTxzJXLYsNaznc=")
+ORIGIN("staticflickr.com", "wVwCdqpEdkQbYr/NdKn/ItkhALOs9l5IxgrvAiwpM7I=")
+ORIGIN("gemius.pl", "Zyt8wXbYXgNo8ghX3AUBSKKhandOIgncb2h+i+vdjao=")
+ORIGIN("google.st", "eaeyk/iXQiGdyGJqRMV96RHIYfbUOIeKoQLVWTeVpLE=")
+ORIGIN("tapad.com", "EONnrXrv9FmjgRMuxLh/TV+AusP9X2DNkpYsIkZ4/50=")
+ORIGIN("everydayhealth.com", "Pkqz0gOFXntdamFdX3KmlQDbLAPMWR4r6W2hB8qKW3Y=")
+ORIGIN("google.sk", "ktU1/HLl/iw6kdQWwx4tUmP1dCOxQErtO3AnEiutvYo=")
+ORIGIN("google.sm", "x8s9pOWlN11/O9zTJ+EJJ/pW1L6UbzSmKzP5tplm5fE=")
+ORIGIN("adgear.com", "YXmzjpfsJJjSQWHAA+QpBtSsx4AuuE2DdhRpVCnlp/k=")
+ORIGIN("google.so", "MMgBIgrI+yiMgNv6/u4wKg7ogDZzSOWnvIEmkuf5LnU=")
+ORIGIN("fwmrm.net", "LmEPTjrQTmUHPpxrbsM9pxSRH6B2sUrOxpAesA+KWOw=")
+ORIGIN("imiclk.com", "qE8fH5cXSeVVWiVlryAU1OSf0rxfCyz3fsI39HkAkME=")
+ORIGIN("sedotracker.com", "3TOBZcWLPifQXy43xBdxJ0E+DmJrkJw5qEDAtDnS7EU=")
+ORIGIN("google.se", "joAlyfsGu04vp25qGR+T/s7kdEuy1oMplsRPTSM60VU=")
+ORIGIN("amobee.com", "G6wEzDDpGznMhiBzonYs0aX/vBsGJeYdVc31qh1OGwY=")
+ORIGIN("belboon.com", "FBG5008rOv4UaE/bzYMsEOvj290vAl87bnSBQ2zrnrY=")
+ORIGIN("glanceguide.com", "8vMvKS7Rk/tEE/O7/WLhAaR4wybAA351dvWq9uEBiGw=")
+ORIGIN("intermarkets.net", "ULcqxNpg6wkrGwvPgP1A6Nlns1vcCTzVeaUY5CRSgiE=")
+ORIGIN("twimg.com", "5IdosM5ZVh5bwUGlIGHdRVJOdbZsrX1Z3ZLkMHYlvcU=")
+ORIGIN("mypressplus.com", "B+qUWdIFy5xbHGBPVC3g4uiofcNlFLjZ1TJU91GV6JA=")
+ORIGIN("krxd.net", "05CDzuimWISXdtXGGSrebKNYar8f0hUVB60MeBuKEL0=")
+ORIGIN("cbsinteractive.com", "UXCyd2a5TrO8M/jp+IYmon8JXovziCOaamm7tx+bs/4=")
+ORIGIN("trovus.co.uk", "lqvfEFpPfYrjk1Msy6afXPcUtIa5xql32VLnXTWwkP0=")
+ORIGIN("reduxmedia.com", "IrT6IERLCT16dCTwVmQpk6SzN0GMwW51kngbp8GA+Ao=")
+ORIGIN("adsty.com", "KvP+te9pIfRvIS17QIUBNNIC9C5uEgg7QeUVMY6DDVo=")
+ORIGIN("pressflex.com", "doS1ZTe+oqthzFqW4+CWuFMs+8kGNbMgTGWsrzim8gQ=")
+ORIGIN("gestionpub.com", "eK8hbSIvyiAzPeh0Lga5MeMTioiAGxgqGn9/gg4CTNI=")
+ORIGIN("ad2onegroup.com", "R0ScrWEWI9rDk8NEYbq5sVm3rWIdHV32YKDBbVHwc2k=")
+ORIGIN("atdmt.com", "tIkn5plScqduaYTo5dNjrBZE9M7vWvrEsBmOCgWmeic=")
+ORIGIN("gmads.net", "01RDIP/5izjdDFQzUWRV9BhcM/cnC3n92h1xE/LCPAk=")
+ORIGIN("px.dynamicyield.com", "mC3pO1Od8RmEa4kLnUJqmCwN1+KankSTYfn9kLw6Uv4=")
+ORIGIN("po.st", "k3z8Rl/3KLaRntfpSN7uc059dxArC7/56NXtldavOkc=")
+ORIGIN("liveperson.net", "g7avo7QJMUzI+Vo1xytOBccz8bodzYR14HYtInXvz5E=")
+ORIGIN("adtrgt.com", "pKfVkY6n6YadT0jNE9reCEbqPLsMd0pUnxInXDUAfQM=")
+ORIGIN("statsit.com", "Rg4KTGaZu3Z2p74dlum9X4ZtB4xLHH5yjR8wvotpT0k=")
+ORIGIN("iovation.com", "cQK6eMjNjaxEQqpJwKlLlcUv9cQTL/rAXuEGa6+hxWo=")
+ORIGIN("ninua.com", "RpwQseBCCcwpfO4Euw+WtTYnXyuSS5A7CSUSIcUQxyc=")
+ORIGIN("sensisdata.com.au", "x9j/s7IOLPKmweVPyzztqsPWqpi44HEU1MHl5t5tLB0=")
+ORIGIN("adzmath.com", "xdMRxmpzZJ6vhq4Yhwtk+8rdPm7vReLeoAYSW6NvibM=")
+ORIGIN("attracto.com", "57eoOX2eQ/xC3vIBFgLJfg3P4p11/syP1LlMlKYwA3I=")
+ORIGIN("aidata.me", "ikrvtTrpo7tEiwYjALQAdOhhEuuI4trIq2jLIjMku6k=")
+ORIGIN("appflood.com", "gaqnKNf3MCUQ0jdsL7re540zf/D3pGhRektjQUkOSck=")
+ORIGIN("harrenmedianetwork.com", "F2x9yWZmSN/gxNzW7U7p2qhu5LjkXA4TGMdpHLEToi0=")
+ORIGIN("kruxdigital.com", "yodKDZ1zKf9OC1ddmHWvJgSmIW72RLvro8SiUzjTqxQ=")
+ORIGIN("net-results.com", "7HSPdnO/QNPponaPfYol74Cqstmtz84gt6VFfA9gA8Q=")
+ORIGIN("spectate.com", "oqxcgX9wisE2tTomwEsX19rOMaXUGbeU2j4L7lMCVJM=")
+ORIGIN("fathomseo.com", "EkTxu1LSTeCsPJ4V1oGadWYEHJkpiPVWPc2t1qdttGY=")
+ORIGIN("pixanalytics.com", "zgA2tvxmmKiOqXCYGRvR5m7hJqYgrMAxAV5JJNTTERI=")
+ORIGIN("extensions.ru", "SqlTNfD6+9nM84d4W26bQvpf2IMjvYL+y08zsgaHJt0=")
+ORIGIN("adbull.com", "NVU5e0JzQvrKSB23q2rySnXcf3HraW3lFJyO06T0HR8=")
+ORIGIN("kitd.com", "zcJMoSK3aOgNycC0StWVwFJQEHNGlR0rRaNdrPYiB7Q=")
+ORIGIN("google.co.bw", "IRpOhDT2uSnA0nV1qg64Tj3IfV7NFS8Aq+IbXSj7AfM=")
+ORIGIN("consumable.com", "UDNa9ARyxyujS2BARyJIeZ9677FfODCTOUNbzcHSxgc=")
+ORIGIN("thinglink.com", "fi5KHehsxMpiwj6PTO3uij/vu5IrlHVLEg4oZwJf0/U=")
+ORIGIN("google.am", "f2MPQCTpZ+bcEELx+H0kLMKjRljmDhHZj7LYDj4uZuo=")
+ORIGIN("google.al", "b7O6vD9FiAZDP7gZuruGbCynLBvI7bhNXytQbwh2YEo=")
+ORIGIN("ads.yahoo.com", "XJ8D/vXoe2S/sPBKJWZS2WNFPZQaN6CPZeMi+bvL6Ig=")
+ORIGIN("rfihub.com", "W+YOvtn/0w7ooGYwyq1ISVSSScb+E/miyZH5+3++mwY=")
+ORIGIN("google.ae", "T2H1sBM1XTnbIjxw+C782Fim3+zHolUS1YykIn2UCQg=")
+ORIGIN("google.ad", "iPjSMgKdog4qgTHhG+JobDQ7l4w5ltJdGN57Gkj9DUg=")
+ORIGIN("nextag.com", "IDjn+yqsnFYOu95zXnvLggfTnLOP/XOA9h7MlBd3ZVA=")
+ORIGIN("yandex.com", "szCORmCiVckiW4GphVM1uUKacSjyajeOSNz2akfxSiA=")
+ORIGIN("listrakbi.com", "6Pdg+DeZxeiiEDpwY9IHHu3YtJcE6T9Hy9g+kdsPdyA=")
+ORIGIN("clearsearchmedia.com", "jbZ3Djf23B0H6YcqwK3Za2gwbPhLNkDG5+6D5PT1qeU=")
+ORIGIN("pricegrabber.com", "Xx5ioATAulUPzjg7zeF561yBLL9pt89vermDv2AHDkY=")
+ORIGIN("google.at", "SavxAochojFlJYJD3WmHR6VX4OQiOZdhXMPR8ft1Fes=")
+ORIGIN("google.as", "P8i+KyaMg+o7M5wMBKlpyVnAACVw3eYw6/h7NKu6LJs=")
+ORIGIN("dsnrmg.com", "U9BMaOdtEj4GN/8Pei7O8uAgAZzhzWk1eLQbdCgmLoI=")
+ORIGIN("unanimis.co.uk", "R+lgQnlS4jpBIJn8OHl7HwDYCDZdGglrwzMqCmK+fjc=")
+ORIGIN("web-visor.com", "1NR3rkUTXWA/d4yZvt9s9OHhVP3r4fMJVDKh+nlfGhs=")
+ORIGIN("ic-live.com", "FDa2VtMLG67ulZIRei59Ef8z518iR2ogb4Syx/rbDcg=")
+ORIGIN("lduhtrp.net", "V6wSW1XuoXDCRhvdgy82/UIjvhXDyaja7u6+FR4XbBo=")
+ORIGIN("adtelligence.de", "QaXAgbrM5GmyGgS/hO1BHXqTzfO8jdxCI81U7ONbVkE=")
+ORIGIN("freewheel.tv", "JIdnfkKeiKC+WCdWPvWc/5V9s0d0Y2BMlW/iBaojQ04=")
+ORIGIN("merkleinc.com", "4n0T05ixutV8TMfO0kmDaIyJmh6V41vi48MBUQYBXAM=")
+ORIGIN("intentmedia.net", "a4kNcHudTEZ+f528ZlGOUnuqBR8P9jGawr5uC8eM50I=")
+ORIGIN("web.com", "0h2kS3ffABPcpTGzSgZFxrHSvis8YT27McTjVmx6Fr8=")
+ORIGIN("chitika.net", "Bo3GEY7GJqWxnJ2R1vX15XN+KW+h8UEZCo10BKL16H4=")
+ORIGIN("nexac.com", "p+qsRXoGxfevmLQwPfFOp8t4FJ31XNHsOa3lP0oHxSE=")
+ORIGIN("ucoz.du", "lQSeHjvK17SkFPOlmGhXZyvjDzKWzkgO9Ekuzy0km5M=")
+ORIGIN("pulse360.com", "8lP1JMw2ny1LMFkMBYN0c0K68ZFT1i7E8tlcNAnR7q8=")
+ORIGIN("ioam.de", "5dTCczF3tZSnB0y5TcD2uYBb5W/WMNiUm2tr3G3SCnw=")
+ORIGIN("marktest.pt", "YXZByR2+aktfIdzE9aplvREo/Pahr/QS9yJYsLXgGac=")
+ORIGIN("clickdimensions.com", "4QxcbtBZYg3J18lKpqQoOL8+ruOlnVARkIUkdSq2ie8=")
+ORIGIN("meteorsolutions.com", "54MSRRekMSU0zB+TqgBFQukPU95pcO9S3HDG9jC/Toc=")
+ORIGIN("dep-x.com", "MFO1THhvciGmsBLfQf41cPd2CjA+OZBxs5T+RCI8B7E=")
+ORIGIN("google.com.tw", "h5Vuqig4CJDn2tj8hKSpyPMmCDk1z10rFmjK7QkLXl0=")
+ORIGIN("google.com.tr", "uu3z+4v0XAc+XTwk2AR17BbedZDaOvT9u7bmOW5h7U8=")
+ORIGIN("clixmetrix.com", "Pbv+gdmerySr8ezzm33rWoXWFnH8jCzr4TkUDv3dBBA=")
+ORIGIN("avantlink.com", "YKYnRmLccGp1XwC7edwOLNB7pVeTV2b0E0lziJyWf24=")
+ORIGIN("maxymiser.com", "zaQcAycl1GNUSAkm/E4KZMhbSTmcENml7ZBh2PEFzz8=")
+ORIGIN("nurago.com", "g5Y5JrH9SyvixlHvGodfFsdrdTWV3pjjAEO4Idc5Z0c=")
+ORIGIN("matomymedia.com", "Uc13vKxootBQuG1CZexP49wJtyJQT79txMbTvNbki/U=")
+ORIGIN("google.com.tj", "pt56tTq+U2MBHZeLQNa1XSb5JJB2bAoYw+wXMWIleKU=")
+ORIGIN("earnify.com", "qT8flbVyHTCafoIbQ9L4tPjn4ivLQjeNls4ZPKpGqvw=")
+ORIGIN("mediavoice.com", "gpihV72NIKgT10xDQgrsKAg6wQOIiMDYN7m5a0JSuXg=")
+ORIGIN("bannerconnect.net", "9+kGb/Uq17wTfDkpql/011TxzTEX3Qzqy8WaUQVa2H4=")
+ORIGIN("cmcore.com", "KNbbZtySXyuxLcZXfaV8JkXvI5aUVW+aJP28DO9G0FI=")
+ORIGIN("google.kg", "1alFigTnbme6rrIoOYzcNVV1aEhfOgfeGfoYA6Ho1vM=")
+ORIGIN("virgul.com", "tZeDvwYyyubXNsuxqfIpoxNRaQSWnXc8Ni1AHnc54mw=")
+ORIGIN("prosperent.com", "cIlXR8IrJtCy0KMWHC5vhio0wDVzN/TVXld4EkzAnZI=")
+ORIGIN("urtbk.com", "NNe3N70/GL6pysWXa+DSJb7U3c+10nrmhnkRWZBDzvA=")
+ORIGIN("forbes.com", "EDcy8uGdIPI1pYvReV9cAQbtT1ePunPfa/oDOod/LuY=")
+ORIGIN("google.com.hk", "GPZzfCAPetVklZNAyWxpgOIrbse8YUmk0zYXK9kBE/g=")
+ORIGIN("chitika.com", "KEyR697+9dET8kCWo77/c7TSKYj+8gYa2gXtpFBhur0=")
+ORIGIN("shorte.st", "qz/FOuVvz3kHWuHh0Ekmx+oZlSUN2DHlamYcz155OHY=")
+ORIGIN("postini.com", "8KmSacDChYECwu1cbh8vh0dyiUYbINrHZdMn6PLjZko=")
+ORIGIN("sweeterge.info", "7CFOh2RP5BT6THtUOzizhXxQIIEVsOMENhJezfltxFI=")
+ORIGIN("hitbox.com", "zdewwLzy+7GpKIrGkLoAYtbfmIoxmF43adhq2kxxNlE=")
+ORIGIN("accordantmedia.com", "zshebYbA05f29uHQCsZj0PKpTGxhhgdiHO5Gp6ITVao=")
+ORIGIN("dsnextgen.com", "/Rv6jowRGRgywAb899zS+CClr/ACKjsXAloSJNHYyC0=")
+ORIGIN("coremotives.com", "oZXM27I044/s9MZexIFdKutzYEP1rR58drf65JVqSeg=")
+ORIGIN("epicmarketplace.com", "T/A3arcD+2zMxWLR//RvB25jnyUlagm+Wo6bnb9NvYI=")
+ORIGIN("dada.pro", "wcEln8vAdV8wHW3utT5P7kXYYBfFrFsyvR9ymtq+Utw=")
+ORIGIN("reztrack.com", "DHERmuORF1j6WtZ5ZRE4yyBfRI5KnNeofcELAlyKaC8=")
+ORIGIN("yldmgrimg.net", "YnkdbBDibMP17p7+MuQr4iVyVtpAuMUzlfmNc1zg6jI=")
+ORIGIN("ampxchange.com", "br65l0gdoCWuJfRqk7DtdNGNKiZx6Txgf37i3CNPMZs=")
+ORIGIN("korrelate.com", "TuwtFAby2dX+QFknAG2Tgp46poptD+QF6Ru1p2/s9Ew=")
+ORIGIN("tuaw.com", "fHZ/tz7/reLDhj9+OvYQO14aLS+RgWAxfpSo3F3pisI=")
+ORIGIN("hands.com.br", "T8TqJ6fcczUWeJifc3+wQyJlpZbmEiyuvEX8KS3e9pM=")
+ORIGIN("googleapis.com", "LI3uwhA/1EY58D7mkiAc67Dd8FIwY3QTsI/gpkA9LAE=")
+ORIGIN("visitstreamer.com", "BWGUJSViGhx4S2d3APvkwdit0XPc26iHsBtdXh2kuWI=")
+ORIGIN("dtmpub.com", "b7/qWZL0LL4zm2AEhqmHyzv/QxbG3SkSuBAa2mt+BHo=")
+ORIGIN("fxj.com.au", "EN0HG/6EXl2HlNJLhSkp0gfsFK4TE9tiyOL1pKHQkp4=")
+ORIGIN("wallet.google.com", "jJTEKHaR54lnane4AlbP/8PY87833GM/W2HRAYzKFyA=")
+ORIGIN("affilinet-inside.de", "HkHcCa9rQKrjVyfMFjNe5zaZRABkawIaxXGFQouNC9I=")
+ORIGIN("google.rs", "eKqzCr3ckBugkmstYVOBOD+xqZD1ONKmEnQZ434vATg=")
+ORIGIN("dataxu.com", "0P9y+oYQbUE84BfYap6Y03wM++K8Iz8yfhLN4Oq5FC0=")
+ORIGIN("brand.net", "NFrp/OkM6pJZTy6vJ6RDk7GjYpuKyrwwYXibzl/fSi4=")
+ORIGIN("google.ru", "LwUlbCSh+qlylIA6MqE6XaL0tWtjeZNIkqujsss+sEY=")
+ORIGIN("google.ro", "45B83S1zgSNgohGuV0MmRzzihtHdktQg5aZWV2Ezu5U=")
+ORIGIN("fastclick.net", "O1ru8zidahl21NNvayIdpa7uMVIzEp49ueYOyPxgEeU=")
+ORIGIN("adspeed.com", "Vd4odYMzLLP5UagKdxnHmFdZpTvI/Xsb+Tpg3CQbTjk=")
+ORIGIN("snap.com", "lk40y9Vjo5TdqwA+tlJFYINeTXVeJygFEDA81zfhb5U=")
+ORIGIN("indieclick.com", "D8BRNJknyHMQUL298KffmF+9uFbCAIAjrzUSFS+wxvA=")
+ORIGIN("appsflyer.com", "Qns0O5n3d6fX6JaxnV0qH1VUXezcMYpPNFFQcOD+F70=")
+ORIGIN("hi-media.com", "UaSbBcft7W9EPr5IdFsaO8NPHwmBQnXLr0R8P7HQFP8=")
+ORIGIN("messages.yahoo.com", "1DRHkQvUvJqiJO8Cw61lQ105gJZsusfFg078tbTqB84=")
+ORIGIN("admicro.vn", "7FnpGp8F0Ov/5mPsMFfXVcFaI3QjyJpn/tLDiVkejjY=")
+ORIGIN("greystripe.com", "4VlgS2JDqRupF1LQ3g8xofDX5H0whoI/GSgip9WVd+4=")
+ORIGIN("bannertgt.com", "TbfC4yAec+L/IX+VtDtAXnlYbhyEVSVwxrD0MQh7ApU=")
+ORIGIN("webmessenger.yahoo.com", "ynBf7ZI6tm1ti/4PZTWaS4cpgb5bzBNk4pqsaTda8yM=")
+ORIGIN("bluecava.com", "6l4/gQ6p+pW0qHkIYPuzX+cUxCHcbcQSmNDi75uFzoo=")
+ORIGIN("snoobi.com", "1UxdmXnMlzbqQUuJTNcR1bFYAp9OsoQVDjPbuVsJML8=")
+ORIGIN("teadma.com", "YzmrBlAW8VAcu8oqMuAZIgXNKWspMss4l5Vc+0FV/FE=")
+ORIGIN("burstnet.com", "hbK4Ocb8t1oQ/mwBx0uX5QrxScy+NmgMAtjLu1vQBWI=")
+ORIGIN("etarget.eu", "L5SENaN1J5oApWxgMYTcf8BtvmJimoXyxJKEQBT1qzA=")
+ORIGIN("tns-cs.net", "/O18XSdu8MjP6xHboYnf/HeodYCbkbiWmIfyYIsrEVU=")
+ORIGIN("adrdgt.com", "EHm6wNRlGcVSHn9ChWB5jLhM39rBIjajEj81mayOIlc=")
+ORIGIN("pardot.com", "ic/jLBhs3O0FuiOdIB/A0hrJ9sl5tPHc7Cl+ZHabfuA=")
+ORIGIN("clicktale.net", "TJTRDSlQjHtxKSYdwJJuMrT0eqSror7oK1W5LTeneMI=")
+ORIGIN("omnitagjs.com", "Kvk1IILHJDRw9Tl11A8MgwnQQOjv9jQNbTgfHvSRK0s=")
+ORIGIN("google.bf", "Lkp9HjaqJWePAQ0Nd8Oveghy0teB1uZpy3BkJooSi+w=")
+ORIGIN("pippio.com", "MwP0BCYfHCsWOMNftLJqF4ugmbdwKlFSUhZj2vJ5jqQ=")
+ORIGIN("estat.com", "sd+EzT2FTPrRTKNUYjt5bXtD8atyUNMXrhrA8wOG80E=")
+ORIGIN("digitaltarget.ru", "j+lUTfTq3I4WoyJamG4LhFxB54gvR504LcHRzDMqNfo=")
+ORIGIN("directadvert.ru", "t4/nJnbgcbZSY2ojkHHkiP1pvKDft1HmwLnO/ZRyxMM=")
+ORIGIN("singlefeed.com", "axpiDxixgcOJXBo/Qmee/j7VFHKpmU3OcHcOvitGmUo=")
+ORIGIN("addthisedge.com", "IhrpoRBKTqL/3w6En7WrTFK5MWxajfF32mr/vQGTWCE=")
+ORIGIN("fb.com", "H+Hpj9uHj4KHUYGuI3oAe+brlavVMzhC0SPEEOXXbBQ=")
+ORIGIN("newtentionassets.net", "YVy5lqo+BbIrdFAIPy7HQ1VgpGpp3USKQXAgXfJxkhg=")
+ORIGIN("enquisite.com", "cBTP8JbmrUFU8BhlRfY6e22HUmjsUV2kUv6e1O7LbSA=")
+ORIGIN("yoc-performance.com", "pqjK4dVdLNWgGSsWnK67lWjQWK8ofIF0C5DrScfTEq4=")
+ORIGIN("netbina.com", "naVcNktvZ9vQyJR+ZNKcuAsYFwq8I/9I4OB3UEN2Z/M=")
+ORIGIN("chartboost.com", "lbhR34CbrYsG3PBEfisXxc0+d2rCcRJWjuSplOvJnyU=")
+ORIGIN("emediate.biz", "mz/rEqdeQIMiqiWSXZsOHV47eO9QAN50AS630VgoFIU=")
+ORIGIN("primevisibility.com", "WKzxRjCI2LsAmfKR0ZuywlrMtu7BUWAYGrPXNvqNTEE=")
+ORIGIN("breaktime.com.tw", "wfHZynwBe/+ulRo1Dk37f7MkmMl7rS1Mad000Br6dnE=")
+ORIGIN("adworx.at", "i+OjoJQtBhILpakCHOxs7orchTHWq3N1Zrvbh0N9UNg=")
+ORIGIN("httpool.com", "rypHdCgNFqXvjc8S8zPigIe6XVeMx+szVnGbqEtBZ8M=")
+ORIGIN("adiant.com", "2RNFuWIF7lb0iv9d01UYQWfBBlvQc2XEGcc10luMFGA=")
+ORIGIN("cadreon.com", "0x/VyVIgJMA+g1QOBputUcz561rzMkh/K5neC3nWhJA=")
+ORIGIN("pixazza.com", "RFrds7rdEWyLV9yz9OrY2JKBWj0gS/4mT4SupSG9hzw=")
+ORIGIN("res-x.com", "4rJ/vsktgwLvWXJfy2ptLrCaE9fENHBRef3gfltKYrI=")
+ORIGIN("mediawhiz.com", "9Wf7dGvSH42bSycmfKpBkUG7v/jsHUjjMYDPSyPrV9Q=")
+ORIGIN("sparklit.com", "5w/mm5Gu4msN6oNa5bVs65vOmeuE6sLxCQGbB2bRKow=")
+ORIGIN("collserve.com", "nejp41+1NWOTxZsQ2ckav/PdsQX02/aoMv05dwwSkt4=")
+ORIGIN("criteo.com", "y8dBpRDXkbOtt2X6wIS6KLuywoF9wF7l+xSNlMpn8vc=")
+ORIGIN("martiniadnetwork.com", "VisBYvaxtDVb2e4WojUr13NVrGO8x4rFvs61nnyoMMQ=")
+ORIGIN("weborama.com", "D2ckaZUTT9nq8cK1I8kbNLeHX4tLUbdSTV3oLAVvEdI=")
+ORIGIN("adtegrity.com", "1luEwM6IzPStIOTAWy2eM1tLBxRITK4xoJwUPEkvUmM=")
+ORIGIN("ubertags.com", "Ul7rfGzAwYPa4b/zuxu3vMcVFc1UYeTzuo3qEc/1T74=")
+ORIGIN("rmbn.ru", "zgwLE4uaOrlX9tqFebVgjJEfOTcRk/2u42sV0LbYCiI=")
+ORIGIN("login.yahoo.com", "AAcwvd3YNublF6sjEQa4uk+8mtPR1dZSZ8Ft7ajxgC8=")
+ORIGIN("undertone.com", "aghoy/szgP2hMSrlyA4kKFaqPolvHF8mCcuuJu3LFDc=")
+ORIGIN("monetate.com", "i4E+zGa0fzEOHgrAYSZCKD4QW5YeHdiabJ/dvso5soQ=")
+ORIGIN("adsco.re", "FQFNYIyaA29JnsBZL3JSE3se2PUh7fE/58BTqV8cAfs=")
+ORIGIN("unicast.com", "oeQrDhaRxITpkDznxfZwvzK2oLmEIvy/JMUvbC5svSw=")
+ORIGIN("pipes.yahoo.com", "QOas0bxWGJzLXdHfYmklVUm4TK6BHaQMOjWnl+rFylo=")
+ORIGIN("4u.pl", "B8ejtJS3200XjNlYelpzoYPA4y+ql1fbwBk2WF4A6bQ=")
+ORIGIN("visistat.com", "Iwlk6HOxB2ja73OL3jnAzEB5IPYsGs4oiZdniDBWFPA=")
+ORIGIN("list.ru", "aGHkoyvwSHJ3qz3U0qefyc0qyxR3yeUUr/IQZzoMvo0=")
+ORIGIN("apis.google.com", "E7joNep+naSYgbzXDiaW7EFtSIoNn27aEAz838SYkYE=")
+ORIGIN("admost.com", "AgxtUuvryxPpngJE4/xxRCOktrh3ohna2mYAT6M/m4I=")
+ORIGIN("infernotions.com", "frEsyLPOJdRIejfsfIS7skvRiWa+vi9In+dFdbbWI1s=")
+ORIGIN("switchconcepts.co.uk", "JOrpwxdAUxT0eiGHU5sZQMYrN7L8kKEB9mYhluO1eh8=")
+ORIGIN("adprs.net", "5h7cuLgWsI7bzDQ8YQq1W0a+fupKzwfWUoYFPkF6hc4=")
+ORIGIN("mobilestorm.com", "c74p3b2Exa+JpCwKah684CMP0biR1YmTVsOeltVfqKM=")
+ORIGIN("zanox-affiliate.de", "nWyLPjsamiA9oJWglDZW95thbtp0KAUN1ccRqjN6r9o=")
+ORIGIN("yumenetworks.com", "2qdNbBfVg/FIZ40B+wu5uuamr0HdHC5pZnbrkK34hy0=")
+ORIGIN("salesforceliveagent.com", "5M3TdZ4YoK5WETCtuHywf7pTZrkSvYgLQdjt/N9rzI0=")
+ORIGIN("adserverplus.com", "lQTR3oqHWPPw0wm0NF9go69zYbRPCpXtO2l1C0AVHfM=")
+ORIGIN("agreensdistra.info", "6jIXgOF8I/+UHMCEO9ZGtwUlCBalSFIpdAiVp1FoXco=")
+ORIGIN("merchantadvantage.com", "Eh9izYGkqpVILTb88CLX2jJNX0rZrbvnrzbM0UgunEk=")
+ORIGIN("financialcontent.com", "c2dtuO5j24CIuKnvcTi9KPBa+K2vlW6gyl62IJpGNK0=")
+ORIGIN("code.google.com", "7KGLEgiPrCIWgpFJEUAzHAsYskLyLyBTRSLpWQPLf5o=")
+ORIGIN("adjuggler.net", "6fwGaL7u8i4KeGO2sqT4xoee6BJwWsrhYI3GBVhBkFM=")
+ORIGIN("rambler.ru", "OQumm40Vu20ipYToWWzzi2DEkfLRHgSBmUK2kg6Zrvg=")
+ORIGIN("mxptint.net", "w37YAI4a/P5OMOWWCdJQ/TQHs595HUUS8TAPDZ4eF08=")
+ORIGIN("opolen.com.br", "o8GHAv8D69Orqt0CC1Z5a1ZSRD1CRYoENLVI1XLnjl8=")
+ORIGIN("segment.io", "hUWQ4a+xVg/U4r8tPDSCEXvCP3m/eHQ6gVLdXp2w35A=")
+ORIGIN("sapient.com", "J/dBlZXUV8vUWa1NECcKG+kME/pTFrAU2j1eOwiak2M=")
+ORIGIN("freedom.com", "hk4MYBJMul023W/yfDM4d4DguyJBuigVQs2GxeuQbMM=")
+ORIGIN("adnetinteractive.com", "mSscpBJYZ6EH34di5CN1jTyo2zmBskWNWSxAEh+KFNA=")
+ORIGIN("sa-as.com", "zGpCteLGWJ4dE+9qT4yXM66Woh7fAkZyTbFiR9WjRIU=")
+ORIGIN("google.com.uy", "Riz72KTW5FeAnAcuhVSwqq7ysug0HIrDzHz41jg92QY=")
+ORIGIN("spongecell.com", "Vyi3Pkp6B/yx83khK9qgW6pbA1lg544VHKK9b/kBtG0=")
+ORIGIN("google.com.ua", "0M1UUWOkI7/7ZDGsdSTbS3Eeqb3YsHfkUWxcSjZgH70=")
+ORIGIN("jinkads.com", "gVdR/PY69jpX3pm1fvNZta57wWD8KZK4hIqfQnvs9p4=")
+ORIGIN("navdmp.com", "9eB0RvB5qL+IK8Op4bMF8wR60G2/ooP+620AMeeTixw=")
+ORIGIN("vcmedia.vn", "Y6cAyU+rmTXKwH7R2Zl+psO94Y0SClV41HBDFVeBFJ8=")
+ORIGIN("friends2follow.com", "hrQNbevAY6ApYAOkeLDxEt+v7pYxCGvAKCvsdvueC4c=")
+ORIGIN("thesearchagency.net", "NOPH/e+lqYmqYihwyGq9l58+DZ+clrBSPj86WNhwezU=")
+ORIGIN("zendesk.com", "Nw861yuwxHKmVmLMIrZIiESc6/oZUHSIZOyLsAb7Af4=")
+ORIGIN("google.co.in", "hdD+adBPmkdn188h4mBJespGBFbJQtijiw5pXruB0yI=")
+ORIGIN("adbrain.com", "vE2Q/1IKuVBLkBb1xT9YL/S9Yx1Pl7fa2/473Yosyy0=")
+ORIGIN("liftdna.com", "yguMgt8Z+XCwmWSVB+2VGzI/tjeMIxono0d0YkOY/Qo=")
+ORIGIN("expo-max.com", "qDD6YbhqqlhW+v84wQXHx80T9BppvDe8wcE40HOfImo=")
+ORIGIN("ppjol.net", "wLRH6n8HEjmiewQ8T4ZY+aMl2sL5NCTb0Xe917VuVuQ=")
+ORIGIN("adparlor.com", "Acz7VYYYqOGtllj9wDOpfQiRKiyhPtedvGOoIjlvGjw=")
+ORIGIN("scanalert.com", "gdkL4lsOTI/+RhcOqmzpW4rpU0DBH5r4VSQR+KxhqME=")
+ORIGIN("vizu.com", "uZEoVo/K/ybrNmArvTomMu6DxlfY0C58WayxpN2LTa8=")
+ORIGIN("datvantage.com", "u8z4bFzxh1Dckxg1n7z1xYdAOLU4amNV97bWIWN6Mj8=")
+ORIGIN("hp.com", "P6RNZI4qik6ho88ACpeGiBXxmDQCQcECpzd3nq8dIWY=")
+ORIGIN("clickyab.com", "0ahzdK/xuf7oFGPNqxwOvzrdq5gjXngKu9rq6cF7Uaw=")
+ORIGIN("thummit.com", "6A+nrofDJfMEYE0srUneOxBzRj330m5sZShwwpXZcy8=")
+ORIGIN("mediaiprom.com", "hSias/zwXnXznXOhv4bih3H+NgP7SUzBkfU2mbxRpZk=")
+ORIGIN("zedo.com", "mcVr9L66mBqLkM+EmeYfVbsyDKZ8/Rcgz8mpnYt6eJk=")
+ORIGIN("lotame.com", "mn693pD1UgKPjF7jVH9NSNzf7wXJtA+JJlYvHolc348=")
+ORIGIN("cpvfeed.com", "aPwKH0W8dLN2CZKFSQiGYWbB34artt6JScwMFgPLO6Q=")
+ORIGIN("disqus.com", "LvpoRjBBWnNZTuI0YEuLDrz0wGk6UQUidJWr4a2XCZs=")
+ORIGIN("adultfriendfinder.com", "8E5MfrKJIYsfJnsR7gr78MeB7PEa2vg9A5Rt2XIU+EI=")
+ORIGIN("brandscreen.com", "DJAfGmVim9Kkn83CyWS3NzxyT4HKs16U0Sc/HEAje6Y=")
+ORIGIN("radiusmarketing.com", "UCkxHPzZFtvjyNuJDvQMFrQOdIxSuVPRlTsDQJ4OAVw=")
+ORIGIN("inmobi.com", "fMeUk5Jf6kXlVyEGLL8FKaIveDsftlHnFWsDteNkJVg=")
+ORIGIN("fout.jp", "/ewU91cdx2O/vLGLxio3NbC1dEd/veJ1Y6s9pi1F+wU=")
+ORIGIN("docs.google.com", "yHZU/TNkHWXVxr80G+N96vv0pjiU8CEsZlfwN2CkRM0=")
+ORIGIN("flashtalking.com", "3Euk9omCSD1WhWa9lxABIwGuCudAUoCSvZMTwGPE48k=")
+ORIGIN("earth.google.com", "HssiCQpV/mQ39XJW9+AgZmGU7oZn1fRw3KxAaF3SHNw=")
+ORIGIN("omtrdc.net", "ISEqrSWx1NIhqnSgg/PyzNCJCS02SIZmiiEZvqaljrg=")
+ORIGIN("adjuggler.com", "CF0TozjbaP7mdmQOrZzCaAU1JaTtIGm/KfAFrhP41CQ=")
+ORIGIN("audienceiq.com", "/CCvQJp3XixfYKB9crV+PppPmrvnmvH/5PB1PhWmj0w=")
+ORIGIN("seevast.com", "Dw3UnaJdSrpGh3LGjWmU6YKnaMy2nxrsg2KPfja6bOo=")
+ORIGIN("mypagerank.net", "u9ypLhZiQA1lmQ6LSV0PHvtoMkyqjUWp2KzCVyJ4b2s=")
+ORIGIN("amazon.co.uk", "38+UAVUFazxjo+3kQ6jeoRLmykDTimzck9uwmrAHVQU=")
+ORIGIN("alexa.com", "ie+T3T0L4P5I9JrXgIOVV98Ymo2MbQNJzzBEews4XXU=")
+ORIGIN("oversee.net", "dPIRqYrwW6XLLdZ0MYqFbSBr2iiDVPNMiHbJiq0lIWk=")
+ORIGIN("chartbeat.com", "j2NMBBHry3P4brbZWdwcEIMvsHI7uB/Y75vyh5jHD6g=")
+ORIGIN("opinionbar.com", "wFY4EibU9YPhb+AGNS9ReZ2Dajm7ExX0it4T6A6ml+Q=")
+ORIGIN("pntrac.com", "qDQ2aF2sIJuGq0iOFwxPVw4WNw8KdydCHtijSzNzKlI=")
+ORIGIN("punchtab.com", "/uZUwhm+IqA5oLQ/BpRg2f9zsmTkU0abON1QOMJyGQY=")
+ORIGIN("heias.com", "BQ0wXEMhvJypZX2q09v2oOHDsNIZD58QyZEgYht7UhQ=")
+ORIGIN("trackset.com", "Yntnx6JmDFVlK2iYXf4IacsE2QxmdMJGQPfzAnf5WF4=")
+ORIGIN("gwallet.com", "tuKRKO+d3lIAd0rObvqUwZFVJ06oM2cKE7f6Fw5W454=")
+ORIGIN("zango.com", "Kl2a6IWkg6b/X+A2/3xX6m9a460PW0pC5XrBe+UzyiM=")
+ORIGIN("i-behavior.com", "NEIexF9T2lCx67266812o2+N59mHnED8VOTfQjGZQtw=")
+ORIGIN("shareasale.com", "Ft9u1frJDLzLqiYrUo//od0kL5Y9Ox8PFvI1AaFFTJU=")
+ORIGIN("domdex.com", "ucaehEL2lqi7E+CMw6ZMgN4bkzi0Sop5AyFkxnHUB+E=")
+ORIGIN("finance.yahoo.com", "SFZo32AGJSYHrjym0b2bo5p0pf7mIv1uvDD5vx2OmwQ=")
+ORIGIN("adabra.com", "PFRqc2VINIAmlQA03hfeyFcdrZbfvz0pF3JrJAijBnc=")
+ORIGIN("media-servers.net", "hm48WYurM06k8uF7+J0xln/+gyftOXxuUZ4qbRrL4Tg=")
+ORIGIN("anadcoads.com", "Is+frX+UQ4nVFrgRV6+dkzMQZMgr1rBqZ42OV/vhBy4=")
+ORIGIN("gmodules.com", "TLRYpSvs8Hr7A8t7zBr8yYNOkuO3zY8Zf9w9hKRdPWI=")
+ORIGIN("thewheelof.com", "izGpWBCERF4PJpF9pUJf7nfjRbQNKFbEW/F6nGqJ+a8=")
+ORIGIN("maxbounty.com", "WoNUZ7NVzPGd/Swtu+8srzOSEUZf4pLnWF9n0QXcv5g=")
+ORIGIN("othersonline.com", "UPv3GkXoyRKPKCFoFmPbc2V462ecEz9YlS320PY2uCc=")
+ORIGIN("swoop.com", "lleDvPk6+c0m9Py8qCDgnfReeHNY0GTW1TbH4A90+Ms=")
+ORIGIN("stumble-upon.com", "v90yIijsw+5b04v3rf82E7BilxKO1t9+2UkNIlzw/c8=")
+ORIGIN("wanmo.com", "Y0rPbGbZk+r3XRTML7Lt+mhYJ+VIL6aWaZ4tGWD6s0g=")
+ORIGIN("vendemore.com", "gSWGOLXT96mOiwND4CmBxCwCcQYY42wLqKYb/p9uIxc=")
+ORIGIN("aivalabs.com", "8iBD7+eqxr+7dgCQDhUjuNSZGBgatCP3X6jGt6CC7wA=")
+ORIGIN("genesismediaus.com", "eHo9/enY49PzTjBGUC2Dq+qNWKd5OW+I1h7z79OWc2g=")
+ORIGIN("adnext.fr", "djLaKej5iQkhwraBVp7G/svHT2gWGV/BBU4hl3R+DOs=")
+ORIGIN("smtad.net", "QLG/MiiiVc/pFvmCkWiLS4IifUIhFsAqULluNWbb3lc=")
+ORIGIN("fuelx.com", "ogS0fiRyQejFY37AahtlIHtTaQ88he7tQItxeeLdDJs=")
+ORIGIN("ads-twitter.com", "yAiTblWPg7GDjndxNVeOaoeCq3/3Sbp5Y5PEQnVEsvY=")
+ORIGIN("meebocdn.net", "B1ERy8s67kFEvcRjJ1OU4hUvZl2JPoRYp9O942ZFHT8=")
+ORIGIN("addynamix.com", "EBQ93OPWXMOJgCjpx0TyICbLymcTodPLFCm774lEaPk=")
+ORIGIN("lockerz.com", "U1nPf3fsWtncJ56UAJiIgsymARQ3HdCr9FIlP+F4bLI=")
+ORIGIN("raasnet.com", "bRVst6UArdcuPt6QJWPkqhEK4AZGnHLoBADI+XCGB7k=")
+ORIGIN("aquantive.com", "3DUOicBt+/Oyw/NOJCwITrFe5CcD0F7K6ihArfom+SM=")
+ORIGIN("google.com.jm", "V027h/I6+W78/QEbScoMobFCQGzr9yKB8uW2j8pCw8E=")
+ORIGIN("connextra.com", "PNhwHtLjRJWQeVS40s67aW4fb1/TXzAcRM0LvCH4pzk=")
+ORIGIN("rmmonline.com", "30ZDT59BPMMsGiAdeKrHBQKTKwbXIJv/wd+pj2eqt6A=")
+ORIGIN("revtrax.com", "NnKlSwlp+400jKPCwHq5onoA7YxvqQ5bYApqQGNvWY8=")
+ORIGIN("hotjar.com", "6HqhIt6tbtcjP2GKEFn9tFFcGo0fKXIX/TZU+5mbjxQ=")
+ORIGIN("rubiconproject.com", "AlN4GrZXB/iKh/6sduQ+N3awneU+8u5nnnuT8xqm2u0=")
+ORIGIN("dmtracker.com", "rbppVK81tQuDT6A1rUsoMHUEBnPaks73T4VANkR/Uas=")
+ORIGIN("smartclip.com", "XGPutSAvcfoaATq51uXiJjlHp0LyB9DT67ookbbozPc=")
+ORIGIN("affiliatetracking.com", "UFXU+rRbQy9limFLbsfqYX/Gu10eBGH76/IIoh4C9aA=")
+ORIGIN("postrelease.com", "zzXH/ypflH6ANBagnNQIIboqISoZ02coMnps9bcLJ1Y=")
+ORIGIN("yp.com", "D+gPAn59DPDdjg6iH2MTdjgpggOvuCh9jthyR7Pu/gM=")
+ORIGIN("biz.yahoo.com", "NCnD2u+oUONKd29pjUObOK4BW+R9KjFPVvJNc5HuUvk=")
+ORIGIN("kissmyads.com", "qQCCIk1GtkWEMVrACn0X5zYRgantEvQOpv0Njf+1Oyc=")
+ORIGIN("skimlinks.com", "uon1qoiH5tWSnP2+iYiQ/MhFCKpZPshOFs76uzXzqOg=")
+ORIGIN("iegallery.com", "xYskBMn5CwXKhPaDNXXJpXDDCanRuUHhLKywwW+1BXs=")
+ORIGIN("adform.net", "rqYZeo23HHblETDnYvi9csSLZU8TBd6FHaoPGaDXzLE=")
+ORIGIN("adtech.de", "isqUS2myN5Dnb5H4iFOeBWTwiCFs3gdcjx4LGZOW/s4=")
+ORIGIN("atrinsic.com", "Bp2I1WbK+O01amKvRgPBRIVusieXVMNLCGguZNA2xcI=")
+ORIGIN("turn.com", "P7WRBC923LIAl2eVGdBZD2Mf/b9oX3k1aRY1Vo+mxhk=")
+ORIGIN("hs-analytics.net", "rsyEvcbWEmO2llyoXRGxZli2qbFgypFlWpmcWuysG+Y=")
+ORIGIN("exelate.com", "TOArPndULz3A3K0sprPTjU66iPR5J82ljiolro7GFmQ=")
+ORIGIN("lkqd.net", "L+tGIJhE9TESmKsozQQB/4vLk0eAs+bZ+vGBVKOxwlw=")
+ORIGIN("unica.com", "x/nUJGu5R1zWCh4nR+CecdJyWpOBGWEZffDdl064CJo=")
+ORIGIN("xertivemedia.com", "kiwG20RdbjCQ0gQCjJaxln9hrp/i1lDilRN1/ckmggQ=")
+ORIGIN("bluekai.com", "NWsUUr3jFAk5zeCTHDQT3zt0koAf0GHltt7M3mc/nTs=")
+ORIGIN("mochila.com", "MCa4mgoalyzNSV7lqcXJi0XO737UA6NTDref1TpgrCU=")
+ORIGIN("globaltakeoff.net", "CvVk4eMQdmGByBYPwshs4CRYMUqH33vAZeL7xJL1ZXo=")
+ORIGIN("adhaven.com", "xnR7P3zHkIkfLEnZt1K8mToL/2hjFNLDDXUYYL36NSo=")
+ORIGIN("aemedia.com", "AlPlfiwolUYHIm/LZKRM/C1QuudoFMWhCHa1tMNcBNU=")
+ORIGIN("reedge.com", "xcOXDjSlAFX9aEJbY4H22L0huLilTDyKs+QviCNsn6s=")
+ORIGIN("iclive.com", "gaM6xcuJvpu9s4a/AsW2d8LuK7bbzpo8xiATKwAzplo=")
+ORIGIN("nuggad.net", "Ze7GmPxncx4T84VqIaH82TIre2IBCnwf+bK8H21RJnI=")
+ORIGIN("shopping.google.com", "el46f2WxSySYQRcWiQ6no5PLhRR0tRvQEMhrcDoLkPU=")
+ORIGIN("hotwords.es", "wAhODNd3j5QXen65KM+R7kSiFSiO+Y+z7WY59bYfLDQ=")
+ORIGIN("cloudfront.net", "7uMtzzi8OPcJEoqOzpKq+yE3M11c+IG+8mILMUBkVsM=")
+ORIGIN("monitus.net", "FL6ypQkaE5LoaFVoUF46MZIX9a0y1/sRZQuG4fi1a3A=")
+ORIGIN("infolinks.com", "iWM+ItOKMoR7EIsoNeV95ZYLWuFBMEgPaNF+pt67Whc=")
+ORIGIN("guoshipartners.com", "nWNcr0EjUnew46zyxB83IfbF+akxDUvz/dfQgMyPFoQ=")
+ORIGIN("flurry.com", "LwR1dZWGcqLzwbBG4TsJQj8xlbUPlxWLJsPHHwtfhAw=")
+ORIGIN("conduit-banners.com", "eksLAIj1W7/B34GSh5l4Jm2zW9hIGFc6KZOwhcGXQso=")
+ORIGIN("kenshoo.com", "B8atPfzSq0T2xYikxvfFZDc7oUs7WysyEIShNUrjhx0=")
+ORIGIN("choicestream.com", "MiCEdrisWBUAOxEbYfZuCZMMGpeCXgeYZQjKt/mfpJg=")
+ORIGIN("admixer.co.kr", "tJ5OeTFiSfasHiIvJHxf/ivuOTt9oRPp5d8QUUkS9zs=")
+ORIGIN("maxpointinteractive.com", "Ihu70DLsM8+QT85EKdLDwl3xgmKE8zMtkqTCnwNir5s=")
+ORIGIN("nextaction.net", "8vxWg5V76+P7kYrEd1+CPtgzz5+9MZtAAS0BHNzSIQA=")
+ORIGIN("emediate.com", "MFLjLKf+yJyrQQrGlVRGCi874LT9i2CJs8YaZz2Qxt8=")
+ORIGIN("pjtra.com", "vwQyU3HhNiIB6uv0oXsmBF2Ex+Qxu1bIodqfO1EATBE=")
+ORIGIN("jasperlabs.com", "8yjT8igTP4uDRRCZjmRpvqF3bGti71N9XN9dCx9Sjkw=")
+ORIGIN("gsimedia.net", "MHOtniRinVniREppVc91+tHJ+rmd2VMtrYnZQHwbvlE=")
+ORIGIN("adserver.com", "372lhGD3hlCBcXaIgtPcdjixJdrqQU5FS0rvbeaprmA=")
+ORIGIN("yieldmanager.com", "/pcETRcGxDc7g9Qj46+PdYKKZYbWCOB5RwFK6FTSEoM=")
+ORIGIN("hilltopads.net", "Fs2uh9HZGmHmy51DTwtRtOzRF3XdXuXnqdWMJLKJ2S8=")
+ORIGIN("addthiscdn.com", "Q8zmlPd25mJDkE0ORkxRuT8ao0h/tstZ2aMcCF48OTw=")
+ORIGIN("extremetracking.com", "GhVAq/u1bp7b0WoL1QFhYKTFoJCEpyj+SS5ljFf62+I=")
+ORIGIN("kissmetrics.com", "motCCpzEa4NaD4y9WR58gzsUCYQvoxIMBUCNMpB3iQk=")
+ORIGIN("ethicalads.net", "XurLv3MBC5CNDKxt1Dq8IMg6L+58mujorFd6rsNpfd8=")
+ORIGIN("oewa.at", "f0dkIxhexhIZqfArinVz1xrLVDwlRw+OLocQHBZQecg=")
+ORIGIN("beencounter.com", "UqXAURFe2r96R+CDgoBehnVSdK2KqG9TuSKDJwkC9fM=")
+ORIGIN("limelight.com", "6UqJ5crDkVOnj3HtRJBuoNyNbKoKEUvSSoJfFonAHtw=")
+ORIGIN("quintelligence.com", "jYxWE+RKyRaV+Hjv6ZSXglS5IlYUpIlsxBhTx8tajjw=")
+ORIGIN("peer39.net", "IkbfBF6Tk/ag8jZxrLjwXV3iYydxi64+CNQkkXNZeuE=")
+ORIGIN("gorillanation.com", "wHwryHTPn1SELd0TCsz/ZAielV3KDGVuFS/doSLoC6Q=")
+ORIGIN("yandex.ru/clck/counter", "ScDl2FLpjUo9TyVTuN/BssG01oUN4LUivDwpc6+HyqA=")
+ORIGIN("visualwebsiteoptimizer.com", "qaNeytsE/TIsadmGqyg/ys6dHjDXhJ+WnJHqkxZjKQ8=")
+ORIGIN("adgibbon.com", "CAGVXh/TZYHO4P/96TVV6yEkoS+fnXni+WSWUljHi8A=")
+ORIGIN("digbro.com", "Ty2uWauQ3KSITWsYDulAlS6E4jmKzvKEqupcPQ4nFZc=")
+ORIGIN("react2media.com", "XK+60uWgTwJzy/c/wGf/jS7EMxrqFqdM2eu56ms75TA=")
+ORIGIN("terra.com.br", "3w6Yul7AhCEvdfZtfEOwkoFjVz+WUHDSCf6aa0+T10k=")
+ORIGIN("fairfax.com.au", "D9SojaSQJFQns6k0VA4JWvy3vOzURuWuueq40DF1aeg=")
+ORIGIN("prometheusintelligencetechnology.com", "qWtOdQUfAjIkHeZl+jJmYnuHpSQ61oKSLi0a+ye32mw=")
+ORIGIN("hilltopads.com", "o9mjL1EVpqpjDGhV+M7S8xXZVKQkjWtKPbJNRT41xis=")
+ORIGIN("mediacom.com", "bnoxalLXWt0sb9Mb97hZITfD1xdYjNgOYaehJ+x3SZY=")
+ORIGIN("attracta.com", "D+VPBDDutNd3j8Wn8AX+s8ZRgjGJmxtYQn/xs/a52g0=")
+ORIGIN("capitaldata.fr", "KtPjgAjOM4mZSHMbgyJ/kZolrK+y+zLV19oSYKiXRB8=")
+ORIGIN("adchemy.com", "PujhbXfMhTorG4qjwpUMfC3lIZ69M7jIutp6WCn4YfQ=")
+ORIGIN("everesttech.net", "rY734YHBfxZlfPOkEZs2rEcwuystf+JTmYqg2h6b/70=")
+ORIGIN("pro-market.net", "w2jMbuFCPYigdrIIPyCPO2C97wBpnBhMWQp2Mz7I4u0=")
+ORIGIN("feedperfect.com", "eZghqjaGJTf0rxOx3vy//0wYWbSXVgpXjYsBtiEtPIE=")
+ORIGIN("google.ge", "4PanC1O4CCrbOIC6A5mkjQPffMzK83OAowADsk+d3/Q=")
+ORIGIN("google.gg", "HolrgkFOtuUfOt/f1sAO11DpyZRK8A2UvlFkPh3KHtY=")
+ORIGIN("google.ga", "2zFIGZwGbLpJDMH2A4RetgZqCMGLHqglkCbQ+e7ppY4=")
+ORIGIN("google.gm", "bOpkdAk8SQq/EccXIDakWehpHq0yVEiIU6Eh1ySk6Gw=")
+ORIGIN("google.gl", "GgGDpfimsA4tbLT88Eh+J4facPoQKHDIGX5vB3wd64c=")
+ORIGIN("belstat.nl", "toL/JqXxiBc0Vw+0wCH+5DvcoyXzrlboZWCW2+/IAps=")
+ORIGIN("google.gp", "68bmHTESsDta36JHibzyO5c6Ucwby3w7c03XQtynbec=")
+ORIGIN("google.gr", "76bKDP1iv99ELo5ckwvc3AgM0Xca6Ms6ztWOVhnmr8A=")
+ORIGIN("vserv.com", "i+Pt0oIqDUTvbaNm77b1pBKo0DesxrbBG3nHKwpU2Rs=")
+ORIGIN("google.gy", "cLMXcM2KNkbsM5as6/1Uf+bkpsr/jqDhrWliZWUobOI=")
+ORIGIN("adfonic.com", "vXVVok5Y4AuVy8b28/hJSYrf6xN8WT1taAz2d60vMFU=")
+ORIGIN("wingify.com", "xQUJbWTqiDfKnLqXujJwAbzrYAD1PFBYWio019c34cM=")
+ORIGIN("inner-active.com", "JbwMJchyrK5GUYqNchmAYa4KDiWT+dv5f3OeBZJYgL8=")
+ORIGIN("feedjit.com", "fn4mGtqS1P2FPxKCFsC9SwMHtCStssd42kx3dxC1V60=")
+ORIGIN("google.com.fj", "ngws0drvFLHWoEax8F113wcUGmijftDZyjmdcyltSgc=")
+ORIGIN("adlucent.com", "9AEZYdU6Hp2Sa18xVkBHdt/oavIFqLQVV+RRLBQDB+E=")
+ORIGIN("dataxu.net", "6t6FpThCn1iUsHB4YU5Nk78JJHxXqgJBBqP/7jPVVZ4=")
+ORIGIN("mobclix.com", "NzY3Au4YypMVH1mSCKJ4x5jYS/x6LktUbSlmct4iM7Q=")
+ORIGIN("traffiq.com", "wyw0KnFgwJscEaUB0dmvu0cIing6x4uKCdzf87Fy5q8=")
+ORIGIN("sitestat.com", "rG3cWO+2RoKzL7rnNgPDEpQvqswkaxsDN5B0ckzUK/M=")
+ORIGIN("websitealive0.com", "kiMGG2sZwloDZqNhAx1Pvxgsn8ycth+gq4gO/hUZBls=")
+ORIGIN("admobile.com", "b0fAcImObRTileeEtC+CdbRa+rk7xCxIEQ3m99b9V3E=")
+ORIGIN("adtegrity.net", "ZRkceLPAXE+2PEWG+gkDfRJdsjH8NOkT97HAZpxJlcw=")
+ORIGIN("media6degrees.com", "83mRm8sevX9pdgT+3pdh5a9yRPtAy/vse1HXtBAj1cQ=")
+ORIGIN("wysistat.com", "aXCjYw5+GfQJ9ARd/13zmlUhNxTtiM50IWqC4A34UZU=")
+ORIGIN("adformdsp.net", "ZVO3iDUcnVmmi4ipnjxgLashnepY30JbADBu1rQbdy8=")
+ORIGIN("eztargetmedia.com", "gyJuTxIf4DFOHi/WO27aShtj6qqoRpA0jo2kW/CJQqE=")
+ORIGIN("awaps.yandex.ru", "Qv0yfUPxopb/eWXI6qDRvIBuAjafWaSPyF67oOz3J+k=")
+ORIGIN("ucoz.fr", "4PtMWZppV4PVP8QkKj0iAACfWjMnRHjBNKEiuRA14kI=")
+ORIGIN("sexinyourcity.com", "2G4ptC7cP8iM59FciAgYDLFsqJp925SAwIW9cbXgAp0=")
+ORIGIN("clearspring.com", "OVAaSC9Y4xtyZcrKbYSmVHZSqHh7zkYw3GUjjx69HmM=")
+ORIGIN("blogger.com", "1PiXFiCCxLm0TmVgtif2UcOszZJAchPOYmYkCJVMzVg=")
+ORIGIN("retirement-living.com", "viLxMTWAQ3Dq25MB5rFjSeY9p5AgO2lyqp3ALiWoAkQ=")
+ORIGIN("nudatasecurity.com", "TghqZbhkf2WJ6vOpSVktVgPiKl/DM/4a0Lq0/Phmwjg=")
+ORIGIN("adbroker.de", "6nccMCRpQWvvhejiRs2u+YP5ZRBapR0hcKX5VGTQUMw=")
+ORIGIN("acxiom.com", "3UWUsxt1OjPPOa7AMuvqWzFBl/KPP3qbsr6PCvHTvSg=")
+ORIGIN("parsely.com", "K25kUIOucmn0ED4d9FoklUHlQ2DJnQHMvSquuaZltHQ=")
+ORIGIN("clickguard.com", "xYodb6xnZdaheyvk7r6D5K1e5OXm2FMchdm3tDER/b4=")
+ORIGIN("dsply.com", "sOK6oILc5ttdTM2vZGmomjADd3ulQezWi014ddXujj8=")
+ORIGIN("xgraph.com", "2NcB6dBwXPzk42hFHi6LVzTeNMsm9wS/kgQheI1p/GU=")
+ORIGIN("rfihub.net", "PN4qA/uVCw60ARW6ZuHO7KHznIq1hb3Gm3DJwRrJsCU=")
+ORIGIN("yadro.ru", "FJVKFF2JkwoKV28kb9N4ZbjoygI40mOXmBTBZcs8Wu8=")
+ORIGIN("decktrade.com", "A4JuevO4vmBTBwtX9FH2F90ngOh8Ey8V2V5ewH09Ovk=")
+ORIGIN("ringier.cz", "a/95Usja37PhowVgoA1zUELRgAO0opFfHaRz8uFzqR4=")
+ORIGIN("lynchpin.com", "TdUFTjt0kUERKU/fLO0l2dCCyzFoh4qEaXP2o8kl3lY=")
+ORIGIN("adacado.com", "mbLeJmN5iAiHkmO/K/bXA6TqheqzNaXgzrAyH52/ROs=")
+ORIGIN("stratigent.com", "RJmNisy7UVBe2wpp7ldsL2LxPJGI7cqglRAwF0Z0BkA=")
+ORIGIN("hitslink.com", "tpW7H8hxe4gIOvoaGtFxM2r3wePPBWTMy4ezYixod+Y=")
+ORIGIN("google.com.mt", "nggYcw27M5xSv0/y3EyUZ+11IDhGHFJdSwrcL9h5r9I=")
+ORIGIN("baynote.com", "c0j9r87+hFXFkdK7EvulRkQm8+KE1cHmMAVZxLugDK0=")
+ORIGIN("cpmadvisors.com", "k1MJxVrtuFYk+/qOmAHOqvhpudfj/23QFO70QnPL1CQ=")
+ORIGIN("movielush.com", "EcCjvIYeVhgcuVrE0sqoX/P8codTJnjJw14eGvY1bbE=")
+ORIGIN("adiquity.com", "Td7G160eLsO7m3jXPnpXCX+m8IVytJMzOzcgh93nYqc=")
+ORIGIN("adventive.com", "DNgJGPSb7vY4K6PpqmsVf9VENB33IDTCEbr2C+cY0dg=")
+ORIGIN("adrevolver.com", "1du4mUa8yW+rQVwO0g9dLokvCUBXHHG0nsCB//wAoBk=")
+ORIGIN("strikead.com", "eam5LMLlToQFqLFycWcQRjZ9mJf8L5qtTX8+V0daEK4=")
+ORIGIN("yandex.by", "ruybcX2pvGMKxCRWpIe6ftHs+yysVAmTL0RiRCGOz9c=")
+ORIGIN("p.brsrvr.com", "RQfs+0UyhHC6WSS+MVC13SllP/SQTQ3zBngjf/qi6IA=")
+ORIGIN("phpmyvisites.us", "wKW6z2fCppAkCyPtY3Gi9sBhd1DY/7rYjIbzv+960GI=")
+ORIGIN("google.com.ng", "RF1g81myQx+FjY23KHB9MEb6wvKVkPO1voajWia2W2U=")
+ORIGIN("google.com.nf", "ny4tSP402SSnapITMrHa59InAzf7/FkEcvgkCD4GnMw=")
+ORIGIN("google.com.ni", "XfF1RSerP19ZtKhF2xcEGix4MMKd++EvlHjiWwrneGo=")
+ORIGIN("kanoodle.com", "TFbDy+H6ih9B+NMHLgDV6/vMZFGrW9wuA/mdqxZOExw=")
+ORIGIN("brealtime.com", "QupUtOi6EP/PV4hr5I02DDaj3ioU1bQEypVJQMd1CsU=")
+ORIGIN("google.com.np", "eigqmhnniiknmHhqzM2jGMal05xwRAHn2pB7yMmYwEA=")
+ORIGIN("googlemail.com", "N0t/oELPOzghf/0pHRZgaD6546SxMAG9cbH1fK1Bd3g=")
+ORIGIN("woopra-ns.com", "QX0gAY/1an2NQ7O8bOTYGrP7lJ7V2Hc6zUlddPsUTNA=")
+ORIGIN("twyn-group.com", "TRYp4MquMlHhktCDwx6gEkDWBsbIN5gqVM2W0r3IsB4=")
+ORIGIN("mystighty.info", "jS7yDRMNabPfqVgpiOqEkMz3CUXtiiEDL7X34Mu0tGE=")
+ORIGIN("doublepimp.com", "E6n3KiepzqEo3SqDPkU/vg7VcMV9C0cEJG2eJXkotso=")
+ORIGIN("keywordmax.com", "gF1pxUvvqLOX8y2Aao2LiYPe3Ts+JZsTkVhp6pRHfzs=")
+ORIGIN("serving-sys.com", "qvtpezST/UNchB+835JP7g3Pan7XKOfpr6hdQqJkaPY=")
+ORIGIN("adzerk.net", "SUuTDIC/x1IiGKSrsTlPq7PGB/T7gltvYqWljO7MeRk=")
+ORIGIN("adelphic.com", "WqJ7xsZBGc20WyvLO3VPv+r9hxD30Q6cUgrSjmnHELU=")
+ORIGIN("godaddy.com", "I22hOROgTcpCJy2EX6aTV/fRYA89CYnMk1Gs3QGDoSI=")
+ORIGIN("bgclck.me", "iICaf0oGYG3m65s4p6fwFgR+JzXBKMGpbyWvd+NvG8k=")
+ORIGIN("geniegroupltd.co.uk", "lqt4aaM+T+Dp/3e65cd4Y3FUF5OQA1sS8tZzmQxbeZI=")
+ORIGIN("amazon.co.jp", "2OH5RfRgt2hBaUyq9fJGfdxMb8JWHm6oO+hLx2CI7+0=")
+ORIGIN("infostars.ru", "Op25M/vdRO3aAQhn3i8KlERlQ7X2MdhaLDg7CbRPFNc=")
+ORIGIN("cpvtgt.com", "WaPOvmbGct+ix6hWzNwlclDrWjYMVj6CQ5FllD/FP+U=")
+ORIGIN("tremorvideo.com", "vaA0YF8auwQJVWMIrAYltELy8vyAlljwsNvU22YMfuA=")
+ORIGIN("adonnetwork.com", "//PeGn3o6STCxDZ1VpLofuodxwZt0U1opGcneUkWDV0=")
+ORIGIN("games2win.com", "3TJK3/hpOAmA0VzZhbdWOR2bnRfyHTR7qxqkq3Ebb6g=")
+ORIGIN("google-analytics.com", "nfLpMnbE3tXvD7L7pWmHcFU/H5UETdtt+b4eZ+WMGB4=")
+ORIGIN("encoremetrics.com", "QKVFQg0HDUGdDnRsSwl+Kxsi3trQ9a/QcJryQyN/71M=")
+ORIGIN("doublepositive.com", "QjPL34/b8nt4z9FtyM0dTjm/6BuYRNdnaF+061VTRN8=")
+ORIGIN("google.ps", "wpmj1RNNAcuqPzyAvsprdbBCrMsMDrJCUyO1142aSic=")
+ORIGIN("google.pt", "8zGrlHHXCN4xvciFBXLiBvdXUXA8kCeT/6c2iiNqPew=")
+ORIGIN("clipsyndicate.com", "pTETTD+kClerTHzyXr1IBzFEbuJKEf8H4U8sslePHH0=")
+ORIGIN("bubblestat.com", "sqRMzifjUEji4lSVMIMYN96XpZq7x2Le0zihwARxCF4=")
+ORIGIN("topsy.com", "tOscJ/xJB7VJ6kCvUGLfE8FlorFLnO7lJt3rWEPyELM=")
+ORIGIN("cart.ro", "p4nTR5fggWlhorq4drMRF31X9y2NGAVZAvtsWqWzUy8=")
+ORIGIN("ytsa.net", "zJUooWTFXXjhoSvGqExgP7YT8LjgadP725w4DN0Qh4A=")
+ORIGIN("addecisive.com", "P51J0uAGOyBG706xipSKS8qpy30joqWqeFTTje3Nn/8=")
+ORIGIN("google.pl", "5MFXUszUVt/uzdvHpagaEXc76HxPOBcPAL4Ik64qaCU=")
+ORIGIN("google.pn", "M6eTcroXtxPb6d6H7W8Mj9Ev88oMDut29TwwETZ0PRw=")
+ORIGIN("google.la", "Y3HXhz0oZa01kk1eseFf7MWGBDNXNN8z1aplFSr5m90=")
+ORIGIN("turnto.com", "MR66YU+XTq943u3cRgFxJIbdq4G5VUdxt3Nx7I/uX6I=")
+ORIGIN("247realmedia.com", "6iVKLG6hd9xUHgryviC/fd0szgriWpnLwnqkkJGz1EY=")
+ORIGIN("euroclick.com", "lDnm+o4Bb+BVt7XegQmGE+90RNAIvpvqXzhhs8mnV+Q=")
+ORIGIN("pubmatic.com", "PDKyNpwWJr7s1Z0D8H224ErzU15UpXfZS/oO+uMxi28=")
+ORIGIN("admeld.com", "vPhSb+wA8xC37SkfoN9MGgoz8Bd8khq7h6xIx/bj8ps=")
+ORIGIN("adultmoda.com", "KGCkYwtjonN3DMN5iYWslOhp05ICtRyAg9kCLKtQ9mc=")
+ORIGIN("inboundwriter.com", "jztADB4HnJQ4UWR0ZE4CFypIndiAIRAyfxmJxeaq/RY=")
+ORIGIN("ipromote.com", "Ekr7FW1KHfaozPAroB30iIndZyIfJ6c9I7rY0sqDIuI=")
+ORIGIN("adrtx.net", "JcYV540d05WrBFxs1neAVstmr7Nq5j1DQeg+AFe0cvY=")
+ORIGIN("adsfac.info", "YQJCLVRlhsvsLeoV2OHCtwi2U5IwCbG2Nb2Mq1adGJs=")
+ORIGIN("knol.google.com", "7isgP9uFg+KQjFEfuK5xMISONKoled2qbvxlQtUm05g=")
+ORIGIN("yahooapis.com", "CGEPsrBnFILwJL11Tjz15N9HaMCFKkrAtdI8/jAzkuw=")
+ORIGIN("insightexpress.com", "5spU+PV8LmCEgTXtG1zmU24eK8wYdCLQcOT3DMublCo=")
+ORIGIN("csdata1.com", "Ao3/NViw6taQuas9rb7V57d6J2MJwl4Hf1Kxe6nvPK0=")
+ORIGIN("adconnexa.com", "e60mDfWpU/yKJK1UTUA9BtZGVb4Rw/sMeUqCtGY+G8E=")
+ORIGIN("stargamesaffiliate.com", "/W+oIX0ant4x6huUbMP90viG4XzISPew443RitgzG7w=")
+ORIGIN("legolas-media.com", "BEWlCJm6oOVmmAMr29kl2WeLDuYTxsq/77K4D9ja4lA=")
+ORIGIN("telstra.com.au", "dngmGPDreuh49ST8eryQQ09yNBb5BjxyMxUle1xmR5Y=")
+ORIGIN("gmail.com", "YUhHHolh1gWizFkn/7n2xXLRKt/yx+HqlL2VHgHmsiE=")
+ORIGIN("awltovhc.com", "0fZETTFfdH+RiH1qKZvS38o8slVx5OqM2E8nNBfWewE=")
+ORIGIN("clicktracks.com", "S/SIW287Xh+s+8yy31egknvgqfo8mw6XmP2pf3M1jqM=")
+ORIGIN("bidswitch.net", "Vt+6m7s33bbqtjHKVqyzHaTr5GaTLbhyzMvKTbuxvJE=")
+ORIGIN("noktamedya.com", "uaYlYoz+kfW9qqzUWCXZ7NZ8l8+FayiEqos6hrc8g68=")
+ORIGIN("google.lt", "yqRhCh6pH1sCMeeF2BO2NPUh8afj33YneSMyc9Xf5+U=")
+ORIGIN("verticalacuity.com", "DoH5AfiNR09T0lBw+xl9SynApuThLFoEGUDlCqZ3UHc=")
+ORIGIN("google.lu", "ZliPg3sbl3Bx/254hCa6FAFUIbpxp5/hQ6J0jOZ78L4=")
+ORIGIN("youknowbest.com", "nTwKg7gVdoLP5FFkdISyNUmU+GtEdVDDKDOLoFkwhSM=")
+ORIGIN("src.kitcode.net", "6vS1p9+ZNIUNbEFZuxthlrVSLuaAZXlvdcNPYe0IHKU=")
+ORIGIN("voice.google.com", "BKDrnMGSsgXVEjkwgL7oLs4oG0Vdx1x6Lp6FGOO/8lk=")
+ORIGIN("gigya.com", "ausWPqhK0ulWOdTDdAi9dDQcIq6euXpoPIBJLGnIfjE=")
+ORIGIN("tacoda.net", "9/lQNfL4ALCNUiG2xzehFPuNTPXvRS1esAODiqdWKVw=")
+ORIGIN("developermedia.com", "GhofUC9cRWzUQ3lIquBESD9LwEgtK32kh6fXeq7Dt3s=")
+ORIGIN("cpmatic.com", "WFo3DXfKT3NbzxmSBaxASgzHblgsWDHQbDM+kFaICQM=")
+ORIGIN("onestat.com", "8n8vYYnRjnHT+T441HztQDqQqaL3TWQR556SAAUqFhs=")
+ORIGIN("plugin.management", "2kBf5YpTrvWm/i6g6zSW2YSRrvmw3b6vwQeYvZ74Go0=")
+ORIGIN("adohana.com", "1V27BFRB/z5RqjIjLELuiwgsc67rRlrGRWKNKXFaKFM=")
+ORIGIN("mindshare.nl", "VBUoneI2ZT2PRbwP3MwpSq0T6HGEsTv2JvvxFP43UOI=")
+ORIGIN("polldaddy.com", "epTHk29UCBP0SErBUiLAee4EMjr4JMc4SAYYLC3yllo=")
+ORIGIN("roiservice.com", "eO5Qfhu4sPKG/9ASWvg4jb2c09NyfgX5C/+/QORornc=")
+ORIGIN("365media.com", "R9UexJHQ5GWrMd2Z9umNuGR3BLNe8ZGArtsl+XK3X2M=")
+ORIGIN("bluelithium.com", "ErxC0eNUlSd9mIj+Hn8QwMpHQVceMJKX03YnLlPe9+8=")
+ORIGIN("adsymptotic.com", "GjFf/iPOdNumdpBLmbrsytpop8J1u5YmFx/qivYKcNs=")
+ORIGIN("qksz.net", "35LtjBK8Hh2WIJUCPzODnoV+nuRPhnw0JBbt+WCmu50=")
+ORIGIN("grvcdn.com", "mN7gWLHi9morQlmMZuUjawP1UwRlYL23PeCDM7x+T4g=")
+ORIGIN("emxdgt.com", "h36/y0E9WmM08TzEAkMy/NbYr0Gn3JE2SJ4gOSwMOg8=")
+ORIGIN("gosquared.com", "wXZEF9KH5DBYnsMTe00kbfQKqGL21Brbt8TdktNS9A0=")
+ORIGIN("google.fm", "CvepxASOdY52KZG811o2T5L5YkKrQgTbvvLG7pZfCWs=")
+ORIGIN("adfunky.com", "Qbpdxqfd6AhnVNWNkmU9oltwVAWBsqzQya2SnrD+ye4=")
+ORIGIN("imedia.cz", "r7YzQp2tLXLcT8ioXEKq3I1FVHTOaO+LcM3mI0m/FTE=")
+ORIGIN("specificclick.net", "kFWtJ9sSLSOEi3qESEmHm5NC+naOI9wyLlZMyZweUN4=")
+ORIGIN("google.fi", "d4tDu7TVVDudsfK/Id7X2Y8FmgCMQGM+VF32HsUdsvw=")
+ORIGIN("google.fr", "yAb5Zg0XLui9Y4dTge+JjDsGyQMHDEbo3kRp/4k87TE=")
+ORIGIN("tnsglobal.com", "CBIO18Gt5/JQpgFNCXY2Dii/57X9mKmgyclXA6M79IE=")
+ORIGIN("ibm.com", "spyx2WTEo/xoN7iRozH3eswO7KFv5kfxufMnazkEaSY=")
+ORIGIN("tweetboard.com", "J79++DC3BgMOGPgect/KQYCW+Rni6u6YC98xqN+ooq4=")
+ORIGIN("adversalservers.com", "PNq+Ls/CJ36SSS0et1HZ/WiDuJNFPblZYfJDhupNFZA=")
+ORIGIN("adtechus.com", "OcOZ6GFT3UfCzFDnTlaLS5Y4z2i4riOK4+d2rasbUkY=")
+ORIGIN("twittercounter.com", "20C2bgBha9BObsHfS+EIpaxu60u72KNcTIjzhIa+HHM=")
+ORIGIN("airpush.com", "b8w8a6jJBf4ypgUmX65kVbjRpjAOXaFqI8ASwRCfRwc=")
+ORIGIN("visualdna.com", "Qs/9pcsgAB61UI+h+gzjCpK0y3PUV5fq6mo/4XHeFOs=")
+ORIGIN("gunggo.com", "ipc638vyxNFkfRNPoON6UGPC1X73IP8yu/PqlIjj/+g=")
+ORIGIN("usabilitysciences.com", "fyIPWRJ5c1ZxN5Hay432FIeUJEJ0OWsC/3/Uh9IBLOM=")
+ORIGIN("abaxinteractive.com", "qS3wJInMwDy8ewhNv+HAE1VSirhJR/RAdMunQyC6FNk=")
+ORIGIN("magnify360.com", "pzbNxzR5ITEiEUGWVXC4T8kyR5aDY6XlBdhrnmyyr8I=")
+ORIGIN("logdy.com", "8maMecTf4YcZJKN1KRdkyFGHstMSMopPSerO7pkNyeA=")
+ORIGIN("maxusglobal.com", "dWNpNiqM8LwVhXoAQJcsyhcLj/995QNCrS4THLgf+CM=")
+ORIGIN("csdata2.com", "1cGmxe2oo6505LUcNSC44Yy2+u9bgHPYYXg+Ea/7V9g=")
+ORIGIN("iac.com", "SZSFBfRirUPuyD1RSoYgVaKYUuGckRJ/6p/+laWEVIQ=")
+ORIGIN("google.co.uz", "PInb5/Jq9gv/RNrpCP26m77RqVEbO9kO9cI4YRDUCqA=")
+ORIGIN("yandex.com.tr", "sawWnWXWNPE1GOvKk3cpJdPbD9kMXOn3GYB6ka5CgQw=")
+ORIGIN("wtp101.com", "b0Dla0qz22tA9rMxagngWG9qoidJ5vCpSk2kmFTzBUk=")
+ORIGIN("svlu.net", "pIAKroYlruDvhh0UbvhNHLnpcxd451SdSeJZRYZjh1U=")
+ORIGIN("google.co.uk", "YhvuR5TjpAp1kBKS4lSAyguLNj9J63lDTeBzBCbZiuI=")
+ORIGIN("sensisdigitalmedia.com.au", "/ObEs/Rgjk+3ICojq3zvp3OUakvz8UoNWuspdvzhMWk=")
+ORIGIN("alerts.yahoo.com", "bqtAdDAxQZy6rjQavUyy2lb7az1CAuiS6aheA8G5jok=")
+ORIGIN("yandex.ua", "dSK456tW1Xwtp/7dg0IamAjlcbzCcoGzZE1sRPNJ5HE=")
+ORIGIN("kitchendaily.com", "8nw4zpVgyAjlzLMekv+ipZCoeR5WEbeNQdtj93BSfzI=")
+ORIGIN("cmads.com.tw", "Wrihb3gYEPhDr08mpjPIigoPJObOUqDnQ2edgS7kfQU=")
+ORIGIN("google.co.ug", "Z2HPupC8Y6jE9LCXSZx7RoyHLclkEJ8Wrb1jD4zn4SY=")
+ORIGIN("veoxa.com", "YXXBSP6+f6hjGRPYE6xmlXAtLug4kvYE9tiopUXkkbI=")
+ORIGIN("live.com", "7M0/r7UjOdP5TKBFvbeDwZpP/nkB0IVaAsuYZSr9mr4=")
+ORIGIN("mail.ru", "h90DtgXXu0dp66WCJrvqvz2CgP5wuzKkNW7tGGMD3Bk=")
+ORIGIN("peerius.com", "tIuO+7hZ8FSFStZSWpcf8fG3OBofLUoghUIOsqHgSm0=")
+ORIGIN("turntonetworks.com", "k6rf0op+YNTTh8xIvB20NTCyhQUaEF3kSV+J2SnPigw=")
+ORIGIN("tonemedia.com", "SOtHIzhQwR0nabTyxr8j64leAy340HgBJufhSayIB3Q=")
+ORIGIN("rtbidder.net", "elX9K92R1Dy+GOxXhxH/NvujK3XfhXVAfXUOas4TLBU=")
+ORIGIN("google.co.id", "zM56p790PS9sxbBCwEyZzCVfbjFQWERCfOu90c8Lbvg=")
+ORIGIN("twitter.com", "7+XSRurpWgtx+tvSD5S4Uq5xvaSXSHEDAV5Gm3pWbkI=")
+ORIGIN("visualrevenue.com", "gCeFfwQFEPxOu0dW+l890WUYwk13VtJG6BdjxnxDFLc=")
+ORIGIN("rlcdn.net", "YeaCqU+nhEHYDxQavoKRozFv/Og0qNSXMjL1+wJPjWw=")
+ORIGIN("eloqua.com", "Cy1S7yfqXzxYxdiqsdYn3/G8YG6XO/QN10C6I99T8s0=")
+ORIGIN("news.google.com", "YuKZZymtdAVFJWRl5UOHwhh3wGxV8xFdOu1thH8J1A8=")
+ORIGIN("p-advg.com", "KbmdPvSSZbUAiAhmCEyFINZb7h8cKALit/i5micympU=")
+ORIGIN("sharethrough.com", "vGNht7RnDTCxFqTZ/Hov8tJi4T/0KKKBH8ZiKwN+kh0=")
+ORIGIN("brandaffinity.net", "np7fAq9SRqRIucYXt2IfdL+JWHEJQTNQnvwmrPF3zsQ=")
+ORIGIN("vendio.com", "7WFklFArow8+3jkPsqCxHd74I9TM7alWjs+daLXBGV0=")
+ORIGIN("techcrunch.com", "8pmXPGLVeYyM/SjC233pn66TsVoLD+TQC2b3iX7ErLI=")
+ORIGIN("yuilibrary.com", "dSMvd7ZTR2FY+a13ONdNFr5V5OY0P2NsfS4pulWjtzQ=")
+ORIGIN("adcde.com", "VaWn7NZSVlwUu9O/9/sEKPs8Ov0RLPKRFCxSpbTcPDI=")
+ORIGIN("pictela.net", "L1NXkj2U5GHf9VQGPFn44doWl2XbhUAnUkwA9/omDds=")
+ORIGIN("adrolays.de", "YAZKQHakPzwzlxfD7g2S4WD35knvRbMP0Ps/yOArfRo=")
+ORIGIN("radiumone.com", "PKJPFns8LP1/wckQhuZlzAfh9QeoAEDqf9igBYojV7U=")
+ORIGIN("developer.yahoo.com", "Nrz1eqSXEPL1rbldIlHGmgFjoAX86HPSeCO0mmxRmrE=")
+ORIGIN("clickable.net", "m4Scz7zc91YiLAaOa8Zx9u4Cm07o3F4CjOYBMeVVRRI=")
+ORIGIN("burstdirectads.com", "Q5pxsv2z/uH+D5vE9kyYfiuFP44lvJ/+1EOVs803zzA=")
+ORIGIN("qsstats.com", "PnEX1rWjkvb5VO1UyANwfDto/goiR53AlPvkASyPuak=")
+ORIGIN("summitmedia.co.uk", "biE216vi8UcYSIKF0Cvq+oAmyRTxXLRbcQuDskwwXfE=")
+ORIGIN("adsmart.com", "vRbCtY83XzQsjWHtA5mVmd/SCRXqXGDxA5vZqsGu03Y=")
+ORIGIN("quinstreet.com", "sG7xOk8q7TwvvpLMPOgvqs9Hk/PrHTR0VFLj86bhrA0=")
+ORIGIN("valueclickmedia.com", "dCEGshDVxQUAtgVmYkbG51eVmN9ix96tS+XkUvlg0qw=")
+ORIGIN("1rx.io", "BNcxlA5DX+h3oeORIwJXFeg2REcCjJk71842xtzdh7k=")
+ORIGIN("webmetro.com", "/nHY1XFXcT0bWmXzwHmfr0FHj97k86KcnxV9tqGotwY=")
+ORIGIN("nuconomy.com", "+7ENDBVruI9RTBg4CPOPh/ygYQ27z69bgx2ae28gb3g=")
+ORIGIN("images.google.com", "g49WVkX52Q4wwr7M3GjOkuL3anN9nKKp1rAMfYag9e0=")
+ORIGIN("thetradedesk.com", "03dyXX8gmppDpuU4stWmxBW0IP7+eQhcywgri7j+uUc=")
+ORIGIN("hooklogic.com", "iFQrpjZWR5FpTNH/kWYRW5j2b2xKkfjDT2xK9fWe2bs=")
+ORIGIN("audienceadnetwork.com", "AxNWTOsyAkC4fSyDPd1CN395M5CdZhmIk9cZ+rChMzE=")
+ORIGIN("mmapiws.com", "CYKUtmNj09mSAWSAKHfuttxbTQ2ynpEqucX7qkLIidY=")
+ORIGIN("b0e8.com", "Co/6TjaoY6VB0iMbU4wySMmOMmavCgrzB/TTfuY4gNo=")
+ORIGIN("nakanohito.jp", "+R01KsDizmcZDta+Q2QkDAoGp5nFwl1oQ/SR5+7CFVc=")
+ORIGIN("de17a.com", "Cmu23OqLpLjQml+U82RtH88RZnFv0u+nMIXdZReeGhc=")
+ORIGIN("renegadeinternet.com", "+8Cu48EizDYBPViZMmAtr+1/l4duO8CkHkBml/cluUY=")
+ORIGIN("verticalresponse.com", "F0ySG5B/4LWEQIfVSDnNdy3H53RusgYF8ljiK7+FAow=")
+ORIGIN("networldmedia.com", "EeIpLPYezBF/e5pjKAl/MHW9nnBQjM3ZbCgXTmhD838=")
+ORIGIN("adform.com", "luc0spL2jZckFpFuyyH3A5ZAsm77xIaDBPO0Ei4uAQU=")
+ORIGIN("yellowtracker.com", "nMFY5n/+qjZB9o4vHUbTI1M1x3BbbKMiWPa8dXH1k9w=")
+ORIGIN("zopim.com", "Eu+zmA1UJmag0axvo1duSSWMfx7/+WpRk/k+lQryHVs=")
+ORIGIN("doclix.com", "iA3vMVkmXYdOIJwrvQ432VIUxoH2GP5EvZKD+MRkZL4=")
+ORIGIN("datasift.com", "AbUpWsNuc7JAG87JtMnLkOy2TwbERoewA2OmOVt4AHA=")
+ORIGIN("mb01.com", "X9/9IgjduKZsZNh+qcaHIfFcczni2UHbBzrCrq3ZiLI=")
+ORIGIN("hittail.com", "bzUrFVcPFDIKxXpMgwwDqYPpRPmqniS/SQO45aJtiX0=")
+ORIGIN("haloscan.com", "AGF75p17F4Y+Lh3uTcS4M7Wip8dbilRIUvTrvj6DHeg=")
+ORIGIN("inbox.google.com", "SETskRfVJfgkrEOzF3bg4OLen9zaSTwhyaUAiKDvD7w=")
+ORIGIN("azetklik.sk", "Z532Goxf9qifO32Ngh6cl1Xu9DWkLX4MsX9CLtMvMVY=")
+ORIGIN("rightaction.com", "gYCy0gvIUKlc4M8tmD/DqJ9QiIYabe/WYf/V3/BZark=")
+ORIGIN("mathtag.com", "2M2U/R8LuShy13GU0Z/8mjX8AnYMz0dnv2I+pxQ49Zc=")
+ORIGIN("mouseflow.com", "5HPwzePIU+NiJI+b90n1o+rxSWzmE90jQaGZbyYkSB0=")
+ORIGIN("patch.com", "hza+WcYpIu98rNjfSmQH9v59EofdLOb7UErhMnSB8SI=")
+ORIGIN("venatusmedia.com", "XZOM5Lv+ZlQT4B6rxS8inCvkqamFmhdbO3UMK89nDmU=")
+ORIGIN("runads.com", "c51XZFKuxR9sAFNDGCxaFCG8HHEKVrpSUn60WmPxOug=")
+ORIGIN("connexity.com", "W6TP0LUmeeDvQ7d5CUNJQBbBgzQca3U3lP93u4IOTJY=")
+ORIGIN("linksynergy.com", "tDvbLbQ+0UMrKE3Rs+LPAdqgIYRRdqr/MW2il8VQr0o=")
+ORIGIN("alexametrics.com", "Hwn39xeqNN0OkHgzsBTNuunG3OpodVTGICWpmJ7lcwM=")
diff --git a/toolkit/components/telemetry/core/TelemetryScalar.cpp b/toolkit/components/telemetry/core/TelemetryScalar.cpp
new file mode 100644
index 0000000000..0f4bace011
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryScalar.cpp
@@ -0,0 +1,4202 @@
+/* -*- 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 "TelemetryScalar.h"
+
+#include "geckoview/streaming/GeckoViewStreamingTelemetry.h"
+#include "ipc/TelemetryComms.h"
+#include "ipc/TelemetryIPCAccumulator.h"
+#include "js/Array.h" // JS::GetArrayLength, JS::IsArrayObject
+#include "mozilla/dom/ContentParent.h"
+#include "mozilla/dom/PContent.h"
+#include "mozilla/JSONWriter.h"
+#include "mozilla/Preferences.h"
+#include "mozilla/StaticMutex.h"
+#include "mozilla/StaticPtr.h"
+#include "mozilla/Unused.h"
+#include "nsBaseHashtable.h"
+#include "nsClassHashtable.h"
+#include "nsContentUtils.h"
+#include "nsDataHashtable.h"
+#include "nsHashKeys.h"
+#include "nsITelemetry.h"
+#include "nsIVariant.h"
+#include "nsIXPConnect.h"
+#include "nsJSUtils.h"
+#include "nsPrintfCString.h"
+#include "nsThreadUtils.h"
+#include "nsVariant.h"
+#include "TelemetryScalarData.h"
+
+using mozilla::Nothing;
+using mozilla::Preferences;
+using mozilla::Some;
+using mozilla::StaticAutoPtr;
+using mozilla::StaticMutex;
+using mozilla::StaticMutexAutoLock;
+using mozilla::Telemetry::DynamicScalarDefinition;
+using mozilla::Telemetry::KeyedScalarAction;
+using mozilla::Telemetry::ProcessID;
+using mozilla::Telemetry::ScalarAction;
+using mozilla::Telemetry::ScalarActionType;
+using mozilla::Telemetry::ScalarID;
+using mozilla::Telemetry::ScalarVariant;
+using mozilla::Telemetry::Common::AutoHashtable;
+using mozilla::Telemetry::Common::CanRecordDataset;
+using mozilla::Telemetry::Common::CanRecordProduct;
+using mozilla::Telemetry::Common::GetCurrentProduct;
+using mozilla::Telemetry::Common::GetIDForProcessName;
+using mozilla::Telemetry::Common::GetNameForProcessID;
+using mozilla::Telemetry::Common::IsExpiredVersion;
+using mozilla::Telemetry::Common::IsInDataset;
+using mozilla::Telemetry::Common::IsValidIdentifierString;
+using mozilla::Telemetry::Common::LogToBrowserConsole;
+using mozilla::Telemetry::Common::RecordedProcessType;
+using mozilla::Telemetry::Common::StringHashSet;
+using mozilla::Telemetry::Common::SupportedProduct;
+
+namespace TelemetryIPCAccumulator = mozilla::TelemetryIPCAccumulator;
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// Naming: there are two kinds of functions in this file:
+//
+// * Functions named internal_*: these can only be reached via an
+// interface function (TelemetryScalar::*). If they access shared
+// state, they require the interface function to have acquired
+// |gTelemetryScalarMutex| to ensure thread safety.
+//
+// * Functions named TelemetryScalar::*. This is the external interface.
+// Entries and exits to these functions are serialised using
+// |gTelemetryScalarsMutex|.
+//
+// Avoiding races and deadlocks:
+//
+// All functions in the external interface (TelemetryScalar::*) are
+// serialised using the mutex |gTelemetryScalarsMutex|. This means
+// that the external interface is thread-safe. But it also brings
+// a danger of deadlock if any function in the external interface can
+// get back to that interface. That is, we will deadlock on any call
+// chain like this
+//
+// TelemetryScalar::* -> .. any functions .. -> TelemetryScalar::*
+//
+// To reduce the danger of that happening, observe the following rules:
+//
+// * No function in TelemetryScalar::* may directly call, nor take the
+// address of, any other function in TelemetryScalar::*.
+//
+// * No internal function internal_* may call, nor take the address
+// of, any function in TelemetryScalar::*.
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE TYPES
+
+namespace {
+
+const uint32_t kMaximumNumberOfKeys = 100;
+const uint32_t kMaxEventSummaryKeys = 500;
+const uint32_t kMaximumKeyStringLength = 72;
+const uint32_t kMaximumStringValueLength = 50;
+// The category and scalar name maximum lengths are used by the dynamic
+// scalar registration function and must match the constants used by
+// the 'parse_scalars.py' script for static scalars.
+const uint32_t kMaximumCategoryNameLength = 40;
+const uint32_t kMaximumScalarNameLength = 40;
+const uint32_t kScalarCount =
+ static_cast<uint32_t>(mozilla::Telemetry::ScalarID::ScalarCount);
+
+// To stop growing unbounded in memory while waiting for scalar deserialization
+// to finish, we immediately apply pending operations if the array reaches
+// a certain high water mark of elements.
+const size_t kScalarActionsArrayHighWaterMark = 10000;
+
+const char* TEST_SCALAR_PREFIX = "telemetry.test.";
+
+// The max offset supported by gScalarStoresTable for static scalars' stores.
+// Also the sentinel value (with store_count == 0) for just the sole "main"
+// store.
+const uint32_t kMaxStaticStoreOffset = UINT16_MAX;
+
+enum class ScalarResult : uint8_t {
+ // Nothing went wrong.
+ Ok,
+ // General Scalar Errors
+ NotInitialized,
+ CannotUnpackVariant,
+ CannotRecordInProcess,
+ CannotRecordDataset,
+ KeyedTypeMismatch,
+ UnknownScalar,
+ OperationNotSupported,
+ InvalidType,
+ InvalidValue,
+ // Keyed Scalar Errors
+ KeyIsEmpty,
+ KeyTooLong,
+ TooManyKeys,
+ KeyNotAllowed,
+ // String Scalar Errors
+ StringTooLong,
+ // Unsigned Scalar Errors
+ UnsignedNegativeValue,
+ UnsignedTruncatedValue,
+};
+
+// A common identifier for both built-in and dynamic scalars.
+struct ScalarKey {
+ uint32_t id;
+ bool dynamic;
+};
+
+// Dynamic scalar store names.
+StaticAutoPtr<nsTArray<RefPtr<nsAtom>>> gDynamicStoreNames;
+
+/**
+ * Scalar information for dynamic definitions.
+ */
+struct DynamicScalarInfo : BaseScalarInfo {
+ nsCString mDynamicName;
+ bool mDynamicExpiration;
+ uint32_t store_count;
+ uint32_t store_offset;
+
+ DynamicScalarInfo(uint32_t aKind, bool aRecordOnRelease, bool aExpired,
+ const nsACString& aName, bool aKeyed, bool aBuiltin,
+ const nsTArray<nsCString>& aStores)
+ : BaseScalarInfo(aKind,
+ aRecordOnRelease
+ ? nsITelemetry::DATASET_ALL_CHANNELS
+ : nsITelemetry::DATASET_PRERELEASE_CHANNELS,
+ RecordedProcessType::All, aKeyed, 0, 0,
+ GetCurrentProduct(), aBuiltin),
+ mDynamicName(aName),
+ mDynamicExpiration(aExpired) {
+ store_count = aStores.Length();
+ if (store_count == 0) {
+ store_count = 1;
+ store_offset = kMaxStaticStoreOffset;
+ } else {
+ store_offset = kMaxStaticStoreOffset + 1 + gDynamicStoreNames->Length();
+ for (const auto& storeName : aStores) {
+ gDynamicStoreNames->AppendElement(NS_Atomize(storeName));
+ }
+ MOZ_ASSERT(
+ gDynamicStoreNames->Length() < UINT32_MAX - kMaxStaticStoreOffset - 1,
+ "Too many dynamic scalar store names. Overflow.");
+ }
+ };
+
+ // The following functions will read the stored text
+ // instead of looking it up in the statically generated
+ // tables.
+ const char* name() const override;
+ const char* expiration() const override;
+
+ uint32_t storeCount() const override;
+ uint32_t storeOffset() const override;
+};
+
+const char* DynamicScalarInfo::name() const { return mDynamicName.get(); }
+
+const char* DynamicScalarInfo::expiration() const {
+ // Dynamic scalars can either be expired or not (boolean flag).
+ // Return an appropriate version string to leverage the scalar expiration
+ // logic.
+ return mDynamicExpiration ? "1.0" : "never";
+}
+
+uint32_t DynamicScalarInfo::storeOffset() const { return store_offset; }
+uint32_t DynamicScalarInfo::storeCount() const { return store_count; }
+
+typedef nsBaseHashtableET<nsDepCharHashKey, ScalarKey> CharPtrEntryType;
+typedef AutoHashtable<CharPtrEntryType> ScalarMapType;
+
+// Dynamic scalar definitions.
+StaticAutoPtr<nsTArray<DynamicScalarInfo>> gDynamicScalarInfo;
+
+const BaseScalarInfo& internal_GetScalarInfo(const StaticMutexAutoLock& lock,
+ const ScalarKey& aId) {
+ if (!aId.dynamic) {
+ return gScalars[aId.id];
+ }
+
+ return (*gDynamicScalarInfo)[aId.id];
+}
+
+bool IsValidEnumId(mozilla::Telemetry::ScalarID aID) {
+ return aID < mozilla::Telemetry::ScalarID::ScalarCount;
+}
+
+bool internal_IsValidId(const StaticMutexAutoLock& lock, const ScalarKey& aId) {
+ // Please note that this function needs to be called with the scalar
+ // mutex being acquired: other functions might be messing with
+ // |gDynamicScalarInfo|.
+ return aId.dynamic
+ ? (aId.id < gDynamicScalarInfo->Length())
+ : IsValidEnumId(static_cast<mozilla::Telemetry::ScalarID>(aId.id));
+}
+
+/**
+ * Convert a nsIVariant to a mozilla::Variant, which is used for
+ * accumulating child process scalars.
+ */
+ScalarResult GetVariantFromIVariant(nsIVariant* aInput, uint32_t aScalarKind,
+ mozilla::Maybe<ScalarVariant>& aOutput) {
+ switch (aScalarKind) {
+ case nsITelemetry::SCALAR_TYPE_COUNT: {
+ uint32_t val = 0;
+ nsresult rv = aInput->GetAsUint32(&val);
+ if (NS_FAILED(rv)) {
+ return ScalarResult::CannotUnpackVariant;
+ }
+ aOutput = mozilla::Some(mozilla::AsVariant(val));
+ break;
+ }
+ case nsITelemetry::SCALAR_TYPE_STRING: {
+ nsString val;
+ nsresult rv = aInput->GetAsAString(val);
+ if (NS_FAILED(rv)) {
+ return ScalarResult::CannotUnpackVariant;
+ }
+ aOutput = mozilla::Some(mozilla::AsVariant(val));
+ break;
+ }
+ case nsITelemetry::SCALAR_TYPE_BOOLEAN: {
+ bool val = false;
+ nsresult rv = aInput->GetAsBool(&val);
+ if (NS_FAILED(rv)) {
+ return ScalarResult::CannotUnpackVariant;
+ }
+ aOutput = mozilla::Some(mozilla::AsVariant(val));
+ break;
+ }
+ default:
+ MOZ_ASSERT(false, "Unknown scalar kind.");
+ return ScalarResult::UnknownScalar;
+ }
+ return ScalarResult::Ok;
+}
+
+/**
+ * Write a nsIVariant with a JSONWriter, used for GeckoView persistence.
+ */
+nsresult WriteVariantToJSONWriter(
+ uint32_t aScalarType, nsIVariant* aInputValue,
+ const mozilla::Span<const char>& aPropertyName,
+ mozilla::JSONWriter& aWriter) {
+ MOZ_ASSERT(aInputValue);
+
+ switch (aScalarType) {
+ case nsITelemetry::SCALAR_TYPE_COUNT: {
+ uint32_t val = 0;
+ nsresult rv = aInputValue->GetAsUint32(&val);
+ NS_ENSURE_SUCCESS(rv, rv);
+ aWriter.IntProperty(aPropertyName, val);
+ break;
+ }
+ case nsITelemetry::SCALAR_TYPE_STRING: {
+ nsCString val;
+ nsresult rv = aInputValue->GetAsACString(val);
+ NS_ENSURE_SUCCESS(rv, rv);
+ aWriter.StringProperty(aPropertyName, val);
+ break;
+ }
+ case nsITelemetry::SCALAR_TYPE_BOOLEAN: {
+ bool val = false;
+ nsresult rv = aInputValue->GetAsBool(&val);
+ NS_ENSURE_SUCCESS(rv, rv);
+ aWriter.BoolProperty(aPropertyName, val);
+ break;
+ }
+ default:
+ MOZ_ASSERT(false, "Unknown scalar kind.");
+ return NS_ERROR_FAILURE;
+ }
+
+ return NS_OK;
+}
+
+// Implements the methods for ScalarInfo.
+const char* ScalarInfo::name() const {
+ return &gScalarsStringTable[this->name_offset];
+}
+
+const char* ScalarInfo::expiration() const {
+ return &gScalarsStringTable[this->expiration_offset];
+}
+
+/**
+ * The base scalar object, that serves as a common ancestor for storage
+ * purposes.
+ */
+class ScalarBase {
+ public:
+ explicit ScalarBase(const BaseScalarInfo& aInfo)
+ : mStoreCount(aInfo.storeCount()),
+ mStoreOffset(aInfo.storeOffset()),
+ mStoreHasValue(mStoreCount),
+ mName(aInfo.name()) {
+ mStoreHasValue.SetLength(mStoreCount);
+ for (auto& val : mStoreHasValue) {
+ val = false;
+ }
+ };
+ virtual ~ScalarBase() = default;
+
+ // Set, Add and SetMaximum functions as described in the Telemetry IDL.
+ virtual ScalarResult SetValue(nsIVariant* aValue) = 0;
+ virtual ScalarResult AddValue(nsIVariant* aValue) {
+ return ScalarResult::OperationNotSupported;
+ }
+ virtual ScalarResult SetMaximum(nsIVariant* aValue) {
+ return ScalarResult::OperationNotSupported;
+ }
+
+ // Convenience methods used by the C++ API.
+ virtual void SetValue(uint32_t aValue) {
+ mozilla::Unused << HandleUnsupported();
+ }
+ virtual ScalarResult SetValue(const nsAString& aValue) {
+ return HandleUnsupported();
+ }
+ virtual void SetValue(bool aValue) { mozilla::Unused << HandleUnsupported(); }
+ virtual void AddValue(uint32_t aValue) {
+ mozilla::Unused << HandleUnsupported();
+ }
+ virtual void SetMaximum(uint32_t aValue) {
+ mozilla::Unused << HandleUnsupported();
+ }
+
+ // GetValue is used to get the value of the scalar when persisting it to JS.
+ virtual nsresult GetValue(const nsACString& aStoreName, bool aClearStore,
+ nsCOMPtr<nsIVariant>& aResult) = 0;
+
+ // To measure the memory stats.
+ size_t SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const;
+ virtual size_t SizeOfIncludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) const = 0;
+
+ protected:
+ bool HasValueInStore(size_t aStoreIndex) const;
+ void ClearValueInStore(size_t aStoreIndex);
+ void SetValueInStores();
+ nsresult StoreIndex(const nsACString& aStoreName, size_t* aStoreIndex) const;
+
+ private:
+ ScalarResult HandleUnsupported() const;
+
+ const uint32_t mStoreCount;
+ const uint32_t mStoreOffset;
+ nsTArray<bool> mStoreHasValue;
+
+ protected:
+ const nsCString mName;
+};
+
+ScalarResult ScalarBase::HandleUnsupported() const {
+ MOZ_ASSERT(false, "This operation is not support for this scalar type.");
+ return ScalarResult::OperationNotSupported;
+}
+
+bool ScalarBase::HasValueInStore(size_t aStoreIndex) const {
+ MOZ_ASSERT(aStoreIndex < mStoreHasValue.Length(),
+ "Invalid scalar store index.");
+ return mStoreHasValue[aStoreIndex];
+}
+
+void ScalarBase::ClearValueInStore(size_t aStoreIndex) {
+ MOZ_ASSERT(aStoreIndex < mStoreHasValue.Length(),
+ "Invalid scalar store index to clear.");
+ mStoreHasValue[aStoreIndex] = false;
+}
+
+void ScalarBase::SetValueInStores() {
+ for (auto& val : mStoreHasValue) {
+ val = true;
+ }
+}
+
+nsresult ScalarBase::StoreIndex(const nsACString& aStoreName,
+ size_t* aStoreIndex) const {
+ if (mStoreCount == 1 && mStoreOffset == kMaxStaticStoreOffset) {
+ // This Scalar is only in the "main" store.
+ if (aStoreName.EqualsLiteral("main")) {
+ *aStoreIndex = 0;
+ return NS_OK;
+ }
+ return NS_ERROR_NO_CONTENT;
+ }
+
+ // Multiple stores. Linear scan to find one that matches aStoreName.
+ // Dynamic Scalars start at kMaxStaticStoreOffset + 1
+ if (mStoreOffset > kMaxStaticStoreOffset) {
+ auto dynamicOffset = mStoreOffset - kMaxStaticStoreOffset - 1;
+ for (uint32_t i = 0; i < mStoreCount; ++i) {
+ auto scalarStore = (*gDynamicStoreNames)[dynamicOffset + i];
+ if (nsAtomCString(scalarStore).Equals(aStoreName)) {
+ *aStoreIndex = i;
+ return NS_OK;
+ }
+ }
+ return NS_ERROR_NO_CONTENT;
+ }
+
+ // Static Scalars are similar.
+ for (uint32_t i = 0; i < mStoreCount; ++i) {
+ uint32_t stringIndex = gScalarStoresTable[mStoreOffset + i];
+ if (aStoreName.EqualsASCII(&gScalarsStringTable[stringIndex])) {
+ *aStoreIndex = i;
+ return NS_OK;
+ }
+ }
+ return NS_ERROR_NO_CONTENT;
+}
+
+size_t ScalarBase::SizeOfExcludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) const {
+ return mStoreHasValue.ShallowSizeOfExcludingThis(aMallocSizeOf);
+}
+
+/**
+ * The implementation for the unsigned int scalar type.
+ */
+class ScalarUnsigned : public ScalarBase {
+ public:
+ using ScalarBase::SetValue;
+
+ explicit ScalarUnsigned(const BaseScalarInfo& aInfo)
+ : ScalarBase(aInfo), mStorage(aInfo.storeCount()) {
+ mStorage.SetLength(aInfo.storeCount());
+ for (auto& val : mStorage) {
+ val = 0;
+ }
+ };
+
+ ~ScalarUnsigned() override = default;
+
+ ScalarResult SetValue(nsIVariant* aValue) final;
+ void SetValue(uint32_t aValue) final;
+ ScalarResult AddValue(nsIVariant* aValue) final;
+ void AddValue(uint32_t aValue) final;
+ ScalarResult SetMaximum(nsIVariant* aValue) final;
+ void SetMaximum(uint32_t aValue) final;
+ nsresult GetValue(const nsACString& aStoreName, bool aClearStore,
+ nsCOMPtr<nsIVariant>& aResult) final;
+ size_t SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const final;
+
+ private:
+ nsTArray<uint32_t> mStorage;
+
+ ScalarResult CheckInput(nsIVariant* aValue);
+
+ // Prevent copying.
+ ScalarUnsigned(const ScalarUnsigned& aOther) = delete;
+ void operator=(const ScalarUnsigned& aOther) = delete;
+};
+
+ScalarResult ScalarUnsigned::SetValue(nsIVariant* aValue) {
+ ScalarResult sr = CheckInput(aValue);
+ if (sr == ScalarResult::UnsignedNegativeValue) {
+ return sr;
+ }
+
+ uint32_t value = 0;
+ if (NS_FAILED(aValue->GetAsUint32(&value))) {
+ return ScalarResult::InvalidValue;
+ }
+
+ SetValue(value);
+ return sr;
+}
+
+void ScalarUnsigned::SetValue(uint32_t aValue) {
+ if (GetCurrentProduct() == SupportedProduct::GeckoviewStreaming) {
+ GeckoViewStreamingTelemetry::UintScalarSet(mName, aValue);
+ return;
+ }
+ for (auto& val : mStorage) {
+ val = aValue;
+ }
+ SetValueInStores();
+}
+
+ScalarResult ScalarUnsigned::AddValue(nsIVariant* aValue) {
+ ScalarResult sr = CheckInput(aValue);
+ if (sr == ScalarResult::UnsignedNegativeValue) {
+ return sr;
+ }
+
+ uint32_t newAddend = 0;
+ nsresult rv = aValue->GetAsUint32(&newAddend);
+ if (NS_FAILED(rv)) {
+ return ScalarResult::InvalidValue;
+ }
+
+ AddValue(newAddend);
+ return sr;
+}
+
+void ScalarUnsigned::AddValue(uint32_t aValue) {
+ MOZ_ASSERT(GetCurrentProduct() != SupportedProduct::GeckoviewStreaming);
+ for (auto& val : mStorage) {
+ val += aValue;
+ }
+ SetValueInStores();
+}
+
+ScalarResult ScalarUnsigned::SetMaximum(nsIVariant* aValue) {
+ MOZ_ASSERT(GetCurrentProduct() != SupportedProduct::GeckoviewStreaming);
+ ScalarResult sr = CheckInput(aValue);
+ if (sr == ScalarResult::UnsignedNegativeValue) {
+ return sr;
+ }
+
+ uint32_t newValue = 0;
+ nsresult rv = aValue->GetAsUint32(&newValue);
+ if (NS_FAILED(rv)) {
+ return ScalarResult::InvalidValue;
+ }
+
+ SetMaximum(newValue);
+ return sr;
+}
+
+void ScalarUnsigned::SetMaximum(uint32_t aValue) {
+ for (auto& val : mStorage) {
+ if (aValue > val) {
+ val = aValue;
+ }
+ }
+ SetValueInStores();
+}
+
+nsresult ScalarUnsigned::GetValue(const nsACString& aStoreName,
+ bool aClearStore,
+ nsCOMPtr<nsIVariant>& aResult) {
+ size_t storeIndex = 0;
+ nsresult rv = StoreIndex(aStoreName, &storeIndex);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ if (!HasValueInStore(storeIndex)) {
+ return NS_ERROR_NO_CONTENT;
+ }
+ nsCOMPtr<nsIWritableVariant> outVar(new nsVariant());
+ rv = outVar->SetAsUint32(mStorage[storeIndex]);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ aResult = std::move(outVar);
+ if (aClearStore) {
+ mStorage[storeIndex] = 0;
+ ClearValueInStore(storeIndex);
+ }
+ return NS_OK;
+}
+
+size_t ScalarUnsigned::SizeOfIncludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) const {
+ size_t n = aMallocSizeOf(this);
+ n += ScalarBase::SizeOfExcludingThis(aMallocSizeOf);
+ n += mStorage.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ return n;
+}
+
+ScalarResult ScalarUnsigned::CheckInput(nsIVariant* aValue) {
+ // If this is a floating point value/double, we will probably get truncated.
+ uint16_t type = aValue->GetDataType();
+ if (type == nsIDataType::VTYPE_FLOAT || type == nsIDataType::VTYPE_DOUBLE) {
+ return ScalarResult::UnsignedTruncatedValue;
+ }
+
+ int32_t signedTest;
+ // If we're able to cast the number to an int, check its sign.
+ // Warn the user if he's trying to set the unsigned scalar to a negative
+ // number.
+ if (NS_SUCCEEDED(aValue->GetAsInt32(&signedTest)) && signedTest < 0) {
+ return ScalarResult::UnsignedNegativeValue;
+ }
+ return ScalarResult::Ok;
+}
+
+/**
+ * The implementation for the string scalar type.
+ */
+class ScalarString : public ScalarBase {
+ public:
+ using ScalarBase::SetValue;
+
+ explicit ScalarString(const BaseScalarInfo& aInfo)
+ : ScalarBase(aInfo), mStorage(aInfo.storeCount()) {
+ mStorage.SetLength(aInfo.storeCount());
+ };
+
+ ~ScalarString() override = default;
+
+ ScalarResult SetValue(nsIVariant* aValue) final;
+ ScalarResult SetValue(const nsAString& aValue) final;
+ nsresult GetValue(const nsACString& aStoreName, bool aClearStore,
+ nsCOMPtr<nsIVariant>& aResult) final;
+ size_t SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const final;
+
+ private:
+ nsTArray<nsString> mStorage;
+
+ // Prevent copying.
+ ScalarString(const ScalarString& aOther) = delete;
+ void operator=(const ScalarString& aOther) = delete;
+};
+
+ScalarResult ScalarString::SetValue(nsIVariant* aValue) {
+ // Check that we got the correct data type.
+ uint16_t type = aValue->GetDataType();
+ if (type != nsIDataType::VTYPE_CHAR && type != nsIDataType::VTYPE_WCHAR &&
+ type != nsIDataType::VTYPE_CHAR_STR &&
+ type != nsIDataType::VTYPE_WCHAR_STR &&
+ type != nsIDataType::VTYPE_STRING_SIZE_IS &&
+ type != nsIDataType::VTYPE_WSTRING_SIZE_IS &&
+ type != nsIDataType::VTYPE_UTF8STRING &&
+ type != nsIDataType::VTYPE_CSTRING &&
+ type != nsIDataType::VTYPE_ASTRING) {
+ return ScalarResult::InvalidType;
+ }
+
+ nsAutoString convertedString;
+ nsresult rv = aValue->GetAsAString(convertedString);
+ if (NS_FAILED(rv)) {
+ return ScalarResult::InvalidValue;
+ }
+ return SetValue(convertedString);
+};
+
+ScalarResult ScalarString::SetValue(const nsAString& aValue) {
+ auto str = Substring(aValue, 0, kMaximumStringValueLength);
+ if (GetCurrentProduct() == SupportedProduct::GeckoviewStreaming) {
+ GeckoViewStreamingTelemetry::StringScalarSet(mName,
+ NS_ConvertUTF16toUTF8(str));
+ return aValue.Length() > kMaximumStringValueLength
+ ? ScalarResult::StringTooLong
+ : ScalarResult::Ok;
+ }
+ for (auto& val : mStorage) {
+ val.Assign(str);
+ }
+ SetValueInStores();
+ if (aValue.Length() > kMaximumStringValueLength) {
+ return ScalarResult::StringTooLong;
+ }
+ return ScalarResult::Ok;
+}
+
+nsresult ScalarString::GetValue(const nsACString& aStoreName, bool aClearStore,
+ nsCOMPtr<nsIVariant>& aResult) {
+ nsCOMPtr<nsIWritableVariant> outVar(new nsVariant());
+ size_t storeIndex = 0;
+ nsresult rv = StoreIndex(aStoreName, &storeIndex);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ if (!HasValueInStore(storeIndex)) {
+ return NS_ERROR_NO_CONTENT;
+ }
+ rv = outVar->SetAsAString(mStorage[storeIndex]);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ if (aClearStore) {
+ ClearValueInStore(storeIndex);
+ }
+ aResult = std::move(outVar);
+ return NS_OK;
+}
+
+size_t ScalarString::SizeOfIncludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) const {
+ size_t n = aMallocSizeOf(this);
+ n += ScalarBase::SizeOfExcludingThis(aMallocSizeOf);
+ n += mStorage.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ for (auto& val : mStorage) {
+ n += val.SizeOfExcludingThisIfUnshared(aMallocSizeOf);
+ }
+ return n;
+}
+
+/**
+ * The implementation for the boolean scalar type.
+ */
+class ScalarBoolean : public ScalarBase {
+ public:
+ using ScalarBase::SetValue;
+
+ explicit ScalarBoolean(const BaseScalarInfo& aInfo)
+ : ScalarBase(aInfo), mStorage(aInfo.storeCount()) {
+ mStorage.SetLength(aInfo.storeCount());
+ for (auto& val : mStorage) {
+ val = false;
+ }
+ };
+
+ ~ScalarBoolean() override = default;
+
+ ScalarResult SetValue(nsIVariant* aValue) final;
+ void SetValue(bool aValue) final;
+ nsresult GetValue(const nsACString& aStoreName, bool aClearStore,
+ nsCOMPtr<nsIVariant>& aResult) final;
+ size_t SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const final;
+
+ private:
+ nsTArray<bool> mStorage;
+
+ // Prevent copying.
+ ScalarBoolean(const ScalarBoolean& aOther) = delete;
+ void operator=(const ScalarBoolean& aOther) = delete;
+};
+
+ScalarResult ScalarBoolean::SetValue(nsIVariant* aValue) {
+ // Check that we got the correct data type.
+ uint16_t type = aValue->GetDataType();
+ if (type != nsIDataType::VTYPE_BOOL && type != nsIDataType::VTYPE_INT8 &&
+ type != nsIDataType::VTYPE_INT16 && type != nsIDataType::VTYPE_INT32 &&
+ type != nsIDataType::VTYPE_INT64 && type != nsIDataType::VTYPE_UINT8 &&
+ type != nsIDataType::VTYPE_UINT16 && type != nsIDataType::VTYPE_UINT32 &&
+ type != nsIDataType::VTYPE_UINT64) {
+ return ScalarResult::InvalidType;
+ }
+
+ bool value = false;
+ if (NS_FAILED(aValue->GetAsBool(&value))) {
+ return ScalarResult::InvalidValue;
+ }
+ SetValue(value);
+ return ScalarResult::Ok;
+};
+
+void ScalarBoolean::SetValue(bool aValue) {
+ if (GetCurrentProduct() == SupportedProduct::GeckoviewStreaming) {
+ GeckoViewStreamingTelemetry::BoolScalarSet(mName, aValue);
+ return;
+ }
+ for (auto& val : mStorage) {
+ val = aValue;
+ }
+ SetValueInStores();
+}
+
+nsresult ScalarBoolean::GetValue(const nsACString& aStoreName, bool aClearStore,
+ nsCOMPtr<nsIVariant>& aResult) {
+ nsCOMPtr<nsIWritableVariant> outVar(new nsVariant());
+ size_t storeIndex = 0;
+ nsresult rv = StoreIndex(aStoreName, &storeIndex);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ if (!HasValueInStore(storeIndex)) {
+ return NS_ERROR_NO_CONTENT;
+ }
+ if (aClearStore) {
+ ClearValueInStore(storeIndex);
+ }
+ rv = outVar->SetAsBool(mStorage[storeIndex]);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ aResult = std::move(outVar);
+ return NS_OK;
+}
+
+size_t ScalarBoolean::SizeOfIncludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) const {
+ size_t n = aMallocSizeOf(this);
+ n += ScalarBase::SizeOfExcludingThis(aMallocSizeOf);
+ n += mStorage.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ return n;
+}
+
+/**
+ * Allocate a scalar class given the scalar info.
+ *
+ * @param aInfo The informations for the scalar coming from the definition file.
+ * @return nullptr if the scalar type is unknown, otherwise a valid pointer to
+ * the scalar type.
+ */
+ScalarBase* internal_ScalarAllocate(const BaseScalarInfo& aInfo) {
+ ScalarBase* scalar = nullptr;
+ switch (aInfo.kind) {
+ case nsITelemetry::SCALAR_TYPE_COUNT:
+ scalar = new ScalarUnsigned(aInfo);
+ break;
+ case nsITelemetry::SCALAR_TYPE_STRING:
+ scalar = new ScalarString(aInfo);
+ break;
+ case nsITelemetry::SCALAR_TYPE_BOOLEAN:
+ scalar = new ScalarBoolean(aInfo);
+ break;
+ default:
+ MOZ_ASSERT(false, "Invalid scalar type");
+ }
+ return scalar;
+}
+
+/**
+ * The implementation for the keyed scalar type.
+ */
+class KeyedScalar {
+ public:
+ typedef std::pair<nsCString, nsCOMPtr<nsIVariant>> KeyValuePair;
+
+ // We store the name instead of a reference to the BaseScalarInfo because
+ // the BaseScalarInfo can move if it's from a dynamic scalar.
+ explicit KeyedScalar(const BaseScalarInfo& info)
+ : mScalarName(info.name()),
+ mScalarKeyCount(info.key_count),
+ mScalarKeyOffset(info.key_offset),
+ mMaximumNumberOfKeys(kMaximumNumberOfKeys){};
+ ~KeyedScalar() = default;
+
+ // Set, Add and SetMaximum functions as described in the Telemetry IDL.
+ // These methods implicitly instantiate a Scalar[*] for each key.
+ ScalarResult SetValue(const StaticMutexAutoLock& locker,
+ const nsAString& aKey, nsIVariant* aValue);
+ ScalarResult AddValue(const StaticMutexAutoLock& locker,
+ const nsAString& aKey, nsIVariant* aValue);
+ ScalarResult SetMaximum(const StaticMutexAutoLock& locker,
+ const nsAString& aKey, nsIVariant* aValue);
+
+ // Convenience methods used by the C++ API.
+ void SetValue(const StaticMutexAutoLock& locker, const nsAString& aKey,
+ uint32_t aValue);
+ void SetValue(const StaticMutexAutoLock& locker, const nsAString& aKey,
+ bool aValue);
+ void AddValue(const StaticMutexAutoLock& locker, const nsAString& aKey,
+ uint32_t aValue);
+ void SetMaximum(const StaticMutexAutoLock& locker, const nsAString& aKey,
+ uint32_t aValue);
+
+ // GetValue is used to get the key-value pairs stored in the keyed scalar
+ // when persisting it to JS.
+ nsresult GetValue(const nsACString& aStoreName, bool aClearStorage,
+ nsTArray<KeyValuePair>& aValues);
+
+ // To measure the memory stats.
+ size_t SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf);
+
+ // To permit more keys than normal.
+ void SetMaximumNumberOfKeys(uint32_t aMaximumNumberOfKeys) {
+ mMaximumNumberOfKeys = aMaximumNumberOfKeys;
+ };
+
+ private:
+ typedef nsClassHashtable<nsCStringHashKey, ScalarBase> ScalarKeysMapType;
+
+ const nsCString mScalarName;
+ ScalarKeysMapType mScalarKeys;
+ uint32_t mScalarKeyCount;
+ uint32_t mScalarKeyOffset;
+ uint32_t mMaximumNumberOfKeys;
+
+ ScalarResult GetScalarForKey(const StaticMutexAutoLock& locker,
+ const nsAString& aKey, ScalarBase** aRet);
+
+ bool AllowsKey(const nsAString& aKey) const;
+};
+
+ScalarResult KeyedScalar::SetValue(const StaticMutexAutoLock& locker,
+ const nsAString& aKey, nsIVariant* aValue) {
+ ScalarBase* scalar = nullptr;
+ ScalarResult sr = GetScalarForKey(locker, aKey, &scalar);
+ if (sr != ScalarResult::Ok) {
+ return sr;
+ }
+
+ return scalar->SetValue(aValue);
+}
+
+ScalarResult KeyedScalar::AddValue(const StaticMutexAutoLock& locker,
+ const nsAString& aKey, nsIVariant* aValue) {
+ ScalarBase* scalar = nullptr;
+ ScalarResult sr = GetScalarForKey(locker, aKey, &scalar);
+ if (sr != ScalarResult::Ok) {
+ return sr;
+ }
+
+ return scalar->AddValue(aValue);
+}
+
+ScalarResult KeyedScalar::SetMaximum(const StaticMutexAutoLock& locker,
+ const nsAString& aKey,
+ nsIVariant* aValue) {
+ ScalarBase* scalar = nullptr;
+ ScalarResult sr = GetScalarForKey(locker, aKey, &scalar);
+ if (sr != ScalarResult::Ok) {
+ return sr;
+ }
+
+ return scalar->SetMaximum(aValue);
+}
+
+void KeyedScalar::SetValue(const StaticMutexAutoLock& locker,
+ const nsAString& aKey, uint32_t aValue) {
+ ScalarBase* scalar = nullptr;
+ ScalarResult sr = GetScalarForKey(locker, aKey, &scalar);
+
+ if (sr != ScalarResult::Ok) {
+ // Bug 1451813 - We now report which scalars exceed the key limit in
+ // telemetry.keyed_scalars_exceed_limit.
+ if (sr == ScalarResult::KeyTooLong) {
+ MOZ_ASSERT(false, "Key too long to be recorded in the scalar.");
+ }
+ return;
+ }
+
+ return scalar->SetValue(aValue);
+}
+
+void KeyedScalar::SetValue(const StaticMutexAutoLock& locker,
+ const nsAString& aKey, bool aValue) {
+ ScalarBase* scalar = nullptr;
+ ScalarResult sr = GetScalarForKey(locker, aKey, &scalar);
+
+ if (sr != ScalarResult::Ok) {
+ // Bug 1451813 - We now report which scalars exceed the key limit in
+ // telemetry.keyed_scalars_exceed_limit.
+ if (sr == ScalarResult::KeyTooLong) {
+ MOZ_ASSERT(false, "Key too long to be recorded in the scalar.");
+ }
+ return;
+ }
+
+ return scalar->SetValue(aValue);
+}
+
+void KeyedScalar::AddValue(const StaticMutexAutoLock& locker,
+ const nsAString& aKey, uint32_t aValue) {
+ ScalarBase* scalar = nullptr;
+ ScalarResult sr = GetScalarForKey(locker, aKey, &scalar);
+
+ if (sr != ScalarResult::Ok) {
+ // Bug 1451813 - We now report which scalars exceed the key limit in
+ // telemetry.keyed_scalars_exceed_limit.
+ if (sr == ScalarResult::KeyTooLong) {
+ MOZ_ASSERT(false, "Key too long to be recorded in the scalar.");
+ }
+ return;
+ }
+
+ return scalar->AddValue(aValue);
+}
+
+void KeyedScalar::SetMaximum(const StaticMutexAutoLock& locker,
+ const nsAString& aKey, uint32_t aValue) {
+ ScalarBase* scalar = nullptr;
+ ScalarResult sr = GetScalarForKey(locker, aKey, &scalar);
+
+ if (sr != ScalarResult::Ok) {
+ // Bug 1451813 - We now report which scalars exceed the key limit in
+ // telemetry.keyed_scalars_exceed_limit.
+ if (sr == ScalarResult::KeyTooLong) {
+ MOZ_ASSERT(false, "Key too long to be recorded in the scalar.");
+ }
+
+ return;
+ }
+
+ return scalar->SetMaximum(aValue);
+}
+
+/**
+ * Get a key-value array with the values for the Keyed Scalar.
+ * @param aValue The array that will hold the key-value pairs.
+ * @return {nsresult} NS_OK or an error value as reported by the
+ * the specific scalar objects implementations (e.g.
+ * ScalarUnsigned).
+ */
+nsresult KeyedScalar::GetValue(const nsACString& aStoreName, bool aClearStorage,
+ nsTArray<KeyValuePair>& aValues) {
+ for (auto iter = mScalarKeys.ConstIter(); !iter.Done(); iter.Next()) {
+ ScalarBase* scalar = iter.UserData();
+
+ // Get the scalar value.
+ nsCOMPtr<nsIVariant> scalarValue;
+ nsresult rv = scalar->GetValue(aStoreName, aClearStorage, scalarValue);
+ if (rv == NS_ERROR_NO_CONTENT) {
+ // No value for this store.
+ continue;
+ }
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ // Append it to value list.
+ aValues.AppendElement(std::make_pair(nsCString(iter.Key()), scalarValue));
+ }
+
+ return NS_OK;
+}
+
+// Forward declaration
+nsresult internal_GetKeyedScalarByEnum(const StaticMutexAutoLock& lock,
+ const ScalarKey& aId,
+ ProcessID aProcessStorage,
+ KeyedScalar** aRet);
+
+// Forward declaration
+nsresult internal_GetEnumByScalarName(const StaticMutexAutoLock& lock,
+ const nsACString& aName, ScalarKey* aId);
+
+/**
+ * Get the scalar for the referenced key.
+ * If there's no such key, instantiate a new Scalar object with the
+ * same type of the Keyed scalar and create the key.
+ */
+ScalarResult KeyedScalar::GetScalarForKey(const StaticMutexAutoLock& locker,
+ const nsAString& aKey,
+ ScalarBase** aRet) {
+ if (aKey.IsEmpty()) {
+ return ScalarResult::KeyIsEmpty;
+ }
+
+ if (!AllowsKey(aKey)) {
+ KeyedScalar* scalarUnknown = nullptr;
+ ScalarKey scalarUnknownUniqueId{
+ static_cast<uint32_t>(
+ mozilla::Telemetry::ScalarID::TELEMETRY_KEYED_SCALARS_UNKNOWN_KEYS),
+ false};
+ ProcessID process = ProcessID::Parent;
+ nsresult rv = internal_GetKeyedScalarByEnum(locker, scalarUnknownUniqueId,
+ process, &scalarUnknown);
+ if (NS_FAILED(rv)) {
+ return ScalarResult::TooManyKeys;
+ }
+ scalarUnknown->AddValue(locker, NS_ConvertUTF8toUTF16(mScalarName), 1);
+
+ return ScalarResult::KeyNotAllowed;
+ }
+
+ if (aKey.Length() > kMaximumKeyStringLength) {
+ return ScalarResult::KeyTooLong;
+ }
+
+ NS_ConvertUTF16toUTF8 utf8Key(aKey);
+
+ ScalarBase* scalar = nullptr;
+ if (mScalarKeys.Get(utf8Key, &scalar)) {
+ *aRet = scalar;
+ return ScalarResult::Ok;
+ }
+
+ ScalarKey uniqueId;
+ nsresult rv = internal_GetEnumByScalarName(locker, mScalarName, &uniqueId);
+ if (NS_FAILED(rv)) {
+ return (rv == NS_ERROR_FAILURE) ? ScalarResult::NotInitialized
+ : ScalarResult::UnknownScalar;
+ }
+
+ const BaseScalarInfo& info = internal_GetScalarInfo(locker, uniqueId);
+ if (mScalarKeys.Count() >= mMaximumNumberOfKeys) {
+ if (aKey.EqualsLiteral("telemetry.keyed_scalars_exceed_limit")) {
+ return ScalarResult::TooManyKeys;
+ }
+
+ KeyedScalar* scalarExceed = nullptr;
+
+ ScalarKey uniqueId{
+ static_cast<uint32_t>(
+ mozilla::Telemetry::ScalarID::TELEMETRY_KEYED_SCALARS_EXCEED_LIMIT),
+ false};
+
+ ProcessID process = ProcessID::Parent;
+ nsresult rv =
+ internal_GetKeyedScalarByEnum(locker, uniqueId, process, &scalarExceed);
+
+ if (NS_FAILED(rv)) {
+ return ScalarResult::TooManyKeys;
+ }
+
+ scalarExceed->AddValue(locker, NS_ConvertUTF8toUTF16(info.name()), 1);
+
+ return ScalarResult::TooManyKeys;
+ }
+
+ scalar = internal_ScalarAllocate(info);
+ if (!scalar) {
+ return ScalarResult::InvalidType;
+ }
+
+ mScalarKeys.Put(utf8Key, scalar);
+
+ *aRet = scalar;
+ return ScalarResult::Ok;
+}
+
+size_t KeyedScalar::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) {
+ size_t n = aMallocSizeOf(this);
+ for (auto iter = mScalarKeys.Iter(); !iter.Done(); iter.Next()) {
+ ScalarBase* scalar = iter.UserData();
+ n += scalar->SizeOfIncludingThis(aMallocSizeOf);
+ }
+ return n;
+}
+
+bool KeyedScalar::AllowsKey(const nsAString& aKey) const {
+ // If we didn't specify a list of allowed keys, just return true.
+ if (mScalarKeyCount == 0) {
+ return true;
+ }
+
+ for (uint32_t i = 0; i < mScalarKeyCount; ++i) {
+ uint32_t stringIndex = gScalarKeysTable[mScalarKeyOffset + i];
+ if (aKey.EqualsASCII(&gScalarsStringTable[stringIndex])) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+typedef nsUint32HashKey ScalarIDHashKey;
+typedef nsUint32HashKey ProcessIDHashKey;
+typedef nsClassHashtable<ScalarIDHashKey, ScalarBase> ScalarStorageMapType;
+typedef nsClassHashtable<ScalarIDHashKey, KeyedScalar>
+ KeyedScalarStorageMapType;
+typedef nsClassHashtable<ProcessIDHashKey, ScalarStorageMapType>
+ ProcessesScalarsMapType;
+typedef nsClassHashtable<ProcessIDHashKey, KeyedScalarStorageMapType>
+ ProcessesKeyedScalarsMapType;
+
+typedef mozilla::Tuple<const char*, nsCOMPtr<nsIVariant>, uint32_t>
+ ScalarDataTuple;
+typedef nsTArray<ScalarDataTuple> ScalarTupleArray;
+typedef nsDataHashtable<ProcessIDHashKey, ScalarTupleArray> ScalarSnapshotTable;
+
+typedef mozilla::Tuple<const char*, nsTArray<KeyedScalar::KeyValuePair>,
+ uint32_t>
+ KeyedScalarDataTuple;
+typedef nsTArray<KeyedScalarDataTuple> KeyedScalarTupleArray;
+typedef nsDataHashtable<ProcessIDHashKey, KeyedScalarTupleArray>
+ KeyedScalarSnapshotTable;
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE STATE, SHARED BY ALL THREADS
+
+namespace {
+
+// Set to true once this global state has been initialized.
+bool gInitDone = false;
+
+bool gCanRecordBase;
+bool gCanRecordExtended;
+
+// The Name -> ID cache map.
+ScalarMapType gScalarNameIDMap(kScalarCount);
+
+// The (Process Id -> (Scalar ID -> Scalar Object)) map. This is a
+// nsClassHashtable, it owns the scalar instances and takes care of deallocating
+// them when they are removed from the map.
+ProcessesScalarsMapType gScalarStorageMap;
+// As above, for the keyed scalars.
+ProcessesKeyedScalarsMapType gKeyedScalarStorageMap;
+// Provide separate storage for "dynamic builtin" plain and keyed scalars,
+// needed to support "build faster" in local developer builds.
+ProcessesScalarsMapType gDynamicBuiltinScalarStorageMap;
+ProcessesKeyedScalarsMapType gDynamicBuiltinKeyedScalarStorageMap;
+
+// Whether or not the deserialization of persisted scalars is still in progress.
+// This is never the case on Desktop or Fennec.
+// Only GeckoView restores persisted scalars.
+bool gIsDeserializing = false;
+// This batches scalar accumulations that should be applied once loading
+// finished.
+StaticAutoPtr<nsTArray<ScalarAction>> gScalarsActions;
+StaticAutoPtr<nsTArray<KeyedScalarAction>> gKeyedScalarsActions;
+
+bool internal_IsScalarDeserializing(const StaticMutexAutoLock& lock) {
+ return gIsDeserializing;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: Function that may call JS code.
+
+// NOTE: the functions in this section all run without protection from
+// |gTelemetryScalarsMutex|. If they held the mutex, there would be the
+// possibility of deadlock because the JS_ calls that they make may call
+// back into the TelemetryScalar interface, hence trying to re-acquire the
+// mutex.
+//
+// This means that these functions potentially race against threads, but
+// that seems preferable to risking deadlock.
+
+namespace {
+
+/**
+ * Converts the error code to a human readable error message and prints it to
+ * the browser console.
+ *
+ * @param aScalarName The name of the scalar that raised the error.
+ * @param aSr The error code.
+ */
+void internal_LogScalarError(const nsACString& aScalarName, ScalarResult aSr) {
+ nsAutoString errorMessage;
+ AppendUTF8toUTF16(aScalarName, errorMessage);
+
+ switch (aSr) {
+ case ScalarResult::NotInitialized:
+ errorMessage.AppendLiteral(u" - Telemetry was not yet initialized.");
+ break;
+ case ScalarResult::CannotUnpackVariant:
+ errorMessage.AppendLiteral(
+ u" - Cannot convert the provided JS value to nsIVariant.");
+ break;
+ case ScalarResult::CannotRecordInProcess:
+ errorMessage.AppendLiteral(
+ u" - Cannot record the scalar in the current process.");
+ break;
+ case ScalarResult::KeyedTypeMismatch:
+ errorMessage.AppendLiteral(
+ u" - Attempting to manage a keyed scalar as a scalar (or "
+ u"vice-versa).");
+ break;
+ case ScalarResult::UnknownScalar:
+ errorMessage.AppendLiteral(u" - Unknown scalar.");
+ break;
+ case ScalarResult::OperationNotSupported:
+ errorMessage.AppendLiteral(
+ u" - The requested operation is not supported on this scalar.");
+ break;
+ case ScalarResult::InvalidType:
+ errorMessage.AppendLiteral(
+ u" - Attempted to set the scalar to an invalid data type.");
+ break;
+ case ScalarResult::InvalidValue:
+ errorMessage.AppendLiteral(
+ u" - Attempted to set the scalar to an incompatible value.");
+ break;
+ case ScalarResult::StringTooLong:
+ AppendUTF8toUTF16(
+ nsPrintfCString(" - Truncating scalar value to %d characters.",
+ kMaximumStringValueLength),
+ errorMessage);
+ break;
+ case ScalarResult::KeyIsEmpty:
+ errorMessage.AppendLiteral(u" - The key must not be empty.");
+ break;
+ case ScalarResult::KeyTooLong:
+ AppendUTF8toUTF16(
+ nsPrintfCString(" - The key length must be limited to %d characters.",
+ kMaximumKeyStringLength),
+ errorMessage);
+ break;
+ case ScalarResult::KeyNotAllowed:
+ AppendUTF8toUTF16(
+ nsPrintfCString(" - The key is not allowed for this scalar."),
+ errorMessage);
+ break;
+ case ScalarResult::TooManyKeys:
+ AppendUTF8toUTF16(
+ nsPrintfCString(" - Keyed scalars cannot have more than %d keys.",
+ kMaximumNumberOfKeys),
+ errorMessage);
+ break;
+ case ScalarResult::UnsignedNegativeValue:
+ errorMessage.AppendLiteral(
+ u" - Trying to set an unsigned scalar to a negative number.");
+ break;
+ case ScalarResult::UnsignedTruncatedValue:
+ errorMessage.AppendLiteral(u" - Truncating float/double number.");
+ break;
+ default:
+ // Nothing.
+ return;
+ }
+
+ LogToBrowserConsole(nsIScriptError::warningFlag, errorMessage);
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: helpers for the external interface
+
+namespace {
+
+bool internal_CanRecordBase(const StaticMutexAutoLock& lock) {
+ return gCanRecordBase;
+}
+
+bool internal_CanRecordExtended(const StaticMutexAutoLock& lock) {
+ return gCanRecordExtended;
+}
+
+/**
+ * Check if the given scalar is a keyed scalar.
+ *
+ * @param lock Instance of a lock locking gTelemetryHistogramMutex
+ * @param aId The scalar identifier.
+ * @return true if aId refers to a keyed scalar, false otherwise.
+ */
+bool internal_IsKeyedScalar(const StaticMutexAutoLock& lock,
+ const ScalarKey& aId) {
+ return internal_GetScalarInfo(lock, aId).keyed;
+}
+
+/**
+ * Check if we're allowed to record the given scalar in the current
+ * process.
+ *
+ * @param lock Instance of a lock locking gTelemetryHistogramMutex
+ * @param aId The scalar identifier.
+ * @return true if the scalar is allowed to be recorded in the current process,
+ * false otherwise.
+ */
+bool internal_CanRecordProcess(const StaticMutexAutoLock& lock,
+ const ScalarKey& aId) {
+ const BaseScalarInfo& info = internal_GetScalarInfo(lock, aId);
+ return CanRecordInProcess(info.record_in_processes, XRE_GetProcessType());
+}
+
+bool internal_CanRecordProduct(const StaticMutexAutoLock& lock,
+ const ScalarKey& aId) {
+ const BaseScalarInfo& info = internal_GetScalarInfo(lock, aId);
+ return CanRecordProduct(info.products);
+}
+
+bool internal_CanRecordForScalarID(const StaticMutexAutoLock& lock,
+ const ScalarKey& aId) {
+ // Get the scalar info from the id.
+ const BaseScalarInfo& info = internal_GetScalarInfo(lock, aId);
+
+ // Can we record at all?
+ bool canRecordBase = internal_CanRecordBase(lock);
+ if (!canRecordBase) {
+ return false;
+ }
+
+ bool canRecordDataset = CanRecordDataset(info.dataset, canRecordBase,
+ internal_CanRecordExtended(lock));
+ if (!canRecordDataset) {
+ return false;
+ }
+
+ return true;
+}
+
+/**
+ * Check if we are allowed to record the provided scalar.
+ *
+ * @param lock Instance of a lock locking gTelemetryHistogramMutex
+ * @param aId The scalar identifier.
+ * @param aKeyed Are we attempting to write a keyed scalar?
+ * @param aForce Whether to allow recording even if the probe is not allowed on
+ * the current process.
+ * This must only be true for GeckoView persistence and recorded
+ * actions.
+ * @return ScalarResult::Ok if we can record, an error code otherwise.
+ */
+ScalarResult internal_CanRecordScalar(const StaticMutexAutoLock& lock,
+ const ScalarKey& aId, bool aKeyed,
+ bool aForce = false) {
+ // Make sure that we have a keyed scalar if we are trying to change one.
+ if (internal_IsKeyedScalar(lock, aId) != aKeyed) {
+ return ScalarResult::KeyedTypeMismatch;
+ }
+
+ // Are we allowed to record this scalar based on the current Telemetry
+ // settings?
+ if (!internal_CanRecordForScalarID(lock, aId)) {
+ return ScalarResult::CannotRecordDataset;
+ }
+
+ // Can we record in this process?
+ if (!aForce && !internal_CanRecordProcess(lock, aId)) {
+ return ScalarResult::CannotRecordInProcess;
+ }
+
+ // Can we record on this product?
+ if (!internal_CanRecordProduct(lock, aId)) {
+ return ScalarResult::CannotRecordDataset;
+ }
+
+ return ScalarResult::Ok;
+}
+
+/**
+ * Get the scalar enum id from the scalar name.
+ *
+ * @param lock Instance of a lock locking gTelemetryHistogramMutex
+ * @param aName The scalar name.
+ * @param aId The output variable to contain the enum.
+ * @return
+ * NS_ERROR_FAILURE if this was called before init is completed.
+ * NS_ERROR_INVALID_ARG if the name can't be found in the scalar definitions.
+ * NS_OK if the scalar was found and aId contains a valid enum id.
+ */
+nsresult internal_GetEnumByScalarName(const StaticMutexAutoLock& lock,
+ const nsACString& aName, ScalarKey* aId) {
+ if (!gInitDone) {
+ return NS_ERROR_FAILURE;
+ }
+
+ CharPtrEntryType* entry =
+ gScalarNameIDMap.GetEntry(PromiseFlatCString(aName).get());
+ if (!entry) {
+ return NS_ERROR_INVALID_ARG;
+ }
+ *aId = entry->GetData();
+ return NS_OK;
+}
+
+/**
+ * Get a scalar object by its enum id. This implicitly allocates the scalar
+ * object in the storage if it wasn't previously allocated.
+ *
+ * @param lock Instance of a lock locking gTelemetryHistogramMutex
+ * @param aId The scalar identifier.
+ * @param aProcessStorage This drives the selection of the map to use to store
+ * the scalar data coming from child processes. This is only meaningful
+ * when this function is called in parent process. If that's the case,
+ * if this is not |GeckoProcessType_Default|, the process id is used to
+ * allocate and store the scalars.
+ * @param aRes The output variable that stores scalar object.
+ * @return
+ * NS_ERROR_INVALID_ARG if the scalar id is unknown.
+ * NS_ERROR_NOT_AVAILABLE if the scalar is expired.
+ * NS_OK if the scalar was found. If that's the case, aResult contains a
+ * valid pointer to a scalar type.
+ */
+nsresult internal_GetScalarByEnum(const StaticMutexAutoLock& lock,
+ const ScalarKey& aId,
+ ProcessID aProcessStorage,
+ ScalarBase** aRet) {
+ if (!internal_IsValidId(lock, aId)) {
+ MOZ_ASSERT(false, "Requested a scalar with an invalid id.");
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ const BaseScalarInfo& info = internal_GetScalarInfo(lock, aId);
+
+ // Dynamic scalars fixup: they are always stored in the "dynamic" process,
+ // unless they are part of the "builtin" Firefox probes. Please note that
+ // "dynamic builtin" probes are meant to support "artifact" and "build faster"
+ // builds.
+ if (aId.dynamic && !info.builtin) {
+ aProcessStorage = ProcessID::Dynamic;
+ }
+
+ ScalarBase* scalar = nullptr;
+ ScalarStorageMapType* scalarStorage = nullptr;
+ // Initialize the scalar storage to the parent storage. This will get
+ // set to the child storage if needed.
+ uint32_t storageId = static_cast<uint32_t>(aProcessStorage);
+
+ // Put dynamic-builtin scalars (used to support "build faster") in a
+ // separate storage.
+ ProcessesScalarsMapType& processStorage =
+ (aId.dynamic && info.builtin) ? gDynamicBuiltinScalarStorageMap
+ : gScalarStorageMap;
+
+ // Get the process-specific storage or create one if it's not
+ // available.
+ if (!processStorage.Get(storageId, &scalarStorage)) {
+ scalarStorage = new ScalarStorageMapType();
+ processStorage.Put(storageId, scalarStorage);
+ }
+
+ // Check if the scalar is already allocated in the parent or in the child
+ // storage.
+ if (scalarStorage->Get(aId.id, &scalar)) {
+ // Dynamic scalars can expire at any time during the session (e.g. an
+ // add-on was updated). Check if it expired.
+ if (aId.dynamic) {
+ const DynamicScalarInfo& dynInfo =
+ static_cast<const DynamicScalarInfo&>(info);
+ if (dynInfo.mDynamicExpiration) {
+ // The Dynamic scalar is expired.
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+ }
+ // This was not a dynamic scalar or was not expired.
+ *aRet = scalar;
+ return NS_OK;
+ }
+
+ // The scalar storage wasn't already allocated. Check if the scalar is expired
+ // and then allocate the storage, if needed.
+ if (IsExpiredVersion(info.expiration())) {
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+
+ scalar = internal_ScalarAllocate(info);
+ if (!scalar) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ scalarStorage->Put(aId.id, scalar);
+ *aRet = scalar;
+ return NS_OK;
+}
+
+void internal_ApplyPendingOperations(const StaticMutexAutoLock& lock);
+
+/**
+ * Record the given action on a scalar into the pending actions list.
+ *
+ * If the pending actions list overflows the high water mark length
+ * all operations are immediately applied, including the passed action.
+ *
+ * @param aScalarAction The action to record.
+ */
+void internal_RecordScalarAction(const StaticMutexAutoLock& lock,
+ const ScalarAction& aScalarAction) {
+ // Make sure to have the storage.
+ if (!gScalarsActions) {
+ gScalarsActions = new nsTArray<ScalarAction>();
+ }
+
+ // Store the action.
+ gScalarsActions->AppendElement(aScalarAction);
+
+ // If this action overflows the pending actions array, we immediately apply
+ // pending operations and assume loading is over. If loading still happens
+ // afterwards, some scalar values might be overwritten and inconsistent, but
+ // we won't lose operations on otherwise untouched probes.
+ if (gScalarsActions->Length() > kScalarActionsArrayHighWaterMark) {
+ internal_ApplyPendingOperations(lock);
+ return;
+ }
+}
+
+/**
+ * Record the given action on a scalar on the main process into the pending
+ * actions list.
+ *
+ * If the pending actions list overflows the high water mark length
+ * all operations are immediately applied, including the passed action.
+ *
+ * @param aId The scalar's ID this action applies to
+ * @param aDynamic Determines if the scalar is dynamic
+ * @param aAction The action to record
+ * @param aValue The additional data for the recorded action
+ */
+void internal_RecordScalarAction(const StaticMutexAutoLock& lock, uint32_t aId,
+ bool aDynamic, ScalarActionType aAction,
+ const ScalarVariant& aValue) {
+ internal_RecordScalarAction(
+ lock,
+ ScalarAction{aId, aDynamic, aAction, Some(aValue), ProcessID::Parent});
+}
+
+/**
+ * Record the given action on a keyed scalar into the pending actions list.
+ *
+ * If the pending actions list overflows the high water mark length
+ * all operations are immediately applied, including the passed action.
+ *
+ * @param aScalarAction The action to record.
+ */
+void internal_RecordKeyedScalarAction(const StaticMutexAutoLock& lock,
+ const KeyedScalarAction& aScalarAction) {
+ // Make sure to have the storage.
+ if (!gKeyedScalarsActions) {
+ gKeyedScalarsActions = new nsTArray<KeyedScalarAction>();
+ }
+
+ // Store the action.
+ gKeyedScalarsActions->AppendElement(aScalarAction);
+
+ // If this action overflows the pending actions array, we immediately apply
+ // pending operations and assume loading is over. If loading still happens
+ // afterwards, some scalar values might be overwritten and inconsistent, but
+ // we won't lose operations on otherwise untouched probes.
+ if (gKeyedScalarsActions->Length() > kScalarActionsArrayHighWaterMark) {
+ internal_ApplyPendingOperations(lock);
+ return;
+ }
+}
+
+/**
+ * Record the given action on a keyed scalar on the main process into the
+ * pending actions list.
+ *
+ * If the pending actions list overflows the high water mark length
+ * all operations are immediately applied, including the passed action.
+ *
+ * @param aId The scalar's ID this action applies to
+ * @param aDynamic Determines if the scalar is dynamic
+ * @param aKey The scalar's key
+ * @param aAction The action to record
+ * @param aValue The additional data for the recorded action
+ */
+void internal_RecordKeyedScalarAction(const StaticMutexAutoLock& lock,
+ uint32_t aId, bool aDynamic,
+ const nsAString& aKey,
+ ScalarActionType aAction,
+ const ScalarVariant& aValue) {
+ internal_RecordKeyedScalarAction(
+ lock,
+ KeyedScalarAction{aId, aDynamic, aAction, NS_ConvertUTF16toUTF8(aKey),
+ Some(aValue), ProcessID::Parent});
+}
+
+/**
+ * Update the scalar with the provided value. This is used by the JS API.
+ *
+ * @param lock Instance of a lock locking gTelemetryHistogramMutex
+ * @param aName The scalar name.
+ * @param aType The action type for updating the scalar.
+ * @param aValue The value to use for updating the scalar.
+ * @param aProcessOverride The process for which the scalar must be updated.
+ * This must only be used for GeckoView persistence. It must be
+ * set to the ProcessID::Parent for all the other cases.
+ * @param aForce Whether to force updating even if load is in progress.
+ * @return a ScalarResult error value.
+ */
+ScalarResult internal_UpdateScalar(
+ const StaticMutexAutoLock& lock, const nsACString& aName,
+ ScalarActionType aType, nsIVariant* aValue,
+ ProcessID aProcessOverride = ProcessID::Parent, bool aForce = false) {
+ ScalarKey uniqueId;
+ nsresult rv = internal_GetEnumByScalarName(lock, aName, &uniqueId);
+ if (NS_FAILED(rv)) {
+ return (rv == NS_ERROR_FAILURE) ? ScalarResult::NotInitialized
+ : ScalarResult::UnknownScalar;
+ }
+
+ ScalarResult sr = internal_CanRecordScalar(lock, uniqueId, false, aForce);
+ if (sr != ScalarResult::Ok) {
+ if (sr == ScalarResult::CannotRecordDataset) {
+ return ScalarResult::Ok;
+ }
+ return sr;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ const BaseScalarInfo& info = internal_GetScalarInfo(lock, uniqueId);
+ // Convert the nsIVariant to a Variant.
+ mozilla::Maybe<ScalarVariant> variantValue;
+ sr = GetVariantFromIVariant(aValue, info.kind, variantValue);
+ if (sr != ScalarResult::Ok) {
+ MOZ_ASSERT(false, "Unable to convert nsIVariant to mozilla::Variant.");
+ return sr;
+ }
+ TelemetryIPCAccumulator::RecordChildScalarAction(
+ uniqueId.id, uniqueId.dynamic, aType, variantValue.ref());
+ return ScalarResult::Ok;
+ }
+
+ if (!aForce && internal_IsScalarDeserializing(lock)) {
+ const BaseScalarInfo& info = internal_GetScalarInfo(lock, uniqueId);
+ // Convert the nsIVariant to a Variant.
+ mozilla::Maybe<ScalarVariant> variantValue;
+ sr = GetVariantFromIVariant(aValue, info.kind, variantValue);
+ if (sr != ScalarResult::Ok) {
+ MOZ_ASSERT(false, "Unable to convert nsIVariant to mozilla::Variant.");
+ return sr;
+ }
+ internal_RecordScalarAction(lock, uniqueId.id, uniqueId.dynamic, aType,
+ variantValue.ref());
+ return ScalarResult::Ok;
+ }
+
+ // Finally get the scalar.
+ ScalarBase* scalar = nullptr;
+ rv = internal_GetScalarByEnum(lock, uniqueId, aProcessOverride, &scalar);
+ if (NS_FAILED(rv)) {
+ // Don't throw on expired scalars.
+ if (rv == NS_ERROR_NOT_AVAILABLE) {
+ return ScalarResult::Ok;
+ }
+ return ScalarResult::UnknownScalar;
+ }
+
+ if (aType == ScalarActionType::eAdd) {
+ return scalar->AddValue(aValue);
+ }
+ if (aType == ScalarActionType::eSet) {
+ return scalar->SetValue(aValue);
+ }
+
+ return scalar->SetMaximum(aValue);
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE: thread-unsafe helpers for the keyed scalars
+
+namespace {
+
+/**
+ * Get a keyed scalar object by its enum id. This implicitly allocates the keyed
+ * scalar object in the storage if it wasn't previously allocated.
+ *
+ * @param lock Instance of a lock locking gTelemetryHistogramMutex
+ * @param aId The scalar identifier.
+ * @param aProcessStorage This drives the selection of the map to use to store
+ * the scalar data coming from child processes. This is only meaningful
+ * when this function is called in parent process. If that's the case,
+ * if this is not |GeckoProcessType_Default|, the process id is used to
+ * allocate and store the scalars.
+ * @param aRet The output variable that stores scalar object.
+ * @return
+ * NS_ERROR_INVALID_ARG if the scalar id is unknown or a this is a keyed
+ * string scalar.
+ * NS_ERROR_NOT_AVAILABLE if the scalar is expired.
+ * NS_OK if the scalar was found. If that's the case, aResult contains a
+ * valid pointer to a scalar type.
+ */
+nsresult internal_GetKeyedScalarByEnum(const StaticMutexAutoLock& lock,
+ const ScalarKey& aId,
+ ProcessID aProcessStorage,
+ KeyedScalar** aRet) {
+ if (!internal_IsValidId(lock, aId)) {
+ MOZ_ASSERT(false, "Requested a keyed scalar with an invalid id.");
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ const BaseScalarInfo& info = internal_GetScalarInfo(lock, aId);
+
+ // Dynamic scalars fixup: they are always stored in the "dynamic" process,
+ // unless they are part of the "builtin" Firefox probes. Please note that
+ // "dynamic builtin" probes are meant to support "artifact" and "build faster"
+ // builds.
+ if (aId.dynamic && !info.builtin) {
+ aProcessStorage = ProcessID::Dynamic;
+ }
+
+ KeyedScalar* scalar = nullptr;
+ KeyedScalarStorageMapType* scalarStorage = nullptr;
+ // Initialize the scalar storage to the parent storage. This will get
+ // set to the child storage if needed.
+ uint32_t storageId = static_cast<uint32_t>(aProcessStorage);
+
+ // Put dynamic-builtin scalars (used to support "build faster") in a
+ // separate storage.
+ ProcessesKeyedScalarsMapType& processStorage =
+ (aId.dynamic && info.builtin) ? gDynamicBuiltinKeyedScalarStorageMap
+ : gKeyedScalarStorageMap;
+
+ // Get the process-specific storage or create one if it's not
+ // available.
+ if (!processStorage.Get(storageId, &scalarStorage)) {
+ scalarStorage = new KeyedScalarStorageMapType();
+ processStorage.Put(storageId, scalarStorage);
+ }
+
+ if (scalarStorage->Get(aId.id, &scalar)) {
+ *aRet = scalar;
+ return NS_OK;
+ }
+
+ if (IsExpiredVersion(info.expiration())) {
+ return NS_ERROR_NOT_AVAILABLE;
+ }
+
+ // We don't currently support keyed string scalars. Disable them.
+ if (info.kind == nsITelemetry::SCALAR_TYPE_STRING) {
+ MOZ_ASSERT(false, "Keyed string scalars are not currently supported.");
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ scalar = new KeyedScalar(info);
+ if (!scalar) {
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ scalarStorage->Put(aId.id, scalar);
+ *aRet = scalar;
+ return NS_OK;
+}
+
+/**
+ * Update the keyed scalar with the provided value. This is used by the JS API.
+ *
+ * @param lock Instance of a lock locking gTelemetryHistogramMutex
+ * @param aName The scalar name.
+ * @param aKey The key name.
+ * @param aType The action type for updating the scalar.
+ * @param aValue The value to use for updating the scalar.
+ * @param aProcessOverride The process for which the scalar must be updated.
+ * This must only be used for GeckoView persistence. It must be
+ * set to the ProcessID::Parent for all the other cases.
+ * @return a ScalarResult error value.
+ */
+ScalarResult internal_UpdateKeyedScalar(
+ const StaticMutexAutoLock& lock, const nsACString& aName,
+ const nsAString& aKey, ScalarActionType aType, nsIVariant* aValue,
+ ProcessID aProcessOverride = ProcessID::Parent, bool aForce = false) {
+ ScalarKey uniqueId;
+ nsresult rv = internal_GetEnumByScalarName(lock, aName, &uniqueId);
+ if (NS_FAILED(rv)) {
+ return (rv == NS_ERROR_FAILURE) ? ScalarResult::NotInitialized
+ : ScalarResult::UnknownScalar;
+ }
+
+ ScalarResult sr = internal_CanRecordScalar(lock, uniqueId, true, aForce);
+ if (sr != ScalarResult::Ok) {
+ if (sr == ScalarResult::CannotRecordDataset) {
+ return ScalarResult::Ok;
+ }
+ return sr;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ const BaseScalarInfo& info = internal_GetScalarInfo(lock, uniqueId);
+ // Convert the nsIVariant to a Variant.
+ mozilla::Maybe<ScalarVariant> variantValue;
+ sr = GetVariantFromIVariant(aValue, info.kind, variantValue);
+ if (sr != ScalarResult::Ok) {
+ MOZ_ASSERT(false, "Unable to convert nsIVariant to mozilla::Variant.");
+ return sr;
+ }
+ TelemetryIPCAccumulator::RecordChildKeyedScalarAction(
+ uniqueId.id, uniqueId.dynamic, aKey, aType, variantValue.ref());
+ return ScalarResult::Ok;
+ }
+
+ if (!aForce && internal_IsScalarDeserializing(lock)) {
+ const BaseScalarInfo& info = internal_GetScalarInfo(lock, uniqueId);
+ // Convert the nsIVariant to a Variant.
+ mozilla::Maybe<ScalarVariant> variantValue;
+ sr = GetVariantFromIVariant(aValue, info.kind, variantValue);
+ if (sr != ScalarResult::Ok) {
+ MOZ_ASSERT(false, "Unable to convert nsIVariant to mozilla::Variant.");
+ return sr;
+ }
+ internal_RecordKeyedScalarAction(lock, uniqueId.id, uniqueId.dynamic, aKey,
+ aType, variantValue.ref());
+ return ScalarResult::Ok;
+ }
+
+ // Finally get the scalar.
+ KeyedScalar* scalar = nullptr;
+ rv = internal_GetKeyedScalarByEnum(lock, uniqueId, aProcessOverride, &scalar);
+ if (NS_FAILED(rv)) {
+ // Don't throw on expired scalars.
+ if (rv == NS_ERROR_NOT_AVAILABLE) {
+ return ScalarResult::Ok;
+ }
+ return ScalarResult::UnknownScalar;
+ }
+
+ if (aType == ScalarActionType::eAdd) {
+ return scalar->AddValue(lock, aKey, aValue);
+ }
+ if (aType == ScalarActionType::eSet) {
+ return scalar->SetValue(lock, aKey, aValue);
+ }
+
+ return scalar->SetMaximum(lock, aKey, aValue);
+}
+
+/**
+ * Helper function to convert an array of |DynamicScalarInfo|
+ * to |DynamicScalarDefinition| used by the IPC calls.
+ */
+void internal_DynamicScalarToIPC(
+ const StaticMutexAutoLock& lock,
+ const nsTArray<DynamicScalarInfo>& aDynamicScalarInfos,
+ nsTArray<DynamicScalarDefinition>& aIPCDefs) {
+ for (auto& info : aDynamicScalarInfos) {
+ DynamicScalarDefinition stubDefinition;
+ stubDefinition.type = info.kind;
+ stubDefinition.dataset = info.dataset;
+ stubDefinition.expired = info.mDynamicExpiration;
+ stubDefinition.keyed = info.keyed;
+ stubDefinition.name = info.mDynamicName;
+ stubDefinition.builtin = info.builtin;
+ aIPCDefs.AppendElement(stubDefinition);
+ }
+}
+
+/**
+ * Broadcasts the dynamic scalar definitions to all the other
+ * content processes.
+ */
+void internal_BroadcastDefinitions(
+ const nsTArray<DynamicScalarDefinition>& scalarDefs) {
+ nsTArray<mozilla::dom::ContentParent*> parents;
+ mozilla::dom::ContentParent::GetAll(parents);
+ if (!parents.Length()) {
+ return;
+ }
+
+ // Broadcast the definitions to the other content processes.
+ for (auto parent : parents) {
+ mozilla::Unused << parent->SendAddDynamicScalars(scalarDefs);
+ }
+}
+
+void internal_RegisterScalars(const StaticMutexAutoLock& lock,
+ const nsTArray<DynamicScalarInfo>& scalarInfos) {
+ // Register the new scalars.
+ if (!gDynamicScalarInfo) {
+ gDynamicScalarInfo = new nsTArray<DynamicScalarInfo>();
+ }
+ if (!gDynamicStoreNames) {
+ gDynamicStoreNames = new nsTArray<RefPtr<nsAtom>>();
+ }
+
+ for (auto& scalarInfo : scalarInfos) {
+ // Allow expiring scalars that were already registered.
+ CharPtrEntryType* existingKey =
+ gScalarNameIDMap.GetEntry(scalarInfo.name());
+ if (existingKey) {
+ // Change the scalar to expired if needed.
+ if (scalarInfo.mDynamicExpiration && !scalarInfo.builtin) {
+ DynamicScalarInfo& scalarData =
+ (*gDynamicScalarInfo)[existingKey->GetData().id];
+ scalarData.mDynamicExpiration = true;
+ }
+ continue;
+ }
+
+ gDynamicScalarInfo->AppendElement(scalarInfo);
+ uint32_t scalarId = gDynamicScalarInfo->Length() - 1;
+ CharPtrEntryType* entry = gScalarNameIDMap.PutEntry(scalarInfo.name());
+ entry->SetData(ScalarKey{scalarId, true});
+ }
+}
+
+/**
+ * Creates a snapshot of the desired scalar storage.
+ * @param {aLock} The proof of lock to access scalar data.
+ * @param {aScalarsToReflect} The table that will contain the snapshot.
+ * @param {aDataset} The dataset we're asking the snapshot for.
+ * @param {aProcessStorage} The scalar storage to take a snapshot of.
+ * @param {aIsBuiltinDynamic} Whether or not the storage is for dynamic builtin
+ * scalars.
+ * @return NS_OK or the error code describing the failure reason.
+ */
+nsresult internal_ScalarSnapshotter(const StaticMutexAutoLock& aLock,
+ ScalarSnapshotTable& aScalarsToReflect,
+ unsigned int aDataset,
+ ProcessesScalarsMapType& aProcessStorage,
+ bool aIsBuiltinDynamic, bool aClearScalars,
+ const nsACString& aStoreName) {
+ // Iterate the scalars in aProcessStorage. The storage may contain empty or
+ // yet to be initialized scalars from all the supported processes.
+ for (auto iter = aProcessStorage.Iter(); !iter.Done(); iter.Next()) {
+ ScalarStorageMapType* scalarStorage = iter.UserData();
+ ScalarTupleArray& processScalars =
+ aScalarsToReflect.GetOrInsert(iter.Key());
+
+ // Are we in the "Dynamic" process?
+ bool isDynamicProcess =
+ ProcessID::Dynamic == static_cast<ProcessID>(iter.Key());
+
+ // Iterate each available child storage.
+ for (auto childIter = scalarStorage->Iter(); !childIter.Done();
+ childIter.Next()) {
+ ScalarBase* scalar = childIter.UserData();
+
+ // Get the informations for this scalar.
+ const BaseScalarInfo& info = internal_GetScalarInfo(
+ aLock, ScalarKey{childIter.Key(),
+ aIsBuiltinDynamic ? true : isDynamicProcess});
+
+ // Serialize the scalar if it's in the desired dataset.
+ if (IsInDataset(info.dataset, aDataset)) {
+ // Get the scalar value.
+ nsCOMPtr<nsIVariant> scalarValue;
+ nsresult rv = scalar->GetValue(aStoreName, aClearScalars, scalarValue);
+ if (rv == NS_ERROR_NO_CONTENT) {
+ // No value for this store. Proceed.
+ continue;
+ }
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ // Append it to our list.
+ processScalars.AppendElement(
+ mozilla::MakeTuple(info.name(), scalarValue, info.kind));
+ }
+ }
+ if (processScalars.Length() == 0) {
+ aScalarsToReflect.Remove(iter.Key());
+ }
+ }
+ return NS_OK;
+}
+
+/**
+ * Creates a snapshot of the desired keyed scalar storage.
+ * @param {aLock} The proof of lock to access scalar data.
+ * @param {aScalarsToReflect} The table that will contain the snapshot.
+ * @param {aDataset} The dataset we're asking the snapshot for.
+ * @param {aProcessStorage} The scalar storage to take a snapshot of.
+ * @param {aIsBuiltinDynamic} Whether or not the storage is for dynamic builtin
+ * scalars.
+ * @return NS_OK or the error code describing the failure reason.
+ */
+nsresult internal_KeyedScalarSnapshotter(
+ const StaticMutexAutoLock& aLock,
+ KeyedScalarSnapshotTable& aScalarsToReflect, unsigned int aDataset,
+ ProcessesKeyedScalarsMapType& aProcessStorage, bool aIsBuiltinDynamic,
+ bool aClearScalars, const nsACString& aStoreName) {
+ // Iterate the scalars in aProcessStorage. The storage may contain empty or
+ // yet to be initialized scalars from all the supported processes.
+ for (auto iter = aProcessStorage.Iter(); !iter.Done(); iter.Next()) {
+ KeyedScalarStorageMapType* scalarStorage = iter.UserData();
+ KeyedScalarTupleArray& processScalars =
+ aScalarsToReflect.GetOrInsert(iter.Key());
+
+ // Are we in the "Dynamic" process?
+ bool isDynamicProcess =
+ ProcessID::Dynamic == static_cast<ProcessID>(iter.Key());
+
+ for (auto childIter = scalarStorage->Iter(); !childIter.Done();
+ childIter.Next()) {
+ KeyedScalar* scalar = childIter.UserData();
+
+ // Get the informations for this scalar.
+ const BaseScalarInfo& info = internal_GetScalarInfo(
+ aLock, ScalarKey{childIter.Key(),
+ aIsBuiltinDynamic ? true : isDynamicProcess});
+
+ // Serialize the scalar if it's in the desired dataset.
+ if (IsInDataset(info.dataset, aDataset)) {
+ // Get the keys for this scalar.
+ nsTArray<KeyedScalar::KeyValuePair> scalarKeyedData;
+ nsresult rv =
+ scalar->GetValue(aStoreName, aClearScalars, scalarKeyedData);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ if (scalarKeyedData.Length() == 0) {
+ // Don't bother with empty keyed scalars.
+ continue;
+ }
+ // Append it to our list.
+ processScalars.AppendElement(mozilla::MakeTuple(
+ info.name(), std::move(scalarKeyedData), info.kind));
+ }
+ }
+ if (processScalars.Length() == 0) {
+ aScalarsToReflect.Remove(iter.Key());
+ }
+ }
+ return NS_OK;
+}
+
+/**
+ * Helper function to get a snapshot of the scalars.
+ *
+ * @param {aLock} The proof of lock to access scalar data.
+ * @param {aScalarsToReflect} The table that will contain the snapshot.
+ * @param {aDataset} The dataset we're asking the snapshot for.
+ * @param {aClearScalars} Whether or not to clear the scalar storage.
+ * @param {aStoreName} The name of the store to snapshot.
+ * @return NS_OK or the error code describing the failure reason.
+ */
+nsresult internal_GetScalarSnapshot(const StaticMutexAutoLock& aLock,
+ ScalarSnapshotTable& aScalarsToReflect,
+ unsigned int aDataset, bool aClearScalars,
+ const nsACString& aStoreName) {
+ // Take a snapshot of the scalars.
+ nsresult rv =
+ internal_ScalarSnapshotter(aLock, aScalarsToReflect, aDataset,
+ gScalarStorageMap, false, /*aIsBuiltinDynamic*/
+ aClearScalars, aStoreName);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ // And a snapshot of the dynamic builtin ones.
+ rv = internal_ScalarSnapshotter(aLock, aScalarsToReflect, aDataset,
+ gDynamicBuiltinScalarStorageMap,
+ true, /*aIsBuiltinDynamic*/
+ aClearScalars, aStoreName);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Helper function to get a snapshot of the keyed scalars.
+ *
+ * @param {aLock} The proof of lock to access scalar data.
+ * @param {aScalarsToReflect} The table that will contain the snapshot.
+ * @param {aDataset} The dataset we're asking the snapshot for.
+ * @param {aClearScalars} Whether or not to clear the scalar storage.
+ * @param {aStoreName} The name of the store to snapshot.
+ * @return NS_OK or the error code describing the failure reason.
+ */
+nsresult internal_GetKeyedScalarSnapshot(
+ const StaticMutexAutoLock& aLock,
+ KeyedScalarSnapshotTable& aScalarsToReflect, unsigned int aDataset,
+ bool aClearScalars, const nsACString& aStoreName) {
+ // Take a snapshot of the scalars.
+ nsresult rv = internal_KeyedScalarSnapshotter(
+ aLock, aScalarsToReflect, aDataset, gKeyedScalarStorageMap,
+ false, /*aIsBuiltinDynamic*/
+ aClearScalars, aStoreName);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ // And a snapshot of the dynamic builtin ones.
+ rv = internal_KeyedScalarSnapshotter(aLock, aScalarsToReflect, aDataset,
+ gDynamicBuiltinKeyedScalarStorageMap,
+ true, /*aIsBuiltinDynamic*/
+ aClearScalars, aStoreName);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ return NS_OK;
+}
+
+} // namespace
+
+// helpers for recording/applying scalar operations
+namespace {
+
+void internal_ApplyScalarActions(
+ const StaticMutexAutoLock& lock,
+ const nsTArray<mozilla::Telemetry::ScalarAction>& aScalarActions,
+ const mozilla::Maybe<ProcessID>& aProcessType = Nothing()) {
+ if (!internal_CanRecordBase(lock)) {
+ return;
+ }
+
+ for (auto& upd : aScalarActions) {
+ ScalarKey uniqueId{upd.mId, upd.mDynamic};
+ if (NS_WARN_IF(!internal_IsValidId(lock, uniqueId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ continue;
+ }
+
+ if (internal_IsKeyedScalar(lock, uniqueId)) {
+ continue;
+ }
+
+ // Are we allowed to record this scalar? We don't need to check for
+ // allowed processes here, that's taken care of when recording
+ // in child processes.
+ if (!internal_CanRecordForScalarID(lock, uniqueId)) {
+ continue;
+ }
+
+ // Either we got passed a process type or it was explicitely set on the
+ // recorded action. It should never happen that it is set to an invalid
+ // value (such as ProcessID::Count)
+ ProcessID processType = aProcessType.valueOr(upd.mProcessType);
+ MOZ_ASSERT(processType != ProcessID::Count);
+
+ // Refresh the data in the parent process with the data coming from the
+ // child processes.
+ ScalarBase* scalar = nullptr;
+ nsresult rv =
+ internal_GetScalarByEnum(lock, uniqueId, processType, &scalar);
+ if (NS_FAILED(rv)) {
+ // Bug 1513496 - We no longer log a warning if the scalar is expired.
+ if (rv != NS_ERROR_NOT_AVAILABLE) {
+ NS_WARNING("NS_FAILED internal_GetScalarByEnum for CHILD");
+ }
+ continue;
+ }
+
+ if (upd.mData.isNothing()) {
+ MOZ_ASSERT(false, "There is no data in the ScalarActionType.");
+ continue;
+ }
+
+ // Get the type of this scalar from the scalar ID. We already checked
+ // for its validity a few lines above.
+ const uint32_t scalarType = internal_GetScalarInfo(lock, uniqueId).kind;
+
+ // Extract the data from the mozilla::Variant.
+ switch (upd.mActionType) {
+ case ScalarActionType::eSet: {
+ switch (scalarType) {
+ case nsITelemetry::SCALAR_TYPE_COUNT:
+ if (!upd.mData->is<uint32_t>()) {
+ NS_WARNING("Attempting to set a count scalar to a non-integer.");
+ continue;
+ }
+ scalar->SetValue(upd.mData->as<uint32_t>());
+ break;
+ case nsITelemetry::SCALAR_TYPE_BOOLEAN:
+ if (!upd.mData->is<bool>()) {
+ NS_WARNING(
+ "Attempting to set a boolean scalar to a non-boolean.");
+ continue;
+ }
+ scalar->SetValue(upd.mData->as<bool>());
+ break;
+ case nsITelemetry::SCALAR_TYPE_STRING:
+ if (!upd.mData->is<nsString>()) {
+ NS_WARNING("Attempting to set a string scalar to a non-string.");
+ continue;
+ }
+ scalar->SetValue(upd.mData->as<nsString>());
+ break;
+ }
+ break;
+ }
+ case ScalarActionType::eAdd: {
+ if (scalarType != nsITelemetry::SCALAR_TYPE_COUNT) {
+ NS_WARNING("Attempting to add on a non count scalar.");
+ continue;
+ }
+ // We only support adding uint32_t.
+ if (!upd.mData->is<uint32_t>()) {
+ NS_WARNING("Attempting to add to a count scalar with a non-integer.");
+ continue;
+ }
+ scalar->AddValue(upd.mData->as<uint32_t>());
+ break;
+ }
+ case ScalarActionType::eSetMaximum: {
+ if (scalarType != nsITelemetry::SCALAR_TYPE_COUNT) {
+ NS_WARNING("Attempting to setMaximum on a non count scalar.");
+ continue;
+ }
+ // We only support SetMaximum on uint32_t.
+ if (!upd.mData->is<uint32_t>()) {
+ NS_WARNING(
+ "Attempting to setMaximum a count scalar to a non-integer.");
+ continue;
+ }
+ scalar->SetMaximum(upd.mData->as<uint32_t>());
+ break;
+ }
+ default:
+ NS_WARNING("Unsupported action coming from scalar child updates.");
+ }
+ }
+}
+
+void internal_ApplyKeyedScalarActions(
+ const StaticMutexAutoLock& lock,
+ const nsTArray<mozilla::Telemetry::KeyedScalarAction>& aScalarActions,
+ const mozilla::Maybe<ProcessID>& aProcessType = Nothing()) {
+ if (!internal_CanRecordBase(lock)) {
+ return;
+ }
+
+ for (auto& upd : aScalarActions) {
+ ScalarKey uniqueId{upd.mId, upd.mDynamic};
+ if (NS_WARN_IF(!internal_IsValidId(lock, uniqueId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ continue;
+ }
+
+ if (!internal_IsKeyedScalar(lock, uniqueId)) {
+ continue;
+ }
+
+ // Are we allowed to record this scalar? We don't need to check for
+ // allowed processes here, that's taken care of when recording
+ // in child processes.
+ if (!internal_CanRecordForScalarID(lock, uniqueId)) {
+ continue;
+ }
+
+ // Either we got passed a process type or it was explicitely set on the
+ // recorded action. It should never happen that it is set to an invalid
+ // value (such as ProcessID::Count)
+ ProcessID processType = aProcessType.valueOr(upd.mProcessType);
+ MOZ_ASSERT(processType != ProcessID::Count);
+
+ // Refresh the data in the parent process with the data coming from the
+ // child processes.
+ KeyedScalar* scalar = nullptr;
+ nsresult rv =
+ internal_GetKeyedScalarByEnum(lock, uniqueId, processType, &scalar);
+ if (NS_FAILED(rv)) {
+ // Bug 1513496 - We no longer log a warning if the scalar is expired.
+ if (rv != NS_ERROR_NOT_AVAILABLE) {
+ NS_WARNING("NS_FAILED internal_GetKeyedScalarByEnum for CHILD");
+ }
+ continue;
+ }
+
+ if (upd.mData.isNothing()) {
+ MOZ_ASSERT(false, "There is no data in the KeyedScalarAction.");
+ continue;
+ }
+
+ // Get the type of this scalar from the scalar ID. We already checked
+ // for its validity a few lines above.
+ const uint32_t scalarType = internal_GetScalarInfo(lock, uniqueId).kind;
+
+ // Extract the data from the mozilla::Variant.
+ switch (upd.mActionType) {
+ case ScalarActionType::eSet: {
+ switch (scalarType) {
+ case nsITelemetry::SCALAR_TYPE_COUNT:
+ if (!upd.mData->is<uint32_t>()) {
+ NS_WARNING("Attempting to set a count scalar to a non-integer.");
+ continue;
+ }
+ scalar->SetValue(lock, NS_ConvertUTF8toUTF16(upd.mKey),
+ upd.mData->as<uint32_t>());
+ break;
+ case nsITelemetry::SCALAR_TYPE_BOOLEAN:
+ if (!upd.mData->is<bool>()) {
+ NS_WARNING(
+ "Attempting to set a boolean scalar to a non-boolean.");
+ continue;
+ }
+ scalar->SetValue(lock, NS_ConvertUTF8toUTF16(upd.mKey),
+ upd.mData->as<bool>());
+ break;
+ default:
+ NS_WARNING("Unsupported type coming from scalar child updates.");
+ }
+ break;
+ }
+ case ScalarActionType::eAdd: {
+ if (scalarType != nsITelemetry::SCALAR_TYPE_COUNT) {
+ NS_WARNING("Attempting to add on a non count scalar.");
+ continue;
+ }
+ // We only support adding on uint32_t.
+ if (!upd.mData->is<uint32_t>()) {
+ NS_WARNING("Attempting to add to a count scalar with a non-integer.");
+ continue;
+ }
+ scalar->AddValue(lock, NS_ConvertUTF8toUTF16(upd.mKey),
+ upd.mData->as<uint32_t>());
+ break;
+ }
+ case ScalarActionType::eSetMaximum: {
+ if (scalarType != nsITelemetry::SCALAR_TYPE_COUNT) {
+ NS_WARNING("Attempting to setMaximum on a non count scalar.");
+ continue;
+ }
+ // We only support SetMaximum on uint32_t.
+ if (!upd.mData->is<uint32_t>()) {
+ NS_WARNING(
+ "Attempting to setMaximum a count scalar to a non-integer.");
+ continue;
+ }
+ scalar->SetMaximum(lock, NS_ConvertUTF8toUTF16(upd.mKey),
+ upd.mData->as<uint32_t>());
+ break;
+ }
+ default:
+ NS_WARNING(
+ "Unsupported action coming from keyed scalar child updates.");
+ }
+ }
+}
+
+void internal_ApplyPendingOperations(const StaticMutexAutoLock& lock) {
+ if (gScalarsActions && gScalarsActions->Length() > 0) {
+ internal_ApplyScalarActions(lock, *gScalarsActions);
+ gScalarsActions->Clear();
+ }
+
+ if (gKeyedScalarsActions && gKeyedScalarsActions->Length() > 0) {
+ internal_ApplyKeyedScalarActions(lock, *gKeyedScalarsActions);
+ gKeyedScalarsActions->Clear();
+ }
+
+ // After all pending operations are applied deserialization is done
+ gIsDeserializing = false;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// EXTERNALLY VISIBLE FUNCTIONS in namespace TelemetryScalars::
+
+// This is a StaticMutex rather than a plain Mutex (1) so that
+// it gets initialised in a thread-safe manner the first time
+// it is used, and (2) because it is never de-initialised, and
+// a normal Mutex would show up as a leak in BloatView. StaticMutex
+// also has the "OffTheBooks" property, so it won't show as a leak
+// in BloatView.
+// Another reason to use a StaticMutex instead of a plain Mutex is
+// that, due to the nature of Telemetry, we cannot rely on having a
+// mutex initialized in InitializeGlobalState. Unfortunately, we
+// cannot make sure that no other function is called before this point.
+static StaticMutex gTelemetryScalarsMutex;
+
+void TelemetryScalar::InitializeGlobalState(bool aCanRecordBase,
+ bool aCanRecordExtended) {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ MOZ_ASSERT(!gInitDone,
+ "TelemetryScalar::InitializeGlobalState "
+ "may only be called once");
+
+ gCanRecordBase = aCanRecordBase;
+ gCanRecordExtended = aCanRecordExtended;
+
+ // Populate the static scalar name->id cache. Note that the scalar names are
+ // statically allocated and come from the automatically generated
+ // TelemetryScalarData.h.
+ uint32_t scalarCount =
+ static_cast<uint32_t>(mozilla::Telemetry::ScalarID::ScalarCount);
+ for (uint32_t i = 0; i < scalarCount; i++) {
+ CharPtrEntryType* entry = gScalarNameIDMap.PutEntry(gScalars[i].name());
+ entry->SetData(ScalarKey{i, false});
+ }
+
+ // To summarize dynamic events we need a dynamic scalar.
+ const nsTArray<DynamicScalarInfo> initialDynamicScalars({
+ DynamicScalarInfo{
+ nsITelemetry::SCALAR_TYPE_COUNT,
+ true /* recordOnRelease */,
+ false /* expired */,
+ nsAutoCString("telemetry.dynamic_event_counts"),
+ true /* keyed */,
+ false /* built-in */,
+ {} /* stores */,
+ },
+ });
+ internal_RegisterScalars(locker, initialDynamicScalars);
+
+ gInitDone = true;
+}
+
+void TelemetryScalar::DeInitializeGlobalState() {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ gCanRecordBase = false;
+ gCanRecordExtended = false;
+ gScalarNameIDMap.Clear();
+ gScalarStorageMap.Clear();
+ gKeyedScalarStorageMap.Clear();
+ gDynamicBuiltinScalarStorageMap.Clear();
+ gDynamicBuiltinKeyedScalarStorageMap.Clear();
+ gDynamicScalarInfo = nullptr;
+ gDynamicStoreNames = nullptr;
+ gInitDone = false;
+}
+
+void TelemetryScalar::DeserializationStarted() {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ gIsDeserializing = true;
+}
+
+void TelemetryScalar::ApplyPendingOperations() {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ internal_ApplyPendingOperations(locker);
+}
+
+void TelemetryScalar::SetCanRecordBase(bool b) {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ gCanRecordBase = b;
+}
+
+void TelemetryScalar::SetCanRecordExtended(bool b) {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ gCanRecordExtended = b;
+}
+
+/**
+ * Adds the value to the given scalar.
+ *
+ * @param aName The scalar name.
+ * @param aVal The numeric value to add to the scalar.
+ * @param aCx The JS context.
+ * @return NS_OK (always) so that the JS API call doesn't throw. In case of
+ * errors, a warning level message is printed in the browser console.
+ */
+nsresult TelemetryScalar::Add(const nsACString& aName, JS::HandleValue aVal,
+ JSContext* aCx) {
+ // Unpack the aVal to nsIVariant. This uses the JS context.
+ nsCOMPtr<nsIVariant> unpackedVal;
+ nsresult rv = nsContentUtils::XPConnect()->JSToVariant(
+ aCx, aVal, getter_AddRefs(unpackedVal));
+ if (NS_FAILED(rv)) {
+ internal_LogScalarError(aName, ScalarResult::CannotUnpackVariant);
+ return NS_OK;
+ }
+
+ ScalarResult sr;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ sr = internal_UpdateScalar(locker, aName, ScalarActionType::eAdd,
+ unpackedVal);
+ }
+
+ // Warn the user about the error if we need to.
+ if (sr != ScalarResult::Ok) {
+ internal_LogScalarError(aName, sr);
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Adds the value to the given scalar.
+ *
+ * @param aName The scalar name.
+ * @param aKey The key name.
+ * @param aVal The numeric value to add to the scalar.
+ * @param aCx The JS context.
+ * @return NS_OK (always) so that the JS API call doesn't throw. In case of
+ * errors, a warning level message is printed in the browser console.
+ */
+nsresult TelemetryScalar::Add(const nsACString& aName, const nsAString& aKey,
+ JS::HandleValue aVal, JSContext* aCx) {
+ // Unpack the aVal to nsIVariant. This uses the JS context.
+ nsCOMPtr<nsIVariant> unpackedVal;
+ nsresult rv = nsContentUtils::XPConnect()->JSToVariant(
+ aCx, aVal, getter_AddRefs(unpackedVal));
+ if (NS_FAILED(rv)) {
+ internal_LogScalarError(aName, ScalarResult::CannotUnpackVariant);
+ return NS_OK;
+ }
+
+ ScalarResult sr;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ sr = internal_UpdateKeyedScalar(locker, aName, aKey, ScalarActionType::eAdd,
+ unpackedVal);
+ }
+
+ // Warn the user about the error if we need to.
+ if (sr != ScalarResult::Ok) {
+ internal_LogScalarError(aName, sr);
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Adds the value to the given scalar.
+ *
+ * @param aId The scalar enum id.
+ * @param aVal The numeric value to add to the scalar.
+ */
+void TelemetryScalar::Add(mozilla::Telemetry::ScalarID aId, uint32_t aValue) {
+ if (NS_WARN_IF(!IsValidEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ return;
+ }
+
+ ScalarKey uniqueId{static_cast<uint32_t>(aId), false};
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ if (internal_CanRecordScalar(locker, uniqueId, false) != ScalarResult::Ok) {
+ // We can't record this scalar. Bail out.
+ return;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ TelemetryIPCAccumulator::RecordChildScalarAction(
+ uniqueId.id, uniqueId.dynamic, ScalarActionType::eAdd,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ if (internal_IsScalarDeserializing(locker)) {
+ internal_RecordScalarAction(locker, uniqueId.id, uniqueId.dynamic,
+ ScalarActionType::eAdd, ScalarVariant(aValue));
+ return;
+ }
+
+ ScalarBase* scalar = nullptr;
+ nsresult rv =
+ internal_GetScalarByEnum(locker, uniqueId, ProcessID::Parent, &scalar);
+ if (NS_FAILED(rv)) {
+ return;
+ }
+
+ scalar->AddValue(aValue);
+}
+
+/**
+ * Adds the value to the given keyed scalar.
+ *
+ * @param aId The scalar enum id.
+ * @param aKey The key name.
+ * @param aVal The numeric value to add to the scalar.
+ */
+void TelemetryScalar::Add(mozilla::Telemetry::ScalarID aId,
+ const nsAString& aKey, uint32_t aValue) {
+ if (NS_WARN_IF(!IsValidEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ return;
+ }
+
+ ScalarKey uniqueId{static_cast<uint32_t>(aId), false};
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ if (internal_CanRecordScalar(locker, uniqueId, true) != ScalarResult::Ok) {
+ // We can't record this scalar. Bail out.
+ return;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ TelemetryIPCAccumulator::RecordChildKeyedScalarAction(
+ uniqueId.id, uniqueId.dynamic, aKey, ScalarActionType::eAdd,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ if (internal_IsScalarDeserializing(locker)) {
+ internal_RecordKeyedScalarAction(locker, uniqueId.id, uniqueId.dynamic,
+ aKey, ScalarActionType::eAdd,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ KeyedScalar* scalar = nullptr;
+ nsresult rv = internal_GetKeyedScalarByEnum(locker, uniqueId,
+ ProcessID::Parent, &scalar);
+ if (NS_FAILED(rv)) {
+ return;
+ }
+
+ scalar->AddValue(locker, aKey, aValue);
+}
+
+/**
+ * Sets the scalar to the given value.
+ *
+ * @param aName The scalar name.
+ * @param aVal The value to set the scalar to.
+ * @param aCx The JS context.
+ * @return NS_OK (always) so that the JS API call doesn't throw. In case of
+ * errors, a warning level message is printed in the browser console.
+ */
+nsresult TelemetryScalar::Set(const nsACString& aName, JS::HandleValue aVal,
+ JSContext* aCx) {
+ // Unpack the aVal to nsIVariant. This uses the JS context.
+ nsCOMPtr<nsIVariant> unpackedVal;
+ nsresult rv = nsContentUtils::XPConnect()->JSToVariant(
+ aCx, aVal, getter_AddRefs(unpackedVal));
+ if (NS_FAILED(rv)) {
+ internal_LogScalarError(aName, ScalarResult::CannotUnpackVariant);
+ return NS_OK;
+ }
+
+ ScalarResult sr;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ sr = internal_UpdateScalar(locker, aName, ScalarActionType::eSet,
+ unpackedVal);
+ }
+
+ // Warn the user about the error if we need to.
+ if (sr != ScalarResult::Ok) {
+ internal_LogScalarError(aName, sr);
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Sets the keyed scalar to the given value.
+ *
+ * @param aName The scalar name.
+ * @param aKey The key name.
+ * @param aVal The value to set the scalar to.
+ * @param aCx The JS context.
+ * @return NS_OK (always) so that the JS API call doesn't throw. In case of
+ * errors, a warning level message is printed in the browser console.
+ */
+nsresult TelemetryScalar::Set(const nsACString& aName, const nsAString& aKey,
+ JS::HandleValue aVal, JSContext* aCx) {
+ // Unpack the aVal to nsIVariant. This uses the JS context.
+ nsCOMPtr<nsIVariant> unpackedVal;
+ nsresult rv = nsContentUtils::XPConnect()->JSToVariant(
+ aCx, aVal, getter_AddRefs(unpackedVal));
+ if (NS_FAILED(rv)) {
+ internal_LogScalarError(aName, ScalarResult::CannotUnpackVariant);
+ return NS_OK;
+ }
+
+ ScalarResult sr;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ sr = internal_UpdateKeyedScalar(locker, aName, aKey, ScalarActionType::eSet,
+ unpackedVal);
+ }
+
+ // Warn the user about the error if we need to.
+ if (sr != ScalarResult::Ok) {
+ internal_LogScalarError(aName, sr);
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Sets the scalar to the given numeric value.
+ *
+ * @param aId The scalar enum id.
+ * @param aValue The numeric, unsigned value to set the scalar to.
+ */
+void TelemetryScalar::Set(mozilla::Telemetry::ScalarID aId, uint32_t aValue) {
+ if (NS_WARN_IF(!IsValidEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ return;
+ }
+
+ ScalarKey uniqueId{static_cast<uint32_t>(aId), false};
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ if (internal_CanRecordScalar(locker, uniqueId, false) != ScalarResult::Ok) {
+ // We can't record this scalar. Bail out.
+ return;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ TelemetryIPCAccumulator::RecordChildScalarAction(
+ uniqueId.id, uniqueId.dynamic, ScalarActionType::eSet,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ if (internal_IsScalarDeserializing(locker)) {
+ internal_RecordScalarAction(locker, uniqueId.id, uniqueId.dynamic,
+ ScalarActionType::eSet, ScalarVariant(aValue));
+ return;
+ }
+
+ ScalarBase* scalar = nullptr;
+ nsresult rv =
+ internal_GetScalarByEnum(locker, uniqueId, ProcessID::Parent, &scalar);
+ if (NS_FAILED(rv)) {
+ return;
+ }
+
+ scalar->SetValue(aValue);
+}
+
+/**
+ * Sets the scalar to the given string value.
+ *
+ * @param aId The scalar enum id.
+ * @param aValue The string value to set the scalar to.
+ */
+void TelemetryScalar::Set(mozilla::Telemetry::ScalarID aId,
+ const nsAString& aValue) {
+ if (NS_WARN_IF(!IsValidEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ return;
+ }
+
+ ScalarKey uniqueId{static_cast<uint32_t>(aId), false};
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ if (internal_CanRecordScalar(locker, uniqueId, false) != ScalarResult::Ok) {
+ // We can't record this scalar. Bail out.
+ return;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ TelemetryIPCAccumulator::RecordChildScalarAction(
+ uniqueId.id, uniqueId.dynamic, ScalarActionType::eSet,
+ ScalarVariant(nsString(aValue)));
+ return;
+ }
+
+ if (internal_IsScalarDeserializing(locker)) {
+ internal_RecordScalarAction(locker, uniqueId.id, uniqueId.dynamic,
+ ScalarActionType::eSet,
+ ScalarVariant(nsString(aValue)));
+ return;
+ }
+
+ ScalarBase* scalar = nullptr;
+ nsresult rv =
+ internal_GetScalarByEnum(locker, uniqueId, ProcessID::Parent, &scalar);
+ if (NS_FAILED(rv)) {
+ return;
+ }
+
+ scalar->SetValue(aValue);
+}
+
+/**
+ * Sets the scalar to the given boolean value.
+ *
+ * @param aId The scalar enum id.
+ * @param aValue The boolean value to set the scalar to.
+ */
+void TelemetryScalar::Set(mozilla::Telemetry::ScalarID aId, bool aValue) {
+ if (NS_WARN_IF(!IsValidEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ return;
+ }
+
+ ScalarKey uniqueId{static_cast<uint32_t>(aId), false};
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ if (internal_CanRecordScalar(locker, uniqueId, false) != ScalarResult::Ok) {
+ // We can't record this scalar. Bail out.
+ return;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ TelemetryIPCAccumulator::RecordChildScalarAction(
+ uniqueId.id, uniqueId.dynamic, ScalarActionType::eSet,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ if (internal_IsScalarDeserializing(locker)) {
+ internal_RecordScalarAction(locker, uniqueId.id, uniqueId.dynamic,
+ ScalarActionType::eSet, ScalarVariant(aValue));
+ return;
+ }
+
+ ScalarBase* scalar = nullptr;
+ nsresult rv =
+ internal_GetScalarByEnum(locker, uniqueId, ProcessID::Parent, &scalar);
+ if (NS_FAILED(rv)) {
+ return;
+ }
+
+ scalar->SetValue(aValue);
+}
+
+/**
+ * Sets the keyed scalar to the given numeric value.
+ *
+ * @param aId The scalar enum id.
+ * @param aKey The scalar key.
+ * @param aValue The numeric, unsigned value to set the scalar to.
+ */
+void TelemetryScalar::Set(mozilla::Telemetry::ScalarID aId,
+ const nsAString& aKey, uint32_t aValue) {
+ if (NS_WARN_IF(!IsValidEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ return;
+ }
+
+ ScalarKey uniqueId{static_cast<uint32_t>(aId), false};
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ if (internal_CanRecordScalar(locker, uniqueId, true) != ScalarResult::Ok) {
+ // We can't record this scalar. Bail out.
+ return;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ TelemetryIPCAccumulator::RecordChildKeyedScalarAction(
+ uniqueId.id, uniqueId.dynamic, aKey, ScalarActionType::eSet,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ if (internal_IsScalarDeserializing(locker)) {
+ internal_RecordKeyedScalarAction(locker, uniqueId.id, uniqueId.dynamic,
+ aKey, ScalarActionType::eSet,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ KeyedScalar* scalar = nullptr;
+ nsresult rv = internal_GetKeyedScalarByEnum(locker, uniqueId,
+ ProcessID::Parent, &scalar);
+ if (NS_FAILED(rv)) {
+ return;
+ }
+
+ scalar->SetValue(locker, aKey, aValue);
+}
+
+/**
+ * Sets the scalar to the given boolean value.
+ *
+ * @param aId The scalar enum id.
+ * @param aKey The scalar key.
+ * @param aValue The boolean value to set the scalar to.
+ */
+void TelemetryScalar::Set(mozilla::Telemetry::ScalarID aId,
+ const nsAString& aKey, bool aValue) {
+ if (NS_WARN_IF(!IsValidEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ return;
+ }
+
+ ScalarKey uniqueId{static_cast<uint32_t>(aId), false};
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ if (internal_CanRecordScalar(locker, uniqueId, true) != ScalarResult::Ok) {
+ // We can't record this scalar. Bail out.
+ return;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ TelemetryIPCAccumulator::RecordChildKeyedScalarAction(
+ uniqueId.id, uniqueId.dynamic, aKey, ScalarActionType::eSet,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ if (internal_IsScalarDeserializing(locker)) {
+ internal_RecordKeyedScalarAction(locker, uniqueId.id, uniqueId.dynamic,
+ aKey, ScalarActionType::eSet,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ KeyedScalar* scalar = nullptr;
+ nsresult rv = internal_GetKeyedScalarByEnum(locker, uniqueId,
+ ProcessID::Parent, &scalar);
+ if (NS_FAILED(rv)) {
+ return;
+ }
+
+ scalar->SetValue(locker, aKey, aValue);
+}
+
+/**
+ * Sets the scalar to the maximum of the current and the passed value.
+ *
+ * @param aName The scalar name.
+ * @param aVal The numeric value to set the scalar to.
+ * @param aCx The JS context.
+ * @return NS_OK (always) so that the JS API call doesn't throw. In case of
+ * errors, a warning level message is printed in the browser console.
+ */
+nsresult TelemetryScalar::SetMaximum(const nsACString& aName,
+ JS::HandleValue aVal, JSContext* aCx) {
+ // Unpack the aVal to nsIVariant. This uses the JS context.
+ nsCOMPtr<nsIVariant> unpackedVal;
+ nsresult rv = nsContentUtils::XPConnect()->JSToVariant(
+ aCx, aVal, getter_AddRefs(unpackedVal));
+ if (NS_FAILED(rv)) {
+ internal_LogScalarError(aName, ScalarResult::CannotUnpackVariant);
+ return NS_OK;
+ }
+
+ ScalarResult sr;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ sr = internal_UpdateScalar(locker, aName, ScalarActionType::eSetMaximum,
+ unpackedVal);
+ }
+
+ // Warn the user about the error if we need to.
+ if (sr != ScalarResult::Ok) {
+ internal_LogScalarError(aName, sr);
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Sets the scalar to the maximum of the current and the passed value.
+ *
+ * @param aName The scalar name.
+ * @param aKey The key name.
+ * @param aVal The numeric value to set the scalar to.
+ * @param aCx The JS context.
+ * @return NS_OK (always) so that the JS API call doesn't throw. In case of
+ * errors, a warning level message is printed in the browser console.
+ */
+nsresult TelemetryScalar::SetMaximum(const nsACString& aName,
+ const nsAString& aKey,
+ JS::HandleValue aVal, JSContext* aCx) {
+ // Unpack the aVal to nsIVariant. This uses the JS context.
+ nsCOMPtr<nsIVariant> unpackedVal;
+ nsresult rv = nsContentUtils::XPConnect()->JSToVariant(
+ aCx, aVal, getter_AddRefs(unpackedVal));
+ if (NS_FAILED(rv)) {
+ internal_LogScalarError(aName, ScalarResult::CannotUnpackVariant);
+ return NS_OK;
+ }
+
+ ScalarResult sr;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ sr = internal_UpdateKeyedScalar(locker, aName, aKey,
+ ScalarActionType::eSetMaximum, unpackedVal);
+ }
+
+ // Warn the user about the error if we need to.
+ if (sr != ScalarResult::Ok) {
+ internal_LogScalarError(aName, sr);
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Sets the scalar to the maximum of the current and the passed value.
+ *
+ * @param aId The scalar enum id.
+ * @param aValue The numeric value to set the scalar to.
+ */
+void TelemetryScalar::SetMaximum(mozilla::Telemetry::ScalarID aId,
+ uint32_t aValue) {
+ if (NS_WARN_IF(!IsValidEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ return;
+ }
+
+ ScalarKey uniqueId{static_cast<uint32_t>(aId), false};
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ if (internal_CanRecordScalar(locker, uniqueId, false) != ScalarResult::Ok) {
+ // We can't record this scalar. Bail out.
+ return;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ TelemetryIPCAccumulator::RecordChildScalarAction(
+ uniqueId.id, uniqueId.dynamic, ScalarActionType::eSetMaximum,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ if (internal_IsScalarDeserializing(locker)) {
+ internal_RecordScalarAction(locker, uniqueId.id, uniqueId.dynamic,
+ ScalarActionType::eSetMaximum,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ ScalarBase* scalar = nullptr;
+ nsresult rv =
+ internal_GetScalarByEnum(locker, uniqueId, ProcessID::Parent, &scalar);
+ if (NS_FAILED(rv)) {
+ return;
+ }
+
+ scalar->SetMaximum(aValue);
+}
+
+/**
+ * Sets the keyed scalar to the maximum of the current and the passed value.
+ *
+ * @param aId The scalar enum id.
+ * @param aKey The key name.
+ * @param aValue The numeric value to set the scalar to.
+ */
+void TelemetryScalar::SetMaximum(mozilla::Telemetry::ScalarID aId,
+ const nsAString& aKey, uint32_t aValue) {
+ if (NS_WARN_IF(!IsValidEnumId(aId))) {
+ MOZ_ASSERT_UNREACHABLE("Scalar usage requires valid ids.");
+ return;
+ }
+
+ ScalarKey uniqueId{static_cast<uint32_t>(aId), false};
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ if (internal_CanRecordScalar(locker, uniqueId, true) != ScalarResult::Ok) {
+ // We can't record this scalar. Bail out.
+ return;
+ }
+
+ // Accumulate in the child process if needed.
+ if (!XRE_IsParentProcess()) {
+ TelemetryIPCAccumulator::RecordChildKeyedScalarAction(
+ uniqueId.id, uniqueId.dynamic, aKey, ScalarActionType::eSetMaximum,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ if (internal_IsScalarDeserializing(locker)) {
+ internal_RecordKeyedScalarAction(locker, uniqueId.id, uniqueId.dynamic,
+ aKey, ScalarActionType::eSetMaximum,
+ ScalarVariant(aValue));
+ return;
+ }
+
+ KeyedScalar* scalar = nullptr;
+ nsresult rv = internal_GetKeyedScalarByEnum(locker, uniqueId,
+ ProcessID::Parent, &scalar);
+ if (NS_FAILED(rv)) {
+ return;
+ }
+
+ scalar->SetMaximum(locker, aKey, aValue);
+}
+
+nsresult TelemetryScalar::CreateSnapshots(unsigned int aDataset,
+ bool aClearScalars, JSContext* aCx,
+ uint8_t optional_argc,
+ JS::MutableHandle<JS::Value> aResult,
+ bool aFilterTest,
+ const nsACString& aStoreName) {
+ MOZ_ASSERT(
+ XRE_IsParentProcess(),
+ "Snapshotting scalars should only happen in the parent processes.");
+ // If no arguments were passed in, apply the default value.
+ if (!optional_argc) {
+ aClearScalars = false;
+ }
+
+ JS::Rooted<JSObject*> root_obj(aCx, JS_NewPlainObject(aCx));
+ if (!root_obj) {
+ return NS_ERROR_FAILURE;
+ }
+ aResult.setObject(*root_obj);
+
+ // Return `{}` in child processes.
+ if (!XRE_IsParentProcess()) {
+ return NS_OK;
+ }
+
+ // Only lock the mutex while accessing our data, without locking any JS
+ // related code.
+ ScalarSnapshotTable scalarsToReflect;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ nsresult rv = internal_GetScalarSnapshot(locker, scalarsToReflect, aDataset,
+ aClearScalars, aStoreName);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ }
+
+ // Reflect it to JS.
+ for (auto iter = scalarsToReflect.Iter(); !iter.Done(); iter.Next()) {
+ ScalarTupleArray& processScalars = iter.Data();
+ const char* processName = GetNameForProcessID(ProcessID(iter.Key()));
+
+ // Create the object that will hold the scalars for this process and add it
+ // to the returned root object.
+ JS::RootedObject processObj(aCx, JS_NewPlainObject(aCx));
+ if (!processObj || !JS_DefineProperty(aCx, root_obj, processName,
+ processObj, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ for (ScalarTupleArray::size_type i = 0; i < processScalars.Length(); i++) {
+ const ScalarDataTuple& scalar = processScalars[i];
+
+ const char* scalarName = mozilla::Get<0>(scalar);
+ if (aFilterTest && strncmp(TEST_SCALAR_PREFIX, scalarName,
+ strlen(TEST_SCALAR_PREFIX)) == 0) {
+ continue;
+ }
+
+ // Convert it to a JS Val.
+ JS::Rooted<JS::Value> scalarJsValue(aCx);
+ nsresult rv = nsContentUtils::XPConnect()->VariantToJS(
+ aCx, processObj, mozilla::Get<1>(scalar), &scalarJsValue);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ // Add it to the scalar object.
+ if (!JS_DefineProperty(aCx, processObj, scalarName, scalarJsValue,
+ JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult TelemetryScalar::CreateKeyedSnapshots(
+ unsigned int aDataset, bool aClearScalars, JSContext* aCx,
+ uint8_t optional_argc, JS::MutableHandle<JS::Value> aResult,
+ bool aFilterTest, const nsACString& aStoreName) {
+ MOZ_ASSERT(
+ XRE_IsParentProcess(),
+ "Snapshotting scalars should only happen in the parent processes.");
+ // If no arguments were passed in, apply the default value.
+ if (!optional_argc) {
+ aClearScalars = false;
+ }
+
+ JS::Rooted<JSObject*> root_obj(aCx, JS_NewPlainObject(aCx));
+ if (!root_obj) {
+ return NS_ERROR_FAILURE;
+ }
+ aResult.setObject(*root_obj);
+
+ // Return `{}` in child processes.
+ if (!XRE_IsParentProcess()) {
+ return NS_OK;
+ }
+
+ // Only lock the mutex while accessing our data, without locking any JS
+ // related code.
+ KeyedScalarSnapshotTable scalarsToReflect;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ nsresult rv = internal_GetKeyedScalarSnapshot(
+ locker, scalarsToReflect, aDataset, aClearScalars, aStoreName);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ }
+
+ // Reflect it to JS.
+ for (auto iter = scalarsToReflect.Iter(); !iter.Done(); iter.Next()) {
+ KeyedScalarTupleArray& processScalars = iter.Data();
+ const char* processName = GetNameForProcessID(ProcessID(iter.Key()));
+
+ // Create the object that will hold the scalars for this process and add it
+ // to the returned root object.
+ JS::RootedObject processObj(aCx, JS_NewPlainObject(aCx));
+ if (!processObj || !JS_DefineProperty(aCx, root_obj, processName,
+ processObj, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ for (KeyedScalarTupleArray::size_type i = 0; i < processScalars.Length();
+ i++) {
+ const KeyedScalarDataTuple& keyedScalarData = processScalars[i];
+
+ const char* scalarName = mozilla::Get<0>(keyedScalarData);
+ if (aFilterTest && strncmp(TEST_SCALAR_PREFIX, scalarName,
+ strlen(TEST_SCALAR_PREFIX)) == 0) {
+ continue;
+ }
+
+ // Go through each keyed scalar and create a keyed scalar object.
+ // This object will hold the values for all the keyed scalar keys.
+ JS::RootedObject keyedScalarObj(aCx, JS_NewPlainObject(aCx));
+
+ // Define a property for each scalar key, then add it to the keyed scalar
+ // object.
+ const nsTArray<KeyedScalar::KeyValuePair>& keyProps =
+ mozilla::Get<1>(keyedScalarData);
+ for (uint32_t i = 0; i < keyProps.Length(); i++) {
+ const KeyedScalar::KeyValuePair& keyData = keyProps[i];
+
+ // Convert the value for the key to a JSValue.
+ JS::Rooted<JS::Value> keyJsValue(aCx);
+ nsresult rv = nsContentUtils::XPConnect()->VariantToJS(
+ aCx, keyedScalarObj, keyData.second, &keyJsValue);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+
+ // Add the key to the scalar representation.
+ const NS_ConvertUTF8toUTF16 key(keyData.first);
+ if (!JS_DefineUCProperty(aCx, keyedScalarObj, key.Data(), key.Length(),
+ keyJsValue, JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ // Add the scalar to the root object.
+ if (!JS_DefineProperty(aCx, processObj, scalarName, keyedScalarObj,
+ JSPROP_ENUMERATE)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+ }
+
+ return NS_OK;
+}
+
+nsresult TelemetryScalar::RegisterScalars(const nsACString& aCategoryName,
+ JS::Handle<JS::Value> aScalarData,
+ bool aBuiltin, JSContext* cx) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Dynamic scalars should only be created in the parent process.");
+
+ if (!IsValidIdentifierString(aCategoryName, kMaximumCategoryNameLength, true,
+ false)) {
+ JS_ReportErrorASCII(cx, "Invalid category name %s.",
+ PromiseFlatCString(aCategoryName).get());
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ if (!aScalarData.isObject()) {
+ JS_ReportErrorASCII(cx, "Scalar data parameter should be an object");
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ JS::RootedObject obj(cx, &aScalarData.toObject());
+ JS::Rooted<JS::IdVector> scalarPropertyIds(cx, JS::IdVector(cx));
+ if (!JS_Enumerate(cx, obj, &scalarPropertyIds)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ // Collect the scalar data into local storage first.
+ // Only after successfully validating all contained scalars will we register
+ // them into global storage.
+ nsTArray<DynamicScalarInfo> newScalarInfos;
+
+ for (size_t i = 0, n = scalarPropertyIds.length(); i < n; i++) {
+ nsAutoJSString scalarName;
+ if (!scalarName.init(cx, scalarPropertyIds[i])) {
+ return NS_ERROR_FAILURE;
+ }
+
+ if (!IsValidIdentifierString(NS_ConvertUTF16toUTF8(scalarName),
+ kMaximumScalarNameLength, false, true)) {
+ JS_ReportErrorASCII(
+ cx, "Invalid scalar name %s.",
+ PromiseFlatCString(NS_ConvertUTF16toUTF8(scalarName)).get());
+ return NS_ERROR_INVALID_ARG;
+ }
+
+ // Join the category and the probe names.
+ nsPrintfCString fullName("%s.%s", PromiseFlatCString(aCategoryName).get(),
+ NS_ConvertUTF16toUTF8(scalarName).get());
+
+ JS::RootedValue value(cx);
+ if (!JS_GetPropertyById(cx, obj, scalarPropertyIds[i], &value) ||
+ !value.isObject()) {
+ return NS_ERROR_FAILURE;
+ }
+ JS::RootedObject scalarDef(cx, &value.toObject());
+
+ // Get the scalar's kind.
+ if (!JS_GetProperty(cx, scalarDef, "kind", &value) || !value.isInt32()) {
+ JS_ReportErrorASCII(cx, "Invalid or missing 'kind' for scalar %s.",
+ PromiseFlatCString(fullName).get());
+ return NS_ERROR_FAILURE;
+ }
+ uint32_t kind = static_cast<uint32_t>(value.toInt32());
+
+ // Get the optional scalar's recording policy (default to false).
+ bool hasProperty = false;
+ bool recordOnRelease = false;
+ if (JS_HasProperty(cx, scalarDef, "record_on_release", &hasProperty) &&
+ hasProperty) {
+ if (!JS_GetProperty(cx, scalarDef, "record_on_release", &value) ||
+ !value.isBoolean()) {
+ JS_ReportErrorASCII(cx, "Invalid 'record_on_release' for scalar %s.",
+ PromiseFlatCString(fullName).get());
+ return NS_ERROR_FAILURE;
+ }
+ recordOnRelease = static_cast<bool>(value.toBoolean());
+ }
+
+ // Get the optional scalar's keyed (default to false).
+ bool keyed = false;
+ if (JS_HasProperty(cx, scalarDef, "keyed", &hasProperty) && hasProperty) {
+ if (!JS_GetProperty(cx, scalarDef, "keyed", &value) ||
+ !value.isBoolean()) {
+ JS_ReportErrorASCII(cx, "Invalid 'keyed' for scalar %s.",
+ PromiseFlatCString(fullName).get());
+ return NS_ERROR_FAILURE;
+ }
+ keyed = static_cast<bool>(value.toBoolean());
+ }
+
+ // Get the optional scalar's expired state (default to false).
+ bool expired = false;
+ if (JS_HasProperty(cx, scalarDef, "expired", &hasProperty) && hasProperty) {
+ if (!JS_GetProperty(cx, scalarDef, "expired", &value) ||
+ !value.isBoolean()) {
+ JS_ReportErrorASCII(cx, "Invalid 'expired' for scalar %s.",
+ PromiseFlatCString(fullName).get());
+ return NS_ERROR_FAILURE;
+ }
+ expired = static_cast<bool>(value.toBoolean());
+ }
+
+ // Get the scalar's optional stores list (default to ["main"]).
+ nsTArray<nsCString> stores;
+ if (JS_HasProperty(cx, scalarDef, "stores", &hasProperty) && hasProperty) {
+ bool isArray = false;
+ if (!JS_GetProperty(cx, scalarDef, "stores", &value) ||
+ !JS::IsArrayObject(cx, value, &isArray) || !isArray) {
+ JS_ReportErrorASCII(cx, "Invalid 'stores' for scalar %s.",
+ PromiseFlatCString(fullName).get());
+ return NS_ERROR_FAILURE;
+ }
+
+ JS::RootedObject arrayObj(cx, &value.toObject());
+ uint32_t storesLength = 0;
+ if (!JS::GetArrayLength(cx, arrayObj, &storesLength)) {
+ JS_ReportErrorASCII(cx,
+ "Can't get 'stores' array length for scalar %s.",
+ PromiseFlatCString(fullName).get());
+ return NS_ERROR_FAILURE;
+ }
+
+ for (uint32_t i = 0; i < storesLength; ++i) {
+ JS::Rooted<JS::Value> elt(cx);
+ if (!JS_GetElement(cx, arrayObj, i, &elt)) {
+ JS_ReportErrorASCII(
+ cx, "Can't get element from scalar %s 'stores' array.",
+ PromiseFlatCString(fullName).get());
+ return NS_ERROR_FAILURE;
+ }
+ if (!elt.isString()) {
+ JS_ReportErrorASCII(cx,
+ "Element in scalar %s 'stores' array isn't a "
+ "string.",
+ PromiseFlatCString(fullName).get());
+ return NS_ERROR_FAILURE;
+ }
+
+ nsAutoJSString jsStr;
+ if (!jsStr.init(cx, elt)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ stores.AppendElement(NS_ConvertUTF16toUTF8(jsStr));
+ }
+ // In the event of the usual case (just "main"), save the storage.
+ if (stores.Length() == 1 && stores[0].EqualsLiteral("main")) {
+ stores.TruncateLength(0);
+ }
+ }
+
+ // We defer the actual registration here in case any other event description
+ // is invalid. In that case we don't need to roll back any partial
+ // registration.
+ newScalarInfos.AppendElement(
+ DynamicScalarInfo{kind, recordOnRelease, expired, fullName, keyed,
+ aBuiltin, std::move(stores)});
+ }
+
+ // Register the dynamic definition on the parent process.
+ nsTArray<DynamicScalarDefinition> ipcDefinitions;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ ::internal_RegisterScalars(locker, newScalarInfos);
+
+ // Convert the internal scalar representation to a stripped down IPC one.
+ ::internal_DynamicScalarToIPC(locker, newScalarInfos, ipcDefinitions);
+ }
+
+ // Propagate the registration to all the content-processes.
+ // Do not hold the mutex while calling IPC.
+ ::internal_BroadcastDefinitions(ipcDefinitions);
+
+ return NS_OK;
+}
+
+/**
+ * Count in Scalars how many of which events were recorded. See bug 1440673
+ *
+ * Event Telemetry unfortunately cannot use vanilla ScalarAdd because it needs
+ * to summarize events recorded in different processes to the
+ * telemetry.event_counts of the same process. Including "dynamic".
+ *
+ * @param aUniqueEventName - expected to be category#object#method
+ * @param aProcessType - the process of the event being summarized
+ * @param aDynamic - whether the event being summarized was dynamic
+ */
+void TelemetryScalar::SummarizeEvent(const nsCString& aUniqueEventName,
+ ProcessID aProcessType, bool aDynamic) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Only summarize events in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return;
+ }
+
+ StaticMutexAutoLock lock(gTelemetryScalarsMutex);
+
+ ScalarKey scalarKey{static_cast<uint32_t>(ScalarID::TELEMETRY_EVENT_COUNTS),
+ aDynamic};
+ if (aDynamic) {
+ nsresult rv = internal_GetEnumByScalarName(
+ lock, nsAutoCString("telemetry.dynamic_event_counts"), &scalarKey);
+ if (NS_FAILED(rv)) {
+ NS_WARNING(
+ "NS_FAILED getting ScalarKey for telemetry.dynamic_event_counts");
+ return;
+ }
+ }
+
+ KeyedScalar* scalar = nullptr;
+ nsresult rv =
+ internal_GetKeyedScalarByEnum(lock, scalarKey, aProcessType, &scalar);
+
+ if (NS_FAILED(rv)) {
+ NS_WARNING("NS_FAILED getting keyed scalar for event summary. Wut.");
+ return;
+ }
+
+ // Set this each time as it may have been cleared and recreated between calls
+ scalar->SetMaximumNumberOfKeys(kMaxEventSummaryKeys);
+
+ scalar->AddValue(lock, NS_ConvertASCIItoUTF16(aUniqueEventName), 1);
+}
+
+/**
+ * Resets all the stored scalars. This is intended to be only used in tests.
+ */
+void TelemetryScalar::ClearScalars() {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Scalars should only be cleared in the parent process.");
+ if (!XRE_IsParentProcess()) {
+ return;
+ }
+
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ gScalarStorageMap.Clear();
+ gKeyedScalarStorageMap.Clear();
+ gDynamicBuiltinScalarStorageMap.Clear();
+ gDynamicBuiltinKeyedScalarStorageMap.Clear();
+ gScalarsActions = nullptr;
+ gKeyedScalarsActions = nullptr;
+}
+
+size_t TelemetryScalar::GetMapShallowSizesOfExcludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ return gScalarNameIDMap.ShallowSizeOfExcludingThis(aMallocSizeOf);
+}
+
+size_t TelemetryScalar::GetScalarSizesOfIncludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ size_t n = 0;
+
+ auto getSizeOf = [aMallocSizeOf](auto& storageMap) {
+ size_t partial = 0;
+ for (auto iter = storageMap.Iter(); !iter.Done(); iter.Next()) {
+ auto scalarStorage = iter.UserData();
+ for (auto childIter = scalarStorage->Iter(); !childIter.Done();
+ childIter.Next()) {
+ auto scalar = childIter.UserData();
+ partial += scalar->SizeOfIncludingThis(aMallocSizeOf);
+ }
+ }
+ return partial;
+ };
+
+ // Account for all the storage used for the different scalar types.
+ n += getSizeOf(gScalarStorageMap);
+ n += getSizeOf(gKeyedScalarStorageMap);
+ n += getSizeOf(gDynamicBuiltinScalarStorageMap);
+ n += getSizeOf(gDynamicBuiltinKeyedScalarStorageMap);
+
+ return n;
+}
+
+void TelemetryScalar::UpdateChildData(
+ ProcessID aProcessType,
+ const nsTArray<mozilla::Telemetry::ScalarAction>& aScalarActions) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "The stored child processes scalar data must be updated from the "
+ "parent process.");
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ // If scalars are still being deserialized, we need to record the incoming
+ // operations as well.
+ if (internal_IsScalarDeserializing(locker)) {
+ for (const ScalarAction& action : aScalarActions) {
+ // We're only getting immutable access, so let's copy it
+ ScalarAction copy = action;
+ // Fix up the process type
+ copy.mProcessType = aProcessType;
+ internal_RecordScalarAction(locker, copy);
+ }
+
+ return;
+ }
+
+ internal_ApplyScalarActions(locker, aScalarActions, Some(aProcessType));
+}
+
+void TelemetryScalar::UpdateChildKeyedData(
+ ProcessID aProcessType,
+ const nsTArray<mozilla::Telemetry::KeyedScalarAction>& aScalarActions) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "The stored child processes keyed scalar data must be updated "
+ "from the parent process.");
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+
+ // If scalars are still being deserialized, we need to record the incoming
+ // operations as well.
+ if (internal_IsScalarDeserializing(locker)) {
+ for (const KeyedScalarAction& action : aScalarActions) {
+ // We're only getting immutable access, so let's copy it
+ KeyedScalarAction copy = action;
+ // Fix up the process type
+ copy.mProcessType = aProcessType;
+ internal_RecordKeyedScalarAction(locker, copy);
+ }
+
+ return;
+ }
+
+ internal_ApplyKeyedScalarActions(locker, aScalarActions, Some(aProcessType));
+}
+
+void TelemetryScalar::RecordDiscardedData(
+ ProcessID aProcessType,
+ const mozilla::Telemetry::DiscardedData& aDiscardedData) {
+ MOZ_ASSERT(XRE_IsParentProcess(),
+ "Discarded Data must be updated from the parent process.");
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ if (!internal_CanRecordBase(locker)) {
+ return;
+ }
+
+ if (GetCurrentProduct() == SupportedProduct::GeckoviewStreaming) {
+ return;
+ }
+
+ ScalarBase* scalar = nullptr;
+ mozilla::DebugOnly<nsresult> rv;
+
+ rv = internal_GetScalarByEnum(
+ locker,
+ ScalarKey{
+ static_cast<uint32_t>(ScalarID::TELEMETRY_DISCARDED_ACCUMULATIONS),
+ false},
+ aProcessType, &scalar);
+ MOZ_ASSERT(NS_SUCCEEDED(rv));
+ scalar->AddValue(aDiscardedData.mDiscardedHistogramAccumulations);
+
+ rv = internal_GetScalarByEnum(
+ locker,
+ ScalarKey{static_cast<uint32_t>(
+ ScalarID::TELEMETRY_DISCARDED_KEYED_ACCUMULATIONS),
+ false},
+ aProcessType, &scalar);
+ MOZ_ASSERT(NS_SUCCEEDED(rv));
+ scalar->AddValue(aDiscardedData.mDiscardedKeyedHistogramAccumulations);
+
+ rv = internal_GetScalarByEnum(
+ locker,
+ ScalarKey{
+ static_cast<uint32_t>(ScalarID::TELEMETRY_DISCARDED_SCALAR_ACTIONS),
+ false},
+ aProcessType, &scalar);
+ MOZ_ASSERT(NS_SUCCEEDED(rv));
+ scalar->AddValue(aDiscardedData.mDiscardedScalarActions);
+
+ rv = internal_GetScalarByEnum(
+ locker,
+ ScalarKey{static_cast<uint32_t>(
+ ScalarID::TELEMETRY_DISCARDED_KEYED_SCALAR_ACTIONS),
+ false},
+ aProcessType, &scalar);
+ MOZ_ASSERT(NS_SUCCEEDED(rv));
+ scalar->AddValue(aDiscardedData.mDiscardedKeyedScalarActions);
+
+ rv = internal_GetScalarByEnum(
+ locker,
+ ScalarKey{
+ static_cast<uint32_t>(ScalarID::TELEMETRY_DISCARDED_CHILD_EVENTS),
+ false},
+ aProcessType, &scalar);
+ MOZ_ASSERT(NS_SUCCEEDED(rv));
+ scalar->AddValue(aDiscardedData.mDiscardedChildEvents);
+}
+
+/**
+ * Get the dynamic scalar definitions in an IPC-friendly
+ * structure.
+ */
+void TelemetryScalar::GetDynamicScalarDefinitions(
+ nsTArray<DynamicScalarDefinition>& aDefArray) {
+ MOZ_ASSERT(XRE_IsParentProcess());
+ if (!gDynamicScalarInfo) {
+ // Don't have dynamic scalar definitions. Bail out!
+ return;
+ }
+
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ internal_DynamicScalarToIPC(locker, *gDynamicScalarInfo, aDefArray);
+}
+
+/**
+ * This adds the dynamic scalar definitions coming from
+ * the parent process to this child process. If a dynamic
+ * scalar definition is already defined, check if the new definition
+ * makes the scalar expired and eventually update the expiration
+ * state.
+ */
+void TelemetryScalar::AddDynamicScalarDefinitions(
+ const nsTArray<DynamicScalarDefinition>& aDefs) {
+ MOZ_ASSERT(!XRE_IsParentProcess());
+
+ nsTArray<DynamicScalarInfo> dynamicStubs;
+
+ // Populate the definitions array before acquiring the lock.
+ for (auto& def : aDefs) {
+ bool recordOnRelease = def.dataset == nsITelemetry::DATASET_ALL_CHANNELS;
+ dynamicStubs.AppendElement(DynamicScalarInfo{def.type,
+ recordOnRelease,
+ def.expired,
+ def.name,
+ def.keyed,
+ def.builtin,
+ {} /* stores */});
+ }
+
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ internal_RegisterScalars(locker, dynamicStubs);
+ }
+}
+
+nsresult TelemetryScalar::GetAllStores(StringHashSet& set) {
+ // Static stores
+ for (uint32_t storeIdx : gScalarStoresTable) {
+ const char* name = &gScalarsStringTable[storeIdx];
+ nsAutoCString store;
+ store.AssignASCII(name);
+ if (!set.PutEntry(store)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ // Dynamic stores
+ for (auto& ptr : *gDynamicStoreNames) {
+ nsAutoCString store;
+ ptr->ToUTF8String(store);
+ if (!set.PutEntry(store)) {
+ return NS_ERROR_FAILURE;
+ }
+ }
+
+ return NS_OK;
+}
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PUBLIC: GeckoView serialization/deserialization functions.
+
+/**
+ * Write the scalar data to the provided Json object, for
+ * GeckoView measurement persistence. The output format is the same one used
+ * for snapshotting the scalars.
+ *
+ * @param {aWriter} The JSON object to write to.
+ * @returns NS_OK or a failure value explaining why persistence failed.
+ */
+nsresult TelemetryScalar::SerializeScalars(mozilla::JSONWriter& aWriter) {
+ // Get a copy of the data, without clearing.
+ ScalarSnapshotTable scalarsToReflect;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ // For persistence, we care about all the datasets. Worst case, they
+ // will be empty.
+ nsresult rv = internal_GetScalarSnapshot(
+ locker, scalarsToReflect, nsITelemetry::DATASET_PRERELEASE_CHANNELS,
+ false, /*aClearScalars*/
+ "main"_ns);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ }
+
+ // Persist the scalars to the JSON object.
+ for (auto iter = scalarsToReflect.Iter(); !iter.Done(); iter.Next()) {
+ ScalarTupleArray& processScalars = iter.Data();
+ const char* processName = GetNameForProcessID(ProcessID(iter.Key()));
+
+ aWriter.StartObjectProperty(mozilla::MakeStringSpan(processName));
+
+ for (const ScalarDataTuple& scalar : processScalars) {
+ nsresult rv = WriteVariantToJSONWriter(
+ mozilla::Get<2>(scalar) /*aScalarType*/,
+ mozilla::Get<1>(scalar) /*aInputValue*/,
+ mozilla::MakeStringSpan(mozilla::Get<0>(scalar)) /*aPropertyName*/,
+ aWriter /*aWriter*/);
+ if (NS_FAILED(rv)) {
+ // Skip this scalar if we failed to write it. We don't bail out just
+ // yet as we may salvage other scalars. We eventually need to call
+ // EndObject.
+ continue;
+ }
+ }
+
+ aWriter.EndObject();
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Write the keyed scalar data to the provided Json object, for
+ * GeckoView measurement persistence. The output format is the same
+ * one used for snapshotting the keyed scalars.
+ *
+ * @param {aWriter} The JSON object to write to.
+ * @returns NS_OK or a failure value explaining why persistence failed.
+ */
+nsresult TelemetryScalar::SerializeKeyedScalars(mozilla::JSONWriter& aWriter) {
+ // Get a copy of the data, without clearing.
+ KeyedScalarSnapshotTable keyedScalarsToReflect;
+ {
+ StaticMutexAutoLock locker(gTelemetryScalarsMutex);
+ // For persistence, we care about all the datasets. Worst case, they
+ // will be empty.
+ nsresult rv = internal_GetKeyedScalarSnapshot(
+ locker, keyedScalarsToReflect,
+ nsITelemetry::DATASET_PRERELEASE_CHANNELS, false, /*aClearScalars*/
+ "main"_ns);
+ if (NS_FAILED(rv)) {
+ return rv;
+ }
+ }
+
+ // Persist the scalars to the JSON object.
+ for (auto iter = keyedScalarsToReflect.Iter(); !iter.Done(); iter.Next()) {
+ KeyedScalarTupleArray& processScalars = iter.Data();
+ const char* processName = GetNameForProcessID(ProcessID(iter.Key()));
+
+ aWriter.StartObjectProperty(mozilla::MakeStringSpan(processName));
+
+ for (const KeyedScalarDataTuple& keyedScalarData : processScalars) {
+ aWriter.StartObjectProperty(
+ mozilla::MakeStringSpan(mozilla::Get<0>(keyedScalarData)));
+
+ // Define a property for each scalar key, then add it to the keyed scalar
+ // object.
+ const nsTArray<KeyedScalar::KeyValuePair>& keyProps =
+ mozilla::Get<1>(keyedScalarData);
+ for (const KeyedScalar::KeyValuePair& keyData : keyProps) {
+ nsresult rv = WriteVariantToJSONWriter(
+ mozilla::Get<2>(keyedScalarData) /*aScalarType*/,
+ keyData.second /*aInputValue*/,
+ PromiseFlatCString(keyData.first) /*aOutKey*/, aWriter /*aWriter*/);
+ if (NS_FAILED(rv)) {
+ // Skip this scalar if we failed to write it. We don't bail out just
+ // yet as we may salvage other scalars. We eventually need to call
+ // EndObject.
+ continue;
+ }
+ }
+ aWriter.EndObject();
+ }
+ aWriter.EndObject();
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Load the persisted measurements from a Json object and inject them
+ * in the relevant process storage.
+ *
+ * @param {aData} The input Json object.
+ * @returns NS_OK if loading was performed, an error code explaining the
+ * failure reason otherwise.
+ */
+nsresult TelemetryScalar::DeserializePersistedScalars(JSContext* aCx,
+ JS::HandleValue aData) {
+ MOZ_ASSERT(XRE_IsParentProcess(), "Only load scalars in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ typedef std::pair<nsCString, nsCOMPtr<nsIVariant>> PersistedScalarPair;
+ typedef nsTArray<PersistedScalarPair> PersistedScalarArray;
+ typedef nsDataHashtable<ProcessIDHashKey, PersistedScalarArray>
+ PeristedScalarStorage;
+
+ PeristedScalarStorage scalarsToUpdate;
+
+ // Before updating the scalars, we need to get the data out of the JS
+ // wrappers. We can't hold the scalars mutex while handling JS stuff.
+ // Build a <scalar name, value> map.
+ JS::RootedObject scalarDataObj(aCx, &aData.toObject());
+ JS::Rooted<JS::IdVector> processes(aCx, JS::IdVector(aCx));
+ if (!JS_Enumerate(aCx, scalarDataObj, &processes)) {
+ // We can't even enumerate the processes in the loaded data, so
+ // there is nothing we could recover from the persistence file. Bail out.
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+
+ // The following block of code attempts to extract as much data as possible
+ // from the serialized JSON, even in case of light data corruptions: if, for
+ // example, the data for a single process is corrupted or is in an unexpected
+ // form, we press on and attempt to load the data for the other processes.
+ JS::RootedId process(aCx);
+ for (auto& processVal : processes) {
+ // This is required as JS API calls require an Handle<jsid> and not a
+ // plain jsid.
+ process = processVal;
+ // Get the process name.
+ nsAutoJSString processNameJS;
+ if (!processNameJS.init(aCx, process)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Make sure it's valid. Note that this is safe to call outside
+ // of a locked section.
+ NS_ConvertUTF16toUTF8 processName(processNameJS);
+ ProcessID processID = GetIDForProcessName(processName.get());
+ if (processID == ProcessID::Count) {
+ NS_WARNING(
+ nsPrintfCString("Failed to get process ID for %s", processName.get())
+ .get());
+ continue;
+ }
+
+ // And its probes.
+ JS::RootedValue processData(aCx);
+ if (!JS_GetPropertyById(aCx, scalarDataObj, process, &processData)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ if (!processData.isObject()) {
+ // |processData| should be an object containing scalars. If this is
+ // not the case, silently skip and try to load the data for the other
+ // processes.
+ continue;
+ }
+
+ // Iterate through each scalar.
+ JS::RootedObject processDataObj(aCx, &processData.toObject());
+ JS::Rooted<JS::IdVector> scalars(aCx, JS::IdVector(aCx));
+ if (!JS_Enumerate(aCx, processDataObj, &scalars)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ JS::RootedId scalar(aCx);
+ for (auto& scalarVal : scalars) {
+ scalar = scalarVal;
+ // Get the scalar name.
+ nsAutoJSString scalarName;
+ if (!scalarName.init(aCx, scalar)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Get the scalar value as a JS value.
+ JS::RootedValue scalarValue(aCx);
+ if (!JS_GetPropertyById(aCx, processDataObj, scalar, &scalarValue)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ if (scalarValue.isNullOrUndefined()) {
+ // We can't set scalars to null or undefined values, skip this
+ // and try to load other scalars.
+ continue;
+ }
+
+ // Unpack the aVal to nsIVariant.
+ nsCOMPtr<nsIVariant> unpackedVal;
+ nsresult rv = nsContentUtils::XPConnect()->JSToVariant(
+ aCx, scalarValue, getter_AddRefs(unpackedVal));
+ if (NS_FAILED(rv)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Add the scalar to the map.
+ PersistedScalarArray& processScalars =
+ scalarsToUpdate.GetOrInsert(static_cast<uint32_t>(processID));
+ processScalars.AppendElement(std::make_pair(
+ nsCString(NS_ConvertUTF16toUTF8(scalarName)), unpackedVal));
+ }
+ }
+
+ // Now that all the JS specific operations are finished, update the scalars.
+ {
+ StaticMutexAutoLock lock(gTelemetryScalarsMutex);
+
+ for (auto iter = scalarsToUpdate.ConstIter(); !iter.Done(); iter.Next()) {
+ PersistedScalarArray& processScalars = iter.Data();
+ for (PersistedScalarArray::size_type i = 0; i < processScalars.Length();
+ i++) {
+ mozilla::Unused << internal_UpdateScalar(
+ lock, processScalars[i].first, ScalarActionType::eSet,
+ processScalars[i].second, ProcessID(iter.Key()), true /* aForce */);
+ }
+ }
+ }
+
+ return NS_OK;
+}
+
+/**
+ * Load the persisted measurements from a Json object and injects them
+ * in the relevant process storage.
+ *
+ * @param {aData} The input Json object.
+ * @returns NS_OK if loading was performed, an error code explaining the
+ * failure reason otherwise.
+ */
+nsresult TelemetryScalar::DeserializePersistedKeyedScalars(
+ JSContext* aCx, JS::HandleValue aData) {
+ MOZ_ASSERT(XRE_IsParentProcess(), "Only load scalars in the parent process");
+ if (!XRE_IsParentProcess()) {
+ return NS_ERROR_FAILURE;
+ }
+
+ typedef mozilla::Tuple<nsCString, nsString, nsCOMPtr<nsIVariant>>
+ PersistedKeyedScalarTuple;
+ typedef nsTArray<PersistedKeyedScalarTuple> PersistedKeyedScalarArray;
+ typedef nsDataHashtable<ProcessIDHashKey, PersistedKeyedScalarArray>
+ PeristedKeyedScalarStorage;
+
+ PeristedKeyedScalarStorage scalarsToUpdate;
+
+ // Before updating the keyed scalars, we need to get the data out of the JS
+ // wrappers. We can't hold the scalars mutex while handling JS stuff.
+ // Build a <scalar name, value> map.
+ JS::RootedObject scalarDataObj(aCx, &aData.toObject());
+ JS::Rooted<JS::IdVector> processes(aCx, JS::IdVector(aCx));
+ if (!JS_Enumerate(aCx, scalarDataObj, &processes)) {
+ // We can't even enumerate the processes in the loaded data, so
+ // there is nothing we could recover from the persistence file. Bail out.
+ JS_ClearPendingException(aCx);
+ return NS_ERROR_FAILURE;
+ }
+
+ // The following block of code attempts to extract as much data as possible
+ // from the serialized JSON, even in case of light data corruptions: if, for
+ // example, the data for a single process is corrupted or is in an unexpected
+ // form, we press on and attempt to load the data for the other processes.
+ JS::RootedId process(aCx);
+ for (auto& processVal : processes) {
+ process = processVal;
+ // Get the process name.
+ nsAutoJSString processNameJS;
+ if (!processNameJS.init(aCx, process)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Make sure it's valid. Note that this is safe to call outside
+ // of a locked section.
+ NS_ConvertUTF16toUTF8 processName(processNameJS);
+ ProcessID processID = GetIDForProcessName(processName.get());
+ if (processID == ProcessID::Count) {
+ NS_WARNING(
+ nsPrintfCString("Failed to get process ID for %s", processName.get())
+ .get());
+ continue;
+ }
+
+ // And its probes.
+ JS::RootedValue processData(aCx);
+ if (!JS_GetPropertyById(aCx, scalarDataObj, process, &processData)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ if (!processData.isObject()) {
+ // |processData| should be an object containing scalars. If this is
+ // not the case, silently skip and try to load the data for the other
+ // processes.
+ continue;
+ }
+
+ // Iterate through each keyed scalar.
+ JS::RootedObject processDataObj(aCx, &processData.toObject());
+ JS::Rooted<JS::IdVector> keyedScalars(aCx, JS::IdVector(aCx));
+ if (!JS_Enumerate(aCx, processDataObj, &keyedScalars)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ JS::RootedId keyedScalar(aCx);
+ for (auto& keyedScalarVal : keyedScalars) {
+ keyedScalar = keyedScalarVal;
+ // Get the scalar name.
+ nsAutoJSString scalarName;
+ if (!scalarName.init(aCx, keyedScalar)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Get the data for this keyed scalar.
+ JS::RootedValue keyedScalarData(aCx);
+ if (!JS_GetPropertyById(aCx, processDataObj, keyedScalar,
+ &keyedScalarData)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ if (!keyedScalarData.isObject()) {
+ // Keyed scalar data need to be an object. If that's not the case, skip
+ // it and try to load the rest of the data.
+ continue;
+ }
+
+ // Get the keys in the keyed scalar.
+ JS::RootedObject keyedScalarDataObj(aCx, &keyedScalarData.toObject());
+ JS::Rooted<JS::IdVector> keys(aCx, JS::IdVector(aCx));
+ if (!JS_Enumerate(aCx, keyedScalarDataObj, &keys)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ JS::RootedId key(aCx);
+ for (auto keyVal : keys) {
+ key = keyVal;
+ // Get the process name.
+ nsAutoJSString keyName;
+ if (!keyName.init(aCx, key)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Get the scalar value as a JS value.
+ JS::RootedValue scalarValue(aCx);
+ if (!JS_GetPropertyById(aCx, keyedScalarDataObj, key, &scalarValue)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ if (scalarValue.isNullOrUndefined()) {
+ // We can't set scalars to null or undefined values, skip this
+ // and try to load other scalars.
+ continue;
+ }
+
+ // Unpack the aVal to nsIVariant.
+ nsCOMPtr<nsIVariant> unpackedVal;
+ nsresult rv = nsContentUtils::XPConnect()->JSToVariant(
+ aCx, scalarValue, getter_AddRefs(unpackedVal));
+ if (NS_FAILED(rv)) {
+ JS_ClearPendingException(aCx);
+ continue;
+ }
+
+ // Add the scalar to the map.
+ PersistedKeyedScalarArray& processScalars =
+ scalarsToUpdate.GetOrInsert(static_cast<uint32_t>(processID));
+ processScalars.AppendElement(
+ mozilla::MakeTuple(nsCString(NS_ConvertUTF16toUTF8(scalarName)),
+ nsString(keyName), unpackedVal));
+ }
+ }
+ }
+
+ // Now that all the JS specific operations are finished, update the scalars.
+ {
+ StaticMutexAutoLock lock(gTelemetryScalarsMutex);
+
+ for (auto iter = scalarsToUpdate.ConstIter(); !iter.Done(); iter.Next()) {
+ PersistedKeyedScalarArray& processScalars = iter.Data();
+ for (PersistedKeyedScalarArray::size_type i = 0;
+ i < processScalars.Length(); i++) {
+ mozilla::Unused << internal_UpdateKeyedScalar(
+ lock, mozilla::Get<0>(processScalars[i]),
+ mozilla::Get<1>(processScalars[i]), ScalarActionType::eSet,
+ mozilla::Get<2>(processScalars[i]), ProcessID(iter.Key()),
+ true /* aForce */);
+ }
+ }
+ }
+
+ return NS_OK;
+}
diff --git a/toolkit/components/telemetry/core/TelemetryScalar.h b/toolkit/components/telemetry/core/TelemetryScalar.h
new file mode 100644
index 0000000000..00fa85835e
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryScalar.h
@@ -0,0 +1,130 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryScalar_h__
+#define TelemetryScalar_h__
+
+#include <stdint.h>
+#include "mozilla/TelemetryProcessEnums.h"
+#include "mozilla/TelemetryScalarEnums.h"
+#include "nsTArray.h"
+#include "TelemetryCommon.h"
+
+// This module is internal to Telemetry. It encapsulates Telemetry's
+// scalar accumulation and storage logic. It should only be used by
+// Telemetry.cpp. These functions should not be used anywhere else.
+// For the public interface to Telemetry functionality, see Telemetry.h.
+
+namespace mozilla {
+// This is only used for the GeckoView persistence.
+class JSONWriter;
+namespace Telemetry {
+struct ScalarAction;
+struct KeyedScalarAction;
+struct DiscardedData;
+struct DynamicScalarDefinition;
+} // namespace Telemetry
+} // namespace mozilla
+
+namespace TelemetryScalar {
+
+void InitializeGlobalState(bool canRecordBase, bool canRecordExtended);
+void DeInitializeGlobalState();
+
+void SetCanRecordBase(bool b);
+void SetCanRecordExtended(bool b);
+
+// JS API Endpoints.
+nsresult Add(const nsACString& aName, JS::HandleValue aVal, JSContext* aCx);
+nsresult Set(const nsACString& aName, JS::HandleValue aVal, JSContext* aCx);
+nsresult SetMaximum(const nsACString& aName, JS::HandleValue aVal,
+ JSContext* aCx);
+nsresult CreateSnapshots(unsigned int aDataset, bool aClearScalars,
+ JSContext* aCx, uint8_t optional_argc,
+ JS::MutableHandle<JS::Value> aResult, bool aFilterTest,
+ const nsACString& aStoreName);
+
+// Keyed JS API Endpoints.
+nsresult Add(const nsACString& aName, const nsAString& aKey,
+ JS::HandleValue aVal, JSContext* aCx);
+nsresult Set(const nsACString& aName, const nsAString& aKey,
+ JS::HandleValue aVal, JSContext* aCx);
+nsresult SetMaximum(const nsACString& aName, const nsAString& aKey,
+ JS::HandleValue aVal, JSContext* aCx);
+nsresult CreateKeyedSnapshots(unsigned int aDataset, bool aClearScalars,
+ JSContext* aCx, uint8_t optional_argc,
+ JS::MutableHandle<JS::Value> aResult,
+ bool aFilterTest, const nsACString& aStoreName);
+
+// C++ API Endpoints.
+void Add(mozilla::Telemetry::ScalarID aId, uint32_t aValue);
+void Set(mozilla::Telemetry::ScalarID aId, uint32_t aValue);
+void Set(mozilla::Telemetry::ScalarID aId, const nsAString& aValue);
+void Set(mozilla::Telemetry::ScalarID aId, bool aValue);
+void SetMaximum(mozilla::Telemetry::ScalarID aId, uint32_t aValue);
+
+// Keyed C++ API Endpoints.
+void Add(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ uint32_t aValue);
+void Set(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ uint32_t aValue);
+void Set(mozilla::Telemetry::ScalarID aId, const nsAString& aKey, bool aValue);
+void SetMaximum(mozilla::Telemetry::ScalarID aId, const nsAString& aKey,
+ uint32_t aValue);
+
+nsresult RegisterScalars(const nsACString& aCategoryName,
+ JS::Handle<JS::Value> aScalarData, bool aBuiltin,
+ JSContext* cx);
+
+// Event Summary
+void SummarizeEvent(const nsCString& aUniqueEventName,
+ mozilla::Telemetry::ProcessID aProcessType, bool aDynamic);
+
+// Only to be used for testing.
+void ClearScalars();
+
+size_t GetMapShallowSizesOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf);
+size_t GetScalarSizesOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf);
+
+void UpdateChildData(
+ mozilla::Telemetry::ProcessID aProcessType,
+ const nsTArray<mozilla::Telemetry::ScalarAction>& aScalarActions);
+
+void UpdateChildKeyedData(
+ mozilla::Telemetry::ProcessID aProcessType,
+ const nsTArray<mozilla::Telemetry::KeyedScalarAction>& aScalarActions);
+
+void RecordDiscardedData(
+ mozilla::Telemetry::ProcessID aProcessType,
+ const mozilla::Telemetry::DiscardedData& aDiscardedData);
+
+void GetDynamicScalarDefinitions(
+ nsTArray<mozilla::Telemetry::DynamicScalarDefinition>&);
+void AddDynamicScalarDefinitions(
+ const nsTArray<mozilla::Telemetry::DynamicScalarDefinition>&);
+
+/**
+ * Append the list of registered stores to the given set.
+ * This includes dynamic stores.
+ */
+nsresult GetAllStores(mozilla::Telemetry::Common::StringHashSet& set);
+
+// They are responsible for updating in-memory probes with the data persisted
+// on the disk and vice-versa.
+nsresult SerializeScalars(mozilla::JSONWriter& aWriter);
+nsresult SerializeKeyedScalars(mozilla::JSONWriter& aWriter);
+nsresult DeserializePersistedScalars(JSContext* aCx, JS::HandleValue aData);
+nsresult DeserializePersistedKeyedScalars(JSContext* aCx,
+ JS::HandleValue aData);
+// Mark deserialization as in progress.
+// After this, all scalar operations are recorded into the pending operations
+// list.
+void DeserializationStarted();
+// Apply all operations from the pending operations list and mark
+// deserialization finished afterwards.
+void ApplyPendingOperations();
+} // namespace TelemetryScalar
+
+#endif // TelemetryScalar_h__
diff --git a/toolkit/components/telemetry/core/TelemetryUserInteraction.cpp b/toolkit/components/telemetry/core/TelemetryUserInteraction.cpp
new file mode 100644
index 0000000000..8de3666992
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryUserInteraction.cpp
@@ -0,0 +1,103 @@
+/* -*- 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 "nsAString.h"
+#include "nsClassHashtable.h"
+#include "nsITelemetry.h"
+#include "MainThreadUtils.h"
+#include "TelemetryUserInteraction.h"
+#include "TelemetryUserInteractionData.h"
+#include "TelemetryUserInteractionNameMap.h"
+#include "UserInteractionInfo.h"
+
+using mozilla::Telemetry::UserInteractionIDByNameLookup;
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE TYPES
+namespace {
+
+struct UserInteractionKey {
+ uint32_t id;
+};
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// PRIVATE STATE, SHARED BY ALL THREADS
+
+namespace {
+// Set to true once this global state has been initialized.
+bool gInitDone = false;
+
+bool gCanRecordBase;
+bool gCanRecordExtended;
+} // namespace
+
+namespace {
+// Implements the methods for UserInteractionInfo.
+const char* UserInteractionInfo::name() const {
+ return &gUserInteractionsStringTable[this->name_offset];
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// EXTERNALLY VISIBLE FUNCTIONS in namespace TelemetryUserInteraction::
+
+void TelemetryUserInteraction::InitializeGlobalState(bool aCanRecordBase,
+ bool aCanRecordExtended) {
+ if (!XRE_IsParentProcess()) {
+ return;
+ }
+
+ MOZ_ASSERT(NS_IsMainThread());
+ MOZ_ASSERT(!gInitDone,
+ "TelemetryUserInteraction::InitializeGlobalState "
+ "may only be called once");
+
+ gCanRecordBase = aCanRecordBase;
+ gCanRecordExtended = aCanRecordExtended;
+ gInitDone = true;
+}
+
+void TelemetryUserInteraction::DeInitializeGlobalState() {
+ if (!XRE_IsParentProcess()) {
+ return;
+ }
+
+ MOZ_ASSERT(NS_IsMainThread());
+ MOZ_ASSERT(gInitDone);
+ if (!gInitDone) {
+ return;
+ }
+
+ gInitDone = false;
+}
+
+bool TelemetryUserInteraction::CanRecord(const nsAString& aName) {
+ if (!gCanRecordBase) {
+ return false;
+ }
+
+ nsCString name = NS_ConvertUTF16toUTF8(aName);
+ const uint32_t idx = UserInteractionIDByNameLookup(name);
+
+ MOZ_DIAGNOSTIC_ASSERT(
+ idx < mozilla::Telemetry::UserInteractionID::UserInteractionCount,
+ "Intermediate lookup should always give a valid index.");
+
+ if (name.Equals(gUserInteractions[idx].name())) {
+ return true;
+ }
+
+ return false;
+}
diff --git a/toolkit/components/telemetry/core/TelemetryUserInteraction.h b/toolkit/components/telemetry/core/TelemetryUserInteraction.h
new file mode 100644
index 0000000000..ea03533ec3
--- /dev/null
+++ b/toolkit/components/telemetry/core/TelemetryUserInteraction.h
@@ -0,0 +1,18 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryUserInteraction_h__
+#define TelemetryUserInteraction_h__
+
+namespace TelemetryUserInteraction {
+
+void InitializeGlobalState(bool canRecordBase, bool canRecordExtended);
+void DeInitializeGlobalState();
+
+bool CanRecord(const nsAString& aName);
+
+} // namespace TelemetryUserInteraction
+
+#endif // TelemetryUserInteraction_h__
diff --git a/toolkit/components/telemetry/core/UserInteractionInfo.h b/toolkit/components/telemetry/core/UserInteractionInfo.h
new file mode 100644
index 0000000000..e20017b790
--- /dev/null
+++ b/toolkit/components/telemetry/core/UserInteractionInfo.h
@@ -0,0 +1,30 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryUserInteractionInfo_h__
+#define TelemetryUserInteractionInfo_h__
+
+#include "TelemetryCommon.h"
+
+// This module is internal to Telemetry. It defines a structure that holds the
+// UserInteraction info. It should only be used by
+// TelemetryUserInteractionData.h automatically generated file and
+// TelemetryUserInteraction.cpp. This should not be used anywhere else. For the
+// public interface to Telemetry functionality, see Telemetry.h.
+
+namespace {
+
+struct UserInteractionInfo {
+ const uint32_t name_offset;
+
+ explicit constexpr UserInteractionInfo(const uint32_t aNameOffset)
+ : name_offset(aNameOffset) {}
+
+ const char* name() const;
+};
+
+} // namespace
+
+#endif // TelemetryUserInteractionInfo_h__
diff --git a/toolkit/components/telemetry/core/components.conf b/toolkit/components/telemetry/core/components.conf
new file mode 100644
index 0000000000..4bf617aa6a
--- /dev/null
+++ b/toolkit/components/telemetry/core/components.conf
@@ -0,0 +1,21 @@
+# -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
+# vim: set filetype=python:
+# 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/.
+
+Headers = ['mozilla/Telemetry.h']
+
+UnloadFunc = 'mozilla::Telemetry::ShutdownTelemetry'
+
+Classes = [
+ {
+ 'js_name': 'telemetry',
+ 'cid': '{aea477f2-b3a2-469c-aa29-0a82d132b829}',
+ 'contract_ids': ['@mozilla.org/base/telemetry;1'],
+ 'interfaces': ['nsITelemetry'],
+ 'singleton': True,
+ 'type': 'nsITelemetry',
+ 'processes': ProcessSelector.ALLOW_IN_GPU_RDD_VR_AND_SOCKET_PROCESS,
+ },
+]
diff --git a/toolkit/components/telemetry/core/ipc/TelemetryComms.h b/toolkit/components/telemetry/core/ipc/TelemetryComms.h
new file mode 100644
index 0000000000..452d8e4938
--- /dev/null
+++ b/toolkit/components/telemetry/core/ipc/TelemetryComms.h
@@ -0,0 +1,416 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef Telemetry_Comms_h__
+#define Telemetry_Comms_h__
+
+#include "ipc/IPCMessageUtils.h"
+#include "ipc/IPCMessageUtilsSpecializations.h"
+#include "mozilla/TelemetryProcessEnums.h"
+#include "mozilla/TimeStamp.h"
+#include "mozilla/Variant.h"
+#include "nsITelemetry.h"
+
+namespace mozilla {
+namespace Telemetry {
+
+// Histogram accumulation types.
+enum HistogramID : uint32_t;
+
+struct HistogramAccumulation {
+ mozilla::Telemetry::HistogramID mId;
+ uint32_t mSample;
+};
+
+struct KeyedHistogramAccumulation {
+ mozilla::Telemetry::HistogramID mId;
+ uint32_t mSample;
+ nsCString mKey;
+};
+
+// Scalar accumulation types.
+enum class ScalarID : uint32_t;
+
+enum class ScalarActionType : uint32_t { eSet = 0, eAdd = 1, eSetMaximum = 2 };
+
+typedef mozilla::Variant<uint32_t, bool, nsString> ScalarVariant;
+
+struct ScalarAction {
+ uint32_t mId;
+ bool mDynamic;
+ ScalarActionType mActionType;
+ // We need to wrap mData in a Maybe otherwise the IPC system
+ // is unable to instantiate a ScalarAction.
+ Maybe<ScalarVariant> mData;
+ // The process type this scalar should be recorded for.
+ // The IPC system will determine the process this action was coming from
+ // later.
+ mozilla::Telemetry::ProcessID mProcessType;
+};
+
+struct KeyedScalarAction {
+ uint32_t mId;
+ bool mDynamic;
+ ScalarActionType mActionType;
+ nsCString mKey;
+ // We need to wrap mData in a Maybe otherwise the IPC system
+ // is unable to instantiate a ScalarAction.
+ Maybe<ScalarVariant> mData;
+ // The process type this scalar should be recorded for.
+ // The IPC system will determine the process this action was coming from
+ // later.
+ mozilla::Telemetry::ProcessID mProcessType;
+};
+
+// Dynamic scalars support.
+struct DynamicScalarDefinition {
+ uint32_t type;
+ uint32_t dataset;
+ bool expired;
+ bool keyed;
+ bool builtin;
+ nsCString name;
+
+ bool operator==(const DynamicScalarDefinition& rhs) const {
+ return type == rhs.type && dataset == rhs.dataset &&
+ expired == rhs.expired && keyed == rhs.keyed &&
+ builtin == rhs.builtin && name.Equals(rhs.name);
+ }
+};
+
+struct EventExtraEntry {
+ nsCString key;
+ nsCString value;
+};
+
+struct ChildEventData {
+ mozilla::TimeStamp timestamp;
+ nsCString category;
+ nsCString method;
+ nsCString object;
+ mozilla::Maybe<nsCString> value;
+ CopyableTArray<EventExtraEntry> extra;
+};
+
+struct DiscardedData {
+ uint32_t mDiscardedHistogramAccumulations;
+ uint32_t mDiscardedKeyedHistogramAccumulations;
+ uint32_t mDiscardedScalarActions;
+ uint32_t mDiscardedKeyedScalarActions;
+ uint32_t mDiscardedChildEvents;
+};
+
+} // namespace Telemetry
+} // namespace mozilla
+
+namespace IPC {
+
+template <>
+struct ParamTraits<mozilla::Telemetry::HistogramAccumulation> {
+ typedef mozilla::Telemetry::HistogramAccumulation paramType;
+
+ static void Write(Message* aMsg, const paramType& aParam) {
+ aMsg->WriteUInt32(aParam.mId);
+ WriteParam(aMsg, aParam.mSample);
+ }
+
+ static bool Read(const Message* aMsg, PickleIterator* aIter,
+ paramType* aResult) {
+ if (!aMsg->ReadUInt32(aIter,
+ reinterpret_cast<uint32_t*>(&(aResult->mId))) ||
+ !ReadParam(aMsg, aIter, &(aResult->mSample))) {
+ return false;
+ }
+
+ return true;
+ }
+};
+
+template <>
+struct ParamTraits<mozilla::Telemetry::KeyedHistogramAccumulation> {
+ typedef mozilla::Telemetry::KeyedHistogramAccumulation paramType;
+
+ static void Write(Message* aMsg, const paramType& aParam) {
+ aMsg->WriteUInt32(aParam.mId);
+ WriteParam(aMsg, aParam.mSample);
+ WriteParam(aMsg, aParam.mKey);
+ }
+
+ static bool Read(const Message* aMsg, PickleIterator* aIter,
+ paramType* aResult) {
+ if (!aMsg->ReadUInt32(aIter,
+ reinterpret_cast<uint32_t*>(&(aResult->mId))) ||
+ !ReadParam(aMsg, aIter, &(aResult->mSample)) ||
+ !ReadParam(aMsg, aIter, &(aResult->mKey))) {
+ return false;
+ }
+
+ return true;
+ }
+};
+
+/**
+ * IPC scalar data message serialization and de-serialization.
+ */
+template <>
+struct ParamTraits<mozilla::Telemetry::ScalarAction> {
+ typedef mozilla::Telemetry::ScalarAction paramType;
+
+ static void Write(Message* aMsg, const paramType& aParam) {
+ // Write the message type
+ aMsg->WriteUInt32(aParam.mId);
+ WriteParam(aMsg, aParam.mDynamic);
+ WriteParam(aMsg, static_cast<uint32_t>(aParam.mActionType));
+
+ if (aParam.mData.isNothing()) {
+ MOZ_CRASH("There is no data in the ScalarAction.");
+ return;
+ }
+
+ if (aParam.mData->is<uint32_t>()) {
+ // That's a nsITelemetry::SCALAR_TYPE_COUNT.
+ WriteParam(aMsg, static_cast<uint32_t>(nsITelemetry::SCALAR_TYPE_COUNT));
+ WriteParam(aMsg, aParam.mData->as<uint32_t>());
+ } else if (aParam.mData->is<nsString>()) {
+ // That's a nsITelemetry::SCALAR_TYPE_STRING.
+ WriteParam(aMsg, static_cast<uint32_t>(nsITelemetry::SCALAR_TYPE_STRING));
+ WriteParam(aMsg, aParam.mData->as<nsString>());
+ } else if (aParam.mData->is<bool>()) {
+ // That's a nsITelemetry::SCALAR_TYPE_BOOLEAN.
+ WriteParam(aMsg,
+ static_cast<uint32_t>(nsITelemetry::SCALAR_TYPE_BOOLEAN));
+ WriteParam(aMsg, aParam.mData->as<bool>());
+ } else {
+ MOZ_CRASH("Unknown scalar type.");
+ }
+ }
+
+ static bool Read(const Message* aMsg, PickleIterator* aIter,
+ paramType* aResult) {
+ // Read the scalar ID and the scalar type.
+ uint32_t scalarType = 0;
+ if (!aMsg->ReadUInt32(aIter,
+ reinterpret_cast<uint32_t*>(&(aResult->mId))) ||
+ !ReadParam(aMsg, aIter,
+ reinterpret_cast<bool*>(&(aResult->mDynamic))) ||
+ !ReadParam(aMsg, aIter,
+ reinterpret_cast<uint32_t*>(&(aResult->mActionType))) ||
+ !ReadParam(aMsg, aIter, &scalarType)) {
+ return false;
+ }
+
+ // De-serialize the data based on the scalar type.
+ switch (scalarType) {
+ case nsITelemetry::SCALAR_TYPE_COUNT: {
+ uint32_t data = 0;
+ // De-serialize the data.
+ if (!ReadParam(aMsg, aIter, &data)) {
+ return false;
+ }
+ aResult->mData = mozilla::Some(mozilla::AsVariant(data));
+ break;
+ }
+ case nsITelemetry::SCALAR_TYPE_STRING: {
+ nsString data;
+ // De-serialize the data.
+ if (!ReadParam(aMsg, aIter, &data)) {
+ return false;
+ }
+ aResult->mData = mozilla::Some(mozilla::AsVariant(data));
+ break;
+ }
+ case nsITelemetry::SCALAR_TYPE_BOOLEAN: {
+ bool data = false;
+ // De-serialize the data.
+ if (!ReadParam(aMsg, aIter, &data)) {
+ return false;
+ }
+ aResult->mData = mozilla::Some(mozilla::AsVariant(data));
+ break;
+ }
+ default:
+ MOZ_ASSERT(false, "Unknown scalar type.");
+ return false;
+ }
+
+ return true;
+ }
+};
+
+/**
+ * IPC keyed scalar data message serialization and de-serialization.
+ */
+template <>
+struct ParamTraits<mozilla::Telemetry::KeyedScalarAction> {
+ typedef mozilla::Telemetry::KeyedScalarAction paramType;
+
+ static void Write(Message* aMsg, const paramType& aParam) {
+ // Write the message type
+ aMsg->WriteUInt32(static_cast<uint32_t>(aParam.mId));
+ WriteParam(aMsg, aParam.mDynamic);
+ WriteParam(aMsg, static_cast<uint32_t>(aParam.mActionType));
+ WriteParam(aMsg, aParam.mKey);
+
+ if (aParam.mData.isNothing()) {
+ MOZ_CRASH("There is no data in the KeyedScalarAction.");
+ return;
+ }
+
+ if (aParam.mData->is<uint32_t>()) {
+ // That's a nsITelemetry::SCALAR_TYPE_COUNT.
+ WriteParam(aMsg, static_cast<uint32_t>(nsITelemetry::SCALAR_TYPE_COUNT));
+ WriteParam(aMsg, aParam.mData->as<uint32_t>());
+ } else if (aParam.mData->is<nsString>()) {
+ // That's a nsITelemetry::SCALAR_TYPE_STRING.
+ // Keyed string scalars are not supported.
+ MOZ_ASSERT(false,
+ "Keyed String Scalar unable to be write from child process. "
+ "Not supported.");
+ } else if (aParam.mData->is<bool>()) {
+ // That's a nsITelemetry::SCALAR_TYPE_BOOLEAN.
+ WriteParam(aMsg,
+ static_cast<uint32_t>(nsITelemetry::SCALAR_TYPE_BOOLEAN));
+ WriteParam(aMsg, aParam.mData->as<bool>());
+ } else {
+ MOZ_CRASH("Unknown keyed scalar type.");
+ }
+ }
+
+ static bool Read(const Message* aMsg, PickleIterator* aIter,
+ paramType* aResult) {
+ // Read the scalar ID and the scalar type.
+ uint32_t scalarType = 0;
+ if (!aMsg->ReadUInt32(aIter,
+ reinterpret_cast<uint32_t*>(&(aResult->mId))) ||
+ !ReadParam(aMsg, aIter,
+ reinterpret_cast<bool*>(&(aResult->mDynamic))) ||
+ !ReadParam(aMsg, aIter,
+ reinterpret_cast<uint32_t*>(&(aResult->mActionType))) ||
+ !ReadParam(aMsg, aIter, &(aResult->mKey)) ||
+ !ReadParam(aMsg, aIter, &scalarType)) {
+ return false;
+ }
+
+ // De-serialize the data based on the scalar type.
+ switch (scalarType) {
+ case nsITelemetry::SCALAR_TYPE_COUNT: {
+ uint32_t data = 0;
+ // De-serialize the data.
+ if (!ReadParam(aMsg, aIter, &data)) {
+ return false;
+ }
+ aResult->mData = mozilla::Some(mozilla::AsVariant(data));
+ break;
+ }
+ case nsITelemetry::SCALAR_TYPE_STRING: {
+ // Keyed string scalars are not supported.
+ MOZ_ASSERT(false,
+ "Keyed String Scalar unable to be read from child process. "
+ "Not supported.");
+ return false;
+ }
+ case nsITelemetry::SCALAR_TYPE_BOOLEAN: {
+ bool data = false;
+ // De-serialize the data.
+ if (!ReadParam(aMsg, aIter, &data)) {
+ return false;
+ }
+ aResult->mData = mozilla::Some(mozilla::AsVariant(data));
+ break;
+ }
+ default:
+ MOZ_ASSERT(false, "Unknown keyed scalar type.");
+ return false;
+ }
+
+ return true;
+ }
+};
+
+template <>
+struct ParamTraits<mozilla::Telemetry::DynamicScalarDefinition> {
+ typedef mozilla::Telemetry::DynamicScalarDefinition paramType;
+
+ static void Write(Message* aMsg, const paramType& aParam) {
+ nsCString name;
+ WriteParam(aMsg, aParam.type);
+ WriteParam(aMsg, aParam.dataset);
+ WriteParam(aMsg, aParam.expired);
+ WriteParam(aMsg, aParam.keyed);
+ WriteParam(aMsg, aParam.builtin);
+ WriteParam(aMsg, aParam.name);
+ }
+
+ static bool Read(const Message* aMsg, PickleIterator* aIter,
+ paramType* aResult) {
+ if (!ReadParam(aMsg, aIter,
+ reinterpret_cast<uint32_t*>(&(aResult->type))) ||
+ !ReadParam(aMsg, aIter,
+ reinterpret_cast<uint32_t*>(&(aResult->dataset))) ||
+ !ReadParam(aMsg, aIter, reinterpret_cast<bool*>(&(aResult->expired))) ||
+ !ReadParam(aMsg, aIter, reinterpret_cast<bool*>(&(aResult->keyed))) ||
+ !ReadParam(aMsg, aIter, reinterpret_cast<bool*>(&(aResult->builtin))) ||
+ !ReadParam(aMsg, aIter, &(aResult->name))) {
+ return false;
+ }
+ return true;
+ }
+};
+
+template <>
+struct ParamTraits<mozilla::Telemetry::ChildEventData> {
+ typedef mozilla::Telemetry::ChildEventData paramType;
+
+ static void Write(Message* aMsg, const paramType& aParam) {
+ WriteParam(aMsg, aParam.timestamp);
+ WriteParam(aMsg, aParam.category);
+ WriteParam(aMsg, aParam.method);
+ WriteParam(aMsg, aParam.object);
+ WriteParam(aMsg, aParam.value);
+ WriteParam(aMsg, aParam.extra);
+ }
+
+ static bool Read(const Message* aMsg, PickleIterator* aIter,
+ paramType* aResult) {
+ if (!ReadParam(aMsg, aIter, &(aResult->timestamp)) ||
+ !ReadParam(aMsg, aIter, &(aResult->category)) ||
+ !ReadParam(aMsg, aIter, &(aResult->method)) ||
+ !ReadParam(aMsg, aIter, &(aResult->object)) ||
+ !ReadParam(aMsg, aIter, &(aResult->value)) ||
+ !ReadParam(aMsg, aIter, &(aResult->extra))) {
+ return false;
+ }
+
+ return true;
+ }
+};
+
+template <>
+struct ParamTraits<mozilla::Telemetry::EventExtraEntry> {
+ typedef mozilla::Telemetry::EventExtraEntry paramType;
+
+ static void Write(Message* aMsg, const paramType& aParam) {
+ WriteParam(aMsg, aParam.key);
+ WriteParam(aMsg, aParam.value);
+ }
+
+ static bool Read(const Message* aMsg, PickleIterator* aIter,
+ paramType* aResult) {
+ if (!ReadParam(aMsg, aIter, &(aResult->key)) ||
+ !ReadParam(aMsg, aIter, &(aResult->value))) {
+ return false;
+ }
+
+ return true;
+ }
+};
+
+template <>
+struct ParamTraits<mozilla::Telemetry::DiscardedData>
+ : public PlainOldDataSerializer<mozilla::Telemetry::DiscardedData> {};
+
+} // namespace IPC
+
+#endif // Telemetry_Comms_h__
diff --git a/toolkit/components/telemetry/core/ipc/TelemetryIPC.cpp b/toolkit/components/telemetry/core/ipc/TelemetryIPC.cpp
new file mode 100644
index 0000000000..daeaeece65
--- /dev/null
+++ b/toolkit/components/telemetry/core/ipc/TelemetryIPC.cpp
@@ -0,0 +1,59 @@
+/* -*- 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 "TelemetryIPC.h"
+#include "../TelemetryScalar.h"
+#include "../TelemetryHistogram.h"
+#include "../TelemetryEvent.h"
+
+namespace mozilla {
+
+void TelemetryIPC::AccumulateChildHistograms(
+ Telemetry::ProcessID aProcessType,
+ const nsTArray<Telemetry::HistogramAccumulation>& aAccumulations) {
+ TelemetryHistogram::AccumulateChild(aProcessType, aAccumulations);
+}
+
+void TelemetryIPC::AccumulateChildKeyedHistograms(
+ Telemetry::ProcessID aProcessType,
+ const nsTArray<Telemetry::KeyedHistogramAccumulation>& aAccumulations) {
+ TelemetryHistogram::AccumulateChildKeyed(aProcessType, aAccumulations);
+}
+
+void TelemetryIPC::UpdateChildScalars(
+ Telemetry::ProcessID aProcessType,
+ const nsTArray<Telemetry::ScalarAction>& aScalarActions) {
+ TelemetryScalar::UpdateChildData(aProcessType, aScalarActions);
+}
+
+void TelemetryIPC::UpdateChildKeyedScalars(
+ Telemetry::ProcessID aProcessType,
+ const nsTArray<Telemetry::KeyedScalarAction>& aScalarActions) {
+ TelemetryScalar::UpdateChildKeyedData(aProcessType, aScalarActions);
+}
+
+void TelemetryIPC::GetDynamicScalarDefinitions(
+ nsTArray<mozilla::Telemetry::DynamicScalarDefinition>& aDefs) {
+ TelemetryScalar::GetDynamicScalarDefinitions(aDefs);
+}
+
+void TelemetryIPC::AddDynamicScalarDefinitions(
+ const nsTArray<mozilla::Telemetry::DynamicScalarDefinition>& aDefs) {
+ TelemetryScalar::AddDynamicScalarDefinitions(aDefs);
+}
+
+void TelemetryIPC::RecordChildEvents(
+ Telemetry::ProcessID aProcessType,
+ const nsTArray<Telemetry::ChildEventData>& aEvents) {
+ TelemetryEvent::RecordChildEvents(aProcessType, aEvents);
+}
+
+void TelemetryIPC::RecordDiscardedData(
+ Telemetry::ProcessID aProcessType,
+ const Telemetry::DiscardedData& aDiscardedData) {
+ TelemetryScalar::RecordDiscardedData(aProcessType, aDiscardedData);
+}
+} // namespace mozilla
diff --git a/toolkit/components/telemetry/core/ipc/TelemetryIPC.h b/toolkit/components/telemetry/core/ipc/TelemetryIPC.h
new file mode 100644
index 0000000000..ef3bcddc49
--- /dev/null
+++ b/toolkit/components/telemetry/core/ipc/TelemetryIPC.h
@@ -0,0 +1,116 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryIPC_h__
+#define TelemetryIPC_h__
+
+#include <stdint.h>
+#include "mozilla/TelemetryProcessEnums.h"
+#include "nsTArray.h"
+#include "nsXULAppAPI.h"
+
+// This module provides the interface to accumulate Telemetry from child
+// processes. Top-level actors for different child processes types
+// (ContentParent, GPUChild) will call this for messages from their respective
+// processes.
+
+namespace mozilla {
+
+namespace Telemetry {
+
+struct HistogramAccumulation;
+struct KeyedHistogramAccumulation;
+struct ScalarAction;
+struct KeyedScalarAction;
+struct DynamicScalarDefinition;
+struct ChildEventData;
+struct DiscardedData;
+
+} // namespace Telemetry
+
+namespace TelemetryIPC {
+
+/**
+ * Accumulate child process data into histograms for the given process type.
+ *
+ * @param aProcessType - the process type to accumulate the histograms for
+ * @param aAccumulations - accumulation actions to perform
+ */
+void AccumulateChildHistograms(
+ Telemetry::ProcessID aProcessType,
+ const nsTArray<Telemetry::HistogramAccumulation>& aAccumulations);
+
+/**
+ * Accumulate child process data into keyed histograms for the given process
+ * type.
+ *
+ * @param aProcessType - the process type to accumulate the keyed histograms for
+ * @param aAccumulations - accumulation actions to perform
+ */
+void AccumulateChildKeyedHistograms(
+ Telemetry::ProcessID aProcessType,
+ const nsTArray<Telemetry::KeyedHistogramAccumulation>& aAccumulations);
+
+/**
+ * Update scalars for the given process type with the data coming from child
+ * process.
+ *
+ * @param aProcessType - the process type to process the scalar actions for
+ * @param aScalarActions - actions to update the scalar data
+ */
+void UpdateChildScalars(
+ Telemetry::ProcessID aProcessType,
+ const nsTArray<Telemetry::ScalarAction>& aScalarActions);
+
+/**
+ * Update keyed scalars for the given process type with the data coming from
+ * child process.
+ *
+ * @param aProcessType - the process type to process the keyed scalar actions
+ * for
+ * @param aScalarActions - actions to update the keyed scalar data
+ */
+void UpdateChildKeyedScalars(
+ Telemetry::ProcessID aProcessType,
+ const nsTArray<Telemetry::KeyedScalarAction>& aScalarActions);
+
+/**
+ * Record events for the given process type with the data coming from child
+ * process.
+ *
+ * @param aProcessType - the process type to record the events for
+ * @param aEvents - events to record
+ */
+void RecordChildEvents(Telemetry::ProcessID aProcessType,
+ const nsTArray<Telemetry::ChildEventData>& aEvents);
+
+/**
+ * Record the counts of data the child process had to discard
+ *
+ * @param aProcessType - the process reporting the discarded data
+ * @param aDiscardedData - stats about the discarded data
+ */
+void RecordDiscardedData(Telemetry::ProcessID aProcessType,
+ const Telemetry::DiscardedData& aDiscardedData);
+
+/**
+ * Get the dynamic scalar definitions from the parent process.
+ * @param aDefs - The array that will contain the scalar definitions.
+ */
+void GetDynamicScalarDefinitions(
+ nsTArray<mozilla::Telemetry::DynamicScalarDefinition>& aDefs);
+
+/**
+ * Add the dynamic scalar definitions coming from the parent process
+ * to the current child process.
+ * @param aDefs - The array that contains the scalar definitions.
+ */
+void AddDynamicScalarDefinitions(
+ const nsTArray<mozilla::Telemetry::DynamicScalarDefinition>& aDefs);
+
+} // namespace TelemetryIPC
+} // namespace mozilla
+
+#endif // TelemetryIPC_h__
diff --git a/toolkit/components/telemetry/core/ipc/TelemetryIPCAccumulator.cpp b/toolkit/components/telemetry/core/ipc/TelemetryIPCAccumulator.cpp
new file mode 100644
index 0000000000..bb9a112fd1
--- /dev/null
+++ b/toolkit/components/telemetry/core/ipc/TelemetryIPCAccumulator.cpp
@@ -0,0 +1,345 @@
+/* -*- 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 "TelemetryIPCAccumulator.h"
+
+#include "core/TelemetryHistogram.h"
+#include "core/TelemetryScalar.h"
+#include "mozilla/dom/ContentChild.h"
+#include "mozilla/gfx/GPUParent.h"
+#include "mozilla/gfx/GPUProcessManager.h"
+#include "mozilla/net/SocketProcessChild.h"
+#include "mozilla/SchedulerGroup.h"
+#include "mozilla/StaticMutex.h"
+#include "mozilla/StaticPrefs_toolkit.h"
+#include "mozilla/StaticPtr.h"
+#include "mozilla/Unused.h"
+#include "nsComponentManagerUtils.h"
+#include "nsITimer.h"
+#include "nsThreadUtils.h"
+
+using mozilla::StaticAutoPtr;
+using mozilla::StaticMutex;
+using mozilla::StaticMutexAutoLock;
+using mozilla::TaskCategory;
+using mozilla::Telemetry::ChildEventData;
+using mozilla::Telemetry::DiscardedData;
+using mozilla::Telemetry::HistogramAccumulation;
+using mozilla::Telemetry::KeyedHistogramAccumulation;
+using mozilla::Telemetry::KeyedScalarAction;
+using mozilla::Telemetry::ScalarAction;
+using mozilla::Telemetry::ScalarActionType;
+using mozilla::Telemetry::ScalarVariant;
+
+namespace TelemetryIPCAccumulator = mozilla::TelemetryIPCAccumulator;
+
+// To stop growing unbounded in memory while waiting for
+// StaticPrefs::toolkit_telemetry_ipcBatchTimeout() milliseconds to drain the
+// probe accumulation arrays, we request an immediate flush if the arrays
+// manage to reach certain high water mark of elements.
+const size_t kHistogramAccumulationsArrayHighWaterMark = 5 * 1024;
+const size_t kScalarActionsArrayHighWaterMark = 10000;
+// With the current limits, events cost us about 1100 bytes each.
+// This limits memory use to about 10MB.
+const size_t kEventsArrayHighWaterMark = 10000;
+// If we are starved we can overshoot the watermark.
+// This is the multiplier over which we will discard data.
+const size_t kWaterMarkDiscardFactor = 5;
+
+// Counts of how many pieces of data we have discarded.
+DiscardedData gDiscardedData = {0};
+
+// This timer is used for batching and sending child process accumulations to
+// the parent.
+nsITimer* gIPCTimer = nullptr;
+mozilla::Atomic<bool, mozilla::Relaxed> gIPCTimerArmed(false);
+mozilla::Atomic<bool, mozilla::Relaxed> gIPCTimerArming(false);
+
+// This batches child process accumulations that should be sent to the parent.
+StaticAutoPtr<nsTArray<HistogramAccumulation>> gHistogramAccumulations;
+StaticAutoPtr<nsTArray<KeyedHistogramAccumulation>>
+ gKeyedHistogramAccumulations;
+StaticAutoPtr<nsTArray<ScalarAction>> gChildScalarsActions;
+StaticAutoPtr<nsTArray<KeyedScalarAction>> gChildKeyedScalarsActions;
+StaticAutoPtr<nsTArray<ChildEventData>> gChildEvents;
+
+// This is a StaticMutex rather than a plain Mutex so that (1)
+// it gets initialised in a thread-safe manner the first time
+// it is used, and (2) because it is never de-initialised, and
+// a normal Mutex would show up as a leak in BloatView. StaticMutex
+// also has the "OffTheBooks" property, so it won't show as a leak
+// in BloatView.
+static StaticMutex gTelemetryIPCAccumulatorMutex;
+
+namespace {
+
+void DoArmIPCTimerMainThread(const StaticMutexAutoLock& lock) {
+ MOZ_ASSERT(NS_IsMainThread());
+ gIPCTimerArming = false;
+ if (gIPCTimerArmed) {
+ return;
+ }
+ if (!gIPCTimer) {
+ gIPCTimer = NS_NewTimer().take();
+ }
+ if (gIPCTimer) {
+ gIPCTimer->InitWithNamedFuncCallback(
+ TelemetryIPCAccumulator::IPCTimerFired, nullptr,
+ mozilla::StaticPrefs::toolkit_telemetry_ipcBatchTimeout(),
+ nsITimer::TYPE_ONE_SHOT_LOW_PRIORITY,
+ "TelemetryIPCAccumulator::IPCTimerFired");
+ gIPCTimerArmed = true;
+ }
+}
+
+void ArmIPCTimer(const StaticMutexAutoLock& lock) {
+ if (gIPCTimerArmed || gIPCTimerArming) {
+ return;
+ }
+ gIPCTimerArming = true;
+ if (NS_IsMainThread()) {
+ DoArmIPCTimerMainThread(lock);
+ } else {
+ TelemetryIPCAccumulator::DispatchToMainThread(NS_NewRunnableFunction(
+ "TelemetryIPCAccumulator::ArmIPCTimer", []() -> void {
+ StaticMutexAutoLock locker(gTelemetryIPCAccumulatorMutex);
+ DoArmIPCTimerMainThread(locker);
+ }));
+ }
+}
+
+void DispatchIPCTimerFired() {
+ TelemetryIPCAccumulator::DispatchToMainThread(NS_NewRunnableFunction(
+ "TelemetryIPCAccumulator::IPCTimerFired", []() -> void {
+ TelemetryIPCAccumulator::IPCTimerFired(nullptr, nullptr);
+ }));
+}
+
+} // anonymous namespace
+
+////////////////////////////////////////////////////////////////////////
+////////////////////////////////////////////////////////////////////////
+//
+// EXTERNALLY VISIBLE FUNCTIONS in namespace TelemetryIPCAccumulator::
+
+void TelemetryIPCAccumulator::AccumulateChildHistogram(
+ mozilla::Telemetry::HistogramID aId, uint32_t aSample) {
+ StaticMutexAutoLock locker(gTelemetryIPCAccumulatorMutex);
+ if (!gHistogramAccumulations) {
+ gHistogramAccumulations = new nsTArray<HistogramAccumulation>();
+ }
+ if (gHistogramAccumulations->Length() >=
+ kWaterMarkDiscardFactor * kHistogramAccumulationsArrayHighWaterMark) {
+ gDiscardedData.mDiscardedHistogramAccumulations++;
+ return;
+ }
+ if (gHistogramAccumulations->Length() ==
+ kHistogramAccumulationsArrayHighWaterMark) {
+ DispatchIPCTimerFired();
+ }
+ gHistogramAccumulations->AppendElement(HistogramAccumulation{aId, aSample});
+ ArmIPCTimer(locker);
+}
+
+void TelemetryIPCAccumulator::AccumulateChildKeyedHistogram(
+ mozilla::Telemetry::HistogramID aId, const nsCString& aKey,
+ uint32_t aSample) {
+ StaticMutexAutoLock locker(gTelemetryIPCAccumulatorMutex);
+ if (!gKeyedHistogramAccumulations) {
+ gKeyedHistogramAccumulations = new nsTArray<KeyedHistogramAccumulation>();
+ }
+ if (gKeyedHistogramAccumulations->Length() >=
+ kWaterMarkDiscardFactor * kHistogramAccumulationsArrayHighWaterMark) {
+ gDiscardedData.mDiscardedKeyedHistogramAccumulations++;
+ return;
+ }
+ if (gKeyedHistogramAccumulations->Length() ==
+ kHistogramAccumulationsArrayHighWaterMark) {
+ DispatchIPCTimerFired();
+ }
+ gKeyedHistogramAccumulations->AppendElement(
+ KeyedHistogramAccumulation{aId, aSample, aKey});
+ ArmIPCTimer(locker);
+}
+
+void TelemetryIPCAccumulator::RecordChildScalarAction(
+ uint32_t aId, bool aDynamic, ScalarActionType aAction,
+ const ScalarVariant& aValue) {
+ StaticMutexAutoLock locker(gTelemetryIPCAccumulatorMutex);
+ // Make sure to have the storage.
+ if (!gChildScalarsActions) {
+ gChildScalarsActions = new nsTArray<ScalarAction>();
+ }
+ if (gChildScalarsActions->Length() >=
+ kWaterMarkDiscardFactor * kScalarActionsArrayHighWaterMark) {
+ gDiscardedData.mDiscardedScalarActions++;
+ return;
+ }
+ if (gChildScalarsActions->Length() == kScalarActionsArrayHighWaterMark) {
+ DispatchIPCTimerFired();
+ }
+ // Store the action. The ProcessID will be determined by the receiver.
+ gChildScalarsActions->AppendElement(ScalarAction{
+ aId, aDynamic, aAction, Some(aValue), Telemetry::ProcessID::Count});
+ ArmIPCTimer(locker);
+}
+
+void TelemetryIPCAccumulator::RecordChildKeyedScalarAction(
+ uint32_t aId, bool aDynamic, const nsAString& aKey,
+ ScalarActionType aAction, const ScalarVariant& aValue) {
+ StaticMutexAutoLock locker(gTelemetryIPCAccumulatorMutex);
+ // Make sure to have the storage.
+ if (!gChildKeyedScalarsActions) {
+ gChildKeyedScalarsActions = new nsTArray<KeyedScalarAction>();
+ }
+ if (gChildKeyedScalarsActions->Length() >=
+ kWaterMarkDiscardFactor * kScalarActionsArrayHighWaterMark) {
+ gDiscardedData.mDiscardedKeyedScalarActions++;
+ return;
+ }
+ if (gChildKeyedScalarsActions->Length() == kScalarActionsArrayHighWaterMark) {
+ DispatchIPCTimerFired();
+ }
+ // Store the action. The ProcessID will be determined by the receiver.
+ gChildKeyedScalarsActions->AppendElement(
+ KeyedScalarAction{aId, aDynamic, aAction, NS_ConvertUTF16toUTF8(aKey),
+ Some(aValue), Telemetry::ProcessID::Count});
+ ArmIPCTimer(locker);
+}
+
+void TelemetryIPCAccumulator::RecordChildEvent(
+ const mozilla::TimeStamp& timestamp, const nsACString& category,
+ const nsACString& method, const nsACString& object,
+ const mozilla::Maybe<nsCString>& value,
+ const nsTArray<mozilla::Telemetry::EventExtraEntry>& extra) {
+ StaticMutexAutoLock locker(gTelemetryIPCAccumulatorMutex);
+
+ if (!gChildEvents) {
+ gChildEvents = new nsTArray<ChildEventData>();
+ }
+
+ if (gChildEvents->Length() >=
+ kWaterMarkDiscardFactor * kEventsArrayHighWaterMark) {
+ gDiscardedData.mDiscardedChildEvents++;
+ return;
+ }
+
+ if (gChildEvents->Length() == kEventsArrayHighWaterMark) {
+ DispatchIPCTimerFired();
+ }
+
+ // Store the event.
+ gChildEvents->AppendElement(
+ ChildEventData{timestamp, nsCString(category), nsCString(method),
+ nsCString(object), value, extra.Clone()});
+ ArmIPCTimer(locker);
+}
+
+// This method takes the lock only to double-buffer the batched telemetry.
+// It releases the lock before calling out to IPC code which can (and does)
+// Accumulate (which would deadlock)
+template <class TActor>
+static void SendAccumulatedData(TActor* ipcActor) {
+ // Get the accumulated data and free the storage buffers.
+ nsTArray<HistogramAccumulation> histogramsToSend;
+ nsTArray<KeyedHistogramAccumulation> keyedHistogramsToSend;
+ nsTArray<ScalarAction> scalarsToSend;
+ nsTArray<KeyedScalarAction> keyedScalarsToSend;
+ nsTArray<ChildEventData> eventsToSend;
+ DiscardedData discardedData;
+
+ {
+ StaticMutexAutoLock locker(gTelemetryIPCAccumulatorMutex);
+ if (gHistogramAccumulations) {
+ histogramsToSend = std::move(*gHistogramAccumulations);
+ }
+ if (gKeyedHistogramAccumulations) {
+ keyedHistogramsToSend = std::move(*gKeyedHistogramAccumulations);
+ }
+ if (gChildScalarsActions) {
+ scalarsToSend = std::move(*gChildScalarsActions);
+ }
+ if (gChildKeyedScalarsActions) {
+ keyedScalarsToSend = std::move(*gChildKeyedScalarsActions);
+ }
+ if (gChildEvents) {
+ eventsToSend = std::move(*gChildEvents);
+ }
+ discardedData = gDiscardedData;
+ gDiscardedData = {0};
+ }
+
+ // Send the accumulated data to the parent process.
+ MOZ_ASSERT(ipcActor);
+ if (histogramsToSend.Length()) {
+ mozilla::Unused << NS_WARN_IF(
+ !ipcActor->SendAccumulateChildHistograms(histogramsToSend));
+ }
+ if (keyedHistogramsToSend.Length()) {
+ mozilla::Unused << NS_WARN_IF(
+ !ipcActor->SendAccumulateChildKeyedHistograms(keyedHistogramsToSend));
+ }
+ if (scalarsToSend.Length()) {
+ mozilla::Unused << NS_WARN_IF(
+ !ipcActor->SendUpdateChildScalars(scalarsToSend));
+ }
+ if (keyedScalarsToSend.Length()) {
+ mozilla::Unused << NS_WARN_IF(
+ !ipcActor->SendUpdateChildKeyedScalars(keyedScalarsToSend));
+ }
+ if (eventsToSend.Length()) {
+ mozilla::Unused << NS_WARN_IF(
+ !ipcActor->SendRecordChildEvents(eventsToSend));
+ }
+ mozilla::Unused << NS_WARN_IF(
+ !ipcActor->SendRecordDiscardedData(discardedData));
+}
+
+// To ensure we don't loop IPCTimerFired->AccumulateChild->arm timer, we don't
+// unset gIPCTimerArmed until the IPC completes
+//
+// This function must be called on the main thread, otherwise IPC will fail.
+void TelemetryIPCAccumulator::IPCTimerFired(nsITimer* aTimer, void* aClosure) {
+ MOZ_ASSERT(NS_IsMainThread());
+
+ // Send accumulated data to the correct parent process.
+ switch (XRE_GetProcessType()) {
+ case GeckoProcessType_Content:
+ SendAccumulatedData(mozilla::dom::ContentChild::GetSingleton());
+ break;
+ case GeckoProcessType_GPU:
+ SendAccumulatedData(mozilla::gfx::GPUParent::GetSingleton());
+ break;
+ case GeckoProcessType_Socket:
+ SendAccumulatedData(mozilla::net::SocketProcessChild::GetSingleton());
+ break;
+ default:
+ MOZ_ASSERT_UNREACHABLE("Unsupported process type");
+ break;
+ }
+
+ gIPCTimerArmed = false;
+}
+
+void TelemetryIPCAccumulator::DeInitializeGlobalState() {
+ MOZ_ASSERT(NS_IsMainThread());
+
+ StaticMutexAutoLock locker(gTelemetryIPCAccumulatorMutex);
+ if (gIPCTimer) {
+ NS_RELEASE(gIPCTimer);
+ }
+
+ gHistogramAccumulations = nullptr;
+ gKeyedHistogramAccumulations = nullptr;
+ gChildScalarsActions = nullptr;
+ gChildKeyedScalarsActions = nullptr;
+ gChildEvents = nullptr;
+}
+
+void TelemetryIPCAccumulator::DispatchToMainThread(
+ already_AddRefed<nsIRunnable>&& aEvent) {
+ SchedulerGroup::Dispatch(TaskCategory::Other, std::move(aEvent));
+}
diff --git a/toolkit/components/telemetry/core/ipc/TelemetryIPCAccumulator.h b/toolkit/components/telemetry/core/ipc/TelemetryIPCAccumulator.h
new file mode 100644
index 0000000000..67c5019a25
--- /dev/null
+++ b/toolkit/components/telemetry/core/ipc/TelemetryIPCAccumulator.h
@@ -0,0 +1,54 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
+/* 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/. */
+
+#ifndef TelemetryIPCAccumulator_h__
+#define TelemetryIPCAccumulator_h__
+
+#include "mozilla/AlreadyAddRefed.h"
+#include "mozilla/Maybe.h"
+#include "nsStringFwd.h"
+#include "TelemetryComms.h"
+
+class nsIRunnable;
+class nsITimer;
+
+namespace mozilla {
+
+class TimeStamp;
+
+namespace TelemetryIPCAccumulator {
+
+// Histogram accumulation functions.
+void AccumulateChildHistogram(mozilla::Telemetry::HistogramID aId,
+ uint32_t aSample);
+void AccumulateChildKeyedHistogram(mozilla::Telemetry::HistogramID aId,
+ const nsCString& aKey, uint32_t aSample);
+
+// Scalar accumulation functions.
+void RecordChildScalarAction(uint32_t aId, bool aDynamic,
+ mozilla::Telemetry::ScalarActionType aAction,
+ const mozilla::Telemetry::ScalarVariant& aValue);
+
+void RecordChildKeyedScalarAction(
+ uint32_t aId, bool aDynamic, const nsAString& aKey,
+ mozilla::Telemetry::ScalarActionType aAction,
+ const mozilla::Telemetry::ScalarVariant& aValue);
+
+void RecordChildEvent(
+ const mozilla::TimeStamp& timestamp, const nsACString& category,
+ const nsACString& method, const nsACString& object,
+ const mozilla::Maybe<nsCString>& value,
+ const nsTArray<mozilla::Telemetry::EventExtraEntry>& extra);
+
+void IPCTimerFired(nsITimer* aTimer, void* aClosure);
+
+void DeInitializeGlobalState();
+
+void DispatchToMainThread(already_AddRefed<nsIRunnable>&& aEvent);
+
+} // namespace TelemetryIPCAccumulator
+} // namespace mozilla
+
+#endif // TelemetryIPCAccumulator_h__
diff --git a/toolkit/components/telemetry/core/nsITelemetry.idl b/toolkit/components/telemetry/core/nsITelemetry.idl
new file mode 100644
index 0000000000..4ae28c008f
--- /dev/null
+++ b/toolkit/components/telemetry/core/nsITelemetry.idl
@@ -0,0 +1,702 @@
+/* -*- Mode: C++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 8 -*- */
+/* 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 "nsISupports.idl"
+#include "nsIFile.idl"
+
+[scriptable,function, uuid(3d3b9075-5549-4244-9c08-b64fefa1dd60)]
+interface nsIFetchTelemetryDataCallback : nsISupports
+{
+ void complete();
+};
+
+[scriptable, uuid(273d2dd0-6c63-475a-b864-cb65160a1909)]
+interface nsITelemetry : nsISupports
+{
+ /**
+ * Histogram types:
+ * HISTOGRAM_EXPONENTIAL - buckets increase exponentially
+ * HISTOGRAM_LINEAR - buckets increase linearly
+ * HISTOGRAM_BOOLEAN - For storing 0/1 values
+ * HISTOGRAM_FLAG - For storing a single value; its count is always == 1.
+ * HISTOGRAM_COUNT - For storing counter values without bucketing.
+ * HISTOGRAM_CATEGORICAL - For storing enumerated values by label.
+ */
+ const unsigned long HISTOGRAM_EXPONENTIAL = 0;
+ const unsigned long HISTOGRAM_LINEAR = 1;
+ const unsigned long HISTOGRAM_BOOLEAN = 2;
+ const unsigned long HISTOGRAM_FLAG = 3;
+ const unsigned long HISTOGRAM_COUNT = 4;
+ const unsigned long HISTOGRAM_CATEGORICAL = 5;
+
+ /**
+ * Scalar types:
+ * SCALAR_TYPE_COUNT - for storing a numeric value
+ * SCALAR_TYPE_STRING - for storing a string value
+ * SCALAR_TYPE_BOOLEAN - for storing a boolean value
+ */
+ const unsigned long SCALAR_TYPE_COUNT = 0;
+ const unsigned long SCALAR_TYPE_STRING = 1;
+ const unsigned long SCALAR_TYPE_BOOLEAN = 2;
+
+ /**
+ * Dataset types:
+ * DATASET_ALL_CHANNELS - the basic dataset that is on-by-default on all channels
+ * DATASET_PRERELEASE_CHANNELS - the extended dataset that is opt-in on release,
+ * opt-out on pre-release channels.
+ */
+ const unsigned long DATASET_ALL_CHANNELS = 0;
+ const unsigned long DATASET_PRERELEASE_CHANNELS = 1;
+
+ /**
+ * Serializes the histogram labels for categorical hitograms.
+ * The returned structure looks like:
+ * { "histogram1": [ "histogram1_label1", "histogram1_label2", ...],
+ * "histogram2": [ "histogram2_label1", "histogram2_label2", ...]
+ * ...
+ * }
+ *
+ * Note that this function should only be used in tests and about:telemetry.
+ */
+ [implicit_jscontext]
+ jsval getCategoricalLabels();
+
+ /**
+ * Serializes the histograms from the given store to a JSON-style object.
+ * The returned structure looks like:
+ * { "process": { "name1": histogramData1, "name2": histogramData2 }, ... }
+ *
+ * Each histogram is represented in a packed format and has the following properties:
+ * bucket_count - Number of buckets of this histogram
+ * histogram_type - HISTOGRAM_EXPONENTIAL, HISTOGRAM_LINEAR, HISTOGRAM_BOOLEAN,
+ * HISTOGRAM_FLAG, HISTOGRAM_COUNT, or HISTOGRAM_CATEGORICAL
+ * sum - sum of the bucket contents
+ * range - A 2-item array of minimum and maximum bucket size
+ * values - Map from bucket to the bucket's count
+ *
+ * @param aStoreName The name of the store to snapshot.
+ * Defaults to "main".
+ * Custom stores are available when probes have them defined.
+ * See the `record_into_store` attribute on histograms.
+ * @see https://firefox-source-docs.mozilla.org/toolkit/components/telemetry/telemetry/collection/histograms.html#record-into-store
+ * @param aClearStore Whether to clear out the histograms in the named store after snapshotting.
+ * Defaults to false.
+ * @param aFilterTest If true, `TELEMETRY_TEST_` histograms will be filtered out.
+ Filtered histograms are still cleared if `aClearStore` is true.
+ * Defaults to false.
+ */
+ [implicit_jscontext]
+ jsval getSnapshotForHistograms([optional] in ACString aStoreName, [optional] in boolean aClearStore, [optional] in boolean aFilterTest);
+
+ /**
+ * Serializes the keyed histograms from the given store to a JSON-style object.
+ * The returned structure looks like:
+ * { "process": { "name1": { "key_1": histogramData1, "key_2": histogramData2 }, ...}, ... }
+ *
+ * @param aStoreName The name of the store to snapshot.
+ * Defaults to "main".
+ * Custom stores are available when probes have them defined.
+ * See the `record_into_store` attribute on histograms.
+ * @see https://firefox-source-docs.mozilla.org/toolkit/components/telemetry/telemetry/collection/histograms.html#record-into-store
+ * @param aClearStore Whether to clear out the keyed histograms in the named store after snapshotting.
+ * Defaults to false.
+ * @param aFilterTest If true, `TELEMETRY_TEST_` histograms will be filtered out.
+ Filtered histograms are still cleared if `aClearStore` is true.
+ * Defaults to false.
+ */
+ [implicit_jscontext]
+ jsval getSnapshotForKeyedHistograms([optional] in ACString aStoreName, [optional] in boolean aClearStore, [optional] in boolean aFilterTest);
+
+ /**
+ * Serializes the scalars from the given store to a JSON-style object.
+ * The returned structure looks like:
+ * { "process": { "category1.probe": 1,"category1.other_probe": false, ... }, ... }.
+ *
+ * @param aStoreName The name of the store to snapshot.
+ * Defaults to "main".
+ * Custom stores are available when probes have them defined.
+ * See the `record_into_store` attribute on scalars.
+ * @see https://firefox-source-docs.mozilla.org/toolkit/components/telemetry/telemetry/collection/scalars.html#optional-fields
+ * @param aClearStore Whether to clear out the scalars in the named store after snapshotting.
+ * Defaults to false.
+ * @param aFilterTest If true, `telemetry.test` scalars will be filtered out.
+ Filtered scalars are still cleared if `aClearStore` is true.
+ * Defaults to false.
+ */
+ [implicit_jscontext]
+ jsval getSnapshotForScalars([optional] in ACString aStoreName, [optional] in boolean aClearStore, [optional] in boolean aFilterTest);
+
+ /**
+ * Serializes the keyed scalars from the given store to a JSON-style object.
+ * The returned structure looks like:
+ * { "process": { "category1.probe": { "key_1": 2, "key_2": 1, ... }, ... }, ... }
+ *
+ * @param aStoreName The name of the store to snapshot.
+ * Defaults to "main".
+ * Custom stores are available when probes have them defined.
+ * See the `record_into_store` attribute on scalars.
+ * @see https://firefox-source-docs.mozilla.org/toolkit/components/telemetry/telemetry/collection/scalars.html#optional-fields
+ * @param aClearStore Whether to clear out the keyed scalars in the named store after snapshotting.
+ * Defaults to false.
+ * @param aFilterTest If true, `telemetry.test` scalars will be filtered out.
+ Filtered scalars are still cleared if `aClearStore` is true.
+ * Defaults to false.
+ */
+ [implicit_jscontext]
+ jsval getSnapshotForKeyedScalars([optional] in ACString aStoreName, [optional] in boolean aClearStore, [optional] in boolean aFilterTest);
+
+ /**
+ * The amount of time, in milliseconds, that the last session took
+ * to shutdown. Reads as 0 to indicate failure.
+ */
+ readonly attribute uint32_t lastShutdownDuration;
+
+ /**
+ * The number of failed profile lock attempts that have occurred prior to
+ * successfully locking the profile
+ */
+ readonly attribute uint32_t failedProfileLockCount;
+
+ /*
+ * An object containing information about slow SQL statements.
+ *
+ * {
+ * mainThread: { "sqlString1": [<hit count>, <total time>], "sqlString2": [...], ... },
+ * otherThreads: { "sqlString3": [<hit count>, <total time>], "sqlString4": [...], ... }
+ * }
+ *
+ * where:
+ * mainThread: Slow statements that executed on the main thread
+ * otherThreads: Slow statements that executed on a non-main thread
+ * sqlString - String of the offending statement (see note)
+ * hit count - The number of times this statement required longer than the threshold time to execute
+ * total time - The sum of all execution times above the threshold time for this statement
+ *
+ * Note that dynamic SQL strings and SQL strings executed against addon DBs could contain private information.
+ * This property represents such SQL as aggregate database-level stats and the sqlString contains the database
+ * filename instead.
+ */
+ [implicit_jscontext]
+ readonly attribute jsval slowSQL;
+
+ /*
+ * See slowSQL above.
+ *
+ * An object containing full strings of every slow SQL statement if toolkit.telemetry.debugSlowSql = true
+ * The returned SQL strings may contain private information and should not be reported to Telemetry.
+ */
+ [implicit_jscontext]
+ readonly attribute jsval debugSlowSQL;
+
+ /**
+ * A number representing the highest number of concurrent threads
+ * reached during this session.
+ */
+ readonly attribute uint32_t maximalNumberOfConcurrentThreads;
+
+ /**
+ * Flags for getUntrustedModuleLoadEvents.
+ */
+
+ /**
+ * This flag is set to retrieve all data including instances which have been
+ * retrieved before. If not set, only new instances since the last call
+ * will be returned.
+ * If this flag is set, KEEP_LOADEVENTS_NEW must not be set unless
+ * EXCLUDE_STACKINFO_FROM_LOADEVENTS is set.
+ * (See also MultiGetUntrustedModulesData::Serialize.)
+ */
+ const unsigned long INCLUDE_OLD_LOADEVENTS = 1 << 0;
+
+ /**
+ * This flag is set to keep the returned instances as if they were not
+ * retrieved, meaning those instances will be returned by a next method
+ * call without INCLUDE_OLD_LOADEVENTS. If not set, the returned instances
+ * can be re-retrieved only when INCLUDE_OLD_LOADEVENTS is specified.
+ * If this flag is set, INCLUDE_OLD_LOADEVENTS must not be set unless
+ * EXCLUDE_STACKINFO_FROM_LOADEVENTS is set.
+ * (See also MultiGetUntrustedModulesData::Serialize.)
+ */
+ const unsigned long KEEP_LOADEVENTS_NEW = 1 << 1;
+
+ /**
+ * This flag is set to include private fields.
+ * Do not specify this flag to retrieve data to be submitted.
+ */
+ const unsigned long INCLUDE_PRIVATE_FIELDS_IN_LOADEVENTS = 1 << 2;
+
+ /**
+ * This flag is set to exclude the "combinedStacks" field.
+ * Without this flag, the flags INCLUDE_OLD_LOADEVENTS and KEEP_LOADEVENTS_NEW
+ * cannot be set at the same time.
+ */
+ const unsigned long EXCLUDE_STACKINFO_FROM_LOADEVENTS = 1 << 3;
+
+ /*
+ * An array of untrusted module load events. Each element describes one or
+ * more modules that were loaded, contextual information at the time of the
+ * event (including stack trace), and flags describing the module's
+ * trustworthiness.
+ *
+ * @param aFlags Combination (bitwise OR) of the flags specified above.
+ * Defaults to 0.
+ */
+ [implicit_jscontext]
+ Promise getUntrustedModuleLoadEvents([optional] in unsigned long aFlags);
+
+ /*
+ * Asynchronously get an array of the modules loaded in the process.
+ *
+ * The data has the following structure:
+ *
+ * [
+ * {
+ * "name": <string>, // Name of the module file (e.g. xul.dll)
+ * "version": <string>, // Version of the module
+ * "debugName": <string>, // ID of the debug information file
+ * "debugID": <string>, // Name of the debug information file
+ * "certSubject": <string>, // Name of the organization that signed the binary (Optional, only defined when present)
+ * },
+ * ...
+ * ]
+ *
+ * @return A promise that resolves to an array of modules or rejects with
+ NS_ERROR_FAILURE on failure.
+ * @throws NS_ERROR_NOT_IMPLEMENTED if the Gecko profiler is not enabled.
+ */
+ [implicit_jscontext]
+ Promise getLoadedModules();
+
+ /*
+ * An object with two fields: memoryMap and stacks.
+ * * memoryMap is a list of loaded libraries.
+ * * stacks is a list of stacks. Each stack is a list of pairs of the form
+ * [moduleIndex, offset]. The moduleIndex is an index into the memoryMap and
+ * offset is an offset in the library at memoryMap[moduleIndex].
+ * This format is used to make it easier to send the stacks to the
+ * symbolication server.
+ */
+ [implicit_jscontext]
+ readonly attribute jsval lateWrites;
+
+ /**
+ * Create and return a histogram registered in TelemetryHistograms.h.
+ *
+ * @param id - unique identifier from TelemetryHistograms.h
+ * The returned object has the following functions:
+ * add(value) - Adds a sample of `value` to the histogram.
+ `value` may be a categorical histogram's label as a string,
+ a boolean histogram's value as a boolean,
+ or a number that fits inside a uint32_t.
+ * snapshot([optional] {store}) - Returns a snapshot of the histogram with the same data fields
+ as in getSnapshotForHistograms().
+ Defaults to the "main" store.
+ * clear([optional] {store}) - Zeros out the histogram's buckets and sum.
+ Defaults to the "main" store.
+ Note: This is intended to be only used in tests.
+ */
+ [implicit_jscontext]
+ jsval getHistogramById(in ACString id);
+
+ /**
+ * Create and return a histogram registered in TelemetryHistograms.h.
+ *
+ * @param id - unique identifier from TelemetryHistograms.h
+ * The returned object has the following functions:
+ * add(string key, [optional] value) - Adds a sample of `value` to the histogram for that key.
+ If no histogram for that key exists yet, it is created.
+ `value` may be a categorical histogram's label as a string,
+ a boolean histogram's value as a boolean,
+ or a number that fits inside a uint32_t.
+ * snapshot([optional] {store}) - Returns the snapshots of all the registered keys in the form
+ {key1: snapshot1, key2: snapshot2, ...} in the specified store.
+ * Defaults to the "main" store.
+ * keys([optional] {store}) - Returns an array with the string keys of the currently registered
+ histograms in the given store.
+ Defaults to "main".
+ * clear([optional] {store}) - Clears the registered histograms from this.
+ * Defaults to the "main" store.
+ * Note: This is intended to be only used in tests.
+ */
+ [implicit_jscontext]
+ jsval getKeyedHistogramById(in ACString id);
+
+ /**
+ * A flag indicating if Telemetry can record base data (FHR data). This is true if the
+ * FHR data reporting service or self-support are enabled.
+ *
+ * In the unlikely event that adding a new base probe is needed, please check the data
+ * collection wiki at https://wiki.mozilla.org/Firefox/Data_Collection and talk to the
+ * Telemetry team.
+ */
+ attribute boolean canRecordBase;
+
+ /**
+ * A flag indicating if Telemetry is allowed to record extended data. Returns false if
+ * the user hasn't opted into "extended Telemetry" on the Release channel, when the
+ * user has explicitly opted out of Telemetry on Nightly/Aurora/Beta or if manually
+ * set to false during tests.
+ *
+ * Set this to false in tests to disable gathering of extended telemetry statistics.
+ */
+ attribute boolean canRecordExtended;
+
+ /**
+ * A flag indicating whether Telemetry is recording release data, which is a
+ * smallish subset of our usage data that we're prepared to handle from our
+ * largish release population.
+ *
+ * This is true most of the time.
+ *
+ * This does not indicate whether Telemetry will send any data. That is
+ * governed by user preference and other mechanisms.
+ *
+ * You may use this to determine if it's okay to record your data.
+ */
+ readonly attribute boolean canRecordReleaseData;
+
+ /**
+ * A flag indicating whether Telemetry is recording prerelease data, which is
+ * a largish amount of usage data that we're prepared to handle from our
+ * smallish pre-release population.
+ *
+ * This is true on pre-release branches of Firefox.
+ *
+ * This does not indicate whether Telemetry will send any data. That is
+ * governed by user preference and other mechanisms.
+ *
+ * You may use this to determine if it's okay to record your data.
+ */
+ readonly attribute boolean canRecordPrereleaseData;
+
+ /**
+ * A flag indicating whether Telemetry can submit official results (for base or extended
+ * data). This is true on official, non-debug builds with built in support for Mozilla
+ * Telemetry reporting.
+ */
+ readonly attribute boolean isOfficialTelemetry;
+
+ /**
+ * Enable/disable recording for this histogram at runtime.
+ * Recording is enabled by default, unless listed at kRecordingInitiallyDisabledIDs[].
+ * Name must be a valid Histogram identifier, otherwise an assertion will be triggered.
+ *
+ * @param id - unique identifier from histograms.json
+ * @param enabled - whether or not to enable recording from now on.
+ */
+ void setHistogramRecordingEnabled(in ACString id, in boolean enabled);
+
+ /**
+ * Read data from the previous run. After the callback is called, the last
+ * shutdown time is available in lastShutdownDuration and any late
+ * writes in lateWrites.
+ */
+ void asyncFetchTelemetryData(in nsIFetchTelemetryDataCallback aCallback);
+
+ /**
+ * Get statistics of file IO reports, null, if not recorded.
+ *
+ * The statistics are returned as an object whose propoerties are the names
+ * of the files that have been accessed and whose corresponding values are
+ * arrays of size three, representing startup, normal, and shutdown stages.
+ * Each stage's entry is either null or an array with the layout
+ * [total_time, #creates, #reads, #writes, #fsyncs, #stats]
+ */
+ [implicit_jscontext]
+ readonly attribute jsval fileIOReports;
+
+ /**
+ * Return the number of milliseconds since process start using monotonic
+ * timestamps (unaffected by system clock changes). On Windows, this includes
+ * the period of time the device was suspended. On Linux and macOS, this does
+ * not include the period of time the device was suspneded.
+ */
+ double msSinceProcessStart();
+
+ /**
+ * Return the number of milliseconds since process start using monotonic
+ * timestamps (unaffected by system clock changes), including the periods of
+ * time the device was suspended.
+ * @throws NS_ERROR_NOT_AVAILABLE if unavailable.
+ */
+ double msSinceProcessStartIncludingSuspend();
+
+ /**
+ * Return the number of milliseconds since process start using monotonic
+ * timestamps (unaffected by system clock changes), excluding the periods of
+ * time the device was suspended.
+ * @throws NS_ERROR_NOT_AVAILABLE if unavailable.
+ */
+ double msSinceProcessStartExcludingSuspend();
+
+ /**
+ * Time since the system wide epoch. This is not a monotonic timer but
+ * can be used across process boundaries.
+ */
+ double msSystemNow();
+
+ /**
+ * Adds the value to the given scalar.
+ *
+ * @param aName The scalar name.
+ * @param aValue The numeric value to add to the scalar. Only unsigned integers supported.
+ */
+ [implicit_jscontext]
+ void scalarAdd(in ACString aName, in jsval aValue);
+
+ /**
+ * Sets the scalar to the given value.
+ *
+ * @param aName The scalar name.
+ * @param aValue The value to set the scalar to. If the type of aValue doesn't match the
+ * type of the scalar, the function will fail. For scalar string types, the this
+ * is truncated to 50 characters.
+ */
+ [implicit_jscontext]
+ void scalarSet(in ACString aName, in jsval aValue);
+
+ /**
+ * Sets the scalar to the maximum of the current and the passed value.
+ *
+ * @param aName The scalar name.
+ * @param aValue The numeric value to set the scalar to. Only unsigned integers supported.
+ */
+ [implicit_jscontext]
+ void scalarSetMaximum(in ACString aName, in jsval aValue);
+
+ /**
+ * Adds the value to the given keyed scalar.
+ *
+ * @param aName The scalar name.
+ * @param aKey The key name.
+ * @param aValue The numeric value to add to the scalar. Only unsigned integers supported.
+ */
+ [implicit_jscontext]
+ void keyedScalarAdd(in ACString aName, in AString aKey, in jsval aValue);
+
+ /**
+ * Sets the keyed scalar to the given value.
+ *
+ * @param aName The scalar name.
+ * @param aKey The key name.
+ * @param aValue The value to set the scalar to. If the type of aValue doesn't match the
+ * type of the scalar, the function will fail.
+ */
+ [implicit_jscontext]
+ void keyedScalarSet(in ACString aName, in AString aKey, in jsval aValue);
+
+ /**
+ * Sets the keyed scalar to the maximum of the current and the passed value.
+ *
+ * @param aName The scalar name.
+ * @param aKey The key name.
+ * @param aValue The numeric value to set the scalar to. Only unsigned integers supported.
+ */
+ [implicit_jscontext]
+ void keyedScalarSetMaximum(in ACString aName, in AString aKey, in jsval aValue);
+
+ /**
+ * Resets all the stored scalars. This is intended to be only used in tests.
+ */
+ void clearScalars();
+
+ /**
+ * Immediately sends any Telemetry batched on this process to the parent
+ * process. This is intended only to be used on process shutdown.
+ */
+ void flushBatchedChildTelemetry();
+
+ /**
+ * Record an event in Telemetry.
+ *
+ * @param aCategory The category name.
+ * @param aMethod The method name.
+ * @param aObject The object name.
+ * @param aValue An optional string value to record.
+ * @param aExtra An optional object of the form (string -> string).
+ * It should only contain registered extra keys.
+ *
+ * @throws NS_ERROR_INVALID_ARG When trying to record an unknown event.
+ */
+ [implicit_jscontext, optional_argc]
+ void recordEvent(in ACString aCategory, in ACString aMethod, in ACString aObject, [optional] in jsval aValue, [optional] in jsval extra);
+
+ /**
+ * Enable recording of events in a category.
+ * Events default to recording disabled. This allows to toggle recording for all events
+ * in the specified category.
+ *
+ * @param aCategory The category name.
+ * @param aEnabled Whether recording is enabled for events in that category.
+ */
+ void setEventRecordingEnabled(in ACString aCategory, in boolean aEnabled);
+
+ /**
+ * Serializes the recorded events to a JSON-appropriate array and optionally resets them.
+ * The returned structure looks like this:
+ * [
+ * // [timestamp, category, method, object, stringValue, extraValues]
+ * [43245, "category1", "method1", "object1", "string value", null],
+ * [43258, "category1", "method2", "object1", null, {"key1": "string value"}],
+ * ...
+ * ]
+ *
+ * @param aDataset DATASET_ALL_CHANNELS or DATASET_PRERELEASE_CHANNELS.
+ * @param [aClear=false] Whether to clear out the flushed events after snapshotting.
+ * @param aEventLimit How many events per process to limit the snapshot to contain, all if unspecified.
+ * Even if aClear, the leftover event records are not cleared.
+ */
+ [implicit_jscontext, optional_argc]
+ jsval snapshotEvents(in uint32_t aDataset, [optional] in boolean aClear, [optional] in uint32_t aEventLimit);
+
+ /**
+ * Register new events to record them from addons. This allows registering multiple
+ * events for a category. They will be valid only for the current Firefox session.
+ * Note that events shipping in Firefox should be registered in Events.yaml.
+ *
+ * @param aCategory The unique category the events are registered in.
+ * @param aEventData An object that contains registration data for 1-N events of the form:
+ * {
+ * "categoryName": {
+ * "methods": ["test1"],
+ * "objects": ["object1"],
+ * "record_on_release": false,
+ * "extra_keys": ["key1", "key2"], // optional
+ * "expired": false // optional, defaults to false.
+ * },
+ * ...
+ * }
+ * @param aEventData.<name>.methods List of methods for this event entry.
+ * @param aEventData.<name>.objects List of objects for this event entry.
+ * @param aEventData.<name>.extra_keys Optional, list of allowed extra keys for this event entry.
+ * @param aEventData.<name>.record_on_release Optional, whether to record this data on release.
+ * Defaults to false.
+ * @param aEventData.<name>.expired Optional, whether this event entry is expired. This allows
+ * recording it without error, but it will be discarded. Defaults to false.
+ */
+ [implicit_jscontext]
+ void registerEvents(in ACString aCategory, in jsval aEventData);
+
+ /**
+ * Parent process only. Register dynamic builtin events. The parameters
+ * have the same meaning as the usual |registerEvents| function.
+ *
+ * This function is only meant to be used to support the "artifact build"/
+ * "build faster" developers by allowing to add new events without rebuilding
+ * the C++ components including the headers files.
+ */
+ [implicit_jscontext]
+ void registerBuiltinEvents(in ACString aCategory, in jsval aEventData);
+
+ /**
+ * Parent process only. Register new scalars to record them from addons. This
+ * allows registering multiple scalars for a category. They will be valid only for
+ * the current Firefox session.
+ * Note that scalars shipping in Firefox should be registered in Scalars.yaml.
+ *
+ * @param aCategoryName The unique category the scalars are registered in.
+ * @param aScalarData An object that contains registration data for multiple scalars in the form:
+ * {
+ * "sample_scalar": {
+ * "kind": Ci.nsITelemetry.SCALAR_TYPE_COUNT,
+ * "keyed": true, //optional, defaults to false
+ * "record_on_release: true, // optional, defaults to false
+ * "expired": false // optional, defaults to false.
+ * },
+ * ...
+ * }
+ * @param aScalarData.<name>.kind One of the scalar types defined in this file (SCALAR_TYPE_*)
+ * @param aScalarData.<name>.keyed Optional, whether this is a keyed scalar or not. Defaults to false.
+ * @param aScalarData.<name>.record_on_release Optional, whether to record this data on release.
+ * Defaults to false.
+ * @param aScalarData.<name>.expired Optional, whether this scalar entry is expired. This allows
+ * recording it without error, but it will be discarded. Defaults to false.
+ */
+ [implicit_jscontext]
+ void registerScalars(in ACString aCategoryName, in jsval aScalarData);
+
+ /**
+ * Parent process only. Register dynamic builtin scalars. The parameters
+ * have the same meaning as the usual |registerScalars| function.
+ *
+ * This function is only meant to be used to support the "artifact build"/
+ * "build faster" developers by allowing to add new scalars without rebuilding
+ * the C++ components including the headers files.
+ */
+ [implicit_jscontext]
+ void registerBuiltinScalars(in ACString aCategoryName, in jsval aScalarData);
+
+ /**
+ * Resets all the stored events. This is intended to be only used in tests.
+ * Events recorded but not yet flushed to the parent process storage won't be cleared.
+ * Override the pref. `toolkit.telemetry.ipcBatchTimeout` to reduce the time to flush events.
+ */
+ void clearEvents();
+
+ /**
+ * Get a list of all registered stores.
+ *
+ * The list is deduplicated, but unordered.
+ */
+ [implicit_jscontext]
+ jsval getAllStores();
+
+ /**
+ * Does early, cheap initialization for native telemetry data providers.
+ * Currently, this includes only MemoryTelemetry.
+ */
+ void earlyInit();
+
+ /**
+ * Does late, expensive initialization for native telemetry data providers.
+ * Currently, this includes only MemoryTelemetry.
+ *
+ * This should only be called after startup has completed and the event loop
+ * is idle.
+ */
+ void delayedInit();
+
+ /**
+ * Shuts down native telemetry providers. Currently, this includes only
+ * MemoryTelemetry.
+ */
+ void shutdown();
+
+ /**
+ * Gathers telemetry data for memory usage and records it to the data store.
+ * Returns a promise which resolves when asynchronous data collection has
+ * completed and all data has been recorded.
+ */
+ [implicit_jscontext]
+ Promise gatherMemory();
+
+ /**
+ * Serializes the per-origin data in plain text, optionally clearing
+ * the storage. Only to be used by about:telemetry.
+ *
+ * The returned structure looks like:
+ * { metric: {origin1: count1, origin2: count2, ...}, ...}
+ *
+ * @param aClear Whether to clear the storage. Default false.
+ * @return a snapshot of the per-origin data.
+ */
+ [implicit_jscontext]
+ jsval getOriginSnapshot([optional] in boolean aClear);
+
+ /**
+ * Encodes the per-origin information then serializes it.
+ * Returns a Promise.
+ *
+ * @param aClear Whether to clear the storage. Default false.
+ * @return Promise that resolves to the serialized encoded data.
+ */
+ [implicit_jscontext]
+ Promise getEncodedOriginSnapshot([optional] in boolean aClear);
+
+ /**
+ * Clears Firefox Origin Telemetry. Only to be used in tests.
+ */
+ void clearOrigins();
+};