diff options
author | Allan Stephens <allan.stephens@windriver.com> | 2010-08-17 07:00:07 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2010-08-17 20:31:53 -0400 |
commit | b02b69c8a403859ec72090742727e853d606a325 (patch) | |
tree | 6c46c41cc46fde8ea8a45d6265a3381556e0f84e /net/tipc | |
parent | f662c07058f7e6365ae65080d772f9122f6f50a9 (diff) |
tipc: Check for disabled bearer when processing incoming messages
Add a check to tipc_recv_msg() to ensure it discards messages
arriving on a newly disabled bearer. This is needed to deal with a
race condition that can arise if the bearer is in the midst of being
disabled when it receives a message. Performing the check after
tipc_net_lock has been taken ensures that TIPC's bearers are in a
stable state while the message is being processed.
Signed-off-by: Allan Stephens <allan.stephens@windriver.com>
Signed-off-by: Paul Gortmaker <paul.gortmaker@windriver.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/tipc')
-rw-r--r-- | net/tipc/link.c | 14 |
1 files changed, 14 insertions, 0 deletions
diff --git a/net/tipc/link.c b/net/tipc/link.c index a3616b99529b..9d18c9b7638b 100644 --- a/net/tipc/link.c +++ b/net/tipc/link.c | |||
@@ -1802,6 +1802,15 @@ static int link_recv_buf_validate(struct sk_buff *buf) | |||
1802 | return pskb_may_pull(buf, hdr_size); | 1802 | return pskb_may_pull(buf, hdr_size); |
1803 | } | 1803 | } |
1804 | 1804 | ||
1805 | /** | ||
1806 | * tipc_recv_msg - process TIPC messages arriving from off-node | ||
1807 | * @head: pointer to message buffer chain | ||
1808 | * @tb_ptr: pointer to bearer message arrived on | ||
1809 | * | ||
1810 | * Invoked with no locks held. Bearer pointer must point to a valid bearer | ||
1811 | * structure (i.e. cannot be NULL), but bearer can be inactive. | ||
1812 | */ | ||
1813 | |||
1805 | void tipc_recv_msg(struct sk_buff *head, struct tipc_bearer *tb_ptr) | 1814 | void tipc_recv_msg(struct sk_buff *head, struct tipc_bearer *tb_ptr) |
1806 | { | 1815 | { |
1807 | read_lock_bh(&tipc_net_lock); | 1816 | read_lock_bh(&tipc_net_lock); |
@@ -1819,6 +1828,11 @@ void tipc_recv_msg(struct sk_buff *head, struct tipc_bearer *tb_ptr) | |||
1819 | 1828 | ||
1820 | head = head->next; | 1829 | head = head->next; |
1821 | 1830 | ||
1831 | /* Ensure bearer is still enabled */ | ||
1832 | |||
1833 | if (unlikely(!b_ptr->active)) | ||
1834 | goto cont; | ||
1835 | |||
1822 | /* Ensure message is well-formed */ | 1836 | /* Ensure message is well-formed */ |
1823 | 1837 | ||
1824 | if (unlikely(!link_recv_buf_validate(buf))) | 1838 | if (unlikely(!link_recv_buf_validate(buf))) |