2010-01-18 16:44:55 +03:00
/*
* Copyright ( C ) 2006 , 2007 , 2009 Rusty Russell , IBM Corporation
2009-12-21 18:33:25 +03:00
* Copyright ( C ) 2009 , 2010 Red Hat , Inc .
2007-10-22 05:03:39 +04:00
*
* 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
*/
2009-12-21 19:06:04 +03:00
# include <linux/cdev.h>
2009-12-21 20:06:21 +03:00
# include <linux/debugfs.h>
2009-12-21 19:06:04 +03:00
# include <linux/device.h>
2007-10-22 05:03:39 +04:00
# include <linux/err.h>
2009-12-21 19:19:30 +03:00
# include <linux/fs.h>
2007-10-22 05:03:39 +04:00
# include <linux/init.h>
2010-01-18 16:45:05 +03:00
# include <linux/list.h>
2009-12-21 19:19:30 +03:00
# include <linux/poll.h>
# include <linux/sched.h>
2010-01-18 16:45:05 +03:00
# include <linux/spinlock.h>
2007-10-22 05:03:39 +04:00
# include <linux/virtio.h>
# include <linux/virtio_console.h>
2009-12-21 19:19:30 +03:00
# include <linux/wait.h>
2009-12-21 18:33:25 +03:00
# include <linux/workqueue.h>
2007-10-22 05:03:39 +04:00
# include "hvc_console.h"
2010-01-18 16:45:05 +03:00
/*
* This is a global struct for storing common data for all the devices
* this driver handles .
*
* Mainly , it has a linked list for all the consoles in one place so
* that callbacks from hvc for get_chars ( ) , put_chars ( ) work properly
* across multiple devices and multiple ports per device .
*/
struct ports_driver_data {
2009-12-21 19:06:04 +03:00
/* Used for registering chardevs */
struct class * class ;
2009-12-21 20:06:21 +03:00
/* Used for exporting per-port information to debugfs */
struct dentry * debugfs_dir ;
2009-12-21 19:06:04 +03:00
/* Number of devices this driver is handling */
unsigned int index ;
2010-01-18 16:45:06 +03:00
/*
* This is used to keep track of the number of hvc consoles
* spawned by this driver . This number is given as the first
* argument to hvc_alloc ( ) . To correctly map an initial
* console spawned via hvc_instantiate to the console being
* hooked up via hvc_alloc , we need to pass the same vtermno .
*
* We also just assume the first console being initialised was
* the first one that got used as the initial console .
*/
unsigned int next_vtermno ;
2010-01-18 16:45:05 +03:00
/* All the console devices handled by this driver */
struct list_head consoles ;
} ;
static struct ports_driver_data pdrvdata ;
DEFINE_SPINLOCK ( pdrvdata_lock ) ;
2010-01-18 16:45:09 +03:00
/* This struct holds information that's relevant only for console ports */
struct console {
/* We'll place all consoles in a list in the pdrvdata struct */
struct list_head list ;
/* The hvc device associated with this console port */
struct hvc_struct * hvc ;
/*
* This number identifies the number that we used to register
* with hvc in hvc_instantiate ( ) and hvc_alloc ( ) ; this is the
* number passed on by the hvc callbacks to us to
* differentiate between the other console ports handled by
* this driver
*/
u32 vtermno ;
} ;
2010-01-18 16:45:01 +03:00
struct port_buffer {
char * buf ;
/* size of the buffer in *buf above */
size_t size ;
/* used length of the buffer */
size_t len ;
/* offset in the buf from which to consume data */
size_t offset ;
} ;
2009-12-21 18:33:25 +03:00
/*
* This is a per - device struct that stores data common to all the
* ports for that device ( vdev - > priv ) .
*/
struct ports_device {
/*
* Workqueue handlers where we process deferred work after
* notification
*/
struct work_struct control_work ;
2009-12-21 19:52:08 +03:00
struct work_struct config_work ;
2009-12-21 18:33:25 +03:00
struct list_head ports ;
/* To protect the list of ports */
spinlock_t ports_lock ;
/* To protect the vq operations for the control channel */
spinlock_t cvq_lock ;
/* The current config space is stored here */
struct virtio_console_config config ;
/* The virtio device we're associated with */
struct virtio_device * vdev ;
/*
* A couple of virtqueues for the control channel : one for
* guest - > host transfers , one for host - > guest transfers
*/
struct virtqueue * c_ivq , * c_ovq ;
/* Array of per-port IO virtqueues */
struct virtqueue * * in_vqs , * * out_vqs ;
2009-12-21 19:06:04 +03:00
/* Used for numbering devices for sysfs and debugfs */
unsigned int drv_index ;
/* Major number for this device. Ports will be created as minors. */
int chr_major ;
2009-12-21 18:33:25 +03:00
} ;
2010-01-18 16:45:07 +03:00
/* This struct holds the per-port data */
2010-01-18 16:45:00 +03:00
struct port {
2009-12-21 18:33:25 +03:00
/* Next port in the list, head is in the ports_device */
struct list_head list ;
2010-01-18 16:45:07 +03:00
/* Pointer to the parent virtio_console device */
struct ports_device * portdev ;
2010-01-18 16:45:01 +03:00
/* The current buffer from which data has to be fed to readers */
struct port_buffer * inbuf ;
2010-01-18 16:45:00 +03:00
2010-01-18 16:45:12 +03:00
/*
* To protect the operations on the in_vq associated with this
* port . Has to be a spinlock because it can be called from
* interrupt context ( get_char ( ) ) .
*/
spinlock_t inbuf_lock ;
2010-01-18 16:45:07 +03:00
/* The IO vqs for this port */
struct virtqueue * in_vq , * out_vq ;
2009-12-21 20:06:21 +03:00
/* File in the debugfs directory that exposes this port's information */
struct dentry * debugfs_file ;
2010-01-18 16:45:09 +03:00
/*
* The entries in this struct will be valid if this port is
* hooked up to an hvc console
*/
struct console cons ;
2009-12-21 18:33:25 +03:00
2009-12-21 19:06:04 +03:00
/* Each port associates with a separate char device */
struct cdev cdev ;
struct device * dev ;
2009-12-21 19:19:30 +03:00
/* A waitqueue for poll() or blocking read operations */
wait_queue_head_t waitqueue ;
2009-12-21 19:27:40 +03:00
/* The 'name' of the port that we expose via sysfs properties */
char * name ;
2009-12-21 18:33:25 +03:00
/* The 'id' to identify the port with the Host */
u32 id ;
2009-12-21 19:19:30 +03:00
/* Is the host device open */
bool host_connected ;
2009-11-26 08:55:38 +03:00
/* We should allow only one process to open a port */
bool guest_connected ;
2010-01-18 16:45:00 +03:00
} ;
2007-10-22 05:03:39 +04:00
2010-01-18 16:44:56 +03:00
/* This is the very early arch-specified put chars function. */
static int ( * early_put_chars ) ( u32 , const char * , int ) ;
2010-01-18 16:45:05 +03:00
static struct port * find_port_by_vtermno ( u32 vtermno )
{
struct port * port ;
2010-01-18 16:45:09 +03:00
struct console * cons ;
2010-01-18 16:45:05 +03:00
unsigned long flags ;
spin_lock_irqsave ( & pdrvdata_lock , flags ) ;
2010-01-18 16:45:09 +03:00
list_for_each_entry ( cons , & pdrvdata . consoles , list ) {
if ( cons - > vtermno = = vtermno ) {
port = container_of ( cons , struct port , cons ) ;
2010-01-18 16:45:05 +03:00
goto out ;
2010-01-18 16:45:09 +03:00
}
2010-01-18 16:45:05 +03:00
}
port = NULL ;
out :
spin_unlock_irqrestore ( & pdrvdata_lock , flags ) ;
return port ;
}
2009-12-21 18:33:25 +03:00
static struct port * find_port_by_id ( struct ports_device * portdev , u32 id )
{
struct port * port ;
unsigned long flags ;
spin_lock_irqsave ( & portdev - > ports_lock , flags ) ;
list_for_each_entry ( port , & portdev - > ports , list )
if ( port - > id = = id )
goto out ;
port = NULL ;
out :
spin_unlock_irqrestore ( & portdev - > ports_lock , flags ) ;
return port ;
}
2010-01-18 16:45:12 +03:00
static struct port * find_port_by_vq ( struct ports_device * portdev ,
struct virtqueue * vq )
{
struct port * port ;
unsigned long flags ;
2009-12-21 18:33:25 +03:00
spin_lock_irqsave ( & portdev - > ports_lock , flags ) ;
list_for_each_entry ( port , & portdev - > ports , list )
2010-01-18 16:45:12 +03:00
if ( port - > in_vq = = vq | | port - > out_vq = = vq )
goto out ;
port = NULL ;
out :
2009-12-21 18:33:25 +03:00
spin_unlock_irqrestore ( & portdev - > ports_lock , flags ) ;
2010-01-18 16:45:12 +03:00
return port ;
}
2009-12-21 18:33:25 +03:00
static bool is_console_port ( struct port * port )
{
if ( port - > cons . hvc )
return true ;
return false ;
}
static inline bool use_multiport ( struct ports_device * portdev )
{
/*
* This condition can be true when put_chars is called from
* early_init
*/
if ( ! portdev - > vdev )
return 0 ;
return portdev - > vdev - > features [ 0 ] & ( 1 < < VIRTIO_CONSOLE_F_MULTIPORT ) ;
}
2010-01-18 16:45:01 +03:00
static void free_buf ( struct port_buffer * buf )
{
kfree ( buf - > buf ) ;
kfree ( buf ) ;
}
static struct port_buffer * alloc_buf ( size_t buf_size )
{
struct port_buffer * buf ;
buf = kmalloc ( sizeof ( * buf ) , GFP_KERNEL ) ;
if ( ! buf )
goto fail ;
buf - > buf = kzalloc ( buf_size , GFP_KERNEL ) ;
if ( ! buf - > buf )
goto free_buf ;
buf - > len = 0 ;
buf - > offset = 0 ;
buf - > size = buf_size ;
return buf ;
free_buf :
kfree ( buf ) ;
fail :
return NULL ;
}
2010-01-18 16:45:03 +03:00
/* Callers should take appropriate locks */
static void * get_inbuf ( struct port * port )
{
struct port_buffer * buf ;
struct virtqueue * vq ;
unsigned int len ;
vq = port - > in_vq ;
buf = vq - > vq_ops - > get_buf ( vq , & len ) ;
if ( buf ) {
buf - > len = len ;
buf - > offset = 0 ;
}
return buf ;
}
2010-01-18 16:45:02 +03:00
/*
* Create a scatter - gather list representing our input buffer and put
* it in the queue .
*
* Callers should take appropriate locks .
*/
2010-01-18 16:45:12 +03:00
static int add_inbuf ( struct virtqueue * vq , struct port_buffer * buf )
2010-01-18 16:45:02 +03:00
{
struct scatterlist sg [ 1 ] ;
2010-01-18 16:45:12 +03:00
int ret ;
2010-01-18 16:45:07 +03:00
2010-01-18 16:45:02 +03:00
sg_init_one ( sg , buf - > buf , buf - > size ) ;
2010-01-18 16:45:12 +03:00
ret = vq - > vq_ops - > add_buf ( vq , sg , 0 , 1 , buf ) ;
2010-01-18 16:45:02 +03:00
vq - > vq_ops - > kick ( vq ) ;
2010-01-18 16:45:12 +03:00
return ret ;
}
2009-12-21 19:45:30 +03:00
/* Discard any unread data this port has. Callers lockers. */
static void discard_port_data ( struct port * port )
{
struct port_buffer * buf ;
struct virtqueue * vq ;
unsigned int len ;
2010-02-12 08:02:18 +03:00
int ret ;
2009-12-21 19:45:30 +03:00
vq = port - > in_vq ;
if ( port - > inbuf )
buf = port - > inbuf ;
else
buf = vq - > vq_ops - > get_buf ( vq , & len ) ;
2010-02-12 08:02:18 +03:00
ret = 0 ;
while ( buf ) {
if ( add_inbuf ( vq , buf ) < 0 ) {
ret + + ;
free_buf ( buf ) ;
}
buf = vq - > vq_ops - > get_buf ( vq , & len ) ;
2009-12-21 19:45:30 +03:00
}
port - > inbuf = NULL ;
2010-02-12 08:02:18 +03:00
if ( ret )
dev_warn ( port - > dev , " Errors adding %d buffers back to vq \n " ,
ret ) ;
2009-12-21 19:45:30 +03:00
}
2010-01-18 16:45:12 +03:00
static bool port_has_data ( struct port * port )
{
unsigned long flags ;
bool ret ;
spin_lock_irqsave ( & port - > inbuf_lock , flags ) ;
2010-02-12 08:02:18 +03:00
if ( port - > inbuf ) {
2010-01-18 16:45:12 +03:00
ret = true ;
2010-02-12 08:02:18 +03:00
goto out ;
}
port - > inbuf = get_inbuf ( port ) ;
if ( port - > inbuf ) {
ret = true ;
goto out ;
}
ret = false ;
out :
2010-01-18 16:45:12 +03:00
spin_unlock_irqrestore ( & port - > inbuf_lock , flags ) ;
return ret ;
}
2009-12-21 18:33:25 +03:00
static ssize_t send_control_msg ( struct port * port , unsigned int event ,
unsigned int value )
{
struct scatterlist sg [ 1 ] ;
struct virtio_console_control cpkt ;
struct virtqueue * vq ;
2010-02-24 08:06:51 +03:00
unsigned int len ;
2009-12-21 18:33:25 +03:00
if ( ! use_multiport ( port - > portdev ) )
return 0 ;
cpkt . id = port - > id ;
cpkt . event = event ;
cpkt . value = value ;
vq = port - > portdev - > c_ovq ;
sg_init_one ( sg , & cpkt , sizeof ( cpkt ) ) ;
if ( vq - > vq_ops - > add_buf ( vq , sg , 1 , 0 , & cpkt ) > = 0 ) {
vq - > vq_ops - > kick ( vq ) ;
while ( ! vq - > vq_ops - > get_buf ( vq , & len ) )
cpu_relax ( ) ;
}
return 0 ;
}
2009-12-21 14:58:51 +03:00
static ssize_t send_buf ( struct port * port , void * in_buf , size_t in_count )
{
struct scatterlist sg [ 1 ] ;
struct virtqueue * out_vq ;
ssize_t ret ;
unsigned int len ;
out_vq = port - > out_vq ;
sg_init_one ( sg , in_buf , in_count ) ;
ret = out_vq - > vq_ops - > add_buf ( out_vq , sg , 1 , 0 , in_buf ) ;
/* Tell Host to go! */
out_vq - > vq_ops - > kick ( out_vq ) ;
if ( ret < 0 ) {
len = 0 ;
goto fail ;
}
/*
* Wait till the host acknowledges it pushed out the data we
* sent . Also ensure we return to userspace the number of
* bytes that were successfully consumed by the host .
*/
while ( ! out_vq - > vq_ops - > get_buf ( out_vq , & len ) )
cpu_relax ( ) ;
fail :
/* We're expected to return the amount of data we wrote */
return len ;
}
2010-01-18 16:45:12 +03:00
/*
* Give out the data that ' s requested from the buffer that we have
* queued up .
*/
2009-12-21 18:56:45 +03:00
static ssize_t fill_readbuf ( struct port * port , char * out_buf , size_t out_count ,
bool to_user )
2010-01-18 16:45:12 +03:00
{
struct port_buffer * buf ;
unsigned long flags ;
if ( ! out_count | | ! port_has_data ( port ) )
return 0 ;
buf = port - > inbuf ;
2009-12-21 18:56:45 +03:00
out_count = min ( out_count , buf - > len - buf - > offset ) ;
2010-01-18 16:45:12 +03:00
2009-12-21 18:56:45 +03:00
if ( to_user ) {
ssize_t ret ;
ret = copy_to_user ( out_buf , buf - > buf + buf - > offset , out_count ) ;
if ( ret )
return - EFAULT ;
} else {
memcpy ( out_buf , buf - > buf + buf - > offset , out_count ) ;
}
2010-01-18 16:45:12 +03:00
buf - > offset + = out_count ;
if ( buf - > offset = = buf - > len ) {
/*
* We ' re done using all the data in this buffer .
* Re - queue so that the Host can send us more data .
*/
spin_lock_irqsave ( & port - > inbuf_lock , flags ) ;
port - > inbuf = NULL ;
if ( add_inbuf ( port - > in_vq , buf ) < 0 )
2009-12-21 19:06:04 +03:00
dev_warn ( port - > dev , " failed add_buf \n " ) ;
2010-01-18 16:45:12 +03:00
spin_unlock_irqrestore ( & port - > inbuf_lock , flags ) ;
}
2009-12-21 18:56:45 +03:00
/* Return the number of bytes actually copied */
2010-01-18 16:45:12 +03:00
return out_count ;
2010-01-18 16:45:02 +03:00
}
2009-12-21 19:19:30 +03:00
/* The condition that must be true for polling to end */
static bool wait_is_over ( struct port * port )
{
return port_has_data ( port ) | | ! port - > host_connected ;
}
static ssize_t port_fops_read ( struct file * filp , char __user * ubuf ,
size_t count , loff_t * offp )
{
struct port * port ;
ssize_t ret ;
port = filp - > private_data ;
if ( ! port_has_data ( port ) ) {
/*
* If nothing ' s connected on the host just return 0 in
* case of list_empty ; this tells the userspace app
* that there ' s no connection
*/
if ( ! port - > host_connected )
return 0 ;
if ( filp - > f_flags & O_NONBLOCK )
return - EAGAIN ;
ret = wait_event_interruptible ( port - > waitqueue ,
wait_is_over ( port ) ) ;
if ( ret < 0 )
return ret ;
}
/*
* We could ' ve received a disconnection message while we were
* waiting for more data .
*
* This check is not clubbed in the if ( ) statement above as we
* might receive some data as well as the host could get
* disconnected after we got woken up from our wait . So we
* really want to give off whatever data we have and only then
* check for host_connected .
*/
if ( ! port_has_data ( port ) & & ! port - > host_connected )
return 0 ;
return fill_readbuf ( port , ubuf , count , true ) ;
}
static ssize_t port_fops_write ( struct file * filp , const char __user * ubuf ,
size_t count , loff_t * offp )
{
struct port * port ;
char * buf ;
ssize_t ret ;
port = filp - > private_data ;
count = min ( ( size_t ) ( 32 * 1024 ) , count ) ;
buf = kmalloc ( count , GFP_KERNEL ) ;
if ( ! buf )
return - ENOMEM ;
ret = copy_from_user ( buf , ubuf , count ) ;
if ( ret ) {
ret = - EFAULT ;
goto free_buf ;
}
ret = send_buf ( port , buf , count ) ;
free_buf :
kfree ( buf ) ;
return ret ;
}
static unsigned int port_fops_poll ( struct file * filp , poll_table * wait )
{
struct port * port ;
unsigned int ret ;
port = filp - > private_data ;
poll_wait ( filp , & port - > waitqueue , wait ) ;
ret = 0 ;
if ( port - > inbuf )
ret | = POLLIN | POLLRDNORM ;
if ( port - > host_connected )
ret | = POLLOUT ;
if ( ! port - > host_connected )
ret | = POLLHUP ;
return ret ;
}
static int port_fops_release ( struct inode * inode , struct file * filp )
{
struct port * port ;
port = filp - > private_data ;
/* Notify host of port being closed */
send_control_msg ( port , VIRTIO_CONSOLE_PORT_OPEN , 0 ) ;
2009-12-21 19:45:30 +03:00
spin_lock_irq ( & port - > inbuf_lock ) ;
2009-11-26 08:55:38 +03:00
port - > guest_connected = false ;
2009-12-21 19:45:30 +03:00
discard_port_data ( port ) ;
spin_unlock_irq ( & port - > inbuf_lock ) ;
2009-12-21 19:19:30 +03:00
return 0 ;
}
static int port_fops_open ( struct inode * inode , struct file * filp )
{
struct cdev * cdev = inode - > i_cdev ;
struct port * port ;
port = container_of ( cdev , struct port , cdev ) ;
filp - > private_data = port ;
/*
* Don ' t allow opening of console port devices - - that ' s done
* via / dev / hvc
*/
if ( is_console_port ( port ) )
return - ENXIO ;
2009-11-26 08:55:38 +03:00
/* Allow only one process to open a particular port at a time */
spin_lock_irq ( & port - > inbuf_lock ) ;
if ( port - > guest_connected ) {
spin_unlock_irq ( & port - > inbuf_lock ) ;
return - EMFILE ;
}
port - > guest_connected = true ;
spin_unlock_irq ( & port - > inbuf_lock ) ;
2009-12-21 19:19:30 +03:00
/* Notify host of port being opened */
send_control_msg ( filp - > private_data , VIRTIO_CONSOLE_PORT_OPEN , 1 ) ;
return 0 ;
}
/*
* The file operations that we support : programs in the guest can open
* a console device , read from it , write to it , poll for data and
* close it . The devices are at
* / dev / vport < device number > p < port number >
*/
static const struct file_operations port_fops = {
. owner = THIS_MODULE ,
. open = port_fops_open ,
. read = port_fops_read ,
. write = port_fops_write ,
. poll = port_fops_poll ,
. release = port_fops_release ,
} ;
2010-01-18 16:44:55 +03:00
/*
* The put_chars ( ) callback is pretty straightforward .
2007-10-22 05:03:39 +04:00
*
2010-01-18 16:44:55 +03:00
* We turn the characters into a scatter - gather list , add it to the
* output queue and then kick the Host . Then we sit here waiting for
* it to finish : inefficient in theory , but in practice
* implementations will do it immediately ( lguest ' s Launcher does ) .
*/
2007-10-22 05:03:39 +04:00
static int put_chars ( u32 vtermno , const char * buf , int count )
{
2010-01-18 16:45:00 +03:00
struct port * port ;
2010-01-18 16:45:05 +03:00
port = find_port_by_vtermno ( vtermno ) ;
if ( ! port )
return 0 ;
2007-10-22 05:03:39 +04:00
2010-01-18 16:44:56 +03:00
if ( unlikely ( early_put_chars ) )
return early_put_chars ( vtermno , buf , count ) ;
2009-12-21 14:58:51 +03:00
return send_buf ( port , ( void * ) buf , count ) ;
2007-10-22 05:03:39 +04:00
}
2010-01-18 16:44:55 +03:00
/*
* get_chars ( ) is the callback from the hvc_console infrastructure
* when an interrupt is received .
2007-10-22 05:03:39 +04:00
*
2010-01-18 16:45:12 +03:00
* We call out to fill_readbuf that gets us the required data from the
* buffers that are queued up .
2010-01-18 16:44:55 +03:00
*/
2007-10-22 05:03:39 +04:00
static int get_chars ( u32 vtermno , char * buf , int count )
{
2010-01-18 16:45:00 +03:00
struct port * port ;
2010-01-18 16:45:05 +03:00
port = find_port_by_vtermno ( vtermno ) ;
if ( ! port )
return 0 ;
2010-01-18 16:45:00 +03:00
2007-10-22 05:03:39 +04:00
/* If we don't have an input queue yet, we can't get input. */
2010-01-18 16:45:00 +03:00
BUG_ON ( ! port - > in_vq ) ;
2007-10-22 05:03:39 +04:00
2009-12-21 18:56:45 +03:00
return fill_readbuf ( port , buf , count , false ) ;
2007-10-22 05:03:39 +04:00
}
2010-01-18 16:45:08 +03:00
static void resize_console ( struct port * port )
2008-11-25 15:36:26 +03:00
{
2010-01-18 16:45:08 +03:00
struct virtio_device * vdev ;
2008-11-25 15:36:26 +03:00
struct winsize ws ;
2010-01-18 16:45:08 +03:00
vdev = port - > portdev - > vdev ;
if ( virtio_has_feature ( vdev , VIRTIO_CONSOLE_F_SIZE ) ) {
vdev - > config - > get ( vdev ,
offsetof ( struct virtio_console_config , cols ) ,
& ws . ws_col , sizeof ( u16 ) ) ;
vdev - > config - > get ( vdev ,
offsetof ( struct virtio_console_config , rows ) ,
& ws . ws_row , sizeof ( u16 ) ) ;
2010-01-18 16:45:09 +03:00
hvc_resize ( port - > cons . hvc , ws ) ;
2008-11-25 15:36:26 +03:00
}
}
2010-01-18 16:45:05 +03:00
/* We set the configuration at this point, since we now have a tty */
2008-06-20 17:24:15 +04:00
static int notifier_add_vio ( struct hvc_struct * hp , int data )
{
2010-01-18 16:45:05 +03:00
struct port * port ;
port = find_port_by_vtermno ( hp - > vtermno ) ;
if ( ! port )
return - EINVAL ;
2008-06-20 17:24:15 +04:00
hp - > irq_requested = 1 ;
2010-01-18 16:45:08 +03:00
resize_console ( port ) ;
2008-11-25 15:36:26 +03:00
2008-06-20 17:24:15 +04:00
return 0 ;
}
static void notifier_del_vio ( struct hvc_struct * hp , int data )
{
hp - > irq_requested = 0 ;
}
2009-12-21 18:33:25 +03:00
/* The operations for console ports. */
2009-11-28 09:50:26 +03:00
static const struct hv_ops hv_ops = {
2010-01-18 16:44:56 +03:00
. get_chars = get_chars ,
. put_chars = put_chars ,
. notifier_add = notifier_add_vio ,
. notifier_del = notifier_del_vio ,
. notifier_hangup = notifier_del_vio ,
} ;
/*
* Console drivers are initialized very early so boot messages can go
* out , so we do things slightly differently from the generic virtio
* initialization of the net and block drivers .
*
* At this stage , the console is output - only . It ' s too early to set
* up a virtqueue , so we let the drivers do some boutique early - output
* thing .
*/
int __init virtio_cons_early_init ( int ( * put_chars ) ( u32 , const char * , int ) )
{
early_put_chars = put_chars ;
return hvc_instantiate ( 0 , 0 , & hv_ops ) ;
}
2009-12-21 18:33:25 +03:00
int init_port_console ( struct port * port )
2010-01-18 16:45:10 +03:00
{
int ret ;
/*
* The Host ' s telling us this port is a console port . Hook it
* up with an hvc console .
*
* To set up and manage our virtual console , we call
* hvc_alloc ( ) .
*
* The first argument of hvc_alloc ( ) is the virtual console
* number . The second argument is the parameter for the
* notification mechanism ( like irq number ) . We currently
* leave this as zero , virtqueues have implicit notifications .
*
* The third argument is a " struct hv_ops " containing the
* put_chars ( ) get_chars ( ) , notifier_add ( ) and notifier_del ( )
* pointers . The final argument is the output buffer size : we
* can do any size , so we put PAGE_SIZE here .
*/
port - > cons . vtermno = pdrvdata . next_vtermno ;
port - > cons . hvc = hvc_alloc ( port - > cons . vtermno , 0 , & hv_ops , PAGE_SIZE ) ;
if ( IS_ERR ( port - > cons . hvc ) ) {
ret = PTR_ERR ( port - > cons . hvc ) ;
2010-01-18 14:05:23 +03:00
dev_err ( port - > dev ,
" error %d allocating hvc for port \n " , ret ) ;
2010-01-18 16:45:10 +03:00
port - > cons . hvc = NULL ;
return ret ;
}
spin_lock_irq ( & pdrvdata_lock ) ;
pdrvdata . next_vtermno + + ;
list_add_tail ( & port - > cons . list , & pdrvdata . consoles ) ;
spin_unlock_irq ( & pdrvdata_lock ) ;
2009-11-26 08:55:38 +03:00
port - > guest_connected = true ;
2010-01-18 16:45:10 +03:00
2009-12-21 19:19:30 +03:00
/* Notify host of port being opened */
send_control_msg ( port , VIRTIO_CONSOLE_PORT_OPEN , 1 ) ;
2010-01-18 16:45:10 +03:00
return 0 ;
}
2009-12-21 19:27:40 +03:00
static ssize_t show_port_name ( struct device * dev ,
struct device_attribute * attr , char * buffer )
{
struct port * port ;
port = dev_get_drvdata ( dev ) ;
return sprintf ( buffer , " %s \n " , port - > name ) ;
}
static DEVICE_ATTR ( name , S_IRUGO , show_port_name , NULL ) ;
static struct attribute * port_sysfs_entries [ ] = {
& dev_attr_name . attr ,
NULL
} ;
static struct attribute_group port_attribute_group = {
. name = NULL , /* put in device directory */
. attrs = port_sysfs_entries ,
} ;
2009-12-21 20:06:21 +03:00
static int debugfs_open ( struct inode * inode , struct file * filp )
{
filp - > private_data = inode - > i_private ;
return 0 ;
}
static ssize_t debugfs_read ( struct file * filp , char __user * ubuf ,
size_t count , loff_t * offp )
{
struct port * port ;
char * buf ;
ssize_t ret , out_offset , out_count ;
out_count = 1024 ;
buf = kmalloc ( out_count , GFP_KERNEL ) ;
if ( ! buf )
return - ENOMEM ;
port = filp - > private_data ;
out_offset = 0 ;
out_offset + = snprintf ( buf + out_offset , out_count ,
" name: %s \n " , port - > name ? port - > name : " " ) ;
out_offset + = snprintf ( buf + out_offset , out_count - out_offset ,
" guest_connected: %d \n " , port - > guest_connected ) ;
out_offset + = snprintf ( buf + out_offset , out_count - out_offset ,
" host_connected: %d \n " , port - > host_connected ) ;
out_offset + = snprintf ( buf + out_offset , out_count - out_offset ,
" is_console: %s \n " ,
is_console_port ( port ) ? " yes " : " no " ) ;
out_offset + = snprintf ( buf + out_offset , out_count - out_offset ,
" console_vtermno: %u \n " , port - > cons . vtermno ) ;
ret = simple_read_from_buffer ( ubuf , count , offp , buf , out_offset ) ;
kfree ( buf ) ;
return ret ;
}
static const struct file_operations port_debugfs_ops = {
. owner = THIS_MODULE ,
. open = debugfs_open ,
. read = debugfs_read ,
} ;
2009-12-21 19:57:31 +03:00
/* Remove all port-specific data. */
static int remove_port ( struct port * port )
{
2010-02-12 08:02:15 +03:00
struct port_buffer * buf ;
2009-12-21 19:57:31 +03:00
spin_lock_irq ( & port - > portdev - > ports_lock ) ;
list_del ( & port - > list ) ;
spin_unlock_irq ( & port - > portdev - > ports_lock ) ;
if ( is_console_port ( port ) ) {
spin_lock_irq ( & pdrvdata_lock ) ;
list_del ( & port - > cons . list ) ;
spin_unlock_irq ( & pdrvdata_lock ) ;
hvc_remove ( port - > cons . hvc ) ;
}
if ( port - > guest_connected )
send_control_msg ( port , VIRTIO_CONSOLE_PORT_OPEN , 0 ) ;
sysfs_remove_group ( & port - > dev - > kobj , & port_attribute_group ) ;
device_destroy ( pdrvdata . class , port - > dev - > devt ) ;
cdev_del ( & port - > cdev ) ;
2010-02-12 08:02:15 +03:00
/* Remove unused data this port might have received. */
2009-12-21 19:57:31 +03:00
discard_port_data ( port ) ;
2010-02-12 08:02:15 +03:00
/* Remove buffers we queued up for the Host to send us data in. */
while ( ( buf = port - > in_vq - > vq_ops - > detach_unused_buf ( port - > in_vq ) ) )
free_buf ( buf ) ;
2009-12-21 19:57:31 +03:00
kfree ( port - > name ) ;
2009-12-21 20:06:21 +03:00
debugfs_remove ( port - > debugfs_file ) ;
2009-12-21 19:57:31 +03:00
kfree ( port ) ;
return 0 ;
}
2009-12-21 18:33:25 +03:00
/* Any private messages that the Host and Guest want to share */
static void handle_control_message ( struct ports_device * portdev ,
struct port_buffer * buf )
{
struct virtio_console_control * cpkt ;
struct port * port ;
2009-12-21 19:27:40 +03:00
size_t name_size ;
int err ;
2009-12-21 18:33:25 +03:00
cpkt = ( struct virtio_console_control * ) ( buf - > buf + buf - > offset ) ;
port = find_port_by_id ( portdev , cpkt - > id ) ;
if ( ! port ) {
/* No valid header at start of buffer. Drop it. */
dev_dbg ( & portdev - > vdev - > dev ,
" Invalid index %u in control packet \n " , cpkt - > id ) ;
return ;
}
switch ( cpkt - > event ) {
case VIRTIO_CONSOLE_CONSOLE_PORT :
if ( ! cpkt - > value )
break ;
if ( is_console_port ( port ) )
break ;
init_port_console ( port ) ;
/*
* Could remove the port here in case init fails - but
* have to notify the host first .
*/
break ;
case VIRTIO_CONSOLE_RESIZE :
if ( ! is_console_port ( port ) )
break ;
port - > cons . hvc - > irq_requested = 1 ;
resize_console ( port ) ;
break ;
2009-12-21 19:19:30 +03:00
case VIRTIO_CONSOLE_PORT_OPEN :
port - > host_connected = cpkt - > value ;
wake_up_interruptible ( & port - > waitqueue ) ;
break ;
2009-12-21 19:27:40 +03:00
case VIRTIO_CONSOLE_PORT_NAME :
/*
* Skip the size of the header and the cpkt to get the size
* of the name that was sent
*/
name_size = buf - > len - buf - > offset - sizeof ( * cpkt ) + 1 ;
port - > name = kmalloc ( name_size , GFP_KERNEL ) ;
if ( ! port - > name ) {
dev_err ( port - > dev ,
" Not enough space to store port name \n " ) ;
break ;
}
strncpy ( port - > name , buf - > buf + buf - > offset + sizeof ( * cpkt ) ,
name_size - 1 ) ;
port - > name [ name_size - 1 ] = 0 ;
/*
* Since we only have one sysfs attribute , ' name ' ,
* create it only if we have a name for the port .
*/
err = sysfs_create_group ( & port - > dev - > kobj ,
& port_attribute_group ) ;
if ( err )
dev_err ( port - > dev ,
" Error %d creating sysfs device attributes \n " ,
err ) ;
break ;
2009-12-21 19:57:31 +03:00
case VIRTIO_CONSOLE_PORT_REMOVE :
/*
* Hot unplug the port . We don ' t decrement nr_ports
* since we don ' t want to deal with extra complexities
* of using the lowest - available port id : We can just
* pick up the nr_ports number as the id and not have
* userspace send it to us . This helps us in two
* ways :
*
* - We don ' t need to have a ' port_id ' field in the
* config space when a port is hot - added . This is a
* good thing as we might queue up multiple hotplug
* requests issued in our workqueue .
*
* - Another way to deal with this would have been to
* use a bitmap of the active ports and select the
* lowest non - active port from that map . That
* bloats the already tight config space and we
* would end up artificially limiting the
* max . number of ports to sizeof ( bitmap ) . Right
* now we can support 2 ^ 32 ports ( as the port id is
* stored in a u32 type ) .
*
*/
remove_port ( port ) ;
break ;
2009-12-21 18:33:25 +03:00
}
}
static void control_work_handler ( struct work_struct * work )
{
struct ports_device * portdev ;
struct virtqueue * vq ;
struct port_buffer * buf ;
unsigned int len ;
portdev = container_of ( work , struct ports_device , control_work ) ;
vq = portdev - > c_ivq ;
spin_lock ( & portdev - > cvq_lock ) ;
while ( ( buf = vq - > vq_ops - > get_buf ( vq , & len ) ) ) {
spin_unlock ( & portdev - > cvq_lock ) ;
buf - > len = len ;
buf - > offset = 0 ;
handle_control_message ( portdev , buf ) ;
spin_lock ( & portdev - > cvq_lock ) ;
if ( add_inbuf ( portdev - > c_ivq , buf ) < 0 ) {
dev_warn ( & portdev - > vdev - > dev ,
" Error adding buffer to queue \n " ) ;
free_buf ( buf ) ;
}
}
spin_unlock ( & portdev - > cvq_lock ) ;
}
static void in_intr ( struct virtqueue * vq )
{
struct port * port ;
unsigned long flags ;
port = find_port_by_vq ( vq - > vdev - > priv , vq ) ;
if ( ! port )
return ;
spin_lock_irqsave ( & port - > inbuf_lock , flags ) ;
2010-02-12 08:02:18 +03:00
if ( ! port - > inbuf )
port - > inbuf = get_inbuf ( port ) ;
2009-12-21 18:33:25 +03:00
2009-12-21 19:45:30 +03:00
/*
* Don ' t queue up data when port is closed . This condition
* can be reached when a console port is not yet connected ( no
* tty is spawned ) and the host sends out data to console
* ports . For generic serial ports , the host won ' t
* ( shouldn ' t ) send data till the guest is connected .
*/
if ( ! port - > guest_connected )
discard_port_data ( port ) ;
2009-12-21 18:33:25 +03:00
spin_unlock_irqrestore ( & port - > inbuf_lock , flags ) ;
2009-12-21 19:19:30 +03:00
wake_up_interruptible ( & port - > waitqueue ) ;
2009-12-21 18:33:25 +03:00
if ( is_console_port ( port ) & & hvc_poll ( port - > cons . hvc ) )
hvc_kick ( ) ;
}
static void control_intr ( struct virtqueue * vq )
{
struct ports_device * portdev ;
portdev = vq - > vdev - > priv ;
schedule_work ( & portdev - > control_work ) ;
}
2009-12-21 19:52:08 +03:00
static void config_intr ( struct virtio_device * vdev )
{
struct ports_device * portdev ;
portdev = vdev - > priv ;
if ( use_multiport ( portdev ) ) {
/* Handle port hot-add */
schedule_work ( & portdev - > config_work ) ;
}
/*
* We ' ll use this way of resizing only for legacy support .
* For newer userspace ( VIRTIO_CONSOLE_F_MULTPORT + ) , use
* control messages to indicate console size changes so that
* it can be done per - port
*/
resize_console ( find_port_by_id ( portdev , 0 ) ) ;
}
2010-02-12 08:02:17 +03:00
static unsigned int fill_queue ( struct virtqueue * vq , spinlock_t * lock )
2009-12-21 18:33:25 +03:00
{
struct port_buffer * buf ;
2010-02-12 08:02:17 +03:00
unsigned int ret ;
int err ;
2009-12-21 18:33:25 +03:00
2010-02-12 08:02:17 +03:00
ret = 0 ;
2009-12-21 18:33:25 +03:00
do {
buf = alloc_buf ( PAGE_SIZE ) ;
if ( ! buf )
break ;
spin_lock_irq ( lock ) ;
2010-02-12 08:02:17 +03:00
err = add_inbuf ( vq , buf ) ;
if ( err < 0 ) {
2009-12-21 18:33:25 +03:00
spin_unlock_irq ( lock ) ;
free_buf ( buf ) ;
break ;
}
2010-02-12 08:02:17 +03:00
ret + + ;
2009-12-21 18:33:25 +03:00
spin_unlock_irq ( lock ) ;
2010-02-12 08:02:17 +03:00
} while ( err > 0 ) ;
return ret ;
2009-12-21 18:33:25 +03:00
}
static int add_port ( struct ports_device * portdev , u32 id )
2010-01-18 16:45:06 +03:00
{
2009-12-21 20:06:21 +03:00
char debugfs_name [ 16 ] ;
2010-01-18 16:45:00 +03:00
struct port * port ;
2010-02-12 08:02:18 +03:00
struct port_buffer * buf ;
2009-12-21 19:06:04 +03:00
dev_t devt ;
2007-10-22 05:03:39 +04:00
int err ;
2010-01-18 16:45:07 +03:00
port = kmalloc ( sizeof ( * port ) , GFP_KERNEL ) ;
2010-01-18 16:45:06 +03:00
if ( ! port ) {
err = - ENOMEM ;
goto fail ;
2010-01-18 16:44:59 +03:00
}
2010-01-18 16:45:04 +03:00
2010-01-18 16:45:07 +03:00
port - > portdev = portdev ;
2009-12-21 18:33:25 +03:00
port - > id = id ;
2010-01-18 16:45:12 +03:00
2009-12-21 19:27:40 +03:00
port - > name = NULL ;
2010-01-18 16:45:12 +03:00
port - > inbuf = NULL ;
2009-12-21 18:33:25 +03:00
port - > cons . hvc = NULL ;
2010-01-18 16:45:12 +03:00
2009-11-26 08:55:38 +03:00
port - > host_connected = port - > guest_connected = false ;
2009-12-21 19:19:30 +03:00
2009-12-21 18:33:25 +03:00
port - > in_vq = portdev - > in_vqs [ port - > id ] ;
port - > out_vq = portdev - > out_vqs [ port - > id ] ;
2007-10-22 05:03:39 +04:00
2009-12-21 19:19:30 +03:00
cdev_init ( & port - > cdev , & port_fops ) ;
2009-12-21 19:06:04 +03:00
devt = MKDEV ( portdev - > chr_major , id ) ;
err = cdev_add ( & port - > cdev , devt , 1 ) ;
if ( err < 0 ) {
dev_err ( & port - > portdev - > vdev - > dev ,
" Error %d adding cdev for port %u \n " , err , id ) ;
goto free_port ;
}
port - > dev = device_create ( pdrvdata . class , & port - > portdev - > vdev - > dev ,
devt , port , " vport%up%u " ,
port - > portdev - > drv_index , id ) ;
if ( IS_ERR ( port - > dev ) ) {
err = PTR_ERR ( port - > dev ) ;
dev_err ( & port - > portdev - > vdev - > dev ,
" Error %d creating device for port %u \n " ,
err , id ) ;
goto free_cdev ;
}
2010-01-18 16:45:12 +03:00
spin_lock_init ( & port - > inbuf_lock ) ;
2009-12-21 19:19:30 +03:00
init_waitqueue_head ( & port - > waitqueue ) ;
2010-01-18 16:45:12 +03:00
2010-02-12 08:02:18 +03:00
/* Fill the in_vq with buffers so the host can send us data. */
err = fill_queue ( port - > in_vq , & port - > inbuf_lock ) ;
if ( ! err ) {
dev_err ( port - > dev , " Error allocating inbufs \n " ) ;
2010-01-18 16:45:07 +03:00
err = - ENOMEM ;
2009-12-21 19:06:04 +03:00
goto free_device ;
2010-01-18 16:45:07 +03:00
}
2007-10-22 05:03:39 +04:00
2009-12-21 18:33:25 +03:00
/*
* If we ' re not using multiport support , this has to be a console port
*/
if ( ! use_multiport ( port - > portdev ) ) {
err = init_port_console ( port ) ;
if ( err )
2010-02-12 08:02:18 +03:00
goto free_inbufs ;
2009-12-21 18:33:25 +03:00
}
spin_lock_irq ( & portdev - > ports_lock ) ;
list_add_tail ( & port - > list , & port - > portdev - > ports ) ;
spin_unlock_irq ( & portdev - > ports_lock ) ;
/*
* Tell the Host we ' re set so that it can send us various
* configuration parameters for this port ( eg , port name ,
* caching , whether this is a console port , etc . )
*/
send_control_msg ( port , VIRTIO_CONSOLE_PORT_READY , 1 ) ;
2010-01-18 16:45:05 +03:00
2009-12-21 20:06:21 +03:00
if ( pdrvdata . debugfs_dir ) {
/*
* Finally , create the debugfs file that we can use to
* inspect a port ' s state at any time
*/
sprintf ( debugfs_name , " vport%up%u " ,
port - > portdev - > drv_index , id ) ;
port - > debugfs_file = debugfs_create_file ( debugfs_name , 0444 ,
pdrvdata . debugfs_dir ,
port ,
& port_debugfs_ops ) ;
}
2010-01-18 16:45:07 +03:00
return 0 ;
2010-02-12 08:02:18 +03:00
free_inbufs :
while ( ( buf = port - > in_vq - > vq_ops - > detach_unused_buf ( port - > in_vq ) ) )
free_buf ( buf ) ;
2009-12-21 19:06:04 +03:00
free_device :
device_destroy ( pdrvdata . class , port - > dev - > devt ) ;
free_cdev :
cdev_del ( & port - > cdev ) ;
2010-01-18 16:45:07 +03:00
free_port :
kfree ( port ) ;
fail :
return err ;
}
2009-12-21 19:52:08 +03:00
/*
* The workhandler for config - space updates .
*
* This is called when ports are hot - added .
*/
static void config_work_handler ( struct work_struct * work )
{
struct virtio_console_config virtconconf ;
struct ports_device * portdev ;
struct virtio_device * vdev ;
int err ;
portdev = container_of ( work , struct ports_device , config_work ) ;
vdev = portdev - > vdev ;
vdev - > config - > get ( vdev ,
offsetof ( struct virtio_console_config , nr_ports ) ,
& virtconconf . nr_ports ,
sizeof ( virtconconf . nr_ports ) ) ;
if ( portdev - > config . nr_ports = = virtconconf . nr_ports ) {
/*
* Port 0 got hot - added . Since we already did all the
* other initialisation for it , just tell the Host
2009-12-21 19:57:31 +03:00
* that the port is ready if we find the port . In
* case the port was hot - removed earlier , we call
* add_port to add the port .
2009-12-21 19:52:08 +03:00
*/
struct port * port ;
port = find_port_by_id ( portdev , 0 ) ;
2009-12-21 19:57:31 +03:00
if ( ! port )
add_port ( portdev , 0 ) ;
else
send_control_msg ( port , VIRTIO_CONSOLE_PORT_READY , 1 ) ;
2009-12-21 19:52:08 +03:00
return ;
}
if ( virtconconf . nr_ports > portdev - > config . max_nr_ports ) {
dev_warn ( & vdev - > dev ,
" More ports specified (%u) than allowed (%u) " ,
portdev - > config . nr_ports + 1 ,
portdev - > config . max_nr_ports ) ;
return ;
}
if ( virtconconf . nr_ports < portdev - > config . nr_ports )
return ;
/* Hot-add ports */
while ( virtconconf . nr_ports - portdev - > config . nr_ports ) {
err = add_port ( portdev , portdev - > config . nr_ports ) ;
if ( err )
break ;
portdev - > config . nr_ports + + ;
}
}
2010-01-18 16:45:11 +03:00
static int init_vqs ( struct ports_device * portdev )
{
vq_callback_t * * io_callbacks ;
char * * io_names ;
struct virtqueue * * vqs ;
2009-12-21 18:33:25 +03:00
u32 i , j , nr_ports , nr_queues ;
2010-01-18 16:45:11 +03:00
int err ;
2009-12-21 18:33:25 +03:00
nr_ports = portdev - > config . max_nr_ports ;
nr_queues = use_multiport ( portdev ) ? ( nr_ports + 1 ) * 2 : 2 ;
2010-01-18 16:45:11 +03:00
vqs = kmalloc ( nr_queues * sizeof ( struct virtqueue * ) , GFP_KERNEL ) ;
if ( ! vqs ) {
err = - ENOMEM ;
goto fail ;
}
io_callbacks = kmalloc ( nr_queues * sizeof ( vq_callback_t * ) , GFP_KERNEL ) ;
if ( ! io_callbacks ) {
err = - ENOMEM ;
goto free_vqs ;
}
io_names = kmalloc ( nr_queues * sizeof ( char * ) , GFP_KERNEL ) ;
if ( ! io_names ) {
err = - ENOMEM ;
goto free_callbacks ;
}
portdev - > in_vqs = kmalloc ( nr_ports * sizeof ( struct virtqueue * ) ,
GFP_KERNEL ) ;
if ( ! portdev - > in_vqs ) {
err = - ENOMEM ;
goto free_names ;
}
portdev - > out_vqs = kmalloc ( nr_ports * sizeof ( struct virtqueue * ) ,
GFP_KERNEL ) ;
if ( ! portdev - > out_vqs ) {
err = - ENOMEM ;
goto free_invqs ;
}
2009-12-21 18:33:25 +03:00
/*
* For backward compat ( newer host but older guest ) , the host
* spawns a console port first and also inits the vqs for port
* 0 before others .
*/
j = 0 ;
io_callbacks [ j ] = in_intr ;
io_callbacks [ j + 1 ] = NULL ;
io_names [ j ] = " input " ;
io_names [ j + 1 ] = " output " ;
j + = 2 ;
if ( use_multiport ( portdev ) ) {
io_callbacks [ j ] = control_intr ;
io_callbacks [ j + 1 ] = NULL ;
io_names [ j ] = " control-i " ;
io_names [ j + 1 ] = " control-o " ;
for ( i = 1 ; i < nr_ports ; i + + ) {
j + = 2 ;
io_callbacks [ j ] = in_intr ;
io_callbacks [ j + 1 ] = NULL ;
io_names [ j ] = " input " ;
io_names [ j + 1 ] = " output " ;
}
}
2010-01-18 16:45:11 +03:00
/* Find the queues. */
err = portdev - > vdev - > config - > find_vqs ( portdev - > vdev , nr_queues , vqs ,
io_callbacks ,
( const char * * ) io_names ) ;
if ( err )
goto free_outvqs ;
2009-12-21 18:33:25 +03:00
j = 0 ;
2010-01-18 16:45:11 +03:00
portdev - > in_vqs [ 0 ] = vqs [ 0 ] ;
portdev - > out_vqs [ 0 ] = vqs [ 1 ] ;
2009-12-21 18:33:25 +03:00
j + = 2 ;
if ( use_multiport ( portdev ) ) {
portdev - > c_ivq = vqs [ j ] ;
portdev - > c_ovq = vqs [ j + 1 ] ;
for ( i = 1 ; i < nr_ports ; i + + ) {
j + = 2 ;
portdev - > in_vqs [ i ] = vqs [ j ] ;
portdev - > out_vqs [ i ] = vqs [ j + 1 ] ;
}
}
2010-01-18 16:45:11 +03:00
kfree ( io_callbacks ) ;
kfree ( io_names ) ;
kfree ( vqs ) ;
return 0 ;
free_names :
kfree ( io_names ) ;
free_callbacks :
kfree ( io_callbacks ) ;
free_outvqs :
kfree ( portdev - > out_vqs ) ;
free_invqs :
kfree ( portdev - > in_vqs ) ;
free_vqs :
kfree ( vqs ) ;
fail :
return err ;
}
2009-12-21 19:06:04 +03:00
static const struct file_operations portdev_fops = {
. owner = THIS_MODULE ,
} ;
2010-01-18 16:45:07 +03:00
/*
* Once we ' re further in boot , we get probed like any other virtio
* device .
2009-12-21 18:33:25 +03:00
*
* If the host also supports multiple console ports , we check the
* config space to see how many ports the host has spawned . We
* initialize each port found .
2010-01-18 16:45:07 +03:00
*/
static int __devinit virtcons_probe ( struct virtio_device * vdev )
{
struct ports_device * portdev ;
2009-12-21 18:33:25 +03:00
u32 i ;
2010-01-18 16:45:07 +03:00
int err ;
2009-12-21 18:33:25 +03:00
bool multiport ;
2010-01-18 16:45:07 +03:00
portdev = kmalloc ( sizeof ( * portdev ) , GFP_KERNEL ) ;
if ( ! portdev ) {
err = - ENOMEM ;
goto fail ;
}
/* Attach this portdev to this virtio_device, and vice-versa. */
portdev - > vdev = vdev ;
vdev - > priv = portdev ;
2009-12-21 19:06:04 +03:00
spin_lock_irq ( & pdrvdata_lock ) ;
portdev - > drv_index = pdrvdata . index + + ;
spin_unlock_irq ( & pdrvdata_lock ) ;
portdev - > chr_major = register_chrdev ( 0 , " virtio-portsdev " ,
& portdev_fops ) ;
if ( portdev - > chr_major < 0 ) {
dev_err ( & vdev - > dev ,
" Error %d registering chrdev for device %u \n " ,
portdev - > chr_major , portdev - > drv_index ) ;
err = portdev - > chr_major ;
goto free ;
}
2009-12-21 18:33:25 +03:00
multiport = false ;
portdev - > config . nr_ports = 1 ;
portdev - > config . max_nr_ports = 1 ;
if ( virtio_has_feature ( vdev , VIRTIO_CONSOLE_F_MULTIPORT ) ) {
multiport = true ;
vdev - > features [ 0 ] | = 1 < < VIRTIO_CONSOLE_F_MULTIPORT ;
vdev - > config - > get ( vdev , offsetof ( struct virtio_console_config ,
nr_ports ) ,
& portdev - > config . nr_ports ,
sizeof ( portdev - > config . nr_ports ) ) ;
vdev - > config - > get ( vdev , offsetof ( struct virtio_console_config ,
max_nr_ports ) ,
& portdev - > config . max_nr_ports ,
sizeof ( portdev - > config . max_nr_ports ) ) ;
if ( portdev - > config . nr_ports > portdev - > config . max_nr_ports ) {
dev_warn ( & vdev - > dev ,
" More ports (%u) specified than allowed (%u). Will init %u ports. " ,
portdev - > config . nr_ports ,
portdev - > config . max_nr_ports ,
portdev - > config . max_nr_ports ) ;
portdev - > config . nr_ports = portdev - > config . max_nr_ports ;
}
}
/* Let the Host know we support multiple ports.*/
vdev - > config - > finalize_features ( vdev ) ;
2010-01-18 16:45:11 +03:00
err = init_vqs ( portdev ) ;
if ( err < 0 ) {
dev_err ( & vdev - > dev , " Error %d initializing vqs \n " , err ) ;
2009-12-21 19:06:04 +03:00
goto free_chrdev ;
2010-01-18 16:45:11 +03:00
}
2010-01-18 16:45:07 +03:00
2009-12-21 18:33:25 +03:00
spin_lock_init ( & portdev - > ports_lock ) ;
INIT_LIST_HEAD ( & portdev - > ports ) ;
if ( multiport ) {
spin_lock_init ( & portdev - > cvq_lock ) ;
INIT_WORK ( & portdev - > control_work , & control_work_handler ) ;
2009-12-21 19:52:08 +03:00
INIT_WORK ( & portdev - > config_work , & config_work_handler ) ;
2009-12-21 18:33:25 +03:00
2010-02-12 08:02:17 +03:00
err = fill_queue ( portdev - > c_ivq , & portdev - > cvq_lock ) ;
if ( ! err ) {
dev_err ( & vdev - > dev ,
" Error allocating buffers for control queue \n " ) ;
err = - ENOMEM ;
goto free_vqs ;
}
2009-12-21 18:33:25 +03:00
}
for ( i = 0 ; i < portdev - > config . nr_ports ; i + + )
add_port ( portdev , i ) ;
2010-01-18 16:45:07 +03:00
2010-01-18 16:44:56 +03:00
/* Start using the new console output. */
early_put_chars = NULL ;
2007-10-22 05:03:39 +04:00
return 0 ;
2010-02-12 08:02:17 +03:00
free_vqs :
vdev - > config - > del_vqs ( vdev ) ;
kfree ( portdev - > in_vqs ) ;
kfree ( portdev - > out_vqs ) ;
2009-12-21 19:06:04 +03:00
free_chrdev :
unregister_chrdev ( portdev - > chr_major , " virtio-portsdev " ) ;
2007-10-22 05:03:39 +04:00
free :
2010-01-18 16:45:07 +03:00
kfree ( portdev ) ;
2007-10-22 05:03:39 +04:00
fail :
return err ;
}
2010-02-12 08:02:16 +03:00
static void virtcons_remove ( struct virtio_device * vdev )
{
struct ports_device * portdev ;
struct port * port , * port2 ;
struct port_buffer * buf ;
unsigned int len ;
portdev = vdev - > priv ;
cancel_work_sync ( & portdev - > control_work ) ;
cancel_work_sync ( & portdev - > config_work ) ;
list_for_each_entry_safe ( port , port2 , & portdev - > ports , list )
remove_port ( port ) ;
unregister_chrdev ( portdev - > chr_major , " virtio-portsdev " ) ;
while ( ( buf = portdev - > c_ivq - > vq_ops - > get_buf ( portdev - > c_ivq , & len ) ) )
free_buf ( buf ) ;
while ( ( buf = portdev - > c_ivq - > vq_ops - > detach_unused_buf ( portdev - > c_ivq ) ) )
free_buf ( buf ) ;
vdev - > config - > del_vqs ( vdev ) ;
kfree ( portdev - > in_vqs ) ;
kfree ( portdev - > out_vqs ) ;
kfree ( portdev ) ;
}
2007-10-22 05:03:39 +04:00
static struct virtio_device_id id_table [ ] = {
{ VIRTIO_ID_CONSOLE , VIRTIO_DEV_ANY_ID } ,
{ 0 } ,
} ;
2008-11-25 15:36:26 +03:00
static unsigned int features [ ] = {
VIRTIO_CONSOLE_F_SIZE ,
2009-12-21 18:33:25 +03:00
VIRTIO_CONSOLE_F_MULTIPORT ,
2008-11-25 15:36:26 +03:00
} ;
2007-10-22 05:03:39 +04:00
static struct virtio_driver virtio_console = {
2008-11-25 15:36:26 +03:00
. feature_table = features ,
. feature_table_size = ARRAY_SIZE ( features ) ,
2007-10-22 05:03:39 +04:00
. driver . name = KBUILD_MODNAME ,
. driver . owner = THIS_MODULE ,
. id_table = id_table ,
. probe = virtcons_probe ,
2010-02-12 08:02:16 +03:00
. remove = virtcons_remove ,
2009-12-21 19:52:08 +03:00
. config_changed = config_intr ,
2007-10-22 05:03:39 +04:00
} ;
static int __init init ( void )
{
2009-12-21 19:06:04 +03:00
int err ;
pdrvdata . class = class_create ( THIS_MODULE , " virtio-ports " ) ;
if ( IS_ERR ( pdrvdata . class ) ) {
err = PTR_ERR ( pdrvdata . class ) ;
pr_err ( " Error %d creating virtio-ports class \n " , err ) ;
return err ;
}
2009-12-21 20:06:21 +03:00
pdrvdata . debugfs_dir = debugfs_create_dir ( " virtio-ports " , NULL ) ;
if ( ! pdrvdata . debugfs_dir ) {
pr_warning ( " Error %ld creating debugfs dir for virtio-ports \n " ,
PTR_ERR ( pdrvdata . debugfs_dir ) ) ;
}
2010-01-18 16:45:05 +03:00
INIT_LIST_HEAD ( & pdrvdata . consoles ) ;
2007-10-22 05:03:39 +04:00
return register_virtio_driver ( & virtio_console ) ;
}
2010-02-12 08:02:16 +03:00
static void __exit fini ( void )
{
unregister_virtio_driver ( & virtio_console ) ;
class_destroy ( pdrvdata . class ) ;
if ( pdrvdata . debugfs_dir )
debugfs_remove_recursive ( pdrvdata . debugfs_dir ) ;
}
2007-10-22 05:03:39 +04:00
module_init ( init ) ;
2010-02-12 08:02:16 +03:00
module_exit ( fini ) ;
2007-10-22 05:03:39 +04:00
MODULE_DEVICE_TABLE ( virtio , id_table ) ;
MODULE_DESCRIPTION ( " Virtio console driver " ) ;
MODULE_LICENSE ( " GPL " ) ;