gecko-dev/toolkit/components/telemetry/WebrtcTelemetry.cpp
2016-06-25 01:01:00 +02:00

134 lines
3.9 KiB
C++

/* -*- 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 "TelemetryCommon.h"
#include "WebrtcTelemetry.h"
#include "jsapi.h"
#include "nsPrintfCString.h"
#include "nsTHashtable.h"
using mozilla::Telemetry::Common::AutoHashtable;
void
WebrtcTelemetry::RecordIceCandidateMask(const uint32_t iceCandidateBitmask,
const bool success,
const bool loop)
{
WebrtcIceCandidateType *entry = mWebrtcIceCandidates.GetEntry(iceCandidateBitmask);
if (!entry) {
entry = mWebrtcIceCandidates.PutEntry(iceCandidateBitmask);
if (MOZ_UNLIKELY(!entry))
return;
}
if (loop) {
if (success) {
entry->mData.loop.successCount++;
} else {
entry->mData.loop.failureCount++;
}
} else {
if (success) {
entry->mData.webrtc.successCount++;
} else {
entry->mData.webrtc.failureCount++;
}
}
}
bool
ReflectIceEntry(const WebrtcTelemetry::WebrtcIceCandidateType *entry,
const WebrtcTelemetry::WebrtcIceCandidateStats *stat, JSContext *cx,
JS::Handle<JSObject*> obj)
{
if ((stat->successCount == 0) && (stat->failureCount == 0))
return true;
const uint32_t &bitmask = entry->GetKey();
JS::Rooted<JSObject*> statsObj(cx, JS_NewPlainObject(cx));
if (!statsObj)
return false;
if (!JS_DefineProperty(cx, obj,
nsPrintfCString("%lu", bitmask).BeginReading(),
statsObj, JSPROP_ENUMERATE)) {
return false;
}
if (stat->successCount && !JS_DefineProperty(cx, statsObj, "successCount",
stat->successCount,
JSPROP_ENUMERATE)) {
return false;
}
if (stat->failureCount && !JS_DefineProperty(cx, statsObj, "failureCount",
stat->failureCount,
JSPROP_ENUMERATE)) {
return false;
}
return true;
}
bool
ReflectIceWebrtc(WebrtcTelemetry::WebrtcIceCandidateType *entry, JSContext *cx,
JS::Handle<JSObject*> obj)
{
return ReflectIceEntry(entry, &entry->mData.webrtc, cx, obj);
}
bool
ReflectIceLoop(WebrtcTelemetry::WebrtcIceCandidateType *entry, JSContext *cx,
JS::Handle<JSObject*> obj)
{
return ReflectIceEntry(entry, &entry->mData.loop, cx, obj);
}
bool
WebrtcTelemetry::AddIceInfo(JSContext *cx, JS::Handle<JSObject*> iceObj,
const bool loop)
{
JS::Rooted<JSObject*> statsObj(cx, JS_NewPlainObject(cx));
if (!statsObj)
return false;
AutoHashtable<WebrtcIceCandidateType>::ReflectEntryFunc reflectFunction =
loop ? ReflectIceLoop : ReflectIceWebrtc;
if (!mWebrtcIceCandidates.ReflectIntoJS(reflectFunction, cx, statsObj)) {
return false;
}
return JS_DefineProperty(cx, iceObj, loop ? "loop" : "webrtc",
statsObj, JSPROP_ENUMERATE);
}
bool
WebrtcTelemetry::GetWebrtcStats(JSContext *cx, JS::MutableHandle<JS::Value> ret)
{
JS::Rooted<JSObject*> root_obj(cx, JS_NewPlainObject(cx));
if (!root_obj)
return false;
ret.setObject(*root_obj);
JS::Rooted<JSObject*> ice_obj(cx, JS_NewPlainObject(cx));
if (!ice_obj)
return false;
JS_DefineProperty(cx, root_obj, "IceCandidatesStats", ice_obj,
JSPROP_ENUMERATE);
if (!AddIceInfo(cx, ice_obj, false))
return false;
if (!AddIceInfo(cx, ice_obj, true))
return false;
return true;
}
size_t
WebrtcTelemetry::SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
{
return mWebrtcIceCandidates.ShallowSizeOfExcludingThis(aMallocSizeOf);
}