/* -*- 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/glean/bindings/Counter.h" #include "nsITelemetry.h" #include "nsString.h" #include "mozilla/ResultVariant.h" #include "mozilla/dom/GleanMetricsBinding.h" #include "mozilla/glean/bindings/HistogramGIFFTMap.h" #include "mozilla/glean/bindings/ScalarGIFFTMap.h" #include "mozilla/glean/fog_ffi_generated.h" #include "GIFFTFwd.h" using mozilla::Telemetry::HistogramID; /* static */ void accumulateToBoolean(HistogramID aId, const nsACString& aLabel, int32_t aAmount) { MOZ_ASSERT(aAmount == 1, "When mirroring to boolean histograms, we only support " "accumulating one sample at a time."); if (aLabel.EqualsASCII("true")) { TelemetryHistogram::Accumulate(aId, true); } else if (aLabel.EqualsASCII("false")) { TelemetryHistogram::Accumulate(aId, false); } else { MOZ_ASSERT_UNREACHABLE( "When mirroring to boolean histograms, we only support labels 'true' " "and 'false'"); } } /* static */ void accumulateToKeyedCount(HistogramID aId, const nsCString& aLabel, int32_t aAmount) { TelemetryHistogram::Accumulate(aId, aLabel, aAmount); } /* static */ void accumulateToCategorical(HistogramID aId, const nsCString& aLabel, int32_t aAmount) { MOZ_ASSERT(aAmount == 1, "When mirroring to categorical histograms, we only support " "accumulating one sample at a time."); TelemetryHistogram::AccumulateCategorical(aId, aLabel); } namespace mozilla::glean { namespace impl { void CounterMetric::Add(int32_t aAmount) const { auto scalarId = ScalarIdForMetric(mId); if (aAmount >= 0) { if (scalarId) { TelemetryScalar::Add(scalarId.extract(), aAmount); } else if (IsSubmetricId(mId)) { bool mirrorsToKeyedScalar = false; GetLabeledMirrorLock().apply([&](const auto& lock) { auto tuple = lock.ref()->MaybeGet(mId); mirrorsToKeyedScalar = !!tuple; if (tuple && aAmount > 0) { TelemetryScalar::Add(std::get<0>(tuple.ref()), std::get<1>(tuple.ref()), (uint32_t)aAmount); } }); if (!mirrorsToKeyedScalar) { GetLabeledDistributionMirrorLock().apply([&](const auto& lock) { auto tuple = lock.ref()->MaybeGet(mId); if (tuple) { HistogramID hId = std::get<0>(tuple.ref()); switch (TelemetryHistogram::GetHistogramType(hId)) { case nsITelemetry::HISTOGRAM_BOOLEAN: accumulateToBoolean(hId, std::get<1>(tuple.ref()), aAmount); break; case nsITelemetry::HISTOGRAM_COUNT: accumulateToKeyedCount(hId, std::get<1>(tuple.ref()), aAmount); break; case nsITelemetry::HISTOGRAM_CATEGORICAL: accumulateToCategorical(hId, std::get<1>(tuple.ref()), aAmount); break; default: MOZ_ASSERT_UNREACHABLE( "Asked to mirror labeled_counter to unsupported histogram " "type."); break; } } }); } } else { auto hgramId = HistogramIdForMetric(mId); if (hgramId) { TelemetryHistogram::Accumulate(hgramId.extract(), aAmount); } } } fog_counter_add(mId, aAmount); } Result, nsCString> CounterMetric::TestGetValue( const nsACString& aPingName) const { nsCString err; if (fog_counter_test_get_error(mId, &err)) { return Err(err); } if (!fog_counter_test_has_value(mId, &aPingName)) { return Maybe(); // can't use Nothing() or templates will fail. } return Some(fog_counter_test_get_value(mId, &aPingName)); } } // namespace impl /* virtual */ JSObject* GleanCounter::WrapObject(JSContext* aCx, JS::Handle aGivenProto) { return dom::GleanCounter_Binding::Wrap(aCx, this, aGivenProto); } void GleanCounter::Add(int32_t aAmount) { mCounter.Add(aAmount); } dom::Nullable GleanCounter::TestGetValue(const nsACString& aPingName, ErrorResult& aRv) { dom::Nullable ret; auto result = mCounter.TestGetValue(aPingName); if (result.isErr()) { aRv.ThrowDataError(result.unwrapErr()); return ret; } auto optresult = result.unwrap(); if (!optresult.isNothing()) { ret.SetValue(optresult.value()); } return ret; } } // namespace mozilla::glean