diff options
-rw-r--r-- | include/linux/mroute.h | 13 | ||||
-rw-r--r-- | net/ipv4/ipmr.c | 12 |
2 files changed, 20 insertions, 5 deletions
diff --git a/include/linux/mroute.h b/include/linux/mroute.h index 08bc776d05e2..d5f69151f692 100644 --- a/include/linux/mroute.h +++ b/include/linux/mroute.h | |||
@@ -59,13 +59,18 @@ struct vifctl { | |||
59 | unsigned char vifc_flags; /* VIFF_ flags */ | 59 | unsigned char vifc_flags; /* VIFF_ flags */ |
60 | unsigned char vifc_threshold; /* ttl limit */ | 60 | unsigned char vifc_threshold; /* ttl limit */ |
61 | unsigned int vifc_rate_limit; /* Rate limiter values (NI) */ | 61 | unsigned int vifc_rate_limit; /* Rate limiter values (NI) */ |
62 | struct in_addr vifc_lcl_addr; /* Our address */ | 62 | union { |
63 | struct in_addr vifc_lcl_addr; /* Local interface address */ | ||
64 | int vifc_lcl_ifindex; /* Local interface index */ | ||
65 | }; | ||
63 | struct in_addr vifc_rmt_addr; /* IPIP tunnel addr */ | 66 | struct in_addr vifc_rmt_addr; /* IPIP tunnel addr */ |
64 | }; | 67 | }; |
65 | 68 | ||
66 | #define VIFF_TUNNEL 0x1 /* IPIP tunnel */ | 69 | #define VIFF_TUNNEL 0x1 /* IPIP tunnel */ |
67 | #define VIFF_SRCRT 0x2 /* NI */ | 70 | #define VIFF_SRCRT 0x2 /* NI */ |
68 | #define VIFF_REGISTER 0x4 /* register vif */ | 71 | #define VIFF_REGISTER 0x4 /* register vif */ |
72 | #define VIFF_USE_IFINDEX 0x8 /* use vifc_lcl_ifindex instead of | ||
73 | vifc_lcl_addr to find an interface */ | ||
69 | 74 | ||
70 | /* | 75 | /* |
71 | * Cache manipulation structures for mrouted and PIMd | 76 | * Cache manipulation structures for mrouted and PIMd |
diff --git a/net/ipv4/ipmr.c b/net/ipv4/ipmr.c index 630a56df7b47..c757f0b4b74c 100644 --- a/net/ipv4/ipmr.c +++ b/net/ipv4/ipmr.c | |||
@@ -469,8 +469,18 @@ static int vif_add(struct net *net, struct vifctl *vifc, int mrtsock) | |||
469 | return err; | 469 | return err; |
470 | } | 470 | } |
471 | break; | 471 | break; |
472 | |||
473 | case VIFF_USE_IFINDEX: | ||
472 | case 0: | 474 | case 0: |
473 | dev = ip_dev_find(net, vifc->vifc_lcl_addr.s_addr); | 475 | if (vifc->vifc_flags == VIFF_USE_IFINDEX) { |
476 | dev = dev_get_by_index(net, vifc->vifc_lcl_ifindex); | ||
477 | if (dev && dev->ip_ptr == NULL) { | ||
478 | dev_put(dev); | ||
479 | return -EADDRNOTAVAIL; | ||
480 | } | ||
481 | } else | ||
482 | dev = ip_dev_find(net, vifc->vifc_lcl_addr.s_addr); | ||
483 | |||
474 | if (!dev) | 484 | if (!dev) |
475 | return -EADDRNOTAVAIL; | 485 | return -EADDRNOTAVAIL; |
476 | err = dev_set_allmulti(dev, 1); | 486 | err = dev_set_allmulti(dev, 1); |