1 /******************************************************************************/
2 #ifdef JEMALLOC_H_TYPES
3 
4 typedef struct extent_node_s extent_node_t;
5 
6 #endif /* JEMALLOC_H_TYPES */
7 /******************************************************************************/
8 #ifdef JEMALLOC_H_STRUCTS
9 
10 /* Tree of extents.  Use accessor functions for en_* fields. */
11 struct extent_node_s {
12 	/* Arena from which this extent came, if any. */
13 	arena_t			*en_arena;
14 
15 	/* Pointer to the extent that this tree node is responsible for. */
16 	void			*en_addr;
17 
18 	/* Total region size. */
19 	size_t			en_size;
20 
21 	/*
22 	 * Serial number (potentially non-unique).
23 	 *
24 	 * In principle serial numbers can wrap around on 32-bit systems if
25 	 * JEMALLOC_MUNMAP is defined, but as long as comparison functions fall
26 	 * back on address comparison for equal serial numbers, stable (if
27 	 * imperfect) ordering is maintained.
28 	 *
29 	 * Serial numbers may not be unique even in the absence of wrap-around,
30 	 * e.g. when splitting an extent and assigning the same serial number to
31 	 * both resulting adjacent extents.
32 	 */
33 	size_t			en_sn;
34 
35 	/*
36 	 * The zeroed flag is used by chunk recycling code to track whether
37 	 * memory is zero-filled.
38 	 */
39 	bool			en_zeroed;
40 
41 	/*
42 	 * True if physical memory is committed to the extent, whether
43 	 * explicitly or implicitly as on a system that overcommits and
44 	 * satisfies physical memory needs on demand via soft page faults.
45 	 */
46 	bool			en_committed;
47 
48 	/*
49 	 * The achunk flag is used to validate that huge allocation lookups
50 	 * don't return arena chunks.
51 	 */
52 	bool			en_achunk;
53 
54 	/* Profile counters, used for huge objects. */
55 	prof_tctx_t		*en_prof_tctx;
56 
57 	/* Linkage for arena's runs_dirty and chunks_cache rings. */
58 	arena_runs_dirty_link_t	rd;
59 	qr(extent_node_t)	cc_link;
60 
61 	union {
62 		/* Linkage for the size/sn/address-ordered tree. */
63 		rb_node(extent_node_t)	szsnad_link;
64 
65 		/* Linkage for arena's achunks, huge, and node_cache lists. */
66 		ql_elm(extent_node_t)	ql_link;
67 	};
68 
69 	/* Linkage for the address-ordered tree. */
70 	rb_node(extent_node_t)	ad_link;
71 };
72 typedef rb_tree(extent_node_t) extent_tree_t;
73 
74 #endif /* JEMALLOC_H_STRUCTS */
75 /******************************************************************************/
76 #ifdef JEMALLOC_H_EXTERNS
77 
78 rb_proto(, extent_tree_szsnad_, extent_tree_t, extent_node_t)
79 
80 rb_proto(, extent_tree_ad_, extent_tree_t, extent_node_t)
81 
82 #endif /* JEMALLOC_H_EXTERNS */
83 /******************************************************************************/
84 #ifdef JEMALLOC_H_INLINES
85 
86 #ifndef JEMALLOC_ENABLE_INLINE
87 arena_t	*extent_node_arena_get(const extent_node_t *node);
88 void	*extent_node_addr_get(const extent_node_t *node);
89 size_t	extent_node_size_get(const extent_node_t *node);
90 size_t	extent_node_sn_get(const extent_node_t *node);
91 bool	extent_node_zeroed_get(const extent_node_t *node);
92 bool	extent_node_committed_get(const extent_node_t *node);
93 bool	extent_node_achunk_get(const extent_node_t *node);
94 prof_tctx_t	*extent_node_prof_tctx_get(const extent_node_t *node);
95 void	extent_node_arena_set(extent_node_t *node, arena_t *arena);
96 void	extent_node_addr_set(extent_node_t *node, void *addr);
97 void	extent_node_size_set(extent_node_t *node, size_t size);
98 void	extent_node_sn_set(extent_node_t *node, size_t sn);
99 void	extent_node_zeroed_set(extent_node_t *node, bool zeroed);
100 void	extent_node_committed_set(extent_node_t *node, bool committed);
101 void	extent_node_achunk_set(extent_node_t *node, bool achunk);
102 void	extent_node_prof_tctx_set(extent_node_t *node, prof_tctx_t *tctx);
103 void	extent_node_init(extent_node_t *node, arena_t *arena, void *addr,
104     size_t size, size_t sn, bool zeroed, bool committed);
105 void	extent_node_dirty_linkage_init(extent_node_t *node);
106 void	extent_node_dirty_insert(extent_node_t *node,
107     arena_runs_dirty_link_t *runs_dirty, extent_node_t *chunks_dirty);
108 void	extent_node_dirty_remove(extent_node_t *node);
109 #endif
110 
111 #if (defined(JEMALLOC_ENABLE_INLINE) || defined(JEMALLOC_EXTENT_C_))
112 JEMALLOC_INLINE arena_t *
extent_node_arena_get(const extent_node_t * node)113 extent_node_arena_get(const extent_node_t *node)
114 {
115 
116 	return (node->en_arena);
117 }
118 
119 JEMALLOC_INLINE void *
extent_node_addr_get(const extent_node_t * node)120 extent_node_addr_get(const extent_node_t *node)
121 {
122 
123 	return (node->en_addr);
124 }
125 
126 JEMALLOC_INLINE size_t
extent_node_size_get(const extent_node_t * node)127 extent_node_size_get(const extent_node_t *node)
128 {
129 
130 	return (node->en_size);
131 }
132 
133 JEMALLOC_INLINE size_t
extent_node_sn_get(const extent_node_t * node)134 extent_node_sn_get(const extent_node_t *node)
135 {
136 
137 	return (node->en_sn);
138 }
139 
140 JEMALLOC_INLINE bool
extent_node_zeroed_get(const extent_node_t * node)141 extent_node_zeroed_get(const extent_node_t *node)
142 {
143 
144 	return (node->en_zeroed);
145 }
146 
147 JEMALLOC_INLINE bool
extent_node_committed_get(const extent_node_t * node)148 extent_node_committed_get(const extent_node_t *node)
149 {
150 
151 	assert(!node->en_achunk);
152 	return (node->en_committed);
153 }
154 
155 JEMALLOC_INLINE bool
extent_node_achunk_get(const extent_node_t * node)156 extent_node_achunk_get(const extent_node_t *node)
157 {
158 
159 	return (node->en_achunk);
160 }
161 
162 JEMALLOC_INLINE prof_tctx_t *
extent_node_prof_tctx_get(const extent_node_t * node)163 extent_node_prof_tctx_get(const extent_node_t *node)
164 {
165 
166 	return (node->en_prof_tctx);
167 }
168 
169 JEMALLOC_INLINE void
extent_node_arena_set(extent_node_t * node,arena_t * arena)170 extent_node_arena_set(extent_node_t *node, arena_t *arena)
171 {
172 
173 	node->en_arena = arena;
174 }
175 
176 JEMALLOC_INLINE void
extent_node_addr_set(extent_node_t * node,void * addr)177 extent_node_addr_set(extent_node_t *node, void *addr)
178 {
179 
180 	node->en_addr = addr;
181 }
182 
183 JEMALLOC_INLINE void
extent_node_size_set(extent_node_t * node,size_t size)184 extent_node_size_set(extent_node_t *node, size_t size)
185 {
186 
187 	node->en_size = size;
188 }
189 
190 JEMALLOC_INLINE void
extent_node_sn_set(extent_node_t * node,size_t sn)191 extent_node_sn_set(extent_node_t *node, size_t sn)
192 {
193 
194 	node->en_sn = sn;
195 }
196 
197 JEMALLOC_INLINE void
extent_node_zeroed_set(extent_node_t * node,bool zeroed)198 extent_node_zeroed_set(extent_node_t *node, bool zeroed)
199 {
200 
201 	node->en_zeroed = zeroed;
202 }
203 
204 JEMALLOC_INLINE void
extent_node_committed_set(extent_node_t * node,bool committed)205 extent_node_committed_set(extent_node_t *node, bool committed)
206 {
207 
208 	node->en_committed = committed;
209 }
210 
211 JEMALLOC_INLINE void
extent_node_achunk_set(extent_node_t * node,bool achunk)212 extent_node_achunk_set(extent_node_t *node, bool achunk)
213 {
214 
215 	node->en_achunk = achunk;
216 }
217 
218 JEMALLOC_INLINE void
extent_node_prof_tctx_set(extent_node_t * node,prof_tctx_t * tctx)219 extent_node_prof_tctx_set(extent_node_t *node, prof_tctx_t *tctx)
220 {
221 
222 	node->en_prof_tctx = tctx;
223 }
224 
225 JEMALLOC_INLINE void
extent_node_init(extent_node_t * node,arena_t * arena,void * addr,size_t size,size_t sn,bool zeroed,bool committed)226 extent_node_init(extent_node_t *node, arena_t *arena, void *addr, size_t size,
227     size_t sn, bool zeroed, bool committed)
228 {
229 
230 	extent_node_arena_set(node, arena);
231 	extent_node_addr_set(node, addr);
232 	extent_node_size_set(node, size);
233 	extent_node_sn_set(node, sn);
234 	extent_node_zeroed_set(node, zeroed);
235 	extent_node_committed_set(node, committed);
236 	extent_node_achunk_set(node, false);
237 	if (config_prof)
238 		extent_node_prof_tctx_set(node, NULL);
239 }
240 
241 JEMALLOC_INLINE void
extent_node_dirty_linkage_init(extent_node_t * node)242 extent_node_dirty_linkage_init(extent_node_t *node)
243 {
244 
245 	qr_new(&node->rd, rd_link);
246 	qr_new(node, cc_link);
247 }
248 
249 JEMALLOC_INLINE void
extent_node_dirty_insert(extent_node_t * node,arena_runs_dirty_link_t * runs_dirty,extent_node_t * chunks_dirty)250 extent_node_dirty_insert(extent_node_t *node,
251     arena_runs_dirty_link_t *runs_dirty, extent_node_t *chunks_dirty)
252 {
253 
254 	qr_meld(runs_dirty, &node->rd, rd_link);
255 	qr_meld(chunks_dirty, node, cc_link);
256 }
257 
258 JEMALLOC_INLINE void
extent_node_dirty_remove(extent_node_t * node)259 extent_node_dirty_remove(extent_node_t *node)
260 {
261 
262 	qr_remove(&node->rd, rd_link);
263 	qr_remove(node, cc_link);
264 }
265 
266 #endif
267 
268 #endif /* JEMALLOC_H_INLINES */
269 /******************************************************************************/
270 
271