Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 221903
b: refs/heads/master
c: 100eeae
h: refs/heads/master
i:
  221901: 1628de9
  221899: 8dea15e
  221895: c4ade9c
  221887: be68127
v: v3
  • Loading branch information
Jiri Slaby authored and Greg Kroah-Hartman committed Nov 9, 2010
1 parent b10675c commit df528ea
Show file tree
Hide file tree
Showing 2 changed files with 30 additions and 1 deletion.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: ebf7c06635fbcf21a59e60187e166c5c23c57b06
refs/heads/master: 100eeae2c5ce23b4db93ff320ee330ef1d740151
29 changes: 29 additions & 0 deletions trunk/drivers/tty/tty_ldisc.c
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@

static DEFINE_SPINLOCK(tty_ldisc_lock);
static DECLARE_WAIT_QUEUE_HEAD(tty_ldisc_wait);
static DECLARE_WAIT_QUEUE_HEAD(tty_ldisc_idle);
/* Line disc dispatch table */
static struct tty_ldisc_ops *tty_ldiscs[NR_LDISCS];

Expand Down Expand Up @@ -83,6 +84,7 @@ static void put_ldisc(struct tty_ldisc *ld)
return;
}
local_irq_restore(flags);
wake_up(&tty_ldisc_idle);
}

/**
Expand Down Expand Up @@ -530,6 +532,23 @@ static int tty_ldisc_halt(struct tty_struct *tty)
return cancel_delayed_work_sync(&tty->buf.work);
}

/**
* tty_ldisc_wait_idle - wait for the ldisc to become idle
* @tty: tty to wait for
*
* Wait for the line discipline to become idle. The discipline must
* have been halted for this to guarantee it remains idle.
*/
static int tty_ldisc_wait_idle(struct tty_struct *tty)
{
int ret;
ret = wait_event_interruptible_timeout(tty_ldisc_idle,
atomic_read(&tty->ldisc->users) == 1, 5 * HZ);
if (ret < 0)
return ret;
return ret > 0 ? 0 : -EBUSY;
}

/**
* tty_set_ldisc - set line discipline
* @tty: the terminal to set
Expand Down Expand Up @@ -634,8 +653,17 @@ int tty_set_ldisc(struct tty_struct *tty, int ldisc)

flush_scheduled_work();

retval = tty_ldisc_wait_idle(tty);

tty_lock();
mutex_lock(&tty->ldisc_mutex);

/* handle wait idle failure locked */
if (retval) {
tty_ldisc_put(new_ldisc);
goto enable;
}

if (test_bit(TTY_HUPPED, &tty->flags)) {
/* We were raced by the hangup method. It will have stomped
the ldisc data and closed the ldisc down */
Expand Down Expand Up @@ -669,6 +697,7 @@ int tty_set_ldisc(struct tty_struct *tty, int ldisc)

tty_ldisc_put(o_ldisc);

enable:
/*
* Allow ldisc referencing to occur again
*/
Expand Down

0 comments on commit df528ea

Please sign in to comment.