if (tty->ops->close)
                tty->ops->close(tty, filp);
 
-       tty_unlock(tty);
+       /* If tty is pty master, lock the slave pty (stable lock order) */
+       tty_lock_slave(o_tty);
+
        /*
         * Sanity check: if tty->count is going to zero, there shouldn't be
         * any waiters on tty->read_wait or tty->write_wait.  We test the
         * Thus this test wouldn't be triggered at the time the slave closed,
         * so we do it now.
         */
-       tty_lock_pair(tty, o_tty);
-
        while (1) {
                do_sleep = 0;
 
        /* check whether both sides are closing ... */
        final = !tty->count && !(o_tty && o_tty->count);
 
-       tty_unlock_pair(tty, o_tty);
+       tty_unlock_slave(o_tty);
+       tty_unlock(tty);
+
        /* At this point, the tty->count == 0 should ensure a dead tty
           cannot be re-opened by a racing opener */
 
 
 #include <linux/semaphore.h>
 #include <linux/sched.h>
 
+/*
+ * Nested tty locks are necessary for releasing pty pairs.
+ * The stable lock order is master pty first, then slave pty.
+ */
+
 /* Legacy tty mutex glue */
 
 enum {
 }
 EXPORT_SYMBOL(tty_unlock);
 
-/*
- * Getting the big tty mutex for a pair of ttys with lock ordering
- * On a non pty/tty pair tty2 can be NULL which is just fine.
- */
-void __lockfunc tty_lock_pair(struct tty_struct *tty,
-                                       struct tty_struct *tty2)
+void __lockfunc tty_lock_slave(struct tty_struct *tty)
 {
-       if (tty < tty2) {
-               tty_lock(tty);
-               tty_lock_nested(tty2, TTY_MUTEX_NESTED);
-       } else {
-               if (tty2 && tty2 != tty)
-                       tty_lock(tty2);
+       if (tty && tty != tty->link) {
+               WARN_ON(!mutex_is_locked(&tty->link->legacy_mutex) ||
+                       !tty->driver->type == TTY_DRIVER_TYPE_PTY ||
+                       !tty->driver->type == PTY_TYPE_SLAVE);
                tty_lock_nested(tty, TTY_MUTEX_NESTED);
        }
 }
-EXPORT_SYMBOL(tty_lock_pair);
 
-void __lockfunc tty_unlock_pair(struct tty_struct *tty,
-                                               struct tty_struct *tty2)
+void __lockfunc tty_unlock_slave(struct tty_struct *tty)
 {
-       tty_unlock(tty);
-       if (tty2 && tty2 != tty)
-               tty_unlock(tty2);
+       if (tty && tty != tty->link)
+               tty_unlock(tty);
 }
-EXPORT_SYMBOL(tty_unlock_pair);
 
 /* functions for preparation of BKL removal */
 extern void __lockfunc tty_lock(struct tty_struct *tty);
 extern void __lockfunc tty_unlock(struct tty_struct *tty);
-extern void __lockfunc tty_lock_pair(struct tty_struct *tty,
-                               struct tty_struct *tty2);
-extern void __lockfunc tty_unlock_pair(struct tty_struct *tty,
-                               struct tty_struct *tty2);
-
+extern void __lockfunc tty_lock_slave(struct tty_struct *tty);
+extern void __lockfunc tty_unlock_slave(struct tty_struct *tty);
 /*
  * this shall be called only from where BTM is held (like close)
  *