| /* vcc.c: sun4v virtual channel concentrator | 
 |  * | 
 |  * Copyright (C) 2017 Oracle. All rights reserved. | 
 |  */ | 
 |  | 
 | #include <linux/delay.h> | 
 | #include <linux/interrupt.h> | 
 | #include <linux/module.h> | 
 | #include <linux/slab.h> | 
 | #include <linux/sysfs.h> | 
 | #include <linux/tty.h> | 
 | #include <linux/tty_flip.h> | 
 | #include <asm/vio.h> | 
 | #include <asm/ldc.h> | 
 |  | 
 | #define DRV_MODULE_NAME		"vcc" | 
 | #define DRV_MODULE_VERSION	"1.1" | 
 | #define DRV_MODULE_RELDATE	"July 1, 2017" | 
 |  | 
 | static char version[] = | 
 | 	DRV_MODULE_NAME ".c:v" DRV_MODULE_VERSION " (" DRV_MODULE_RELDATE ")"; | 
 |  | 
 | MODULE_DESCRIPTION("Sun LDOM virtual console concentrator driver"); | 
 | MODULE_LICENSE("GPL"); | 
 | MODULE_VERSION(DRV_MODULE_VERSION); | 
 |  | 
 | struct vcc_port { | 
 | 	struct vio_driver_state vio; | 
 |  | 
 | 	spinlock_t lock; | 
 | 	char *domain; | 
 | 	struct tty_struct *tty;	/* only populated while dev is open */ | 
 | 	unsigned long index;	/* index into the vcc_table */ | 
 |  | 
 | 	u64 refcnt; | 
 | 	bool excl_locked; | 
 |  | 
 | 	bool removed; | 
 |  | 
 | 	/* This buffer is required to support the tty write_room interface | 
 | 	 * and guarantee that any characters that the driver accepts will | 
 | 	 * be eventually sent, either immediately or later. | 
 | 	 */ | 
 | 	int chars_in_buffer; | 
 | 	struct vio_vcc buffer; | 
 |  | 
 | 	struct timer_list rx_timer; | 
 | 	struct timer_list tx_timer; | 
 | }; | 
 |  | 
 | /* Microseconds that thread will delay waiting for a vcc port ref */ | 
 | #define VCC_REF_DELAY		100 | 
 |  | 
 | #define VCC_MAX_PORTS		1024 | 
 | #define VCC_MINOR_START		0	/* must be zero */ | 
 | #define VCC_BUFF_LEN		VIO_VCC_MTU_SIZE | 
 |  | 
 | #define VCC_CTL_BREAK		-1 | 
 | #define VCC_CTL_HUP		-2 | 
 |  | 
 | static const char vcc_driver_name[] = "vcc"; | 
 | static const char vcc_device_node[] = "vcc"; | 
 | static struct tty_driver *vcc_tty_driver; | 
 |  | 
 | static struct vcc_port *vcc_table[VCC_MAX_PORTS]; | 
 | static DEFINE_SPINLOCK(vcc_table_lock); | 
 |  | 
 | int vcc_dbg; | 
 | int vcc_dbg_ldc; | 
 | int vcc_dbg_vio; | 
 |  | 
 | module_param(vcc_dbg, uint, 0664); | 
 | module_param(vcc_dbg_ldc, uint, 0664); | 
 | module_param(vcc_dbg_vio, uint, 0664); | 
 |  | 
 | #define VCC_DBG_DRV	0x1 | 
 | #define VCC_DBG_LDC	0x2 | 
 | #define VCC_DBG_PKT	0x4 | 
 |  | 
 | #define vccdbg(f, a...)						\ | 
 | 	do {							\ | 
 | 		if (vcc_dbg & VCC_DBG_DRV)			\ | 
 | 			pr_info(f, ## a);			\ | 
 | 	} while (0)						\ | 
 |  | 
 | #define vccdbgl(l)						\ | 
 | 	do {							\ | 
 | 		if (vcc_dbg & VCC_DBG_LDC)			\ | 
 | 			ldc_print(l);				\ | 
 | 	} while (0)						\ | 
 |  | 
 | #define vccdbgp(pkt)						\ | 
 | 	do {							\ | 
 | 		if (vcc_dbg & VCC_DBG_PKT) {			\ | 
 | 			int i;					\ | 
 | 			for (i = 0; i < pkt.tag.stype; i++)	\ | 
 | 				pr_info("[%c]", pkt.data[i]);	\ | 
 | 		}						\ | 
 | 	} while (0)						\ | 
 |  | 
 | /* Note: Be careful when adding flags to this line discipline.  Don't | 
 |  * add anything that will cause echoing or we'll go into recursive | 
 |  * loop echoing chars back and forth with the console drivers. | 
 |  */ | 
 | static const struct ktermios vcc_tty_termios = { | 
 | 	.c_iflag = IGNBRK | IGNPAR, | 
 | 	.c_oflag = OPOST, | 
 | 	.c_cflag = B38400 | CS8 | CREAD | HUPCL, | 
 | 	.c_cc = INIT_C_CC, | 
 | 	.c_ispeed = 38400, | 
 | 	.c_ospeed = 38400 | 
 | }; | 
 |  | 
 | /** | 
 |  * vcc_table_add() - Add VCC port to the VCC table | 
 |  * @port: pointer to the VCC port | 
 |  * | 
 |  * Return: index of the port in the VCC table on success, | 
 |  *	   -1 on failure | 
 |  */ | 
 | static int vcc_table_add(struct vcc_port *port) | 
 | { | 
 | 	unsigned long flags; | 
 | 	int i; | 
 |  | 
 | 	spin_lock_irqsave(&vcc_table_lock, flags); | 
 | 	for (i = VCC_MINOR_START; i < VCC_MAX_PORTS; i++) { | 
 | 		if (!vcc_table[i]) { | 
 | 			vcc_table[i] = port; | 
 | 			break; | 
 | 		} | 
 | 	} | 
 | 	spin_unlock_irqrestore(&vcc_table_lock, flags); | 
 |  | 
 | 	if (i < VCC_MAX_PORTS) | 
 | 		return i; | 
 | 	else | 
 | 		return -1; | 
 | } | 
 |  | 
 | /** | 
 |  * vcc_table_remove() - Removes a VCC port from the VCC table | 
 |  * @index: Index into the VCC table | 
 |  */ | 
 | static void vcc_table_remove(unsigned long index) | 
 | { | 
 | 	unsigned long flags; | 
 |  | 
 | 	if (WARN_ON(index >= VCC_MAX_PORTS)) | 
 | 		return; | 
 |  | 
 | 	spin_lock_irqsave(&vcc_table_lock, flags); | 
 | 	vcc_table[index] = NULL; | 
 | 	spin_unlock_irqrestore(&vcc_table_lock, flags); | 
 | } | 
 |  | 
 | /** | 
 |  * vcc_get() - Gets a reference to VCC port | 
 |  * @index: Index into the VCC table | 
 |  * @excl: Indicates if an exclusive access is requested | 
 |  * | 
 |  * Return: reference to the VCC port, if found | 
 |  *	   NULL, if port not found | 
 |  */ | 
 | static struct vcc_port *vcc_get(unsigned long index, bool excl) | 
 | { | 
 | 	struct vcc_port *port; | 
 | 	unsigned long flags; | 
 |  | 
 | try_again: | 
 | 	spin_lock_irqsave(&vcc_table_lock, flags); | 
 |  | 
 | 	port = vcc_table[index]; | 
 | 	if (!port) { | 
 | 		spin_unlock_irqrestore(&vcc_table_lock, flags); | 
 | 		return NULL; | 
 | 	} | 
 |  | 
 | 	if (!excl) { | 
 | 		if (port->excl_locked) { | 
 | 			spin_unlock_irqrestore(&vcc_table_lock, flags); | 
 | 			udelay(VCC_REF_DELAY); | 
 | 			goto try_again; | 
 | 		} | 
 | 		port->refcnt++; | 
 | 		spin_unlock_irqrestore(&vcc_table_lock, flags); | 
 | 		return port; | 
 | 	} | 
 |  | 
 | 	if (port->refcnt) { | 
 | 		spin_unlock_irqrestore(&vcc_table_lock, flags); | 
 | 		/* Threads wanting exclusive access will wait half the time, | 
 | 		 * probably giving them higher priority in the case of | 
 | 		 * multiple waiters. | 
 | 		 */ | 
 | 		udelay(VCC_REF_DELAY/2); | 
 | 		goto try_again; | 
 | 	} | 
 |  | 
 | 	port->refcnt++; | 
 | 	port->excl_locked = true; | 
 | 	spin_unlock_irqrestore(&vcc_table_lock, flags); | 
 |  | 
 | 	return port; | 
 | } | 
 |  | 
 | /** | 
 |  * vcc_put() - Returns a reference to VCC port | 
 |  * @port: pointer to VCC port | 
 |  * @excl: Indicates if the returned reference is an exclusive reference | 
 |  * | 
 |  * Note: It's the caller's responsibility to ensure the correct value | 
 |  *	 for the excl flag | 
 |  */ | 
 | static void vcc_put(struct vcc_port *port, bool excl) | 
 | { | 
 | 	unsigned long flags; | 
 |  | 
 | 	if (!port) | 
 | 		return; | 
 |  | 
 | 	spin_lock_irqsave(&vcc_table_lock, flags); | 
 |  | 
 | 	/* check if caller attempted to put with the wrong flags */ | 
 | 	if (WARN_ON((excl && !port->excl_locked) || | 
 | 		    (!excl && port->excl_locked))) | 
 | 		goto done; | 
 |  | 
 | 	port->refcnt--; | 
 |  | 
 | 	if (excl) | 
 | 		port->excl_locked = false; | 
 |  | 
 | done: | 
 | 	spin_unlock_irqrestore(&vcc_table_lock, flags); | 
 | } | 
 |  | 
 | /** | 
 |  * vcc_get_ne() - Get a non-exclusive reference to VCC port | 
 |  * @index: Index into the VCC table | 
 |  * | 
 |  * Gets a non-exclusive reference to VCC port, if it's not removed | 
 |  * | 
 |  * Return: pointer to the VCC port, if found | 
 |  *	   NULL, if port not found | 
 |  */ | 
 | static struct vcc_port *vcc_get_ne(unsigned long index) | 
 | { | 
 | 	struct vcc_port *port; | 
 |  | 
 | 	port = vcc_get(index, false); | 
 |  | 
 | 	if (port && port->removed) { | 
 | 		vcc_put(port, false); | 
 | 		return NULL; | 
 | 	} | 
 |  | 
 | 	return port; | 
 | } | 
 |  | 
 | static void vcc_kick_rx(struct vcc_port *port) | 
 | { | 
 | 	struct vio_driver_state *vio = &port->vio; | 
 |  | 
 | 	assert_spin_locked(&port->lock); | 
 |  | 
 | 	if (!timer_pending(&port->rx_timer) && !port->removed) { | 
 | 		disable_irq_nosync(vio->vdev->rx_irq); | 
 | 		port->rx_timer.expires = (jiffies + 1); | 
 | 		add_timer(&port->rx_timer); | 
 | 	} | 
 | } | 
 |  | 
 | static void vcc_kick_tx(struct vcc_port *port) | 
 | { | 
 | 	assert_spin_locked(&port->lock); | 
 |  | 
 | 	if (!timer_pending(&port->tx_timer) && !port->removed) { | 
 | 		port->tx_timer.expires = (jiffies + 1); | 
 | 		add_timer(&port->tx_timer); | 
 | 	} | 
 | } | 
 |  | 
 | static int vcc_rx_check(struct tty_struct *tty, int size) | 
 | { | 
 | 	if (WARN_ON(!tty || !tty->port)) | 
 | 		return 1; | 
 |  | 
 | 	/* tty_buffer_request_room won't sleep because it uses | 
 | 	 * GFP_ATOMIC flag to allocate buffer | 
 | 	 */ | 
 | 	if (test_bit(TTY_THROTTLED, &tty->flags) || | 
 | 	    (tty_buffer_request_room(tty->port, VCC_BUFF_LEN) < VCC_BUFF_LEN)) | 
 | 		return 0; | 
 |  | 
 | 	return 1; | 
 | } | 
 |  | 
 | static int vcc_rx(struct tty_struct *tty, char *buf, int size) | 
 | { | 
 | 	int len = 0; | 
 |  | 
 | 	if (WARN_ON(!tty || !tty->port)) | 
 | 		return len; | 
 |  | 
 | 	len = tty_insert_flip_string(tty->port, buf, size); | 
 | 	if (len) | 
 | 		tty_flip_buffer_push(tty->port); | 
 |  | 
 | 	return len; | 
 | } | 
 |  | 
 | static int vcc_ldc_read(struct vcc_port *port) | 
 | { | 
 | 	struct vio_driver_state *vio = &port->vio; | 
 | 	struct tty_struct *tty; | 
 | 	struct vio_vcc pkt; | 
 | 	int rv = 0; | 
 |  | 
 | 	tty = port->tty; | 
 | 	if (!tty) { | 
 | 		rv = ldc_rx_reset(vio->lp); | 
 | 		vccdbg("VCC: reset rx q: rv=%d\n", rv); | 
 | 		goto done; | 
 | 	} | 
 |  | 
 | 	/* Read as long as LDC has incoming data. */ | 
 | 	while (1) { | 
 | 		if (!vcc_rx_check(tty, VIO_VCC_MTU_SIZE)) { | 
 | 			vcc_kick_rx(port); | 
 | 			break; | 
 | 		} | 
 |  | 
 | 		vccdbgl(vio->lp); | 
 |  | 
 | 		rv = ldc_read(vio->lp, &pkt, sizeof(pkt)); | 
 | 		if (rv <= 0) | 
 | 			break; | 
 |  | 
 | 		vccdbg("VCC: ldc_read()=%d\n", rv); | 
 | 		vccdbg("TAG [%02x:%02x:%04x:%08x]\n", | 
 | 		       pkt.tag.type, pkt.tag.stype, | 
 | 		       pkt.tag.stype_env, pkt.tag.sid); | 
 |  | 
 | 		if (pkt.tag.type == VIO_TYPE_DATA) { | 
 | 			vccdbgp(pkt); | 
 | 			/* vcc_rx_check ensures memory availability */ | 
 | 			vcc_rx(tty, pkt.data, pkt.tag.stype); | 
 | 		} else { | 
 | 			pr_err("VCC: unknown msg [%02x:%02x:%04x:%08x]\n", | 
 | 			       pkt.tag.type, pkt.tag.stype, | 
 | 			       pkt.tag.stype_env, pkt.tag.sid); | 
 | 			rv = -ECONNRESET; | 
 | 			break; | 
 | 		} | 
 |  | 
 | 		WARN_ON(rv != LDC_PACKET_SIZE); | 
 | 	} | 
 |  | 
 | done: | 
 | 	return rv; | 
 | } | 
 |  | 
 | static void vcc_rx_timer(struct timer_list *t) | 
 | { | 
 | 	struct vcc_port *port = from_timer(port, t, rx_timer); | 
 | 	struct vio_driver_state *vio; | 
 | 	unsigned long flags; | 
 | 	int rv; | 
 |  | 
 | 	spin_lock_irqsave(&port->lock, flags); | 
 | 	port->rx_timer.expires = 0; | 
 |  | 
 | 	vio = &port->vio; | 
 |  | 
 | 	enable_irq(vio->vdev->rx_irq); | 
 |  | 
 | 	if (!port->tty || port->removed) | 
 | 		goto done; | 
 |  | 
 | 	rv = vcc_ldc_read(port); | 
 | 	if (rv == -ECONNRESET) | 
 | 		vio_conn_reset(vio); | 
 |  | 
 | done: | 
 | 	spin_unlock_irqrestore(&port->lock, flags); | 
 | 	vcc_put(port, false); | 
 | } | 
 |  | 
 | static void vcc_tx_timer(struct timer_list *t) | 
 | { | 
 | 	struct vcc_port *port = from_timer(port, t, tx_timer); | 
 | 	struct vio_vcc *pkt; | 
 | 	unsigned long flags; | 
 | 	int tosend = 0; | 
 | 	int rv; | 
 |  | 
 | 	spin_lock_irqsave(&port->lock, flags); | 
 | 	port->tx_timer.expires = 0; | 
 |  | 
 | 	if (!port->tty || port->removed) | 
 | 		goto done; | 
 |  | 
 | 	tosend = min(VCC_BUFF_LEN, port->chars_in_buffer); | 
 | 	if (!tosend) | 
 | 		goto done; | 
 |  | 
 | 	pkt = &port->buffer; | 
 | 	pkt->tag.type = VIO_TYPE_DATA; | 
 | 	pkt->tag.stype = tosend; | 
 | 	vccdbgl(port->vio.lp); | 
 |  | 
 | 	rv = ldc_write(port->vio.lp, pkt, (VIO_TAG_SIZE + tosend)); | 
 | 	WARN_ON(!rv); | 
 |  | 
 | 	if (rv < 0) { | 
 | 		vccdbg("VCC: ldc_write()=%d\n", rv); | 
 | 		vcc_kick_tx(port); | 
 | 	} else { | 
 | 		struct tty_struct *tty = port->tty; | 
 |  | 
 | 		port->chars_in_buffer = 0; | 
 | 		if (tty) | 
 | 			tty_wakeup(tty); | 
 | 	} | 
 |  | 
 | done: | 
 | 	spin_unlock_irqrestore(&port->lock, flags); | 
 | 	vcc_put(port, false); | 
 | } | 
 |  | 
 | /** | 
 |  * vcc_event() - LDC event processing engine | 
 |  * @arg: VCC private data | 
 |  * @event: LDC event | 
 |  * | 
 |  * Handles LDC events for VCC | 
 |  */ | 
 | static void vcc_event(void *arg, int event) | 
 | { | 
 | 	struct vio_driver_state *vio; | 
 | 	struct vcc_port *port; | 
 | 	unsigned long flags; | 
 | 	int rv; | 
 |  | 
 | 	port = arg; | 
 | 	vio = &port->vio; | 
 |  | 
 | 	spin_lock_irqsave(&port->lock, flags); | 
 |  | 
 | 	switch (event) { | 
 | 	case LDC_EVENT_RESET: | 
 | 	case LDC_EVENT_UP: | 
 | 		vio_link_state_change(vio, event); | 
 | 		break; | 
 |  | 
 | 	case LDC_EVENT_DATA_READY: | 
 | 		rv = vcc_ldc_read(port); | 
 | 		if (rv == -ECONNRESET) | 
 | 			vio_conn_reset(vio); | 
 | 		break; | 
 |  | 
 | 	default: | 
 | 		pr_err("VCC: unexpected LDC event(%d)\n", event); | 
 | 	} | 
 |  | 
 | 	spin_unlock_irqrestore(&port->lock, flags); | 
 | } | 
 |  | 
 | static struct ldc_channel_config vcc_ldc_cfg = { | 
 | 	.event		= vcc_event, | 
 | 	.mtu		= VIO_VCC_MTU_SIZE, | 
 | 	.mode		= LDC_MODE_RAW, | 
 | 	.debug		= 0, | 
 | }; | 
 |  | 
 | /* Ordered from largest major to lowest */ | 
 | static struct vio_version vcc_versions[] = { | 
 | 	{ .major = 1, .minor = 0 }, | 
 | }; | 
 |  | 
 | static struct tty_port_operations vcc_port_ops = { 0 }; | 
 |  | 
 | static ssize_t vcc_sysfs_domain_show(struct device *dev, | 
 | 				     struct device_attribute *attr, | 
 | 				     char *buf) | 
 | { | 
 | 	struct vcc_port *port; | 
 | 	int rv; | 
 |  | 
 | 	port = dev_get_drvdata(dev); | 
 | 	if (!port) | 
 | 		return -ENODEV; | 
 |  | 
 | 	rv = scnprintf(buf, PAGE_SIZE, "%s\n", port->domain); | 
 |  | 
 | 	return rv; | 
 | } | 
 |  | 
 | static int vcc_send_ctl(struct vcc_port *port, int ctl) | 
 | { | 
 | 	struct vio_vcc pkt; | 
 | 	int rv; | 
 |  | 
 | 	pkt.tag.type = VIO_TYPE_CTRL; | 
 | 	pkt.tag.sid = ctl; | 
 | 	pkt.tag.stype = 0; | 
 |  | 
 | 	rv = ldc_write(port->vio.lp, &pkt, sizeof(pkt.tag)); | 
 | 	WARN_ON(!rv); | 
 | 	vccdbg("VCC: ldc_write(%ld)=%d\n", sizeof(pkt.tag), rv); | 
 |  | 
 | 	return rv; | 
 | } | 
 |  | 
 | static ssize_t vcc_sysfs_break_store(struct device *dev, | 
 | 				     struct device_attribute *attr, | 
 | 				     const char *buf, size_t count) | 
 | { | 
 | 	struct vcc_port *port; | 
 | 	unsigned long flags; | 
 | 	int rv = count; | 
 | 	int brk; | 
 |  | 
 | 	port = dev_get_drvdata(dev); | 
 | 	if (!port) | 
 | 		return -ENODEV; | 
 |  | 
 | 	spin_lock_irqsave(&port->lock, flags); | 
 |  | 
 | 	if (sscanf(buf, "%ud", &brk) != 1 || brk != 1) | 
 | 		rv = -EINVAL; | 
 | 	else if (vcc_send_ctl(port, VCC_CTL_BREAK) < 0) | 
 | 		vcc_kick_tx(port); | 
 |  | 
 | 	spin_unlock_irqrestore(&port->lock, flags); | 
 |  | 
 | 	return rv; | 
 | } | 
 |  | 
 | static DEVICE_ATTR(domain, 0400, vcc_sysfs_domain_show, NULL); | 
 | static DEVICE_ATTR(break, 0200, NULL, vcc_sysfs_break_store); | 
 |  | 
 | static struct attribute *vcc_sysfs_entries[] = { | 
 | 	&dev_attr_domain.attr, | 
 | 	&dev_attr_break.attr, | 
 | 	NULL | 
 | }; | 
 |  | 
 | static struct attribute_group vcc_attribute_group = { | 
 | 	.name = NULL, | 
 | 	.attrs = vcc_sysfs_entries, | 
 | }; | 
 |  | 
 | /** | 
 |  * vcc_probe() - Initialize VCC port | 
 |  * @vdev: Pointer to VIO device of the new VCC port | 
 |  * @id: VIO device ID | 
 |  * | 
 |  * Initializes a VCC port to receive serial console data from | 
 |  * the guest domain. Sets up a TTY end point on the control | 
 |  * domain. Sets up VIO/LDC link between the guest & control | 
 |  * domain endpoints. | 
 |  * | 
 |  * Return: status of the probe | 
 |  */ | 
 | static int vcc_probe(struct vio_dev *vdev, const struct vio_device_id *id) | 
 | { | 
 | 	struct mdesc_handle *hp; | 
 | 	struct vcc_port *port; | 
 | 	struct device *dev; | 
 | 	const char *domain; | 
 | 	char *name; | 
 | 	u64 node; | 
 | 	int rv; | 
 |  | 
 | 	vccdbg("VCC: name=%s\n", dev_name(&vdev->dev)); | 
 |  | 
 | 	if (!vcc_tty_driver) { | 
 | 		pr_err("VCC: TTY driver not registered\n"); | 
 | 		return -ENODEV; | 
 | 	} | 
 |  | 
 | 	port = kzalloc(sizeof(struct vcc_port), GFP_KERNEL); | 
 | 	if (!port) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	name = kstrdup(dev_name(&vdev->dev), GFP_KERNEL); | 
 |  | 
 | 	rv = vio_driver_init(&port->vio, vdev, VDEV_CONSOLE_CON, vcc_versions, | 
 | 			     ARRAY_SIZE(vcc_versions), NULL, name); | 
 | 	if (rv) | 
 | 		goto free_port; | 
 |  | 
 | 	port->vio.debug = vcc_dbg_vio; | 
 | 	vcc_ldc_cfg.debug = vcc_dbg_ldc; | 
 |  | 
 | 	rv = vio_ldc_alloc(&port->vio, &vcc_ldc_cfg, port); | 
 | 	if (rv) | 
 | 		goto free_port; | 
 |  | 
 | 	spin_lock_init(&port->lock); | 
 |  | 
 | 	port->index = vcc_table_add(port); | 
 | 	if (port->index == -1) { | 
 | 		pr_err("VCC: no more TTY indices left for allocation\n"); | 
 | 		goto free_ldc; | 
 | 	} | 
 |  | 
 | 	/* Register the device using VCC table index as TTY index */ | 
 | 	dev = tty_register_device(vcc_tty_driver, port->index, &vdev->dev); | 
 | 	if (IS_ERR(dev)) { | 
 | 		rv = PTR_ERR(dev); | 
 | 		goto free_table; | 
 | 	} | 
 |  | 
 | 	hp = mdesc_grab(); | 
 |  | 
 | 	node = vio_vdev_node(hp, vdev); | 
 | 	if (node == MDESC_NODE_NULL) { | 
 | 		rv = -ENXIO; | 
 | 		mdesc_release(hp); | 
 | 		goto unreg_tty; | 
 | 	} | 
 |  | 
 | 	domain = mdesc_get_property(hp, node, "vcc-domain-name", NULL); | 
 | 	if (!domain) { | 
 | 		rv = -ENXIO; | 
 | 		mdesc_release(hp); | 
 | 		goto unreg_tty; | 
 | 	} | 
 | 	port->domain = kstrdup(domain, GFP_KERNEL); | 
 |  | 
 | 	mdesc_release(hp); | 
 |  | 
 | 	rv = sysfs_create_group(&vdev->dev.kobj, &vcc_attribute_group); | 
 | 	if (rv) | 
 | 		goto free_domain; | 
 |  | 
 | 	timer_setup(&port->rx_timer, vcc_rx_timer, 0); | 
 | 	timer_setup(&port->tx_timer, vcc_tx_timer, 0); | 
 |  | 
 | 	dev_set_drvdata(&vdev->dev, port); | 
 |  | 
 | 	/* It's possible to receive IRQs in the middle of vio_port_up. Disable | 
 | 	 * IRQs until the port is up. | 
 | 	 */ | 
 | 	disable_irq_nosync(vdev->rx_irq); | 
 | 	vio_port_up(&port->vio); | 
 | 	enable_irq(vdev->rx_irq); | 
 |  | 
 | 	return 0; | 
 |  | 
 | free_domain: | 
 | 	kfree(port->domain); | 
 | unreg_tty: | 
 | 	tty_unregister_device(vcc_tty_driver, port->index); | 
 | free_table: | 
 | 	vcc_table_remove(port->index); | 
 | free_ldc: | 
 | 	vio_ldc_free(&port->vio); | 
 | free_port: | 
 | 	kfree(name); | 
 | 	kfree(port); | 
 |  | 
 | 	return rv; | 
 | } | 
 |  | 
 | /** | 
 |  * vcc_remove() - Terminate a VCC port | 
 |  * @vdev: Pointer to VIO device of the VCC port | 
 |  * | 
 |  * Terminates a VCC port. Sets up the teardown of TTY and | 
 |  * VIO/LDC link between guest and primary domains. | 
 |  * | 
 |  * Return: status of removal | 
 |  */ | 
 | static int vcc_remove(struct vio_dev *vdev) | 
 | { | 
 | 	struct vcc_port *port = dev_get_drvdata(&vdev->dev); | 
 |  | 
 | 	if (!port) | 
 | 		return -ENODEV; | 
 |  | 
 | 	del_timer_sync(&port->rx_timer); | 
 | 	del_timer_sync(&port->tx_timer); | 
 |  | 
 | 	/* If there's a process with the device open, do a synchronous | 
 | 	 * hangup of the TTY. This *may* cause the process to call close | 
 | 	 * asynchronously, but it's not guaranteed. | 
 | 	 */ | 
 | 	if (port->tty) | 
 | 		tty_vhangup(port->tty); | 
 |  | 
 | 	/* Get exclusive reference to VCC, ensures that there are no other | 
 | 	 * clients to this port | 
 | 	 */ | 
 | 	port = vcc_get(port->index, true); | 
 |  | 
 | 	if (WARN_ON(!port)) | 
 | 		return -ENODEV; | 
 |  | 
 | 	tty_unregister_device(vcc_tty_driver, port->index); | 
 |  | 
 | 	del_timer_sync(&port->vio.timer); | 
 | 	vio_ldc_free(&port->vio); | 
 | 	sysfs_remove_group(&vdev->dev.kobj, &vcc_attribute_group); | 
 | 	dev_set_drvdata(&vdev->dev, NULL); | 
 | 	if (port->tty) { | 
 | 		port->removed = true; | 
 | 		vcc_put(port, true); | 
 | 	} else { | 
 | 		vcc_table_remove(port->index); | 
 |  | 
 | 		kfree(port->vio.name); | 
 | 		kfree(port->domain); | 
 | 		kfree(port); | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static const struct vio_device_id vcc_match[] = { | 
 | 	{ | 
 | 		.type = "vcc-port", | 
 | 	}, | 
 | 	{}, | 
 | }; | 
 | MODULE_DEVICE_TABLE(vio, vcc_match); | 
 |  | 
 | static struct vio_driver vcc_driver = { | 
 | 	.id_table	= vcc_match, | 
 | 	.probe		= vcc_probe, | 
 | 	.remove		= vcc_remove, | 
 | 	.name		= "vcc", | 
 | }; | 
 |  | 
 | static int vcc_open(struct tty_struct *tty, struct file *vcc_file) | 
 | { | 
 | 	struct vcc_port *port; | 
 |  | 
 | 	if (unlikely(!tty)) { | 
 | 		pr_err("VCC: open: Invalid TTY handle\n"); | 
 | 		return -ENXIO; | 
 | 	} | 
 |  | 
 | 	if (tty->count > 1) | 
 | 		return -EBUSY; | 
 |  | 
 | 	port = vcc_get_ne(tty->index); | 
 | 	if (unlikely(!port)) { | 
 | 		pr_err("VCC: open: Failed to find VCC port\n"); | 
 | 		return -ENODEV; | 
 | 	} | 
 |  | 
 | 	if (unlikely(!port->vio.lp)) { | 
 | 		pr_err("VCC: open: LDC channel not configured\n"); | 
 | 		vcc_put(port, false); | 
 | 		return -EPIPE; | 
 | 	} | 
 | 	vccdbgl(port->vio.lp); | 
 |  | 
 | 	vcc_put(port, false); | 
 |  | 
 | 	if (unlikely(!tty->port)) { | 
 | 		pr_err("VCC: open: TTY port not found\n"); | 
 | 		return -ENXIO; | 
 | 	} | 
 |  | 
 | 	if (unlikely(!tty->port->ops)) { | 
 | 		pr_err("VCC: open: TTY ops not defined\n"); | 
 | 		return -ENXIO; | 
 | 	} | 
 |  | 
 | 	return tty_port_open(tty->port, tty, vcc_file); | 
 | } | 
 |  | 
 | static void vcc_close(struct tty_struct *tty, struct file *vcc_file) | 
 | { | 
 | 	if (unlikely(!tty)) { | 
 | 		pr_err("VCC: close: Invalid TTY handle\n"); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	if (unlikely(tty->count > 1)) | 
 | 		return; | 
 |  | 
 | 	if (unlikely(!tty->port)) { | 
 | 		pr_err("VCC: close: TTY port not found\n"); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	tty_port_close(tty->port, tty, vcc_file); | 
 | } | 
 |  | 
 | static void vcc_ldc_hup(struct vcc_port *port) | 
 | { | 
 | 	unsigned long flags; | 
 |  | 
 | 	spin_lock_irqsave(&port->lock, flags); | 
 |  | 
 | 	if (vcc_send_ctl(port, VCC_CTL_HUP) < 0) | 
 | 		vcc_kick_tx(port); | 
 |  | 
 | 	spin_unlock_irqrestore(&port->lock, flags); | 
 | } | 
 |  | 
 | static void vcc_hangup(struct tty_struct *tty) | 
 | { | 
 | 	struct vcc_port *port; | 
 |  | 
 | 	if (unlikely(!tty)) { | 
 | 		pr_err("VCC: hangup: Invalid TTY handle\n"); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	port = vcc_get_ne(tty->index); | 
 | 	if (unlikely(!port)) { | 
 | 		pr_err("VCC: hangup: Failed to find VCC port\n"); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	if (unlikely(!tty->port)) { | 
 | 		pr_err("VCC: hangup: TTY port not found\n"); | 
 | 		vcc_put(port, false); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	vcc_ldc_hup(port); | 
 |  | 
 | 	vcc_put(port, false); | 
 |  | 
 | 	tty_port_hangup(tty->port); | 
 | } | 
 |  | 
 | static int vcc_write(struct tty_struct *tty, const unsigned char *buf, | 
 | 		     int count) | 
 | { | 
 | 	struct vcc_port *port; | 
 | 	struct vio_vcc *pkt; | 
 | 	unsigned long flags; | 
 | 	int total_sent = 0; | 
 | 	int tosend = 0; | 
 | 	int rv = -EINVAL; | 
 |  | 
 | 	if (unlikely(!tty)) { | 
 | 		pr_err("VCC: write: Invalid TTY handle\n"); | 
 | 		return -ENXIO; | 
 | 	} | 
 |  | 
 | 	port = vcc_get_ne(tty->index); | 
 | 	if (unlikely(!port)) { | 
 | 		pr_err("VCC: write: Failed to find VCC port"); | 
 | 		return -ENODEV; | 
 | 	} | 
 |  | 
 | 	spin_lock_irqsave(&port->lock, flags); | 
 |  | 
 | 	pkt = &port->buffer; | 
 | 	pkt->tag.type = VIO_TYPE_DATA; | 
 |  | 
 | 	while (count > 0) { | 
 | 		/* Minimum of data to write and space available */ | 
 | 		tosend = min(count, (VCC_BUFF_LEN - port->chars_in_buffer)); | 
 |  | 
 | 		if (!tosend) | 
 | 			break; | 
 |  | 
 | 		memcpy(&pkt->data[port->chars_in_buffer], &buf[total_sent], | 
 | 		       tosend); | 
 | 		port->chars_in_buffer += tosend; | 
 | 		pkt->tag.stype = tosend; | 
 |  | 
 | 		vccdbg("TAG [%02x:%02x:%04x:%08x]\n", pkt->tag.type, | 
 | 		       pkt->tag.stype, pkt->tag.stype_env, pkt->tag.sid); | 
 | 		vccdbg("DATA [%s]\n", pkt->data); | 
 | 		vccdbgl(port->vio.lp); | 
 |  | 
 | 		/* Since we know we have enough room in VCC buffer for tosend | 
 | 		 * we record that it was sent regardless of whether the | 
 | 		 * hypervisor actually took it because we have it buffered. | 
 | 		 */ | 
 | 		rv = ldc_write(port->vio.lp, pkt, (VIO_TAG_SIZE + tosend)); | 
 | 		vccdbg("VCC: write: ldc_write(%d)=%d\n", | 
 | 		       (VIO_TAG_SIZE + tosend), rv); | 
 |  | 
 | 		total_sent += tosend; | 
 | 		count -= tosend; | 
 | 		if (rv < 0) { | 
 | 			vcc_kick_tx(port); | 
 | 			break; | 
 | 		} | 
 |  | 
 | 		port->chars_in_buffer = 0; | 
 | 	} | 
 |  | 
 | 	spin_unlock_irqrestore(&port->lock, flags); | 
 |  | 
 | 	vcc_put(port, false); | 
 |  | 
 | 	vccdbg("VCC: write: total=%d rv=%d", total_sent, rv); | 
 |  | 
 | 	return total_sent ? total_sent : rv; | 
 | } | 
 |  | 
 | static int vcc_write_room(struct tty_struct *tty) | 
 | { | 
 | 	struct vcc_port *port; | 
 | 	u64 num; | 
 |  | 
 | 	if (unlikely(!tty)) { | 
 | 		pr_err("VCC: write_room: Invalid TTY handle\n"); | 
 | 		return -ENXIO; | 
 | 	} | 
 |  | 
 | 	port = vcc_get_ne(tty->index); | 
 | 	if (unlikely(!port)) { | 
 | 		pr_err("VCC: write_room: Failed to find VCC port\n"); | 
 | 		return -ENODEV; | 
 | 	} | 
 |  | 
 | 	num = VCC_BUFF_LEN - port->chars_in_buffer; | 
 |  | 
 | 	vcc_put(port, false); | 
 |  | 
 | 	return num; | 
 | } | 
 |  | 
 | static int vcc_chars_in_buffer(struct tty_struct *tty) | 
 | { | 
 | 	struct vcc_port *port; | 
 | 	u64 num; | 
 |  | 
 | 	if (unlikely(!tty)) { | 
 | 		pr_err("VCC: chars_in_buffer: Invalid TTY handle\n"); | 
 | 		return -ENXIO; | 
 | 	} | 
 |  | 
 | 	port = vcc_get_ne(tty->index); | 
 | 	if (unlikely(!port)) { | 
 | 		pr_err("VCC: chars_in_buffer: Failed to find VCC port\n"); | 
 | 		return -ENODEV; | 
 | 	} | 
 |  | 
 | 	num = port->chars_in_buffer; | 
 |  | 
 | 	vcc_put(port, false); | 
 |  | 
 | 	return num; | 
 | } | 
 |  | 
 | static int vcc_break_ctl(struct tty_struct *tty, int state) | 
 | { | 
 | 	struct vcc_port *port; | 
 | 	unsigned long flags; | 
 |  | 
 | 	if (unlikely(!tty)) { | 
 | 		pr_err("VCC: break_ctl: Invalid TTY handle\n"); | 
 | 		return -ENXIO; | 
 | 	} | 
 |  | 
 | 	port = vcc_get_ne(tty->index); | 
 | 	if (unlikely(!port)) { | 
 | 		pr_err("VCC: break_ctl: Failed to find VCC port\n"); | 
 | 		return -ENODEV; | 
 | 	} | 
 |  | 
 | 	/* Turn off break */ | 
 | 	if (state == 0) { | 
 | 		vcc_put(port, false); | 
 | 		return 0; | 
 | 	} | 
 |  | 
 | 	spin_lock_irqsave(&port->lock, flags); | 
 |  | 
 | 	if (vcc_send_ctl(port, VCC_CTL_BREAK) < 0) | 
 | 		vcc_kick_tx(port); | 
 |  | 
 | 	spin_unlock_irqrestore(&port->lock, flags); | 
 |  | 
 | 	vcc_put(port, false); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int vcc_install(struct tty_driver *driver, struct tty_struct *tty) | 
 | { | 
 | 	struct vcc_port *port_vcc; | 
 | 	struct tty_port *port_tty; | 
 | 	int ret; | 
 |  | 
 | 	if (unlikely(!tty)) { | 
 | 		pr_err("VCC: install: Invalid TTY handle\n"); | 
 | 		return -ENXIO; | 
 | 	} | 
 |  | 
 | 	if (tty->index >= VCC_MAX_PORTS) | 
 | 		return -EINVAL; | 
 |  | 
 | 	ret = tty_standard_install(driver, tty); | 
 | 	if (ret) | 
 | 		return ret; | 
 |  | 
 | 	port_tty = kzalloc(sizeof(struct tty_port), GFP_KERNEL); | 
 | 	if (!port_tty) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	port_vcc = vcc_get(tty->index, true); | 
 | 	if (!port_vcc) { | 
 | 		pr_err("VCC: install: Failed to find VCC port\n"); | 
 | 		tty->port = NULL; | 
 | 		kfree(port_tty); | 
 | 		return -ENODEV; | 
 | 	} | 
 |  | 
 | 	tty_port_init(port_tty); | 
 | 	port_tty->ops = &vcc_port_ops; | 
 | 	tty->port = port_tty; | 
 |  | 
 | 	port_vcc->tty = tty; | 
 |  | 
 | 	vcc_put(port_vcc, true); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void vcc_cleanup(struct tty_struct *tty) | 
 | { | 
 | 	struct vcc_port *port; | 
 |  | 
 | 	if (unlikely(!tty)) { | 
 | 		pr_err("VCC: cleanup: Invalid TTY handle\n"); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	port = vcc_get(tty->index, true); | 
 | 	if (port) { | 
 | 		port->tty = NULL; | 
 |  | 
 | 		if (port->removed) { | 
 | 			vcc_table_remove(tty->index); | 
 | 			kfree(port->vio.name); | 
 | 			kfree(port->domain); | 
 | 			kfree(port); | 
 | 		} else { | 
 | 			vcc_put(port, true); | 
 | 		} | 
 | 	} | 
 |  | 
 | 	tty_port_destroy(tty->port); | 
 | 	kfree(tty->port); | 
 | 	tty->port = NULL; | 
 | } | 
 |  | 
 | static const struct tty_operations vcc_ops = { | 
 | 	.open			= vcc_open, | 
 | 	.close			= vcc_close, | 
 | 	.hangup			= vcc_hangup, | 
 | 	.write			= vcc_write, | 
 | 	.write_room		= vcc_write_room, | 
 | 	.chars_in_buffer	= vcc_chars_in_buffer, | 
 | 	.break_ctl		= vcc_break_ctl, | 
 | 	.install		= vcc_install, | 
 | 	.cleanup		= vcc_cleanup, | 
 | }; | 
 |  | 
 | #define VCC_TTY_FLAGS   (TTY_DRIVER_DYNAMIC_DEV | TTY_DRIVER_REAL_RAW) | 
 |  | 
 | static int vcc_tty_init(void) | 
 | { | 
 | 	int rv; | 
 |  | 
 | 	pr_info("VCC: %s\n", version); | 
 |  | 
 | 	vcc_tty_driver = tty_alloc_driver(VCC_MAX_PORTS, VCC_TTY_FLAGS); | 
 | 	if (IS_ERR(vcc_tty_driver)) { | 
 | 		pr_err("VCC: TTY driver alloc failed\n"); | 
 | 		return PTR_ERR(vcc_tty_driver); | 
 | 	} | 
 |  | 
 | 	vcc_tty_driver->driver_name = vcc_driver_name; | 
 | 	vcc_tty_driver->name = vcc_device_node; | 
 |  | 
 | 	vcc_tty_driver->minor_start = VCC_MINOR_START; | 
 | 	vcc_tty_driver->type = TTY_DRIVER_TYPE_SYSTEM; | 
 | 	vcc_tty_driver->init_termios = vcc_tty_termios; | 
 |  | 
 | 	tty_set_operations(vcc_tty_driver, &vcc_ops); | 
 |  | 
 | 	rv = tty_register_driver(vcc_tty_driver); | 
 | 	if (rv) { | 
 | 		pr_err("VCC: TTY driver registration failed\n"); | 
 | 		put_tty_driver(vcc_tty_driver); | 
 | 		vcc_tty_driver = NULL; | 
 | 		return rv; | 
 | 	} | 
 |  | 
 | 	vccdbg("VCC: TTY driver registered\n"); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void vcc_tty_exit(void) | 
 | { | 
 | 	tty_unregister_driver(vcc_tty_driver); | 
 | 	put_tty_driver(vcc_tty_driver); | 
 | 	vccdbg("VCC: TTY driver unregistered\n"); | 
 |  | 
 | 	vcc_tty_driver = NULL; | 
 | } | 
 |  | 
 | static int __init vcc_init(void) | 
 | { | 
 | 	int rv; | 
 |  | 
 | 	rv = vcc_tty_init(); | 
 | 	if (rv) { | 
 | 		pr_err("VCC: TTY init failed\n"); | 
 | 		return rv; | 
 | 	} | 
 |  | 
 | 	rv = vio_register_driver(&vcc_driver); | 
 | 	if (rv) { | 
 | 		pr_err("VCC: VIO driver registration failed\n"); | 
 | 		vcc_tty_exit(); | 
 | 	} else { | 
 | 		vccdbg("VCC: VIO driver registered successfully\n"); | 
 | 	} | 
 |  | 
 | 	return rv; | 
 | } | 
 |  | 
 | static void __exit vcc_exit(void) | 
 | { | 
 | 	vio_unregister_driver(&vcc_driver); | 
 | 	vccdbg("VCC: VIO driver unregistered\n"); | 
 | 	vcc_tty_exit(); | 
 | 	vccdbg("VCC: TTY driver unregistered\n"); | 
 | } | 
 |  | 
 | module_init(vcc_init); | 
 | module_exit(vcc_exit); |