cregit-Linux how code gets into the kernel

Release 4.11 drivers/tty/serdev/serdev-ttyport.c

/*
 * Copyright (C) 2016-2017 Linaro Ltd., Rob Herring <robh@kernel.org>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 and
 * only version 2 as published by the Free Software Foundation.
 *
 * 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.
 */
#include <linux/kernel.h>
#include <linux/serdev.h>
#include <linux/tty.h>
#include <linux/tty_driver.h>


#define SERPORT_ACTIVE		1


struct serport {
	
struct tty_port *port;
	
struct tty_struct *tty;
	
struct tty_driver *tty_drv;
	
int tty_idx;
	
unsigned long flags;
};

/*
 * Callback functions from the tty port.
 */


static int ttyport_receive_buf(struct tty_port *port, const unsigned char *cp, const unsigned char *fp, size_t count) { struct serdev_controller *ctrl = port->client_data; struct serport *serport = serdev_controller_get_drvdata(ctrl); if (!test_bit(SERPORT_ACTIVE, &serport->flags)) return 0; return serdev_controller_receive_buf(ctrl, cp, count); }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring71100.00%1100.00%
Total71100.00%1100.00%


static void ttyport_write_wakeup(struct tty_port *port) { struct serdev_controller *ctrl = port->client_data; struct serport *serport = serdev_controller_get_drvdata(ctrl); if (!test_and_clear_bit(TTY_DO_WRITE_WAKEUP, &port->tty->flags)) return; if (test_bit(SERPORT_ACTIVE, &serport->flags)) serdev_controller_write_wakeup(ctrl); }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring63100.00%1100.00%
Total63100.00%1100.00%

static const struct tty_port_client_operations client_ops = { .receive_buf = ttyport_receive_buf, .write_wakeup = ttyport_write_wakeup, }; /* * Callback functions from the serdev core. */
static int ttyport_write_buf(struct serdev_controller *ctrl, const unsigned char *data, size_t len) { struct serport *serport = serdev_controller_get_drvdata(ctrl); struct tty_struct *tty = serport->tty; if (!test_bit(SERPORT_ACTIVE, &serport->flags)) return 0; set_bit(TTY_DO_WRITE_WAKEUP, &tty->flags); return tty->ops->write(serport->tty, data, len); }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring81100.00%1100.00%
Total81100.00%1100.00%


static void ttyport_write_flush(struct serdev_controller *ctrl) { struct serport *serport = serdev_controller_get_drvdata(ctrl); struct tty_struct *tty = serport->tty; tty_driver_flush_buffer(tty); }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring35100.00%1100.00%
Total35100.00%1100.00%


static int ttyport_write_room(struct serdev_controller *ctrl) { struct serport *serport = serdev_controller_get_drvdata(ctrl); struct tty_struct *tty = serport->tty; return tty_write_room(tty); }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring36100.00%1100.00%
Total36100.00%1100.00%


static int ttyport_open(struct serdev_controller *ctrl) { struct serport *serport = serdev_controller_get_drvdata(ctrl); struct tty_struct *tty; struct ktermios ktermios; tty = tty_init_dev(serport->tty_drv, serport->tty_idx); if (IS_ERR(tty)) return PTR_ERR(tty); serport->tty = tty; serport->port->client_ops = &client_ops; serport->port->client_data = ctrl; if (tty->ops->open) tty->ops->open(serport->tty, NULL); else tty_port_open(serport->port, tty, NULL); /* Bring the UART into a known 8 bits no parity hw fc state */ ktermios = tty->termios; ktermios.c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP | INLCR | IGNCR | ICRNL | IXON); ktermios.c_oflag &= ~OPOST; ktermios.c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN); ktermios.c_cflag &= ~(CSIZE | PARENB); ktermios.c_cflag |= CS8; ktermios.c_cflag |= CRTSCTS; tty_set_termios(tty, &ktermios); set_bit(SERPORT_ACTIVE, &serport->flags); tty_unlock(serport->tty); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring20494.01%150.00%
Dan Carpenter135.99%150.00%
Total217100.00%2100.00%


static void ttyport_close(struct serdev_controller *ctrl) { struct serport *serport = serdev_controller_get_drvdata(ctrl); struct tty_struct *tty = serport->tty; clear_bit(SERPORT_ACTIVE, &serport->flags); if (tty->ops->close) tty->ops->close(tty, NULL); tty_release_struct(tty, serport->tty_idx); }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring68100.00%1100.00%
Total68100.00%1100.00%


static unsigned int ttyport_set_baudrate(struct serdev_controller *ctrl, unsigned int speed) { struct serport *serport = serdev_controller_get_drvdata(ctrl); struct tty_struct *tty = serport->tty; struct ktermios ktermios = tty->termios; ktermios.c_cflag &= ~CBAUD; tty_termios_encode_baud_rate(&ktermios, speed, speed); /* tty_set_termios() return not checked as it is always 0 */ tty_set_termios(tty, &ktermios); return speed; }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring72100.00%1100.00%
Total72100.00%1100.00%


static void ttyport_set_flow_control(struct serdev_controller *ctrl, bool enable) { struct serport *serport = serdev_controller_get_drvdata(ctrl); struct tty_struct *tty = serport->tty; struct ktermios ktermios = tty->termios; if (enable) ktermios.c_cflag |= CRTSCTS; else ktermios.c_cflag &= ~CRTSCTS; tty_set_termios(tty, &ktermios); }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring67100.00%1100.00%
Total67100.00%1100.00%

static const struct serdev_controller_ops ctrl_ops = { .write_buf = ttyport_write_buf, .write_flush = ttyport_write_flush, .write_room = ttyport_write_room, .open = ttyport_open, .close = ttyport_close, .set_flow_control = ttyport_set_flow_control, .set_baudrate = ttyport_set_baudrate, };
struct device *serdev_tty_port_register(struct tty_port *port, struct device *parent, struct tty_driver *drv, int idx) { struct serdev_controller *ctrl; struct serport *serport; int ret; if (!port || !drv || !parent) return ERR_PTR(-ENODEV); ctrl = serdev_controller_alloc(parent, sizeof(struct serport)); if (!ctrl) return ERR_PTR(-ENOMEM); serport = serdev_controller_get_drvdata(ctrl); serport->port = port; serport->tty_idx = idx; serport->tty_drv = drv; ctrl->ops = &ctrl_ops; ret = serdev_controller_add(ctrl); if (ret) goto err_controller_put; dev_info(&ctrl->dev, "tty port %s%d registered\n", drv->name, idx); return &ctrl->dev; err_controller_put: serdev_controller_put(ctrl); return ERR_PTR(ret); }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring162100.00%1100.00%
Total162100.00%1100.00%


void serdev_tty_port_unregister(struct tty_port *port) { struct serdev_controller *ctrl = port->client_data; struct serport *serport = serdev_controller_get_drvdata(ctrl); if (!serport) return; serdev_controller_remove(ctrl); port->client_ops = NULL; port->client_data = NULL; serdev_controller_put(ctrl); }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring57100.00%1100.00%
Total57100.00%1100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Rob Herring102398.75%150.00%
Dan Carpenter131.25%150.00%
Total1036100.00%2100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.