public inbox for linux-kernel@vger.kernel.org
 help / color / mirror / Atom feed
* [RFC, 2.6] a simple FIFO implementation
@ 2004-09-13 13:52 Stelian Pop
  2004-09-15 10:20 ` Stelian Pop
                   ` (3 more replies)
  0 siblings, 4 replies; 64+ messages in thread
From: Stelian Pop @ 2004-09-13 13:52 UTC (permalink / raw)
  To: Linux Kernel Mailing List

Hi all,

Is there a reason there is no API implementing a simple in-kernel
FIFO ? A linked list is a bit overkill...

Besides my sonypi and meye drivers which could use this, there are
quite a few other drivers which re-implement the circular buffer
over and over again...

An initial implementation follows below. Comments ?

Thanks,

Stelian.

--- /dev/null	2004-02-23 22:02:56.000000000 +0100
+++ linux-2.6/include/linux/kfifo.h	2004-09-13 14:58:37.000000000 +0200
@@ -0,0 +1,179 @@
+/*
+ * A simple kernel FIFO implementation.
+ *
+ * Copyright (C) 2004 Stelian Pop <stelian@popies.net>
+ *
+ * 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., 675 Mass Ave, Cambridge, MA 02139, USA.
+ *
+ */
+#ifndef _LINUX_KFIFO_H
+#define _LINUX_KFIFO_H
+
+#ifdef __KERNEL__
+
+#include <linux/kernel.h>
+#include <linux/slab.h>
+#include <linux/spinlock.h>
+
+/*
+ * This is a simple FIFO implementation using a circular buffer.
+ */
+struct kfifo {
+	unsigned int head;
+	unsigned int tail;
+	unsigned int size;
+	unsigned int len;
+	spinlock_t lock;
+	unsigned char *buffer;
+};
+
+/*
+ * kfifo_alloc - allocates a new FIFO
+ * @size: the size of the internal buffer.
+ *
+ * Note that the allocation uses kmalloc(GFP_KERNEL).
+ */
+static inline struct kfifo *kfifo_alloc(unsigned int size) {
+	struct kfifo *fifo;
+
+	fifo = kmalloc(sizeof(struct kfifo), GFP_KERNEL);
+	if (!fifo)
+		return NULL;
+	
+	fifo->buffer = kmalloc(size, GFP_KERNEL);
+	if (!fifo->buffer) {
+		kfree(fifo);
+		return NULL;
+	}
+
+	fifo->head = fifo->tail = 0;
+	fifo->size = size;
+	fifo->len = 0;
+	fifo->lock = SPIN_LOCK_UNLOCKED;
+
+	return fifo;
+}
+
+/*
+ * kfifo_free - frees the FIFO
+ * @fifo: the fifo to be freed.
+ */
+static inline void kfifo_free(struct kfifo *fifo) {
+	kfree(fifo->buffer);
+	kfree(fifo);
+}
+
+/*
+ * kfifo_reset - removes the entire FIFO contents
+ * @fifo: the fifo to be emptied.
+ */
+static inline void kfifo_reset(struct kfifo *fifo) {
+	unsigned long flags;
+
+	spin_lock_irqsave(&fifo->lock, flags);
+	
+	fifo->head = fifo->tail = 0;
+	fifo->len = 0;
+
+	spin_unlock_irqrestore(&fifo->lock, flags);
+}
+
+/*
+ * kfifo_put - puts some data into the FIFO
+ * @fifo: the fifo to be used.
+ * @buffer: the data to be added.
+ * @len: the length of the data to be added.
+ *
+ * This function copies at most 'len' bytes from the 'buffer' into
+ * the FIFO depending on the free space, and returns the number of
+ * bytes copied. 
+ */
+static inline unsigned int kfifo_put(struct kfifo *fifo, 
+				     unsigned char *buffer, unsigned int len) {
+	unsigned long flags;
+	unsigned int total, remaining;
+	
+	spin_lock_irqsave(&fifo->lock, flags);
+
+	total = remaining = min(len, fifo->size - fifo->len);
+	while (remaining > 0) {
+		unsigned int l = min(remaining, fifo->size - fifo->tail);
+		memcpy(fifo->buffer + fifo->tail, buffer, l);
+		fifo->tail += l;
+		fifo->tail %= fifo->size;
+		fifo->len += l;
+		buffer += l;
+		remaining -= l;
+	}
+
+	spin_unlock_irqrestore(&fifo->lock, flags);
+
+	return total;
+}
+
+/*
+ * kfifo_get - gets some data from the FIFO
+ * @fifo: the fifo to be used.
+ * @buffer: where the data must be copied.
+ * @len: the size of the destination buffer.
+ *
+ * This function copies at most 'len' bytes from the FIFO into the
+ * 'buffer' and returns the number of copied bytes.
+ */
+static inline unsigned int kfifo_get(struct kfifo *fifo, 
+				     unsigned char *buffer, unsigned int len) {
+	unsigned long flags;
+	unsigned int total, remaining;
+
+	spin_lock_irqsave(&fifo->lock, flags);
+
+	total = remaining = min(len, fifo->len);
+	while (remaining > 0) {
+		unsigned int l = min(remaining, fifo->size - fifo->head);
+		memcpy(buffer, fifo->buffer + fifo->head, l);
+		fifo->head += l;
+		fifo->head %= fifo->size;
+		fifo->len -= l;
+		buffer += l;
+		remaining -= l;
+	}
+
+	spin_unlock_irqrestore(&fifo->lock, flags);
+
+        return total;
+}
+
+/*
+ * kfifo_len - returns the number of bytes available in the FIFO
+ * @fifo: the fifo to be used.
+ */
+static inline unsigned int kfifo_len(struct kfifo *fifo) {
+	unsigned long flags;
+	unsigned int result;
+	
+	spin_lock_irqsave(&fifo->lock, flags);
+	
+	result = fifo->len;
+
+	spin_unlock_irqrestore(&fifo->lock, flags);
+
+	return result;
+}
+
+
+#else
+#warning "don't include kernel headers in userspace"
+#endif /* __KERNEL__ */
+#endif
-- 
Stelian Pop <stelian@popies.net>    

^ permalink raw reply	[flat|nested] 64+ messages in thread
[parent not found: <260353727@toto.iv>]

end of thread, other threads:[~2004-09-20 19:01 UTC | newest]

Thread overview: 64+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2004-09-13 13:52 [RFC, 2.6] a simple FIFO implementation Stelian Pop
2004-09-15 10:20 ` Stelian Pop
2004-09-15 11:10   ` Tonnerre
2004-09-15 11:25   ` Alan Cox
2004-09-15 13:27 ` Dmitry Torokhov
2004-09-15 14:21   ` Stelian Pop
2004-09-15 22:30 ` Andrew Morton
2004-09-16  6:43   ` Stelian Pop
2004-09-16  7:04     ` Andrew Morton
2004-09-16 10:45       ` Stelian Pop
2004-09-16 13:57         ` Paul Jackson
2004-09-16 14:09           ` Stelian Pop
2004-09-16 15:45             ` Paul Jackson
2004-09-16 15:59               ` Stelian Pop
2004-09-16 15:09         ` Buddy Lucas
2004-09-16 15:29           ` Stelian Pop
2004-09-16 15:51             ` Buddy Lucas
2004-09-16 15:52               ` Stelian Pop
2004-09-16 16:07                 ` Buddy Lucas
2004-09-16 18:30                   ` Stelian Pop
2004-09-16 22:52                     ` Buddy Lucas
2004-09-16 17:00         ` Andrew Morton
2004-09-16 18:09           ` Stelian Pop
2004-09-17  0:18             ` Andrew Morton
2004-09-17 10:24               ` Stelian Pop
2004-09-17 11:28                 ` Paul Jackson
2004-09-17 11:44                 ` Hugh Dickins
2004-09-17 12:24                   ` Stelian Pop
2004-09-17 12:37                     ` Duncan Sands
2004-09-17 12:48                       ` Stelian Pop
2004-09-17 13:00                         ` Duncan Sands
2004-09-17 13:05                           ` Stelian Pop
2004-09-17 13:16                             ` Duncan Sands
2004-09-17 13:15                     ` Andrea Arcangeli
2004-09-17 13:36                       ` Stelian Pop
2004-09-17 13:41                         ` Andrea Arcangeli
2004-09-17 14:00                           ` Stelian Pop
2004-09-17 15:14                             ` Andrea Arcangeli
2004-09-17 14:47                         ` Paul Jackson
2004-09-17 12:52       ` James R Bruce
2004-09-17 15:48         ` Stelian Pop
2004-09-17 16:01           ` Andrea Arcangeli
2004-09-17 16:14           ` Hugh Dickins
2004-09-17 20:50             ` Stelian Pop
2004-09-17 21:28               ` Andrea Arcangeli
2004-09-17 21:54                 ` Ryan Cumming
2004-09-17 22:00                   ` Andrea Arcangeli
2004-09-17 22:14                     ` Ryan Cumming
2004-09-17 22:35                       ` Andrew Morton
2004-09-18  0:07                         ` Ryan Cumming
2004-09-17 22:29                     ` Ryan Cumming
2004-09-18  3:41                       ` Kyle Moffett
2004-09-18  7:56                         ` Ryan Cumming
2004-09-17 22:07                 ` Andrew Morton
2004-09-20 15:14                 ` Stelian Pop
2004-09-20 18:01                   ` Sasha Khapyorsky
2004-09-20 18:22                     ` Stelian Pop
2004-09-20 19:00                       ` Richard B. Johnson
2004-09-16 15:57 ` Bill Davidsen
2004-09-17 10:25   ` Stelian Pop
     [not found] <260353727@toto.iv>
2004-09-17  1:00 ` Peter Chubb
2004-09-17  5:45   ` Chris Friesen
2004-09-17 11:58     ` Anton Blanchard
2004-09-17 10:32   ` Stelian Pop

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox