tty: Make core responsible for synchronizing its work
The tty core relies on the ldisc layer for synchronizing destruction of the tty. Instead, the final tty release must wait for any pending tty work to complete prior to tty destruction. Signed-off-by: Peter Hurley <peter@hurleysoftware.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
4f98d46751
commit
a2965b7bee
2 changed files with 21 additions and 20 deletions
|
@ -1510,6 +1510,17 @@ void tty_free_termios(struct tty_struct *tty)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(tty_free_termios);
|
EXPORT_SYMBOL(tty_free_termios);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* tty_flush_works - flush all works of a tty
|
||||||
|
* @tty: tty device to flush works for
|
||||||
|
*
|
||||||
|
* Sync flush all works belonging to @tty.
|
||||||
|
*/
|
||||||
|
static void tty_flush_works(struct tty_struct *tty)
|
||||||
|
{
|
||||||
|
flush_work(&tty->SAK_work);
|
||||||
|
flush_work(&tty->hangup_work);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* release_one_tty - release tty structure memory
|
* release_one_tty - release tty structure memory
|
||||||
|
@ -1831,6 +1842,12 @@ int tty_release(struct inode *inode, struct file *filp)
|
||||||
* Ask the line discipline code to release its structures
|
* Ask the line discipline code to release its structures
|
||||||
*/
|
*/
|
||||||
tty_ldisc_release(tty, o_tty);
|
tty_ldisc_release(tty, o_tty);
|
||||||
|
|
||||||
|
/* Wait for pending work before tty destruction commmences */
|
||||||
|
tty_flush_works(tty);
|
||||||
|
if (o_tty)
|
||||||
|
tty_flush_works(o_tty);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The release_tty 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 tty_unlock_pair
|
* the slots and preserving the termios structure. The tty_unlock_pair
|
||||||
|
|
|
@ -498,18 +498,6 @@ static void tty_ldisc_restore(struct tty_struct *tty, struct tty_ldisc *old)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* tty_ldisc_flush_works - flush all works of a tty
|
|
||||||
* @tty: tty device to flush works for
|
|
||||||
*
|
|
||||||
* Sync flush all works belonging to @tty.
|
|
||||||
*/
|
|
||||||
static void tty_ldisc_flush_works(struct tty_struct *tty)
|
|
||||||
{
|
|
||||||
flush_work(&tty->SAK_work);
|
|
||||||
flush_work(&tty->hangup_work);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* tty_ldisc_wait_idle - wait for the ldisc to become idle
|
* tty_ldisc_wait_idle - wait for the ldisc to become idle
|
||||||
* @tty: tty to wait for
|
* @tty: tty to wait for
|
||||||
|
@ -698,13 +686,13 @@ int tty_set_ldisc(struct tty_struct *tty, int ldisc)
|
||||||
retval = tty_ldisc_halt(tty, o_tty, 5 * HZ);
|
retval = tty_ldisc_halt(tty, o_tty, 5 * HZ);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Wait for ->hangup_work and ->buf.work handlers to terminate.
|
* Wait for hangup to complete, if pending.
|
||||||
* We must drop the mutex here in case a hangup is also in process.
|
* We must drop the mutex here in case a hangup is also in process.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
mutex_unlock(&tty->ldisc_mutex);
|
mutex_unlock(&tty->ldisc_mutex);
|
||||||
|
|
||||||
tty_ldisc_flush_works(tty);
|
flush_work(&tty->hangup_work);
|
||||||
|
|
||||||
tty_lock(tty);
|
tty_lock(tty);
|
||||||
mutex_lock(&tty->ldisc_mutex);
|
mutex_lock(&tty->ldisc_mutex);
|
||||||
|
@ -951,15 +939,11 @@ static void tty_ldisc_kill(struct tty_struct *tty)
|
||||||
void tty_ldisc_release(struct tty_struct *tty, struct tty_struct *o_tty)
|
void tty_ldisc_release(struct tty_struct *tty, struct tty_struct *o_tty)
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
* Prevent flush_to_ldisc() from rescheduling the work for later. Then
|
* Shutdown this line discipline. As this is the final close,
|
||||||
* kill any delayed work. As this is the final close it does not
|
* it does not race with the set_ldisc code path.
|
||||||
* race with the set_ldisc code path.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
tty_ldisc_halt(tty, o_tty, MAX_SCHEDULE_TIMEOUT);
|
tty_ldisc_halt(tty, o_tty, MAX_SCHEDULE_TIMEOUT);
|
||||||
tty_ldisc_flush_works(tty);
|
|
||||||
if (o_tty)
|
|
||||||
tty_ldisc_flush_works(o_tty);
|
|
||||||
|
|
||||||
tty_lock_pair(tty, o_tty);
|
tty_lock_pair(tty, o_tty);
|
||||||
/* This will need doing differently if we need to lock */
|
/* This will need doing differently if we need to lock */
|
||||||
|
|
Loading…
Reference in a new issue