2014-06-08 10:51:01 +04:00
|
|
|
/* vim: tabstop=4 shiftwidth=4 noexpandtab
|
2014-06-08 10:58:31 +04:00
|
|
|
* This file is part of ToaruOS and is released under the terms
|
2014-06-08 10:13:29 +04:00
|
|
|
* of the NCSA / University of Illinois License - see LICENSE.md
|
|
|
|
* Copyright (C) 2012-2014 Kevin Lange
|
2014-06-08 10:43:21 +04:00
|
|
|
*
|
2012-01-25 03:56:35 +04:00
|
|
|
* Buffered Pipe
|
2014-06-08 10:43:21 +04:00
|
|
|
*
|
2012-01-25 03:56:35 +04:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <system.h>
|
|
|
|
#include <fs.h>
|
2014-04-06 03:36:17 +04:00
|
|
|
#include <printf.h>
|
2012-01-25 03:56:35 +04:00
|
|
|
#include <pipe.h>
|
2012-12-03 09:43:54 +04:00
|
|
|
#include <logging.h>
|
2012-01-25 03:56:35 +04:00
|
|
|
|
2012-01-25 10:54:59 +04:00
|
|
|
#define DEBUG_PIPES 0
|
|
|
|
|
2012-01-25 03:56:35 +04:00
|
|
|
uint32_t read_pipe(fs_node_t *node, uint32_t offset, uint32_t size, uint8_t *buffer);
|
|
|
|
uint32_t write_pipe(fs_node_t *node, uint32_t offset, uint32_t size, uint8_t *buffer);
|
2013-04-24 11:19:08 +04:00
|
|
|
void open_pipe(fs_node_t *node, unsigned int flags);
|
2012-01-25 03:56:35 +04:00
|
|
|
void close_pipe(fs_node_t *node);
|
|
|
|
|
|
|
|
static inline size_t pipe_unread(pipe_device_t * pipe) {
|
2012-01-25 10:54:59 +04:00
|
|
|
if (pipe->read_ptr == pipe->write_ptr) {
|
|
|
|
return 0;
|
|
|
|
}
|
2012-01-25 03:56:35 +04:00
|
|
|
if (pipe->read_ptr > pipe->write_ptr) {
|
|
|
|
return (pipe->size - pipe->read_ptr) + pipe->write_ptr;
|
|
|
|
} else {
|
|
|
|
return (pipe->write_ptr - pipe->read_ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-23 02:11:19 +04:00
|
|
|
int pipe_size(fs_node_t * node) {
|
2013-03-26 08:48:16 +04:00
|
|
|
pipe_device_t * pipe = (pipe_device_t *)node->device;
|
2012-01-31 10:16:09 +04:00
|
|
|
return pipe_unread(pipe);
|
|
|
|
}
|
|
|
|
|
2012-01-25 03:56:35 +04:00
|
|
|
static inline size_t pipe_available(pipe_device_t * pipe) {
|
2012-01-25 10:54:59 +04:00
|
|
|
if (pipe->read_ptr == pipe->write_ptr) {
|
|
|
|
return pipe->size - 1;
|
|
|
|
}
|
|
|
|
|
2012-01-25 03:56:35 +04:00
|
|
|
if (pipe->read_ptr > pipe->write_ptr) {
|
2012-01-25 10:54:59 +04:00
|
|
|
return pipe->read_ptr - pipe->write_ptr - 1;
|
2012-01-25 03:56:35 +04:00
|
|
|
} else {
|
2012-01-25 10:54:59 +04:00
|
|
|
return (pipe->size - pipe->write_ptr) + pipe->read_ptr - 1;
|
2012-01-25 03:56:35 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-03 09:50:32 +04:00
|
|
|
int pipe_unsize(fs_node_t * node) {
|
|
|
|
pipe_device_t * pipe = (pipe_device_t *)node->device;
|
|
|
|
return pipe_available(pipe);
|
|
|
|
}
|
|
|
|
|
2012-01-25 03:56:35 +04:00
|
|
|
static inline void pipe_increment_read(pipe_device_t * pipe) {
|
|
|
|
pipe->read_ptr++;
|
|
|
|
if (pipe->read_ptr == pipe->size) {
|
|
|
|
pipe->read_ptr = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void pipe_increment_write(pipe_device_t * pipe) {
|
|
|
|
pipe->write_ptr++;
|
|
|
|
if (pipe->write_ptr == pipe->size) {
|
|
|
|
pipe->write_ptr = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-13 08:21:38 +04:00
|
|
|
static inline void pipe_increment_write_by(pipe_device_t * pipe, size_t amount) {
|
|
|
|
pipe->write_ptr = (pipe->write_ptr + amount) % pipe->size;
|
|
|
|
}
|
|
|
|
|
2017-01-06 13:01:22 +03:00
|
|
|
static void pipe_alert_waiters(pipe_device_t * pipe) {
|
2017-01-06 12:30:17 +03:00
|
|
|
if (pipe->alert_waiters) {
|
|
|
|
while (pipe->alert_waiters->head) {
|
|
|
|
node_t * node = list_dequeue(pipe->alert_waiters);
|
|
|
|
process_t * p = node->value;
|
2017-01-06 13:01:22 +03:00
|
|
|
process_alert_node(p, pipe);
|
2017-01-06 12:30:17 +03:00
|
|
|
free(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-25 03:56:35 +04:00
|
|
|
uint32_t read_pipe(fs_node_t *node, uint32_t offset, uint32_t size, uint8_t *buffer) {
|
2013-03-26 08:48:16 +04:00
|
|
|
assert(node->device != 0 && "Attempted to read from a fully-closed pipe.");
|
2012-01-25 03:56:35 +04:00
|
|
|
|
|
|
|
/* Retreive the pipe object associated with this file node */
|
2013-03-26 08:48:16 +04:00
|
|
|
pipe_device_t * pipe = (pipe_device_t *)node->device;
|
2012-01-25 03:56:35 +04:00
|
|
|
|
2012-01-25 10:54:59 +04:00
|
|
|
#if DEBUG_PIPES
|
|
|
|
if (pipe->size > 300) { /* Ignore small pipes (ie, keyboard) */
|
2014-04-06 02:23:17 +04:00
|
|
|
debug_print(INFO, "[debug] Call to read from pipe 0x%x", node->device);
|
|
|
|
debug_print(INFO, " Unread bytes: %d", pipe_unread(pipe));
|
|
|
|
debug_print(INFO, " Total size: %d", pipe->size);
|
|
|
|
debug_print(INFO, " Request size: %d", size);
|
|
|
|
debug_print(INFO, " Write pointer: %d", pipe->write_ptr);
|
|
|
|
debug_print(INFO, " Read pointer: %d", pipe->read_ptr);
|
|
|
|
debug_print(INFO, " Buffer address: 0x%x", pipe->buffer);
|
2012-01-25 10:54:59 +04:00
|
|
|
}
|
2012-01-25 10:19:52 +04:00
|
|
|
#endif
|
|
|
|
|
2013-08-20 06:38:15 +04:00
|
|
|
if (pipe->dead) {
|
|
|
|
debug_print(WARNING, "Pipe is dead?");
|
|
|
|
send_signal(getpid(), SIGPIPE);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-01-25 03:56:35 +04:00
|
|
|
size_t collected = 0;
|
2012-01-25 10:19:52 +04:00
|
|
|
while (collected == 0) {
|
2015-05-21 11:28:14 +03:00
|
|
|
spin_lock(pipe->lock_read);
|
2012-01-25 04:40:25 +04:00
|
|
|
while (pipe_unread(pipe) > 0 && collected < size) {
|
2012-01-25 03:56:35 +04:00
|
|
|
buffer[collected] = pipe->buffer[pipe->read_ptr];
|
|
|
|
pipe_increment_read(pipe);
|
|
|
|
collected++;
|
2012-02-01 05:27:38 +04:00
|
|
|
}
|
2015-05-21 11:28:14 +03:00
|
|
|
spin_unlock(pipe->lock_read);
|
2014-04-13 05:14:35 +04:00
|
|
|
wakeup_queue(pipe->wait_queue_writers);
|
2012-02-01 05:27:38 +04:00
|
|
|
/* Deschedule and switch */
|
|
|
|
if (collected == 0) {
|
2014-04-13 05:14:35 +04:00
|
|
|
sleep_on(pipe->wait_queue_readers);
|
2012-01-25 03:56:35 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-25 10:19:52 +04:00
|
|
|
return collected;
|
2012-01-25 03:56:35 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t write_pipe(fs_node_t *node, uint32_t offset, uint32_t size, uint8_t *buffer) {
|
2013-03-26 08:48:16 +04:00
|
|
|
assert(node->device != 0 && "Attempted to write to a fully-closed pipe.");
|
2012-01-25 03:56:35 +04:00
|
|
|
|
|
|
|
/* Retreive the pipe object associated with this file node */
|
2013-03-26 08:48:16 +04:00
|
|
|
pipe_device_t * pipe = (pipe_device_t *)node->device;
|
2012-01-25 03:56:35 +04:00
|
|
|
|
2012-01-25 10:54:59 +04:00
|
|
|
#if DEBUG_PIPES
|
|
|
|
if (pipe->size > 300) { /* Ignore small pipes (ie, keyboard) */
|
2014-04-06 02:23:17 +04:00
|
|
|
debug_print(INFO, "[debug] Call to write to pipe 0x%x", node->device);
|
|
|
|
debug_print(INFO, " Available space: %d", pipe_available(pipe));
|
|
|
|
debug_print(INFO, " Total size: %d", pipe->size);
|
|
|
|
debug_print(INFO, " Request size: %d", size);
|
|
|
|
debug_print(INFO, " Write pointer: %d", pipe->write_ptr);
|
|
|
|
debug_print(INFO, " Read pointer: %d", pipe->read_ptr);
|
|
|
|
debug_print(INFO, " Buffer address: 0x%x", pipe->buffer);
|
|
|
|
debug_print(INFO, " Write: %s", buffer);
|
2012-01-25 10:54:59 +04:00
|
|
|
}
|
2012-01-25 04:40:25 +04:00
|
|
|
#endif
|
|
|
|
|
2013-08-20 06:38:15 +04:00
|
|
|
if (pipe->dead) {
|
|
|
|
debug_print(WARNING, "Pipe is dead?");
|
|
|
|
send_signal(getpid(), SIGPIPE);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-01-25 03:56:35 +04:00
|
|
|
size_t written = 0;
|
|
|
|
while (written < size) {
|
2015-05-21 11:28:14 +03:00
|
|
|
spin_lock(pipe->lock_write);
|
2012-04-13 08:21:38 +04:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
size_t available = 0;
|
|
|
|
if (pipe->read_ptr <= pipe->write_ptr) {
|
|
|
|
available = pipe->size - pipe->write_ptr;
|
|
|
|
} else {
|
|
|
|
available = pipe->read_ptr - pipe->write_ptr - 1;
|
|
|
|
}
|
|
|
|
if (available) {
|
|
|
|
available = min(available, size - written);
|
|
|
|
memcpy(&pipe->buffer[pipe->write_ptr], buffer, available);
|
|
|
|
pipe_increment_write_by(pipe, available);
|
|
|
|
written += available;
|
|
|
|
}
|
|
|
|
#else
|
2012-01-25 04:40:25 +04:00
|
|
|
while (pipe_available(pipe) > 0 && written < size) {
|
2012-01-25 03:56:35 +04:00
|
|
|
pipe->buffer[pipe->write_ptr] = buffer[written];
|
|
|
|
pipe_increment_write(pipe);
|
|
|
|
written++;
|
2012-02-01 05:27:38 +04:00
|
|
|
}
|
2012-04-13 08:21:38 +04:00
|
|
|
#endif
|
|
|
|
|
2015-05-21 11:28:14 +03:00
|
|
|
spin_unlock(pipe->lock_write);
|
2014-04-13 05:14:35 +04:00
|
|
|
wakeup_queue(pipe->wait_queue_readers);
|
2017-01-06 13:01:22 +03:00
|
|
|
pipe_alert_waiters(pipe);
|
2012-02-01 05:27:38 +04:00
|
|
|
if (written < size) {
|
2014-04-13 05:14:35 +04:00
|
|
|
sleep_on(pipe->wait_queue_writers);
|
2012-01-25 03:56:35 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-25 10:19:52 +04:00
|
|
|
return written;
|
2012-01-25 03:56:35 +04:00
|
|
|
}
|
|
|
|
|
2013-04-24 11:19:08 +04:00
|
|
|
void open_pipe(fs_node_t * node, unsigned int flags) {
|
2013-03-26 08:48:16 +04:00
|
|
|
assert(node->device != 0 && "Attempted to open a fully-closed pipe.");
|
2012-01-25 03:56:35 +04:00
|
|
|
|
|
|
|
/* Retreive the pipe object associated with this file node */
|
2013-03-26 08:48:16 +04:00
|
|
|
pipe_device_t * pipe = (pipe_device_t *)node->device;
|
2012-01-25 03:56:35 +04:00
|
|
|
|
|
|
|
/* Add a reference */
|
|
|
|
pipe->refcount++;
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void close_pipe(fs_node_t * node) {
|
2013-03-26 08:48:16 +04:00
|
|
|
assert(node->device != 0 && "Attempted to close an already fully-closed pipe.");
|
2012-01-25 03:56:35 +04:00
|
|
|
|
|
|
|
/* Retreive the pipe object associated with this file node */
|
2013-03-26 08:48:16 +04:00
|
|
|
pipe_device_t * pipe = (pipe_device_t *)node->device;
|
2012-01-25 03:56:35 +04:00
|
|
|
|
|
|
|
/* Drop one reference */
|
|
|
|
pipe->refcount--;
|
|
|
|
|
|
|
|
/* Check the reference count number */
|
|
|
|
if (pipe->refcount == 0) {
|
2012-03-28 01:52:46 +04:00
|
|
|
#if 0
|
2012-01-25 03:56:35 +04:00
|
|
|
/* No other references exist, free the pipe (but not its buffer) */
|
2012-01-25 04:40:25 +04:00
|
|
|
free(pipe->buffer);
|
2012-02-01 05:27:38 +04:00
|
|
|
list_free(pipe->wait_queue);
|
|
|
|
free(pipe->wait_queue);
|
2012-01-25 03:56:35 +04:00
|
|
|
free(pipe);
|
|
|
|
/* And let the creator know there are no more references */
|
2013-03-26 08:48:16 +04:00
|
|
|
node->device = 0;
|
2012-03-28 01:52:46 +04:00
|
|
|
#endif
|
2012-01-25 03:56:35 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-06 12:30:17 +03:00
|
|
|
static int pipe_check(fs_node_t * node) {
|
|
|
|
pipe_device_t * pipe = (pipe_device_t *)node->device;
|
|
|
|
|
|
|
|
if (pipe_unread(pipe) > 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int pipe_wait(fs_node_t * node, void * process) {
|
|
|
|
pipe_device_t * pipe = (pipe_device_t *)node->device;
|
|
|
|
|
|
|
|
if (!pipe->alert_waiters) {
|
|
|
|
pipe->alert_waiters = list_create();
|
|
|
|
}
|
|
|
|
|
|
|
|
list_insert(pipe->alert_waiters, process);
|
|
|
|
list_insert(((process_t *)process)->node_waits, pipe);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-01-25 04:40:25 +04:00
|
|
|
fs_node_t * make_pipe(size_t size) {
|
2012-01-25 03:56:35 +04:00
|
|
|
fs_node_t * fnode = malloc(sizeof(fs_node_t));
|
2012-11-29 11:05:19 +04:00
|
|
|
pipe_device_t * pipe = malloc(sizeof(pipe_device_t));
|
2014-05-28 08:38:30 +04:00
|
|
|
memset(fnode, 0, sizeof(fs_node_t));
|
2017-01-06 12:30:17 +03:00
|
|
|
memset(pipe, 0, sizeof(pipe_device_t));
|
2012-01-25 03:56:35 +04:00
|
|
|
|
2013-03-26 08:48:16 +04:00
|
|
|
fnode->device = 0;
|
2012-01-25 03:56:35 +04:00
|
|
|
fnode->name[0] = '\0';
|
2012-02-13 04:47:01 +04:00
|
|
|
sprintf(fnode->name, "[pipe]");
|
2012-01-25 03:56:35 +04:00
|
|
|
fnode->uid = 0;
|
|
|
|
fnode->gid = 0;
|
2016-12-14 15:21:32 +03:00
|
|
|
fnode->mask = 0666;
|
2012-01-27 13:12:29 +04:00
|
|
|
fnode->flags = FS_PIPE;
|
2012-01-25 03:56:35 +04:00
|
|
|
fnode->read = read_pipe;
|
|
|
|
fnode->write = write_pipe;
|
|
|
|
fnode->open = open_pipe;
|
|
|
|
fnode->close = close_pipe;
|
|
|
|
fnode->readdir = NULL;
|
|
|
|
fnode->finddir = NULL;
|
2013-03-19 10:57:40 +04:00
|
|
|
fnode->ioctl = NULL; /* TODO ioctls for pipes? maybe */
|
2013-03-22 22:58:22 +04:00
|
|
|
fnode->get_size = pipe_size;
|
2012-01-25 03:56:35 +04:00
|
|
|
|
2017-01-06 12:30:17 +03:00
|
|
|
fnode->selectcheck = pipe_check;
|
|
|
|
fnode->selectwait = pipe_wait;
|
|
|
|
|
2012-12-10 04:59:55 +04:00
|
|
|
fnode->atime = now();
|
|
|
|
fnode->mtime = fnode->atime;
|
|
|
|
fnode->ctime = fnode->atime;
|
|
|
|
|
2013-03-26 08:48:16 +04:00
|
|
|
fnode->device = pipe;
|
2012-01-25 03:56:35 +04:00
|
|
|
|
2012-01-25 04:40:25 +04:00
|
|
|
pipe->buffer = malloc(size);
|
2012-01-25 03:56:35 +04:00
|
|
|
pipe->write_ptr = 0;
|
|
|
|
pipe->read_ptr = 0;
|
|
|
|
pipe->size = size;
|
|
|
|
pipe->refcount = 0;
|
2013-08-20 06:38:15 +04:00
|
|
|
pipe->dead = 0;
|
2012-01-25 03:56:35 +04:00
|
|
|
|
2015-05-21 11:28:14 +03:00
|
|
|
spin_init(pipe->lock_read);
|
|
|
|
spin_init(pipe->lock_write);
|
|
|
|
|
2014-04-13 05:14:35 +04:00
|
|
|
pipe->wait_queue_writers = list_create();
|
|
|
|
pipe->wait_queue_readers = list_create();
|
2012-02-01 05:27:38 +04:00
|
|
|
|
2012-01-25 03:56:35 +04:00
|
|
|
return fnode;
|
|
|
|
}
|