|
@@ -1122,14 +1122,12 @@ int tty_mode_ioctl(struct tty_struct *tty, struct file *file,
|
|
|
}
|
|
|
EXPORT_SYMBOL_GPL(tty_mode_ioctl);
|
|
|
|
|
|
-int tty_perform_flush(struct tty_struct *tty, unsigned long arg)
|
|
|
+
|
|
|
+/* Caller guarantees ldisc reference is held */
|
|
|
+static int __tty_perform_flush(struct tty_struct *tty, unsigned long arg)
|
|
|
{
|
|
|
- struct tty_ldisc *ld;
|
|
|
- int retval = tty_check_change(tty);
|
|
|
- if (retval)
|
|
|
- return retval;
|
|
|
+ struct tty_ldisc *ld = tty->ldisc;
|
|
|
|
|
|
- ld = tty_ldisc_ref_wait(tty);
|
|
|
switch (arg) {
|
|
|
case TCIFLUSH:
|
|
|
if (ld && ld->ops->flush_buffer) {
|
|
@@ -1147,12 +1145,24 @@ int tty_perform_flush(struct tty_struct *tty, unsigned long arg)
|
|
|
tty_driver_flush_buffer(tty);
|
|
|
break;
|
|
|
default:
|
|
|
- tty_ldisc_deref(ld);
|
|
|
return -EINVAL;
|
|
|
}
|
|
|
- tty_ldisc_deref(ld);
|
|
|
return 0;
|
|
|
}
|
|
|
+
|
|
|
+int tty_perform_flush(struct tty_struct *tty, unsigned long arg)
|
|
|
+{
|
|
|
+ struct tty_ldisc *ld;
|
|
|
+ int retval = tty_check_change(tty);
|
|
|
+ if (retval)
|
|
|
+ return retval;
|
|
|
+
|
|
|
+ ld = tty_ldisc_ref_wait(tty);
|
|
|
+ retval = __tty_perform_flush(tty, arg);
|
|
|
+ if (ld)
|
|
|
+ tty_ldisc_deref(ld);
|
|
|
+ return retval;
|
|
|
+}
|
|
|
EXPORT_SYMBOL_GPL(tty_perform_flush);
|
|
|
|
|
|
int n_tty_ioctl_helper(struct tty_struct *tty, struct file *file,
|
|
@@ -1191,7 +1201,7 @@ int n_tty_ioctl_helper(struct tty_struct *tty, struct file *file,
|
|
|
}
|
|
|
return 0;
|
|
|
case TCFLSH:
|
|
|
- return tty_perform_flush(tty, arg);
|
|
|
+ return __tty_perform_flush(tty, arg);
|
|
|
default:
|
|
|
/* Try the mode commands */
|
|
|
return tty_mode_ioctl(tty, file, cmd, arg);
|