[NETNS][IPV6] addrconf - Pass the proper network namespace parameters to addrconf

This patch propagates the network namespace pointer to the address
configuration routines which need it, which means adding a new
parameter to these functions, and make them use it instead of using
the initial network namespace.

Signed-off-by: Daniel Lezcano <dlezcano@fr.ibm.com>
Signed-off-by: Benjamin Thery <benjamin.thery@bull.net>
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
Daniel Lezcano 2008-03-05 10:46:57 -08:00 committed by David S. Miller
parent 300bf591de
commit af2849377e
3 changed files with 57 additions and 24 deletions

View File

@ -55,9 +55,12 @@ struct prefix_info {
extern int addrconf_init(void); extern int addrconf_init(void);
extern void addrconf_cleanup(void); extern void addrconf_cleanup(void);
extern int addrconf_add_ifaddr(void __user *arg); extern int addrconf_add_ifaddr(struct net *net,
extern int addrconf_del_ifaddr(void __user *arg); void __user *arg);
extern int addrconf_set_dstaddr(void __user *arg); extern int addrconf_del_ifaddr(struct net *net,
void __user *arg);
extern int addrconf_set_dstaddr(struct net *net,
void __user *arg);
extern int ipv6_chk_addr(struct net *net, extern int ipv6_chk_addr(struct net *net,
struct in6_addr *addr, struct in6_addr *addr,

View File

@ -1866,7 +1866,7 @@ void addrconf_prefix_rcv(struct net_device *dev, u8 *opt, int len)
* Special case for SIT interfaces where we create a new "virtual" * Special case for SIT interfaces where we create a new "virtual"
* device. * device.
*/ */
int addrconf_set_dstaddr(void __user *arg) int addrconf_set_dstaddr(struct net *net, void __user *arg)
{ {
struct in6_ifreq ireq; struct in6_ifreq ireq;
struct net_device *dev; struct net_device *dev;
@ -1878,7 +1878,7 @@ int addrconf_set_dstaddr(void __user *arg)
if (copy_from_user(&ireq, arg, sizeof(struct in6_ifreq))) if (copy_from_user(&ireq, arg, sizeof(struct in6_ifreq)))
goto err_exit; goto err_exit;
dev = __dev_get_by_index(&init_net, ireq.ifr6_ifindex); dev = __dev_get_by_index(net, ireq.ifr6_ifindex);
err = -ENODEV; err = -ENODEV;
if (dev == NULL) if (dev == NULL)
@ -1909,7 +1909,8 @@ int addrconf_set_dstaddr(void __user *arg)
if (err == 0) { if (err == 0) {
err = -ENOBUFS; err = -ENOBUFS;
if ((dev = __dev_get_by_name(&init_net, p.name)) == NULL) dev = __dev_get_by_name(net, p.name);
if (!dev)
goto err_exit; goto err_exit;
err = dev_open(dev); err = dev_open(dev);
} }
@ -1924,8 +1925,9 @@ int addrconf_set_dstaddr(void __user *arg)
/* /*
* Manual configuration of address on an interface * Manual configuration of address on an interface
*/ */
static int inet6_addr_add(int ifindex, struct in6_addr *pfx, int plen, static int inet6_addr_add(struct net *net, int ifindex, struct in6_addr *pfx,
__u8 ifa_flags, __u32 prefered_lft, __u32 valid_lft) int plen, __u8 ifa_flags, __u32 prefered_lft,
__u32 valid_lft)
{ {
struct inet6_ifaddr *ifp; struct inet6_ifaddr *ifp;
struct inet6_dev *idev; struct inet6_dev *idev;
@ -1939,7 +1941,8 @@ static int inet6_addr_add(int ifindex, struct in6_addr *pfx, int plen,
if (!valid_lft || prefered_lft > valid_lft) if (!valid_lft || prefered_lft > valid_lft)
return -EINVAL; return -EINVAL;
if ((dev = __dev_get_by_index(&init_net, ifindex)) == NULL) dev = __dev_get_by_index(net, ifindex);
if (!dev)
return -ENODEV; return -ENODEV;
if ((idev = addrconf_add_dev(dev)) == NULL) if ((idev = addrconf_add_dev(dev)) == NULL)
@ -1984,13 +1987,15 @@ static int inet6_addr_add(int ifindex, struct in6_addr *pfx, int plen,
return PTR_ERR(ifp); return PTR_ERR(ifp);
} }
static int inet6_addr_del(int ifindex, struct in6_addr *pfx, int plen) static int inet6_addr_del(struct net *net, int ifindex, struct in6_addr *pfx,
int plen)
{ {
struct inet6_ifaddr *ifp; struct inet6_ifaddr *ifp;
struct inet6_dev *idev; struct inet6_dev *idev;
struct net_device *dev; struct net_device *dev;
if ((dev = __dev_get_by_index(&init_net, ifindex)) == NULL) dev = __dev_get_by_index(net, ifindex);
if (!dev)
return -ENODEV; return -ENODEV;
if ((idev = __in6_dev_get(dev)) == NULL) if ((idev = __in6_dev_get(dev)) == NULL)
@ -2018,7 +2023,7 @@ static int inet6_addr_del(int ifindex, struct in6_addr *pfx, int plen)
} }
int addrconf_add_ifaddr(void __user *arg) int addrconf_add_ifaddr(struct net *net, void __user *arg)
{ {
struct in6_ifreq ireq; struct in6_ifreq ireq;
int err; int err;
@ -2030,13 +2035,14 @@ int addrconf_add_ifaddr(void __user *arg)
return -EFAULT; return -EFAULT;
rtnl_lock(); rtnl_lock();
err = inet6_addr_add(ireq.ifr6_ifindex, &ireq.ifr6_addr, ireq.ifr6_prefixlen, err = inet6_addr_add(net, ireq.ifr6_ifindex, &ireq.ifr6_addr,
IFA_F_PERMANENT, INFINITY_LIFE_TIME, INFINITY_LIFE_TIME); ireq.ifr6_prefixlen, IFA_F_PERMANENT,
INFINITY_LIFE_TIME, INFINITY_LIFE_TIME);
rtnl_unlock(); rtnl_unlock();
return err; return err;
} }
int addrconf_del_ifaddr(void __user *arg) int addrconf_del_ifaddr(struct net *net, void __user *arg)
{ {
struct in6_ifreq ireq; struct in6_ifreq ireq;
int err; int err;
@ -2048,7 +2054,8 @@ int addrconf_del_ifaddr(void __user *arg)
return -EFAULT; return -EFAULT;
rtnl_lock(); rtnl_lock();
err = inet6_addr_del(ireq.ifr6_ifindex, &ireq.ifr6_addr, ireq.ifr6_prefixlen); err = inet6_addr_del(net, ireq.ifr6_ifindex, &ireq.ifr6_addr,
ireq.ifr6_prefixlen);
rtnl_unlock(); rtnl_unlock();
return err; return err;
} }
@ -3061,7 +3068,7 @@ inet6_rtm_deladdr(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)
if (pfx == NULL) if (pfx == NULL)
return -EINVAL; return -EINVAL;
return inet6_addr_del(ifm->ifa_index, pfx, ifm->ifa_prefixlen); return inet6_addr_del(net, ifm->ifa_index, pfx, ifm->ifa_prefixlen);
} }
static int inet6_addr_modify(struct inet6_ifaddr *ifp, u8 ifa_flags, static int inet6_addr_modify(struct inet6_ifaddr *ifp, u8 ifa_flags,
@ -3137,7 +3144,7 @@ inet6_rtm_newaddr(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)
valid_lft = INFINITY_LIFE_TIME; valid_lft = INFINITY_LIFE_TIME;
} }
dev = __dev_get_by_index(&init_net, ifm->ifa_index); dev = __dev_get_by_index(net, ifm->ifa_index);
if (dev == NULL) if (dev == NULL)
return -ENODEV; return -ENODEV;
@ -3150,8 +3157,9 @@ inet6_rtm_newaddr(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)
* It would be best to check for !NLM_F_CREATE here but * It would be best to check for !NLM_F_CREATE here but
* userspace alreay relies on not having to provide this. * userspace alreay relies on not having to provide this.
*/ */
return inet6_addr_add(ifm->ifa_index, pfx, ifm->ifa_prefixlen, return inet6_addr_add(net, ifm->ifa_index, pfx,
ifa_flags, preferred_lft, valid_lft); ifm->ifa_prefixlen, ifa_flags,
preferred_lft, valid_lft);
} }
if (nlh->nlmsg_flags & NLM_F_EXCL || if (nlh->nlmsg_flags & NLM_F_EXCL ||
@ -4260,6 +4268,22 @@ int unregister_inet6addr_notifier(struct notifier_block *nb)
EXPORT_SYMBOL(unregister_inet6addr_notifier); EXPORT_SYMBOL(unregister_inet6addr_notifier);
static int addrconf_net_init(struct net *net)
{
return 0;
}
static void addrconf_net_exit(struct net *net)
{
;
}
static struct pernet_operations addrconf_net_ops = {
.init = addrconf_net_init,
.exit = addrconf_net_exit,
};
/* /*
* Init / cleanup code * Init / cleanup code
*/ */
@ -4301,6 +4325,10 @@ int __init addrconf_init(void)
if (err) if (err)
goto errlo; goto errlo;
err = register_pernet_device(&addrconf_net_ops);
if (err)
return err;
register_netdevice_notifier(&ipv6_dev_notf); register_netdevice_notifier(&ipv6_dev_notf);
addrconf_verify(0); addrconf_verify(0);
@ -4334,6 +4362,7 @@ void addrconf_cleanup(void)
int i; int i;
unregister_netdevice_notifier(&ipv6_dev_notf); unregister_netdevice_notifier(&ipv6_dev_notf);
unregister_pernet_device(&addrconf_net_ops);
unregister_pernet_subsys(&addrconf_ops); unregister_pernet_subsys(&addrconf_ops);
@ -4370,6 +4399,7 @@ void addrconf_cleanup(void)
write_unlock_bh(&addrconf_hash_lock); write_unlock_bh(&addrconf_hash_lock);
del_timer(&addr_chk_timer); del_timer(&addr_chk_timer);
rtnl_unlock(); rtnl_unlock();
unregister_pernet_subsys(&addrconf_net_ops);
} }

View File

@ -454,11 +454,11 @@ int inet6_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg)
return(ipv6_route_ioctl(net, cmd, (void __user *)arg)); return(ipv6_route_ioctl(net, cmd, (void __user *)arg));
case SIOCSIFADDR: case SIOCSIFADDR:
return addrconf_add_ifaddr((void __user *) arg); return addrconf_add_ifaddr(net, (void __user *) arg);
case SIOCDIFADDR: case SIOCDIFADDR:
return addrconf_del_ifaddr((void __user *) arg); return addrconf_del_ifaddr(net, (void __user *) arg);
case SIOCSIFDSTADDR: case SIOCSIFDSTADDR:
return addrconf_set_dstaddr((void __user *) arg); return addrconf_set_dstaddr(net, (void __user *) arg);
default: default:
if (!sk->sk_prot->ioctl) if (!sk->sk_prot->ioctl)
return -ENOIOCTLCMD; return -ENOIOCTLCMD;