netdev.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH 1/1] net: netlink: Fix multicast group storage allocation for families with more than one groups
@ 2016-01-11 12:26 Matti Vaittinen
  2016-01-12 21:42 ` David Miller
  0 siblings, 1 reply; 3+ messages in thread
From: Matti Vaittinen @ 2016-01-11 12:26 UTC (permalink / raw)
  To: David S. Miller, Johannes Berg, Jiri Benc, Yaowei Bai, Matti Vaittinen
  Cc: netdev, linux-kernel, Sverdlin Alexander, Pennanen Teppo Olavi

Hello,

    I got an oops when developing a driver utilizing more than 32
    multicast groups. After a little print adding I saw that part of 
    the address of policy struct was overwritten. It seems to me that
    amount of longs needed to store the multicast addresses is
    calculated wrong for families with more than one multicast groups.
    Problem is easy to reproduce by: 
    
    1. generating a module which registers large amount of multicast 
    groups.
    2. Loading the module
    3. Sending the netlink message which requests multicast groups for
    this family.

    I believe step 3 may not be even needed if there is enough of the
    groups registered.

    Patch created against just cloned net repository below.

    Br.
        Matti Vaittinen



Multicast groups are stored in global buffer. Check for needed buffer size
incorrectly compares buffer size to first id for family. This means that
for families with more than one mcast id one may allocate too small buffer
and end up writing rest of the groups to some unallocated memory. Fix the 
buffer size check to compare allocated space to last mcast id for the 
family.

Tested on ARM using kernel 3.14

Signed-off-by: Matti Vaittinen <matti.vaittinen@nokia.com>
---
 net/netlink/genetlink.c | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/net/netlink/genetlink.c b/net/netlink/genetlink.c
index bc0e504..a992083 100644
--- a/net/netlink/genetlink.c
+++ b/net/netlink/genetlink.c
@@ -185,7 +185,7 @@ static int genl_allocate_reserve_groups(int n_groups, int *first_id)
 			}
 		}
 
-		if (id >= mc_groups_longs * BITS_PER_LONG) {
+		if (id + n_groups >= mc_groups_longs * BITS_PER_LONG) {
 			unsigned long new_longs = mc_groups_longs +
 						  BITS_TO_LONGS(n_groups);
 			size_t nlen = new_longs * sizeof(unsigned long);
-- 
2.1.0

^ permalink raw reply related	[flat|nested] 3+ messages in thread

* Re: [PATCH 1/1] net: netlink: Fix multicast group storage allocation for families with more than one groups
  2016-01-11 12:26 [PATCH 1/1] net: netlink: Fix multicast group storage allocation for families with more than one groups Matti Vaittinen
@ 2016-01-12 21:42 ` David Miller
  2016-01-13  7:53   ` Matti Vaittinen
  0 siblings, 1 reply; 3+ messages in thread
From: David Miller @ 2016-01-12 21:42 UTC (permalink / raw)
  To: matti.vaittinen
  Cc: johannes.berg, jbenc, bywxiaobai, netdev, linux-kernel,
	alexander.sverdlin, teppo.o.pennanen

From: Matti Vaittinen <matti.vaittinen@nokia.com>
Date: Mon, 11 Jan 2016 14:26:19 +0200

> Multicast groups are stored in global buffer. Check for needed buffer size
> incorrectly compares buffer size to first id for family. This means that
> for families with more than one mcast id one may allocate too small buffer
> and end up writing rest of the groups to some unallocated memory. Fix the 
> buffer size check to compare allocated space to last mcast id for the 
> family.
> 
> Tested on ARM using kernel 3.14
> 
> Signed-off-by: Matti Vaittinen <matti.vaittinen@nokia.com>

Indeed, it looks like this function was never tested with any value
of n_groups other than one.

But I think your change has an off-by-one bug:

> -		if (id >= mc_groups_longs * BITS_PER_LONG) {
> +		if (id + n_groups >= mc_groups_longs * BITS_PER_LONG) {

I think this needs to be "id + n_groups > ".  Consider the existing,
working, case of "n_groups == 1".  Now you're adding '1' and therefore
the test needs to be adjusted from >= to >.

Thanks.

^ permalink raw reply	[flat|nested] 3+ messages in thread

* Re: [PATCH 1/1] net: netlink: Fix multicast group storage allocation for families with more than one groups
  2016-01-12 21:42 ` David Miller
@ 2016-01-13  7:53   ` Matti Vaittinen
  0 siblings, 0 replies; 3+ messages in thread
From: Matti Vaittinen @ 2016-01-13  7:53 UTC (permalink / raw)
  To: EXT David Miller
  Cc: johannes.berg, jbenc, netdev, linux-kernel, alexander.sverdlin,
	teppo.o.pennanen

Hello David & others,

On Tue, Jan 12, 2016 at 04:42:11PM -0500, EXT David Miller wrote:
> But I think your change has an off-by-one bug:
> 
> From: Matti Vaittinen <matti.vaittinen@nokia.com>
> > -		if (id >= mc_groups_longs * BITS_PER_LONG) {
> > +		if (id + n_groups >= mc_groups_longs * BITS_PER_LONG) {
> 
> I think this needs to be "id + n_groups > ".  Consider the existing,
> working, case of "n_groups == 1".  Now you're adding '1' and therefore
> the test needs to be adjusted from >= to >.

Absolutely. I did patch v2.

Br.
    Matti Vaittinen

^ permalink raw reply	[flat|nested] 3+ messages in thread

end of thread, other threads:[~2016-01-13  7:53 UTC | newest]

Thread overview: 3+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2016-01-11 12:26 [PATCH 1/1] net: netlink: Fix multicast group storage allocation for families with more than one groups Matti Vaittinen
2016-01-12 21:42 ` David Miller
2016-01-13  7:53   ` Matti Vaittinen

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).