diff options
author | Jaganath Kanakkassery <jaganath.k@samsung.com> | 2012-07-13 08:47:55 -0400 |
---|---|---|
committer | Gustavo Padovan <gustavo.padovan@collabora.co.uk> | 2012-08-06 14:02:58 -0400 |
commit | 4af66c691f4e5c2db9bb00793669a548e9db1974 (patch) | |
tree | 2ea27d4467c635d78d5b001ee4bee384b1ab7eea /net | |
parent | 3064837289259843310b266a9422aca5f5b4b9c7 (diff) |
Bluetooth: Free the l2cap channel list only when refcount is zero
Move the l2cap channel list chan->global_l under the refcnt
protection and free it based on the refcnt.
Signed-off-by: Jaganath Kanakkassery <jaganath.k@samsung.com>
Signed-off-by: Syam Sidhardhan <s.syam@samsung.com>
Reviewed-by: Andrei Emeltchenko <andrei.emeltchenko@intel.com>
Signed-off-by: Gustavo Padovan <gustavo.padovan@collabora.co.uk>
Diffstat (limited to 'net')
-rw-r--r-- | net/bluetooth/a2mp.c | 2 | ||||
-rw-r--r-- | net/bluetooth/l2cap_core.c | 8 | ||||
-rw-r--r-- | net/bluetooth/l2cap_sock.c | 2 |
3 files changed, 7 insertions, 5 deletions
diff --git a/net/bluetooth/a2mp.c b/net/bluetooth/a2mp.c index 79af661a58dd..0760d1fed6f0 100644 --- a/net/bluetooth/a2mp.c +++ b/net/bluetooth/a2mp.c | |||
@@ -416,7 +416,7 @@ static int a2mp_chan_recv_cb(struct l2cap_chan *chan, struct sk_buff *skb) | |||
416 | 416 | ||
417 | static void a2mp_chan_close_cb(struct l2cap_chan *chan) | 417 | static void a2mp_chan_close_cb(struct l2cap_chan *chan) |
418 | { | 418 | { |
419 | l2cap_chan_destroy(chan); | 419 | l2cap_chan_put(chan); |
420 | } | 420 | } |
421 | 421 | ||
422 | static void a2mp_chan_state_change_cb(struct l2cap_chan *chan, int state) | 422 | static void a2mp_chan_state_change_cb(struct l2cap_chan *chan, int state) |
diff --git a/net/bluetooth/l2cap_core.c b/net/bluetooth/l2cap_core.c index 79923d8bbe97..9f8b29ef5b68 100644 --- a/net/bluetooth/l2cap_core.c +++ b/net/bluetooth/l2cap_core.c | |||
@@ -416,13 +416,15 @@ struct l2cap_chan *l2cap_chan_create(void) | |||
416 | return chan; | 416 | return chan; |
417 | } | 417 | } |
418 | 418 | ||
419 | void l2cap_chan_destroy(struct l2cap_chan *chan) | 419 | static void l2cap_chan_destroy(struct l2cap_chan *chan) |
420 | { | 420 | { |
421 | BT_DBG("chan %p", chan); | ||
422 | |||
421 | write_lock(&chan_list_lock); | 423 | write_lock(&chan_list_lock); |
422 | list_del(&chan->global_l); | 424 | list_del(&chan->global_l); |
423 | write_unlock(&chan_list_lock); | 425 | write_unlock(&chan_list_lock); |
424 | 426 | ||
425 | l2cap_chan_put(chan); | 427 | kfree(chan); |
426 | } | 428 | } |
427 | 429 | ||
428 | void l2cap_chan_hold(struct l2cap_chan *c) | 430 | void l2cap_chan_hold(struct l2cap_chan *c) |
@@ -437,7 +439,7 @@ void l2cap_chan_put(struct l2cap_chan *c) | |||
437 | BT_DBG("chan %p orig refcnt %d", c, atomic_read(&c->refcnt)); | 439 | BT_DBG("chan %p orig refcnt %d", c, atomic_read(&c->refcnt)); |
438 | 440 | ||
439 | if (atomic_dec_and_test(&c->refcnt)) | 441 | if (atomic_dec_and_test(&c->refcnt)) |
440 | kfree(c); | 442 | l2cap_chan_destroy(c); |
441 | } | 443 | } |
442 | 444 | ||
443 | void l2cap_chan_set_defaults(struct l2cap_chan *chan) | 445 | void l2cap_chan_set_defaults(struct l2cap_chan *chan) |
diff --git a/net/bluetooth/l2cap_sock.c b/net/bluetooth/l2cap_sock.c index a4bb27e8427e..79350d10087c 100644 --- a/net/bluetooth/l2cap_sock.c +++ b/net/bluetooth/l2cap_sock.c | |||
@@ -823,7 +823,7 @@ static void l2cap_sock_kill(struct sock *sk) | |||
823 | 823 | ||
824 | /* Kill poor orphan */ | 824 | /* Kill poor orphan */ |
825 | 825 | ||
826 | l2cap_chan_destroy(l2cap_pi(sk)->chan); | 826 | l2cap_chan_put(l2cap_pi(sk)->chan); |
827 | sock_set_flag(sk, SOCK_DEAD); | 827 | sock_set_flag(sk, SOCK_DEAD); |
828 | sock_put(sk); | 828 | sock_put(sk); |
829 | } | 829 | } |