Lines Matching +full:port +full:-
1 // SPDX-License-Identifier: GPL-2.0
46 /* Microseconds that thread will delay waiting for a vcc port ref */
53 #define VCC_CTL_BREAK -1
54 #define VCC_CTL_HUP -2
108 * vcc_table_add() - Add VCC port to the VCC table
109 * @port: pointer to the VCC port
111 * Return: index of the port in the VCC table on success,
112 * -1 on failure
114 static int vcc_table_add(struct vcc_port *port) in vcc_table_add() argument
122 vcc_table[i] = port; in vcc_table_add()
131 return -1; in vcc_table_add()
135 * vcc_table_remove() - Removes a VCC port from the VCC table
151 * vcc_get() - Gets a reference to VCC port
155 * Return: reference to the VCC port, if found
156 * NULL, if port not found
160 struct vcc_port *port; in vcc_get() local
166 port = vcc_table[index]; in vcc_get()
167 if (!port) { in vcc_get()
173 if (port->excl_locked) { in vcc_get()
178 port->refcnt++; in vcc_get()
180 return port; in vcc_get()
183 if (port->refcnt) { in vcc_get()
193 port->refcnt++; in vcc_get()
194 port->excl_locked = true; in vcc_get()
197 return port; in vcc_get()
201 * vcc_put() - Returns a reference to VCC port
202 * @port: pointer to VCC port
208 static void vcc_put(struct vcc_port *port, bool excl) in vcc_put() argument
212 if (!port) in vcc_put()
218 if (WARN_ON((excl && !port->excl_locked) || in vcc_put()
219 (!excl && port->excl_locked))) in vcc_put()
222 port->refcnt--; in vcc_put()
225 port->excl_locked = false; in vcc_put()
232 * vcc_get_ne() - Get a non-exclusive reference to VCC port
235 * Gets a non-exclusive reference to VCC port, if it's not removed
237 * Return: pointer to the VCC port, if found
238 * NULL, if port not found
242 struct vcc_port *port; in vcc_get_ne() local
244 port = vcc_get(index, false); in vcc_get_ne()
246 if (port && port->removed) { in vcc_get_ne()
247 vcc_put(port, false); in vcc_get_ne()
251 return port; in vcc_get_ne()
254 static void vcc_kick_rx(struct vcc_port *port) in vcc_kick_rx() argument
256 struct vio_driver_state *vio = &port->vio; in vcc_kick_rx()
258 assert_spin_locked(&port->lock); in vcc_kick_rx()
260 if (!timer_pending(&port->rx_timer) && !port->removed) { in vcc_kick_rx()
261 disable_irq_nosync(vio->vdev->rx_irq); in vcc_kick_rx()
262 port->rx_timer.expires = (jiffies + 1); in vcc_kick_rx()
263 add_timer(&port->rx_timer); in vcc_kick_rx()
267 static void vcc_kick_tx(struct vcc_port *port) in vcc_kick_tx() argument
269 assert_spin_locked(&port->lock); in vcc_kick_tx()
271 if (!timer_pending(&port->tx_timer) && !port->removed) { in vcc_kick_tx()
272 port->tx_timer.expires = (jiffies + 1); in vcc_kick_tx()
273 add_timer(&port->tx_timer); in vcc_kick_tx()
279 if (WARN_ON(!tty || !tty->port)) in vcc_rx_check()
285 if (test_bit(TTY_THROTTLED, &tty->flags) || in vcc_rx_check()
286 (tty_buffer_request_room(tty->port, VCC_BUFF_LEN) < VCC_BUFF_LEN)) in vcc_rx_check()
296 if (WARN_ON(!tty || !tty->port)) in vcc_rx()
299 len = tty_insert_flip_string(tty->port, buf, size); in vcc_rx()
301 tty_flip_buffer_push(tty->port); in vcc_rx()
306 static int vcc_ldc_read(struct vcc_port *port) in vcc_ldc_read() argument
308 struct vio_driver_state *vio = &port->vio; in vcc_ldc_read()
313 tty = port->tty; in vcc_ldc_read()
315 rv = ldc_rx_reset(vio->lp); in vcc_ldc_read()
323 vcc_kick_rx(port); in vcc_ldc_read()
327 vccdbgl(vio->lp); in vcc_ldc_read()
329 rv = ldc_read(vio->lp, &pkt, sizeof(pkt)); in vcc_ldc_read()
346 rv = -ECONNRESET; in vcc_ldc_read()
359 struct vcc_port *port = from_timer(port, t, rx_timer); in vcc_rx_timer() local
364 spin_lock_irqsave(&port->lock, flags); in vcc_rx_timer()
365 port->rx_timer.expires = 0; in vcc_rx_timer()
367 vio = &port->vio; in vcc_rx_timer()
369 enable_irq(vio->vdev->rx_irq); in vcc_rx_timer()
371 if (!port->tty || port->removed) in vcc_rx_timer()
374 rv = vcc_ldc_read(port); in vcc_rx_timer()
375 if (rv == -ECONNRESET) in vcc_rx_timer()
379 spin_unlock_irqrestore(&port->lock, flags); in vcc_rx_timer()
380 vcc_put(port, false); in vcc_rx_timer()
385 struct vcc_port *port = from_timer(port, t, tx_timer); in vcc_tx_timer() local
391 spin_lock_irqsave(&port->lock, flags); in vcc_tx_timer()
392 port->tx_timer.expires = 0; in vcc_tx_timer()
394 if (!port->tty || port->removed) in vcc_tx_timer()
397 tosend = min(VCC_BUFF_LEN, port->chars_in_buffer); in vcc_tx_timer()
401 pkt = &port->buffer; in vcc_tx_timer()
402 pkt->tag.type = VIO_TYPE_DATA; in vcc_tx_timer()
403 pkt->tag.stype = tosend; in vcc_tx_timer()
404 vccdbgl(port->vio.lp); in vcc_tx_timer()
406 rv = ldc_write(port->vio.lp, pkt, (VIO_TAG_SIZE + tosend)); in vcc_tx_timer()
411 vcc_kick_tx(port); in vcc_tx_timer()
413 struct tty_struct *tty = port->tty; in vcc_tx_timer()
415 port->chars_in_buffer = 0; in vcc_tx_timer()
421 spin_unlock_irqrestore(&port->lock, flags); in vcc_tx_timer()
422 vcc_put(port, false); in vcc_tx_timer()
426 * vcc_event() - LDC event processing engine
435 struct vcc_port *port; in vcc_event() local
439 port = arg; in vcc_event()
440 vio = &port->vio; in vcc_event()
442 spin_lock_irqsave(&port->lock, flags); in vcc_event()
451 rv = vcc_ldc_read(port); in vcc_event()
452 if (rv == -ECONNRESET) in vcc_event()
460 spin_unlock_irqrestore(&port->lock, flags); in vcc_event()
481 struct vcc_port *port; in domain_show() local
484 port = dev_get_drvdata(dev); in domain_show()
485 if (!port) in domain_show()
486 return -ENODEV; in domain_show()
488 rv = scnprintf(buf, PAGE_SIZE, "%s\n", port->domain); in domain_show()
493 static int vcc_send_ctl(struct vcc_port *port, int ctl) in vcc_send_ctl() argument
502 rv = ldc_write(port->vio.lp, &pkt, sizeof(pkt.tag)); in vcc_send_ctl()
513 struct vcc_port *port; in break_store() local
518 port = dev_get_drvdata(dev); in break_store()
519 if (!port) in break_store()
520 return -ENODEV; in break_store()
522 spin_lock_irqsave(&port->lock, flags); in break_store()
525 rv = -EINVAL; in break_store()
526 else if (vcc_send_ctl(port, VCC_CTL_BREAK) < 0) in break_store()
527 vcc_kick_tx(port); in break_store()
529 spin_unlock_irqrestore(&port->lock, flags); in break_store()
549 * vcc_probe() - Initialize VCC port
550 * @vdev: Pointer to VIO device of the new VCC port
553 * Initializes a VCC port to receive serial console data from
563 struct vcc_port *port; in vcc_probe() local
570 vccdbg("VCC: name=%s\n", dev_name(&vdev->dev)); in vcc_probe()
574 return -ENODEV; in vcc_probe()
577 port = kzalloc(sizeof(struct vcc_port), GFP_KERNEL); in vcc_probe()
578 if (!port) in vcc_probe()
579 return -ENOMEM; in vcc_probe()
581 name = kstrdup(dev_name(&vdev->dev), GFP_KERNEL); in vcc_probe()
583 rv = -ENOMEM; in vcc_probe()
587 rv = vio_driver_init(&port->vio, vdev, VDEV_CONSOLE_CON, vcc_versions, in vcc_probe()
592 port->vio.debug = vcc_dbg_vio; in vcc_probe()
595 rv = vio_ldc_alloc(&port->vio, &vcc_ldc_cfg, port); in vcc_probe()
599 spin_lock_init(&port->lock); in vcc_probe()
601 port->index = vcc_table_add(port); in vcc_probe()
602 if (port->index == -1) { in vcc_probe()
604 rv = -ENOMEM; in vcc_probe()
609 dev = tty_register_device(vcc_tty_driver, port->index, &vdev->dev); in vcc_probe()
619 rv = -ENXIO; in vcc_probe()
624 domain = mdesc_get_property(hp, node, "vcc-domain-name", NULL); in vcc_probe()
626 rv = -ENXIO; in vcc_probe()
630 port->domain = kstrdup(domain, GFP_KERNEL); in vcc_probe()
631 if (!port->domain) { in vcc_probe()
632 rv = -ENOMEM; in vcc_probe()
639 rv = sysfs_create_group(&vdev->dev.kobj, &vcc_attribute_group); in vcc_probe()
643 timer_setup(&port->rx_timer, vcc_rx_timer, 0); in vcc_probe()
644 timer_setup(&port->tx_timer, vcc_tx_timer, 0); in vcc_probe()
646 dev_set_drvdata(&vdev->dev, port); in vcc_probe()
649 * IRQs until the port is up. in vcc_probe()
651 disable_irq_nosync(vdev->rx_irq); in vcc_probe()
652 vio_port_up(&port->vio); in vcc_probe()
653 enable_irq(vdev->rx_irq); in vcc_probe()
658 kfree(port->domain); in vcc_probe()
660 tty_unregister_device(vcc_tty_driver, port->index); in vcc_probe()
662 vcc_table_remove(port->index); in vcc_probe()
664 vio_ldc_free(&port->vio); in vcc_probe()
668 kfree(port); in vcc_probe()
674 * vcc_remove() - Terminate a VCC port
675 * @vdev: Pointer to VIO device of the VCC port
677 * Terminates a VCC port. Sets up the teardown of TTY and
684 struct vcc_port *port = dev_get_drvdata(&vdev->dev); in vcc_remove() local
686 del_timer_sync(&port->rx_timer); in vcc_remove()
687 del_timer_sync(&port->tx_timer); in vcc_remove()
693 if (port->tty) in vcc_remove()
694 tty_vhangup(port->tty); in vcc_remove()
697 * clients to this port. This cannot fail. in vcc_remove()
699 vcc_get(port->index, true); in vcc_remove()
701 tty_unregister_device(vcc_tty_driver, port->index); in vcc_remove()
703 del_timer_sync(&port->vio.timer); in vcc_remove()
704 vio_ldc_free(&port->vio); in vcc_remove()
705 sysfs_remove_group(&vdev->dev.kobj, &vcc_attribute_group); in vcc_remove()
706 dev_set_drvdata(&vdev->dev, NULL); in vcc_remove()
707 if (port->tty) { in vcc_remove()
708 port->removed = true; in vcc_remove()
709 vcc_put(port, true); in vcc_remove()
711 vcc_table_remove(port->index); in vcc_remove()
713 kfree(port->vio.name); in vcc_remove()
714 kfree(port->domain); in vcc_remove()
715 kfree(port); in vcc_remove()
721 .type = "vcc-port",
736 struct vcc_port *port; in vcc_open() local
738 if (tty->count > 1) in vcc_open()
739 return -EBUSY; in vcc_open()
741 port = vcc_get_ne(tty->index); in vcc_open()
742 if (unlikely(!port)) { in vcc_open()
743 pr_err("VCC: open: Failed to find VCC port\n"); in vcc_open()
744 return -ENODEV; in vcc_open()
747 if (unlikely(!port->vio.lp)) { in vcc_open()
749 vcc_put(port, false); in vcc_open()
750 return -EPIPE; in vcc_open()
752 vccdbgl(port->vio.lp); in vcc_open()
754 vcc_put(port, false); in vcc_open()
756 if (unlikely(!tty->port)) { in vcc_open()
757 pr_err("VCC: open: TTY port not found\n"); in vcc_open()
758 return -ENXIO; in vcc_open()
761 if (unlikely(!tty->port->ops)) { in vcc_open()
763 return -ENXIO; in vcc_open()
766 return tty_port_open(tty->port, tty, vcc_file); in vcc_open()
771 if (unlikely(tty->count > 1)) in vcc_close()
774 if (unlikely(!tty->port)) { in vcc_close()
775 pr_err("VCC: close: TTY port not found\n"); in vcc_close()
779 tty_port_close(tty->port, tty, vcc_file); in vcc_close()
782 static void vcc_ldc_hup(struct vcc_port *port) in vcc_ldc_hup() argument
786 spin_lock_irqsave(&port->lock, flags); in vcc_ldc_hup()
788 if (vcc_send_ctl(port, VCC_CTL_HUP) < 0) in vcc_ldc_hup()
789 vcc_kick_tx(port); in vcc_ldc_hup()
791 spin_unlock_irqrestore(&port->lock, flags); in vcc_ldc_hup()
796 struct vcc_port *port; in vcc_hangup() local
798 port = vcc_get_ne(tty->index); in vcc_hangup()
799 if (unlikely(!port)) { in vcc_hangup()
800 pr_err("VCC: hangup: Failed to find VCC port\n"); in vcc_hangup()
804 if (unlikely(!tty->port)) { in vcc_hangup()
805 pr_err("VCC: hangup: TTY port not found\n"); in vcc_hangup()
806 vcc_put(port, false); in vcc_hangup()
810 vcc_ldc_hup(port); in vcc_hangup()
812 vcc_put(port, false); in vcc_hangup()
814 tty_port_hangup(tty->port); in vcc_hangup()
819 struct vcc_port *port; in vcc_write() local
824 int rv = -EINVAL; in vcc_write()
826 port = vcc_get_ne(tty->index); in vcc_write()
827 if (unlikely(!port)) { in vcc_write()
828 pr_err("VCC: write: Failed to find VCC port"); in vcc_write()
829 return -ENODEV; in vcc_write()
832 spin_lock_irqsave(&port->lock, flags); in vcc_write()
834 pkt = &port->buffer; in vcc_write()
835 pkt->tag.type = VIO_TYPE_DATA; in vcc_write()
840 (VCC_BUFF_LEN - port->chars_in_buffer)); in vcc_write()
845 memcpy(&pkt->data[port->chars_in_buffer], &buf[total_sent], in vcc_write()
847 port->chars_in_buffer += tosend; in vcc_write()
848 pkt->tag.stype = tosend; in vcc_write()
850 vccdbg("TAG [%02x:%02x:%04x:%08x]\n", pkt->tag.type, in vcc_write()
851 pkt->tag.stype, pkt->tag.stype_env, pkt->tag.sid); in vcc_write()
852 vccdbg("DATA [%s]\n", pkt->data); in vcc_write()
853 vccdbgl(port->vio.lp); in vcc_write()
859 rv = ldc_write(port->vio.lp, pkt, (VIO_TAG_SIZE + tosend)); in vcc_write()
864 count -= tosend; in vcc_write()
866 vcc_kick_tx(port); in vcc_write()
870 port->chars_in_buffer = 0; in vcc_write()
873 spin_unlock_irqrestore(&port->lock, flags); in vcc_write()
875 vcc_put(port, false); in vcc_write()
884 struct vcc_port *port; in vcc_write_room() local
887 port = vcc_get_ne(tty->index); in vcc_write_room()
888 if (unlikely(!port)) { in vcc_write_room()
889 pr_err("VCC: write_room: Failed to find VCC port\n"); in vcc_write_room()
893 num = VCC_BUFF_LEN - port->chars_in_buffer; in vcc_write_room()
895 vcc_put(port, false); in vcc_write_room()
902 struct vcc_port *port; in vcc_chars_in_buffer() local
905 port = vcc_get_ne(tty->index); in vcc_chars_in_buffer()
906 if (unlikely(!port)) { in vcc_chars_in_buffer()
907 pr_err("VCC: chars_in_buffer: Failed to find VCC port\n"); in vcc_chars_in_buffer()
911 num = port->chars_in_buffer; in vcc_chars_in_buffer()
913 vcc_put(port, false); in vcc_chars_in_buffer()
920 struct vcc_port *port; in vcc_break_ctl() local
923 port = vcc_get_ne(tty->index); in vcc_break_ctl()
924 if (unlikely(!port)) { in vcc_break_ctl()
925 pr_err("VCC: break_ctl: Failed to find VCC port\n"); in vcc_break_ctl()
926 return -ENODEV; in vcc_break_ctl()
931 vcc_put(port, false); in vcc_break_ctl()
935 spin_lock_irqsave(&port->lock, flags); in vcc_break_ctl()
937 if (vcc_send_ctl(port, VCC_CTL_BREAK) < 0) in vcc_break_ctl()
938 vcc_kick_tx(port); in vcc_break_ctl()
940 spin_unlock_irqrestore(&port->lock, flags); in vcc_break_ctl()
942 vcc_put(port, false); in vcc_break_ctl()
953 if (tty->index >= VCC_MAX_PORTS) in vcc_install()
954 return -EINVAL; in vcc_install()
962 return -ENOMEM; in vcc_install()
964 port_vcc = vcc_get(tty->index, true); in vcc_install()
966 pr_err("VCC: install: Failed to find VCC port\n"); in vcc_install()
967 tty->port = NULL; in vcc_install()
969 return -ENODEV; in vcc_install()
973 port_tty->ops = &vcc_port_ops; in vcc_install()
974 tty->port = port_tty; in vcc_install()
976 port_vcc->tty = tty; in vcc_install()
985 struct vcc_port *port; in vcc_cleanup() local
987 port = vcc_get(tty->index, true); in vcc_cleanup()
988 if (port) { in vcc_cleanup()
989 port->tty = NULL; in vcc_cleanup()
991 if (port->removed) { in vcc_cleanup()
992 vcc_table_remove(tty->index); in vcc_cleanup()
993 kfree(port->vio.name); in vcc_cleanup()
994 kfree(port->domain); in vcc_cleanup()
995 kfree(port); in vcc_cleanup()
997 vcc_put(port, true); in vcc_cleanup()
1001 tty_port_destroy(tty->port); in vcc_cleanup()
1002 kfree(tty->port); in vcc_cleanup()
1003 tty->port = NULL; in vcc_cleanup()
1030 vcc_tty_driver->driver_name = "vcc"; in vcc_tty_init()
1031 vcc_tty_driver->name = "vcc"; in vcc_tty_init()
1033 vcc_tty_driver->minor_start = VCC_MINOR_START; in vcc_tty_init()
1034 vcc_tty_driver->type = TTY_DRIVER_TYPE_SYSTEM; in vcc_tty_init()
1035 vcc_tty_driver->init_termios = vcc_tty_termios; in vcc_tty_init()