[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <BA0C77F83D4255479F53DBBC4C4BAFC55FF699@SHSMSX101.ccr.corp.intel.com>
Date: Tue, 15 Apr 2014 02:49:02 +0000
From: "Chen, Tingjie" <tingjie.chen@...el.com>
To: Greg Kroah-Hartman <gregkh@...uxfoundation.org>
CC: Jiri Slaby <jslaby@...e.cz>,
"linux-kernel@...r.kernel.org" <linux-kernel@...r.kernel.org>,
"Zhang, Jun" <jun.zhang@...el.com>,
"One Thousand Gnomes" <gnomes@...rguk.ukuu.org.uk>,
"Liu, Chuansheng" <chuansheng.liu@...el.com>
Subject: RE: [PATCH] [PATCH V2] tty: memleak in alloc_pid
Hello,
The related code in disassociate_ctty() as following:
---------------------------------------
/* sometimes current->signal->tty_old_pgrp is NULL (a stand for tty_old_pgrp not NULL, !a for tty_old_pgrp is NULL) */
spin_lock_irq(¤t->sighand->siglock);
put_pid(current->signal->tty_old_pgrp); // when a is NULL, will not actually put_pid() by decrease the count
current->signal->tty_old_pgrp = NULL;
spin_unlock_irq(¤t->sighand->siglock);
/* sometimes current->signal->tty is NULL (b stand for signal->tty, !b for signal->tty is NULL) */
tty = get_current_tty();
if (tty) {
unsigned long flags;
spin_lock_irqsave(&tty->ctrl_lock, flags);
put_pid(tty->session);
put_pid(tty->pgrp);
tty->session = NULL;
tty->pgrp = NULL;
spin_unlock_irqrestore(&tty->ctrl_lock, flags);
tty_kref_put(tty);
} else {
#ifdef TTY_DEBUG_HANGUP
printk(KERN_DEBUG "error attempted to write to tty [0x%p]"
" = NULL", tty);
#endif
}
------------------------------------
Commonly there are 2 normal case flow for the code:
1/ in task: getprop: !a --> b
This case, current->signal->tty_old_pgrp is NULL, but get_current_tty() is not NULL,
Really put_pid() as following:
put_pid(tty->session);
put_pid(tty->pgrp);
2/ in task: dumpsys: a --> !b
This case, currnet->signal->tty_old_pgrp is not NULL, but get_current_tty() is NULL,
Really put_pid() as following:
put_pid(current->signal->tty_old_pgrp);
There is one abnormal case:
3/ in task: dumpsys: !a -> !b
This case, because of race, current->signal->tty_old_pgrp and get_current_tty() are NULL,
Lack of put_pid(), pid->count will not be 0 at last, this pid will be leak.
When read the current->signal->tty_old_pgrp and current->signal->tty,
The 2 operations should be atomic, which protected by spinlock: current->sighand->siglock.
So the sentence: spin_unlock_irq(¤t->sighand->siglock);
need to move down after tty has process completed.
Thanks,
-----Original Message-----
From: Greg Kroah-Hartman [mailto:gregkh@...uxfoundation.org]
Sent: Monday, April 14, 2014 7:47 PM
To: Chen, Tingjie
Cc: Jiri Slaby; linux-kernel@...r.kernel.org; Zhang, Jun
Subject: Re: [PATCH] [PATCH V2] tty: memleak in alloc_pid
On Mon, Apr 14, 2014 at 03:31:15PM +0800, Chen Tingjie wrote:
> There is memleak in alloc_pid:
> ------------------------------
> unreferenced object 0xd3453a80 (size 64):
> comm "adbd", pid 1730, jiffies 66363 (age 6586.950s)
> hex dump (first 32 bytes):
> 01 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................
> 00 00 00 00 40 c2 f6 d5 00 d3 25 c1 59 28 00 00 ....@.....%.Y(..
> backtrace:
> [<c1a6f15c>] kmemleak_alloc+0x3c/0xa0
> [<c1320546>] kmem_cache_alloc+0xc6/0x190
> [<c125d51e>] alloc_pid+0x1e/0x400
> [<c123d344>] copy_process.part.39+0xad4/0x1120
> [<c123da59>] do_fork+0x99/0x330
> [<c123dd58>] sys_fork+0x28/0x30
> [<c1a89a08>] syscall_call+0x7/0xb
> [<ffffffff>] 0xffffffff
>
> the leak is due to unreleased pid->count, which execute in function:
> get_pid()(pid->count++) and put_pid()(pid->count--).
>
> The race condition as following:
> task[dumpsys] task[adbd]
> in disassociate_ctty() in tty_signal_session_leader()
> ----------------------- -------------------------
> tty = get_current_tty();
> // tty is not NULL
> ...
> spin_lock_irq(¤t->sighand->siglock);
> put_pid(current->signal->tty_old_pgrp);
> current->signal->tty_old_pgrp = NULL;
> spin_unlock_irq(¤t->sighand->siglock);
>
> spin_lock_irq(&p->sighand->siglock);
> ...
> p->signal->tty = NULL;
> ...
> spin_unlock_irq(&p->sighand->siglock);
>
> tty = get_current_tty();
> // tty NULL, goto else branch by accident.
> if (tty) {
> ...
> put_pid(tty_session);
> put_pid(tty_pgrp);
> ...
> } else {
> print msg
> }
>
> in task[dumpsys], in disassociate_ctty(), tty is set NULL by
> task[adbd], tty_signal_session_leader(), then it goto else branch and
> lack of put_pid(), cause memleak.
>
> move spin_unlock(sighand->siglock) after get_current_tty() can avoid
> the race and fix the memleak.
>
> Change-Id: Ic960dda039c8f99aad3e0f4d176489a966c62f6a
Why is this line here?
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
Powered by blists - more mailing lists