blob: ac8d895ca46d1843d0e76a1565f5b8877b745ec1 [file] [log] [blame]
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef BASE_TRACE_EVENT_HEAP_PROFILER_STACK_FRAME_DEDUPLICATOR_H_
#define BASE_TRACE_EVENT_HEAP_PROFILER_STACK_FRAME_DEDUPLICATOR_H_
#include <string>
#include <unordered_map>
#include "base/base_export.h"
#include "base/containers/circular_deque.h"
#include "base/containers/flat_map.h"
#include "base/macros.h"
#include "base/trace_event/heap_profiler_allocation_context.h"
#include "base/trace_event/trace_event_impl.h"
namespace base {
namespace trace_event {
class TraceEventMemoryOverhead;
// A data structure that allows grouping a set of backtraces in a space-
// efficient manner by creating a call tree and writing it as a set of (node,
// parent) pairs. The tree nodes reference both parent and children. The parent
// is referenced by index into |frames_|. The children are referenced via a map
// of |StackFrame|s to index into |frames_|. So there is a trie for bottum-up
// lookup of a backtrace for deduplication, and a tree for compact storage in
// the trace log.
class BASE_EXPORT StackFrameDeduplicator : public ConvertableToTraceFormat {
public:
// A node in the call tree.
struct FrameNode {
FrameNode(StackFrame frame, int parent_frame_index);
FrameNode(const FrameNode& other);
~FrameNode();
size_t EstimateMemoryUsage() const;
StackFrame frame;
// The index of the parent stack frame in |frames_|, or kInvalidFrameIndex
// if there is no parent frame (when it is at the bottom of the call stack).
int parent_frame_index;
constexpr static int kInvalidFrameIndex = -1;
// Indices into |frames_| of frames called from the current frame.
base::flat_map<StackFrame, int> children;
};
using ConstIterator = base::circular_deque<FrameNode>::const_iterator;
StackFrameDeduplicator();
~StackFrameDeduplicator() override;
// Inserts a backtrace where |beginFrame| is a pointer to the bottom frame
// (e.g. main) and |endFrame| is a pointer past the top frame (most recently
// called function), and returns the index of its leaf node in |frames_|.
// Returns -1 if the backtrace is empty.
int Insert(const StackFrame* beginFrame, const StackFrame* endFrame);
// Iterators over the frame nodes in the call tree.
ConstIterator begin() const { return frames_.begin(); }
ConstIterator end() const { return frames_.end(); }
// Writes the |stackFrames| dictionary as defined in https://goo.gl/GerkV8 to
// the trace log.
void AppendAsTraceFormat(std::string* out) const override;
// Estimates memory overhead including |sizeof(StackFrameDeduplicator)|.
void EstimateTraceMemoryOverhead(TraceEventMemoryOverhead* overhead) override;
private:
// Checks that existing backtrace identified by |frame_index| equals
// to the one identified by |begin_frame|, |end_frame|.
bool Match(int frame_index,
const StackFrame* begin_frame,
const StackFrame* end_frame) const;
base::flat_map<StackFrame, int> roots_;
base::circular_deque<FrameNode> frames_;
// {backtrace_hash -> frame_index} map for finding backtraces that are
// already added. Backtraces themselves are not stored in the map, instead
// Match() is used on the found frame_index to detect collisions.
std::unordered_map<size_t, int> backtrace_lookup_table_;
DISALLOW_COPY_AND_ASSIGN(StackFrameDeduplicator);
};
} // namespace trace_event
} // namespace base
#endif // BASE_TRACE_EVENT_HEAP_PROFILER_STACK_FRAME_DEDUPLICATOR_H_