327 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			327 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* drivers/char/dcc_tty.c
 | 
						|
 *
 | 
						|
 * Copyright (C) 2007 Google, Inc.
 | 
						|
 *
 | 
						|
 * This software is licensed under the terms of the GNU General Public
 | 
						|
 * License version 2, as published by the Free Software Foundation, and
 | 
						|
 * may be copied, distributed, and modified under those terms.
 | 
						|
 *
 | 
						|
 * 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.
 | 
						|
 *
 | 
						|
 */
 | 
						|
 | 
						|
#include <linux/module.h>
 | 
						|
#include <linux/platform_device.h>
 | 
						|
#include <linux/delay.h>
 | 
						|
#include <linux/console.h>
 | 
						|
#include <linux/hrtimer.h>
 | 
						|
#include <linux/tty.h>
 | 
						|
#include <linux/tty_driver.h>
 | 
						|
#include <linux/tty_flip.h>
 | 
						|
 | 
						|
MODULE_DESCRIPTION("DCC TTY Driver");
 | 
						|
MODULE_LICENSE("GPL");
 | 
						|
MODULE_VERSION("1.0");
 | 
						|
 | 
						|
static spinlock_t g_dcc_tty_lock = SPIN_LOCK_UNLOCKED;
 | 
						|
static struct hrtimer g_dcc_timer;
 | 
						|
static char g_dcc_buffer[16];
 | 
						|
static int g_dcc_buffer_head;
 | 
						|
static int g_dcc_buffer_count;
 | 
						|
static unsigned g_dcc_write_delay_usecs = 1;
 | 
						|
static struct tty_driver *g_dcc_tty_driver;
 | 
						|
static struct tty_struct *g_dcc_tty;
 | 
						|
static int g_dcc_tty_open_count;
 | 
						|
 | 
						|
static void dcc_poll_locked(void)
 | 
						|
{
 | 
						|
	char ch;
 | 
						|
	int rch;
 | 
						|
	int written;
 | 
						|
 | 
						|
	while (g_dcc_buffer_count) {
 | 
						|
		ch = g_dcc_buffer[g_dcc_buffer_head];
 | 
						|
		asm(
 | 
						|
			"mrc 14, 0, r15, c0, c1, 0\n"
 | 
						|
			"mcrcc 14, 0, %1, c0, c5, 0\n"
 | 
						|
			"movcc %0, #1\n"
 | 
						|
			"movcs %0, #0\n"
 | 
						|
			: "=r" (written)
 | 
						|
			: "r" (ch)
 | 
						|
		);
 | 
						|
		if (written) {
 | 
						|
			if (ch == '\n')
 | 
						|
				g_dcc_buffer[g_dcc_buffer_head] = '\r';
 | 
						|
			else {
 | 
						|
				g_dcc_buffer_head = (g_dcc_buffer_head + 1) % ARRAY_SIZE(g_dcc_buffer);
 | 
						|
				g_dcc_buffer_count--;
 | 
						|
				if (g_dcc_tty)
 | 
						|
					tty_wakeup(g_dcc_tty);
 | 
						|
			}
 | 
						|
			g_dcc_write_delay_usecs = 1;
 | 
						|
		} else {
 | 
						|
			if (g_dcc_write_delay_usecs > 0x100)
 | 
						|
				break;
 | 
						|
			g_dcc_write_delay_usecs <<= 1;
 | 
						|
			udelay(g_dcc_write_delay_usecs);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (g_dcc_tty && !test_bit(TTY_THROTTLED, &g_dcc_tty->flags)) {
 | 
						|
		asm(
 | 
						|
			"mrc 14, 0, %0, c0, c1, 0\n"
 | 
						|
			"tst %0, #(1 << 30)\n"
 | 
						|
			"moveq %0, #-1\n"
 | 
						|
			"mrcne 14, 0, %0, c0, c5, 0\n"
 | 
						|
			: "=r" (rch)
 | 
						|
		);
 | 
						|
		if (rch >= 0) {
 | 
						|
			ch = rch;
 | 
						|
			tty_insert_flip_string(g_dcc_tty, &ch, 1);
 | 
						|
			tty_flip_buffer_push(g_dcc_tty);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	if (g_dcc_buffer_count)
 | 
						|
		hrtimer_start(&g_dcc_timer, ktime_set(0, g_dcc_write_delay_usecs * NSEC_PER_USEC), HRTIMER_MODE_REL);
 | 
						|
	else
 | 
						|
		hrtimer_start(&g_dcc_timer, ktime_set(0, 20 * NSEC_PER_MSEC), HRTIMER_MODE_REL);
 | 
						|
}
 | 
						|
 | 
						|
static int dcc_tty_open(struct tty_struct * tty, struct file * filp)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	unsigned long irq_flags;
 | 
						|
 | 
						|
	spin_lock_irqsave(&g_dcc_tty_lock, irq_flags);
 | 
						|
	if (g_dcc_tty == NULL || g_dcc_tty == tty) {
 | 
						|
		g_dcc_tty = tty;
 | 
						|
		g_dcc_tty_open_count++;
 | 
						|
		ret = 0;
 | 
						|
	} else
 | 
						|
		ret = -EBUSY;
 | 
						|
	spin_unlock_irqrestore(&g_dcc_tty_lock, irq_flags);
 | 
						|
 | 
						|
	printk("dcc_tty_open, tty %p, f_flags %x, returned %d\n", tty, filp->f_flags, ret);
 | 
						|
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static void dcc_tty_close(struct tty_struct * tty, struct file * filp)
 | 
						|
{
 | 
						|
	printk("dcc_tty_close, tty %p, f_flags %x\n", tty, filp->f_flags);
 | 
						|
	if (g_dcc_tty == tty) {
 | 
						|
		if (--g_dcc_tty_open_count == 0)
 | 
						|
			g_dcc_tty = NULL;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static int dcc_write(const unsigned char *buf_start, int count)
 | 
						|
{
 | 
						|
	const unsigned char *buf = buf_start;
 | 
						|
	unsigned long irq_flags;
 | 
						|
	int copy_len;
 | 
						|
	int space_left;
 | 
						|
	int tail;
 | 
						|
 | 
						|
	if (count < 1)
 | 
						|
		return 0;
 | 
						|
 | 
						|
	spin_lock_irqsave(&g_dcc_tty_lock, irq_flags);
 | 
						|
	do {
 | 
						|
		tail = (g_dcc_buffer_head + g_dcc_buffer_count) % ARRAY_SIZE(g_dcc_buffer);
 | 
						|
		copy_len = ARRAY_SIZE(g_dcc_buffer) - tail;
 | 
						|
		space_left = ARRAY_SIZE(g_dcc_buffer) - g_dcc_buffer_count;
 | 
						|
		if (copy_len > space_left)
 | 
						|
			copy_len = space_left;
 | 
						|
		if (copy_len > count)
 | 
						|
			copy_len = count;
 | 
						|
		memcpy(&g_dcc_buffer[tail], buf, copy_len);
 | 
						|
		g_dcc_buffer_count += copy_len;
 | 
						|
		buf += copy_len;
 | 
						|
		count -= copy_len;
 | 
						|
		if (copy_len < count && copy_len < space_left) {
 | 
						|
			space_left -= copy_len;
 | 
						|
			copy_len = count;
 | 
						|
			if (copy_len > space_left) {
 | 
						|
				copy_len = space_left;
 | 
						|
			}
 | 
						|
			memcpy(g_dcc_buffer, buf, copy_len);
 | 
						|
			buf += copy_len;
 | 
						|
			count -= copy_len;
 | 
						|
			g_dcc_buffer_count += copy_len;
 | 
						|
		}
 | 
						|
		dcc_poll_locked();
 | 
						|
		space_left = ARRAY_SIZE(g_dcc_buffer) - g_dcc_buffer_count;
 | 
						|
	} while(count && space_left);
 | 
						|
	spin_unlock_irqrestore(&g_dcc_tty_lock, irq_flags);
 | 
						|
	return buf - buf_start;
 | 
						|
}
 | 
						|
 | 
						|
static int dcc_tty_write(struct tty_struct * tty, const unsigned char *buf, int count)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	/* printk("dcc_tty_write %p, %d\n", buf, count); */
 | 
						|
	ret = dcc_write(buf, count);
 | 
						|
	if (ret != count)
 | 
						|
		printk("dcc_tty_write %p, %d, returned %d\n", buf, count, ret);
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static int dcc_tty_write_room(struct tty_struct *tty)
 | 
						|
{
 | 
						|
	int space_left;
 | 
						|
	unsigned long irq_flags;
 | 
						|
 | 
						|
	spin_lock_irqsave(&g_dcc_tty_lock, irq_flags);
 | 
						|
	space_left = ARRAY_SIZE(g_dcc_buffer) - g_dcc_buffer_count;
 | 
						|
	spin_unlock_irqrestore(&g_dcc_tty_lock, irq_flags);
 | 
						|
	return space_left;
 | 
						|
}
 | 
						|
 | 
						|
static int dcc_tty_chars_in_buffer(struct tty_struct *tty)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	asm(
 | 
						|
		"mrc 14, 0, %0, c0, c1, 0\n"
 | 
						|
		"mov %0, %0, LSR #30\n"
 | 
						|
		"and %0, %0, #1\n"
 | 
						|
		: "=r" (ret)
 | 
						|
	);
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static void dcc_tty_unthrottle(struct tty_struct * tty)
 | 
						|
{
 | 
						|
	unsigned long irq_flags;
 | 
						|
 | 
						|
	spin_lock_irqsave(&g_dcc_tty_lock, irq_flags);
 | 
						|
	dcc_poll_locked();
 | 
						|
	spin_unlock_irqrestore(&g_dcc_tty_lock, irq_flags);
 | 
						|
}
 | 
						|
 | 
						|
static enum hrtimer_restart dcc_tty_timer_func(struct hrtimer *timer)
 | 
						|
{
 | 
						|
	unsigned long irq_flags;
 | 
						|
 | 
						|
	spin_lock_irqsave(&g_dcc_tty_lock, irq_flags);
 | 
						|
	dcc_poll_locked();
 | 
						|
	spin_unlock_irqrestore(&g_dcc_tty_lock, irq_flags);
 | 
						|
	return HRTIMER_NORESTART;
 | 
						|
}
 | 
						|
 | 
						|
void dcc_console_write(struct console *co, const char *b, unsigned count)
 | 
						|
{
 | 
						|
#if 1
 | 
						|
	dcc_write(b, count);
 | 
						|
#else
 | 
						|
	/* blocking printk */
 | 
						|
	while (count > 0) {
 | 
						|
		int written;
 | 
						|
		written = dcc_write(b, count);
 | 
						|
		if (written) {
 | 
						|
			b += written;
 | 
						|
			count -= written;
 | 
						|
		}
 | 
						|
	}
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
static struct tty_driver *dcc_console_device(struct console *c, int *index)
 | 
						|
{
 | 
						|
	*index = 0;
 | 
						|
	return g_dcc_tty_driver;
 | 
						|
}
 | 
						|
 | 
						|
static int __init dcc_console_setup(struct console *co, char *options)
 | 
						|
{
 | 
						|
	if (co->index != 0)
 | 
						|
		return -ENODEV;
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static struct console dcc_console =
 | 
						|
{
 | 
						|
	.name		= "ttyDCC",
 | 
						|
	.write		= dcc_console_write,
 | 
						|
	.device		= dcc_console_device,
 | 
						|
	.setup		= dcc_console_setup,
 | 
						|
	.flags		= CON_PRINTBUFFER,
 | 
						|
	.index		= -1,
 | 
						|
};
 | 
						|
 | 
						|
static struct tty_operations dcc_tty_ops = {
 | 
						|
	.open = dcc_tty_open,
 | 
						|
	.close = dcc_tty_close,
 | 
						|
	.write = dcc_tty_write,
 | 
						|
	.write_room = dcc_tty_write_room,
 | 
						|
	.chars_in_buffer = dcc_tty_chars_in_buffer,
 | 
						|
	.unthrottle = dcc_tty_unthrottle,
 | 
						|
};
 | 
						|
 | 
						|
static int __init dcc_tty_init(void)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
 | 
						|
	hrtimer_init(&g_dcc_timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
 | 
						|
	g_dcc_timer.function = dcc_tty_timer_func;
 | 
						|
 | 
						|
	g_dcc_tty_driver = alloc_tty_driver(1);
 | 
						|
	if (!g_dcc_tty_driver) {
 | 
						|
		printk(KERN_ERR "dcc_tty_probe: alloc_tty_driver failed\n");
 | 
						|
		ret = -ENOMEM;
 | 
						|
		goto err_alloc_tty_driver_failed;
 | 
						|
	}
 | 
						|
	g_dcc_tty_driver->owner = THIS_MODULE;
 | 
						|
	g_dcc_tty_driver->driver_name = "dcc";
 | 
						|
	g_dcc_tty_driver->name = "ttyDCC";
 | 
						|
	g_dcc_tty_driver->major = 0; // auto assign
 | 
						|
	g_dcc_tty_driver->minor_start = 0;
 | 
						|
	g_dcc_tty_driver->type = TTY_DRIVER_TYPE_SERIAL;
 | 
						|
	g_dcc_tty_driver->subtype = SERIAL_TYPE_NORMAL;
 | 
						|
	g_dcc_tty_driver->init_termios = tty_std_termios;
 | 
						|
	g_dcc_tty_driver->flags = TTY_DRIVER_RESET_TERMIOS | TTY_DRIVER_REAL_RAW | TTY_DRIVER_DYNAMIC_DEV;
 | 
						|
	tty_set_operations(g_dcc_tty_driver, &dcc_tty_ops);
 | 
						|
	ret = tty_register_driver(g_dcc_tty_driver);
 | 
						|
	if (ret) {
 | 
						|
		printk(KERN_ERR "dcc_tty_probe: tty_register_driver failed, %d\n", ret);
 | 
						|
		goto err_tty_register_driver_failed;
 | 
						|
	}
 | 
						|
	tty_register_device(g_dcc_tty_driver, 0, NULL);
 | 
						|
 | 
						|
	register_console(&dcc_console);
 | 
						|
	hrtimer_start(&g_dcc_timer, ktime_set(0, 0), HRTIMER_MODE_REL);
 | 
						|
 | 
						|
	return 0;
 | 
						|
 | 
						|
err_tty_register_driver_failed:
 | 
						|
	put_tty_driver(g_dcc_tty_driver);
 | 
						|
	g_dcc_tty_driver = NULL;
 | 
						|
err_alloc_tty_driver_failed:
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static void  __exit dcc_tty_exit(void)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
 | 
						|
	tty_unregister_device(g_dcc_tty_driver, 0);
 | 
						|
	ret = tty_unregister_driver(g_dcc_tty_driver);
 | 
						|
	if (ret < 0) {
 | 
						|
		printk(KERN_ERR "dcc_tty_remove: tty_unregister_driver failed, %d\n", ret);
 | 
						|
	} else {
 | 
						|
		put_tty_driver(g_dcc_tty_driver);
 | 
						|
	}
 | 
						|
	g_dcc_tty_driver = NULL;
 | 
						|
}
 | 
						|
 | 
						|
module_init(dcc_tty_init);
 | 
						|
module_exit(dcc_tty_exit);
 | 
						|
 | 
						|
 |