diff options
Diffstat (limited to 'net/core')
| -rw-r--r-- | net/core/dev.c | 18 | ||||
| -rw-r--r-- | net/core/neighbour.c | 2 | ||||
| -rw-r--r-- | net/core/rtnetlink.c | 274 | ||||
| -rw-r--r-- | net/core/skbuff.c | 3 | 
4 files changed, 168 insertions, 129 deletions
| diff --git a/net/core/dev.c b/net/core/dev.c index ab9b8d0d115e..ae00b894e675 100644 --- a/net/core/dev.c +++ b/net/core/dev.c @@ -2403,17 +2403,20 @@ static void skb_warn_bad_offload(const struct sk_buff *skb)  {  	static const netdev_features_t null_features = 0;  	struct net_device *dev = skb->dev; -	const char *driver = ""; +	const char *name = "";  	if (!net_ratelimit())  		return; -	if (dev && dev->dev.parent) -		driver = dev_driver_string(dev->dev.parent); - +	if (dev) { +		if (dev->dev.parent) +			name = dev_driver_string(dev->dev.parent); +		else +			name = netdev_name(dev); +	}  	WARN(1, "%s: caps=(%pNF, %pNF) len=%d data_len=%d gso_size=%d "  	     "gso_type=%d ip_summed=%d\n", -	     driver, dev ? &dev->features : &null_features, +	     name, dev ? &dev->features : &null_features,  	     skb->sk ? &skb->sk->sk_route_caps : &null_features,  	     skb->len, skb->data_len, skb_shinfo(skb)->gso_size,  	     skb_shinfo(skb)->gso_type, skb->ip_summed); @@ -6426,11 +6429,16 @@ int __netdev_update_features(struct net_device *dev)  	if (dev->netdev_ops->ndo_set_features)  		err = dev->netdev_ops->ndo_set_features(dev, features); +	else +		err = 0;  	if (unlikely(err < 0)) {  		netdev_err(dev,  			"set_features() failed (%d); wanted %pNF, left %pNF\n",  			err, &features, &dev->features); +		/* return non-0 since some features might have changed and +		 * it's better to fire a spurious notification than miss it +		 */  		return -1;  	} diff --git a/net/core/neighbour.c b/net/core/neighbour.c index 1aa8437ed6c4..e6af42da28d9 100644 --- a/net/core/neighbour.c +++ b/net/core/neighbour.c @@ -857,7 +857,7 @@ static void neigh_probe(struct neighbour *neigh)  	struct sk_buff *skb = skb_peek_tail(&neigh->arp_queue);  	/* keep skb alive even if arp_queue overflows */  	if (skb) -		skb = skb_copy(skb, GFP_ATOMIC); +		skb = skb_clone(skb, GFP_ATOMIC);  	write_unlock(&neigh->lock);  	neigh->ops->solicit(neigh, skb);  	atomic_inc(&neigh->probes); diff --git a/net/core/rtnetlink.c b/net/core/rtnetlink.c index 504bd17b7456..34ba7a08876d 100644 --- a/net/core/rtnetlink.c +++ b/net/core/rtnetlink.c @@ -1045,15 +1045,156 @@ static int rtnl_phys_switch_id_fill(struct sk_buff *skb, struct net_device *dev)  	return 0;  } +static noinline_for_stack int rtnl_fill_stats(struct sk_buff *skb, +					      struct net_device *dev) +{ +	const struct rtnl_link_stats64 *stats; +	struct rtnl_link_stats64 temp; +	struct nlattr *attr; + +	stats = dev_get_stats(dev, &temp); + +	attr = nla_reserve(skb, IFLA_STATS, +			   sizeof(struct rtnl_link_stats)); +	if (!attr) +		return -EMSGSIZE; + +	copy_rtnl_link_stats(nla_data(attr), stats); + +	attr = nla_reserve(skb, IFLA_STATS64, +			   sizeof(struct rtnl_link_stats64)); +	if (!attr) +		return -EMSGSIZE; + +	copy_rtnl_link_stats64(nla_data(attr), stats); + +	return 0; +} + +static noinline_for_stack int rtnl_fill_vfinfo(struct sk_buff *skb, +					       struct net_device *dev, +					       int vfs_num, +					       struct nlattr *vfinfo) +{ +	struct ifla_vf_rss_query_en vf_rss_query_en; +	struct ifla_vf_link_state vf_linkstate; +	struct ifla_vf_spoofchk vf_spoofchk; +	struct ifla_vf_tx_rate vf_tx_rate; +	struct ifla_vf_stats vf_stats; +	struct ifla_vf_trust vf_trust; +	struct ifla_vf_vlan vf_vlan; +	struct ifla_vf_rate vf_rate; +	struct nlattr *vf, *vfstats; +	struct ifla_vf_mac vf_mac; +	struct ifla_vf_info ivi; + +	/* Not all SR-IOV capable drivers support the +	 * spoofcheck and "RSS query enable" query.  Preset to +	 * -1 so the user space tool can detect that the driver +	 * didn't report anything. +	 */ +	ivi.spoofchk = -1; +	ivi.rss_query_en = -1; +	ivi.trusted = -1; +	memset(ivi.mac, 0, sizeof(ivi.mac)); +	/* The default value for VF link state is "auto" +	 * IFLA_VF_LINK_STATE_AUTO which equals zero +	 */ +	ivi.linkstate = 0; +	if (dev->netdev_ops->ndo_get_vf_config(dev, vfs_num, &ivi)) +		return 0; + +	vf_mac.vf = +		vf_vlan.vf = +		vf_rate.vf = +		vf_tx_rate.vf = +		vf_spoofchk.vf = +		vf_linkstate.vf = +		vf_rss_query_en.vf = +		vf_trust.vf = ivi.vf; + +	memcpy(vf_mac.mac, ivi.mac, sizeof(ivi.mac)); +	vf_vlan.vlan = ivi.vlan; +	vf_vlan.qos = ivi.qos; +	vf_tx_rate.rate = ivi.max_tx_rate; +	vf_rate.min_tx_rate = ivi.min_tx_rate; +	vf_rate.max_tx_rate = ivi.max_tx_rate; +	vf_spoofchk.setting = ivi.spoofchk; +	vf_linkstate.link_state = ivi.linkstate; +	vf_rss_query_en.setting = ivi.rss_query_en; +	vf_trust.setting = ivi.trusted; +	vf = nla_nest_start(skb, IFLA_VF_INFO); +	if (!vf) { +		nla_nest_cancel(skb, vfinfo); +		return -EMSGSIZE; +	} +	if (nla_put(skb, IFLA_VF_MAC, sizeof(vf_mac), &vf_mac) || +	    nla_put(skb, IFLA_VF_VLAN, sizeof(vf_vlan), &vf_vlan) || +	    nla_put(skb, IFLA_VF_RATE, sizeof(vf_rate), +		    &vf_rate) || +	    nla_put(skb, IFLA_VF_TX_RATE, sizeof(vf_tx_rate), +		    &vf_tx_rate) || +	    nla_put(skb, IFLA_VF_SPOOFCHK, sizeof(vf_spoofchk), +		    &vf_spoofchk) || +	    nla_put(skb, IFLA_VF_LINK_STATE, sizeof(vf_linkstate), +		    &vf_linkstate) || +	    nla_put(skb, IFLA_VF_RSS_QUERY_EN, +		    sizeof(vf_rss_query_en), +		    &vf_rss_query_en) || +	    nla_put(skb, IFLA_VF_TRUST, +		    sizeof(vf_trust), &vf_trust)) +		return -EMSGSIZE; +	memset(&vf_stats, 0, sizeof(vf_stats)); +	if (dev->netdev_ops->ndo_get_vf_stats) +		dev->netdev_ops->ndo_get_vf_stats(dev, vfs_num, +						&vf_stats); +	vfstats = nla_nest_start(skb, IFLA_VF_STATS); +	if (!vfstats) { +		nla_nest_cancel(skb, vf); +		nla_nest_cancel(skb, vfinfo); +		return -EMSGSIZE; +	} +	if (nla_put_u64(skb, IFLA_VF_STATS_RX_PACKETS, +			vf_stats.rx_packets) || +	    nla_put_u64(skb, IFLA_VF_STATS_TX_PACKETS, +			vf_stats.tx_packets) || +	    nla_put_u64(skb, IFLA_VF_STATS_RX_BYTES, +			vf_stats.rx_bytes) || +	    nla_put_u64(skb, IFLA_VF_STATS_TX_BYTES, +			vf_stats.tx_bytes) || +	    nla_put_u64(skb, IFLA_VF_STATS_BROADCAST, +			vf_stats.broadcast) || +	    nla_put_u64(skb, IFLA_VF_STATS_MULTICAST, +			vf_stats.multicast)) +		return -EMSGSIZE; +	nla_nest_end(skb, vfstats); +	nla_nest_end(skb, vf); +	return 0; +} + +static int rtnl_fill_link_ifmap(struct sk_buff *skb, struct net_device *dev) +{ +	struct rtnl_link_ifmap map = { +		.mem_start   = dev->mem_start, +		.mem_end     = dev->mem_end, +		.base_addr   = dev->base_addr, +		.irq         = dev->irq, +		.dma         = dev->dma, +		.port        = dev->if_port, +	}; +	if (nla_put(skb, IFLA_MAP, sizeof(map), &map)) +		return -EMSGSIZE; + +	return 0; +} +  static int rtnl_fill_ifinfo(struct sk_buff *skb, struct net_device *dev,  			    int type, u32 pid, u32 seq, u32 change,  			    unsigned int flags, u32 ext_filter_mask)  {  	struct ifinfomsg *ifm;  	struct nlmsghdr *nlh; -	struct rtnl_link_stats64 temp; -	const struct rtnl_link_stats64 *stats; -	struct nlattr *attr, *af_spec; +	struct nlattr *af_spec;  	struct rtnl_af_ops *af_ops;  	struct net_device *upper_dev = netdev_master_upper_dev_get(dev); @@ -1096,18 +1237,8 @@ static int rtnl_fill_ifinfo(struct sk_buff *skb, struct net_device *dev,  	    nla_put_u8(skb, IFLA_PROTO_DOWN, dev->proto_down))  		goto nla_put_failure; -	if (1) { -		struct rtnl_link_ifmap map = { -			.mem_start   = dev->mem_start, -			.mem_end     = dev->mem_end, -			.base_addr   = dev->base_addr, -			.irq         = dev->irq, -			.dma         = dev->dma, -			.port        = dev->if_port, -		}; -		if (nla_put(skb, IFLA_MAP, sizeof(map), &map)) -			goto nla_put_failure; -	} +	if (rtnl_fill_link_ifmap(skb, dev)) +		goto nla_put_failure;  	if (dev->addr_len) {  		if (nla_put(skb, IFLA_ADDRESS, dev->addr_len, dev->dev_addr) || @@ -1124,128 +1255,27 @@ static int rtnl_fill_ifinfo(struct sk_buff *skb, struct net_device *dev,  	if (rtnl_phys_switch_id_fill(skb, dev))  		goto nla_put_failure; -	attr = nla_reserve(skb, IFLA_STATS, -			sizeof(struct rtnl_link_stats)); -	if (attr == NULL) -		goto nla_put_failure; - -	stats = dev_get_stats(dev, &temp); -	copy_rtnl_link_stats(nla_data(attr), stats); - -	attr = nla_reserve(skb, IFLA_STATS64, -			sizeof(struct rtnl_link_stats64)); -	if (attr == NULL) +	if (rtnl_fill_stats(skb, dev))  		goto nla_put_failure; -	copy_rtnl_link_stats64(nla_data(attr), stats);  	if (dev->dev.parent && (ext_filter_mask & RTEXT_FILTER_VF) &&  	    nla_put_u32(skb, IFLA_NUM_VF, dev_num_vf(dev->dev.parent)))  		goto nla_put_failure; -	if (dev->netdev_ops->ndo_get_vf_config && dev->dev.parent -	    && (ext_filter_mask & RTEXT_FILTER_VF)) { +	if (dev->netdev_ops->ndo_get_vf_config && dev->dev.parent && +	    ext_filter_mask & RTEXT_FILTER_VF) {  		int i; - -		struct nlattr *vfinfo, *vf, *vfstats; +		struct nlattr *vfinfo;  		int num_vfs = dev_num_vf(dev->dev.parent);  		vfinfo = nla_nest_start(skb, IFLA_VFINFO_LIST);  		if (!vfinfo)  			goto nla_put_failure;  		for (i = 0; i < num_vfs; i++) { -			struct ifla_vf_info ivi; -			struct ifla_vf_mac vf_mac; -			struct ifla_vf_vlan vf_vlan; -			struct ifla_vf_rate vf_rate; -			struct ifla_vf_tx_rate vf_tx_rate; -			struct ifla_vf_spoofchk vf_spoofchk; -			struct ifla_vf_link_state vf_linkstate; -			struct ifla_vf_rss_query_en vf_rss_query_en; -			struct ifla_vf_stats vf_stats; -			struct ifla_vf_trust vf_trust; - -			/* -			 * Not all SR-IOV capable drivers support the -			 * spoofcheck and "RSS query enable" query.  Preset to -			 * -1 so the user space tool can detect that the driver -			 * didn't report anything. -			 */ -			ivi.spoofchk = -1; -			ivi.rss_query_en = -1; -			ivi.trusted = -1; -			memset(ivi.mac, 0, sizeof(ivi.mac)); -			/* The default value for VF link state is "auto" -			 * IFLA_VF_LINK_STATE_AUTO which equals zero -			 */ -			ivi.linkstate = 0; -			if (dev->netdev_ops->ndo_get_vf_config(dev, i, &ivi)) -				break; -			vf_mac.vf = -				vf_vlan.vf = -				vf_rate.vf = -				vf_tx_rate.vf = -				vf_spoofchk.vf = -				vf_linkstate.vf = -				vf_rss_query_en.vf = -				vf_trust.vf = ivi.vf; - -			memcpy(vf_mac.mac, ivi.mac, sizeof(ivi.mac)); -			vf_vlan.vlan = ivi.vlan; -			vf_vlan.qos = ivi.qos; -			vf_tx_rate.rate = ivi.max_tx_rate; -			vf_rate.min_tx_rate = ivi.min_tx_rate; -			vf_rate.max_tx_rate = ivi.max_tx_rate; -			vf_spoofchk.setting = ivi.spoofchk; -			vf_linkstate.link_state = ivi.linkstate; -			vf_rss_query_en.setting = ivi.rss_query_en; -			vf_trust.setting = ivi.trusted; -			vf = nla_nest_start(skb, IFLA_VF_INFO); -			if (!vf) { -				nla_nest_cancel(skb, vfinfo); -				goto nla_put_failure; -			} -			if (nla_put(skb, IFLA_VF_MAC, sizeof(vf_mac), &vf_mac) || -			    nla_put(skb, IFLA_VF_VLAN, sizeof(vf_vlan), &vf_vlan) || -			    nla_put(skb, IFLA_VF_RATE, sizeof(vf_rate), -				    &vf_rate) || -			    nla_put(skb, IFLA_VF_TX_RATE, sizeof(vf_tx_rate), -				    &vf_tx_rate) || -			    nla_put(skb, IFLA_VF_SPOOFCHK, sizeof(vf_spoofchk), -				    &vf_spoofchk) || -			    nla_put(skb, IFLA_VF_LINK_STATE, sizeof(vf_linkstate), -				    &vf_linkstate) || -			    nla_put(skb, IFLA_VF_RSS_QUERY_EN, -				    sizeof(vf_rss_query_en), -				    &vf_rss_query_en) || -			    nla_put(skb, IFLA_VF_TRUST, -				    sizeof(vf_trust), &vf_trust)) +			if (rtnl_fill_vfinfo(skb, dev, i, vfinfo))  				goto nla_put_failure; -			memset(&vf_stats, 0, sizeof(vf_stats)); -			if (dev->netdev_ops->ndo_get_vf_stats) -				dev->netdev_ops->ndo_get_vf_stats(dev, i, -								  &vf_stats); -			vfstats = nla_nest_start(skb, IFLA_VF_STATS); -			if (!vfstats) { -				nla_nest_cancel(skb, vf); -				nla_nest_cancel(skb, vfinfo); -				goto nla_put_failure; -			} -			if (nla_put_u64(skb, IFLA_VF_STATS_RX_PACKETS, -					vf_stats.rx_packets) || -			    nla_put_u64(skb, IFLA_VF_STATS_TX_PACKETS, -					vf_stats.tx_packets) || -			    nla_put_u64(skb, IFLA_VF_STATS_RX_BYTES, -					vf_stats.rx_bytes) || -			    nla_put_u64(skb, IFLA_VF_STATS_TX_BYTES, -					vf_stats.tx_bytes) || -			    nla_put_u64(skb, IFLA_VF_STATS_BROADCAST, -					vf_stats.broadcast) || -			    nla_put_u64(skb, IFLA_VF_STATS_MULTICAST, -					vf_stats.multicast)) -				goto nla_put_failure; -			nla_nest_end(skb, vfstats); -			nla_nest_end(skb, vf);  		} +  		nla_nest_end(skb, vfinfo);  	} diff --git a/net/core/skbuff.c b/net/core/skbuff.c index aa41e6dd6429..152b9c70e252 100644 --- a/net/core/skbuff.c +++ b/net/core/skbuff.c @@ -4268,7 +4268,8 @@ static struct sk_buff *skb_reorder_vlan_header(struct sk_buff *skb)  		return NULL;  	} -	memmove(skb->data - ETH_HLEN, skb->data - VLAN_ETH_HLEN, 2 * ETH_ALEN); +	memmove(skb->data - ETH_HLEN, skb->data - skb->mac_len, +		2 * ETH_ALEN);  	skb->mac_header += VLAN_HLEN;  	return skb;  } | 
