提交 5d249bc6 编写于 作者: J Jiri Slaby 提交者: Greg Kroah-Hartman

PTY: merge pty_install implementations

There are currently two instances of code which handles PTY install.
One for the legacy BSD PTY's, one for unix98's PTY's. Both of them are
very similar and differ only in termios allocation and handling.

Since we will need to allocate a tty_port at that place, this would
require editing two places with the same pattern. Instead, let us move
the implementation to one common place and call it from both places.
Signed-off-by: NJiri Slaby <jslaby@suse.cz>
Signed-off-by: NGreg Kroah-Hartman <gregkh@linuxfoundation.org>
上级 7171604a
...@@ -282,39 +282,53 @@ int pty_resize(struct tty_struct *tty, struct winsize *ws) ...@@ -282,39 +282,53 @@ int pty_resize(struct tty_struct *tty, struct winsize *ws)
return 0; return 0;
} }
/* Traditional BSD devices */ static int pty_common_install(struct tty_driver *driver, struct tty_struct *tty,
#ifdef CONFIG_LEGACY_PTYS bool legacy)
static int pty_install(struct tty_driver *driver, struct tty_struct *tty)
{ {
struct tty_struct *o_tty; struct tty_struct *o_tty;
int idx = tty->index; int idx = tty->index;
int retval; int retval = -ENOMEM;
o_tty = alloc_tty_struct(); o_tty = alloc_tty_struct();
if (!o_tty) if (!o_tty)
return -ENOMEM; goto err;
if (!try_module_get(driver->other->owner)) { if (!try_module_get(driver->other->owner)) {
/* This cannot in fact currently happen */ /* This cannot in fact currently happen */
retval = -ENOMEM;
goto err_free_tty; goto err_free_tty;
} }
initialize_tty_struct(o_tty, driver->other, idx); initialize_tty_struct(o_tty, driver->other, idx);
/* We always use new tty termios data so we can do this if (legacy) {
the easy way .. */ /* We always use new tty termios data so we can do this
retval = tty_init_termios(tty); the easy way .. */
if (retval) retval = tty_init_termios(tty);
goto err_deinit_tty; if (retval)
goto err_deinit_tty;
retval = tty_init_termios(o_tty);
if (retval) retval = tty_init_termios(o_tty);
goto err_free_termios; if (retval)
goto err_free_termios;
driver->other->ttys[idx] = o_tty;
driver->ttys[idx] = tty;
} else {
tty->termios = kzalloc(sizeof(struct ktermios[2]), GFP_KERNEL);
if (tty->termios == NULL)
goto err_deinit_tty;
*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 err_free_termios;
*o_tty->termios = driver->other->init_termios;
o_tty->termios_locked = o_tty->termios + 1;
}
/* /*
* Everything allocated ... set up the o_tty structure. * Everything allocated ... set up the o_tty structure.
*/ */
driver->other->ttys[idx] = o_tty;
tty_driver_kref_get(driver->other); tty_driver_kref_get(driver->other);
if (driver->subtype == PTY_TYPE_MASTER) if (driver->subtype == PTY_TYPE_MASTER)
o_tty->count++; o_tty->count++;
...@@ -324,18 +338,29 @@ static int pty_install(struct tty_driver *driver, struct tty_struct *tty) ...@@ -324,18 +338,29 @@ static int pty_install(struct tty_driver *driver, struct tty_struct *tty)
tty_driver_kref_get(driver); tty_driver_kref_get(driver);
tty->count++; tty->count++;
driver->ttys[idx] = tty;
return 0; return 0;
err_free_termios: err_free_termios:
tty_free_termios(tty); if (legacy)
tty_free_termios(tty);
else
kfree(tty->termios);
err_deinit_tty: err_deinit_tty:
deinitialize_tty_struct(o_tty); deinitialize_tty_struct(o_tty);
module_put(o_tty->driver->owner); module_put(o_tty->driver->owner);
err_free_tty: err_free_tty:
free_tty_struct(o_tty); free_tty_struct(o_tty);
err:
return retval; return retval;
} }
/* 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 int pty_bsd_ioctl(struct tty_struct *tty, static int pty_bsd_ioctl(struct tty_struct *tty,
unsigned int cmd, unsigned long arg) unsigned int cmd, unsigned long arg)
{ {
...@@ -509,52 +534,7 @@ static void pty_unix98_shutdown(struct tty_struct *tty) ...@@ -509,52 +534,7 @@ static void pty_unix98_shutdown(struct tty_struct *tty)
static int pty_unix98_install(struct tty_driver *driver, struct tty_struct *tty) static int pty_unix98_install(struct tty_driver *driver, struct tty_struct *tty)
{ {
struct tty_struct *o_tty; return pty_common_install(driver, tty, false);
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 */
goto err_free_tty;
}
initialize_tty_struct(o_tty, driver->other, idx);
tty->termios = kzalloc(sizeof(struct ktermios[2]), GFP_KERNEL);
if (tty->termios == NULL)
goto err_free_mem;
*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 err_free_mem;
*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++;
return 0;
err_free_mem:
deinitialize_tty_struct(o_tty);
kfree(o_tty->termios);
kfree(tty->termios);
module_put(o_tty->driver->owner);
err_free_tty:
free_tty_struct(o_tty);
return -ENOMEM;
} }
static void pty_unix98_remove(struct tty_driver *driver, struct tty_struct *tty) static void pty_unix98_remove(struct tty_driver *driver, struct tty_struct *tty)
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册