diff options
Diffstat (limited to 'drivers/char/vt.c')
-rw-r--r-- | drivers/char/vt.c | 84 |
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 | |||
2374 | out: | ||
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 | */ | ||
2808 | static void con_close(struct tty_struct *tty, struct file *filp) | 2785 | static 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) | 2790 | static 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 | ||
2831 | static int default_italic_color = 2; // green (ASCII) | 2801 | static 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 | ||
2955 | int __init vty_init(void) | 2926 | static struct cdev vc0_cdev; |
2927 | |||
2928 | int __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 = ®istered_con_driver[i]; | 3555 | struct con_driver *con = ®istered_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); |