/* If console not initialised the printk will be sent to xen serial line
NOTE: you need to enable verbose in xen/Rules.mk for it to work. */
+static struct consfront_dev* xen_console = NULL;
static int console_initialised = 0;
__attribute__((weak)) void console_input(char * buf, unsigned len)
void init_console(void)
{
printk("Initialising console ... ");
- xencons_ring_init();
+ xen_console = xencons_ring_init();
console_initialised = 1;
/* This is also required to notify the daemon */
printk("done.\n");
}
+
+void suspend_console(void)
+{
+ console_initialised = 0;
+ xencons_ring_fini(xen_console);
+}
+
+void resume_console(void)
+{
+ xencons_ring_resume(xen_console);
+ console_initialised = 1;
+}
\ No newline at end of file
static struct xencons_interface *console_ring;
uint32_t console_evtchn;
+static struct consfront_dev* resume_xen_console(struct consfront_dev* dev);
+
#ifdef CONFIG_PARAVIRT
void get_console(void *p)
{
{
uint64_t v = -1;
- hvm_get_parameter(HVM_PARAM_CONSOLE_EVTCHN, &v);
+ if (hvm_get_parameter(HVM_PARAM_CONSOLE_EVTCHN, &v))
+ BUG();
console_evtchn = v;
- hvm_get_parameter(HVM_PARAM_CONSOLE_PFN, &v);
+ if (hvm_get_parameter(HVM_PARAM_CONSOLE_PFN, &v))
+ BUG();
console_ring = (struct xencons_interface *)map_frame_virt(v);
}
#endif
notify_daemon(dev);
return sent;
-}
-
-
+}
void console_handle_input(evtchn_port_t port, struct pt_regs *regs, void *data)
{
struct consfront_dev *xencons_ring_init(void)
{
- int err;
- struct consfront_dev *dev;
+ struct consfront_dev *dev;
- if (!console_evtchn)
- return 0;
+ if (!console_evtchn)
+ return 0;
- dev = malloc(sizeof(struct consfront_dev));
- memset(dev, 0, sizeof(struct consfront_dev));
- dev->nodename = "device/console";
- dev->dom = 0;
- dev->backend = 0;
- dev->ring_ref = 0;
+ dev = malloc(sizeof(struct consfront_dev));
+ memset(dev, 0, sizeof(struct consfront_dev));
+ dev->nodename = "device/console";
+ dev->dom = 0;
+ dev->backend = 0;
+ dev->ring_ref = 0;
#ifdef HAVE_LIBC
- dev->fd = -1;
+ dev->fd = -1;
#endif
- dev->evtchn = console_evtchn;
- dev->ring = xencons_interface();
-
- err = bind_evtchn(dev->evtchn, console_handle_input, dev);
- if (err <= 0) {
- printk("XEN console request chn bind failed %i\n", err);
- free(dev);
- return NULL;
- }
- unmask_evtchn(dev->evtchn);
- /* In case we have in-flight data after save/restore... */
- notify_daemon(dev);
+ return resume_xen_console(dev);
+}
+
+static struct consfront_dev* resume_xen_console(struct consfront_dev* dev)
+{
+ int err;
- return dev;
+ dev->evtchn = console_evtchn;
+ dev->ring = xencons_interface();
+
+ err = bind_evtchn(dev->evtchn, console_handle_input, dev);
+ if (err <= 0) {
+ printk("XEN console request chn bind failed %i\n", err);
+ free(dev);
+ return NULL;
+ }
+ unmask_evtchn(dev->evtchn);
+
+ /* In case we have in-flight data after save/restore... */
+ notify_daemon(dev);
+
+ return dev;
}
-void xencons_resume(void)
+void xencons_ring_fini(struct consfront_dev* dev)
{
- (void)xencons_ring_init();
+ if (dev)
+ mask_evtchn(dev->evtchn);
}
+void xencons_ring_resume(struct consfront_dev* dev)
+{
+ if (dev) {
+#if CONFIG_PARAVIRT
+ get_console(&start_info);
+#else
+ get_console(0);
+#endif
+ resume_xen_console(dev);
+ }
+}
void get_console(void *p);
void init_console(void);
void console_print(struct consfront_dev *dev, char *data, int length);
-void fini_console(struct consfront_dev *dev);
+void fini_consfront(struct consfront_dev *dev);
+void suspend_console(void);
+void resume_console(void);
/* Low level functions defined in xencons_ring.c */
extern struct wait_queue_head console_queue;
struct consfront_dev *xencons_ring_init(void);
+void xencons_ring_fini(struct consfront_dev* dev);
+void xencons_ring_resume(struct consfront_dev* dev);
struct consfront_dev *init_consfront(char *_nodename);
int xencons_ring_send(struct consfront_dev *dev, const char *data, unsigned len);
int xencons_ring_send_no_notify(struct consfront_dev *dev, const char *data, unsigned len);