blob: 010e0187b11948601cd6466a1b5b415b3d19d134 [file] [log] [blame]
reveman94ff17c2014-12-24 00:52:301// Copyright 2014 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#ifndef CONTENT_COMMON_DISCARDABLE_SHARED_MEMORY_HEAP_H_
6#define CONTENT_COMMON_DISCARDABLE_SHARED_MEMORY_HEAP_H_
7
avia9aa7a82015-12-25 03:06:318#include <stddef.h>
9#include <stdint.h>
10
reveman1500e1a72015-03-17 02:31:3011#include "base/callback.h"
reveman94ff17c2014-12-24 00:52:3012#include "base/containers/hash_tables.h"
13#include "base/containers/linked_list.h"
avia9aa7a82015-12-25 03:06:3114#include "base/macros.h"
reveman94ff17c2014-12-24 00:52:3015#include "base/memory/scoped_ptr.h"
16#include "base/memory/scoped_vector.h"
ssid3c7fbd4b2015-05-05 17:43:0017#include "base/trace_event/process_memory_dump.h"
reveman94ff17c2014-12-24 00:52:3018#include "content/common/content_export.h"
19
20namespace base {
21class DiscardableSharedMemory;
22}
23
24namespace content {
25
reveman3e0e72c02015-03-11 03:55:5926// Implements a heap of discardable shared memory. An array of free lists
27// is used to keep track of free blocks.
reveman94ff17c2014-12-24 00:52:3028class CONTENT_EXPORT DiscardableSharedMemoryHeap {
29 public:
30 class CONTENT_EXPORT Span : public base::LinkNode<Span> {
31 public:
32 ~Span();
33
reveman549bd27a2015-02-18 02:52:1934 base::DiscardableSharedMemory* shared_memory() { return shared_memory_; }
reveman94ff17c2014-12-24 00:52:3035 size_t start() const { return start_; }
36 size_t length() const { return length_; }
ssidb16ac462015-10-13 20:10:2737 void set_is_locked(bool is_locked) { is_locked_ = is_locked; }
reveman94ff17c2014-12-24 00:52:3038
39 private:
40 friend class DiscardableSharedMemoryHeap;
41
reveman549bd27a2015-02-18 02:52:1942 Span(base::DiscardableSharedMemory* shared_memory,
reveman94ff17c2014-12-24 00:52:3043 size_t start,
44 size_t length);
45
reveman549bd27a2015-02-18 02:52:1946 base::DiscardableSharedMemory* shared_memory_;
reveman94ff17c2014-12-24 00:52:3047 size_t start_;
48 size_t length_;
ssidb16ac462015-10-13 20:10:2749 bool is_locked_;
reveman94ff17c2014-12-24 00:52:3050
51 DISALLOW_COPY_AND_ASSIGN(Span);
52 };
53
54 explicit DiscardableSharedMemoryHeap(size_t block_size);
55 ~DiscardableSharedMemoryHeap();
56
reveman549bd27a2015-02-18 02:52:1957 // Grow heap using |shared_memory| and return a span for this new memory.
58 // |shared_memory| must be aligned to the block size and |size| must be a
reveman1500e1a72015-03-17 02:31:3059 // multiple of the block size. |deleted_callback| is called when
60 // |shared_memory| has been deleted.
reveman549bd27a2015-02-18 02:52:1961 scoped_ptr<Span> Grow(scoped_ptr<base::DiscardableSharedMemory> shared_memory,
reveman1500e1a72015-03-17 02:31:3062 size_t size,
ssid3c7fbd4b2015-05-05 17:43:0063 int32_t id,
reveman1500e1a72015-03-17 02:31:3064 const base::Closure& deleted_callback);
reveman94ff17c2014-12-24 00:52:3065
reveman3e0e72c02015-03-11 03:55:5966 // Merge |span| into the free lists. This will coalesce |span| with
67 // neighboring free spans when possible.
68 void MergeIntoFreeLists(scoped_ptr<Span> span);
reveman94ff17c2014-12-24 00:52:3069
70 // Split an allocated span into two spans, one of length |blocks| followed
71 // by another span of length "span->length - blocks" blocks. Modifies |span|
72 // to point to the first span of length |blocks|. Return second span.
73 scoped_ptr<Span> Split(Span* span, size_t blocks);
74
reveman3e0e72c02015-03-11 03:55:5975 // Search free lists for span that satisfies the request for |blocks| of
reveman94ff17c2014-12-24 00:52:3076 // memory. If found, the span is removed from the free list and returned.
reveman3e0e72c02015-03-11 03:55:5977 // |slack| determines the fitness requirement. Only spans that are less
78 // or equal to |blocks| + |slack| are considered, worse fitting spans are
79 // ignored.
80 scoped_ptr<Span> SearchFreeLists(size_t blocks, size_t slack);
reveman94ff17c2014-12-24 00:52:3081
reveman2ce02282015-03-09 23:57:2682 // Release free shared memory segments.
83 void ReleaseFreeMemory();
84
85 // Release shared memory segments that have been purged.
86 void ReleasePurgedMemory();
87
88 // Returns total bytes of memory in heap.
89 size_t GetSize() const;
90
reveman3e0e72c02015-03-11 03:55:5991 // Returns bytes of memory currently in the free lists.
92 size_t GetSizeOfFreeLists() const;
reveman94ff17c2014-12-24 00:52:3093
ssid3c7fbd4b2015-05-05 17:43:0094 // Dumps memory statistics for chrome://tracing.
95 bool OnMemoryDump(base::trace_event::ProcessMemoryDump* pmd);
96
primianoc255f38d2015-07-01 16:17:4797 // Returns a unique identifier for a given tuple of (process id, segment id)
98 // that can be used to match memory dumps across different processes.
99 static base::trace_event::MemoryAllocatorDumpGuid GetSegmentGUIDForTracing(
avia9aa7a82015-12-25 03:06:31100 uint64_t tracing_process_id,
101 int32_t segment_id);
primianoc255f38d2015-07-01 16:17:47102
ssid3e371552015-08-25 15:33:20103 // Returns a MemoryAllocatorDump for a given span on |pmd| with the size of
104 // the span.
105 base::trace_event::MemoryAllocatorDump* CreateMemoryAllocatorDump(
106 Span* span,
107 const char* name,
108 base::trace_event::ProcessMemoryDump* pmd) const;
109
reveman94ff17c2014-12-24 00:52:30110 private:
reveman2ce02282015-03-09 23:57:26111 class ScopedMemorySegment {
112 public:
113 ScopedMemorySegment(DiscardableSharedMemoryHeap* heap,
114 scoped_ptr<base::DiscardableSharedMemory> shared_memory,
reveman1500e1a72015-03-17 02:31:30115 size_t size,
ssid3c7fbd4b2015-05-05 17:43:00116 int32_t id,
reveman1500e1a72015-03-17 02:31:30117 const base::Closure& deleted_callback);
reveman2ce02282015-03-09 23:57:26118 ~ScopedMemorySegment();
119
120 bool IsUsed() const;
121 bool IsResident() const;
122
ssid3e371552015-08-25 15:33:20123 bool ContainsSpan(Span* span) const;
124
125 base::trace_event::MemoryAllocatorDump* CreateMemoryAllocatorDump(
126 Span* span,
ssid053b1f22015-08-27 11:14:59127 size_t block_size,
ssid3e371552015-08-25 15:33:20128 const char* name,
129 base::trace_event::ProcessMemoryDump* pmd) const;
130
ssid3c7fbd4b2015-05-05 17:43:00131 // Used for dumping memory statistics from the segment to chrome://tracing.
132 void OnMemoryDump(base::trace_event::ProcessMemoryDump* pmd) const;
133
reveman2ce02282015-03-09 23:57:26134 private:
135 DiscardableSharedMemoryHeap* const heap_;
136 scoped_ptr<base::DiscardableSharedMemory> shared_memory_;
137 const size_t size_;
ssid3c7fbd4b2015-05-05 17:43:00138 const int32_t id_;
reveman1500e1a72015-03-17 02:31:30139 const base::Closure deleted_callback_;
reveman2ce02282015-03-09 23:57:26140
141 DISALLOW_COPY_AND_ASSIGN(ScopedMemorySegment);
142 };
143
reveman3e0e72c02015-03-11 03:55:59144 void InsertIntoFreeList(scoped_ptr<Span> span);
reveman94ff17c2014-12-24 00:52:30145 scoped_ptr<Span> RemoveFromFreeList(Span* span);
146 scoped_ptr<Span> Carve(Span* span, size_t blocks);
147 void RegisterSpan(Span* span);
reveman549bd27a2015-02-18 02:52:19148 void UnregisterSpan(Span* span);
reveman2ce02282015-03-09 23:57:26149 bool IsMemoryUsed(const base::DiscardableSharedMemory* shared_memory,
150 size_t size);
151 bool IsMemoryResident(const base::DiscardableSharedMemory* shared_memory);
152 void ReleaseMemory(const base::DiscardableSharedMemory* shared_memory,
153 size_t size);
reveman94ff17c2014-12-24 00:52:30154
ssid3c7fbd4b2015-05-05 17:43:00155 // Dumps memory statistics about a memory segment for chrome://tracing.
156 void OnMemoryDump(const base::DiscardableSharedMemory* shared_memory,
157 size_t size,
primianoc255f38d2015-07-01 16:17:47158 int32_t segment_id,
ssid3c7fbd4b2015-05-05 17:43:00159 base::trace_event::ProcessMemoryDump* pmd);
160
reveman94ff17c2014-12-24 00:52:30161 size_t block_size_;
reveman2ce02282015-03-09 23:57:26162 size_t num_blocks_;
163 size_t num_free_blocks_;
reveman94ff17c2014-12-24 00:52:30164
reveman2ce02282015-03-09 23:57:26165 // Vector of memory segments.
166 ScopedVector<ScopedMemorySegment> memory_segments_;
reveman549bd27a2015-02-18 02:52:19167
reveman94ff17c2014-12-24 00:52:30168 // Mapping from first/last block of span to Span instance.
169 typedef base::hash_map<size_t, Span*> SpanMap;
170 SpanMap spans_;
171
reveman3e0e72c02015-03-11 03:55:59172 // Array of linked-lists with free discardable memory regions. For i < 256,
173 // where the 1st entry is located at index 0 of the array, the kth entry
174 // is a free list of runs that consist of k blocks. The 256th entry is a
175 // free list of runs that have length >= 256 blocks.
176 base::LinkedList<Span> free_spans_[256];
reveman94ff17c2014-12-24 00:52:30177
178 DISALLOW_COPY_AND_ASSIGN(DiscardableSharedMemoryHeap);
179};
180
181} // namespace content
182
183#endif // CONTENT_COMMON_DISCARDABLE_SHARED_MEMORY_HEAP_H_