aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/char/vt.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/char/vt.c')
-rw-r--r--drivers/char/vt.c84
1 files changed, 31 insertions, 53 deletions
diff --git a/drivers/char/vt.c b/drivers/char/vt.c
index 60359c360912..57029fefd64a 100644
--- a/drivers/char/vt.c
+++ b/drivers/char/vt.c
@@ -100,10 +100,10 @@
100#include <linux/font.h> 100#include <linux/font.h>
101#include <linux/bitops.h> 101#include <linux/bitops.h>
102#include <linux/notifier.h> 102#include <linux/notifier.h>
103 103#include <linux/device.h>
104#include <asm/io.h> 104#include <linux/io.h>
105#include <asm/system.h> 105#include <asm/system.h>
106#include <asm/uaccess.h> 106#include <linux/uaccess.h>
107 107
108#define MAX_NR_CON_DRIVER 16 108#define MAX_NR_CON_DRIVER 16
109 109
@@ -2136,27 +2136,9 @@ static int do_con_write(struct tty_struct *tty, const unsigned char *buf, int co
2136 release_console_sem(); 2136 release_console_sem();
2137 return 0; 2137 return 0;
2138 } 2138 }
2139 release_console_sem();
2140
2141 orig_buf = buf; 2139 orig_buf = buf;
2142 orig_count = count; 2140 orig_count = count;
2143 2141
2144 /* At this point 'buf' is guaranteed to be a kernel buffer
2145 * and therefore no access to userspace (and therefore sleeping)
2146 * will be needed. The con_buf_mtx serializes all tty based
2147 * console rendering and vcs write/read operations. We hold
2148 * the console spinlock during the entire write.
2149 */
2150
2151 acquire_console_sem();
2152
2153 vc = tty->driver_data;
2154 if (vc == NULL) {
2155 printk(KERN_ERR "vt: argh, driver_data _became_ NULL !\n");
2156 release_console_sem();
2157 goto out;
2158 }
2159
2160 himask = vc->vc_hi_font_mask; 2142 himask = vc->vc_hi_font_mask;
2161 charmask = himask ? 0x1ff : 0xff; 2143 charmask = himask ? 0x1ff : 0xff;
2162 2144
@@ -2370,8 +2352,6 @@ rescan_last_byte:
2370 FLUSH 2352 FLUSH
2371 console_conditional_schedule(); 2353 console_conditional_schedule();
2372 release_console_sem(); 2354 release_console_sem();
2373
2374out:
2375 notify_update(vc); 2355 notify_update(vc);
2376 return n; 2356 return n;
2377#undef FLUSH 2357#undef FLUSH
@@ -2583,8 +2563,6 @@ int tioclinux(struct tty_struct *tty, unsigned long arg)
2583 int lines; 2563 int lines;
2584 int ret; 2564 int ret;
2585 2565
2586 if (tty->driver->type != TTY_DRIVER_TYPE_CONSOLE)
2587 return -EINVAL;
2588 if (current->signal->tty != tty && !capable(CAP_SYS_ADMIN)) 2566 if (current->signal->tty != tty && !capable(CAP_SYS_ADMIN))
2589 return -EPERM; 2567 return -EPERM;
2590 if (get_user(type, p)) 2568 if (get_user(type, p))
@@ -2778,6 +2756,12 @@ static int con_open(struct tty_struct *tty, struct file *filp)
2778 ret = vc_allocate(currcons); 2756 ret = vc_allocate(currcons);
2779 if (ret == 0) { 2757 if (ret == 0) {
2780 struct vc_data *vc = vc_cons[currcons].d; 2758 struct vc_data *vc = vc_cons[currcons].d;
2759
2760 /* Still being freed */
2761 if (vc->vc_tty) {
2762 release_console_sem();
2763 return -ERESTARTSYS;
2764 }
2781 tty->driver_data = vc; 2765 tty->driver_data = vc;
2782 vc->vc_tty = tty; 2766 vc->vc_tty = tty;
2783 2767
@@ -2798,34 +2782,20 @@ static int con_open(struct tty_struct *tty, struct file *filp)
2798 return ret; 2782 return ret;
2799} 2783}
2800 2784
2801/*
2802 * We take tty_mutex in here to prevent another thread from coming in via init_dev
2803 * and taking a ref against the tty while we're in the process of forgetting
2804 * about it and cleaning things up.
2805 *
2806 * This is because vcs_remove_sysfs() can sleep and will drop the BKL.
2807 */
2808static void con_close(struct tty_struct *tty, struct file *filp) 2785static void con_close(struct tty_struct *tty, struct file *filp)
2809{ 2786{
2810 mutex_lock(&tty_mutex); 2787 /* Nothing to do - we defer to shutdown */
2811 acquire_console_sem(); 2788}
2812 if (tty && tty->count == 1) {
2813 struct vc_data *vc = tty->driver_data;
2814 2789
2815 if (vc) 2790static void con_shutdown(struct tty_struct *tty)
2816 vc->vc_tty = NULL; 2791{
2817 tty->driver_data = NULL; 2792 struct vc_data *vc = tty->driver_data;
2818 vcs_remove_sysfs(tty); 2793 BUG_ON(vc == NULL);
2819 release_console_sem(); 2794 acquire_console_sem();
2820 mutex_unlock(&tty_mutex); 2795 vc->vc_tty = NULL;
2821 /* 2796 vcs_remove_sysfs(tty);
2822 * tty_mutex is released, but we still hold BKL, so there is
2823 * still exclusion against init_dev()
2824 */
2825 return;
2826 }
2827 release_console_sem(); 2797 release_console_sem();
2828 mutex_unlock(&tty_mutex); 2798 tty_shutdown(tty);
2829} 2799}
2830 2800
2831static int default_italic_color = 2; // green (ASCII) 2801static int default_italic_color = 2; // green (ASCII)
@@ -2950,10 +2920,19 @@ static const struct tty_operations con_ops = {
2950 .throttle = con_throttle, 2920 .throttle = con_throttle,
2951 .unthrottle = con_unthrottle, 2921 .unthrottle = con_unthrottle,
2952 .resize = vt_resize, 2922 .resize = vt_resize,
2923 .shutdown = con_shutdown
2953}; 2924};
2954 2925
2955int __init vty_init(void) 2926static struct cdev vc0_cdev;
2927
2928int __init vty_init(const struct file_operations *console_fops)
2956{ 2929{
2930 cdev_init(&vc0_cdev, console_fops);
2931 if (cdev_add(&vc0_cdev, MKDEV(TTY_MAJOR, 0), 1) ||
2932 register_chrdev_region(MKDEV(TTY_MAJOR, 0), 1, "/dev/vc/0") < 0)
2933 panic("Couldn't register /dev/tty0 driver\n");
2934 device_create(tty_class, NULL, MKDEV(TTY_MAJOR, 0), NULL, "tty0");
2935
2957 vcs_init(); 2936 vcs_init();
2958 2937
2959 console_driver = alloc_tty_driver(MAX_NR_CONSOLES); 2938 console_driver = alloc_tty_driver(MAX_NR_CONSOLES);
@@ -2972,7 +2951,6 @@ int __init vty_init(void)
2972 tty_set_operations(console_driver, &con_ops); 2951 tty_set_operations(console_driver, &con_ops);
2973 if (tty_register_driver(console_driver)) 2952 if (tty_register_driver(console_driver))
2974 panic("Couldn't register console driver\n"); 2953 panic("Couldn't register console driver\n");
2975
2976 kbd_init(); 2954 kbd_init();
2977 console_map_init(); 2955 console_map_init();
2978#ifdef CONFIG_PROM_CONSOLE 2956#ifdef CONFIG_PROM_CONSOLE
@@ -3466,7 +3444,7 @@ int register_con_driver(const struct consw *csw, int first, int last)
3466 if (retval) 3444 if (retval)
3467 goto err; 3445 goto err;
3468 3446
3469 con_driver->dev = device_create_drvdata(vtconsole_class, NULL, 3447 con_driver->dev = device_create(vtconsole_class, NULL,
3470 MKDEV(0, con_driver->node), 3448 MKDEV(0, con_driver->node),
3471 NULL, "vtcon%i", 3449 NULL, "vtcon%i",
3472 con_driver->node); 3450 con_driver->node);
@@ -3577,7 +3555,7 @@ static int __init vtconsole_class_init(void)
3577 struct con_driver *con = &registered_con_driver[i]; 3555 struct con_driver *con = &registered_con_driver[i];
3578 3556
3579 if (con->con && !con->dev) { 3557 if (con->con && !con->dev) {
3580 con->dev = device_create_drvdata(vtconsole_class, NULL, 3558 con->dev = device_create(vtconsole_class, NULL,
3581 MKDEV(0, con->node), 3559 MKDEV(0, con->node),
3582 NULL, "vtcon%i", 3560 NULL, "vtcon%i",
3583 con->node); 3561 con->node);