Skip to content

Commit 835cc0c

Browse files
dmullistorvalds
authored andcommitted
lib: more scalable list_sort()
XFS and UBIFS can pass long lists to list_sort(); this alternative implementation scales better, reaching ~3x performance gain when list length exceeds the L2 cache size. Stand-alone program timings were run on a Core 2 duo L1=32KB L2=4MB, gcc-4.4, with flags extracted from an Ubuntu kernel build. Object size is 581 bytes compared to 455 for Mark J. Roberts' code. Worst case for either implementation is a list length just over a power of two, and to roughly the same degree, so here are timing results for a range of 2^N+1 lengths. List elements were 16 bytes each including malloc overhead; initial order was random. time (msec) Tatham-Roberts | generic-Mullis-v2 loop_count length | | ratio 4000000 2 206 294 1.427 2000000 3 176 227 1.289 1000000 5 199 172 0.864 500000 9 235 178 0.757 250000 17 243 182 0.748 125000 33 261 196 0.750 62500 65 277 209 0.754 31250 129 292 219 0.75 15625 257 317 235 0.741 7812 513 340 252 0.741 3906 1025 362 267 0.737 1953 2049 388 283 0.729 ~ L1 size 976 4097 556 323 0.580 488 8193 678 361 0.532 244 16385 773 395 0.510 122 32769 844 418 0.495 61 65537 917 454 0.495 30 131073 1128 543 0.481 15 262145 2355 869 0.369 ~ L2 size 7 524289 5597 1714 0.306 3 1048577 6218 2022 0.325 Mark's code does not actually implement the usual or generic mergesort, but rather a variant from Simon Tatham described here: http://www.chiark.greenend.org.uk/~sgtatham/algorithms/listsort.html Simon's algorithm performs O(log N) passes over the entire input list, doing merges of sublists that double in size on each pass. The generic algorithm instead merges pairs of equal length lists as early as possible, in recursive order. For either algorithm, the elements that extend the list beyond power-of-two length are a special case, handled as nearly as possible as a "rounding-up" to a full POT. Some intuition for the locality of reference implications of merge order may be gotten by watching this animation: http://www.sorting-algorithms.com/merge-sort Simon's algorithm requires only O(1) extra space rather than the generic algorithm's O(log N), but in my non-recursive implementation the actual O(log N) data is merely a vector of ~20 pointers, which I've put on the stack. Long-running list_sort() calls: If the list passed in may be long, or the client's cmp() callback function is slow, the client's cmp() may periodically invoke cond_resched() to voluntarily yield the CPU. All inner loops of list_sort() call back to cmp(). Stability of the sort: distinct elements that compare equal emerge from the sort in the same order as with Mark's code, for simple test cases. A boot-time test is provided to verify this and other correctness requirements. A kernel that uses drm.ko appears to run normally with this change; I have no suitable hardware to similarly test the use by UBIFS. [[email protected]: style tweaks, fix comment, make list_sort_test __init] Signed-off-by: Don Mullis <[email protected]> Cc: Dave Airlie <[email protected]> Cc: Andi Kleen <[email protected]> Cc: Dave Chinner <[email protected]> Cc: Artem Bityutskiy <[email protected]> Signed-off-by: Andrew Morton <[email protected]> Signed-off-by: Linus Torvalds <[email protected]>
1 parent d6a2eed commit 835cc0c

File tree

1 file changed

+183
-69
lines changed

1 file changed

+183
-69
lines changed

lib/list_sort.c

Lines changed: 183 additions & 69 deletions
Original file line numberDiff line numberDiff line change
@@ -4,99 +4,213 @@
44
#include <linux/slab.h>
55
#include <linux/list.h>
66

7+
#define MAX_LIST_LENGTH_BITS 20
8+
9+
/*
10+
* Returns a list organized in an intermediate format suited
11+
* to chaining of merge() calls: null-terminated, no reserved or
12+
* sentinel head node, "prev" links not maintained.
13+
*/
14+
static struct list_head *merge(void *priv,
15+
int (*cmp)(void *priv, struct list_head *a,
16+
struct list_head *b),
17+
struct list_head *a, struct list_head *b)
18+
{
19+
struct list_head head, *tail = &head;
20+
21+
while (a && b) {
22+
/* if equal, take 'a' -- important for sort stability */
23+
if ((*cmp)(priv, a, b) <= 0) {
24+
tail->next = a;
25+
a = a->next;
26+
} else {
27+
tail->next = b;
28+
b = b->next;
29+
}
30+
tail = tail->next;
31+
}
32+
tail->next = a?:b;
33+
return head.next;
34+
}
35+
36+
/*
37+
* Combine final list merge with restoration of standard doubly-linked
38+
* list structure. This approach duplicates code from merge(), but
39+
* runs faster than the tidier alternatives of either a separate final
40+
* prev-link restoration pass, or maintaining the prev links
41+
* throughout.
42+
*/
43+
static void merge_and_restore_back_links(void *priv,
44+
int (*cmp)(void *priv, struct list_head *a,
45+
struct list_head *b),
46+
struct list_head *head,
47+
struct list_head *a, struct list_head *b)
48+
{
49+
struct list_head *tail = head;
50+
51+
while (a && b) {
52+
/* if equal, take 'a' -- important for sort stability */
53+
if ((*cmp)(priv, a, b) <= 0) {
54+
tail->next = a;
55+
a->prev = tail;
56+
a = a->next;
57+
} else {
58+
tail->next = b;
59+
b->prev = tail;
60+
b = b->next;
61+
}
62+
tail = tail->next;
63+
}
64+
tail->next = a ? : b;
65+
66+
do {
67+
/*
68+
* In worst cases this loop may run many iterations.
69+
* Continue callbacks to the client even though no
70+
* element comparison is needed, so the client's cmp()
71+
* routine can invoke cond_resched() periodically.
72+
*/
73+
(*cmp)(priv, tail, tail);
74+
75+
tail->next->prev = tail;
76+
tail = tail->next;
77+
} while (tail->next);
78+
79+
tail->next = head;
80+
head->prev = tail;
81+
}
82+
783
/**
884
* list_sort - sort a list.
985
* @priv: private data, passed to @cmp
1086
* @head: the list to sort
1187
* @cmp: the elements comparison function
1288
*
13-
* This function has been implemented by Mark J Roberts <[email protected]>. It
14-
* implements "merge sort" which has O(nlog(n)) complexity. The list is sorted
15-
* in ascending order.
89+
* This function implements "merge sort" which has O(nlog(n)) complexity.
90+
* The list is sorted in ascending order.
1691
*
1792
* The comparison function @cmp is supposed to return a negative value if @a is
1893
* less than @b, and a positive value if @a is greater than @b. If @a and @b
1994
* are equivalent, then it does not matter what this function returns.
2095
*/
2196
void list_sort(void *priv, struct list_head *head,
22-
int (*cmp)(void *priv, struct list_head *a,
23-
struct list_head *b))
97+
int (*cmp)(void *priv, struct list_head *a,
98+
struct list_head *b))
2499
{
25-
struct list_head *p, *q, *e, *list, *tail, *oldhead;
26-
int insize, nmerges, psize, qsize, i;
100+
struct list_head *part[MAX_LIST_LENGTH_BITS+1]; /* sorted partial lists
101+
-- last slot is a sentinel */
102+
int lev; /* index into part[] */
103+
int max_lev = 0;
104+
struct list_head *list;
27105

28106
if (list_empty(head))
29107
return;
30108

109+
memset(part, 0, sizeof(part));
110+
111+
head->prev->next = NULL;
31112
list = head->next;
32-
list_del(head);
33-
insize = 1;
34-
for (;;) {
35-
p = oldhead = list;
36-
list = tail = NULL;
37-
nmerges = 0;
38-
39-
while (p) {
40-
nmerges++;
41-
q = p;
42-
psize = 0;
43-
for (i = 0; i < insize; i++) {
44-
psize++;
45-
q = q->next == oldhead ? NULL : q->next;
46-
if (!q)
47-
break;
48-
}
49113

50-
qsize = insize;
51-
while (psize > 0 || (qsize > 0 && q)) {
52-
if (!psize) {
53-
e = q;
54-
q = q->next;
55-
qsize--;
56-
if (q == oldhead)
57-
q = NULL;
58-
} else if (!qsize || !q) {
59-
e = p;
60-
p = p->next;
61-
psize--;
62-
if (p == oldhead)
63-
p = NULL;
64-
} else if (cmp(priv, p, q) <= 0) {
65-
e = p;
66-
p = p->next;
67-
psize--;
68-
if (p == oldhead)
69-
p = NULL;
70-
} else {
71-
e = q;
72-
q = q->next;
73-
qsize--;
74-
if (q == oldhead)
75-
q = NULL;
76-
}
77-
if (tail)
78-
tail->next = e;
79-
else
80-
list = e;
81-
e->prev = tail;
82-
tail = e;
114+
while (list) {
115+
struct list_head *cur = list;
116+
list = list->next;
117+
cur->next = NULL;
118+
119+
for (lev = 0; part[lev]; lev++) {
120+
cur = merge(priv, cmp, part[lev], cur);
121+
part[lev] = NULL;
122+
}
123+
if (lev > max_lev) {
124+
if (unlikely(lev >= ARRAY_SIZE(part)-1)) {
125+
printk_once(KERN_DEBUG "list passed to"
126+
" list_sort() too long for"
127+
" efficiency\n");
128+
lev--;
83129
}
84-
p = q;
130+
max_lev = lev;
85131
}
132+
part[lev] = cur;
133+
}
86134

87-
tail->next = list;
88-
list->prev = tail;
135+
for (lev = 0; lev < max_lev; lev++)
136+
if (part[lev])
137+
list = merge(priv, cmp, part[lev], list);
89138

90-
if (nmerges <= 1)
91-
break;
139+
merge_and_restore_back_links(priv, cmp, head, part[max_lev], list);
140+
}
141+
EXPORT_SYMBOL(list_sort);
92142

93-
insize *= 2;
94-
}
143+
#ifdef DEBUG_LIST_SORT
144+
struct debug_el {
145+
struct list_head l_h;
146+
int value;
147+
unsigned serial;
148+
};
95149

96-
head->next = list;
97-
head->prev = list->prev;
98-
list->prev->next = head;
99-
list->prev = head;
150+
static int cmp(void *priv, struct list_head *a, struct list_head *b)
151+
{
152+
return container_of(a, struct debug_el, l_h)->value
153+
- container_of(b, struct debug_el, l_h)->value;
100154
}
101155

102-
EXPORT_SYMBOL(list_sort);
156+
/*
157+
* The pattern of set bits in the list length determines which cases
158+
* are hit in list_sort().
159+
*/
160+
#define LIST_SORT_TEST_LENGTH (512+128+2) /* not including head */
161+
162+
static int __init list_sort_test(void)
163+
{
164+
int i, r = 1, count;
165+
struct list_head *head = kmalloc(sizeof(*head), GFP_KERNEL);
166+
struct list_head *cur;
167+
168+
printk(KERN_WARNING "testing list_sort()\n");
169+
170+
cur = head;
171+
for (i = 0; i < LIST_SORT_TEST_LENGTH; i++) {
172+
struct debug_el *el = kmalloc(sizeof(*el), GFP_KERNEL);
173+
BUG_ON(!el);
174+
/* force some equivalencies */
175+
el->value = (r = (r * 725861) % 6599) % (LIST_SORT_TEST_LENGTH/3);
176+
el->serial = i;
177+
178+
el->l_h.prev = cur;
179+
cur->next = &el->l_h;
180+
cur = cur->next;
181+
}
182+
head->prev = cur;
183+
184+
list_sort(NULL, head, cmp);
185+
186+
count = 1;
187+
for (cur = head->next; cur->next != head; cur = cur->next) {
188+
struct debug_el *el = container_of(cur, struct debug_el, l_h);
189+
int cmp_result = cmp(NULL, cur, cur->next);
190+
if (cur->next->prev != cur) {
191+
printk(KERN_EMERG "list_sort() returned "
192+
"a corrupted list!\n");
193+
return 1;
194+
} else if (cmp_result > 0) {
195+
printk(KERN_EMERG "list_sort() failed to sort!\n");
196+
return 1;
197+
} else if (cmp_result == 0 &&
198+
el->serial >= container_of(cur->next,
199+
struct debug_el, l_h)->serial) {
200+
printk(KERN_EMERG "list_sort() failed to preserve order"
201+
" of equivalent elements!\n");
202+
return 1;
203+
}
204+
kfree(cur->prev);
205+
count++;
206+
}
207+
kfree(cur);
208+
if (count != LIST_SORT_TEST_LENGTH) {
209+
printk(KERN_EMERG "list_sort() returned list of"
210+
"different length!\n");
211+
return 1;
212+
}
213+
return 0;
214+
}
215+
module_init(list_sort_test);
216+
#endif

0 commit comments

Comments
 (0)