ethtool: Extend the ethtool API to obtain plugin module eeprom data
ETHTOOL_GMODULEINFO returns a new struct ethtool_modinfo that will return the type and size of plug-in module eeprom (such as SFP+) for parsing by userland program. ETHTOOL_GMODULEEEPROM returns the raw eeprom information using the existing ethtool_eeprom structture to return the data Signed-off-by: Stuart Hodgson <smhodgson@solarflare.com> Signed-off-by: Ben Hutchings <bhutchings@solarflare.com>
This commit is contained in:
parent
081d094eaa
commit
41c3cb6d20
2 changed files with 80 additions and 0 deletions
|
@ -136,6 +136,23 @@ struct ethtool_eeprom {
|
||||||
__u8 data[0];
|
__u8 data[0];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct ethtool_modinfo - plugin module eeprom information
|
||||||
|
* @cmd: %ETHTOOL_GMODULEINFO
|
||||||
|
* @type: Standard the module information conforms to %ETH_MODULE_SFF_xxxx
|
||||||
|
* @eeprom_len: Length of the eeprom
|
||||||
|
*
|
||||||
|
* This structure is used to return the information to
|
||||||
|
* properly size memory for a subsequent call to %ETHTOOL_GMODULEEEPROM.
|
||||||
|
* The type code indicates the eeprom data format
|
||||||
|
*/
|
||||||
|
struct ethtool_modinfo {
|
||||||
|
__u32 cmd;
|
||||||
|
__u32 type;
|
||||||
|
__u32 eeprom_len;
|
||||||
|
__u32 reserved[8];
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct ethtool_coalesce - coalescing parameters for IRQs and stats updates
|
* struct ethtool_coalesce - coalescing parameters for IRQs and stats updates
|
||||||
* @cmd: ETHTOOL_{G,S}COALESCE
|
* @cmd: ETHTOOL_{G,S}COALESCE
|
||||||
|
@ -920,6 +937,9 @@ static inline u32 ethtool_rxfh_indir_default(u32 index, u32 n_rx_rings)
|
||||||
* @get_ts_info: Get the time stamping and PTP hardware clock capabilities.
|
* @get_ts_info: Get the time stamping and PTP hardware clock capabilities.
|
||||||
* Drivers supporting transmit time stamps in software should set this to
|
* Drivers supporting transmit time stamps in software should set this to
|
||||||
* ethtool_op_get_ts_info().
|
* ethtool_op_get_ts_info().
|
||||||
|
* @get_module_info: Get the size and type of the eeprom contained within
|
||||||
|
* a plug-in module.
|
||||||
|
* @get_module_eeprom: Get the eeprom information from the plug-in module
|
||||||
*
|
*
|
||||||
* All operations are optional (i.e. the function pointer may be set
|
* All operations are optional (i.e. the function pointer may be set
|
||||||
* to %NULL) and callers must take this into account. Callers must
|
* to %NULL) and callers must take this into account. Callers must
|
||||||
|
@ -982,6 +1002,11 @@ struct ethtool_ops {
|
||||||
struct ethtool_dump *, void *);
|
struct ethtool_dump *, void *);
|
||||||
int (*set_dump)(struct net_device *, struct ethtool_dump *);
|
int (*set_dump)(struct net_device *, struct ethtool_dump *);
|
||||||
int (*get_ts_info)(struct net_device *, struct ethtool_ts_info *);
|
int (*get_ts_info)(struct net_device *, struct ethtool_ts_info *);
|
||||||
|
int (*get_module_info)(struct net_device *,
|
||||||
|
struct ethtool_modinfo *);
|
||||||
|
int (*get_module_eeprom)(struct net_device *,
|
||||||
|
struct ethtool_eeprom *, u8 *);
|
||||||
|
|
||||||
|
|
||||||
};
|
};
|
||||||
#endif /* __KERNEL__ */
|
#endif /* __KERNEL__ */
|
||||||
|
@ -1057,6 +1082,8 @@ struct ethtool_ops {
|
||||||
#define ETHTOOL_GET_DUMP_FLAG 0x0000003f /* Get dump settings */
|
#define ETHTOOL_GET_DUMP_FLAG 0x0000003f /* Get dump settings */
|
||||||
#define ETHTOOL_GET_DUMP_DATA 0x00000040 /* Get dump data */
|
#define ETHTOOL_GET_DUMP_DATA 0x00000040 /* Get dump data */
|
||||||
#define ETHTOOL_GET_TS_INFO 0x00000041 /* Get time stamping and PHC info */
|
#define ETHTOOL_GET_TS_INFO 0x00000041 /* Get time stamping and PHC info */
|
||||||
|
#define ETHTOOL_GMODULEINFO 0x00000042 /* Get plug-in module information */
|
||||||
|
#define ETHTOOL_GMODULEEEPROM 0x00000043 /* Get plug-in module eeprom */
|
||||||
|
|
||||||
/* compatibility with older code */
|
/* compatibility with older code */
|
||||||
#define SPARC_ETH_GSET ETHTOOL_GSET
|
#define SPARC_ETH_GSET ETHTOOL_GSET
|
||||||
|
@ -1206,6 +1233,12 @@ struct ethtool_ops {
|
||||||
#define RX_CLS_LOC_FIRST 0xfffffffe
|
#define RX_CLS_LOC_FIRST 0xfffffffe
|
||||||
#define RX_CLS_LOC_LAST 0xfffffffd
|
#define RX_CLS_LOC_LAST 0xfffffffd
|
||||||
|
|
||||||
|
/* EEPROM Standards for plug in modules */
|
||||||
|
#define ETH_MODULE_SFF_8079 0x1
|
||||||
|
#define ETH_MODULE_SFF_8079_LEN 256
|
||||||
|
#define ETH_MODULE_SFF_8472 0x2
|
||||||
|
#define ETH_MODULE_SFF_8472_LEN 512
|
||||||
|
|
||||||
/* Reset flags */
|
/* Reset flags */
|
||||||
/* The reset() operation must clear the flags for the components which
|
/* The reset() operation must clear the flags for the components which
|
||||||
* were actually reset. On successful return, the flags indicate the
|
* were actually reset. On successful return, the flags indicate the
|
||||||
|
|
|
@ -1335,6 +1335,47 @@ static int ethtool_get_ts_info(struct net_device *dev, void __user *useraddr)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int ethtool_get_module_info(struct net_device *dev,
|
||||||
|
void __user *useraddr)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
struct ethtool_modinfo modinfo;
|
||||||
|
const struct ethtool_ops *ops = dev->ethtool_ops;
|
||||||
|
|
||||||
|
if (!ops->get_module_info)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
if (copy_from_user(&modinfo, useraddr, sizeof(modinfo)))
|
||||||
|
return -EFAULT;
|
||||||
|
|
||||||
|
ret = ops->get_module_info(dev, &modinfo);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
if (copy_to_user(useraddr, &modinfo, sizeof(modinfo)))
|
||||||
|
return -EFAULT;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ethtool_get_module_eeprom(struct net_device *dev,
|
||||||
|
void __user *useraddr)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
struct ethtool_modinfo modinfo;
|
||||||
|
const struct ethtool_ops *ops = dev->ethtool_ops;
|
||||||
|
|
||||||
|
if (!ops->get_module_info || !ops->get_module_eeprom)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
ret = ops->get_module_info(dev, &modinfo);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
return ethtool_get_any_eeprom(dev, useraddr, ops->get_module_eeprom,
|
||||||
|
modinfo.eeprom_len);
|
||||||
|
}
|
||||||
|
|
||||||
/* The main entry point in this file. Called from net/core/dev.c */
|
/* The main entry point in this file. Called from net/core/dev.c */
|
||||||
|
|
||||||
int dev_ethtool(struct net *net, struct ifreq *ifr)
|
int dev_ethtool(struct net *net, struct ifreq *ifr)
|
||||||
|
@ -1559,6 +1600,12 @@ int dev_ethtool(struct net *net, struct ifreq *ifr)
|
||||||
case ETHTOOL_GET_TS_INFO:
|
case ETHTOOL_GET_TS_INFO:
|
||||||
rc = ethtool_get_ts_info(dev, useraddr);
|
rc = ethtool_get_ts_info(dev, useraddr);
|
||||||
break;
|
break;
|
||||||
|
case ETHTOOL_GMODULEINFO:
|
||||||
|
rc = ethtool_get_module_info(dev, useraddr);
|
||||||
|
break;
|
||||||
|
case ETHTOOL_GMODULEEEPROM:
|
||||||
|
rc = ethtool_get_module_eeprom(dev, useraddr);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
rc = -EOPNOTSUPP;
|
rc = -EOPNOTSUPP;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue