2005-04-16 15:20:36 -07:00
|
|
|
/*
|
|
|
|
* drivers/power/process.c - Functions for saving/restoring console.
|
|
|
|
*
|
|
|
|
* Originally from swsusp.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/vt_kern.h>
|
|
|
|
#include <linux/kbd_kern.h>
|
|
|
|
#include <linux/console.h>
|
2008-04-28 02:15:03 -07:00
|
|
|
#include <linux/module.h>
|
2005-04-16 15:20:36 -07:00
|
|
|
#include "power.h"
|
|
|
|
|
2006-02-07 12:58:50 -08:00
|
|
|
#if defined(CONFIG_VT) && defined(CONFIG_VT_CONSOLE)
|
|
|
|
#define SUSPEND_CONSOLE (MAX_NR_CONSOLES-1)
|
|
|
|
|
2005-04-16 15:20:36 -07:00
|
|
|
static int orig_fgconsole, orig_kmsg;
|
2008-04-28 02:15:03 -07:00
|
|
|
static int disable_vt_switch;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Normally during a suspend, we allocate a new console and switch to it.
|
|
|
|
* When we resume, we switch back to the original console. This switch
|
|
|
|
* can be slow, so on systems where the framebuffer can handle restoration
|
|
|
|
* of video registers anyways, there's little point in doing the console
|
|
|
|
* switch. This function allows you to disable it by passing it '0'.
|
|
|
|
*/
|
|
|
|
void pm_set_vt_switch(int do_switch)
|
|
|
|
{
|
|
|
|
acquire_console_sem();
|
|
|
|
disable_vt_switch = !do_switch;
|
|
|
|
release_console_sem();
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(pm_set_vt_switch);
|
2005-04-16 15:20:36 -07:00
|
|
|
|
|
|
|
int pm_prepare_console(void)
|
|
|
|
{
|
|
|
|
acquire_console_sem();
|
|
|
|
|
2008-04-28 02:15:03 -07:00
|
|
|
if (disable_vt_switch) {
|
|
|
|
release_console_sem();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-04-16 15:20:36 -07:00
|
|
|
orig_fgconsole = fg_console;
|
|
|
|
|
|
|
|
if (vc_allocate(SUSPEND_CONSOLE)) {
|
|
|
|
/* we can't have a free VC for now. Too bad,
|
|
|
|
* we don't want to mess the screen for now. */
|
|
|
|
release_console_sem();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2007-03-16 13:38:24 -08:00
|
|
|
if (set_console(SUSPEND_CONSOLE)) {
|
|
|
|
/*
|
|
|
|
* We're unable to switch to the SUSPEND_CONSOLE.
|
|
|
|
* Let the calling function know so it can decide
|
|
|
|
* what to do.
|
|
|
|
*/
|
|
|
|
release_console_sem();
|
|
|
|
return 1;
|
|
|
|
}
|
2005-04-16 15:20:36 -07:00
|
|
|
release_console_sem();
|
|
|
|
|
|
|
|
if (vt_waitactive(SUSPEND_CONSOLE)) {
|
|
|
|
pr_debug("Suspend: Can't switch VCs.");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
orig_kmsg = kmsg_redirect;
|
|
|
|
kmsg_redirect = SUSPEND_CONSOLE;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pm_restore_console(void)
|
|
|
|
{
|
|
|
|
acquire_console_sem();
|
2008-04-28 02:15:03 -07:00
|
|
|
if (disable_vt_switch) {
|
|
|
|
release_console_sem();
|
|
|
|
return;
|
|
|
|
}
|
2005-04-16 15:20:36 -07:00
|
|
|
set_console(orig_fgconsole);
|
|
|
|
release_console_sem();
|
2009-02-14 02:06:17 +01:00
|
|
|
|
|
|
|
if (vt_waitactive(orig_fgconsole)) {
|
|
|
|
pr_debug("Resume: Can't switch VCs.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2005-04-16 15:20:36 -07:00
|
|
|
kmsg_redirect = orig_kmsg;
|
|
|
|
}
|
2006-02-01 03:05:21 -08:00
|
|
|
#endif
|