1 /* Copyright 2001, 2002, 2003 by Hans Reiser, licensing governed by
4 /* Fast pool allocation.
6 There are situations when some sub-system normally asks memory allocator
7 for only few objects, but under some circumstances could require much
8 more. Typical and actually motivating example is tree balancing. It needs
9 to keep track of nodes that were involved into it, and it is well-known
10 that in reasonable packed balanced tree most (92.938121%) percent of all
11 balancings end up after working with only few nodes (3.141592 on
12 average). But in rare cases balancing can involve much more nodes
13 (3*tree_height+1 in extremal situation).
15 On the one hand, we don't want to resort to dynamic allocation (slab,
16 malloc(), etc.) to allocate data structures required to keep track of
17 nodes during balancing. On the other hand, we cannot statically allocate
18 required amount of space on the stack, because first: it is useless wastage
19 of precious resource, and second: this amount is unknown in advance (tree
22 Pools, implemented in this file are solution for this problem:
24 - some configurable amount of objects is statically preallocated on the
27 - if this preallocated pool is exhausted and more objects is requested
28 they are allocated dynamically.
30 Pools encapsulate distinction between statically and dynamically allocated
31 objects. Both allocation and recycling look exactly the same.
33 To keep track of dynamically allocated objects, pool adds its own linkage
36 NOTE-NIKITA This linkage also contains some balancing-specific data. This
37 is not perfect. On the other hand, balancing is currently the only client
40 NOTE-NIKITA Another desirable feature is to rewrite all pool manipulation
41 functions in the style of tslist/tshash, i.e., make them unreadable, but
50 #include <linux/types.h>
51 #include <linux/err.h>
53 /* initialize new pool object @h */
54 static void reiser4_init_pool_obj(struct reiser4_pool_header
* h
)
56 INIT_LIST_HEAD(&h
->usage_linkage
);
57 INIT_LIST_HEAD(&h
->level_linkage
);
58 INIT_LIST_HEAD(&h
->extra_linkage
);
61 /* initialize new pool */
62 void reiser4_init_pool(struct reiser4_pool
* pool
/* pool to initialize */ ,
63 size_t obj_size
/* size of objects in @pool */ ,
64 int num_of_objs
/* number of preallocated objects */ ,
65 char *data
/* area for preallocated objects */ )
67 struct reiser4_pool_header
*h
;
70 assert("nikita-955", pool
!= NULL
);
71 assert("nikita-1044", obj_size
> 0);
72 assert("nikita-956", num_of_objs
>= 0);
73 assert("nikita-957", data
!= NULL
);
75 memset(pool
, 0, sizeof *pool
);
76 pool
->obj_size
= obj_size
;
78 INIT_LIST_HEAD(&pool
->free
);
79 INIT_LIST_HEAD(&pool
->used
);
80 INIT_LIST_HEAD(&pool
->extra
);
81 memset(data
, 0, obj_size
* num_of_objs
);
82 for (i
= 0; i
< num_of_objs
; ++i
) {
83 h
= (struct reiser4_pool_header
*) (data
+ i
* obj_size
);
84 reiser4_init_pool_obj(h
);
85 /* add pool header to the end of pool's free list */
86 list_add_tail(&h
->usage_linkage
, &pool
->free
);
90 /* release pool resources
92 Release all resources acquired by this pool, specifically, dynamically
96 void reiser4_done_pool(struct reiser4_pool
* pool UNUSED_ARG
)
100 /* allocate carry object from @pool
102 First, try to get preallocated object. If this fails, resort to dynamic
106 static void *reiser4_pool_alloc(struct reiser4_pool
* pool
)
108 struct reiser4_pool_header
*result
;
110 assert("nikita-959", pool
!= NULL
);
112 if (!list_empty(&pool
->free
)) {
113 struct list_head
*linkage
;
115 linkage
= pool
->free
.next
;
117 INIT_LIST_HEAD(linkage
);
118 result
= list_entry(linkage
, struct reiser4_pool_header
,
120 BUG_ON(!list_empty(&result
->level_linkage
) ||
121 !list_empty(&result
->extra_linkage
));
123 /* pool is empty. Extra allocations don't deserve dedicated
124 slab to be served from, as they are expected to be rare. */
125 result
= kmalloc(pool
->obj_size
, reiser4_ctx_gfp_mask_get());
127 reiser4_init_pool_obj(result
);
128 list_add(&result
->extra_linkage
, &pool
->extra
);
130 return ERR_PTR(RETERR(-ENOMEM
));
131 BUG_ON(!list_empty(&result
->usage_linkage
) ||
132 !list_empty(&result
->level_linkage
));
135 list_add(&result
->usage_linkage
, &pool
->used
);
136 memset(result
+ 1, 0, pool
->obj_size
- sizeof *result
);
140 /* return object back to the pool */
141 void reiser4_pool_free(struct reiser4_pool
* pool
,
142 struct reiser4_pool_header
* h
)
144 assert("nikita-961", h
!= NULL
);
145 assert("nikita-962", pool
!= NULL
);
148 assert("nikita-963", pool
->objs
>= 0);
150 list_del_init(&h
->usage_linkage
);
151 list_del_init(&h
->level_linkage
);
153 if (list_empty(&h
->extra_linkage
))
155 * pool header is not an extra one. Push it onto free list
156 * using usage_linkage
158 list_add(&h
->usage_linkage
, &pool
->free
);
160 /* remove pool header from pool's extra list and kfree it */
161 list_del(&h
->extra_linkage
);
166 /* add new object to the carry level list
168 Carry level is FIFO most of the time, but not always. Complications arise
169 when make_space() function tries to go to the left neighbor and thus adds
170 carry node before existing nodes, and also, when updating delimiting keys
171 after moving data between two nodes, we want left node to be locked before
174 Latter case is confusing at the first glance. Problem is that COP_UPDATE
175 opration that updates delimiting keys is sometimes called with two nodes
176 (when data are moved between two nodes) and sometimes with only one node
177 (when leftmost item is deleted in a node). In any case operation is
178 supplied with at least node whose left delimiting key is to be updated
179 (that is "right" node).
181 @pool - from which to allocate new object;
182 @list - where to add object;
183 @reference - after (or before) which existing object to add
185 struct reiser4_pool_header
*reiser4_add_obj(struct reiser4_pool
* pool
,
186 struct list_head
*list
,
188 struct reiser4_pool_header
* reference
)
190 struct reiser4_pool_header
*result
;
192 assert("nikita-972", pool
!= NULL
);
194 result
= reiser4_pool_alloc(pool
);
198 assert("nikita-973", result
!= NULL
);
202 __list_add(&result
->level_linkage
,
203 reference
->level_linkage
.prev
,
204 &reference
->level_linkage
);
207 __list_add(&result
->level_linkage
,
208 &reference
->level_linkage
,
209 reference
->level_linkage
.next
);
212 list_add_tail(&result
->level_linkage
, list
);
215 list_add(&result
->level_linkage
, list
);
218 wrong_return_value("nikita-927", "order");
225 c-indentation-style: "K&R"