/* GNU Mailutils -- a suite of utilities for electronic mail
Copyright (C) 1999-2021 Free Software Foundation, Inc.
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 3 of the License, or (at your option) any later version.
This library 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
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General
Public License along with this library. If not, see
. */
#ifdef HAVE_CONFIG_H
# include
#endif
#include
#include
#include
#include
void
_mu_list_insert_sublist (mu_list_t list,
struct list_data *current,
struct list_data *head,
struct list_data *tail,
size_t count,
int insert_before)
{
if (insert_before)
{
head->prev = current->prev;
tail->next = current;
if (current->prev != &list->head)
current->prev->next = head;
else
list->head.next = head;
current->prev = tail;
}
else
{
tail->next = current->next;
head->prev = current;
if (current->next != &list->head)
current->next->prev = tail;
else
list->head.prev = tail;
current->next = head;
}
list->count += count;
}
void
_mu_list_clear (mu_list_t list)
{
list->head.next = list->head.prev = &list->head;
list->count = 0;
}
int
mu_list_insert_list (mu_list_t list, void *item, mu_list_t new_list,
int insert_before)
{
struct list_data *current;
mu_list_comparator_t comp;
int status = MU_ERR_NOENT;
if (list == NULL)
return EINVAL;
comp = list->comp ? list->comp : _mu_list_ptr_comparator;
mu_monitor_wrlock (list->monitor);
for (current = list->head.next;
current != &list->head;
current = current->next)
{
if (comp (current->item, item) == 0)
{
_mu_list_insert_sublist (list, current,
new_list->head.next, new_list->head.prev,
new_list->count,
insert_before);
_mu_list_clear (new_list);
status = 0;
break;
}
}
mu_monitor_unlock (list->monitor);
return status;
}
void
mu_list_append_list (mu_list_t list, mu_list_t new_list)
{
if (new_list->count == 0)
return;
else if (list->count == 0)
{
list->head = new_list->head;
list->head.next->prev = list->head.prev->next = &list->head;
list->count = new_list->count;
}
else
_mu_list_insert_sublist (list, list->head.prev,
new_list->head.next, new_list->head.prev,
new_list->count,
0);
_mu_list_clear (new_list);
}
void
mu_list_prepend_list (mu_list_t list, mu_list_t new_list)
{
if (list->count == 0)
{
list->head = new_list->head;
list->head.next->prev = list->head.prev->next = &list->head;
list->count = new_list->count;
}
else
_mu_list_insert_sublist (list, list->head.next,
new_list->head.next, new_list->head.prev,
new_list->count,
1);
_mu_list_clear (new_list);
}