/* -*- Mode: C ; c-basic-offset: 2 -*- */ /***************************************************************************** * * list_sort() adapted from linux kernel. * * 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; version 2 of the License * * 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, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * *****************************************************************************/ #include #include "list.h" /* list sort from Mark J Roberts (mjr@znex.org) */ void __list_sort ( struct list_head *head, int member_offset, int (*cmp)(void * a, void * b)) { struct list_head *p, *q, *e, *list, *tail, *oldhead; int insize, nmerges, psize, qsize, i; list = head->next; list_del (head); insize = 1; for (;; ) { p = oldhead = list; list = tail = NULL; nmerges = 0; while (p) { nmerges++; q = p; psize = 0; for (i = 0; i < insize; i++) { psize++; q = q->next == oldhead ? NULL : q->next; if (!q) { break; } } qsize = insize; while (psize > 0 || (qsize > 0 && q)) { if (!psize) { e = q; q = q->next; qsize--; if (q == oldhead) { q = NULL; } } else if (!qsize || !q) { e = p; p = p->next; psize--; if (p == oldhead) { p = NULL; } } else if (cmp ((void*)p - member_offset, (void*)q - member_offset) <= 0) { e = p; p = p->next; psize--; if (p == oldhead) { p = NULL; } } else { e = q; q = q->next; qsize--; if (q == oldhead) { q = NULL; } } if (tail) { tail->next = e; } else { list = e; } e->prev = tail; tail = e; } p = q; } tail->next = list; list->prev = tail; if (nmerges <= 1) { break; } insize *= 2; } head->next = list; head->prev = list->prev; list->prev->next = head; list->prev = head; } struct test_list_el { int value; struct list_head test_list_node; }; int test_list_sort_comparator (struct test_list_el * e1, struct test_list_el * e2) { return e1->value - e2->value; } void test_list_sort (void) { struct list_head test_list; struct test_list_el *el, *next; struct test_list_el te1 = { .value = 1 }; struct test_list_el te2 = { .value = 2 }; struct test_list_el te3 = { .value = 3 }; struct test_list_el te4 = { .value = 4 }; struct test_list_el te5 = { .value = 5 }; struct test_list_el te6 = { .value = 6 }; struct test_list_el te7 = { .value = 7 }; const int expected[] = { 1, 2, 3, 4, 5, 6, 7 }; int i; INIT_LIST_HEAD (&test_list); list_add_tail (&te2.test_list_node, &test_list); list_add_tail (&te6.test_list_node, &test_list); list_add_tail (&te4.test_list_node, &test_list); list_add_tail (&te5.test_list_node, &test_list); list_add_tail (&te7.test_list_node, &test_list); list_add_tail (&te1.test_list_node, &test_list); list_add_tail (&te3.test_list_node, &test_list); list_sort (&test_list, struct test_list_el, test_list_node, test_list_sort_comparator); i = 0; list_for_each_entry_safe (el, next, &test_list, test_list_node) { assert (el->value == expected[i]); i++; } }