Abseil Common Libraries (C++) (grcp 依赖)
https://abseil.io/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
169 lines
7.0 KiB
169 lines
7.0 KiB
4 years ago
|
// Copyright 2019 The Abseil Authors.
|
||
|
//
|
||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
// you may not use this file except in compliance with the License.
|
||
|
// You may obtain a copy of the License at
|
||
|
//
|
||
|
// https://www.apache.org/licenses/LICENSE-2.0
|
||
|
//
|
||
|
// Unless required by applicable law or agreed to in writing, software
|
||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
// See the License for the specific language governing permissions and
|
||
|
// limitations under the License.
|
||
|
|
||
|
#ifndef ABSL_STRINGS_CORDZ_INFO_H_
|
||
|
#define ABSL_STRINGS_CORDZ_INFO_H_
|
||
|
|
||
|
#include <atomic>
|
||
|
#include <cstdint>
|
||
|
#include <functional>
|
||
|
|
||
|
#include "absl/base/config.h"
|
||
|
#include "absl/base/thread_annotations.h"
|
||
|
#include "absl/strings/internal/cord_internal.h"
|
||
|
#include "absl/strings/internal/cordz_handle.h"
|
||
|
#include "absl/strings/internal/cordz_statistics.h"
|
||
|
#include "absl/synchronization/mutex.h"
|
||
|
#include "absl/types/span.h"
|
||
|
|
||
|
namespace absl {
|
||
|
ABSL_NAMESPACE_BEGIN
|
||
|
namespace cord_internal {
|
||
|
|
||
|
// CordzInfo tracks a profiled Cord. Each of these objects can be in two places.
|
||
|
// If a Cord is alive, the CordzInfo will be in the global_cordz_infos map, and
|
||
|
// can also be retrieved via the linked list starting with
|
||
|
// global_cordz_infos_head and continued via the cordz_info_next() method. When
|
||
|
// a Cord has reached the end of its lifespan, the CordzInfo object will be
|
||
|
// migrated out of the global_cordz_infos list and the global_cordz_infos_map,
|
||
|
// and will either be deleted or appended to the global_delete_queue. If it is
|
||
|
// placed on the global_delete_queue, the CordzInfo object will be cleaned in
|
||
|
// the destructor of a CordzSampleToken object.
|
||
|
class CordzInfo : public CordzHandle {
|
||
|
public:
|
||
|
// All profiled Cords should be accompanied by a call to TrackCord.
|
||
|
// TrackCord creates a CordzInfo instance which tracks important metrics of
|
||
|
// the sampled cord. CordzInfo instances are placed in a global list which is
|
||
|
// used to discover and snapshot all actively tracked cords.
|
||
|
// Callers are responsible for calling UntrackCord() before the tracked Cord
|
||
|
// instance is deleted, or to stop tracking the sampled Cord.
|
||
|
static CordzInfo* TrackCord(CordRep* rep);
|
||
|
|
||
|
// Stops tracking changes for a sampled cord, and deletes the provided info.
|
||
|
// This function must be called before the sampled cord instance is deleted,
|
||
|
// and before the root cordrep of the sampled cord is unreffed.
|
||
|
// This function may extend the lifetime of the cordrep in cases where the
|
||
|
// CordInfo instance is being held by a concurrent collection thread.
|
||
|
static void UntrackCord(CordzInfo* cordz_info);
|
||
|
|
||
|
// Identical to TrackCord(), except that this function fills the
|
||
|
// 'parent_stack' property of the returned CordzInfo instance from the
|
||
|
// provided `src` instance if `src` is not null.
|
||
|
// This function should be used for sampling 'copy constructed' cords.
|
||
|
static CordzInfo* TrackCord(CordRep* rep, const CordzInfo* src);
|
||
|
|
||
|
CordzInfo() = delete;
|
||
|
CordzInfo(const CordzInfo&) = delete;
|
||
|
CordzInfo& operator=(const CordzInfo&) = delete;
|
||
|
|
||
|
// Retrieves the oldest existing CordzInfo.
|
||
|
static CordzInfo* Head(const CordzSnapshot& snapshot);
|
||
|
|
||
|
// Retrieves the next oldest existing CordzInfo older than 'this' instance.
|
||
|
CordzInfo* Next(const CordzSnapshot& snapshot) const;
|
||
|
|
||
|
// Returns a reference to the mutex guarding the `rep` property of this
|
||
|
// instance. CordzInfo instances hold a weak reference to the rep pointer of
|
||
|
// sampled cords, and rely on Cord logic to update the rep pointer when the
|
||
|
// underlying root tree or ring of the cord changes.
|
||
|
absl::Mutex& mutex() const { return mutex_; }
|
||
|
|
||
|
// Updates the `rep' property of this instance. This methods is invoked by
|
||
|
// Cord logic each time the root node of a sampled Cord changes, and before
|
||
|
// the old root reference count is deleted. This guarantees that collection
|
||
|
// code can always safely take a reference on the tracked cord.
|
||
|
// Requires `mutex()` to be held.
|
||
|
// TODO(b/117940323): annotate with ABSL_EXCLUSIVE_LOCKS_REQUIRED once all
|
||
|
// Cord code is in a state where this can be proven true by the compiler.
|
||
|
void SetCordRep(CordRep* rep);
|
||
|
|
||
|
// Returns the current value of `rep_` for testing purposes only.
|
||
|
CordRep* GetCordRepForTesting() const ABSL_NO_THREAD_SAFETY_ANALYSIS {
|
||
|
return rep_;
|
||
|
}
|
||
|
|
||
|
// Returns the stack trace for where the cord was first sampled. Cords are
|
||
|
// potentially sampled when they promote from an inlined cord to a tree or
|
||
|
// ring representation, which is not necessarily the location where the cord
|
||
|
// was first created. Some cords are created as inlined cords, and only as
|
||
|
// data is added do they become a non-inlined cord. However, typically the
|
||
|
// location represents reasonably well where the cord is 'created'.
|
||
|
absl::Span<void* const> GetStack() const;
|
||
|
|
||
|
// Returns the stack trace for a sampled cord's 'parent stack trace'. This
|
||
|
// value may be set if the cord is sampled (promoted) after being created
|
||
|
// from, or being assigned the value of an existing (sampled) cord.
|
||
|
absl::Span<void* const> GetParentStack() const;
|
||
|
|
||
|
// Retrieve the CordzStatistics associated with this Cord. The statistics are
|
||
|
// only updated when a Cord goes through a mutation, such as an Append or
|
||
|
// RemovePrefix. The refcounts can change due to external events, so the
|
||
|
// reported refcount stats might be incorrect.
|
||
|
CordzStatistics GetCordzStatistics() const {
|
||
|
CordzStatistics stats;
|
||
|
stats.size = size_.load(std::memory_order_relaxed);
|
||
|
return stats;
|
||
|
}
|
||
|
|
||
|
// Records size metric for this CordzInfo instance.
|
||
|
void RecordMetrics(int64_t size) {
|
||
|
size_.store(size, std::memory_order_relaxed);
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
static constexpr int kMaxStackDepth = 64;
|
||
|
|
||
|
explicit CordzInfo(CordRep* tree);
|
||
|
~CordzInfo() override;
|
||
|
|
||
|
void Track();
|
||
|
void Untrack();
|
||
|
|
||
|
// 'Unsafe' head/next/prev accessors not requiring the lock being held.
|
||
|
// These are used exclusively for iterations (Head / Next) where we enforce
|
||
|
// a token being held, so reading an 'old' / deleted pointer is fine.
|
||
|
static CordzInfo* ci_head_unsafe() ABSL_NO_THREAD_SAFETY_ANALYSIS {
|
||
|
return ci_head_.load(std::memory_order_acquire);
|
||
|
}
|
||
|
CordzInfo* ci_next_unsafe() const ABSL_NO_THREAD_SAFETY_ANALYSIS {
|
||
|
return ci_next_.load(std::memory_order_acquire);
|
||
|
}
|
||
|
CordzInfo* ci_prev_unsafe() const ABSL_NO_THREAD_SAFETY_ANALYSIS {
|
||
|
return ci_prev_.load(std::memory_order_acquire);
|
||
|
}
|
||
|
|
||
|
static absl::Mutex ci_mutex_;
|
||
|
static std::atomic<CordzInfo*> ci_head_ ABSL_GUARDED_BY(ci_mutex_);
|
||
|
std::atomic<CordzInfo*> ci_prev_ ABSL_GUARDED_BY(ci_mutex_){nullptr};
|
||
|
std::atomic<CordzInfo*> ci_next_ ABSL_GUARDED_BY(ci_mutex_){nullptr};
|
||
|
|
||
|
mutable absl::Mutex mutex_;
|
||
|
CordRep* rep_ ABSL_GUARDED_BY(mutex());
|
||
|
|
||
|
void* stack_[kMaxStackDepth];
|
||
|
void* parent_stack_[kMaxStackDepth];
|
||
|
const int stack_depth_;
|
||
|
int parent_stack_depth_;
|
||
|
const absl::Time create_time_;
|
||
|
|
||
|
// Last recorded size for the cord.
|
||
|
std::atomic<int64_t> size_{0};
|
||
|
};
|
||
|
|
||
|
} // namespace cord_internal
|
||
|
ABSL_NAMESPACE_END
|
||
|
} // namespace absl
|
||
|
|
||
|
#endif // ABSL_STRINGS_CORDZ_INFO_H_
|