summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDmitri Pal <dpal@redhat.com>2009-07-01 18:22:19 -0400
committerStephen Gallagher <sgallagh@redhat.com>2009-07-02 08:16:35 -0400
commit9a28c36d323c14e31a3895ffe7dc905b815291f8 (patch)
tree9f43b63b340092f9e63d7e403a36a22f51643124
parent203d8d4ffb03897ab995bc2a6311f2a603245269 (diff)
downloadsssd-9a28c36d323c14e31a3895ffe7dc905b815291f8.tar.gz
sssd-9a28c36d323c14e31a3895ffe7dc905b815291f8.tar.bz2
sssd-9a28c36d323c14e31a3895ffe7dc905b815291f8.zip
Queue and stack APIs using collection.
Patch adds two interfaces for stack and queue using collection as underlaying implementation. It also fixes a bug in the insterion code - the unsigned properties were created as long.
-rw-r--r--common/collection/Makefile.am10
-rw-r--r--common/collection/collection_cnv.c2
-rw-r--r--common/collection/collection_queue.c350
-rw-r--r--common/collection/collection_queue.h81
-rw-r--r--common/collection/collection_queue_ut.c101
-rw-r--r--common/collection/collection_stack.c350
-rw-r--r--common/collection/collection_stack.h81
-rw-r--r--common/collection/collection_stack_ut.c108
8 files changed, 1081 insertions, 2 deletions
diff --git a/common/collection/Makefile.am b/common/collection/Makefile.am
index 8df9fc6b..fe074904 100644
--- a/common/collection/Makefile.am
+++ b/common/collection/Makefile.am
@@ -15,12 +15,20 @@ libcollection_la_SOURCES = \
collection.c \
collection_tools.c \
collection_cnv.c \
+ collection_queue.c \
+ collection_stack.c \
collection.h \
collection_tools.h \
collection_priv.h \
+ collection_queue.h \
+ collection_stack.h \
../trace/trace.h
# Build unit test
-noinst_PROGRAMS = collection_ut
+noinst_PROGRAMS = collection_ut collection_stack_ut collection_queue_ut
collection_ut_SOURCES = collection_ut.c
collection_ut_LDADD = libcollection.la
+collection_stack_ut_SOURCES = collection_stack_ut.c
+collection_stack_ut_LDADD = libcollection.la
+collection_queue_ut_SOURCES = collection_queue_ut.c
+collection_queue_ut_LDADD = libcollection.la
diff --git a/common/collection/collection_cnv.c b/common/collection/collection_cnv.c
index ca9c483e..1ead75f5 100644
--- a/common/collection/collection_cnv.c
+++ b/common/collection/collection_cnv.c
@@ -151,7 +151,7 @@ int insert_unsigned_property(struct collection_item *ci,
index,
flags,
property,
- COL_TYPE_LONG,
+ COL_TYPE_UNSIGNED,
(void *)&number,
sizeof(unsigned),
NULL);
diff --git a/common/collection/collection_queue.c b/common/collection/collection_queue.c
new file mode 100644
index 00000000..0d5c0ac3
--- /dev/null
+++ b/common/collection/collection_queue.c
@@ -0,0 +1,350 @@
+/*
+ QUEUE
+
+ Implementation of the queue on top of collection library interface.
+
+ Copyright (C) Dmitri Pal <dpal@redhat.com> 2009
+
+ Collection 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.
+
+ Collection 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 Collection Library. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+#include "stdlib.h"
+#include "errno.h"
+#include "collection_queue.h"
+#include "trace.h"
+
+/* Function that creates a queue object */
+int create_queue(struct collection_item **queue)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("create_queue", "Entry point.");
+
+ error = create_collection(queue, COL_NAME_QUEUE, COL_CLASS_QUEUE);
+
+ TRACE_FLOW_STRING("create_queue", "Exit.");
+ return error;
+}
+
+/* Function that destroys a queue object */
+void destroy_queue(struct collection_item *queue)
+{
+ TRACE_FLOW_STRING("destroy_queue", "Entry point.");
+
+ destroy_collection(queue);
+
+ TRACE_FLOW_STRING("destroy_queue", "Exit");
+}
+
+
+/* Put a string property into a queue. */
+int enqueue_str_property(struct collection_item *queue,
+ const char *property, char *string, int length)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("enqueue_str_property", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_str_property(queue, NULL, property, string, length);
+
+ TRACE_FLOW_STRING("enqueue_str_property", "Exit.");
+ return error;
+}
+
+/* Put a binary property into a queue. */
+int enqueue_binary_property(struct collection_item *queue,
+ const char *property, void *binary_data, int length)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("enqueue_binary_property", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_binary_property(queue, NULL, property, binary_data, length);
+
+ TRACE_FLOW_STRING("enqueue_binary_property", "Exit.");
+ return error;
+}
+
+
+/* Put an int property into a queue. */
+int enqueue_int_property(struct collection_item *queue,
+ const char *property, int number)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("enqueue_int_property", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_int_property(queue, NULL, property, number);
+
+ TRACE_FLOW_STRING("enqueue_int_property", "Exit.");
+ return error;
+}
+
+/* Put an unsigned int property into a queue. */
+int enqueue_unsigned_property(struct collection_item *queue,
+ const char *property, unsigned int number)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("enqueue_unsigned_property", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_unsigned_property(queue, NULL, property, number);
+
+ TRACE_FLOW_STRING("enqueue_unsigned_property", "Exit.");
+ return error;
+}
+
+
+/* Put a long property. */
+int enqueue_long_property(struct collection_item *queue,
+ const char *property, long number)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("enqueue_long_property", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_long_property(queue, NULL, property, number);
+
+ TRACE_FLOW_STRING("enqueue_long_property", "Exit.");
+ return error;
+}
+
+/* Put an unsigned long property. */
+int enqueue_ulong_property(struct collection_item *queue,
+ const char *property, unsigned long number)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("enqueue_ulong_property", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_ulong_property(queue, NULL, property, number);
+
+ TRACE_FLOW_STRING("enqueue_ulong_property", "Exit.");
+ return error;
+}
+
+/* Put a double property. */
+int enqueue_double_property(struct collection_item *queue,
+ const char *property, double number)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("enqueue_double_property", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_double_property(queue, NULL, property, number);
+
+ TRACE_FLOW_STRING("enqueue_double_property", "Exit.");
+ return error;
+}
+
+/* Put a bool property. */
+int enqueue_bool_property(struct collection_item *queue,
+ const char *property, unsigned char logical)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("enqueue_bool_property", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_bool_property(queue, NULL, property, logical);
+
+ TRACE_FLOW_STRING("enqueue_bool_property", "Exit.");
+ return error;
+}
+
+/* Put any property */
+int enqueue_any_property(struct collection_item *queue,
+ const char *property,
+ int type,
+ void *data,
+ int length)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("enqueue_any_property", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_any_property(queue, NULL, property, type, data, length);
+
+ TRACE_FLOW_STRING("enqueue_any_property", "Exit.");
+ return error;
+}
+
+/* Enqueue item */
+int enqueue_item(struct collection_item *queue,
+ struct collection_item *item)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("enqueue_item", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = insert_item_into_current(queue,
+ item,
+ COL_DSP_END,
+ NULL,
+ 0,
+ COL_INSERT_NOCHECK);
+
+ TRACE_FLOW_STRING("enqueue_item", "Exit.");
+ return error;
+}
+
+/* Dequeue item */
+int dequeue_item(struct collection_item *queue,
+ struct collection_item **item)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("dequeue_item", "Entry point.");
+
+ /* Check that queue is not empty */
+ if (queue == NULL) {
+ TRACE_ERROR_STRING("queue can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a queue */
+ if (!is_of_class(queue, COL_CLASS_QUEUE)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = extract_item_from_current(queue,
+ COL_DSP_FRONT,
+ NULL,
+ 0,
+ 0,
+ item);
+
+ TRACE_FLOW_STRING("dequeue_item", "Exit.");
+ return error;
+}
diff --git a/common/collection/collection_queue.h b/common/collection/collection_queue.h
new file mode 100644
index 00000000..113fcbfb
--- /dev/null
+++ b/common/collection/collection_queue.h
@@ -0,0 +1,81 @@
+/*
+ QUEUE
+
+ Header file for queue implemented using collection interface.
+
+ Copyright (C) Dmitri Pal <dpal@redhat.com> 2009
+
+ Collection 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.
+
+ Collection 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 Collection Library. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+#ifndef COLLECTION_QUEUE_H
+#define COLLECTION_QUEUE_H
+
+#include "collection.h"
+
+
+#define COL_CLASS_QUEUE 40000
+#define COL_NAME_QUEUE "queue"
+
+/* Function that creates a queue object */
+int create_queue(struct collection_item **queue);
+
+/* Function that destroys a queue object */
+void destroy_queue(struct collection_item *queue);
+
+/* Family of functions that add property to a queue */
+/* Put a string property to queue. */
+int enqueue_str_property(struct collection_item *queue,
+ const char *property, char *string, int length);
+/* Put a binary property to queue. */
+int enqueue_binary_property(struct collection_item *queue,
+ const char *property, void *binary_data, int length);
+/* Put an int property to queue. */
+int enqueue_int_property(struct collection_item *queue,
+ const char *property, int number);
+/* Put an unsigned int property to queue. */
+int enqueue_unsigned_property(struct collection_item *queue,
+ const char *property, unsigned int number);
+/* Put a long property. */
+int enqueue_long_property(struct collection_item *queue,
+ const char *property, long number);
+/* Put an unsigned long property. */
+int enqueue_ulong_property(struct collection_item *queue,
+ const char *property, unsigned long number);
+/* Put a double property. */
+int enqueue_double_property(struct collection_item *queue,
+ const char *property, double number);
+/* Put a bool property. */
+int enqueue_bool_property(struct collection_item *queue,
+ const char *property, unsigned char logical);
+
+/* Put any property */
+int enqueue_any_property(struct collection_item *queue, /* Queue */
+ const char *property, /* Name */
+ int type, /* Data type */
+ void *data, /* Pointer to the data */
+ int length); /* Length of the data. For
+ strings it includes the
+ trailing 0 */
+/* Push item */
+int enqueue_item(struct collection_item *queue,
+ struct collection_item *item);
+
+
+/* Get item from queue */
+int dequeue_item(struct collection_item *queue,
+ struct collection_item **item);
+
+
+#endif
diff --git a/common/collection/collection_queue_ut.c b/common/collection/collection_queue_ut.c
new file mode 100644
index 00000000..b4bd5c28
--- /dev/null
+++ b/common/collection/collection_queue_ut.c
@@ -0,0 +1,101 @@
+/*
+ QUEUE INTERFACE
+
+ Queue unit test.
+
+ Copyright (C) Dmitri Pal <dpal@redhat.com> 2009
+
+ Collection 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.
+
+ Collection 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 Collection Library. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+
+#include <stdio.h>
+#include <string.h>
+#include <errno.h>
+#define TRACE_HOME
+#include "trace.h"
+#include "collection_queue.h"
+#include "collection_tools.h"
+
+
+int queue_test()
+{
+ struct collection_item *queue = NULL;
+ char binary_dump[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8 };
+ struct collection_item *item = NULL;
+ int i;
+ unsigned count;
+ int error = EOK;
+
+ TRACE_FLOW_STRING("queue_test","Entry.");
+
+ printf("\n\nQUEUE TEST!!!.\n\n\n");
+
+ if((error = create_queue(&queue)) ||
+ (error = enqueue_str_property(queue, "item1","value 1" ,0)) ||
+ (error = enqueue_int_property(queue, "item2", -1)) ||
+ (error = enqueue_unsigned_property(queue, "item3", 1)) ||
+ (error = enqueue_long_property(queue, "item4", 100)) ||
+ (error = enqueue_ulong_property(queue, "item5", 1000)) ||
+ (error = enqueue_double_property(queue, "item6", 1.1)) ||
+ (error = enqueue_bool_property(queue, "item7", 1)) ||
+ (error = enqueue_binary_property(queue, "item8", binary_dump, sizeof(binary_dump)))) {
+ printf("Failed to enqueue property. Error %d\n", error);
+ destroy_collection(queue);
+ return error;
+ }
+
+ debug_collection(queue,COL_TRAVERSE_DEFAULT);
+
+ error = get_collection_count(queue, &count);
+ if (error) {
+ printf("Failed to get count. Error %d\n", error);
+ destroy_collection(queue);
+ return error;
+ }
+
+ count--;
+
+ printf("Rotate the queue.\n");
+
+ for (i = 0; i < count; i++) {
+ if ((error = dequeue_item(queue, &item)) ||
+ (error = enqueue_item(queue, item))) {
+ printf("Failed to dequeue or enqueue items. Error %d\n", error);
+ destroy_collection(queue);
+ return error;
+ }
+ debug_collection(queue,COL_TRAVERSE_DEFAULT);
+ }
+
+ destroy_collection(queue);
+ TRACE_FLOW_NUMBER("queue_test. Returning", error);
+
+ printf("\n\nEND OF QUEUE TEST!!!.\n\n\n");
+
+ return error;
+}
+
+/* Main function of the unit test */
+
+int main()
+{
+ int error = EOK;
+
+ printf("Start\n");
+ if ((error = queue_test())) printf("Failed!\n");
+ else printf("Success!\n");
+
+ return error;
+}
diff --git a/common/collection/collection_stack.c b/common/collection/collection_stack.c
new file mode 100644
index 00000000..347d9b6c
--- /dev/null
+++ b/common/collection/collection_stack.c
@@ -0,0 +1,350 @@
+/*
+ STACK
+
+ Implementation of the stack on top of collection library interface.
+
+ Copyright (C) Dmitri Pal <dpal@redhat.com> 2009
+
+ Collection 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.
+
+ Collection 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 Collection Library. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+#include "stdlib.h"
+#include "errno.h"
+#include "collection_stack.h"
+#include "trace.h"
+
+/* Function that creates a stack object */
+int create_stack(struct collection_item **stack)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("create_stack", "Entry point.");
+
+ error = create_collection(stack, COL_NAME_STACK, COL_CLASS_STACK);
+
+ TRACE_FLOW_STRING("create_stack", "Exit.");
+ return error;
+}
+
+/* Function that destroys a stack object */
+void destroy_stack(struct collection_item *stack)
+{
+ TRACE_FLOW_STRING("destroy_stack", "Entry point.");
+
+ destroy_collection(stack);
+
+ TRACE_FLOW_STRING("destroy_stack", "Exit");
+}
+
+
+
+int push_str_property(struct collection_item *stack,
+ const char *property, char *string, int length)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("push_str_property", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_str_property(stack, NULL, property, string, length);
+
+ TRACE_FLOW_STRING("push_str_property", "Exit.");
+ return error;
+}
+
+/* Push a binary property to stack. */
+int push_binary_property(struct collection_item *stack,
+ const char *property, void *binary_data, int length)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("push_binary_property", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_binary_property(stack, NULL, property, binary_data, length)
+
+ TRACE_FLOW_STRING("push_binary_property", "Exit.");
+ return error;
+}
+
+
+/* Push an int property to stack. */
+int push_int_property(struct collection_item *stack,
+ const char *property, int number)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("push_int_property", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_int_property(stack, NULL, property, number);
+
+ TRACE_FLOW_STRING("push_int_property", "Exit.");
+ return error;
+}
+
+/* Push an unsigned int property to stack. */
+int push_unsigned_property(struct collection_item *stack,
+ const char *property, unsigned int number)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("push_unsigned_property", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_unsigned_property(stack, NULL, property, number);
+
+ TRACE_FLOW_STRING("push_unsigned_property", "Exit.");
+ return error;
+}
+
+
+/* Push a long property. */
+int push_long_property(struct collection_item *stack,
+ const char *property, long number)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("push_long_property", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_long_property(stack, NULL, property, number);
+
+ TRACE_FLOW_STRING("push_long_property", "Exit.");
+ return error;
+}
+
+/* Push an unsigned long property. */
+int push_ulong_property(struct collection_item *stack,
+ const char *property, unsigned long number)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("push_ulong_property", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_ulong_property(stack, NULL, property, number);
+
+ TRACE_FLOW_STRING("push_ulong_property", "Exit.");
+ return error;
+}
+
+/* Push a double property. */
+int push_double_property(struct collection_item *stack,
+ const char *property, double number)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("push_double_property", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_double_property(stack, NULL, property, number);
+
+ TRACE_FLOW_STRING("push_double_property", "Exit.");
+ return error;
+}
+
+/* Push a bool property. */
+int push_bool_property(struct collection_item *stack,
+ const char *property, unsigned char logical)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("push_bool_property", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_bool_property(stack, NULL, property, logical);
+
+ TRACE_FLOW_STRING("push_double_property", "Exit.");
+ return error;
+}
+
+/* Push any property */
+int push_any_property(struct collection_item *stack,
+ const char *property,
+ int type,
+ void *data,
+ int length)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("push_bool_property", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = add_any_property(stack, NULL, property, type, data, length);
+
+ TRACE_FLOW_STRING("push_bool_property", "Exit.");
+ return error;
+}
+
+/* Push item */
+int push_item(struct collection_item *stack,
+ struct collection_item *item)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("push_item", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = insert_item_into_current(stack,
+ item,
+ COL_DSP_END,
+ NULL,
+ 0,
+ COL_INSERT_NOCHECK);
+
+ TRACE_FLOW_STRING("push_item", "Exit.");
+ return error;
+}
+
+/* Pop_item */
+int pop_item(struct collection_item *stack,
+ struct collection_item **item)
+{
+ int error = EOK;
+
+ TRACE_FLOW_STRING("pop_item", "Entry point.");
+
+ /* Check that stack is not empty */
+ if (stack == NULL) {
+ TRACE_ERROR_STRING("Stack can't be NULL", "");
+ return EINVAL;
+ }
+
+ /* Make sure it is a stack */
+ if (!is_of_class(stack, COL_CLASS_STACK)) {
+ TRACE_ERROR_STRING("Wrong class", "");
+ return EINVAL;
+ }
+
+ error = extract_item_from_current(stack,
+ COL_DSP_END,
+ NULL,
+ 0,
+ 0,
+ item);
+
+ TRACE_FLOW_STRING("pop_item", "Exit.");
+ return error;
+}
diff --git a/common/collection/collection_stack.h b/common/collection/collection_stack.h
new file mode 100644
index 00000000..447b3df5
--- /dev/null
+++ b/common/collection/collection_stack.h
@@ -0,0 +1,81 @@
+/*
+ STACK
+
+ Header file for stack implemented using collection interface.
+
+ Copyright (C) Dmitri Pal <dpal@redhat.com> 2009
+
+ Collection 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.
+
+ Collection 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 Collection Library. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+#ifndef COLLECTION_STACK_H
+#define COLLECTION_STACK_H
+
+#include <collection.h>
+
+
+#define COL_CLASS_STACK 30000
+#define COL_NAME_STACK "stack"
+
+/* Function that creates a stack object */
+int create_stack(struct collection_item **stack);
+
+/* Function that destroys a stack object */
+void destroy_stack(struct collection_item *stack);
+
+/* Family of functions that push property to stack */
+/* Push a string property to stack. */
+int push_str_property(struct collection_item *stack,
+ const char *property, char *string, int length);
+/* Push a binary property to stack. */
+int push_binary_property(struct collection_item *stack,
+ const char *property, void *binary_data, int length);
+/* Push an int property to stack. */
+int push_int_property(struct collection_item *stack,
+ const char *property, int number);
+/* Push an unsigned int property to stack. */
+int push_unsigned_property(struct collection_item *stack,
+ const char *property, unsigned int number);
+/* Push a long property. */
+int push_long_property(struct collection_item *stack,
+ const char *property, long number);
+/* Push an unsigned long property. */
+int push_ulong_property(struct collection_item *stack,
+ const char *property, unsigned long number);
+/* Push a double property. */
+int push_double_property(struct collection_item *stack,
+ const char *property, double number);
+/* Push a bool property. */
+int push_bool_property(struct collection_item *stack,
+ const char *property, unsigned char logical);
+
+/* Push any property */
+int push_any_property(struct collection_item *stack, /* Stack */
+ const char *property, /* Name */
+ int type, /* Data type */
+ void *data, /* Pointer to the data */
+ int length); /* Length of the data. For
+ strings it includes the
+ trailing 0 */
+/* Push item */
+int push_item(struct collection_item *stack,
+ struct collection_item *item);
+
+
+/* Pop item */
+int pop_item(struct collection_item *stack,
+ struct collection_item **item);
+
+
+#endif
diff --git a/common/collection/collection_stack_ut.c b/common/collection/collection_stack_ut.c
new file mode 100644
index 00000000..667af103
--- /dev/null
+++ b/common/collection/collection_stack_ut.c
@@ -0,0 +1,108 @@
+/*
+ STACK INTERFACE
+
+ Stack unit test.
+
+ Copyright (C) Dmitri Pal <dpal@redhat.com> 2009
+
+ Collection 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.
+
+ Collection 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 Collection Library. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+
+#include <stdio.h>
+#include <string.h>
+#include <errno.h>
+#define TRACE_HOME
+#include "trace.h"
+#include "collection_stack.h"
+#include "collection_tools.h"
+
+
+int stack_test()
+{
+ struct collection_item *stack = NULL;
+ char binary_dump[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8 };
+ struct collection_item *item1 = NULL;
+ struct collection_item *item2 = NULL;
+
+ int error = EOK;
+
+ TRACE_FLOW_STRING("stack_test", "Entry.");
+
+ printf("\n\nSTACK TEST!!!.\n\n\n");
+
+ if ((error = create_stack(&stack)) ||
+ (error = push_str_property(stack, "item1", "value 1", 0)) ||
+ (error = push_int_property(stack, "item2", -1)) ||
+ (error = push_unsigned_property(stack, "item3", 1)) ||
+ (error = push_long_property(stack, "item4", 100)) ||
+ (error = push_ulong_property(stack, "item5", 1000)) ||
+ (error = push_double_property(stack, "item6", 1.1)) ||
+ (error = push_bool_property(stack, "item7", 1)) ||
+ (error = push_binary_property(stack, "item8", binary_dump, sizeof(binary_dump)))) {
+ printf("Failed to push property. Error %d\n", error);
+ destroy_collection(stack);
+ return error;
+ }
+
+ debug_collection(stack, COL_TRAVERSE_DEFAULT);
+
+ printf("Swapping last two items by popping and pushing them back.\n");
+
+ if ((error = pop_item(stack, &item1)) ||
+ (error = pop_item(stack, &item2))) {
+ printf("Failed to pop items. Error %d\n", error);
+ destroy_collection(stack);
+ return error;
+ }
+
+ printf("\nPopped two last items.\n");
+ debug_collection(stack, COL_TRAVERSE_DEFAULT);
+
+ printf("\nLast item.\n");
+ debug_item(item1);
+
+ printf("\nPrevious item.\n");
+ debug_item(item2);
+
+ if ((error = push_item(stack, item1)) ||
+ (error = push_item(stack, item2))) {
+ printf("Failed to pop or push items. Error %d\n", error);
+ destroy_collection(stack);
+ return error;
+ }
+
+ printf("\n\nPushed two items again in reverse order.\n\n");
+
+ debug_collection(stack, COL_TRAVERSE_DEFAULT);
+ destroy_collection(stack);
+ TRACE_FLOW_NUMBER("stack_test. Returning", error);
+
+ printf("\n\nEND OF STACK TEST!!!.\n\n");
+
+ return error;
+}
+
+/* Main function of the unit test */
+
+int main()
+{
+ int error = EOK;
+
+ printf("Start\n");
+ if ((error = stack_test())) printf("Failed!\n");
+ else printf("Success!\n");
+
+ return error;
+}