Commit 3e6c6f63 authored by Rusty Russell's avatar Rusty Russell Committed by Linus Torvalds

Delay creation of khcvd thread

This changes hvc_init() to be called only when someone actually uses the
hvc_console driver.  Dave Jones complained when profiling bootup.

hvc_console used to only be for Power aka pSeries: now lguest and Xen both
want it built-in in case the kernel is a guest under one of those, even
though usually it will be a native boot.
Signed-off-by: default avatarRusty Russell <rusty@rustcorp.com.au>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: Jeremy Fitzhardinge <jeremy@goop.org>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent fd5eea42
...@@ -585,8 +585,8 @@ config TIPAR ...@@ -585,8 +585,8 @@ config TIPAR
config HVC_DRIVER config HVC_DRIVER
bool bool
help help
Users of pSeries machines that want to utilize the hvc console front-end Generic "hypervisor virtual console" infrastructure for various
module for their backend console driver should select this option. hypervisors (pSeries, Xen, lguest).
It will automatically be selected if one of the back-end console drivers It will automatically be selected if one of the back-end console drivers
is selected. is selected.
......
...@@ -69,6 +69,8 @@ static struct task_struct *hvc_task; ...@@ -69,6 +69,8 @@ static struct task_struct *hvc_task;
/* Picks up late kicks after list walk but before schedule() */ /* Picks up late kicks after list walk but before schedule() */
static int hvc_kicked; static int hvc_kicked;
static int hvc_init(void);
#ifdef CONFIG_MAGIC_SYSRQ #ifdef CONFIG_MAGIC_SYSRQ
static int sysrq_pressed; static int sysrq_pressed;
#endif #endif
...@@ -754,6 +756,13 @@ struct hvc_struct __devinit *hvc_alloc(uint32_t vtermno, int irq, ...@@ -754,6 +756,13 @@ struct hvc_struct __devinit *hvc_alloc(uint32_t vtermno, int irq,
struct hvc_struct *hp; struct hvc_struct *hp;
int i; int i;
/* We wait until a driver actually comes along */
if (!hvc_driver) {
int err = hvc_init();
if (err)
return ERR_PTR(err);
}
hp = kmalloc(ALIGN(sizeof(*hp), sizeof(long)) + outbuf_size, hp = kmalloc(ALIGN(sizeof(*hp), sizeof(long)) + outbuf_size,
GFP_KERNEL); GFP_KERNEL);
if (!hp) if (!hp)
...@@ -829,16 +838,18 @@ int __devexit hvc_remove(struct hvc_struct *hp) ...@@ -829,16 +838,18 @@ int __devexit hvc_remove(struct hvc_struct *hp)
return 0; return 0;
} }
/* Driver initialization. Follow console initialization. This is where the TTY /* Driver initialization: called as soon as someone uses hvc_alloc(). */
* interfaces start to become available. */ static int hvc_init(void)
static int __init hvc_init(void)
{ {
struct tty_driver *drv; struct tty_driver *drv;
int err;
/* We need more than hvc_count adapters due to hotplug additions. */ /* We need more than hvc_count adapters due to hotplug additions. */
drv = alloc_tty_driver(HVC_ALLOC_TTY_ADAPTERS); drv = alloc_tty_driver(HVC_ALLOC_TTY_ADAPTERS);
if (!drv) if (!drv) {
return -ENOMEM; err = -ENOMEM;
goto out;
}
drv->owner = THIS_MODULE; drv->owner = THIS_MODULE;
drv->driver_name = "hvc"; drv->driver_name = "hvc";
...@@ -854,30 +865,43 @@ static int __init hvc_init(void) ...@@ -854,30 +865,43 @@ static int __init hvc_init(void)
* added later. */ * added later. */
hvc_task = kthread_run(khvcd, NULL, "khvcd"); hvc_task = kthread_run(khvcd, NULL, "khvcd");
if (IS_ERR(hvc_task)) { if (IS_ERR(hvc_task)) {
panic("Couldn't create kthread for console.\n"); printk(KERN_ERR "Couldn't create kthread for console.\n");
put_tty_driver(drv); err = PTR_ERR(hvc_task);
return -EIO; goto put_tty;
} }
if (tty_register_driver(drv)) err = tty_register_driver(drv);
panic("Couldn't register hvc console driver\n"); if (err) {
printk(KERN_ERR "Couldn't register hvc console driver\n");
goto stop_thread;
}
/* FIXME: This mb() seems completely random. Remove it. */
mb(); mb();
hvc_driver = drv; hvc_driver = drv;
return 0; return 0;
put_tty:
put_tty_driver(hvc_driver);
stop_thread:
kthread_stop(hvc_task);
hvc_task = NULL;
out:
return err;
} }
module_init(hvc_init);
/* This isn't particularly necessary due to this being a console driver /* This isn't particularly necessary due to this being a console driver
* but it is nice to be thorough. * but it is nice to be thorough.
*/ */
static void __exit hvc_exit(void) static void __exit hvc_exit(void)
{ {
kthread_stop(hvc_task); if (hvc_driver) {
kthread_stop(hvc_task);
tty_unregister_driver(hvc_driver); tty_unregister_driver(hvc_driver);
/* return tty_struct instances allocated in hvc_init(). */ /* return tty_struct instances allocated in hvc_init(). */
put_tty_driver(hvc_driver); put_tty_driver(hvc_driver);
unregister_console(&hvc_con_driver); unregister_console(&hvc_con_driver);
}
} }
module_exit(hvc_exit); module_exit(hvc_exit);
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment