cregit-Linux how code gets into the kernel

Release 4.14 arch/um/drivers/pty.c

Directory: arch/um/drivers
/*
 * Copyright (C) 2001 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
 * Licensed under the GPL
 */

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <fcntl.h>
#include <string.h>
#include <termios.h>
#include <sys/stat.h>
#include "chan_user.h"
#include <os.h>
#include <um_malloc.h>


struct pty_chan {
	
void (*announce)(char *dev_name, int dev);
	
int dev;
	
int raw;
	
struct termios tt;
	
char dev_name[sizeof("/dev/pts/0123456\0")];
};


static void *pty_chan_init(char *str, int device, const struct chan_opts *opts) { struct pty_chan *data; data = uml_kmalloc(sizeof(*data), UM_GFP_KERNEL); if (data == NULL) return NULL; *data = ((struct pty_chan) { .announce = opts->announce, .dev = device, .raw = opts->raw }); return data; }

Contributors

PersonTokensPropCommitsCommitProp
Jeff Dike7695.00%685.71%
Paolo 'Blaisorblade' Giarrusso45.00%114.29%
Total80100.00%7100.00%


static int pts_open(int input, int output, int primary, void *d, char **dev_out) { struct pty_chan *data = d; char *dev; int fd, err; fd = get_pty(); if (fd < 0) { err = -errno; printk(UM_KERN_ERR "open_pts : Failed to open pts\n"); return err; } if (data->raw) { CATCH_EINTR(err = tcgetattr(fd, &data->tt)); if (err) goto out_close; err = raw(fd); if (err) goto out_close; } dev = ptsname(fd); sprintf(data->dev_name, "%s", dev); *dev_out = data->dev_name; if (data->announce) (*data->announce)(dev, data->dev); return fd; out_close: close(fd); return err; }

Contributors

PersonTokensPropCommitsCommitProp
Jeff Dike16597.63%787.50%
Paolo 'Blaisorblade' Giarrusso42.37%112.50%
Total169100.00%8100.00%


static int getmaster(char *line) { struct stat buf; char *pty, *bank, *cp; int master, err; pty = &line[strlen("/dev/ptyp")]; for (bank = "pqrs"; *bank; bank++) { line[strlen("/dev/pty")] = *bank; *pty = '0'; /* Did we hit the end ? */ if ((stat(line, &buf) < 0) && (errno == ENOENT)) break; for (cp = "0123456789abcdef"; *cp; cp++) { *pty = *cp; master = open(line, O_RDWR); if (master >= 0) { char *tp = &line[strlen("/dev/")]; /* verify slave side is usable */ *tp = 't'; err = access(line, R_OK | W_OK); *tp = 'p'; if (!err) return master; close(master); } } } printk(UM_KERN_ERR "getmaster - no usable host pty devices\n"); return -ENOENT; }

Contributors

PersonTokensPropCommitsCommitProp
Jeff Dike18397.86%375.00%
Paolo 'Blaisorblade' Giarrusso42.14%125.00%
Total187100.00%4100.00%


static int pty_open(int input, int output, int primary, void *d, char **dev_out) { struct pty_chan *data = d; int fd, err; char dev[sizeof("/dev/ptyxx\0")] = "/dev/ptyxx"; fd = getmaster(dev); if (fd < 0) return fd; if (data->raw) { err = raw(fd); if (err) { close(fd); return err; } } if (data->announce) (*data->announce)(dev, data->dev); sprintf(data->dev_name, "%s", dev); *dev_out = data->dev_name; return fd; }

Contributors

PersonTokensPropCommitsCommitProp
Jeff Dike133100.00%6100.00%
Total133100.00%6100.00%

const struct chan_ops pty_ops = { .type = "pty", .init = pty_chan_init, .open = pty_open, .close = generic_close, .read = generic_read, .write = generic_write, .console_write = generic_console_write, .window_size = generic_window_size, .free = generic_free, .winch = 0, }; const struct chan_ops pts_ops = { .type = "pts", .init = pty_chan_init, .open = pts_open, .close = generic_close, .read = generic_read, .write = generic_write, .console_write = generic_console_write, .window_size = generic_window_size, .free = generic_free, .winch = 0, };

Overall Contributors

PersonTokensPropCommitsCommitProp
Jeff Dike73797.62%1381.25%
Paolo 'Blaisorblade' Giarrusso162.12%212.50%
Al Viro20.26%16.25%
Total755100.00%16100.00%
Directory: arch/um/drivers
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.