Commit 311fc65c authored by Eric W. Biederman's avatar Eric W. Biederman Committed by Linus Torvalds

pty: Repair TIOCGPTPEER

The implementation of TIOCGPTPEER has two issues.

When /dev/ptmx (as opposed to /dev/pts/ptmx) is opened the wrong
vfsmount is passed to dentry_open.  Which results in the kernel displaying
the wrong pathname for the peer.

The second is simply by caching the vfsmount and dentry of the peer it leaves
them open, in a way they were not previously Which because of the inreased
reference counts can cause unnecessary behaviour differences resulting in
regressions.

To fix these move the ioctl into tty_io.c at a generic level allowing
the ioctl to have access to the struct file on which the ioctl is
being called.  This allows the path of the slave to be derived when
opening the slave through TIOCGPTPEER instead of requiring the path to
the slave be cached.  Thus removing the need for caching the path.

A new function devpts_ptmx_path is factored out of devpts_acquire and
used to implement a function devpts_mntget.   The new function devpts_mntget
takes a filp to perform the lookup on and fsi so that it can confirm
that the superblock that is found by devpts_ptmx_path is the proper superblock.

v2: Lots of fixes to make the code actually work
v3: Suggestions by Linus
    - Removed the unnecessary initialization of filp in ptm_open_peer
    - Simplified devpts_ptmx_path as gotos are no longer required

[ This is the fix for the issue that was reverted in commit
  143c97cc, but this time without breaking 'pbuilder' due to
  increased reference counts   - Linus ]

Fixes: 54ebbfb1 ("tty: add TIOCGPTPEER ioctl")
Reported-by: default avatarChristian Brauner <christian.brauner@canonical.com>
Reported-and-tested-by: default avatarStefan Lippers-Hollmann <s.l-h@gmx.de>
Signed-off-by: default avatar"Eric W. Biederman" <ebiederm@xmission.com>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 143c97cc
...@@ -69,13 +69,8 @@ static void pty_close(struct tty_struct *tty, struct file *filp) ...@@ -69,13 +69,8 @@ static void pty_close(struct tty_struct *tty, struct file *filp)
#ifdef CONFIG_UNIX98_PTYS #ifdef CONFIG_UNIX98_PTYS
if (tty->driver == ptm_driver) { if (tty->driver == ptm_driver) {
mutex_lock(&devpts_mutex); mutex_lock(&devpts_mutex);
if (tty->link->driver_data) { if (tty->link->driver_data)
struct path *path = tty->link->driver_data; devpts_pty_kill(tty->link->driver_data);
devpts_pty_kill(path->dentry);
path_put(path);
kfree(path);
}
mutex_unlock(&devpts_mutex); mutex_unlock(&devpts_mutex);
} }
#endif #endif
...@@ -607,25 +602,24 @@ static inline void legacy_pty_init(void) { } ...@@ -607,25 +602,24 @@ static inline void legacy_pty_init(void) { }
static struct cdev ptmx_cdev; static struct cdev ptmx_cdev;
/** /**
* pty_open_peer - open the peer of a pty * ptm_open_peer - open the peer of a pty
* @tty: the peer of the pty being opened * @master: the open struct file of the ptmx device node
* @tty: the master of the pty being opened
* @flags: the flags for open
* *
* Open the cached dentry in tty->link, providing a safe way for userspace * Provide a race free way for userspace to open the slave end of a pty
* to get the slave end of a pty (where they have the master fd and cannot * (where they have the master fd and cannot access or trust the mount
* access or trust the mount namespace /dev/pts was mounted inside). * namespace /dev/pts was mounted inside).
*/ */
static struct file *pty_open_peer(struct tty_struct *tty, int flags) int ptm_open_peer(struct file *master, struct tty_struct *tty, int flags)
{
if (tty->driver->subtype != PTY_TYPE_MASTER)
return ERR_PTR(-EIO);
return dentry_open(tty->link->driver_data, flags, current_cred());
}
static int pty_get_peer(struct tty_struct *tty, int flags)
{ {
int fd = -1; int fd = -1;
struct file *filp = NULL; struct file *filp;
int retval = -EINVAL; int retval = -EINVAL;
struct path path;
if (tty->driver != ptm_driver)
return -EIO;
fd = get_unused_fd_flags(0); fd = get_unused_fd_flags(0);
if (fd < 0) { if (fd < 0) {
...@@ -633,7 +627,16 @@ static int pty_get_peer(struct tty_struct *tty, int flags) ...@@ -633,7 +627,16 @@ static int pty_get_peer(struct tty_struct *tty, int flags)
goto err; goto err;
} }
filp = pty_open_peer(tty, flags); /* Compute the slave's path */
path.mnt = devpts_mntget(master, tty->driver_data);
if (IS_ERR(path.mnt)) {
retval = PTR_ERR(path.mnt);
goto err_put;
}
path.dentry = tty->link->driver_data;
filp = dentry_open(&path, flags, current_cred());
mntput(path.mnt);
if (IS_ERR(filp)) { if (IS_ERR(filp)) {
retval = PTR_ERR(filp); retval = PTR_ERR(filp);
goto err_put; goto err_put;
...@@ -662,8 +665,6 @@ static int pty_unix98_ioctl(struct tty_struct *tty, ...@@ -662,8 +665,6 @@ static int pty_unix98_ioctl(struct tty_struct *tty,
return pty_get_pktmode(tty, (int __user *)arg); return pty_get_pktmode(tty, (int __user *)arg);
case TIOCGPTN: /* Get PT Number */ case TIOCGPTN: /* Get PT Number */
return put_user(tty->index, (unsigned int __user *)arg); return put_user(tty->index, (unsigned int __user *)arg);
case TIOCGPTPEER: /* Open the other end */
return pty_get_peer(tty, (int) arg);
case TIOCSIG: /* Send signal to other side of pty */ case TIOCSIG: /* Send signal to other side of pty */
return pty_signal(tty, (int) arg); return pty_signal(tty, (int) arg);
} }
...@@ -791,7 +792,6 @@ static int ptmx_open(struct inode *inode, struct file *filp) ...@@ -791,7 +792,6 @@ static int ptmx_open(struct inode *inode, struct file *filp)
{ {
struct pts_fs_info *fsi; struct pts_fs_info *fsi;
struct tty_struct *tty; struct tty_struct *tty;
struct path *pts_path;
struct dentry *dentry; struct dentry *dentry;
int retval; int retval;
int index; int index;
...@@ -845,26 +845,16 @@ static int ptmx_open(struct inode *inode, struct file *filp) ...@@ -845,26 +845,16 @@ static int ptmx_open(struct inode *inode, struct file *filp)
retval = PTR_ERR(dentry); retval = PTR_ERR(dentry);
goto err_release; goto err_release;
} }
/* We need to cache a fake path for TIOCGPTPEER. */ tty->link->driver_data = dentry;
pts_path = kmalloc(sizeof(struct path), GFP_KERNEL);
if (!pts_path)
goto err_release;
pts_path->mnt = filp->f_path.mnt;
pts_path->dentry = dentry;
path_get(pts_path);
tty->link->driver_data = pts_path;
retval = ptm_driver->ops->open(tty, filp); retval = ptm_driver->ops->open(tty, filp);
if (retval) if (retval)
goto err_path_put; goto err_release;
tty_debug_hangup(tty, "opening (count=%d)\n", tty->count); tty_debug_hangup(tty, "opening (count=%d)\n", tty->count);
tty_unlock(tty); tty_unlock(tty);
return 0; return 0;
err_path_put:
path_put(pts_path);
kfree(pts_path);
err_release: err_release:
tty_unlock(tty); tty_unlock(tty);
// This will also put-ref the fsi // This will also put-ref the fsi
......
...@@ -2518,6 +2518,9 @@ long tty_ioctl(struct file *file, unsigned int cmd, unsigned long arg) ...@@ -2518,6 +2518,9 @@ long tty_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
case TIOCSSERIAL: case TIOCSSERIAL:
tty_warn_deprecated_flags(p); tty_warn_deprecated_flags(p);
break; break;
case TIOCGPTPEER:
/* Special because the struct file is needed */
return ptm_open_peer(file, tty, (int)arg);
default: default:
retval = tty_jobctrl_ioctl(tty, real_tty, file, cmd, arg); retval = tty_jobctrl_ioctl(tty, real_tty, file, cmd, arg);
if (retval != -ENOIOCTLCMD) if (retval != -ENOIOCTLCMD)
......
...@@ -133,6 +133,50 @@ static inline struct pts_fs_info *DEVPTS_SB(struct super_block *sb) ...@@ -133,6 +133,50 @@ static inline struct pts_fs_info *DEVPTS_SB(struct super_block *sb)
return sb->s_fs_info; return sb->s_fs_info;
} }
static int devpts_ptmx_path(struct path *path)
{
struct super_block *sb;
int err;
/* Has the devpts filesystem already been found? */
if (path->mnt->mnt_sb->s_magic == DEVPTS_SUPER_MAGIC)
return 0;
/* Is a devpts filesystem at "pts" in the same directory? */
err = path_pts(path);
if (err)
return err;
/* Is the path the root of a devpts filesystem? */
sb = path->mnt->mnt_sb;
if ((sb->s_magic != DEVPTS_SUPER_MAGIC) ||
(path->mnt->mnt_root != sb->s_root))
return -ENODEV;
return 0;
}
struct vfsmount *devpts_mntget(struct file *filp, struct pts_fs_info *fsi)
{
struct path path;
int err;
path = filp->f_path;
path_get(&path);
err = devpts_ptmx_path(&path);
dput(path.dentry);
if (err) {
mntput(path.mnt);
path.mnt = ERR_PTR(err);
}
if (DEVPTS_SB(path.mnt->mnt_sb) != fsi) {
mntput(path.mnt);
path.mnt = ERR_PTR(-ENODEV);
}
return path.mnt;
}
struct pts_fs_info *devpts_acquire(struct file *filp) struct pts_fs_info *devpts_acquire(struct file *filp)
{ {
struct pts_fs_info *result; struct pts_fs_info *result;
...@@ -143,27 +187,16 @@ struct pts_fs_info *devpts_acquire(struct file *filp) ...@@ -143,27 +187,16 @@ struct pts_fs_info *devpts_acquire(struct file *filp)
path = filp->f_path; path = filp->f_path;
path_get(&path); path_get(&path);
/* Has the devpts filesystem already been found? */ err = devpts_ptmx_path(&path);
sb = path.mnt->mnt_sb;
if (sb->s_magic != DEVPTS_SUPER_MAGIC) {
/* Is a devpts filesystem at "pts" in the same directory? */
err = path_pts(&path);
if (err) { if (err) {
result = ERR_PTR(err); result = ERR_PTR(err);
goto out; goto out;
} }
/* Is the path the root of a devpts filesystem? */
result = ERR_PTR(-ENODEV);
sb = path.mnt->mnt_sb;
if ((sb->s_magic != DEVPTS_SUPER_MAGIC) ||
(path.mnt->mnt_root != sb->s_root))
goto out;
}
/* /*
* pty code needs to hold extra references in case of last /dev/tty close * pty code needs to hold extra references in case of last /dev/tty close
*/ */
sb = path.mnt->mnt_sb;
atomic_inc(&sb->s_active); atomic_inc(&sb->s_active);
result = DEVPTS_SB(sb); result = DEVPTS_SB(sb);
......
...@@ -19,6 +19,7 @@ ...@@ -19,6 +19,7 @@
struct pts_fs_info; struct pts_fs_info;
struct vfsmount *devpts_mntget(struct file *, struct pts_fs_info *);
struct pts_fs_info *devpts_acquire(struct file *); struct pts_fs_info *devpts_acquire(struct file *);
void devpts_release(struct pts_fs_info *); void devpts_release(struct pts_fs_info *);
...@@ -32,6 +33,15 @@ void *devpts_get_priv(struct dentry *); ...@@ -32,6 +33,15 @@ void *devpts_get_priv(struct dentry *);
/* unlink */ /* unlink */
void devpts_pty_kill(struct dentry *); void devpts_pty_kill(struct dentry *);
/* in pty.c */
int ptm_open_peer(struct file *master, struct tty_struct *tty, int flags);
#else
static inline int
ptm_open_peer(struct file *master, struct tty_struct *tty, int flags)
{
return -EIO;
}
#endif #endif
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment