diff options
Diffstat (limited to 'net/ipv4/devinet.c')
| -rw-r--r-- | net/ipv4/devinet.c | 53 | 
1 files changed, 23 insertions, 30 deletions
| diff --git a/net/ipv4/devinet.c b/net/ipv4/devinet.c index d96f3e452fef..ab76744383cf 100644 --- a/net/ipv4/devinet.c +++ b/net/ipv4/devinet.c @@ -216,17 +216,27 @@ static void devinet_sysctl_unregister(struct in_device *idev)  /* Locks all the inet devices. */ -static struct in_ifaddr *inet_alloc_ifa(void) +static struct in_ifaddr *inet_alloc_ifa(struct in_device *in_dev)  { -	return kzalloc(sizeof(struct in_ifaddr), GFP_KERNEL_ACCOUNT); +	struct in_ifaddr *ifa; + +	ifa = kzalloc(sizeof(*ifa), GFP_KERNEL_ACCOUNT); +	if (!ifa) +		return NULL; + +	in_dev_hold(in_dev); +	ifa->ifa_dev = in_dev; + +	INIT_HLIST_NODE(&ifa->hash); + +	return ifa;  }  static void inet_rcu_free_ifa(struct rcu_head *head)  {  	struct in_ifaddr *ifa = container_of(head, struct in_ifaddr, rcu_head); -	if (ifa->ifa_dev) -		in_dev_put(ifa->ifa_dev); +	in_dev_put(ifa->ifa_dev);  	kfree(ifa);  } @@ -574,17 +584,9 @@ static int inet_set_ifa(struct net_device *dev, struct in_ifaddr *ifa)  	ASSERT_RTNL(); -	if (!in_dev) { -		inet_free_ifa(ifa); -		return -ENOBUFS; -	}  	ipv4_devconf_setall(in_dev);  	neigh_parms_data_state_setall(in_dev->arp_parms); -	if (ifa->ifa_dev != in_dev) { -		WARN_ON(ifa->ifa_dev); -		in_dev_hold(in_dev); -		ifa->ifa_dev = in_dev; -	} +  	if (ipv4_is_loopback(ifa->ifa_local))  		ifa->ifa_scope = RT_SCOPE_HOST;  	return inet_insert_ifa(ifa); @@ -701,8 +703,6 @@ errout:  	return err;  } -#define INFINITY_LIFE_TIME	0xFFFFFFFF -  static void check_lifetime(struct work_struct *work)  {  	unsigned long now, next, next_sec, next_sched; @@ -875,7 +875,7 @@ static struct in_ifaddr *rtm_to_ifaddr(struct net *net, struct nlmsghdr *nlh,  	if (!in_dev)  		goto errout; -	ifa = inet_alloc_ifa(); +	ifa = inet_alloc_ifa(in_dev);  	if (!ifa)  		/*  		 * A potential indev allocation can be left alive, it stays @@ -885,19 +885,15 @@ static struct in_ifaddr *rtm_to_ifaddr(struct net *net, struct nlmsghdr *nlh,  	ipv4_devconf_setall(in_dev);  	neigh_parms_data_state_setall(in_dev->arp_parms); -	in_dev_hold(in_dev);  	if (!tb[IFA_ADDRESS])  		tb[IFA_ADDRESS] = tb[IFA_LOCAL]; -	INIT_HLIST_NODE(&ifa->hash);  	ifa->ifa_prefixlen = ifm->ifa_prefixlen;  	ifa->ifa_mask = inet_make_mask(ifm->ifa_prefixlen);  	ifa->ifa_flags = tb[IFA_FLAGS] ? nla_get_u32(tb[IFA_FLAGS]) :  					 ifm->ifa_flags;  	ifa->ifa_scope = ifm->ifa_scope; -	ifa->ifa_dev = in_dev; -  	ifa->ifa_local = nla_get_in_addr(tb[IFA_LOCAL]);  	ifa->ifa_address = nla_get_in_addr(tb[IFA_ADDRESS]); @@ -1184,10 +1180,12 @@ int devinet_ioctl(struct net *net, unsigned int cmd, struct ifreq *ifr)  		if (!ifa) {  			ret = -ENOBUFS; -			ifa = inet_alloc_ifa(); +			if (!in_dev) +				break; +			ifa = inet_alloc_ifa(in_dev);  			if (!ifa)  				break; -			INIT_HLIST_NODE(&ifa->hash); +  			if (colon)  				memcpy(ifa->ifa_label, ifr->ifr_name, IFNAMSIZ);  			else @@ -1586,16 +1584,13 @@ static int inetdev_event(struct notifier_block *this, unsigned long event,  		if (!inetdev_valid_mtu(dev->mtu))  			break;  		if (dev->flags & IFF_LOOPBACK) { -			struct in_ifaddr *ifa = inet_alloc_ifa(); +			struct in_ifaddr *ifa = inet_alloc_ifa(in_dev);  			if (ifa) { -				INIT_HLIST_NODE(&ifa->hash);  				ifa->ifa_local =  				  ifa->ifa_address = htonl(INADDR_LOOPBACK);  				ifa->ifa_prefixlen = 8;  				ifa->ifa_mask = inet_make_mask(8); -				in_dev_hold(in_dev); -				ifa->ifa_dev = in_dev;  				ifa->ifa_scope = RT_SCOPE_HOST;  				memcpy(ifa->ifa_label, dev->name, IFNAMSIZ);  				set_ifa_lifetime(ifa, INFINITY_LIFE_TIME, @@ -1948,8 +1943,7 @@ static void rtmsg_ifa(int event, struct in_ifaddr *ifa, struct nlmsghdr *nlh,  	rtnl_notify(skb, net, portid, RTNLGRP_IPV4_IFADDR, nlh, GFP_KERNEL);  	return;  errout: -	if (err < 0) -		rtnl_set_sk_err(net, RTNLGRP_IPV4_IFADDR, err); +	rtnl_set_sk_err(net, RTNLGRP_IPV4_IFADDR, err);  }  static size_t inet_get_link_af_size(const struct net_device *dev, @@ -2145,8 +2139,7 @@ void inet_netconf_notify_devconf(struct net *net, int event, int type,  	rtnl_notify(skb, net, 0, RTNLGRP_IPV4_NETCONF, NULL, GFP_KERNEL);  	return;  errout: -	if (err < 0) -		rtnl_set_sk_err(net, RTNLGRP_IPV4_NETCONF, err); +	rtnl_set_sk_err(net, RTNLGRP_IPV4_NETCONF, err);  }  static const struct nla_policy devconf_ipv4_policy[NETCONFA_MAX+1] = { | 
