Some further makefile improvement.
[glib.git] / glist.c
blob04232129aa11ecfb6e8a49c52890262bf696cd79
1 /* GLIB - Library of useful routines for C programming
2 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
21 * Modified by the GLib Team and others 1997-2000. See the AUTHORS
22 * file for a list of people on the GLib Team. See the ChangeLog
23 * files for a list of changes. These files are distributed with
24 * GLib at ftp://ftp.gtk.org/pub/gtk/.
27 /*
28 * MT safe
31 #include "glib.h"
34 struct _GAllocator /* from gmem.c */
36 gchar *name;
37 guint16 n_preallocs;
38 guint is_unused : 1;
39 guint type : 4;
40 GAllocator *last;
41 GMemChunk *mem_chunk;
42 GList *free_lists; /* implementation specific */
45 static GAllocator *current_allocator = NULL;
46 G_LOCK_DEFINE_STATIC (current_allocator);
48 /* HOLDS: current_allocator_lock */
49 static void
50 g_list_validate_allocator (GAllocator *allocator)
52 g_return_if_fail (allocator != NULL);
53 g_return_if_fail (allocator->is_unused == TRUE);
55 if (allocator->type != G_ALLOCATOR_LIST)
57 allocator->type = G_ALLOCATOR_LIST;
58 if (allocator->mem_chunk)
60 g_mem_chunk_destroy (allocator->mem_chunk);
61 allocator->mem_chunk = NULL;
65 if (!allocator->mem_chunk)
67 allocator->mem_chunk = g_mem_chunk_new (allocator->name,
68 sizeof (GList),
69 sizeof (GList) * allocator->n_preallocs,
70 G_ALLOC_ONLY);
71 allocator->free_lists = NULL;
74 allocator->is_unused = FALSE;
77 void
78 g_list_push_allocator(GAllocator *allocator)
80 G_LOCK (current_allocator);
81 g_list_validate_allocator (allocator);
82 allocator->last = current_allocator;
83 current_allocator = allocator;
84 G_UNLOCK (current_allocator);
87 void
88 g_list_pop_allocator (void)
90 G_LOCK (current_allocator);
91 if (current_allocator)
93 GAllocator *allocator;
95 allocator = current_allocator;
96 current_allocator = allocator->last;
97 allocator->last = NULL;
98 allocator->is_unused = TRUE;
100 G_UNLOCK (current_allocator);
103 static inline GList*
104 _g_list_alloc (void)
106 GList *list;
108 G_LOCK (current_allocator);
109 if (!current_allocator)
111 GAllocator *allocator = g_allocator_new ("GLib default GList allocator",
112 128);
113 g_list_validate_allocator (allocator);
114 allocator->last = NULL;
115 current_allocator = allocator;
117 if (!current_allocator->free_lists)
119 list = g_chunk_new (GList, current_allocator->mem_chunk);
120 list->data = NULL;
122 else
124 if (current_allocator->free_lists->data)
126 list = current_allocator->free_lists->data;
127 current_allocator->free_lists->data = list->next;
128 list->data = NULL;
130 else
132 list = current_allocator->free_lists;
133 current_allocator->free_lists = list->next;
136 G_UNLOCK (current_allocator);
137 list->next = NULL;
138 list->prev = NULL;
140 return list;
143 GList*
144 g_list_alloc (void)
146 return _g_list_alloc ();
149 void
150 g_list_free (GList *list)
152 if (list)
154 GList *last_node = list;
156 #ifdef ENABLE_GC_FRIENDLY
157 while (last_node->next)
159 last_node->data = NULL;
160 last_node->prev = NULL;
161 last_node = last_node->next;
163 last_node->data = NULL
164 last_node->prev = NULL
165 #else /* !ENABLE_GC_FRIENDLY */
166 list->data = list->next;
167 #endif /* ENABLE_GC_FRIENDLY */
169 G_LOCK (current_allocator);
170 last_node->next = current_allocator->free_lists;
171 current_allocator->free_lists = list;
172 G_UNLOCK (current_allocator);
176 static inline void
177 _g_list_free_1 (GList *list)
179 if (list)
181 list->data = NULL;
183 #ifdef ENABLE_GC_FRIENDLY
184 list->prev = NULL;
185 #endif /* ENABLE_GC_FRIENDLY */
187 G_LOCK (current_allocator);
188 list->next = current_allocator->free_lists;
189 current_allocator->free_lists = list;
190 G_UNLOCK (current_allocator);
194 void
195 g_list_free_1 (GList *list)
197 _g_list_free_1 (list);
200 GList*
201 g_list_append (GList *list,
202 gpointer data)
204 GList *new_list;
205 GList *last;
207 new_list = _g_list_alloc ();
208 new_list->data = data;
210 if (list)
212 last = g_list_last (list);
213 /* g_assert (last != NULL); */
214 last->next = new_list;
215 new_list->prev = last;
217 return list;
219 else
220 return new_list;
223 GList*
224 g_list_prepend (GList *list,
225 gpointer data)
227 GList *new_list;
229 new_list = _g_list_alloc ();
230 new_list->data = data;
232 if (list)
234 if (list->prev)
236 list->prev->next = new_list;
237 new_list->prev = list->prev;
239 list->prev = new_list;
240 new_list->next = list;
243 return new_list;
246 GList*
247 g_list_insert (GList *list,
248 gpointer data,
249 gint position)
251 GList *new_list;
252 GList *tmp_list;
254 if (position < 0)
255 return g_list_append (list, data);
256 else if (position == 0)
257 return g_list_prepend (list, data);
259 tmp_list = g_list_nth (list, position);
260 if (!tmp_list)
261 return g_list_append (list, data);
263 new_list = _g_list_alloc ();
264 new_list->data = data;
266 if (tmp_list->prev)
268 tmp_list->prev->next = new_list;
269 new_list->prev = tmp_list->prev;
271 new_list->next = tmp_list;
272 tmp_list->prev = new_list;
274 if (tmp_list == list)
275 return new_list;
276 else
277 return list;
280 GList *
281 g_list_concat (GList *list1, GList *list2)
283 GList *tmp_list;
285 if (list2)
287 tmp_list = g_list_last (list1);
288 if (tmp_list)
289 tmp_list->next = list2;
290 else
291 list1 = list2;
292 list2->prev = tmp_list;
295 return list1;
298 GList*
299 g_list_remove (GList *list,
300 gconstpointer data)
302 GList *tmp;
304 tmp = list;
305 while (tmp)
307 if (tmp->data != data)
308 tmp = tmp->next;
309 else
311 if (tmp->prev)
312 tmp->prev->next = tmp->next;
313 if (tmp->next)
314 tmp->next->prev = tmp->prev;
316 if (list == tmp)
317 list = list->next;
319 _g_list_free_1 (tmp);
321 break;
324 return list;
327 static inline GList*
328 _g_list_remove_link (GList *list,
329 GList *link)
331 if (link)
333 if (link->prev)
334 link->prev->next = link->next;
335 if (link->next)
336 link->next->prev = link->prev;
338 if (link == list)
339 list = list->next;
341 link->next = NULL;
342 link->prev = NULL;
345 return list;
348 GList*
349 g_list_remove_link (GList *list,
350 GList *link)
352 return _g_list_remove_link (list, link);
355 GList*
356 g_list_delete_link (GList *list,
357 GList *link)
359 list = _g_list_remove_link (list, link);
360 _g_list_free_1 (link);
362 return list;
365 GList*
366 g_list_copy (GList *list)
368 GList *new_list = NULL;
370 if (list)
372 GList *last;
374 new_list = _g_list_alloc ();
375 new_list->data = list->data;
376 last = new_list;
377 list = list->next;
378 while (list)
380 last->next = _g_list_alloc ();
381 last->next->prev = last;
382 last = last->next;
383 last->data = list->data;
384 list = list->next;
388 return new_list;
391 GList*
392 g_list_reverse (GList *list)
394 GList *last;
396 last = NULL;
397 while (list)
399 last = list;
400 list = last->next;
401 last->next = last->prev;
402 last->prev = list;
405 return last;
408 GList*
409 g_list_nth (GList *list,
410 guint n)
412 while ((n-- > 0) && list)
413 list = list->next;
415 return list;
418 gpointer
419 g_list_nth_data (GList *list,
420 guint n)
422 while ((n-- > 0) && list)
423 list = list->next;
425 return list ? list->data : NULL;
428 GList*
429 g_list_find (GList *list,
430 gconstpointer data)
432 while (list)
434 if (list->data == data)
435 break;
436 list = list->next;
439 return list;
442 GList*
443 g_list_find_custom (GList *list,
444 gconstpointer data,
445 GCompareFunc func)
447 g_return_val_if_fail (func != NULL, list);
449 while (list)
451 if (! func (list->data, data))
452 return list;
453 list = list->next;
456 return NULL;
460 gint
461 g_list_position (GList *list,
462 GList *link)
464 gint i;
466 i = 0;
467 while (list)
469 if (list == link)
470 return i;
471 i++;
472 list = list->next;
475 return -1;
478 gint
479 g_list_index (GList *list,
480 gconstpointer data)
482 gint i;
484 i = 0;
485 while (list)
487 if (list->data == data)
488 return i;
489 i++;
490 list = list->next;
493 return -1;
496 GList*
497 g_list_last (GList *list)
499 if (list)
501 while (list->next)
502 list = list->next;
505 return list;
508 GList*
509 g_list_first (GList *list)
511 if (list)
513 while (list->prev)
514 list = list->prev;
517 return list;
520 guint
521 g_list_length (GList *list)
523 guint length;
525 length = 0;
526 while (list)
528 length++;
529 list = list->next;
532 return length;
535 void
536 g_list_foreach (GList *list,
537 GFunc func,
538 gpointer user_data)
540 while (list)
542 (*func) (list->data, user_data);
543 list = list->next;
548 GList*
549 g_list_insert_sorted (GList *list,
550 gpointer data,
551 GCompareFunc func)
553 GList *tmp_list = list;
554 GList *new_list;
555 gint cmp;
557 g_return_val_if_fail (func != NULL, list);
559 if (!list)
561 new_list = _g_list_alloc ();
562 new_list->data = data;
563 return new_list;
566 cmp = (*func) (data, tmp_list->data);
568 while ((tmp_list->next) && (cmp > 0))
570 tmp_list = tmp_list->next;
571 cmp = (*func) (data, tmp_list->data);
574 new_list = _g_list_alloc ();
575 new_list->data = data;
577 if ((!tmp_list->next) && (cmp > 0))
579 tmp_list->next = new_list;
580 new_list->prev = tmp_list;
581 return list;
584 if (tmp_list->prev)
586 tmp_list->prev->next = new_list;
587 new_list->prev = tmp_list->prev;
589 new_list->next = tmp_list;
590 tmp_list->prev = new_list;
592 if (tmp_list == list)
593 return new_list;
594 else
595 return list;
598 static GList *
599 g_list_sort_merge (GList *l1,
600 GList *l2,
601 GCompareFunc compare_func)
603 GList list, *l, *lprev;
605 l = &list;
606 lprev = NULL;
608 while (l1 && l2)
610 if (compare_func (l1->data, l2->data) < 0)
612 l->next = l1;
613 l = l->next;
614 l->prev = lprev;
615 lprev = l;
616 l1 = l1->next;
618 else
620 l->next = l2;
621 l = l->next;
622 l->prev = lprev;
623 lprev = l;
624 l2 = l2->next;
627 l->next = l1 ? l1 : l2;
628 l->next->prev = l;
630 return list.next;
633 GList*
634 g_list_sort (GList *list,
635 GCompareFunc compare_func)
637 GList *l1, *l2;
639 if (!list)
640 return NULL;
641 if (!list->next)
642 return list;
644 l1 = list;
645 l2 = list->next;
647 while ((l2 = l2->next) != NULL)
649 if ((l2 = l2->next) == NULL)
650 break;
651 l1 = l1->next;
653 l2 = l1->next;
654 l1->next = NULL;
656 return g_list_sort_merge (g_list_sort (list, compare_func),
657 g_list_sort (l2, compare_func),
658 compare_func);
661 GList*
662 g_list_sort2 (GList *list,
663 GCompareFunc compare_func)
665 GSList *runs = NULL;
666 GList *tmp;
668 /* Degenerate case. */
669 if (!list) return NULL;
671 /* Assume: list = [12,2,4,11,2,4,6,1,1,12]. */
672 for (tmp = list; tmp; )
674 GList *tmp2;
675 for (tmp2 = tmp;
676 tmp2->next && compare_func (tmp2->data, tmp2->next->data) <= 0;
677 tmp2 = tmp2->next)
678 /* Nothing */;
679 runs = g_slist_append (runs, tmp);
680 tmp = tmp2->next;
681 tmp2->next = NULL;
683 /* Now: runs = [[12],[2,4,11],[2,4,6],[1,1,12]]. */
685 while (runs->next)
687 /* We have more than one run. Merge pairwise. */
688 GSList *dst, *src, *dstprev = NULL;
689 dst = src = runs;
690 while (src && src->next)
692 dst->data = g_list_sort_merge (src->data,
693 src->next->data,
694 compare_func);
695 dstprev = dst;
696 dst = dst->next;
697 src = src->next->next;
700 /* If number of runs was odd, just keep the last. */
701 if (src)
703 dst->data = src->data;
704 dstprev = dst;
705 dst = dst->next;
708 dstprev->next = NULL;
709 g_slist_free (dst);
712 /* After 1st loop: runs = [[2,4,11,12],[1,1,2,4,6,12]]. */
713 /* After 2nd loop: runs = [[1,1,2,2,4,4,6,11,12,12]]. */
715 list = runs->data;
716 g_slist_free (runs);
717 return list;