2007-07-18 05:37:06 +04:00
/*
* xen console driver interface to hvc_console . c
*
* ( c ) 2007 Gerd Hoffmann < kraxel @ suse . de >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <linux/console.h>
# include <linux/delay.h>
# include <linux/err.h>
# include <linux/init.h>
# include <linux/types.h>
# include <asm/xen/hypervisor.h>
2009-10-07 02:11:14 +04:00
# include <xen/xen.h>
2007-07-18 05:37:06 +04:00
# include <xen/page.h>
# include <xen/events.h>
# include <xen/interface/io/console.h>
# include <xen/hvc-console.h>
# include "hvc_console.h"
# define HVC_COOKIE 0x58656e /* "Xen" in hex */
static struct hvc_struct * hvc ;
static int xencons_irq ;
/* ------------------------------------------------------------------ */
2008-05-27 02:31:25 +04:00
static unsigned long console_pfn = ~ 0ul ;
2007-07-18 05:37:06 +04:00
static inline struct xencons_interface * xencons_interface ( void )
{
2008-05-27 02:31:25 +04:00
if ( console_pfn = = ~ 0ul )
return mfn_to_virt ( xen_start_info - > console . domU . mfn ) ;
else
return __va ( console_pfn < < PAGE_SHIFT ) ;
2007-07-18 05:37:06 +04:00
}
static inline void notify_daemon ( void )
{
/* Use evtchn: this is called early, before irq is set up. */
notify_remote_via_evtchn ( xen_start_info - > console . domU . evtchn ) ;
}
2009-10-20 10:28:21 +04:00
static int __write_console ( const char * data , int len )
2007-07-18 05:37:06 +04:00
{
struct xencons_interface * intf = xencons_interface ( ) ;
XENCONS_RING_IDX cons , prod ;
int sent = 0 ;
cons = intf - > out_cons ;
prod = intf - > out_prod ;
mb ( ) ; /* update queue values before going on */
BUG_ON ( ( prod - cons ) > sizeof ( intf - > out ) ) ;
while ( ( sent < len ) & & ( ( prod - cons ) < sizeof ( intf - > out ) ) )
intf - > out [ MASK_XENCONS_IDX ( prod + + , intf - > out ) ] = data [ sent + + ] ;
wmb ( ) ; /* write ring before updating pointer */
intf - > out_prod = prod ;
2010-10-14 22:38:47 +04:00
if ( sent )
notify_daemon ( ) ;
2007-07-18 05:37:06 +04:00
return sent ;
}
2010-09-02 19:17:06 +04:00
static int domU_write_console ( uint32_t vtermno , const char * data , int len )
2009-10-20 10:28:21 +04:00
{
int ret = len ;
/*
* Make sure the whole buffer is emitted , polling if
* necessary . We don ' t ever want to rely on the hvc daemon
* because the most interesting console output is when the
* kernel is crippled .
*/
while ( len ) {
int sent = __write_console ( data , len ) ;
data + = sent ;
len - = sent ;
if ( unlikely ( len ) )
HYPERVISOR_sched_op ( SCHEDOP_yield , NULL ) ;
}
return ret ;
}
2010-09-02 19:17:06 +04:00
static int domU_read_console ( uint32_t vtermno , char * buf , int len )
2007-07-18 05:37:06 +04:00
{
struct xencons_interface * intf = xencons_interface ( ) ;
XENCONS_RING_IDX cons , prod ;
int recv = 0 ;
cons = intf - > in_cons ;
prod = intf - > in_prod ;
mb ( ) ; /* get pointers before reading ring */
BUG_ON ( ( prod - cons ) > sizeof ( intf - > in ) ) ;
while ( cons ! = prod & & recv < len )
buf [ recv + + ] = intf - > in [ MASK_XENCONS_IDX ( cons + + , intf - > in ) ] ;
mb ( ) ; /* read ring before consuming */
intf - > in_cons = cons ;
notify_daemon ( ) ;
return recv ;
}
2010-09-02 19:17:06 +04:00
static struct hv_ops domU_hvc_ops = {
. get_chars = domU_read_console ,
. put_chars = domU_write_console ,
2008-06-20 17:24:08 +04:00
. notifier_add = notifier_add_irq ,
. notifier_del = notifier_del_irq ,
2008-10-14 03:12:48 +04:00
. notifier_hangup = notifier_hangup_irq ,
2007-07-18 05:37:06 +04:00
} ;
2010-09-02 19:17:06 +04:00
static int dom0_read_console ( uint32_t vtermno , char * buf , int len )
{
return HYPERVISOR_console_io ( CONSOLEIO_read , len , buf ) ;
}
/*
* Either for a dom0 to write to the system console , or a domU with a
* debug version of Xen
*/
static int dom0_write_console ( uint32_t vtermno , const char * str , int len )
{
int rc = HYPERVISOR_console_io ( CONSOLEIO_write , len , ( char * ) str ) ;
if ( rc < 0 )
return 0 ;
return len ;
}
static struct hv_ops dom0_hvc_ops = {
. get_chars = dom0_read_console ,
. put_chars = dom0_write_console ,
. notifier_add = notifier_add_irq ,
. notifier_del = notifier_del_irq ,
. notifier_hangup = notifier_hangup_irq ,
} ;
static int __init xen_hvc_init ( void )
2007-07-18 05:37:06 +04:00
{
struct hvc_struct * hp ;
2010-09-02 19:17:06 +04:00
struct hv_ops * ops ;
2007-07-18 05:37:06 +04:00
2010-09-02 19:17:06 +04:00
if ( ! xen_pv_domain ( ) )
2008-05-27 02:31:25 +04:00
return - ENODEV ;
2007-07-18 05:37:06 +04:00
2010-09-02 19:17:06 +04:00
if ( xen_initial_domain ( ) ) {
ops = & dom0_hvc_ops ;
xencons_irq = bind_virq_to_irq ( VIRQ_CONSOLE , 0 ) ;
} else {
if ( ! xen_start_info - > console . domU . evtchn )
return - ENODEV ;
ops = & domU_hvc_ops ;
xencons_irq = bind_evtchn_to_irq ( xen_start_info - > console . domU . evtchn ) ;
}
2007-07-18 05:37:06 +04:00
if ( xencons_irq < 0 )
2008-05-27 02:31:25 +04:00
xencons_irq = 0 ; /* NO_IRQ */
2010-09-02 19:17:06 +04:00
hp = hvc_alloc ( HVC_COOKIE , xencons_irq , ops , 256 ) ;
2007-07-18 05:37:06 +04:00
if ( IS_ERR ( hp ) )
return PTR_ERR ( hp ) ;
hvc = hp ;
2008-05-27 02:31:25 +04:00
console_pfn = mfn_to_pfn ( xen_start_info - > console . domU . mfn ) ;
2007-07-18 05:37:06 +04:00
return 0 ;
}
2008-05-27 02:31:25 +04:00
void xen_console_resume ( void )
{
if ( xencons_irq )
rebind_evtchn_irq ( xen_start_info - > console . domU . evtchn , xencons_irq ) ;
}
2010-09-02 19:17:06 +04:00
static void __exit xen_hvc_fini ( void )
2007-07-18 05:37:06 +04:00
{
if ( hvc )
hvc_remove ( hvc ) ;
}
static int xen_cons_init ( void )
{
2010-09-02 19:17:06 +04:00
struct hv_ops * ops ;
2008-08-20 00:16:17 +04:00
if ( ! xen_pv_domain ( ) )
2007-07-18 05:37:06 +04:00
return 0 ;
2010-09-02 19:17:06 +04:00
if ( xen_initial_domain ( ) )
ops = & dom0_hvc_ops ;
else
ops = & domU_hvc_ops ;
hvc_instantiate ( HVC_COOKIE , 0 , ops ) ;
2007-07-18 05:37:06 +04:00
return 0 ;
}
2010-09-02 19:17:06 +04:00
module_init ( xen_hvc_init ) ;
module_exit ( xen_hvc_fini ) ;
2007-07-18 05:37:06 +04:00
console_initcall ( xen_cons_init ) ;
2008-05-27 02:31:00 +04:00
# ifdef CONFIG_EARLY_PRINTK
2007-07-18 05:37:06 +04:00
static void xenboot_write_console ( struct console * console , const char * string ,
unsigned len )
{
unsigned int linelen , off = 0 ;
const char * pos ;
2010-09-02 19:17:06 +04:00
dom0_write_console ( 0 , string , len ) ;
if ( xen_initial_domain ( ) )
return ;
2008-05-27 02:31:00 +04:00
2010-09-02 19:17:06 +04:00
domU_write_console ( 0 , " (early) " , 8 ) ;
2007-07-18 05:37:06 +04:00
while ( off < len & & NULL ! = ( pos = strchr ( string + off , ' \n ' ) ) ) {
linelen = pos - string + off ;
if ( off + linelen > len )
break ;
2010-09-02 19:17:06 +04:00
domU_write_console ( 0 , string + off , linelen ) ;
domU_write_console ( 0 , " \r \n " , 2 ) ;
2007-07-18 05:37:06 +04:00
off + = linelen + 1 ;
}
if ( off < len )
2010-09-02 19:17:06 +04:00
domU_write_console ( 0 , string + off , len - off ) ;
2007-07-18 05:37:06 +04:00
}
struct console xenboot_console = {
. name = " xenboot " ,
. write = xenboot_write_console ,
2008-05-27 02:31:00 +04:00
. flags = CON_PRINTBUFFER | CON_BOOT | CON_ANYTIME ,
2007-07-18 05:37:06 +04:00
} ;
2008-05-27 02:31:00 +04:00
# endif /* CONFIG_EARLY_PRINTK */
2008-05-27 02:30:59 +04:00
void xen_raw_console_write ( const char * str )
{
2010-09-02 19:17:06 +04:00
dom0_write_console ( 0 , str , strlen ( str ) ) ;
2008-05-27 02:30:59 +04:00
}
void xen_raw_printk ( const char * fmt , . . . )
{
static char buf [ 512 ] ;
va_list ap ;
va_start ( ap , fmt ) ;
vsnprintf ( buf , sizeof ( buf ) , fmt , ap ) ;
va_end ( ap ) ;
xen_raw_console_write ( buf ) ;
}