mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	ice: add ethtool -m support for reading i2c eeprom modules
Implement ethtool -m support to read eeprom data from SFP/QSFP modules. Signed-off-by: Scott W Taylor <scott.w.taylor@intel.com> Signed-off-by: Tony Nguyen <anthony.l.nguyen@intel.com> Tested-by: Andrew Bowers <andrewx.bowers@intel.com> Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
This commit is contained in:
		
							parent
							
								
									71c780f119
								
							
						
					
					
						commit
						a012dca9f7
					
				| @ -1147,6 +1147,33 @@ struct ice_aqc_set_port_id_led { | ||||
| 	u8 rsvd[13]; | ||||
| }; | ||||
| 
 | ||||
| /* Read/Write SFF EEPROM command (indirect 0x06EE) */ | ||||
| struct ice_aqc_sff_eeprom { | ||||
| 	u8 lport_num; | ||||
| 	u8 lport_num_valid; | ||||
| #define ICE_AQC_SFF_PORT_NUM_VALID	BIT(0) | ||||
| 	__le16 i2c_bus_addr; | ||||
| #define ICE_AQC_SFF_I2CBUS_7BIT_M	0x7F | ||||
| #define ICE_AQC_SFF_I2CBUS_10BIT_M	0x3FF | ||||
| #define ICE_AQC_SFF_I2CBUS_TYPE_M	BIT(10) | ||||
| #define ICE_AQC_SFF_I2CBUS_TYPE_7BIT	0 | ||||
| #define ICE_AQC_SFF_I2CBUS_TYPE_10BIT	ICE_AQC_SFF_I2CBUS_TYPE_M | ||||
| #define ICE_AQC_SFF_SET_EEPROM_PAGE_S	11 | ||||
| #define ICE_AQC_SFF_SET_EEPROM_PAGE_M	(0x3 << ICE_AQC_SFF_SET_EEPROM_PAGE_S) | ||||
| #define ICE_AQC_SFF_NO_PAGE_CHANGE	0 | ||||
| #define ICE_AQC_SFF_SET_23_ON_MISMATCH	1 | ||||
| #define ICE_AQC_SFF_SET_22_ON_MISMATCH	2 | ||||
| #define ICE_AQC_SFF_IS_WRITE		BIT(15) | ||||
| 	__le16 i2c_mem_addr; | ||||
| 	__le16 eeprom_page; | ||||
| #define  ICE_AQC_SFF_EEPROM_BANK_S 0 | ||||
| #define  ICE_AQC_SFF_EEPROM_BANK_M (0xFF << ICE_AQC_SFF_EEPROM_BANK_S) | ||||
| #define  ICE_AQC_SFF_EEPROM_PAGE_S 8 | ||||
| #define  ICE_AQC_SFF_EEPROM_PAGE_M (0xFF << ICE_AQC_SFF_EEPROM_PAGE_S) | ||||
| 	__le32 addr_high; | ||||
| 	__le32 addr_low; | ||||
| }; | ||||
| 
 | ||||
| /* NVM Read command (indirect 0x0701)
 | ||||
|  * NVM Erase commands (direct 0x0702) | ||||
|  * NVM Update commands (indirect 0x0703) | ||||
| @ -1618,6 +1645,7 @@ struct ice_aq_desc { | ||||
| 		struct ice_aqc_get_phy_caps get_phy; | ||||
| 		struct ice_aqc_set_phy_cfg set_phy; | ||||
| 		struct ice_aqc_restart_an restart_an; | ||||
| 		struct ice_aqc_sff_eeprom read_write_sff_param; | ||||
| 		struct ice_aqc_set_port_id_led set_port_id_led; | ||||
| 		struct ice_aqc_get_sw_cfg get_sw_conf; | ||||
| 		struct ice_aqc_sw_rules sw_rules; | ||||
| @ -1741,6 +1769,7 @@ enum ice_adminq_opc { | ||||
| 	ice_aqc_opc_set_event_mask			= 0x0613, | ||||
| 	ice_aqc_opc_set_mac_lb				= 0x0620, | ||||
| 	ice_aqc_opc_set_port_id_led			= 0x06E9, | ||||
| 	ice_aqc_opc_sff_eeprom				= 0x06EE, | ||||
| 
 | ||||
| 	/* NVM commands */ | ||||
| 	ice_aqc_opc_nvm_read				= 0x0701, | ||||
|  | ||||
| @ -2555,6 +2555,52 @@ ice_aq_set_port_id_led(struct ice_port_info *pi, bool is_orig_mode, | ||||
| 	return ice_aq_send_cmd(hw, &desc, NULL, 0, cd); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * ice_aq_sff_eeprom | ||||
|  * @hw: pointer to the HW struct | ||||
|  * @lport: bits [7:0] = logical port, bit [8] = logical port valid | ||||
|  * @bus_addr: I2C bus address of the eeprom (typically 0xA0, 0=topo default) | ||||
|  * @mem_addr: I2C offset. lower 8 bits for address, 8 upper bits zero padding. | ||||
|  * @page: QSFP page | ||||
|  * @set_page: set or ignore the page | ||||
|  * @data: pointer to data buffer to be read/written to the I2C device. | ||||
|  * @length: 1-16 for read, 1 for write. | ||||
|  * @write: 0 read, 1 for write. | ||||
|  * @cd: pointer to command details structure or NULL | ||||
|  * | ||||
|  * Read/Write SFF EEPROM (0x06EE) | ||||
|  */ | ||||
| enum ice_status | ||||
| ice_aq_sff_eeprom(struct ice_hw *hw, u16 lport, u8 bus_addr, | ||||
| 		  u16 mem_addr, u8 page, u8 set_page, u8 *data, u8 length, | ||||
| 		  bool write, struct ice_sq_cd *cd) | ||||
| { | ||||
| 	struct ice_aqc_sff_eeprom *cmd; | ||||
| 	struct ice_aq_desc desc; | ||||
| 	enum ice_status status; | ||||
| 
 | ||||
| 	if (!data || (mem_addr & 0xff00)) | ||||
| 		return ICE_ERR_PARAM; | ||||
| 
 | ||||
| 	ice_fill_dflt_direct_cmd_desc(&desc, ice_aqc_opc_sff_eeprom); | ||||
| 	cmd = &desc.params.read_write_sff_param; | ||||
| 	desc.flags = cpu_to_le16(ICE_AQ_FLAG_RD | ICE_AQ_FLAG_BUF); | ||||
| 	cmd->lport_num = (u8)(lport & 0xff); | ||||
| 	cmd->lport_num_valid = (u8)((lport >> 8) & 0x01); | ||||
| 	cmd->i2c_bus_addr = cpu_to_le16(((bus_addr >> 1) & | ||||
| 					 ICE_AQC_SFF_I2CBUS_7BIT_M) | | ||||
| 					((set_page << | ||||
| 					  ICE_AQC_SFF_SET_EEPROM_PAGE_S) & | ||||
| 					 ICE_AQC_SFF_SET_EEPROM_PAGE_M)); | ||||
| 	cmd->i2c_mem_addr = cpu_to_le16(mem_addr & 0xff); | ||||
| 	cmd->eeprom_page = cpu_to_le16((u16)page << ICE_AQC_SFF_EEPROM_PAGE_S); | ||||
| 	if (write) | ||||
| 		cmd->i2c_bus_addr |= cpu_to_le16(ICE_AQC_SFF_IS_WRITE); | ||||
| 
 | ||||
| 	status = ice_aq_send_cmd(hw, &desc, data, length, cd); | ||||
| 	return status; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * __ice_aq_get_set_rss_lut | ||||
|  * @hw: pointer to the hardware structure | ||||
|  | ||||
| @ -117,6 +117,10 @@ ice_aq_set_mac_loopback(struct ice_hw *hw, bool ena_lpbk, struct ice_sq_cd *cd); | ||||
| enum ice_status | ||||
| ice_aq_set_port_id_led(struct ice_port_info *pi, bool is_orig_mode, | ||||
| 		       struct ice_sq_cd *cd); | ||||
| enum ice_status | ||||
| ice_aq_sff_eeprom(struct ice_hw *hw, u16 lport, u8 bus_addr, | ||||
| 		  u16 mem_addr, u8 page, u8 set_page, u8 *data, u8 length, | ||||
| 		  bool write, struct ice_sq_cd *cd); | ||||
| 
 | ||||
| enum ice_status | ||||
| ice_dis_vsi_txq(struct ice_port_info *pi, u16 vsi_handle, u8 tc, u8 num_queues, | ||||
|  | ||||
| @ -3455,6 +3455,151 @@ ice_set_per_q_coalesce(struct net_device *netdev, u32 q_num, | ||||
| 	return __ice_set_coalesce(netdev, ec, q_num); | ||||
| } | ||||
| 
 | ||||
| #define ICE_I2C_EEPROM_DEV_ADDR		0xA0 | ||||
| #define ICE_I2C_EEPROM_DEV_ADDR2	0xA2 | ||||
| #define ICE_MODULE_TYPE_SFP		0x03 | ||||
| #define ICE_MODULE_TYPE_QSFP_PLUS	0x0D | ||||
| #define ICE_MODULE_TYPE_QSFP28		0x11 | ||||
| #define ICE_MODULE_SFF_ADDR_MODE	0x04 | ||||
| #define ICE_MODULE_SFF_DIAG_CAPAB	0x40 | ||||
| #define ICE_MODULE_REVISION_ADDR	0x01 | ||||
| #define ICE_MODULE_SFF_8472_COMP	0x5E | ||||
| #define ICE_MODULE_SFF_8472_SWAP	0x5C | ||||
| #define ICE_MODULE_QSFP_MAX_LEN		640 | ||||
| 
 | ||||
| /**
 | ||||
|  * ice_get_module_info - get SFF module type and revision information | ||||
|  * @netdev: network interface device structure | ||||
|  * @modinfo: module EEPROM size and layout information structure | ||||
|  */ | ||||
| static int | ||||
| ice_get_module_info(struct net_device *netdev, | ||||
| 		    struct ethtool_modinfo *modinfo) | ||||
| { | ||||
| 	struct ice_netdev_priv *np = netdev_priv(netdev); | ||||
| 	struct ice_vsi *vsi = np->vsi; | ||||
| 	struct ice_pf *pf = vsi->back; | ||||
| 	struct ice_hw *hw = &pf->hw; | ||||
| 	enum ice_status status; | ||||
| 	u8 sff8472_comp = 0; | ||||
| 	u8 sff8472_swap = 0; | ||||
| 	u8 sff8636_rev = 0; | ||||
| 	u8 value = 0; | ||||
| 
 | ||||
| 	status = ice_aq_sff_eeprom(hw, 0, ICE_I2C_EEPROM_DEV_ADDR, 0x00, 0x00, | ||||
| 				   0, &value, 1, 0, NULL); | ||||
| 	if (status) | ||||
| 		return -EIO; | ||||
| 
 | ||||
| 	switch (value) { | ||||
| 	case ICE_MODULE_TYPE_SFP: | ||||
| 		status = ice_aq_sff_eeprom(hw, 0, ICE_I2C_EEPROM_DEV_ADDR, | ||||
| 					   ICE_MODULE_SFF_8472_COMP, 0x00, 0, | ||||
| 					   &sff8472_comp, 1, 0, NULL); | ||||
| 		if (status) | ||||
| 			return -EIO; | ||||
| 		status = ice_aq_sff_eeprom(hw, 0, ICE_I2C_EEPROM_DEV_ADDR, | ||||
| 					   ICE_MODULE_SFF_8472_SWAP, 0x00, 0, | ||||
| 					   &sff8472_swap, 1, 0, NULL); | ||||
| 		if (status) | ||||
| 			return -EIO; | ||||
| 
 | ||||
| 		if (sff8472_swap & ICE_MODULE_SFF_ADDR_MODE) { | ||||
| 			modinfo->type = ETH_MODULE_SFF_8079; | ||||
| 			modinfo->eeprom_len = ETH_MODULE_SFF_8079_LEN; | ||||
| 		} else if (sff8472_comp && | ||||
| 			   (sff8472_swap & ICE_MODULE_SFF_DIAG_CAPAB)) { | ||||
| 			modinfo->type = ETH_MODULE_SFF_8472; | ||||
| 			modinfo->eeprom_len = ETH_MODULE_SFF_8472_LEN; | ||||
| 		} else { | ||||
| 			modinfo->type = ETH_MODULE_SFF_8079; | ||||
| 			modinfo->eeprom_len = ETH_MODULE_SFF_8079_LEN; | ||||
| 		} | ||||
| 		break; | ||||
| 	case ICE_MODULE_TYPE_QSFP_PLUS: | ||||
| 	case ICE_MODULE_TYPE_QSFP28: | ||||
| 		status = ice_aq_sff_eeprom(hw, 0, ICE_I2C_EEPROM_DEV_ADDR, | ||||
| 					   ICE_MODULE_REVISION_ADDR, 0x00, 0, | ||||
| 					   &sff8636_rev, 1, 0, NULL); | ||||
| 		if (status) | ||||
| 			return -EIO; | ||||
| 		/* Check revision compliance */ | ||||
| 		if (sff8636_rev > 0x02) { | ||||
| 			/* Module is SFF-8636 compliant */ | ||||
| 			modinfo->type = ETH_MODULE_SFF_8636; | ||||
| 			modinfo->eeprom_len = ICE_MODULE_QSFP_MAX_LEN; | ||||
| 		} else { | ||||
| 			modinfo->type = ETH_MODULE_SFF_8436; | ||||
| 			modinfo->eeprom_len = ICE_MODULE_QSFP_MAX_LEN; | ||||
| 		} | ||||
| 		break; | ||||
| 	default: | ||||
| 		netdev_warn(netdev, | ||||
| 			    "SFF Module Type not recognized.\n"); | ||||
| 		return -EINVAL; | ||||
| 	} | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * ice_get_module_eeprom - fill buffer with SFF EEPROM contents | ||||
|  * @netdev: network interface device structure | ||||
|  * @ee: EEPROM dump request structure | ||||
|  * @data: buffer to be filled with EEPROM contents | ||||
|  */ | ||||
| static int | ||||
| ice_get_module_eeprom(struct net_device *netdev, | ||||
| 		      struct ethtool_eeprom *ee, u8 *data) | ||||
| { | ||||
| 	struct ice_netdev_priv *np = netdev_priv(netdev); | ||||
| 	u8 addr = ICE_I2C_EEPROM_DEV_ADDR; | ||||
| 	struct ice_vsi *vsi = np->vsi; | ||||
| 	struct ice_pf *pf = vsi->back; | ||||
| 	struct ice_hw *hw = &pf->hw; | ||||
| 	enum ice_status status; | ||||
| 	bool is_sfp = false; | ||||
| 	u16 offset = 0; | ||||
| 	u8 value = 0; | ||||
| 	u8 page = 0; | ||||
| 	int i; | ||||
| 
 | ||||
| 	status = ice_aq_sff_eeprom(hw, 0, addr, offset, page, 0, | ||||
| 				   &value, 1, 0, NULL); | ||||
| 	if (status) | ||||
| 		return -EIO; | ||||
| 
 | ||||
| 	if (!ee || !ee->len || !data) | ||||
| 		return -EINVAL; | ||||
| 
 | ||||
| 	if (value == ICE_MODULE_TYPE_SFP) | ||||
| 		is_sfp = true; | ||||
| 
 | ||||
| 	for (i = 0; i < ee->len; i++) { | ||||
| 		offset = i + ee->offset; | ||||
| 
 | ||||
| 		/* Check if we need to access the other memory page */ | ||||
| 		if (is_sfp) { | ||||
| 			if (offset >= ETH_MODULE_SFF_8079_LEN) { | ||||
| 				offset -= ETH_MODULE_SFF_8079_LEN; | ||||
| 				addr = ICE_I2C_EEPROM_DEV_ADDR2; | ||||
| 			} | ||||
| 		} else { | ||||
| 			while (offset >= ETH_MODULE_SFF_8436_LEN) { | ||||
| 				/* Compute memory page number and offset. */ | ||||
| 				offset -= ETH_MODULE_SFF_8436_LEN / 2; | ||||
| 				page++; | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		status = ice_aq_sff_eeprom(hw, 0, addr, offset, page, !is_sfp, | ||||
| 					   &value, 1, 0, NULL); | ||||
| 		if (status) | ||||
| 			value = 0; | ||||
| 		data[i] = value; | ||||
| 	} | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static const struct ethtool_ops ice_ethtool_ops = { | ||||
| 	.get_link_ksettings	= ice_get_link_ksettings, | ||||
| 	.set_link_ksettings	= ice_set_link_ksettings, | ||||
| @ -3490,6 +3635,8 @@ static const struct ethtool_ops ice_ethtool_ops = { | ||||
| 	.set_per_queue_coalesce = ice_set_per_q_coalesce, | ||||
| 	.get_fecparam		= ice_get_fecparam, | ||||
| 	.set_fecparam		= ice_set_fecparam, | ||||
| 	.get_module_info	= ice_get_module_info, | ||||
| 	.get_module_eeprom	= ice_get_module_eeprom, | ||||
| }; | ||||
| 
 | ||||
| static const struct ethtool_ops ice_ethtool_safe_mode_ops = { | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Scott W Taylor
						Scott W Taylor