From 770dd3ad0d4bffa674b2cd88b6a098de6f76145b Mon Sep 17 00:00:00 2001 From: Lalit Date: Wed, 16 Nov 2022 22:25:32 -0800 Subject: [PATCH 1/2] fix meters access in thread safe manner --- .../opentelemetry/sdk/metrics/meter_context.h | 12 ++++++++++-- sdk/src/metrics/meter_context.cc | 17 +++++++++++++++-- sdk/src/metrics/state/metric_collector.cc | 7 ++++--- 3 files changed, 29 insertions(+), 7 deletions(-) diff --git a/sdk/include/opentelemetry/sdk/metrics/meter_context.h b/sdk/include/opentelemetry/sdk/metrics/meter_context.h index 6a0e3f65d7..56483546eb 100644 --- a/sdk/include/opentelemetry/sdk/metrics/meter_context.h +++ b/sdk/include/opentelemetry/sdk/metrics/meter_context.h @@ -56,7 +56,15 @@ class MeterContext : public std::enable_shared_from_this ViewRegistry *GetViewRegistry() const noexcept; /** - * Obtain the configured meters. + * NOTE - INTERNAL method, can change in future. + * Process callback for each meter in thread-safe manner + */ + bool ForEachMeter(nostd::function_ref &meter)> callback) noexcept; + + /** + * NOTE - INTERNAL method, can change in future. + * Obtain the configured meters. + * This method is NOT thread-safe. * */ nostd::span> GetMeters() noexcept; @@ -124,7 +132,7 @@ class MeterContext : public std::enable_shared_from_this std::atomic_flag shutdown_latch_ = ATOMIC_FLAG_INIT; opentelemetry::common::SpinLockMutex forceflush_lock_; - opentelemetry::common::SpinLockMutex storage_lock_; + opentelemetry::common::SpinLockMutex meter_lock_; }; } // namespace metrics diff --git a/sdk/src/metrics/meter_context.cc b/sdk/src/metrics/meter_context.cc index 497340145b..9360f43a43 100644 --- a/sdk/src/metrics/meter_context.cc +++ b/sdk/src/metrics/meter_context.cc @@ -28,9 +28,22 @@ ViewRegistry *MeterContext::GetViewRegistry() const noexcept return views_.get(); } +bool MeterContext::ForEachMeter(nostd::function_ref &meter)> callback) noexcept +{ + std::lock_guard guard(meter_lock_); + for (auto &meter: meters_) + { + if (!callback(meter)) + { + return false; + } + } + return true; +} + nostd::span> MeterContext::GetMeters() noexcept { - std::lock_guard guard(storage_lock_); + // no lock required, as this is called by MeterProvider in thread-safe manner. return nostd::span>{meters_}; } @@ -59,7 +72,7 @@ void MeterContext::AddView(std::unique_ptr instrument_select void MeterContext::AddMeter(std::shared_ptr meter) { - std::lock_guard guard(storage_lock_); + std::lock_guard guard(meter_lock_); meters_.push_back(meter); } diff --git a/sdk/src/metrics/state/metric_collector.cc b/sdk/src/metrics/state/metric_collector.cc index ed14f96c7d..6285e1f88f 100644 --- a/sdk/src/metrics/state/metric_collector.cc +++ b/sdk/src/metrics/state/metric_collector.cc @@ -38,14 +38,15 @@ bool MetricCollector::Collect( return false; } ResourceMetrics resource_metrics; - for (auto &meter : meter_context_->GetMeters()) + meter_context_->ForEachMeter([&](std::shared_ptr meter) noexcept { auto collection_ts = std::chrono::system_clock::now(); ScopeMetrics scope_metrics; scope_metrics.metric_data_ = meter->Collect(this, collection_ts); scope_metrics.scope_ = meter->GetInstrumentationScope(); - resource_metrics.scope_metric_data_.push_back(scope_metrics); - } + resource_metrics.scope_metric_data_.push_back(scope_metrics); + return true; + }); resource_metrics.resource_ = &meter_context_->GetResource(); callback(resource_metrics); return true; From bd54556feb62d81ba362a11ac8ba7a80d5286fef Mon Sep 17 00:00:00 2001 From: Lalit Date: Wed, 16 Nov 2022 22:29:44 -0800 Subject: [PATCH 2/2] fix format, and comment --- .../opentelemetry/sdk/metrics/meter_context.h | 12 ++++++------ sdk/src/metrics/meter_context.cc | 15 ++++++++------- sdk/src/metrics/state/metric_collector.cc | 5 ++--- 3 files changed, 16 insertions(+), 16 deletions(-) diff --git a/sdk/include/opentelemetry/sdk/metrics/meter_context.h b/sdk/include/opentelemetry/sdk/metrics/meter_context.h index 56483546eb..7943f0fdfd 100644 --- a/sdk/include/opentelemetry/sdk/metrics/meter_context.h +++ b/sdk/include/opentelemetry/sdk/metrics/meter_context.h @@ -59,12 +59,12 @@ class MeterContext : public std::enable_shared_from_this * NOTE - INTERNAL method, can change in future. * Process callback for each meter in thread-safe manner */ - bool ForEachMeter(nostd::function_ref &meter)> callback) noexcept; + bool ForEachMeter(nostd::function_ref &meter)> callback) noexcept; - /** + /** * NOTE - INTERNAL method, can change in future. - * Obtain the configured meters. - * This method is NOT thread-safe. + * Get the configured meters. + * This method is NOT thread safe, and only called through MeterProvider * */ nostd::span> GetMeters() noexcept; @@ -104,8 +104,8 @@ class MeterContext : public std::enable_shared_from_this std::unique_ptr view) noexcept; /** - * Adds a meter to the list of configured meters. - * Note: This method is INTERNAL to sdk not thread safe. + * NOTE - INTERNAL method, can change in future. + * Adds a meter to the list of configured meters in thread safe manner. * * @param meter */ diff --git a/sdk/src/metrics/meter_context.cc b/sdk/src/metrics/meter_context.cc index 9360f43a43..57fa372a2c 100644 --- a/sdk/src/metrics/meter_context.cc +++ b/sdk/src/metrics/meter_context.cc @@ -28,17 +28,18 @@ ViewRegistry *MeterContext::GetViewRegistry() const noexcept return views_.get(); } -bool MeterContext::ForEachMeter(nostd::function_ref &meter)> callback) noexcept +bool MeterContext::ForEachMeter( + nostd::function_ref &meter)> callback) noexcept { std::lock_guard guard(meter_lock_); - for (auto &meter: meters_) + for (auto &meter : meters_) + { + if (!callback(meter)) { - if (!callback(meter)) - { - return false; - } + return false; } - return true; + } + return true; } nostd::span> MeterContext::GetMeters() noexcept diff --git a/sdk/src/metrics/state/metric_collector.cc b/sdk/src/metrics/state/metric_collector.cc index 6285e1f88f..cc1883275b 100644 --- a/sdk/src/metrics/state/metric_collector.cc +++ b/sdk/src/metrics/state/metric_collector.cc @@ -38,13 +38,12 @@ bool MetricCollector::Collect( return false; } ResourceMetrics resource_metrics; - meter_context_->ForEachMeter([&](std::shared_ptr meter) noexcept - { + meter_context_->ForEachMeter([&](std::shared_ptr meter) noexcept { auto collection_ts = std::chrono::system_clock::now(); ScopeMetrics scope_metrics; scope_metrics.metric_data_ = meter->Collect(this, collection_ts); scope_metrics.scope_ = meter->GetInstrumentationScope(); - resource_metrics.scope_metric_data_.push_back(scope_metrics); + resource_metrics.scope_metric_data_.push_back(scope_metrics); return true; }); resource_metrics.resource_ = &meter_context_->GetResource();