Lines Matching +full:byte +full:- +full:len
1 // SPDX-License-Identifier: GPL-2.0
2 /* ePAPR hypervisor byte channel device driver
4 * Copyright 2009-2011 Freescale Semiconductor, Inc.
9 * ePAPR hypervisor byte channels.
11 * 1) An early-console (udbg) driver. This provides early console output
12 * through a byte channel. The byte channel handle must be specified in a
15 * 2) A normal console driver. Output is sent to the byte channel designated
19 * 3) A tty driver, which is used to handle user-space input and output. The
20 * byte channel used for the console is designated as the default tty.
43 /* Per-byte channel private data */
59 /* Array of byte channel objects */
62 /* Byte channel handle for stdout (and stdin), taken from device tree */
65 /* Virtual IRQ for the byte channel handle for stdin, taken from device tree */
73 * Unlike a serial device, byte channels have no mechanism for disabling their
81 * 1. The tty layer makes two back-to-back calls to ehv_bc_tty_write()
88 if (!bc->tx_irq_enabled) { in enable_tx_interrupt()
89 enable_irq(bc->tx_irq); in enable_tx_interrupt()
90 bc->tx_irq_enabled = 1; in enable_tx_interrupt()
96 if (bc->tx_irq_enabled) { in disable_tx_interrupt()
97 disable_irq_nosync(bc->tx_irq); in disable_tx_interrupt()
98 bc->tx_irq_enabled = 0; in disable_tx_interrupt()
103 * find the byte channel handle to use for the console
105 * The byte channel to be used for the console is specified via a "stdout"
114 * care if it's compatible with "epapr,hv-byte-channel", because that in find_console_handle()
115 * indicates that it's a byte channel node. in find_console_handle()
117 if (!np || !of_device_is_compatible(np, "epapr,hv-byte-channel")) in find_console_handle()
122 pr_err("ehv-bc: no 'interrupts' property in %pOF node\n", np); in find_console_handle()
127 * The 'hv-handle' property contains the handle for this byte channel. in find_console_handle()
129 iprop = of_get_property(np, "hv-handle", NULL); in find_console_handle()
131 pr_err("ehv-bc: no 'hv-handle' property in %pOFn node\n", in find_console_handle()
148 memset(&buffer[c], 0, sizeof(buffer) - c); in local_ev_byte_channel_send()
159 * send a byte to a byte channel, wait if necessary
161 * This function sends a byte to a byte channel, and it waits and
162 * retries if the byte channel is full. It returns if the character
195 * byte channel handle for stdout.
205 /* Verify the byte channel handle */ in udbg_init_ehv_bc()
214 udbg_printf("ehv-bc: early console using byte channel handle %u\n", in udbg_init_ehv_bc()
225 * Byte channel console sending worker function.
233 unsigned int len; in ehv_bc_console_byte_channel_send() local
237 len = min_t(unsigned int, count, EV_BYTE_CHANNEL_MAX_BYTES); in ehv_bc_console_byte_channel_send()
239 ret = local_ev_byte_channel_send(handle, &len, s); in ehv_bc_console_byte_channel_send()
241 count -= len; in ehv_bc_console_byte_channel_send()
242 s += len; in ehv_bc_console_byte_channel_send()
272 if (j >= (EV_BYTE_CHANNEL_MAX_BYTES - 1)) { in ehv_bc_console_write()
285 * for one with ->device and then calls that method. On success, it expects
286 * the passed-in int* to contain the minor number to use.
290 *index = co->index; in ehv_bc_console_device()
306 * available, so here is where we determine the byte channel handle and IRQ for
313 pr_debug("ehv-bc: stdout is not a byte channel\n"); in ehv_bc_console_init()
314 return -ENODEV; in ehv_bc_console_init()
318 /* Print a friendly warning if the user chose the wrong byte channel in ehv_bc_console_init()
322 pr_warn("ehv-bc: udbg handle %u is not the stdout handle\n", in ehv_bc_console_init()
328 byte channels here, either, since we only care about one. */ in ehv_bc_console_init()
333 pr_info("ehv-bc: registered console driver for byte channel %u\n", in ehv_bc_console_init()
343 * byte channel receive interrupt handler
345 * This ISR is called whenever data is available on a byte channel.
350 unsigned int rx_count, tx_count, len; in ehv_bc_tty_rx_isr() local
356 * if it can handle that much. We want to ensure that every byte we in ehv_bc_tty_rx_isr()
357 * read from the byte channel will be accepted by the TTY layer. in ehv_bc_tty_rx_isr()
359 ev_byte_channel_poll(bc->handle, &rx_count, &tx_count); in ehv_bc_tty_rx_isr()
360 count = tty_buffer_request_room(&bc->port, rx_count); in ehv_bc_tty_rx_isr()
369 len = min_t(unsigned int, count, sizeof(buffer)); in ehv_bc_tty_rx_isr()
371 /* Read some data from the byte channel. This function will in ehv_bc_tty_rx_isr()
374 ev_byte_channel_receive(bc->handle, &len, buffer); in ehv_bc_tty_rx_isr()
376 /* 'len' is now the amount of data that's been received. 'len' in ehv_bc_tty_rx_isr()
381 ret = tty_insert_flip_string(&bc->port, buffer, len); in ehv_bc_tty_rx_isr()
384 * If it's not equal to 'len', then it means the buffer is in ehv_bc_tty_rx_isr()
386 * exit gracefully, but we drop the last 'len - ret' characters in ehv_bc_tty_rx_isr()
387 * that we read from the byte channel. in ehv_bc_tty_rx_isr()
389 if (ret != len) in ehv_bc_tty_rx_isr()
392 count -= len; in ehv_bc_tty_rx_isr()
396 tty_flip_buffer_push(&bc->port); in ehv_bc_tty_rx_isr()
405 * data as possible from the transmit buffer to the byte channel.
410 unsigned int len, ret; in ehv_bc_tx_dequeue() local
414 spin_lock_irqsave(&bc->lock, flags); in ehv_bc_tx_dequeue()
415 len = min_t(unsigned int, in ehv_bc_tx_dequeue()
416 CIRC_CNT_TO_END(bc->head, bc->tail, BUF_SIZE), in ehv_bc_tx_dequeue()
419 ret = local_ev_byte_channel_send(bc->handle, &len, bc->buf + bc->tail); in ehv_bc_tx_dequeue()
421 /* 'len' is valid only if the return code is 0 or EV_EAGAIN */ in ehv_bc_tx_dequeue()
423 bc->tail = (bc->tail + len) & (BUF_SIZE - 1); in ehv_bc_tx_dequeue()
425 count = CIRC_CNT(bc->head, bc->tail, BUF_SIZE); in ehv_bc_tx_dequeue()
426 spin_unlock_irqrestore(&bc->lock, flags); in ehv_bc_tx_dequeue()
429 spin_lock_irqsave(&bc->lock, flags); in ehv_bc_tx_dequeue()
430 if (CIRC_CNT(bc->head, bc->tail, BUF_SIZE)) in ehv_bc_tx_dequeue()
439 spin_unlock_irqrestore(&bc->lock, flags); in ehv_bc_tx_dequeue()
443 * byte channel transmit interrupt handler
446 * characters on a byte channel.
453 tty_port_tty_wakeup(&bc->port); in ehv_bc_tty_tx_isr()
472 struct ehv_bc_data *bc = ttys->driver_data; in ehv_bc_tty_write()
474 unsigned int len; in ehv_bc_tty_write() local
478 spin_lock_irqsave(&bc->lock, flags); in ehv_bc_tty_write()
479 len = CIRC_SPACE_TO_END(bc->head, bc->tail, BUF_SIZE); in ehv_bc_tty_write()
480 if (count < len) in ehv_bc_tty_write()
481 len = count; in ehv_bc_tty_write()
482 if (len) { in ehv_bc_tty_write()
483 memcpy(bc->buf + bc->head, s, len); in ehv_bc_tty_write()
484 bc->head = (bc->head + len) & (BUF_SIZE - 1); in ehv_bc_tty_write()
486 spin_unlock_irqrestore(&bc->lock, flags); in ehv_bc_tty_write()
487 if (!len) in ehv_bc_tty_write()
490 s += len; in ehv_bc_tty_write()
491 count -= len; in ehv_bc_tty_write()
492 written += len; in ehv_bc_tty_write()
502 * why we initialize bc->ttys in ehv_bc_tty_port_activate() instead.
511 struct ehv_bc_data *bc = &bcs[ttys->index]; in ehv_bc_tty_open()
513 if (!bc->dev) in ehv_bc_tty_open()
514 return -ENODEV; in ehv_bc_tty_open()
516 return tty_port_open(&bc->port, ttys, filp); in ehv_bc_tty_open()
526 struct ehv_bc_data *bc = &bcs[ttys->index]; in ehv_bc_tty_close()
528 if (bc->dev) in ehv_bc_tty_close()
529 tty_port_close(&bc->port, ttys, filp); in ehv_bc_tty_close()
541 struct ehv_bc_data *bc = ttys->driver_data; in ehv_bc_tty_write_room()
545 spin_lock_irqsave(&bc->lock, flags); in ehv_bc_tty_write_room()
546 count = CIRC_SPACE(bc->head, bc->tail, BUF_SIZE); in ehv_bc_tty_write_room()
547 spin_unlock_irqrestore(&bc->lock, flags); in ehv_bc_tty_write_room()
566 struct ehv_bc_data *bc = ttys->driver_data; in ehv_bc_tty_throttle()
568 disable_irq(bc->rx_irq); in ehv_bc_tty_throttle()
580 struct ehv_bc_data *bc = ttys->driver_data; in ehv_bc_tty_unthrottle()
585 enable_irq(bc->rx_irq); in ehv_bc_tty_unthrottle()
590 struct ehv_bc_data *bc = ttys->driver_data; in ehv_bc_tty_hangup()
593 tty_port_hangup(&bc->port); in ehv_bc_tty_hangup()
618 * why we initialize tty_struct-related variables here.
626 ttys->driver_data = bc; in ehv_bc_tty_port_activate()
628 ret = request_irq(bc->rx_irq, ehv_bc_tty_rx_isr, 0, "ehv-bc", bc); in ehv_bc_tty_port_activate()
630 dev_err(bc->dev, "could not request rx irq %u (ret=%i)\n", in ehv_bc_tty_port_activate()
631 bc->rx_irq, ret); in ehv_bc_tty_port_activate()
636 bc->tx_irq_enabled = 1; in ehv_bc_tty_port_activate()
638 ret = request_irq(bc->tx_irq, ehv_bc_tty_tx_isr, 0, "ehv-bc", bc); in ehv_bc_tty_port_activate()
640 dev_err(bc->dev, "could not request tx irq %u (ret=%i)\n", in ehv_bc_tty_port_activate()
641 bc->tx_irq, ret); in ehv_bc_tty_port_activate()
642 free_irq(bc->rx_irq, bc); in ehv_bc_tty_port_activate()
647 * byte channel at once, so by default it's disabled. in ehv_bc_tty_port_activate()
658 free_irq(bc->tx_irq, bc); in ehv_bc_tty_port_shutdown()
659 free_irq(bc->rx_irq, bc); in ehv_bc_tty_port_shutdown()
669 struct device_node *np = pdev->dev.of_node; in ehv_bc_tty_probe()
677 iprop = of_get_property(np, "hv-handle", NULL); in ehv_bc_tty_probe()
679 dev_err(&pdev->dev, "no 'hv-handle' property in %pOFn node\n", in ehv_bc_tty_probe()
681 return -ENODEV; in ehv_bc_tty_probe()
686 * we probe the console byte channel node. in ehv_bc_tty_probe()
692 bc->handle = handle; in ehv_bc_tty_probe()
693 bc->head = 0; in ehv_bc_tty_probe()
694 bc->tail = 0; in ehv_bc_tty_probe()
695 spin_lock_init(&bc->lock); in ehv_bc_tty_probe()
697 bc->rx_irq = irq_of_parse_and_map(np, 0); in ehv_bc_tty_probe()
698 bc->tx_irq = irq_of_parse_and_map(np, 1); in ehv_bc_tty_probe()
699 if ((bc->rx_irq == NO_IRQ) || (bc->tx_irq == NO_IRQ)) { in ehv_bc_tty_probe()
700 dev_err(&pdev->dev, "no 'interrupts' property in %pOFn node\n", in ehv_bc_tty_probe()
702 ret = -ENODEV; in ehv_bc_tty_probe()
706 tty_port_init(&bc->port); in ehv_bc_tty_probe()
707 bc->port.ops = &ehv_bc_tty_port_ops; in ehv_bc_tty_probe()
709 bc->dev = tty_port_register_device(&bc->port, ehv_bc_driver, i, in ehv_bc_tty_probe()
710 &pdev->dev); in ehv_bc_tty_probe()
711 if (IS_ERR(bc->dev)) { in ehv_bc_tty_probe()
712 ret = PTR_ERR(bc->dev); in ehv_bc_tty_probe()
713 dev_err(&pdev->dev, "could not register tty (ret=%i)\n", ret); in ehv_bc_tty_probe()
717 dev_set_drvdata(&pdev->dev, bc); in ehv_bc_tty_probe()
719 dev_info(&pdev->dev, "registered /dev/%s%u for byte channel %u\n", in ehv_bc_tty_probe()
720 ehv_bc_driver->name, i, bc->handle); in ehv_bc_tty_probe()
725 tty_port_destroy(&bc->port); in ehv_bc_tty_probe()
726 irq_dispose_mapping(bc->tx_irq); in ehv_bc_tty_probe()
727 irq_dispose_mapping(bc->rx_irq); in ehv_bc_tty_probe()
734 { .compatible = "epapr,hv-byte-channel" },
740 .name = "ehv-bc",
748 * ehv_bc_init - ePAPR hypervisor byte channel driver initialization
759 pr_info("ePAPR hypervisor byte channel driver\n"); in ehv_bc_init()
761 /* Count the number of byte channels */ in ehv_bc_init()
762 for_each_compatible_node(np, NULL, "epapr,hv-byte-channel") in ehv_bc_init()
766 return -ENODEV; in ehv_bc_init()
770 * array, then you can use pointer math (e.g. "bc - bcs") to get its in ehv_bc_init()
775 return -ENOMEM; in ehv_bc_init()
784 driver->driver_name = "ehv-bc"; in ehv_bc_init()
785 driver->name = ehv_bc_console.name; in ehv_bc_init()
786 driver->type = TTY_DRIVER_TYPE_CONSOLE; in ehv_bc_init()
787 driver->subtype = SYSTEM_TYPE_CONSOLE; in ehv_bc_init()
788 driver->init_termios = tty_std_termios; in ehv_bc_init()
793 pr_err("ehv-bc: could not register tty driver (ret=%i)\n", ret); in ehv_bc_init()
801 pr_err("ehv-bc: could not register platform driver (ret=%i)\n", in ehv_bc_init()