blob: 1a96dc4de2b945b4a8ac0f8f4647de4fd60af1b0 [file] [log] [blame]
/*
* Copyright (c) International Business Machines Corp., 2001-2004
*
* 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, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <stdlib.h>
#include <assert.h>
#include "cirlist.h"
#include "util.h"
void init_cirlist(struct cirlist *cl)
{
cl->count = 0;
cl->head = NULL;
}
int cl_empty(struct cirlist *cl)
{
return !(cl->count);
}
void cl_insert_tail(struct cirlist *cl, cldatatype object)
{
struct cnode *new = ffsb_malloc(sizeof(struct cnode));
new->obj = object;
if (cl->count == 0) {
assert(cl->head == NULL);
cl->head = new;
cl->head->next = cl->head;
cl->head->prev = cl->head;
cl->count = 1;
} else {
if (cl->count == 1) {
assert(cl->head->next == cl->head);
assert(cl->head->prev == cl->head);
cl->head->next = new;
cl->head->prev = new;
new->next = cl->head;
new->prev = cl->head;
} else {
assert(cl->head->next != cl->head);
assert(cl->head->prev != cl->head);
new->next = cl->head;
new->prev = (cl->head)->prev;
cl->head->prev->next = new;
cl->head->prev = new;
}
cl->count++;
}
}
cldatatype cl_remove_head(struct cirlist *cl)
{
struct cnode *oldhead = NULL;
struct cnode *newhead = NULL;
cldatatype ret = NULL;
if (cl->count == 0) {
assert(cl->head == NULL);
return NULL;
}
if (cl->count == 1) {
assert(cl->head->next == cl->head);
assert(cl->head->prev == cl->head);
oldhead = cl->head;
cl->head = NULL;
cl->count = 0;
} else if (cl->count == 2) {
oldhead = cl->head;
newhead = oldhead->next;
newhead->next = newhead;
newhead->prev = newhead;
cl->head = newhead;
cl->count = 1;
} else {
assert(cl->head->next != cl->head);
assert(cl->head->prev != cl->head);
oldhead = cl->head;
newhead = oldhead->next;
newhead->prev = oldhead->prev;
newhead->prev->next = newhead;
cl->head = newhead;
cl->count--;
}
ret = oldhead->obj;
oldhead->obj = (void *)(-1);
oldhead->next = (void *)(-1);
oldhead->prev = (void *)(-1);
free(oldhead);
return ret;
}