forked from cmus/cmus
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmergesort.c
83 lines (77 loc) · 1.9 KB
/
mergesort.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
/*
* Copyright 2008-2013 Various Authors
* Copyright 2004 Timo Hirvonen
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, see <http://www.gnu.org/licenses/>.
*/
#include "mergesort.h"
#include "list.h"
void list_mergesort(struct list_head *head,
int (*compare)(const struct list_head *, const struct list_head *))
{
LIST_HEAD(empty);
struct list_head *unsorted_head, *sorted_head, *p, *q, *tmp;
int psize, qsize, K, count;
if (list_empty(head))
return;
unsorted_head = head;
sorted_head = ∅
K = 1;
while (1) {
p = unsorted_head->next;
count = 0;
do {
q = p;
psize = 0;
while (psize < K) {
if (q == unsorted_head)
break;
psize++;
q = q->next;
}
qsize = K;
while (1) {
struct list_head *e;
if (q == unsorted_head)
qsize = 0;
if (psize == 0 && qsize == 0)
break;
if (!psize || (qsize && compare(p, q) > 0)) {
e = q;
q = q->next;
qsize--;
} else {
e = p;
p = p->next;
psize--;
}
list_del(e);
list_add_tail(e, sorted_head);
}
count++;
p = q;
} while (p != unsorted_head);
if (count == 1) {
head->next = sorted_head->next;
head->prev = sorted_head->prev;
sorted_head->prev->next = head;
sorted_head->next->prev = head;
return;
}
tmp = unsorted_head;
unsorted_head = sorted_head;
sorted_head = tmp;
K *= 2;
}
}