diff options
Diffstat (limited to 'drivers/tty/pty.c')
| -rw-r--r-- | drivers/tty/pty.c | 462 | 
1 files changed, 249 insertions, 213 deletions
diff --git a/drivers/tty/pty.c b/drivers/tty/pty.c index 923a4858550..25c9bc78372 100644 --- a/drivers/tty/pty.c +++ b/drivers/tty/pty.c @@ -1,14 +1,9 @@  /* - *  linux/drivers/char/pty.c - *   *  Copyright (C) 1991, 1992  Linus Torvalds   *   *  Added support for a Unix98-style ptmx device.   *    -- C. Scott Ananian <cananian@alumni.princeton.edu>, 14-Jan-1998   * - *  When reading this code see also fs/devpts. In particular note that the - *  driver_data field is used by the devpts side as a binding to the devpts - *  inode.   */  #include <linux/module.h> @@ -23,19 +18,18 @@  #include <linux/major.h>  #include <linux/mm.h>  #include <linux/init.h> -#include <linux/smp_lock.h> -#include <linux/sysctl.h>  #include <linux/device.h>  #include <linux/uaccess.h>  #include <linux/bitops.h>  #include <linux/devpts_fs.h>  #include <linux/slab.h> +#include <linux/mutex.h> -#include <asm/system.h>  #ifdef CONFIG_UNIX98_PTYS  static struct tty_driver *ptm_driver;  static struct tty_driver *pts_driver; +static DEFINE_MUTEX(devpts_mutex);  #endif  static void pty_close(struct tty_struct *tty, struct file *filp) @@ -44,27 +38,34 @@ static void pty_close(struct tty_struct *tty, struct file *filp)  	if (tty->driver->subtype == PTY_TYPE_MASTER)  		WARN_ON(tty->count > 1);  	else { +		if (test_bit(TTY_IO_ERROR, &tty->flags)) +			return;  		if (tty->count > 2)  			return;  	} +	set_bit(TTY_IO_ERROR, &tty->flags);  	wake_up_interruptible(&tty->read_wait);  	wake_up_interruptible(&tty->write_wait);  	tty->packet = 0; +	/* Review - krefs on tty_link ?? */  	if (!tty->link)  		return; -	tty->link->packet = 0;  	set_bit(TTY_OTHER_CLOSED, &tty->link->flags);  	wake_up_interruptible(&tty->link->read_wait);  	wake_up_interruptible(&tty->link->write_wait);  	if (tty->driver->subtype == PTY_TYPE_MASTER) {  		set_bit(TTY_OTHER_CLOSED, &tty->flags);  #ifdef CONFIG_UNIX98_PTYS -		if (tty->driver == ptm_driver) -			devpts_pty_kill(tty->link); +		if (tty->driver == ptm_driver) { +			mutex_lock(&devpts_mutex); +			if (tty->link->driver_data) +				devpts_pty_kill(tty->link->driver_data); +			mutex_unlock(&devpts_mutex); +		}  #endif -		tty_unlock(); +		tty_unlock(tty);  		tty_vhangup(tty->link); -		tty_lock(); +		tty_lock(tty);  	}  } @@ -88,17 +89,13 @@ static void pty_unthrottle(struct tty_struct *tty)   *	pty_space	-	report space left for writing   *	@to: tty we are writing into   * - *	The tty buffers allow 64K but we sneak a peak and clip at 8K this - *	allows a lot of overspill room for echo and other fun messes to - *	be handled properly + *	Limit the buffer space used by ptys to 8k.   */  static int pty_space(struct tty_struct *to)  { -	int n = 8192 - to->buf.memory_used; -	if (n < 0) -		return 0; -	return n; +	int n = tty_buffer_space_avail(to->port); +	return min(n, 8192);  }  /** @@ -122,12 +119,10 @@ static int pty_write(struct tty_struct *tty, const unsigned char *buf, int c)  	if (c > 0) {  		/* Stuff the data into the input queue of the other end */ -		c = tty_insert_flip_string(to, buf, c); +		c = tty_insert_flip_string(to->port, buf, c);  		/* And shovel */ -		if (c) { -			tty_flip_buffer_push(to); -			tty_wakeup(tty); -		} +		if (c) +			tty_flip_buffer_push(to->port);  	}  	return c;  } @@ -173,6 +168,41 @@ static int pty_set_lock(struct tty_struct *tty, int __user *arg)  	return 0;  } +static int pty_get_lock(struct tty_struct *tty, int __user *arg) +{ +	int locked = test_bit(TTY_PTY_LOCK, &tty->flags); +	return put_user(locked, arg); +} + +/* Set the packet mode on a pty */ +static int pty_set_pktmode(struct tty_struct *tty, int __user *arg) +{ +	unsigned long flags; +	int pktmode; + +	if (get_user(pktmode, arg)) +		return -EFAULT; + +	spin_lock_irqsave(&tty->ctrl_lock, flags); +	if (pktmode) { +		if (!tty->packet) { +			tty->packet = 1; +			tty->link->ctrl_status = 0; +		} +	} else +		tty->packet = 0; +	spin_unlock_irqrestore(&tty->ctrl_lock, flags); + +	return 0; +} + +/* Get the packet mode of a pty */ +static int pty_get_pktmode(struct tty_struct *tty, int __user *arg) +{ +	int pktmode = tty->packet; +	return put_user(pktmode, arg); +} +  /* Send a signal to the slave */  static int pty_signal(struct tty_struct *tty, int sig)  { @@ -208,31 +238,31 @@ static void pty_flush_buffer(struct tty_struct *tty)  static int pty_open(struct tty_struct *tty, struct file *filp)  { -	int	retval = -ENODEV; -  	if (!tty || !tty->link) -		goto out; +		return -ENODEV; -	retval = -EIO;  	if (test_bit(TTY_OTHER_CLOSED, &tty->flags))  		goto out;  	if (test_bit(TTY_PTY_LOCK, &tty->link->flags))  		goto out; -	if (tty->link->count != 1) +	if (tty->driver->subtype == PTY_TYPE_SLAVE && tty->link->count != 1)  		goto out; +	clear_bit(TTY_IO_ERROR, &tty->flags);  	clear_bit(TTY_OTHER_CLOSED, &tty->link->flags);  	set_bit(TTY_THROTTLED, &tty->flags); -	retval = 0; +	return 0; +  out: -	return retval; +	set_bit(TTY_IO_ERROR, &tty->flags); +	return -EIO;  }  static void pty_set_termios(struct tty_struct *tty,  					struct ktermios *old_termios)  { -	tty->termios->c_cflag &= ~(CSIZE | PARENB); -	tty->termios->c_cflag |= (CS8 | CREAD); +	tty->termios.c_cflag &= ~(CSIZE | PARENB); +	tty->termios.c_cflag |= (CS8 | CREAD);  }  /** @@ -244,14 +274,14 @@ static void pty_set_termios(struct tty_struct *tty,   *	peform a terminal resize correctly   */ -int pty_resize(struct tty_struct *tty,  struct winsize *ws) +static int pty_resize(struct tty_struct *tty,  struct winsize *ws)  {  	struct pid *pgrp, *rpgrp;  	unsigned long flags;  	struct tty_struct *pty = tty->link;  	/* For a PTY we need to lock the tty side */ -	mutex_lock(&tty->termios_mutex); +	mutex_lock(&tty->winsize_mutex);  	if (!memcmp(ws, &tty->winsize, sizeof(*ws)))  		goto done; @@ -278,70 +308,131 @@ int pty_resize(struct tty_struct *tty,  struct winsize *ws)  	tty->winsize = *ws;  	pty->winsize = *ws;	/* Never used so will go away soon */  done: -	mutex_unlock(&tty->termios_mutex); +	mutex_unlock(&tty->winsize_mutex);  	return 0;  } -/* Traditional BSD devices */ -#ifdef CONFIG_LEGACY_PTYS - -static int pty_install(struct tty_driver *driver, struct tty_struct *tty) +/** + *	pty_common_install		-	set up the pty pair + *	@driver: the pty driver + *	@tty: the tty being instantiated + *	@bool: legacy, true if this is BSD style + * + *	Perform the initial set up for the tty/pty pair. Called from the + *	tty layer when the port is first opened. + * + *	Locking: the caller must hold the tty_mutex + */ +static int pty_common_install(struct tty_driver *driver, struct tty_struct *tty, +		bool legacy)  {  	struct tty_struct *o_tty; +	struct tty_port *ports[2];  	int idx = tty->index; -	int retval; +	int retval = -ENOMEM;  	o_tty = alloc_tty_struct();  	if (!o_tty) -		return -ENOMEM; +		goto err; +	ports[0] = kmalloc(sizeof **ports, GFP_KERNEL); +	ports[1] = kmalloc(sizeof **ports, GFP_KERNEL); +	if (!ports[0] || !ports[1]) +		goto err_free_tty;  	if (!try_module_get(driver->other->owner)) {  		/* This cannot in fact currently happen */ -		free_tty_struct(o_tty); -		return -ENOMEM; +		goto err_free_tty;  	}  	initialize_tty_struct(o_tty, driver->other, idx); -	/* We always use new tty termios data so we can do this -	   the easy way .. */ -	retval = tty_init_termios(tty); -	if (retval) -		goto free_mem_out; - -	retval = tty_init_termios(o_tty); -	if (retval) { -		tty_free_termios(tty); -		goto free_mem_out; +	if (legacy) { +		/* We always use new tty termios data so we can do this +		   the easy way .. */ +		retval = tty_init_termios(tty); +		if (retval) +			goto err_deinit_tty; + +		retval = tty_init_termios(o_tty); +		if (retval) +			goto err_free_termios; + +		driver->other->ttys[idx] = o_tty; +		driver->ttys[idx] = tty; +	} else { +		memset(&tty->termios_locked, 0, sizeof(tty->termios_locked)); +		tty->termios = driver->init_termios; +		memset(&o_tty->termios_locked, 0, sizeof(tty->termios_locked)); +		o_tty->termios = driver->other->init_termios;  	}  	/*  	 * Everything allocated ... set up the o_tty structure.  	 */ -	driver->other->ttys[idx] = o_tty;  	tty_driver_kref_get(driver->other);  	if (driver->subtype == PTY_TYPE_MASTER)  		o_tty->count++;  	/* Establish the links in both directions */  	tty->link   = o_tty;  	o_tty->link = tty; +	tty_port_init(ports[0]); +	tty_port_init(ports[1]); +	o_tty->port = ports[0]; +	tty->port = ports[1]; +	o_tty->port->itty = o_tty;  	tty_driver_kref_get(driver);  	tty->count++; -	driver->ttys[idx] = tty;  	return 0; -free_mem_out: +err_free_termios: +	if (legacy) +		tty_free_termios(tty); +err_deinit_tty: +	deinitialize_tty_struct(o_tty);  	module_put(o_tty->driver->owner); +err_free_tty: +	kfree(ports[0]); +	kfree(ports[1]);  	free_tty_struct(o_tty); -	return -ENOMEM; +err: +	return retval; +} + +static void pty_cleanup(struct tty_struct *tty) +{ +	tty_port_put(tty->port);  } -static int pty_bsd_ioctl(struct tty_struct *tty, struct file *file, +/* Traditional BSD devices */ +#ifdef CONFIG_LEGACY_PTYS + +static int pty_install(struct tty_driver *driver, struct tty_struct *tty) +{ +	return pty_common_install(driver, tty, true); +} + +static void pty_remove(struct tty_driver *driver, struct tty_struct *tty) +{ +	struct tty_struct *pair = tty->link; +	driver->ttys[tty->index] = NULL; +	if (pair) +		pair->driver->ttys[pair->index] = NULL; +} + +static int pty_bsd_ioctl(struct tty_struct *tty,  			 unsigned int cmd, unsigned long arg)  {  	switch (cmd) {  	case TIOCSPTLCK: /* Set PT Lock (disallow slave open) */  		return pty_set_lock(tty, (int __user *) arg); +	case TIOCGPTLCK: /* Get PT Lock status */ +		return pty_get_lock(tty, (int __user *)arg); +	case TIOCPKT: /* Set PT packet mode */ +		return pty_set_pktmode(tty, (int __user *)arg); +	case TIOCGPKT: /* Get PT packet mode */ +		return pty_get_pktmode(tty, (int __user *)arg);  	case TIOCSIG:    /* Send signal to other side of pty */  		return pty_signal(tty, (int) arg); +	case TIOCGPTN: /* TTY returns ENOTTY, but glibc expects EINVAL here */ +		return -EINVAL;  	}  	return -ENOIOCTLCMD;  } @@ -364,7 +455,9 @@ static const struct tty_operations master_pty_ops_bsd = {  	.unthrottle = pty_unthrottle,  	.set_termios = pty_set_termios,  	.ioctl = pty_bsd_ioctl, -	.resize = pty_resize +	.cleanup = pty_cleanup, +	.resize = pty_resize, +	.remove = pty_remove  };  static const struct tty_operations slave_pty_ops_bsd = { @@ -377,7 +470,9 @@ static const struct tty_operations slave_pty_ops_bsd = {  	.chars_in_buffer = pty_chars_in_buffer,  	.unthrottle = pty_unthrottle,  	.set_termios = pty_set_termios, -	.resize = pty_resize +	.cleanup = pty_cleanup, +	.resize = pty_resize, +	.remove = pty_remove  };  static void __init legacy_pty_init(void) @@ -387,15 +482,20 @@ static void __init legacy_pty_init(void)  	if (legacy_count <= 0)  		return; -	pty_driver = alloc_tty_driver(legacy_count); -	if (!pty_driver) +	pty_driver = tty_alloc_driver(legacy_count, +			TTY_DRIVER_RESET_TERMIOS | +			TTY_DRIVER_REAL_RAW | +			TTY_DRIVER_DYNAMIC_ALLOC); +	if (IS_ERR(pty_driver))  		panic("Couldn't allocate pty driver"); -	pty_slave_driver = alloc_tty_driver(legacy_count); -	if (!pty_slave_driver) +	pty_slave_driver = tty_alloc_driver(legacy_count, +			TTY_DRIVER_RESET_TERMIOS | +			TTY_DRIVER_REAL_RAW | +			TTY_DRIVER_DYNAMIC_ALLOC); +	if (IS_ERR(pty_slave_driver))  		panic("Couldn't allocate pty slave driver"); -	pty_driver->owner = THIS_MODULE;  	pty_driver->driver_name = "pty_master";  	pty_driver->name = "pty";  	pty_driver->major = PTY_MASTER_MAJOR; @@ -409,11 +509,9 @@ static void __init legacy_pty_init(void)  	pty_driver->init_termios.c_lflag = 0;  	pty_driver->init_termios.c_ispeed = 38400;  	pty_driver->init_termios.c_ospeed = 38400; -	pty_driver->flags = TTY_DRIVER_RESET_TERMIOS | TTY_DRIVER_REAL_RAW;  	pty_driver->other = pty_slave_driver;  	tty_set_operations(pty_driver, &master_pty_ops_bsd); -	pty_slave_driver->owner = THIS_MODULE;  	pty_slave_driver->driver_name = "pty_slave";  	pty_slave_driver->name = "ttyp";  	pty_slave_driver->major = PTY_SLAVE_MAJOR; @@ -424,8 +522,6 @@ static void __init legacy_pty_init(void)  	pty_slave_driver->init_termios.c_cflag = B38400 | CS8 | CREAD;  	pty_slave_driver->init_termios.c_ispeed = 38400;  	pty_slave_driver->init_termios.c_ospeed = 38400; -	pty_slave_driver->flags = TTY_DRIVER_RESET_TERMIOS | -					TTY_DRIVER_REAL_RAW;  	pty_slave_driver->other = pty_driver;  	tty_set_operations(pty_slave_driver, &slave_pty_ops_bsd); @@ -440,61 +536,21 @@ static inline void legacy_pty_init(void) { }  /* Unix98 devices */  #ifdef CONFIG_UNIX98_PTYS -/* - * sysctl support for setting limits on the number of Unix98 ptys allocated. - * Otherwise one can eat up all kernel memory by opening /dev/ptmx repeatedly. - */ -int pty_limit = NR_UNIX98_PTY_DEFAULT; -static int pty_limit_min; -static int pty_limit_max = NR_UNIX98_PTY_MAX; -static int pty_count;  static struct cdev ptmx_cdev; -static struct ctl_table pty_table[] = { -	{ -		.procname	= "max", -		.maxlen		= sizeof(int), -		.mode		= 0644, -		.data		= &pty_limit, -		.proc_handler	= proc_dointvec_minmax, -		.extra1		= &pty_limit_min, -		.extra2		= &pty_limit_max, -	}, { -		.procname	= "nr", -		.maxlen		= sizeof(int), -		.mode		= 0444, -		.data		= &pty_count, -		.proc_handler	= proc_dointvec, -	},  -	{} -}; - -static struct ctl_table pty_kern_table[] = { -	{ -		.procname	= "pty", -		.mode		= 0555, -		.child		= pty_table, -	}, -	{} -}; - -static struct ctl_table pty_root_table[] = { -	{ -		.procname	= "kernel", -		.mode		= 0555, -		.child		= pty_kern_table, -	}, -	{} -}; - - -static int pty_unix98_ioctl(struct tty_struct *tty, struct file *file, +static int pty_unix98_ioctl(struct tty_struct *tty,  			    unsigned int cmd, unsigned long arg)  {  	switch (cmd) {  	case TIOCSPTLCK: /* Set PT Lock (disallow slave open) */  		return pty_set_lock(tty, (int __user *)arg); +	case TIOCGPTLCK: /* Get PT Lock status */ +		return pty_get_lock(tty, (int __user *)arg); +	case TIOCPKT: /* Set PT packet mode */ +		return pty_set_pktmode(tty, (int __user *)arg); +	case TIOCGPKT: /* Get PT packet mode */ +		return pty_get_pktmode(tty, (int __user *)arg);  	case TIOCGPTN: /* Get PT Number */  		return put_user(tty->index, (unsigned int __user *)arg);  	case TIOCSIG:    /* Send signal to other side of pty */ @@ -516,10 +572,8 @@ static int pty_unix98_ioctl(struct tty_struct *tty, struct file *file,  static struct tty_struct *ptm_unix98_lookup(struct tty_driver *driver,  		struct inode *ptm_inode, int idx)  { -	struct tty_struct *tty = devpts_get_tty(ptm_inode, idx); -	if (tty) -		tty = tty->link; -	return tty; +	/* Master must be open via /dev/ptmx */ +	return ERR_PTR(-EIO);  }  /** @@ -528,81 +582,39 @@ static struct tty_struct *ptm_unix98_lookup(struct tty_driver *driver,   *	@idx: tty index   *   *	Look up a pty master device. Called under the tty_mutex for now. - *	This provides our locking. + *	This provides our locking for the tty pointer.   */  static struct tty_struct *pts_unix98_lookup(struct tty_driver *driver,  		struct inode *pts_inode, int idx)  { -	struct tty_struct *tty = devpts_get_tty(pts_inode, idx); +	struct tty_struct *tty; + +	mutex_lock(&devpts_mutex); +	tty = devpts_get_priv(pts_inode); +	mutex_unlock(&devpts_mutex);  	/* Master must be open before slave */  	if (!tty)  		return ERR_PTR(-EIO);  	return tty;  } -static void pty_unix98_shutdown(struct tty_struct *tty) -{ -	/* We have our own method as we don't use the tty index */ -	kfree(tty->termios); -} -  /* We have no need to install and remove our tty objects as devpts does all     the work for us */  static int pty_unix98_install(struct tty_driver *driver, struct tty_struct *tty)  { -	struct tty_struct *o_tty; -	int idx = tty->index; - -	o_tty = alloc_tty_struct(); -	if (!o_tty) -		return -ENOMEM; -	if (!try_module_get(driver->other->owner)) { -		/* This cannot in fact currently happen */ -		free_tty_struct(o_tty); -		return -ENOMEM; -	} -	initialize_tty_struct(o_tty, driver->other, idx); - -	tty->termios = kzalloc(sizeof(struct ktermios[2]), GFP_KERNEL); -	if (tty->termios == NULL) -		goto free_mem_out; -	*tty->termios = driver->init_termios; -	tty->termios_locked = tty->termios + 1; - -	o_tty->termios = kzalloc(sizeof(struct ktermios[2]), GFP_KERNEL); -	if (o_tty->termios == NULL) -		goto free_mem_out; -	*o_tty->termios = driver->other->init_termios; -	o_tty->termios_locked = o_tty->termios + 1; - -	tty_driver_kref_get(driver->other); -	if (driver->subtype == PTY_TYPE_MASTER) -		o_tty->count++; -	/* Establish the links in both directions */ -	tty->link   = o_tty; -	o_tty->link = tty; -	/* -	 * All structures have been allocated, so now we install them. -	 * Failures after this point use release_tty to clean up, so -	 * there's no need to null out the local pointers. -	 */ -	tty_driver_kref_get(driver); -	tty->count++; -	pty_count++; -	return 0; -free_mem_out: -	kfree(o_tty->termios); -	module_put(o_tty->driver->owner); -	free_tty_struct(o_tty); -	kfree(tty->termios); -	return -ENOMEM; +	return pty_common_install(driver, tty, false);  }  static void pty_unix98_remove(struct tty_driver *driver, struct tty_struct *tty)  { -	pty_count--; +} + +/* this is called once with whichever end is closed last */ +static void pty_unix98_shutdown(struct tty_struct *tty) +{ +	devpts_kill_index(tty->driver_data, tty->index);  }  static const struct tty_operations ptm_unix98_ops = { @@ -618,8 +630,9 @@ static const struct tty_operations ptm_unix98_ops = {  	.unthrottle = pty_unthrottle,  	.set_termios = pty_set_termios,  	.ioctl = pty_unix98_ioctl, +	.resize = pty_resize,  	.shutdown = pty_unix98_shutdown, -	.resize = pty_resize +	.cleanup = pty_cleanup  };  static const struct tty_operations pty_unix98_ops = { @@ -634,7 +647,8 @@ static const struct tty_operations pty_unix98_ops = {  	.chars_in_buffer = pty_chars_in_buffer,  	.unthrottle = pty_unthrottle,  	.set_termios = pty_set_termios, -	.shutdown = pty_unix98_shutdown +	.shutdown = pty_unix98_shutdown, +	.cleanup = pty_cleanup,  };  /** @@ -645,58 +659,78 @@ static const struct tty_operations pty_unix98_ops = {   *	Allocate a unix98 pty master device from the ptmx driver.   *   *	Locking: tty_mutex protects the init_dev work. tty->count should - * 		protect the rest. + *		protect the rest.   *		allocated_ptys_lock handles the list of free pty numbers   */  static int ptmx_open(struct inode *inode, struct file *filp)  {  	struct tty_struct *tty; +	struct inode *slave_inode;  	int retval;  	int index;  	nonseekable_open(inode, filp); +	/* We refuse fsnotify events on ptmx, since it's a shared resource */ +	filp->f_mode |= FMODE_NONOTIFY; + +	retval = tty_alloc_file(filp); +	if (retval) +		return retval; +  	/* find a device that is not in use. */ -	tty_lock(); +	mutex_lock(&devpts_mutex);  	index = devpts_new_index(inode); -	tty_unlock(); -	if (index < 0) -		return index; +	if (index < 0) { +		retval = index; +		mutex_unlock(&devpts_mutex); +		goto err_file; +	} + +	mutex_unlock(&devpts_mutex);  	mutex_lock(&tty_mutex); -	tty_lock(); -	tty = tty_init_dev(ptm_driver, index, 1); -	mutex_unlock(&tty_mutex); +	tty = tty_init_dev(ptm_driver, index);  	if (IS_ERR(tty)) {  		retval = PTR_ERR(tty);  		goto out;  	} +	/* The tty returned here is locked so we can safely +	   drop the mutex */ +	mutex_unlock(&tty_mutex); +  	set_bit(TTY_PTY_LOCK, &tty->flags); /* LOCK THE SLAVE */ +	tty->driver_data = inode; -	retval = tty_add_file(tty, filp); -	if (retval) -		goto out; +	tty_add_file(tty, filp); -	retval = devpts_pty_new(inode, tty->link); -	if (retval) -		goto out1; +	slave_inode = devpts_pty_new(inode, +			MKDEV(UNIX98_PTY_SLAVE_MAJOR, index), index, +			tty->link); +	if (IS_ERR(slave_inode)) { +		retval = PTR_ERR(slave_inode); +		goto err_release; +	} +	tty->link->driver_data = slave_inode;  	retval = ptm_driver->ops->open(tty, filp);  	if (retval) -		goto out2; -out1: -	tty_unlock(); -	return retval; -out2: -	tty_unlock(); +		goto err_release; + +	tty_unlock(tty); +	return 0; +err_release: +	tty_unlock(tty);  	tty_release(inode, filp);  	return retval;  out: +	mutex_unlock(&tty_mutex);  	devpts_kill_index(inode, index); -	tty_unlock(); +err_file: +	tty_free_file(filp);  	return retval;  } @@ -704,14 +738,23 @@ static struct file_operations ptmx_fops;  static void __init unix98_pty_init(void)  { -	ptm_driver = alloc_tty_driver(NR_UNIX98_PTY_MAX); -	if (!ptm_driver) +	ptm_driver = tty_alloc_driver(NR_UNIX98_PTY_MAX, +			TTY_DRIVER_RESET_TERMIOS | +			TTY_DRIVER_REAL_RAW | +			TTY_DRIVER_DYNAMIC_DEV | +			TTY_DRIVER_DEVPTS_MEM | +			TTY_DRIVER_DYNAMIC_ALLOC); +	if (IS_ERR(ptm_driver))  		panic("Couldn't allocate Unix98 ptm driver"); -	pts_driver = alloc_tty_driver(NR_UNIX98_PTY_MAX); -	if (!pts_driver) +	pts_driver = tty_alloc_driver(NR_UNIX98_PTY_MAX, +			TTY_DRIVER_RESET_TERMIOS | +			TTY_DRIVER_REAL_RAW | +			TTY_DRIVER_DYNAMIC_DEV | +			TTY_DRIVER_DEVPTS_MEM | +			TTY_DRIVER_DYNAMIC_ALLOC); +	if (IS_ERR(pts_driver))  		panic("Couldn't allocate Unix98 pts driver"); -	ptm_driver->owner = THIS_MODULE;  	ptm_driver->driver_name = "pty_master";  	ptm_driver->name = "ptm";  	ptm_driver->major = UNIX98_PTY_MASTER_MAJOR; @@ -725,12 +768,9 @@ static void __init unix98_pty_init(void)  	ptm_driver->init_termios.c_lflag = 0;  	ptm_driver->init_termios.c_ispeed = 38400;  	ptm_driver->init_termios.c_ospeed = 38400; -	ptm_driver->flags = TTY_DRIVER_RESET_TERMIOS | TTY_DRIVER_REAL_RAW | -		TTY_DRIVER_DYNAMIC_DEV | TTY_DRIVER_DEVPTS_MEM;  	ptm_driver->other = pts_driver;  	tty_set_operations(ptm_driver, &ptm_unix98_ops); -	pts_driver->owner = THIS_MODULE;  	pts_driver->driver_name = "pty_slave";  	pts_driver->name = "pts";  	pts_driver->major = UNIX98_PTY_SLAVE_MAJOR; @@ -741,8 +781,6 @@ static void __init unix98_pty_init(void)  	pts_driver->init_termios.c_cflag = B38400 | CS8 | CREAD;  	pts_driver->init_termios.c_ispeed = 38400;  	pts_driver->init_termios.c_ospeed = 38400; -	pts_driver->flags = TTY_DRIVER_RESET_TERMIOS | TTY_DRIVER_REAL_RAW | -		TTY_DRIVER_DYNAMIC_DEV | TTY_DRIVER_DEVPTS_MEM;  	pts_driver->other = ptm_driver;  	tty_set_operations(pts_driver, &pty_unix98_ops); @@ -751,8 +789,6 @@ static void __init unix98_pty_init(void)  	if (tty_register_driver(pts_driver))  		panic("Couldn't register Unix98 pts driver"); -	register_sysctl_table(pty_root_table); -  	/* Now create the /dev/ptmx special device */  	tty_default_fops(&ptmx_fops);  	ptmx_fops.open = ptmx_open; @@ -760,7 +796,7 @@ static void __init unix98_pty_init(void)  	cdev_init(&ptmx_cdev, &ptmx_fops);  	if (cdev_add(&ptmx_cdev, MKDEV(TTYAUX_MAJOR, 2), 1) ||  	    register_chrdev_region(MKDEV(TTYAUX_MAJOR, 2), 1, "/dev/ptmx") < 0) -		panic("Couldn't register /dev/ptmx driver\n"); +		panic("Couldn't register /dev/ptmx driver");  	device_create(tty_class, NULL, MKDEV(TTYAUX_MAJOR, 2), NULL, "ptmx");  }  | 
