[NET]: Make the netlink methods in rtnetlink handle multiple network namespaces
After the previous prep work this just consists of removing checks limiting the code to work in the initial network namespace, and updating rtmsg_ifinfo so we can generate events for devices in something other then the initial network namespace. Referring to network other network devices like the IFLA_LINK and IFLA_MASTER attributes do, gets interesting if those network devices happen to be in other network namespaces. Currently ifindex numbers are allocated globally so I have taken the path of least resistance and not still report the information even though the devices they are talking about are invisible. If applications start getting confused or when ifindex numbers become local to the network namespace we may need to do something different in the future. Signed-off-by: Eric W. Biederman <ebiederm@xmission.com> Signed-off-by: Denis V. Lunev <den@openz.org> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
97c53cacf0
commit
4b3da706bb
@ -708,9 +708,6 @@ static int rtnl_dump_ifinfo(struct sk_buff *skb, struct netlink_callback *cb)
|
|||||||
int s_idx = cb->args[0];
|
int s_idx = cb->args[0];
|
||||||
struct net_device *dev;
|
struct net_device *dev;
|
||||||
|
|
||||||
if (net != &init_net)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
idx = 0;
|
idx = 0;
|
||||||
for_each_netdev(net, dev) {
|
for_each_netdev(net, dev) {
|
||||||
if (idx < s_idx)
|
if (idx < s_idx)
|
||||||
@ -913,9 +910,6 @@ static int rtnl_setlink(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)
|
|||||||
struct nlattr *tb[IFLA_MAX+1];
|
struct nlattr *tb[IFLA_MAX+1];
|
||||||
char ifname[IFNAMSIZ];
|
char ifname[IFNAMSIZ];
|
||||||
|
|
||||||
if (net != &init_net)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
err = nlmsg_parse(nlh, sizeof(*ifm), tb, IFLA_MAX, ifla_policy);
|
err = nlmsg_parse(nlh, sizeof(*ifm), tb, IFLA_MAX, ifla_policy);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
goto errout;
|
goto errout;
|
||||||
@ -964,9 +958,6 @@ static int rtnl_dellink(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)
|
|||||||
struct nlattr *tb[IFLA_MAX+1];
|
struct nlattr *tb[IFLA_MAX+1];
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (net != &init_net)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
err = nlmsg_parse(nlh, sizeof(*ifm), tb, IFLA_MAX, ifla_policy);
|
err = nlmsg_parse(nlh, sizeof(*ifm), tb, IFLA_MAX, ifla_policy);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
return err;
|
return err;
|
||||||
@ -1048,9 +1039,6 @@ static int rtnl_newlink(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)
|
|||||||
struct nlattr *linkinfo[IFLA_INFO_MAX+1];
|
struct nlattr *linkinfo[IFLA_INFO_MAX+1];
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (net != &init_net)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
#ifdef CONFIG_KMOD
|
#ifdef CONFIG_KMOD
|
||||||
replay:
|
replay:
|
||||||
#endif
|
#endif
|
||||||
@ -1177,9 +1165,6 @@ static int rtnl_getlink(struct sk_buff *skb, struct nlmsghdr* nlh, void *arg)
|
|||||||
struct sk_buff *nskb;
|
struct sk_buff *nskb;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (net != &init_net)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
err = nlmsg_parse(nlh, sizeof(*ifm), tb, IFLA_MAX, ifla_policy);
|
err = nlmsg_parse(nlh, sizeof(*ifm), tb, IFLA_MAX, ifla_policy);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
return err;
|
return err;
|
||||||
@ -1215,13 +1200,9 @@ errout:
|
|||||||
|
|
||||||
static int rtnl_dump_all(struct sk_buff *skb, struct netlink_callback *cb)
|
static int rtnl_dump_all(struct sk_buff *skb, struct netlink_callback *cb)
|
||||||
{
|
{
|
||||||
struct net *net = skb->sk->sk_net;
|
|
||||||
int idx;
|
int idx;
|
||||||
int s_idx = cb->family;
|
int s_idx = cb->family;
|
||||||
|
|
||||||
if (net != &init_net)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
if (s_idx == 0)
|
if (s_idx == 0)
|
||||||
s_idx = 1;
|
s_idx = 1;
|
||||||
for (idx=1; idx<NPROTO; idx++) {
|
for (idx=1; idx<NPROTO; idx++) {
|
||||||
@ -1243,6 +1224,7 @@ static int rtnl_dump_all(struct sk_buff *skb, struct netlink_callback *cb)
|
|||||||
|
|
||||||
void rtmsg_ifinfo(int type, struct net_device *dev, unsigned change)
|
void rtmsg_ifinfo(int type, struct net_device *dev, unsigned change)
|
||||||
{
|
{
|
||||||
|
struct net *net = dev->nd_net;
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
int err = -ENOBUFS;
|
int err = -ENOBUFS;
|
||||||
|
|
||||||
@ -1257,10 +1239,10 @@ void rtmsg_ifinfo(int type, struct net_device *dev, unsigned change)
|
|||||||
kfree_skb(skb);
|
kfree_skb(skb);
|
||||||
goto errout;
|
goto errout;
|
||||||
}
|
}
|
||||||
err = rtnl_notify(skb, &init_net, 0, RTNLGRP_LINK, NULL, GFP_KERNEL);
|
err = rtnl_notify(skb, net, 0, RTNLGRP_LINK, NULL, GFP_KERNEL);
|
||||||
errout:
|
errout:
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
rtnl_set_sk_err(&init_net, RTNLGRP_LINK, err);
|
rtnl_set_sk_err(net, RTNLGRP_LINK, err);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Protected by RTNL sempahore. */
|
/* Protected by RTNL sempahore. */
|
||||||
@ -1353,9 +1335,6 @@ static int rtnetlink_event(struct notifier_block *this, unsigned long event, voi
|
|||||||
{
|
{
|
||||||
struct net_device *dev = ptr;
|
struct net_device *dev = ptr;
|
||||||
|
|
||||||
if (dev->nd_net != &init_net)
|
|
||||||
return NOTIFY_DONE;
|
|
||||||
|
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case NETDEV_UNREGISTER:
|
case NETDEV_UNREGISTER:
|
||||||
rtmsg_ifinfo(RTM_DELLINK, dev, ~0U);
|
rtmsg_ifinfo(RTM_DELLINK, dev, ~0U);
|
||||||
|
Loading…
Reference in New Issue
Block a user