/* * arch/v850/kernel/memcons.c -- Console I/O to a memory buffer * * Copyright (C) 2001,02 NEC Corporation * Copyright (C) 2001,02 Miles Bader <miles@gnu.org> * * This file is subject to the terms and conditions of the GNU General * Public License. See the file COPYING in the main directory of this * archive for more details. * * Written by Miles Bader <miles@gnu.org> */ #include <linux/kernel.h> #include <linux/console.h> #include <linux/tty.h> #include <linux/tty_driver.h> #include <linux/init.h> /* If this device is enabled, the linker map should define start and end points for its buffer. */ extern char memcons_output[], memcons_output_end; /* Current offset into the buffer. */ static unsigned long memcons_offs = 0; /* Spinlock protecting memcons_offs. */ static DEFINE_SPINLOCK(memcons_lock); static size_t write (const char *buf, size_t len) { unsigned long flags; char *point; spin_lock_irqsave (memcons_lock, flags); point = memcons_output + memcons_offs; if (point + len >= &memcons_output_end) { len = &memcons_output_end - point; memcons_offs = 0; } else memcons_offs += len; spin_unlock_irqrestore (memcons_lock, flags); memcpy (point, buf, len); return len; } /* Low-level console. */ static void memcons_write (struct console *co, const char *buf, unsigned len) { while (len > 0) len -= write (buf, len); } static struct tty_driver *tty_driver; static struct tty_driver *memcons_device (struct console *co, int *index) { *index = co->index; return tty_driver; } static struct console memcons = { .name = "memcons", .write = memcons_write, .device = memcons_device, .flags = CON_PRINTBUFFER, .index = -1, }; void memcons_setup (void) { register_console (&memcons); printk (KERN_INFO "Console: static memory buffer (memcons)\n"); } /* Higher level TTY interface. */ int memcons_tty_open (struct tty_struct *tty, struct file *filp) { return 0; } int memcons_tty_write (struct tty_struct *tty, const unsigned char *buf, int len) { return write (buf, len); } int memcons_tty_write_room (struct tty_struct *tty) { return &memcons_output_end - (memcons_output + memcons_offs); } int memcons_tty_chars_in_buffer (struct tty_struct *tty) { /* We have no buffer. */ return 0; } static const struct tty_operations ops = { .open = memcons_tty_open, .write = memcons_tty_write, .write_room = memcons_tty_write_room, .chars_in_buffer = memcons_tty_chars_in_buffer, }; int __init memcons_tty_init (void) { int err; struct tty_driver *driver = alloc_tty_driver(1); if (!driver) return -ENOMEM; driver->name = "memcons"; driver->major = TTY_MAJOR; driver->minor_start = 64; driver->type = TTY_DRIVER_TYPE_SYSCONS; driver->init_termios = tty_std_termios; tty_set_operations(driver, &ops); err = tty_register_driver(driver); if (err) { put_tty_driver(driver); return err; } tty_driver = driver; return 0; } __initcall (memcons_tty_init);