tty-solitaire/lib/stack.c

134 lines
2.9 KiB
C
Raw Normal View History

#include <stdio.h>
#include <stdlib.h>
2010-04-04 00:44:59 +00:00
#include <stdbool.h>
#include <malloc.h>
#include <string.h>
#include <errno.h>
2010-04-04 00:44:59 +00:00
#include "stack.h"
void allocate_stack(struct stack **stack) {
if (!(*stack = malloc(sizeof(**stack)))) {
fprintf(stderr, "%s: %s (%s:%d)\n", program_name, strerror(errno), __FILE__, __LINE__ - 1);
exit(errno);
}
2010-04-04 00:44:59 +00:00
allocate_card(&((*stack)->card));
return;
}
void initialize_stack(struct stack *stack) {
initialize_card(stack->card);
stack->next = NULL;
2010-04-04 00:44:59 +00:00
return;
2010-04-04 00:44:59 +00:00
}
2011-05-01 06:06:43 +00:00
struct stack *duplicate_stack(struct stack *stack) {
struct stack *iterator = stack;
struct stack *tmp_stack, *new_stack;
allocate_stack(&new_stack);
allocate_stack(&tmp_stack);
initialize_stack(new_stack);
initialize_stack(tmp_stack);
for (iterator = stack; iterator; iterator = iterator->next) {
push(&tmp_stack, duplicate_card(iterator->card));
}
while (!empty(tmp_stack)) {
push(&new_stack, (pop(&tmp_stack))->card);
}
free_stack(tmp_stack);
return(new_stack);
}
2011-02-14 02:10:47 +00:00
void free_stack(struct stack *stack) {
struct stack *tmp_stack;
2011-02-14 02:05:54 +00:00
while (stack) {
tmp_stack = stack->next;
2011-02-14 02:10:47 +00:00
free_card(stack->card);
2011-02-14 02:05:54 +00:00
free(stack);
stack = tmp_stack;
2011-02-12 03:26:03 +00:00
}
2010-04-07 00:59:21 +00:00
return;
}
2010-04-04 00:44:59 +00:00
bool empty(struct stack *stack) {
return(stack->card->value == NO_VALUE &&
stack->card->suit == NO_SUIT &&
stack->card->face == NO_FACE &&
!stack->next);
2010-04-04 00:44:59 +00:00
}
int length(struct stack *stack) {
int length;
2010-04-04 00:44:59 +00:00
if (!empty(stack)) {
for (length = 1; stack->next; stack = stack->next, length++)
;
} else {
length = 0;
2010-04-04 00:44:59 +00:00
}
return(length);
}
2010-04-04 06:20:56 +00:00
void push(struct stack **stack, struct card *card) {
2010-04-04 00:44:59 +00:00
struct stack *new_stack = NULL;
if (card) {
if (empty(*stack)) {
(*stack)->card = duplicate_card(card);
} else {
allocate_stack(&new_stack);
new_stack->card = duplicate_card(card);
new_stack->next = (*stack);
*stack = new_stack;
}
2010-04-04 00:44:59 +00:00
}
}
2010-04-04 01:12:42 +00:00
2010-04-04 07:15:43 +00:00
struct stack *pop(struct stack **stack) {
2010-04-04 01:12:42 +00:00
struct stack *popped_entry = NULL;
2010-04-04 07:15:43 +00:00
if(!empty(*stack)) {
allocate_stack(&popped_entry);
initialize_stack(popped_entry);
popped_entry->card = duplicate_card((*stack)->card);
popped_entry->next = NULL;
if (length(*stack) == 1) {
/* An empty stack is a stack with a blank top card
* and with stack->next == NULL. */
set_card((*stack)->card,
NO_VALUE,
NO_SUIT,
NO_FACE,
(*stack)->card->frame->start_y,
(*stack)->card->frame->start_x);
(*stack)->next = NULL;
} else {
*stack = (*stack)->next;
}
2010-04-04 01:12:42 +00:00
}
return(popped_entry);
}
2011-05-01 01:13:41 +00:00
2011-05-01 04:13:15 +00:00
struct stack *reverse(struct stack *stack) {
if (length(stack) > 1) {
struct stack *tmp_stack, *iterator;
2011-05-01 01:13:41 +00:00
allocate_stack(&tmp_stack);
initialize_stack(tmp_stack);
for (iterator = stack; iterator; iterator = iterator->next) {
push(&tmp_stack, iterator->card);
2011-05-01 01:13:41 +00:00
}
return(tmp_stack);
} else {
2011-05-01 04:13:15 +00:00
return(stack);
2011-05-01 01:13:41 +00:00
}
}