[PATCH] tty: cleanup release_mem
release_mem contains two copies of exactly the same code. Refactor these into a new helper, release_tty. The only change in behaviour is that the driver reference count is now decremented after the master tty has been freed instead of before. [penberg@cs.helsinki.fi: fix use-after-free in release_tty.] Cc: Alan Cox <alan@redhat.com> Signed-off-by: Christoph Hellwig <hch@lst.de> Signed-off-by: Pekka Enberg <penberg@cs.helsinki.fi> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
4b98d11b40
commit
d5698c28b6
|
@ -154,7 +154,7 @@ static int tty_release(struct inode *, struct file *);
|
||||||
int tty_ioctl(struct inode * inode, struct file * file,
|
int tty_ioctl(struct inode * inode, struct file * file,
|
||||||
unsigned int cmd, unsigned long arg);
|
unsigned int cmd, unsigned long arg);
|
||||||
static int tty_fasync(int fd, struct file * filp, int on);
|
static int tty_fasync(int fd, struct file * filp, int on);
|
||||||
static void release_mem(struct tty_struct *tty, int idx);
|
static void release_tty(struct tty_struct *tty, int idx);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* alloc_tty_struct - allocate a tty object
|
* alloc_tty_struct - allocate a tty object
|
||||||
|
@ -2002,7 +2002,7 @@ static int init_dev(struct tty_driver *driver, int idx,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* All structures have been allocated, so now we install them.
|
* All structures have been allocated, so now we install them.
|
||||||
* Failures after this point use release_mem to clean up, so
|
* Failures after this point use release_tty to clean up, so
|
||||||
* there's no need to null out the local pointers.
|
* there's no need to null out the local pointers.
|
||||||
*/
|
*/
|
||||||
if (!(driver->flags & TTY_DRIVER_DEVPTS_MEM)) {
|
if (!(driver->flags & TTY_DRIVER_DEVPTS_MEM)) {
|
||||||
|
@ -2023,8 +2023,8 @@ static int init_dev(struct tty_driver *driver, int idx,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Structures all installed ... call the ldisc open routines.
|
* Structures all installed ... call the ldisc open routines.
|
||||||
* If we fail here just call release_mem to clean up. No need
|
* If we fail here just call release_tty to clean up. No need
|
||||||
* to decrement the use counts, as release_mem doesn't care.
|
* to decrement the use counts, as release_tty doesn't care.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
if (tty->ldisc.open) {
|
if (tty->ldisc.open) {
|
||||||
|
@ -2094,17 +2094,17 @@ fail_no_mem:
|
||||||
retval = -ENOMEM;
|
retval = -ENOMEM;
|
||||||
goto end_init;
|
goto end_init;
|
||||||
|
|
||||||
/* call the tty release_mem routine to clean out this slot */
|
/* call the tty release_tty routine to clean out this slot */
|
||||||
release_mem_out:
|
release_mem_out:
|
||||||
if (printk_ratelimit())
|
if (printk_ratelimit())
|
||||||
printk(KERN_INFO "init_dev: ldisc open failed, "
|
printk(KERN_INFO "init_dev: ldisc open failed, "
|
||||||
"clearing slot %d\n", idx);
|
"clearing slot %d\n", idx);
|
||||||
release_mem(tty, idx);
|
release_tty(tty, idx);
|
||||||
goto end_init;
|
goto end_init;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* release_mem - release tty structure memory
|
* release_one_tty - release tty structure memory
|
||||||
*
|
*
|
||||||
* Releases memory associated with a tty structure, and clears out the
|
* Releases memory associated with a tty structure, and clears out the
|
||||||
* driver table slots. This function is called when a device is no longer
|
* driver table slots. This function is called when a device is no longer
|
||||||
|
@ -2116,37 +2116,14 @@ release_mem_out:
|
||||||
* of ttys that the driver keeps.
|
* of ttys that the driver keeps.
|
||||||
* FIXME: should we require tty_mutex is held here ??
|
* FIXME: should we require tty_mutex is held here ??
|
||||||
*/
|
*/
|
||||||
|
static void release_one_tty(struct tty_struct *tty, int idx)
|
||||||
static void release_mem(struct tty_struct *tty, int idx)
|
|
||||||
{
|
{
|
||||||
struct tty_struct *o_tty;
|
|
||||||
struct ktermios *tp;
|
|
||||||
int devpts = tty->driver->flags & TTY_DRIVER_DEVPTS_MEM;
|
int devpts = tty->driver->flags & TTY_DRIVER_DEVPTS_MEM;
|
||||||
|
struct ktermios *tp;
|
||||||
if ((o_tty = tty->link) != NULL) {
|
|
||||||
if (!devpts)
|
|
||||||
o_tty->driver->ttys[idx] = NULL;
|
|
||||||
if (o_tty->driver->flags & TTY_DRIVER_RESET_TERMIOS) {
|
|
||||||
tp = o_tty->termios;
|
|
||||||
if (!devpts)
|
|
||||||
o_tty->driver->termios[idx] = NULL;
|
|
||||||
kfree(tp);
|
|
||||||
|
|
||||||
tp = o_tty->termios_locked;
|
|
||||||
if (!devpts)
|
|
||||||
o_tty->driver->termios_locked[idx] = NULL;
|
|
||||||
kfree(tp);
|
|
||||||
}
|
|
||||||
o_tty->magic = 0;
|
|
||||||
o_tty->driver->refcount--;
|
|
||||||
file_list_lock();
|
|
||||||
list_del_init(&o_tty->tty_files);
|
|
||||||
file_list_unlock();
|
|
||||||
free_tty_struct(o_tty);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!devpts)
|
if (!devpts)
|
||||||
tty->driver->ttys[idx] = NULL;
|
tty->driver->ttys[idx] = NULL;
|
||||||
|
|
||||||
if (tty->driver->flags & TTY_DRIVER_RESET_TERMIOS) {
|
if (tty->driver->flags & TTY_DRIVER_RESET_TERMIOS) {
|
||||||
tp = tty->termios;
|
tp = tty->termios;
|
||||||
if (!devpts)
|
if (!devpts)
|
||||||
|
@ -2159,15 +2136,39 @@ static void release_mem(struct tty_struct *tty, int idx)
|
||||||
kfree(tp);
|
kfree(tp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
tty->magic = 0;
|
tty->magic = 0;
|
||||||
tty->driver->refcount--;
|
tty->driver->refcount--;
|
||||||
|
|
||||||
file_list_lock();
|
file_list_lock();
|
||||||
list_del_init(&tty->tty_files);
|
list_del_init(&tty->tty_files);
|
||||||
file_list_unlock();
|
file_list_unlock();
|
||||||
module_put(tty->driver->owner);
|
|
||||||
free_tty_struct(tty);
|
free_tty_struct(tty);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* release_tty - release tty structure memory
|
||||||
|
*
|
||||||
|
* Release both @tty and a possible linked partner (think pty pair),
|
||||||
|
* and decrement the refcount of the backing module.
|
||||||
|
*
|
||||||
|
* Locking:
|
||||||
|
* tty_mutex - sometimes only
|
||||||
|
* takes the file list lock internally when working on the list
|
||||||
|
* of ttys that the driver keeps.
|
||||||
|
* FIXME: should we require tty_mutex is held here ??
|
||||||
|
*/
|
||||||
|
static void release_tty(struct tty_struct *tty, int idx)
|
||||||
|
{
|
||||||
|
struct tty_driver *driver = tty->driver;
|
||||||
|
|
||||||
|
if (tty->link)
|
||||||
|
release_one_tty(tty->link, idx);
|
||||||
|
release_one_tty(tty, idx);
|
||||||
|
module_put(driver->owner);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Even releasing the tty structures is a tricky business.. We have
|
* Even releasing the tty structures is a tricky business.. We have
|
||||||
* to be very careful that the structures are all released at the
|
* to be very careful that the structures are all released at the
|
||||||
|
@ -2435,10 +2436,10 @@ static void release_dev(struct file * filp)
|
||||||
tty_set_termios_ldisc(o_tty,N_TTY);
|
tty_set_termios_ldisc(o_tty,N_TTY);
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
* The release_mem function takes care of the details of clearing
|
* The release_tty function takes care of the details of clearing
|
||||||
* the slots and preserving the termios structure.
|
* the slots and preserving the termios structure.
|
||||||
*/
|
*/
|
||||||
release_mem(tty, idx);
|
release_tty(tty, idx);
|
||||||
|
|
||||||
#ifdef CONFIG_UNIX98_PTYS
|
#ifdef CONFIG_UNIX98_PTYS
|
||||||
/* Make this pty number available for reallocation */
|
/* Make this pty number available for reallocation */
|
||||||
|
|
Loading…
Reference in New Issue