ethtool.c 47.7 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5
/*
 * net/core/ethtool.c - Ethtool ioctl handler
 * Copyright (c) 2003 Matthew Wilcox <matthew@wil.cx>
 *
 * This file is where we call all the ethtool_ops commands to get
6
 * the information ethtool needs.
L
Linus Torvalds 已提交
7
 *
8 9 10 11
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
L
Linus Torvalds 已提交
12 13 14 15
 */

#include <linux/module.h>
#include <linux/types.h>
16
#include <linux/capability.h>
L
Linus Torvalds 已提交
17 18 19
#include <linux/errno.h>
#include <linux/ethtool.h>
#include <linux/netdevice.h>
20 21
#include <linux/net_tstamp.h>
#include <linux/phy.h>
22
#include <linux/bitops.h>
23
#include <linux/uaccess.h>
24
#include <linux/vmalloc.h>
25
#include <linux/slab.h>
26 27
#include <linux/rtnetlink.h>
#include <linux/sched.h>
28
#include <linux/net.h>
L
Linus Torvalds 已提交
29

30
/*
L
Linus Torvalds 已提交
31 32 33 34 35 36 37 38 39
 * Some useful ethtool_ops methods that're device independent.
 * If we find that all drivers want to do the same thing here,
 * we can turn these into dev_() function calls.
 */

u32 ethtool_op_get_link(struct net_device *dev)
{
	return netif_carrier_ok(dev) ? 1 : 0;
}
40
EXPORT_SYMBOL(ethtool_op_get_link);
L
Linus Torvalds 已提交
41

42 43 44 45 46 47 48 49 50 51 52
int ethtool_op_get_ts_info(struct net_device *dev, struct ethtool_ts_info *info)
{
	info->so_timestamping =
		SOF_TIMESTAMPING_TX_SOFTWARE |
		SOF_TIMESTAMPING_RX_SOFTWARE |
		SOF_TIMESTAMPING_SOFTWARE;
	info->phc_index = -1;
	return 0;
}
EXPORT_SYMBOL(ethtool_op_get_ts_info);

L
Linus Torvalds 已提交
53 54
/* Handlers for each ethtool command */

55
#define ETHTOOL_DEV_FEATURE_WORDS	((NETDEV_FEATURE_COUNT + 31) / 32)
56

57 58 59 60 61 62 63
static const char netdev_features_strings[NETDEV_FEATURE_COUNT][ETH_GSTRING_LEN] = {
	[NETIF_F_SG_BIT] =               "tx-scatter-gather",
	[NETIF_F_IP_CSUM_BIT] =          "tx-checksum-ipv4",
	[NETIF_F_HW_CSUM_BIT] =          "tx-checksum-ip-generic",
	[NETIF_F_IPV6_CSUM_BIT] =        "tx-checksum-ipv6",
	[NETIF_F_HIGHDMA_BIT] =          "highdma",
	[NETIF_F_FRAGLIST_BIT] =         "tx-scatter-gather-fraglist",
64
	[NETIF_F_HW_VLAN_CTAG_TX_BIT] =  "tx-vlan-hw-insert",
65

66 67
	[NETIF_F_HW_VLAN_CTAG_RX_BIT] =  "rx-vlan-hw-parse",
	[NETIF_F_HW_VLAN_CTAG_FILTER_BIT] = "rx-vlan-filter",
68 69 70
	[NETIF_F_HW_VLAN_STAG_TX_BIT] =  "tx-vlan-stag-hw-insert",
	[NETIF_F_HW_VLAN_STAG_RX_BIT] =  "rx-vlan-stag-hw-parse",
	[NETIF_F_HW_VLAN_STAG_FILTER_BIT] = "rx-vlan-stag-filter",
71 72 73 74 75 76 77 78 79 80 81 82 83
	[NETIF_F_VLAN_CHALLENGED_BIT] =  "vlan-challenged",
	[NETIF_F_GSO_BIT] =              "tx-generic-segmentation",
	[NETIF_F_LLTX_BIT] =             "tx-lockless",
	[NETIF_F_NETNS_LOCAL_BIT] =      "netns-local",
	[NETIF_F_GRO_BIT] =              "rx-gro",
	[NETIF_F_LRO_BIT] =              "rx-lro",

	[NETIF_F_TSO_BIT] =              "tx-tcp-segmentation",
	[NETIF_F_UFO_BIT] =              "tx-udp-fragmentation",
	[NETIF_F_GSO_ROBUST_BIT] =       "tx-gso-robust",
	[NETIF_F_TSO_ECN_BIT] =          "tx-tcp-ecn-segmentation",
	[NETIF_F_TSO6_BIT] =             "tx-tcp6-segmentation",
	[NETIF_F_FSO_BIT] =              "tx-fcoe-segmentation",
84
	[NETIF_F_GSO_GRE_BIT] =		 "tx-gre-segmentation",
E
Eric Dumazet 已提交
85
	[NETIF_F_GSO_IPIP_BIT] =	 "tx-ipip-segmentation",
E
Eric Dumazet 已提交
86
	[NETIF_F_GSO_SIT_BIT] =		 "tx-sit-segmentation",
87
	[NETIF_F_GSO_UDP_TUNNEL_BIT] =	 "tx-udp_tnl-segmentation",
88 89 90 91 92 93 94 95 96

	[NETIF_F_FCOE_CRC_BIT] =         "tx-checksum-fcoe-crc",
	[NETIF_F_SCTP_CSUM_BIT] =        "tx-checksum-sctp",
	[NETIF_F_FCOE_MTU_BIT] =         "fcoe-mtu",
	[NETIF_F_NTUPLE_BIT] =           "rx-ntuple-filter",
	[NETIF_F_RXHASH_BIT] =           "rx-hashing",
	[NETIF_F_RXCSUM_BIT] =           "rx-checksum",
	[NETIF_F_NOCACHE_COPY_BIT] =     "tx-nocache-copy",
	[NETIF_F_LOOPBACK_BIT] =         "loopback",
B
Ben Greear 已提交
97
	[NETIF_F_RXFCS_BIT] =            "rx-fcs",
B
Ben Greear 已提交
98
	[NETIF_F_RXALL_BIT] =            "rx-all",
99
	[NETIF_F_HW_L2FW_DOFFLOAD_BIT] = "l2-fwd-offload",
J
Jiri Pirko 已提交
100
	[NETIF_F_BUSY_POLL_BIT] =        "busy-poll",
101 102
};

103 104 105 106 107 108
static const char
rss_hash_func_strings[ETH_RSS_HASH_FUNCS_COUNT][ETH_GSTRING_LEN] = {
	[ETH_RSS_HASH_TOP_BIT] =	"toeplitz",
	[ETH_RSS_HASH_XOR_BIT] =	"xor",
};

109 110 111 112 113 114
static int ethtool_get_features(struct net_device *dev, void __user *useraddr)
{
	struct ethtool_gfeatures cmd = {
		.cmd = ETHTOOL_GFEATURES,
		.size = ETHTOOL_DEV_FEATURE_WORDS,
	};
115
	struct ethtool_get_features_block features[ETHTOOL_DEV_FEATURE_WORDS];
116 117
	u32 __user *sizeaddr;
	u32 copy_size;
118 119 120
	int i;

	/* in case feature bits run out again */
M
Michał Mirosław 已提交
121
	BUILD_BUG_ON(ETHTOOL_DEV_FEATURE_WORDS * sizeof(u32) > sizeof(netdev_features_t));
122 123

	for (i = 0; i < ETHTOOL_DEV_FEATURE_WORDS; ++i) {
M
Michał Mirosław 已提交
124 125 126 127 128
		features[i].available = (u32)(dev->hw_features >> (32 * i));
		features[i].requested = (u32)(dev->wanted_features >> (32 * i));
		features[i].active = (u32)(dev->features >> (32 * i));
		features[i].never_changed =
			(u32)(NETIF_F_NEVER_CHANGE >> (32 * i));
129
	}
130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150

	sizeaddr = useraddr + offsetof(struct ethtool_gfeatures, size);
	if (get_user(copy_size, sizeaddr))
		return -EFAULT;

	if (copy_size > ETHTOOL_DEV_FEATURE_WORDS)
		copy_size = ETHTOOL_DEV_FEATURE_WORDS;

	if (copy_to_user(useraddr, &cmd, sizeof(cmd)))
		return -EFAULT;
	useraddr += sizeof(cmd);
	if (copy_to_user(useraddr, features, copy_size * sizeof(*features)))
		return -EFAULT;

	return 0;
}

static int ethtool_set_features(struct net_device *dev, void __user *useraddr)
{
	struct ethtool_sfeatures cmd;
	struct ethtool_set_features_block features[ETHTOOL_DEV_FEATURE_WORDS];
151 152
	netdev_features_t wanted = 0, valid = 0;
	int i, ret = 0;
153 154 155 156 157 158 159 160 161 162 163

	if (copy_from_user(&cmd, useraddr, sizeof(cmd)))
		return -EFAULT;
	useraddr += sizeof(cmd);

	if (cmd.size != ETHTOOL_DEV_FEATURE_WORDS)
		return -EINVAL;

	if (copy_from_user(features, useraddr, sizeof(features)))
		return -EFAULT;

164
	for (i = 0; i < ETHTOOL_DEV_FEATURE_WORDS; ++i) {
M
Michał Mirosław 已提交
165 166
		valid |= (netdev_features_t)features[i].valid << (32 * i);
		wanted |= (netdev_features_t)features[i].requested << (32 * i);
167 168 169
	}

	if (valid & ~NETIF_F_ETHTOOL_BITS)
170 171
		return -EINVAL;

172 173
	if (valid & ~dev->hw_features) {
		valid &= dev->hw_features;
174 175 176
		ret |= ETHTOOL_F_UNSUPPORTED;
	}

177 178
	dev->wanted_features &= ~valid;
	dev->wanted_features |= wanted & valid;
179
	__netdev_update_features(dev);
180

181
	if ((dev->wanted_features ^ dev->features) & valid)
182 183 184 185 186
		ret |= ETHTOOL_F_WISH;

	return ret;
}

187 188 189 190
static int __ethtool_get_sset_count(struct net_device *dev, int sset)
{
	const struct ethtool_ops *ops = dev->ethtool_ops;

191 192 193
	if (sset == ETH_SS_FEATURES)
		return ARRAY_SIZE(netdev_features_strings);

194 195 196
	if (sset == ETH_SS_RSS_HASH_FUNCS)
		return ARRAY_SIZE(rss_hash_func_strings);

197
	if (ops->get_sset_count && ops->get_strings)
198 199 200 201 202 203 204 205 206 207
		return ops->get_sset_count(dev, sset);
	else
		return -EOPNOTSUPP;
}

static void __ethtool_get_strings(struct net_device *dev,
	u32 stringset, u8 *data)
{
	const struct ethtool_ops *ops = dev->ethtool_ops;

208 209 210
	if (stringset == ETH_SS_FEATURES)
		memcpy(data, netdev_features_strings,
			sizeof(netdev_features_strings));
211 212 213
	else if (stringset == ETH_SS_RSS_HASH_FUNCS)
		memcpy(data, rss_hash_func_strings,
		       sizeof(rss_hash_func_strings));
214 215 216
	else
		/* ops->get_strings is valid because checked earlier */
		ops->get_strings(dev, stringset, data);
217 218
}

219
static netdev_features_t ethtool_get_feature_mask(u32 eth_cmd)
220 221 222 223 224 225 226
{
	/* feature masks of legacy discrete ethtool ops */

	switch (eth_cmd) {
	case ETHTOOL_GTXCSUM:
	case ETHTOOL_STXCSUM:
		return NETIF_F_ALL_CSUM | NETIF_F_SCTP_CSUM;
227 228 229
	case ETHTOOL_GRXCSUM:
	case ETHTOOL_SRXCSUM:
		return NETIF_F_RXCSUM;
230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252
	case ETHTOOL_GSG:
	case ETHTOOL_SSG:
		return NETIF_F_SG;
	case ETHTOOL_GTSO:
	case ETHTOOL_STSO:
		return NETIF_F_ALL_TSO;
	case ETHTOOL_GUFO:
	case ETHTOOL_SUFO:
		return NETIF_F_UFO;
	case ETHTOOL_GGSO:
	case ETHTOOL_SGSO:
		return NETIF_F_GSO;
	case ETHTOOL_GGRO:
	case ETHTOOL_SGRO:
		return NETIF_F_GRO;
	default:
		BUG();
	}
}

static int ethtool_get_one_feature(struct net_device *dev,
	char __user *useraddr, u32 ethcmd)
{
253
	netdev_features_t mask = ethtool_get_feature_mask(ethcmd);
254 255
	struct ethtool_value edata = {
		.cmd = ethcmd,
256
		.data = !!(dev->features & mask),
257 258 259 260 261 262 263 264 265 266 267
	};

	if (copy_to_user(useraddr, &edata, sizeof(edata)))
		return -EFAULT;
	return 0;
}

static int ethtool_set_one_feature(struct net_device *dev,
	void __user *useraddr, u32 ethcmd)
{
	struct ethtool_value edata;
268
	netdev_features_t mask;
269 270 271 272

	if (copy_from_user(&edata, useraddr, sizeof(edata)))
		return -EFAULT;

273 274
	mask = ethtool_get_feature_mask(ethcmd);
	mask &= dev->hw_features;
M
Michał Mirosław 已提交
275 276
	if (!mask)
		return -EOPNOTSUPP;
277

M
Michał Mirosław 已提交
278 279 280 281
	if (edata.data)
		dev->wanted_features |= mask;
	else
		dev->wanted_features &= ~mask;
282

M
Michał Mirosław 已提交
283
	__netdev_update_features(dev);
284

M
Michał Mirosław 已提交
285 286 287
	return 0;
}

288 289
#define ETH_ALL_FLAGS    (ETH_FLAG_LRO | ETH_FLAG_RXVLAN | ETH_FLAG_TXVLAN | \
			  ETH_FLAG_NTUPLE | ETH_FLAG_RXHASH)
290 291 292
#define ETH_ALL_FEATURES (NETIF_F_LRO | NETIF_F_HW_VLAN_CTAG_RX | \
			  NETIF_F_HW_VLAN_CTAG_TX | NETIF_F_NTUPLE | \
			  NETIF_F_RXHASH)
M
Michał Mirosław 已提交
293 294 295

static u32 __ethtool_get_flags(struct net_device *dev)
{
296 297
	u32 flags = 0;

298 299 300 301 302 303 304 305 306 307
	if (dev->features & NETIF_F_LRO)
		flags |= ETH_FLAG_LRO;
	if (dev->features & NETIF_F_HW_VLAN_CTAG_RX)
		flags |= ETH_FLAG_RXVLAN;
	if (dev->features & NETIF_F_HW_VLAN_CTAG_TX)
		flags |= ETH_FLAG_TXVLAN;
	if (dev->features & NETIF_F_NTUPLE)
		flags |= ETH_FLAG_NTUPLE;
	if (dev->features & NETIF_F_RXHASH)
		flags |= ETH_FLAG_RXHASH;
308 309

	return flags;
310 311
}

M
Michał Mirosław 已提交
312
static int __ethtool_set_flags(struct net_device *dev, u32 data)
313
{
314
	netdev_features_t features = 0, changed;
315

316
	if (data & ~ETH_ALL_FLAGS)
317 318
		return -EINVAL;

319 320 321 322 323 324 325 326 327 328
	if (data & ETH_FLAG_LRO)
		features |= NETIF_F_LRO;
	if (data & ETH_FLAG_RXVLAN)
		features |= NETIF_F_HW_VLAN_CTAG_RX;
	if (data & ETH_FLAG_TXVLAN)
		features |= NETIF_F_HW_VLAN_CTAG_TX;
	if (data & ETH_FLAG_NTUPLE)
		features |= NETIF_F_NTUPLE;
	if (data & ETH_FLAG_RXHASH)
		features |= NETIF_F_RXHASH;
329

330
	/* allow changing only bits set in hw_features */
331
	changed = (features ^ dev->features) & ETH_ALL_FEATURES;
332 333 334 335
	if (changed & ~dev->hw_features)
		return (changed & dev->hw_features) ? -EINVAL : -EOPNOTSUPP;

	dev->wanted_features =
336
		(dev->wanted_features & ~changed) | (features & changed);
337

338
	__netdev_update_features(dev);
339 340 341 342

	return 0;
}

343
int __ethtool_get_settings(struct net_device *dev, struct ethtool_cmd *cmd)
L
Linus Torvalds 已提交
344
{
345
	ASSERT_RTNL();
L
Linus Torvalds 已提交
346

347
	if (!dev->ethtool_ops->get_settings)
L
Linus Torvalds 已提交
348 349
		return -EOPNOTSUPP;

350 351 352 353 354 355 356 357 358 359 360 361
	memset(cmd, 0, sizeof(struct ethtool_cmd));
	cmd->cmd = ETHTOOL_GSET;
	return dev->ethtool_ops->get_settings(dev, cmd);
}
EXPORT_SYMBOL(__ethtool_get_settings);

static int ethtool_get_settings(struct net_device *dev, void __user *useraddr)
{
	int err;
	struct ethtool_cmd cmd;

	err = __ethtool_get_settings(dev, &cmd);
L
Linus Torvalds 已提交
362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382
	if (err < 0)
		return err;

	if (copy_to_user(useraddr, &cmd, sizeof(cmd)))
		return -EFAULT;
	return 0;
}

static int ethtool_set_settings(struct net_device *dev, void __user *useraddr)
{
	struct ethtool_cmd cmd;

	if (!dev->ethtool_ops->set_settings)
		return -EOPNOTSUPP;

	if (copy_from_user(&cmd, useraddr, sizeof(cmd)))
		return -EFAULT;

	return dev->ethtool_ops->set_settings(dev, &cmd);
}

383 384
static noinline_for_stack int ethtool_get_drvinfo(struct net_device *dev,
						  void __user *useraddr)
L
Linus Torvalds 已提交
385 386
{
	struct ethtool_drvinfo info;
387
	const struct ethtool_ops *ops = dev->ethtool_ops;
L
Linus Torvalds 已提交
388 389 390

	memset(&info, 0, sizeof(info));
	info.cmd = ETHTOOL_GDRVINFO;
391
	if (ops->get_drvinfo) {
392 393 394 395 396 397 398 399 400
		ops->get_drvinfo(dev, &info);
	} else if (dev->dev.parent && dev->dev.parent->driver) {
		strlcpy(info.bus_info, dev_name(dev->dev.parent),
			sizeof(info.bus_info));
		strlcpy(info.driver, dev->dev.parent->driver->name,
			sizeof(info.driver));
	} else {
		return -EOPNOTSUPP;
	}
L
Linus Torvalds 已提交
401

402 403
	/*
	 * this method of obtaining string set info is deprecated;
404
	 * Use ETHTOOL_GSSET_INFO instead.
405
	 */
406
	if (ops->get_sset_count) {
407 408 409 410 411 412 413 414
		int rc;

		rc = ops->get_sset_count(dev, ETH_SS_TEST);
		if (rc >= 0)
			info.testinfo_len = rc;
		rc = ops->get_sset_count(dev, ETH_SS_STATS);
		if (rc >= 0)
			info.n_stats = rc;
415 416 417
		rc = ops->get_sset_count(dev, ETH_SS_PRIV_FLAGS);
		if (rc >= 0)
			info.n_priv_flags = rc;
418
	}
419
	if (ops->get_regs_len)
L
Linus Torvalds 已提交
420
		info.regdump_len = ops->get_regs_len(dev);
421
	if (ops->get_eeprom_len)
L
Linus Torvalds 已提交
422 423 424 425 426 427 428
		info.eedump_len = ops->get_eeprom_len(dev);

	if (copy_to_user(useraddr, &info, sizeof(info)))
		return -EFAULT;
	return 0;
}

E
Eric Dumazet 已提交
429
static noinline_for_stack int ethtool_get_sset_info(struct net_device *dev,
430
						    void __user *useraddr)
431 432 433 434 435 436 437 438 439 440 441 442 443 444 445
{
	struct ethtool_sset_info info;
	u64 sset_mask;
	int i, idx = 0, n_bits = 0, ret, rc;
	u32 *info_buf = NULL;

	if (copy_from_user(&info, useraddr, sizeof(info)))
		return -EFAULT;

	/* store copy of mask, because we zero struct later on */
	sset_mask = info.sset_mask;
	if (!sset_mask)
		return 0;

	/* calculate size of return buffer */
446
	n_bits = hweight64(sset_mask);
447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462

	memset(&info, 0, sizeof(info));
	info.cmd = ETHTOOL_GSSET_INFO;

	info_buf = kzalloc(n_bits * sizeof(u32), GFP_USER);
	if (!info_buf)
		return -ENOMEM;

	/*
	 * fill return buffer based on input bitmask and successful
	 * get_sset_count return
	 */
	for (i = 0; i < 64; i++) {
		if (!(sset_mask & (1ULL << i)))
			continue;

463
		rc = __ethtool_get_sset_count(dev, i);
464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484
		if (rc >= 0) {
			info.sset_mask |= (1ULL << i);
			info_buf[idx++] = rc;
		}
	}

	ret = -EFAULT;
	if (copy_to_user(useraddr, &info, sizeof(info)))
		goto out;

	useraddr += offsetof(struct ethtool_sset_info, data);
	if (copy_to_user(useraddr, info_buf, idx * sizeof(u32)))
		goto out;

	ret = 0;

out:
	kfree(info_buf);
	return ret;
}

485
static noinline_for_stack int ethtool_set_rxnfc(struct net_device *dev,
486
						u32 cmd, void __user *useraddr)
487
{
488 489
	struct ethtool_rxnfc info;
	size_t info_size = sizeof(info);
490
	int rc;
491

492
	if (!dev->ethtool_ops->set_rxnfc)
493 494
		return -EOPNOTSUPP;

495 496 497 498 499 500 501 502 503
	/* struct ethtool_rxnfc was originally defined for
	 * ETHTOOL_{G,S}RXFH with only the cmd, flow_type and data
	 * members.  User-space might still be using that
	 * definition. */
	if (cmd == ETHTOOL_SRXFH)
		info_size = (offsetof(struct ethtool_rxnfc, data) +
			     sizeof(info.data));

	if (copy_from_user(&info, useraddr, info_size))
504 505
		return -EFAULT;

506 507 508 509 510 511 512 513 514
	rc = dev->ethtool_ops->set_rxnfc(dev, &info);
	if (rc)
		return rc;

	if (cmd == ETHTOOL_SRXCLSRLINS &&
	    copy_to_user(useraddr, &info, info_size))
		return -EFAULT;

	return 0;
515 516
}

517
static noinline_for_stack int ethtool_get_rxnfc(struct net_device *dev,
518
						u32 cmd, void __user *useraddr)
519 520
{
	struct ethtool_rxnfc info;
521
	size_t info_size = sizeof(info);
522 523 524
	const struct ethtool_ops *ops = dev->ethtool_ops;
	int ret;
	void *rule_buf = NULL;
525

526
	if (!ops->get_rxnfc)
527 528
		return -EOPNOTSUPP;

529 530 531 532 533 534 535 536 537
	/* struct ethtool_rxnfc was originally defined for
	 * ETHTOOL_{G,S}RXFH with only the cmd, flow_type and data
	 * members.  User-space might still be using that
	 * definition. */
	if (cmd == ETHTOOL_GRXFH)
		info_size = (offsetof(struct ethtool_rxnfc, data) +
			     sizeof(info.data));

	if (copy_from_user(&info, useraddr, info_size))
538 539
		return -EFAULT;

540 541
	if (info.cmd == ETHTOOL_GRXCLSRLALL) {
		if (info.rule_cnt > 0) {
542
			if (info.rule_cnt <= KMALLOC_MAX_SIZE / sizeof(u32))
543
				rule_buf = kzalloc(info.rule_cnt * sizeof(u32),
544
						   GFP_USER);
545 546 547 548
			if (!rule_buf)
				return -ENOMEM;
		}
	}
549

550 551 552 553 554
	ret = ops->get_rxnfc(dev, &info, rule_buf);
	if (ret < 0)
		goto err_out;

	ret = -EFAULT;
555
	if (copy_to_user(useraddr, &info, info_size))
556 557 558 559 560 561 562 563 564 565 566
		goto err_out;

	if (rule_buf) {
		useraddr += offsetof(struct ethtool_rxnfc, rule_locs);
		if (copy_to_user(useraddr, rule_buf,
				 info.rule_cnt * sizeof(u32)))
			goto err_out;
	}
	ret = 0;

err_out:
567
	kfree(rule_buf);
568 569

	return ret;
570 571
}

572 573 574 575
static int ethtool_copy_validate_indir(u32 *indir, void __user *useraddr,
					struct ethtool_rxnfc *rx_rings,
					u32 size)
{
576
	int i;
577 578

	if (copy_from_user(indir, useraddr, size * sizeof(indir[0])))
579
		return -EFAULT;
580 581

	/* Validate ring indices */
582 583 584 585 586
	for (i = 0; i < size; i++)
		if (indir[i] >= rx_rings->data)
			return -EINVAL;

	return 0;
587 588
}

589 590 591 592 593 594 595 596 597 598
u8 netdev_rss_key[NETDEV_RSS_KEY_LEN];

void netdev_rss_key_fill(void *buffer, size_t len)
{
	BUG_ON(len > sizeof(netdev_rss_key));
	net_get_random_once(netdev_rss_key, sizeof(netdev_rss_key));
	memcpy(buffer, netdev_rss_key, len);
}
EXPORT_SYMBOL(netdev_rss_key_fill);

599 600 601
static noinline_for_stack int ethtool_get_rxfh_indir(struct net_device *dev,
						     void __user *useraddr)
{
602 603
	u32 user_size, dev_size;
	u32 *indir;
604 605
	int ret;

606
	if (!dev->ethtool_ops->get_rxfh_indir_size ||
607
	    !dev->ethtool_ops->get_rxfh)
608 609 610
		return -EOPNOTSUPP;
	dev_size = dev->ethtool_ops->get_rxfh_indir_size(dev);
	if (dev_size == 0)
611 612
		return -EOPNOTSUPP;

613
	if (copy_from_user(&user_size,
614
			   useraddr + offsetof(struct ethtool_rxfh_indir, size),
615
			   sizeof(user_size)))
616 617
		return -EFAULT;

618 619 620 621 622 623 624 625 626 627 628 629
	if (copy_to_user(useraddr + offsetof(struct ethtool_rxfh_indir, size),
			 &dev_size, sizeof(dev_size)))
		return -EFAULT;

	/* If the user buffer size is 0, this is just a query for the
	 * device table size.  Otherwise, if it's smaller than the
	 * device table size it's an error.
	 */
	if (user_size < dev_size)
		return user_size == 0 ? 0 : -EINVAL;

	indir = kcalloc(dev_size, sizeof(indir[0]), GFP_USER);
630 631 632
	if (!indir)
		return -ENOMEM;

633
	ret = dev->ethtool_ops->get_rxfh(dev, indir, NULL, NULL);
634 635 636
	if (ret)
		goto out;

637 638 639
	if (copy_to_user(useraddr +
			 offsetof(struct ethtool_rxfh_indir, ring_index[0]),
			 indir, dev_size * sizeof(indir[0])))
640 641 642 643 644 645 646 647 648 649
		ret = -EFAULT;

out:
	kfree(indir);
	return ret;
}

static noinline_for_stack int ethtool_set_rxfh_indir(struct net_device *dev,
						     void __user *useraddr)
{
650 651 652
	struct ethtool_rxnfc rx_rings;
	u32 user_size, dev_size, i;
	u32 *indir;
653
	const struct ethtool_ops *ops = dev->ethtool_ops;
654
	int ret;
655
	u32 ringidx_offset = offsetof(struct ethtool_rxfh_indir, ring_index[0]);
656

657
	if (!ops->get_rxfh_indir_size || !ops->set_rxfh ||
658
	    !ops->get_rxnfc)
659
		return -EOPNOTSUPP;
660 661

	dev_size = ops->get_rxfh_indir_size(dev);
662
	if (dev_size == 0)
663 664
		return -EOPNOTSUPP;

665
	if (copy_from_user(&user_size,
666
			   useraddr + offsetof(struct ethtool_rxfh_indir, size),
667
			   sizeof(user_size)))
668 669
		return -EFAULT;

670
	if (user_size != 0 && user_size != dev_size)
671 672 673
		return -EINVAL;

	indir = kcalloc(dev_size, sizeof(indir[0]), GFP_USER);
674 675 676
	if (!indir)
		return -ENOMEM;

677
	rx_rings.cmd = ETHTOOL_GRXRINGS;
678
	ret = ops->get_rxnfc(dev, &rx_rings, NULL);
679 680
	if (ret)
		goto out;
681 682 683 684 685

	if (user_size == 0) {
		for (i = 0; i < dev_size; i++)
			indir[i] = ethtool_rxfh_indir_default(i, rx_rings.data);
	} else {
686 687 688 689 690 691 692 693
		ret = ethtool_copy_validate_indir(indir,
						  useraddr + ringidx_offset,
						  &rx_rings,
						  dev_size);
		if (ret)
			goto out;
	}

694
	ret = ops->set_rxfh(dev, indir, NULL, ETH_RSS_HASH_NO_CHANGE);
695 696 697 698 699 700 701 702 703 704 705

out:
	kfree(indir);
	return ret;
}

static noinline_for_stack int ethtool_get_rxfh(struct net_device *dev,
					       void __user *useraddr)
{
	int ret;
	const struct ethtool_ops *ops = dev->ethtool_ops;
706
	u32 user_indir_size, user_key_size;
707
	u32 dev_indir_size = 0, dev_key_size = 0;
708
	struct ethtool_rxfh rxfh;
709
	u32 total_size;
710
	u32 indir_bytes;
711
	u32 *indir = NULL;
712
	u8 dev_hfunc = 0;
713 714 715
	u8 *hkey = NULL;
	u8 *rss_config;

716
	if (!ops->get_rxfh)
717 718 719 720 721 722 723
		return -EOPNOTSUPP;

	if (ops->get_rxfh_indir_size)
		dev_indir_size = ops->get_rxfh_indir_size(dev);
	if (ops->get_rxfh_key_size)
		dev_key_size = ops->get_rxfh_key_size(dev);

724
	if (copy_from_user(&rxfh, useraddr, sizeof(rxfh)))
725
		return -EFAULT;
726 727
	user_indir_size = rxfh.indir_size;
	user_key_size = rxfh.key_size;
728

729
	/* Check that reserved fields are 0 for now */
730 731
	if (rxfh.rss_context || rxfh.rsvd8[0] || rxfh.rsvd8[1] ||
	    rxfh.rsvd8[2] || rxfh.rsvd32)
732 733 734 735 736
		return -EINVAL;

	rxfh.indir_size = dev_indir_size;
	rxfh.key_size = dev_key_size;
	if (copy_to_user(useraddr, &rxfh, sizeof(rxfh)))
737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754
		return -EFAULT;

	if ((user_indir_size && (user_indir_size != dev_indir_size)) ||
	    (user_key_size && (user_key_size != dev_key_size)))
		return -EINVAL;

	indir_bytes = user_indir_size * sizeof(indir[0]);
	total_size = indir_bytes + user_key_size;
	rss_config = kzalloc(total_size, GFP_USER);
	if (!rss_config)
		return -ENOMEM;

	if (user_indir_size)
		indir = (u32 *)rss_config;

	if (user_key_size)
		hkey = rss_config + indir_bytes;

755 756 757
	ret = dev->ethtool_ops->get_rxfh(dev, indir, hkey, &dev_hfunc);
	if (ret)
		goto out;
758

759 760 761 762 763 764 765 766 767
	if (copy_to_user(useraddr + offsetof(struct ethtool_rxfh, hfunc),
			 &dev_hfunc, sizeof(rxfh.hfunc))) {
		ret = -EFAULT;
	} else if (copy_to_user(useraddr +
			      offsetof(struct ethtool_rxfh, rss_config[0]),
			      rss_config, total_size)) {
		ret = -EFAULT;
	}
out:
768 769 770 771 772 773 774 775 776 777 778
	kfree(rss_config);

	return ret;
}

static noinline_for_stack int ethtool_set_rxfh(struct net_device *dev,
					       void __user *useraddr)
{
	int ret;
	const struct ethtool_ops *ops = dev->ethtool_ops;
	struct ethtool_rxnfc rx_rings;
779 780
	struct ethtool_rxfh rxfh;
	u32 dev_indir_size = 0, dev_key_size = 0, i;
781 782 783 784 785
	u32 *indir = NULL, indir_bytes = 0;
	u8 *hkey = NULL;
	u8 *rss_config;
	u32 rss_cfg_offset = offsetof(struct ethtool_rxfh, rss_config[0]);

786
	if (!ops->get_rxnfc || !ops->set_rxfh)
787 788 789 790 791 792 793
		return -EOPNOTSUPP;

	if (ops->get_rxfh_indir_size)
		dev_indir_size = ops->get_rxfh_indir_size(dev);
	if (ops->get_rxfh_key_size)
		dev_key_size = dev->ethtool_ops->get_rxfh_key_size(dev);

794
	if (copy_from_user(&rxfh, useraddr, sizeof(rxfh)))
795 796
		return -EFAULT;

797
	/* Check that reserved fields are 0 for now */
798 799
	if (rxfh.rss_context || rxfh.rsvd8[0] || rxfh.rsvd8[1] ||
	    rxfh.rsvd8[2] || rxfh.rsvd32)
800 801
		return -EINVAL;

802 803
	/* If either indir, hash key or function is valid, proceed further.
	 * Must request at least one change: indir size, hash key or function.
804
	 */
805 806 807 808 809
	if ((rxfh.indir_size &&
	     rxfh.indir_size != ETH_RXFH_INDIR_NO_CHANGE &&
	     rxfh.indir_size != dev_indir_size) ||
	    (rxfh.key_size && (rxfh.key_size != dev_key_size)) ||
	    (rxfh.indir_size == ETH_RXFH_INDIR_NO_CHANGE &&
810
	     rxfh.key_size == 0 && rxfh.hfunc == ETH_RSS_HASH_NO_CHANGE))
811 812
		return -EINVAL;

813
	if (rxfh.indir_size != ETH_RXFH_INDIR_NO_CHANGE)
814 815
		indir_bytes = dev_indir_size * sizeof(indir[0]);

816
	rss_config = kzalloc(indir_bytes + rxfh.key_size, GFP_USER);
817 818 819 820 821 822 823 824
	if (!rss_config)
		return -ENOMEM;

	rx_rings.cmd = ETHTOOL_GRXRINGS;
	ret = ops->get_rxnfc(dev, &rx_rings, NULL);
	if (ret)
		goto out;

825 826
	/* rxfh.indir_size == 0 means reset the indir table to default.
	 * rxfh.indir_size == ETH_RXFH_INDIR_NO_CHANGE means leave it unchanged.
827
	 */
828 829
	if (rxfh.indir_size &&
	    rxfh.indir_size != ETH_RXFH_INDIR_NO_CHANGE) {
830 831 832 833
		indir = (u32 *)rss_config;
		ret = ethtool_copy_validate_indir(indir,
						  useraddr + rss_cfg_offset,
						  &rx_rings,
834
						  rxfh.indir_size);
835
		if (ret)
836
			goto out;
837
	} else if (rxfh.indir_size == 0) {
838 839 840 841
		indir = (u32 *)rss_config;
		for (i = 0; i < dev_indir_size; i++)
			indir[i] = ethtool_rxfh_indir_default(i, rx_rings.data);
	}
842

843
	if (rxfh.key_size) {
844 845 846
		hkey = rss_config + indir_bytes;
		if (copy_from_user(hkey,
				   useraddr + rss_cfg_offset + indir_bytes,
847
				   rxfh.key_size)) {
848 849
			ret = -EFAULT;
			goto out;
850
		}
851 852
	}

853
	ret = ops->set_rxfh(dev, indir, hkey, rxfh.hfunc);
854 855

out:
856
	kfree(rss_config);
857 858 859
	return ret;
}

L
Linus Torvalds 已提交
860 861 862
static int ethtool_get_regs(struct net_device *dev, char __user *useraddr)
{
	struct ethtool_regs regs;
863
	const struct ethtool_ops *ops = dev->ethtool_ops;
L
Linus Torvalds 已提交
864 865 866 867 868 869 870 871 872 873 874 875 876
	void *regbuf;
	int reglen, ret;

	if (!ops->get_regs || !ops->get_regs_len)
		return -EOPNOTSUPP;

	if (copy_from_user(&regs, useraddr, sizeof(regs)))
		return -EFAULT;

	reglen = ops->get_regs_len(dev);
	if (regs.len > reglen)
		regs.len = reglen;

877
	regbuf = vzalloc(reglen);
878
	if (reglen && !regbuf)
L
Linus Torvalds 已提交
879 880 881 882 883 884 885 886
		return -ENOMEM;

	ops->get_regs(dev, &regs, regbuf);

	ret = -EFAULT;
	if (copy_to_user(useraddr, &regs, sizeof(regs)))
		goto out;
	useraddr += offsetof(struct ethtool_regs, data);
887
	if (regbuf && copy_to_user(useraddr, regbuf, regs.len))
L
Linus Torvalds 已提交
888 889 890 891
		goto out;
	ret = 0;

 out:
892
	vfree(regbuf);
L
Linus Torvalds 已提交
893 894 895
	return ret;
}

B
Ben Hutchings 已提交
896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915
static int ethtool_reset(struct net_device *dev, char __user *useraddr)
{
	struct ethtool_value reset;
	int ret;

	if (!dev->ethtool_ops->reset)
		return -EOPNOTSUPP;

	if (copy_from_user(&reset, useraddr, sizeof(reset)))
		return -EFAULT;

	ret = dev->ethtool_ops->reset(dev, &reset.data);
	if (ret)
		return ret;

	if (copy_to_user(useraddr, &reset, sizeof(reset)))
		return -EFAULT;
	return 0;
}

L
Linus Torvalds 已提交
916 917
static int ethtool_get_wol(struct net_device *dev, char __user *useraddr)
{
918
	struct ethtool_wolinfo wol = { .cmd = ETHTOOL_GWOL };
L
Linus Torvalds 已提交
919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942

	if (!dev->ethtool_ops->get_wol)
		return -EOPNOTSUPP;

	dev->ethtool_ops->get_wol(dev, &wol);

	if (copy_to_user(useraddr, &wol, sizeof(wol)))
		return -EFAULT;
	return 0;
}

static int ethtool_set_wol(struct net_device *dev, char __user *useraddr)
{
	struct ethtool_wolinfo wol;

	if (!dev->ethtool_ops->set_wol)
		return -EOPNOTSUPP;

	if (copy_from_user(&wol, useraddr, sizeof(wol)))
		return -EFAULT;

	return dev->ethtool_ops->set_wol(dev, &wol);
}

943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976
static int ethtool_get_eee(struct net_device *dev, char __user *useraddr)
{
	struct ethtool_eee edata;
	int rc;

	if (!dev->ethtool_ops->get_eee)
		return -EOPNOTSUPP;

	memset(&edata, 0, sizeof(struct ethtool_eee));
	edata.cmd = ETHTOOL_GEEE;
	rc = dev->ethtool_ops->get_eee(dev, &edata);

	if (rc)
		return rc;

	if (copy_to_user(useraddr, &edata, sizeof(edata)))
		return -EFAULT;

	return 0;
}

static int ethtool_set_eee(struct net_device *dev, char __user *useraddr)
{
	struct ethtool_eee edata;

	if (!dev->ethtool_ops->set_eee)
		return -EOPNOTSUPP;

	if (copy_from_user(&edata, useraddr, sizeof(edata)))
		return -EFAULT;

	return dev->ethtool_ops->set_eee(dev, &edata);
}

L
Linus Torvalds 已提交
977 978 979 980 981 982 983 984
static int ethtool_nway_reset(struct net_device *dev)
{
	if (!dev->ethtool_ops->nway_reset)
		return -EOPNOTSUPP;

	return dev->ethtool_ops->nway_reset(dev);
}

985 986 987 988 989 990 991 992 993 994 995 996 997 998
static int ethtool_get_link(struct net_device *dev, char __user *useraddr)
{
	struct ethtool_value edata = { .cmd = ETHTOOL_GLINK };

	if (!dev->ethtool_ops->get_link)
		return -EOPNOTSUPP;

	edata.data = netif_running(dev) && dev->ethtool_ops->get_link(dev);

	if (copy_to_user(useraddr, &edata, sizeof(edata)))
		return -EFAULT;
	return 0;
}

999 1000 1001 1002
static int ethtool_get_any_eeprom(struct net_device *dev, void __user *useraddr,
				  int (*getter)(struct net_device *,
						struct ethtool_eeprom *, u8 *),
				  u32 total_len)
L
Linus Torvalds 已提交
1003 1004
{
	struct ethtool_eeprom eeprom;
1005 1006
	void __user *userbuf = useraddr + sizeof(eeprom);
	u32 bytes_remaining;
L
Linus Torvalds 已提交
1007
	u8 *data;
1008
	int ret = 0;
L
Linus Torvalds 已提交
1009 1010 1011 1012 1013 1014 1015 1016 1017

	if (copy_from_user(&eeprom, useraddr, sizeof(eeprom)))
		return -EFAULT;

	/* Check for wrap and zero */
	if (eeprom.offset + eeprom.len <= eeprom.offset)
		return -EINVAL;

	/* Check for exceeding total eeprom len */
1018
	if (eeprom.offset + eeprom.len > total_len)
L
Linus Torvalds 已提交
1019 1020
		return -EINVAL;

1021
	data = kmalloc(PAGE_SIZE, GFP_USER);
L
Linus Torvalds 已提交
1022 1023 1024
	if (!data)
		return -ENOMEM;

1025 1026 1027 1028
	bytes_remaining = eeprom.len;
	while (bytes_remaining > 0) {
		eeprom.len = min(bytes_remaining, (u32)PAGE_SIZE);

1029
		ret = getter(dev, &eeprom, data);
1030 1031 1032 1033 1034 1035 1036 1037 1038 1039
		if (ret)
			break;
		if (copy_to_user(userbuf, data, eeprom.len)) {
			ret = -EFAULT;
			break;
		}
		userbuf += eeprom.len;
		eeprom.offset += eeprom.len;
		bytes_remaining -= eeprom.len;
	}
L
Linus Torvalds 已提交
1040

1041 1042 1043 1044 1045
	eeprom.len = userbuf - (useraddr + sizeof(eeprom));
	eeprom.offset -= eeprom.len;
	if (copy_to_user(useraddr, &eeprom, sizeof(eeprom)))
		ret = -EFAULT;

L
Linus Torvalds 已提交
1046 1047 1048 1049
	kfree(data);
	return ret;
}

1050 1051 1052 1053
static int ethtool_get_eeprom(struct net_device *dev, void __user *useraddr)
{
	const struct ethtool_ops *ops = dev->ethtool_ops;

1054 1055
	if (!ops->get_eeprom || !ops->get_eeprom_len ||
	    !ops->get_eeprom_len(dev))
1056 1057 1058 1059 1060 1061
		return -EOPNOTSUPP;

	return ethtool_get_any_eeprom(dev, useraddr, ops->get_eeprom,
				      ops->get_eeprom_len(dev));
}

L
Linus Torvalds 已提交
1062 1063 1064
static int ethtool_set_eeprom(struct net_device *dev, void __user *useraddr)
{
	struct ethtool_eeprom eeprom;
1065
	const struct ethtool_ops *ops = dev->ethtool_ops;
1066 1067
	void __user *userbuf = useraddr + sizeof(eeprom);
	u32 bytes_remaining;
L
Linus Torvalds 已提交
1068
	u8 *data;
1069
	int ret = 0;
L
Linus Torvalds 已提交
1070

1071 1072
	if (!ops->set_eeprom || !ops->get_eeprom_len ||
	    !ops->get_eeprom_len(dev))
L
Linus Torvalds 已提交
1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085
		return -EOPNOTSUPP;

	if (copy_from_user(&eeprom, useraddr, sizeof(eeprom)))
		return -EFAULT;

	/* Check for wrap and zero */
	if (eeprom.offset + eeprom.len <= eeprom.offset)
		return -EINVAL;

	/* Check for exceeding total eeprom len */
	if (eeprom.offset + eeprom.len > ops->get_eeprom_len(dev))
		return -EINVAL;

1086
	data = kmalloc(PAGE_SIZE, GFP_USER);
L
Linus Torvalds 已提交
1087 1088 1089
	if (!data)
		return -ENOMEM;

1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104
	bytes_remaining = eeprom.len;
	while (bytes_remaining > 0) {
		eeprom.len = min(bytes_remaining, (u32)PAGE_SIZE);

		if (copy_from_user(data, userbuf, eeprom.len)) {
			ret = -EFAULT;
			break;
		}
		ret = ops->set_eeprom(dev, &eeprom, data);
		if (ret)
			break;
		userbuf += eeprom.len;
		eeprom.offset += eeprom.len;
		bytes_remaining -= eeprom.len;
	}
L
Linus Torvalds 已提交
1105 1106 1107 1108 1109

	kfree(data);
	return ret;
}

1110 1111
static noinline_for_stack int ethtool_get_coalesce(struct net_device *dev,
						   void __user *useraddr)
L
Linus Torvalds 已提交
1112
{
1113
	struct ethtool_coalesce coalesce = { .cmd = ETHTOOL_GCOALESCE };
L
Linus Torvalds 已提交
1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124

	if (!dev->ethtool_ops->get_coalesce)
		return -EOPNOTSUPP;

	dev->ethtool_ops->get_coalesce(dev, &coalesce);

	if (copy_to_user(useraddr, &coalesce, sizeof(coalesce)))
		return -EFAULT;
	return 0;
}

1125 1126
static noinline_for_stack int ethtool_set_coalesce(struct net_device *dev,
						   void __user *useraddr)
L
Linus Torvalds 已提交
1127 1128 1129
{
	struct ethtool_coalesce coalesce;

1130
	if (!dev->ethtool_ops->set_coalesce)
L
Linus Torvalds 已提交
1131 1132 1133 1134 1135 1136 1137 1138 1139 1140
		return -EOPNOTSUPP;

	if (copy_from_user(&coalesce, useraddr, sizeof(coalesce)))
		return -EFAULT;

	return dev->ethtool_ops->set_coalesce(dev, &coalesce);
}

static int ethtool_get_ringparam(struct net_device *dev, void __user *useraddr)
{
1141
	struct ethtool_ringparam ringparam = { .cmd = ETHTOOL_GRINGPARAM };
L
Linus Torvalds 已提交
1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165

	if (!dev->ethtool_ops->get_ringparam)
		return -EOPNOTSUPP;

	dev->ethtool_ops->get_ringparam(dev, &ringparam);

	if (copy_to_user(useraddr, &ringparam, sizeof(ringparam)))
		return -EFAULT;
	return 0;
}

static int ethtool_set_ringparam(struct net_device *dev, void __user *useraddr)
{
	struct ethtool_ringparam ringparam;

	if (!dev->ethtool_ops->set_ringparam)
		return -EOPNOTSUPP;

	if (copy_from_user(&ringparam, useraddr, sizeof(ringparam)))
		return -EFAULT;

	return dev->ethtool_ops->set_ringparam(dev, &ringparam);
}

1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194
static noinline_for_stack int ethtool_get_channels(struct net_device *dev,
						   void __user *useraddr)
{
	struct ethtool_channels channels = { .cmd = ETHTOOL_GCHANNELS };

	if (!dev->ethtool_ops->get_channels)
		return -EOPNOTSUPP;

	dev->ethtool_ops->get_channels(dev, &channels);

	if (copy_to_user(useraddr, &channels, sizeof(channels)))
		return -EFAULT;
	return 0;
}

static noinline_for_stack int ethtool_set_channels(struct net_device *dev,
						   void __user *useraddr)
{
	struct ethtool_channels channels;

	if (!dev->ethtool_ops->set_channels)
		return -EOPNOTSUPP;

	if (copy_from_user(&channels, useraddr, sizeof(channels)))
		return -EFAULT;

	return dev->ethtool_ops->set_channels(dev, &channels);
}

L
Linus Torvalds 已提交
1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212
static int ethtool_get_pauseparam(struct net_device *dev, void __user *useraddr)
{
	struct ethtool_pauseparam pauseparam = { ETHTOOL_GPAUSEPARAM };

	if (!dev->ethtool_ops->get_pauseparam)
		return -EOPNOTSUPP;

	dev->ethtool_ops->get_pauseparam(dev, &pauseparam);

	if (copy_to_user(useraddr, &pauseparam, sizeof(pauseparam)))
		return -EFAULT;
	return 0;
}

static int ethtool_set_pauseparam(struct net_device *dev, void __user *useraddr)
{
	struct ethtool_pauseparam pauseparam;

1213
	if (!dev->ethtool_ops->set_pauseparam)
L
Linus Torvalds 已提交
1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224
		return -EOPNOTSUPP;

	if (copy_from_user(&pauseparam, useraddr, sizeof(pauseparam)))
		return -EFAULT;

	return dev->ethtool_ops->set_pauseparam(dev, &pauseparam);
}

static int ethtool_self_test(struct net_device *dev, char __user *useraddr)
{
	struct ethtool_test test;
1225
	const struct ethtool_ops *ops = dev->ethtool_ops;
L
Linus Torvalds 已提交
1226
	u64 *data;
1227
	int ret, test_len;
L
Linus Torvalds 已提交
1228

1229
	if (!ops->self_test || !ops->get_sset_count)
L
Linus Torvalds 已提交
1230 1231
		return -EOPNOTSUPP;

1232
	test_len = ops->get_sset_count(dev, ETH_SS_TEST);
1233 1234 1235 1236
	if (test_len < 0)
		return test_len;
	WARN_ON(test_len == 0);

L
Linus Torvalds 已提交
1237 1238 1239
	if (copy_from_user(&test, useraddr, sizeof(test)))
		return -EFAULT;

1240 1241
	test.len = test_len;
	data = kmalloc(test_len * sizeof(u64), GFP_USER);
L
Linus Torvalds 已提交
1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268
	if (!data)
		return -ENOMEM;

	ops->self_test(dev, &test, data);

	ret = -EFAULT;
	if (copy_to_user(useraddr, &test, sizeof(test)))
		goto out;
	useraddr += sizeof(test);
	if (copy_to_user(useraddr, data, test.len * sizeof(u64)))
		goto out;
	ret = 0;

 out:
	kfree(data);
	return ret;
}

static int ethtool_get_strings(struct net_device *dev, void __user *useraddr)
{
	struct ethtool_gstrings gstrings;
	u8 *data;
	int ret;

	if (copy_from_user(&gstrings, useraddr, sizeof(gstrings)))
		return -EFAULT;

1269
	ret = __ethtool_get_sset_count(dev, gstrings.string_set);
1270 1271 1272 1273
	if (ret < 0)
		return ret;

	gstrings.len = ret;
L
Linus Torvalds 已提交
1274 1275 1276 1277 1278

	data = kmalloc(gstrings.len * ETH_GSTRING_LEN, GFP_USER);
	if (!data)
		return -ENOMEM;

1279
	__ethtool_get_strings(dev, gstrings.string_set, data);
L
Linus Torvalds 已提交
1280 1281 1282 1283 1284 1285 1286 1287 1288

	ret = -EFAULT;
	if (copy_to_user(useraddr, &gstrings, sizeof(gstrings)))
		goto out;
	useraddr += sizeof(gstrings);
	if (copy_to_user(useraddr, data, gstrings.len * ETH_GSTRING_LEN))
		goto out;
	ret = 0;

1289
out:
L
Linus Torvalds 已提交
1290 1291 1292 1293 1294 1295 1296
	kfree(data);
	return ret;
}

static int ethtool_phys_id(struct net_device *dev, void __user *useraddr)
{
	struct ethtool_value id;
1297
	static bool busy;
1298
	const struct ethtool_ops *ops = dev->ethtool_ops;
1299
	int rc;
L
Linus Torvalds 已提交
1300

1301
	if (!ops->set_phys_id)
L
Linus Torvalds 已提交
1302 1303
		return -EOPNOTSUPP;

1304 1305 1306
	if (busy)
		return -EBUSY;

L
Linus Torvalds 已提交
1307 1308 1309
	if (copy_from_user(&id, useraddr, sizeof(id)))
		return -EFAULT;

1310
	rc = ops->set_phys_id(dev, ETHTOOL_ID_ACTIVE);
1311
	if (rc < 0)
1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322 1323
		return rc;

	/* Drop the RTNL lock while waiting, but prevent reentry or
	 * removal of the device.
	 */
	busy = true;
	dev_hold(dev);
	rtnl_unlock();

	if (rc == 0) {
		/* Driver will handle this itself */
		schedule_timeout_interruptible(
1324
			id.data ? (id.data * HZ) : MAX_SCHEDULE_TIMEOUT);
1325
	} else {
1326 1327 1328 1329
		/* Driver expects to be called at twice the frequency in rc */
		int n = rc * 2, i, interval = HZ / n;

		/* Count down seconds */
1330
		do {
1331 1332 1333 1334
			/* Count down iterations per second */
			i = n;
			do {
				rtnl_lock();
1335
				rc = ops->set_phys_id(dev,
1336 1337 1338 1339 1340 1341
				    (i & 1) ? ETHTOOL_ID_OFF : ETHTOOL_ID_ON);
				rtnl_unlock();
				if (rc)
					break;
				schedule_timeout_interruptible(interval);
			} while (!signal_pending(current) && --i != 0);
1342 1343 1344 1345 1346 1347 1348 1349
		} while (!signal_pending(current) &&
			 (id.data == 0 || --id.data != 0));
	}

	rtnl_lock();
	dev_put(dev);
	busy = false;

1350
	(void) ops->set_phys_id(dev, ETHTOOL_ID_INACTIVE);
1351
	return rc;
L
Linus Torvalds 已提交
1352 1353 1354 1355 1356
}

static int ethtool_get_stats(struct net_device *dev, void __user *useraddr)
{
	struct ethtool_stats stats;
1357
	const struct ethtool_ops *ops = dev->ethtool_ops;
L
Linus Torvalds 已提交
1358
	u64 *data;
1359
	int ret, n_stats;
L
Linus Torvalds 已提交
1360

1361
	if (!ops->get_ethtool_stats || !ops->get_sset_count)
L
Linus Torvalds 已提交
1362 1363
		return -EOPNOTSUPP;

1364
	n_stats = ops->get_sset_count(dev, ETH_SS_STATS);
1365 1366 1367 1368
	if (n_stats < 0)
		return n_stats;
	WARN_ON(n_stats == 0);

L
Linus Torvalds 已提交
1369 1370 1371
	if (copy_from_user(&stats, useraddr, sizeof(stats)))
		return -EFAULT;

1372 1373
	stats.n_stats = n_stats;
	data = kmalloc(n_stats * sizeof(u64), GFP_USER);
L
Linus Torvalds 已提交
1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 1385 1386 1387 1388 1389 1390 1391
	if (!data)
		return -ENOMEM;

	ops->get_ethtool_stats(dev, &stats, data);

	ret = -EFAULT;
	if (copy_to_user(useraddr, &stats, sizeof(stats)))
		goto out;
	useraddr += sizeof(stats);
	if (copy_to_user(useraddr, data, stats.n_stats * sizeof(u64)))
		goto out;
	ret = 0;

 out:
	kfree(data);
	return ret;
}

1392
static int ethtool_get_perm_addr(struct net_device *dev, void __user *useraddr)
1393 1394 1395
{
	struct ethtool_perm_addr epaddr;

1396
	if (copy_from_user(&epaddr, useraddr, sizeof(epaddr)))
1397 1398
		return -EFAULT;

1399 1400 1401
	if (epaddr.size < dev->addr_len)
		return -ETOOSMALL;
	epaddr.size = dev->addr_len;
1402 1403

	if (copy_to_user(useraddr, &epaddr, sizeof(epaddr)))
1404
		return -EFAULT;
1405
	useraddr += sizeof(epaddr);
1406 1407 1408
	if (copy_to_user(useraddr, dev->perm_addr, epaddr.size))
		return -EFAULT;
	return 0;
1409 1410
}

1411 1412
static int ethtool_get_value(struct net_device *dev, char __user *useraddr,
			     u32 cmd, u32 (*actor)(struct net_device *))
1413
{
1414
	struct ethtool_value edata = { .cmd = cmd };
1415

1416
	if (!actor)
1417 1418
		return -EOPNOTSUPP;

1419
	edata.data = actor(dev);
1420 1421 1422 1423 1424 1425

	if (copy_to_user(useraddr, &edata, sizeof(edata)))
		return -EFAULT;
	return 0;
}

1426 1427
static int ethtool_set_value_void(struct net_device *dev, char __user *useraddr,
			     void (*actor)(struct net_device *, u32))
1428 1429 1430
{
	struct ethtool_value edata;

1431
	if (!actor)
1432 1433 1434 1435 1436
		return -EOPNOTSUPP;

	if (copy_from_user(&edata, useraddr, sizeof(edata)))
		return -EFAULT;

1437
	actor(dev, edata.data);
1438 1439 1440
	return 0;
}

1441 1442
static int ethtool_set_value(struct net_device *dev, char __user *useraddr,
			     int (*actor)(struct net_device *, u32))
1443 1444 1445
{
	struct ethtool_value edata;

1446
	if (!actor)
1447 1448 1449 1450 1451
		return -EOPNOTSUPP;

	if (copy_from_user(&edata, useraddr, sizeof(edata)))
		return -EFAULT;

1452
	return actor(dev, edata.data);
1453 1454
}

1455 1456
static noinline_for_stack int ethtool_flash_device(struct net_device *dev,
						   char __user *useraddr)
1457 1458 1459 1460 1461 1462 1463 1464 1465
{
	struct ethtool_flash efl;

	if (copy_from_user(&efl, useraddr, sizeof(efl)))
		return -EFAULT;

	if (!dev->ethtool_ops->flash_device)
		return -EOPNOTSUPP;

1466 1467
	efl.data[ETHTOOL_FLASH_MAX_FILENAME - 1] = 0;

1468 1469 1470
	return dev->ethtool_ops->flash_device(dev, &efl);
}

1471 1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485 1486 1487 1488 1489 1490 1491
static int ethtool_set_dump(struct net_device *dev,
			void __user *useraddr)
{
	struct ethtool_dump dump;

	if (!dev->ethtool_ops->set_dump)
		return -EOPNOTSUPP;

	if (copy_from_user(&dump, useraddr, sizeof(dump)))
		return -EFAULT;

	return dev->ethtool_ops->set_dump(dev, &dump);
}

static int ethtool_get_dump_flag(struct net_device *dev,
				void __user *useraddr)
{
	int ret;
	struct ethtool_dump dump;
	const struct ethtool_ops *ops = dev->ethtool_ops;

1492
	if (!ops->get_dump_flag)
1493 1494 1495 1496 1497 1498 1499 1500 1501 1502 1503 1504 1505 1506 1507 1508 1509 1510 1511 1512 1513 1514 1515
		return -EOPNOTSUPP;

	if (copy_from_user(&dump, useraddr, sizeof(dump)))
		return -EFAULT;

	ret = ops->get_dump_flag(dev, &dump);
	if (ret)
		return ret;

	if (copy_to_user(useraddr, &dump, sizeof(dump)))
		return -EFAULT;
	return 0;
}

static int ethtool_get_dump_data(struct net_device *dev,
				void __user *useraddr)
{
	int ret;
	__u32 len;
	struct ethtool_dump dump, tmp;
	const struct ethtool_ops *ops = dev->ethtool_ops;
	void *data = NULL;

1516
	if (!ops->get_dump_data || !ops->get_dump_flag)
1517 1518 1519 1520 1521 1522 1523 1524 1525 1526 1527
		return -EOPNOTSUPP;

	if (copy_from_user(&dump, useraddr, sizeof(dump)))
		return -EFAULT;

	memset(&tmp, 0, sizeof(tmp));
	tmp.cmd = ETHTOOL_GET_DUMP_FLAG;
	ret = ops->get_dump_flag(dev, &tmp);
	if (ret)
		return ret;

1528
	len = min(tmp.len, dump.len);
1529 1530 1531
	if (!len)
		return -EFAULT;

1532 1533 1534 1535 1536 1537 1538 1539 1540
	/* Don't ever let the driver think there's more space available
	 * than it requested with .get_dump_flag().
	 */
	dump.len = len;

	/* Always allocate enough space to hold the whole thing so that the
	 * driver does not need to check the length and bother with partial
	 * dumping.
	 */
1541 1542 1543 1544 1545 1546 1547
	data = vzalloc(tmp.len);
	if (!data)
		return -ENOMEM;
	ret = ops->get_dump_data(dev, &dump, data);
	if (ret)
		goto out;

1548 1549 1550 1551 1552 1553 1554 1555 1556 1557
	/* There are two sane possibilities:
	 * 1. The driver's .get_dump_data() does not touch dump.len.
	 * 2. Or it may set dump.len to how much it really writes, which
	 *    should be tmp.len (or len if it can do a partial dump).
	 * In any case respond to userspace with the actual length of data
	 * it's receiving.
	 */
	WARN_ON(dump.len != len && dump.len != tmp.len);
	dump.len = len;

1558 1559 1560 1561 1562 1563 1564 1565 1566 1567 1568 1569
	if (copy_to_user(useraddr, &dump, sizeof(dump))) {
		ret = -EFAULT;
		goto out;
	}
	useraddr += offsetof(struct ethtool_dump, data);
	if (copy_to_user(useraddr, data, len))
		ret = -EFAULT;
out:
	vfree(data);
	return ret;
}

1570 1571 1572 1573 1574 1575 1576 1577 1578 1579 1580 1581
static int ethtool_get_ts_info(struct net_device *dev, void __user *useraddr)
{
	int err = 0;
	struct ethtool_ts_info info;
	const struct ethtool_ops *ops = dev->ethtool_ops;
	struct phy_device *phydev = dev->phydev;

	memset(&info, 0, sizeof(info));
	info.cmd = ETHTOOL_GET_TS_INFO;

	if (phydev && phydev->drv && phydev->drv->ts_info) {
		err = phydev->drv->ts_info(phydev, &info);
1582
	} else if (ops->get_ts_info) {
1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597 1598 1599
		err = ops->get_ts_info(dev, &info);
	} else {
		info.so_timestamping =
			SOF_TIMESTAMPING_RX_SOFTWARE |
			SOF_TIMESTAMPING_SOFTWARE;
		info.phc_index = -1;
	}

	if (err)
		return err;

	if (copy_to_user(useraddr, &info, sizeof(info)))
		err = -EFAULT;

	return err;
}

1600 1601 1602 1603 1604 1605 1606 1607 1608 1609 1610 1611 1612 1613 1614
static int __ethtool_get_module_info(struct net_device *dev,
				     struct ethtool_modinfo *modinfo)
{
	const struct ethtool_ops *ops = dev->ethtool_ops;
	struct phy_device *phydev = dev->phydev;

	if (phydev && phydev->drv && phydev->drv->module_info)
		return phydev->drv->module_info(phydev, modinfo);

	if (ops->get_module_info)
		return ops->get_module_info(dev, modinfo);

	return -EOPNOTSUPP;
}

1615 1616 1617 1618 1619 1620 1621 1622 1623
static int ethtool_get_module_info(struct net_device *dev,
				   void __user *useraddr)
{
	int ret;
	struct ethtool_modinfo modinfo;

	if (copy_from_user(&modinfo, useraddr, sizeof(modinfo)))
		return -EFAULT;

1624
	ret = __ethtool_get_module_info(dev, &modinfo);
1625 1626 1627 1628 1629 1630 1631 1632 1633
	if (ret)
		return ret;

	if (copy_to_user(useraddr, &modinfo, sizeof(modinfo)))
		return -EFAULT;

	return 0;
}

1634 1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648
static int __ethtool_get_module_eeprom(struct net_device *dev,
				       struct ethtool_eeprom *ee, u8 *data)
{
	const struct ethtool_ops *ops = dev->ethtool_ops;
	struct phy_device *phydev = dev->phydev;

	if (phydev && phydev->drv && phydev->drv->module_eeprom)
		return phydev->drv->module_eeprom(phydev, ee, data);

	if (ops->get_module_eeprom)
		return ops->get_module_eeprom(dev, ee, data);

	return -EOPNOTSUPP;
}

1649 1650 1651 1652 1653 1654
static int ethtool_get_module_eeprom(struct net_device *dev,
				     void __user *useraddr)
{
	int ret;
	struct ethtool_modinfo modinfo;

1655
	ret = __ethtool_get_module_info(dev, &modinfo);
1656 1657 1658
	if (ret)
		return ret;

1659 1660
	return ethtool_get_any_eeprom(dev, useraddr,
				      __ethtool_get_module_eeprom,
1661 1662 1663
				      modinfo.eeprom_len);
}

1664 1665 1666 1667
static int ethtool_tunable_valid(const struct ethtool_tunable *tuna)
{
	switch (tuna->id) {
	case ETHTOOL_RX_COPYBREAK:
1668
	case ETHTOOL_TX_COPYBREAK:
1669 1670 1671 1672 1673 1674 1675 1676 1677 1678 1679 1680 1681 1682 1683 1684 1685 1686 1687 1688 1689 1690 1691 1692 1693 1694 1695 1696 1697 1698 1699 1700 1701 1702 1703 1704 1705 1706 1707 1708 1709 1710 1711 1712 1713 1714 1715 1716 1717 1718 1719 1720 1721 1722 1723 1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738
		if (tuna->len != sizeof(u32) ||
		    tuna->type_id != ETHTOOL_TUNABLE_U32)
			return -EINVAL;
		break;
	default:
		return -EINVAL;
	}

	return 0;
}

static int ethtool_get_tunable(struct net_device *dev, void __user *useraddr)
{
	int ret;
	struct ethtool_tunable tuna;
	const struct ethtool_ops *ops = dev->ethtool_ops;
	void *data;

	if (!ops->get_tunable)
		return -EOPNOTSUPP;
	if (copy_from_user(&tuna, useraddr, sizeof(tuna)))
		return -EFAULT;
	ret = ethtool_tunable_valid(&tuna);
	if (ret)
		return ret;
	data = kmalloc(tuna.len, GFP_USER);
	if (!data)
		return -ENOMEM;
	ret = ops->get_tunable(dev, &tuna, data);
	if (ret)
		goto out;
	useraddr += sizeof(tuna);
	ret = -EFAULT;
	if (copy_to_user(useraddr, data, tuna.len))
		goto out;
	ret = 0;

out:
	kfree(data);
	return ret;
}

static int ethtool_set_tunable(struct net_device *dev, void __user *useraddr)
{
	int ret;
	struct ethtool_tunable tuna;
	const struct ethtool_ops *ops = dev->ethtool_ops;
	void *data;

	if (!ops->set_tunable)
		return -EOPNOTSUPP;
	if (copy_from_user(&tuna, useraddr, sizeof(tuna)))
		return -EFAULT;
	ret = ethtool_tunable_valid(&tuna);
	if (ret)
		return ret;
	data = kmalloc(tuna.len, GFP_USER);
	if (!data)
		return -ENOMEM;
	useraddr += sizeof(tuna);
	ret = -EFAULT;
	if (copy_from_user(data, useraddr, tuna.len))
		goto out;
	ret = ops->set_tunable(dev, &tuna, data);

out:
	kfree(data);
	return ret;
}

1739
/* The main entry point in this file.  Called from net/core/dev_ioctl.c */
L
Linus Torvalds 已提交
1740

1741
int dev_ethtool(struct net *net, struct ifreq *ifr)
L
Linus Torvalds 已提交
1742
{
1743
	struct net_device *dev = __dev_get_by_name(net, ifr->ifr_name);
L
Linus Torvalds 已提交
1744 1745 1746
	void __user *useraddr = ifr->ifr_data;
	u32 ethcmd;
	int rc;
1747
	netdev_features_t old_features;
L
Linus Torvalds 已提交
1748 1749 1750 1751

	if (!dev || !netif_device_present(dev))
		return -ENODEV;

1752
	if (copy_from_user(&ethcmd, useraddr, sizeof(ethcmd)))
L
Linus Torvalds 已提交
1753 1754
		return -EFAULT;

1755
	/* Allow some commands to be done by anyone */
1756
	switch (ethcmd) {
1757
	case ETHTOOL_GSET:
1758 1759
	case ETHTOOL_GDRVINFO:
	case ETHTOOL_GMSGLVL:
1760
	case ETHTOOL_GLINK:
1761 1762 1763 1764 1765 1766
	case ETHTOOL_GCOALESCE:
	case ETHTOOL_GRINGPARAM:
	case ETHTOOL_GPAUSEPARAM:
	case ETHTOOL_GRXCSUM:
	case ETHTOOL_GTXCSUM:
	case ETHTOOL_GSG:
1767
	case ETHTOOL_GSSET_INFO:
1768
	case ETHTOOL_GSTRINGS:
1769
	case ETHTOOL_GSTATS:
1770 1771 1772 1773
	case ETHTOOL_GTSO:
	case ETHTOOL_GPERMADDR:
	case ETHTOOL_GUFO:
	case ETHTOOL_GGSO:
1774
	case ETHTOOL_GGRO:
1775 1776
	case ETHTOOL_GFLAGS:
	case ETHTOOL_GPFLAGS:
1777
	case ETHTOOL_GRXFH:
1778 1779 1780 1781
	case ETHTOOL_GRXRINGS:
	case ETHTOOL_GRXCLSRLCNT:
	case ETHTOOL_GRXCLSRULE:
	case ETHTOOL_GRXCLSRLALL:
1782
	case ETHTOOL_GRXFHINDIR:
1783
	case ETHTOOL_GRSSH:
1784
	case ETHTOOL_GFEATURES:
1785
	case ETHTOOL_GCHANNELS:
1786
	case ETHTOOL_GET_TS_INFO:
1787
	case ETHTOOL_GEEE:
1788
	case ETHTOOL_GTUNABLE:
1789 1790
		break;
	default:
1791
		if (!ns_capable(net->user_ns, CAP_NET_ADMIN))
1792 1793 1794
			return -EPERM;
	}

1795 1796 1797
	if (dev->ethtool_ops->begin) {
		rc = dev->ethtool_ops->begin(dev);
		if (rc  < 0)
L
Linus Torvalds 已提交
1798
			return rc;
1799
	}
1800 1801
	old_features = dev->features;

L
Linus Torvalds 已提交
1802 1803 1804 1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815 1816 1817 1818 1819 1820 1821
	switch (ethcmd) {
	case ETHTOOL_GSET:
		rc = ethtool_get_settings(dev, useraddr);
		break;
	case ETHTOOL_SSET:
		rc = ethtool_set_settings(dev, useraddr);
		break;
	case ETHTOOL_GDRVINFO:
		rc = ethtool_get_drvinfo(dev, useraddr);
		break;
	case ETHTOOL_GREGS:
		rc = ethtool_get_regs(dev, useraddr);
		break;
	case ETHTOOL_GWOL:
		rc = ethtool_get_wol(dev, useraddr);
		break;
	case ETHTOOL_SWOL:
		rc = ethtool_set_wol(dev, useraddr);
		break;
	case ETHTOOL_GMSGLVL:
1822 1823
		rc = ethtool_get_value(dev, useraddr, ethcmd,
				       dev->ethtool_ops->get_msglevel);
L
Linus Torvalds 已提交
1824 1825
		break;
	case ETHTOOL_SMSGLVL:
1826 1827
		rc = ethtool_set_value_void(dev, useraddr,
				       dev->ethtool_ops->set_msglevel);
L
Linus Torvalds 已提交
1828
		break;
1829 1830 1831 1832 1833 1834
	case ETHTOOL_GEEE:
		rc = ethtool_get_eee(dev, useraddr);
		break;
	case ETHTOOL_SEEE:
		rc = ethtool_set_eee(dev, useraddr);
		break;
L
Linus Torvalds 已提交
1835 1836 1837 1838
	case ETHTOOL_NWAY_RST:
		rc = ethtool_nway_reset(dev);
		break;
	case ETHTOOL_GLINK:
1839
		rc = ethtool_get_link(dev, useraddr);
L
Linus Torvalds 已提交
1840 1841 1842 1843 1844 1845 1846 1847 1848 1849 1850 1851 1852 1853 1854 1855 1856 1857 1858 1859 1860 1861 1862 1863 1864 1865 1866 1867 1868 1869 1870 1871 1872 1873 1874 1875 1876
		break;
	case ETHTOOL_GEEPROM:
		rc = ethtool_get_eeprom(dev, useraddr);
		break;
	case ETHTOOL_SEEPROM:
		rc = ethtool_set_eeprom(dev, useraddr);
		break;
	case ETHTOOL_GCOALESCE:
		rc = ethtool_get_coalesce(dev, useraddr);
		break;
	case ETHTOOL_SCOALESCE:
		rc = ethtool_set_coalesce(dev, useraddr);
		break;
	case ETHTOOL_GRINGPARAM:
		rc = ethtool_get_ringparam(dev, useraddr);
		break;
	case ETHTOOL_SRINGPARAM:
		rc = ethtool_set_ringparam(dev, useraddr);
		break;
	case ETHTOOL_GPAUSEPARAM:
		rc = ethtool_get_pauseparam(dev, useraddr);
		break;
	case ETHTOOL_SPAUSEPARAM:
		rc = ethtool_set_pauseparam(dev, useraddr);
		break;
	case ETHTOOL_TEST:
		rc = ethtool_self_test(dev, useraddr);
		break;
	case ETHTOOL_GSTRINGS:
		rc = ethtool_get_strings(dev, useraddr);
		break;
	case ETHTOOL_PHYS_ID:
		rc = ethtool_phys_id(dev, useraddr);
		break;
	case ETHTOOL_GSTATS:
		rc = ethtool_get_stats(dev, useraddr);
		break;
1877 1878 1879
	case ETHTOOL_GPERMADDR:
		rc = ethtool_get_perm_addr(dev, useraddr);
		break;
1880
	case ETHTOOL_GFLAGS:
1881
		rc = ethtool_get_value(dev, useraddr, ethcmd,
M
Michał Mirosław 已提交
1882
					__ethtool_get_flags);
1883 1884
		break;
	case ETHTOOL_SFLAGS:
1885
		rc = ethtool_set_value(dev, useraddr, __ethtool_set_flags);
1886
		break;
1887
	case ETHTOOL_GPFLAGS:
1888 1889
		rc = ethtool_get_value(dev, useraddr, ethcmd,
				       dev->ethtool_ops->get_priv_flags);
1890 1891
		break;
	case ETHTOOL_SPFLAGS:
1892 1893
		rc = ethtool_set_value(dev, useraddr,
				       dev->ethtool_ops->set_priv_flags);
1894
		break;
1895
	case ETHTOOL_GRXFH:
1896 1897 1898 1899
	case ETHTOOL_GRXRINGS:
	case ETHTOOL_GRXCLSRLCNT:
	case ETHTOOL_GRXCLSRULE:
	case ETHTOOL_GRXCLSRLALL:
1900
		rc = ethtool_get_rxnfc(dev, ethcmd, useraddr);
1901 1902
		break;
	case ETHTOOL_SRXFH:
1903 1904
	case ETHTOOL_SRXCLSRLDEL:
	case ETHTOOL_SRXCLSRLINS:
1905
		rc = ethtool_set_rxnfc(dev, ethcmd, useraddr);
1906
		break;
1907 1908 1909
	case ETHTOOL_FLASHDEV:
		rc = ethtool_flash_device(dev, useraddr);
		break;
B
Ben Hutchings 已提交
1910 1911 1912
	case ETHTOOL_RESET:
		rc = ethtool_reset(dev, useraddr);
		break;
1913 1914 1915
	case ETHTOOL_GSSET_INFO:
		rc = ethtool_get_sset_info(dev, useraddr);
		break;
1916 1917 1918 1919 1920 1921
	case ETHTOOL_GRXFHINDIR:
		rc = ethtool_get_rxfh_indir(dev, useraddr);
		break;
	case ETHTOOL_SRXFHINDIR:
		rc = ethtool_set_rxfh_indir(dev, useraddr);
		break;
1922 1923 1924 1925 1926 1927
	case ETHTOOL_GRSSH:
		rc = ethtool_get_rxfh(dev, useraddr);
		break;
	case ETHTOOL_SRSSH:
		rc = ethtool_set_rxfh(dev, useraddr);
		break;
1928 1929 1930 1931 1932 1933
	case ETHTOOL_GFEATURES:
		rc = ethtool_get_features(dev, useraddr);
		break;
	case ETHTOOL_SFEATURES:
		rc = ethtool_set_features(dev, useraddr);
		break;
1934
	case ETHTOOL_GTXCSUM:
1935
	case ETHTOOL_GRXCSUM:
1936 1937 1938 1939 1940 1941 1942 1943
	case ETHTOOL_GSG:
	case ETHTOOL_GTSO:
	case ETHTOOL_GUFO:
	case ETHTOOL_GGSO:
	case ETHTOOL_GGRO:
		rc = ethtool_get_one_feature(dev, useraddr, ethcmd);
		break;
	case ETHTOOL_STXCSUM:
1944
	case ETHTOOL_SRXCSUM:
1945 1946 1947 1948 1949 1950 1951
	case ETHTOOL_SSG:
	case ETHTOOL_STSO:
	case ETHTOOL_SUFO:
	case ETHTOOL_SGSO:
	case ETHTOOL_SGRO:
		rc = ethtool_set_one_feature(dev, useraddr, ethcmd);
		break;
1952 1953 1954 1955 1956 1957
	case ETHTOOL_GCHANNELS:
		rc = ethtool_get_channels(dev, useraddr);
		break;
	case ETHTOOL_SCHANNELS:
		rc = ethtool_set_channels(dev, useraddr);
		break;
1958 1959 1960 1961 1962 1963 1964 1965 1966
	case ETHTOOL_SET_DUMP:
		rc = ethtool_set_dump(dev, useraddr);
		break;
	case ETHTOOL_GET_DUMP_FLAG:
		rc = ethtool_get_dump_flag(dev, useraddr);
		break;
	case ETHTOOL_GET_DUMP_DATA:
		rc = ethtool_get_dump_data(dev, useraddr);
		break;
1967 1968 1969
	case ETHTOOL_GET_TS_INFO:
		rc = ethtool_get_ts_info(dev, useraddr);
		break;
1970 1971 1972 1973 1974 1975
	case ETHTOOL_GMODULEINFO:
		rc = ethtool_get_module_info(dev, useraddr);
		break;
	case ETHTOOL_GMODULEEEPROM:
		rc = ethtool_get_module_eeprom(dev, useraddr);
		break;
1976 1977 1978 1979 1980 1981
	case ETHTOOL_GTUNABLE:
		rc = ethtool_get_tunable(dev, useraddr);
		break;
	case ETHTOOL_STUNABLE:
		rc = ethtool_set_tunable(dev, useraddr);
		break;
L
Linus Torvalds 已提交
1982
	default:
1983
		rc = -EOPNOTSUPP;
L
Linus Torvalds 已提交
1984
	}
1985

1986
	if (dev->ethtool_ops->complete)
L
Linus Torvalds 已提交
1987
		dev->ethtool_ops->complete(dev);
1988 1989 1990 1991

	if (old_features != dev->features)
		netdev_features_change(dev);

L
Linus Torvalds 已提交
1992 1993
	return rc;
}