mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	r8152: support request_firmware for RTL8153
This patch supports loading additional firmware file through request_firmware(). A firmware file may include a header followed by several blocks which have different types of firmware. Currently, the supported types are RTL_FW_END, RTL_FW_PLA, and RTL_FW_USB. The firmware is used to fix some compatible or hardware issues. For example, the device couldn't be found after rebooting several times. The supported chips are RTL_VER_04 (rtl8153a-2.fw) RTL_VER_05 (rtl8153a-3.fw) RTL_VER_06 (rtl8153a-4.fw) RTL_VER_09 (rtl8153b-2.fw) Signed-off-by: Hayes Wang <hayeswang@realtek.com> Reviewed-by: Prashant Malani <pmalani@chromium.org> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									d9f45ab9e6
								
							
						
					
					
						commit
						9370f2d05a
					
				| @ -24,9 +24,11 @@ | ||||
| #include <linux/suspend.h> | ||||
| #include <linux/atomic.h> | ||||
| #include <linux/acpi.h> | ||||
| #include <linux/firmware.h> | ||||
| #include <crypto/hash.h> | ||||
| 
 | ||||
| /* Information for net-next */ | ||||
| #define NETNEXT_VERSION		"10" | ||||
| #define NETNEXT_VERSION		"11" | ||||
| 
 | ||||
| /* Information for net */ | ||||
| #define NET_VERSION		"10" | ||||
| @ -54,8 +56,11 @@ | ||||
| #define PLA_BDC_CR		0xd1a0 | ||||
| #define PLA_TEREDO_TIMER	0xd2cc | ||||
| #define PLA_REALWOW_TIMER	0xd2e8 | ||||
| #define PLA_UPHY_TIMER		0xd388 | ||||
| #define PLA_SUSPEND_FLAG	0xd38a | ||||
| #define PLA_INDICATE_FALG	0xd38c | ||||
| #define PLA_MACDBG_PRE		0xd38c	/* RTL_VER_04 only */ | ||||
| #define PLA_MACDBG_POST		0xd38e	/* RTL_VER_04 only */ | ||||
| #define PLA_EXTRA_STATUS	0xd398 | ||||
| #define PLA_EFUSE_DATA		0xdd00 | ||||
| #define PLA_EFUSE_CMD		0xdd02 | ||||
| @ -110,7 +115,12 @@ | ||||
| #define USB_CONNECT_TIMER	0xcbf8 | ||||
| #define USB_MSC_TIMER		0xcbfc | ||||
| #define USB_BURST_SIZE		0xcfc0 | ||||
| #define USB_FW_FIX_EN0		0xcfca | ||||
| #define USB_FW_FIX_EN1		0xcfcc | ||||
| #define USB_LPM_CONFIG		0xcfd8 | ||||
| #define USB_CSTMR		0xcfef	/* RTL8153A */ | ||||
| #define USB_FW_CTRL		0xd334	/* RTL8153B */ | ||||
| #define USB_FC_TIMER		0xd340 | ||||
| #define USB_USB_CTRL		0xd406 | ||||
| #define USB_PHY_CTRL		0xd408 | ||||
| #define USB_TX_AGG		0xd40a | ||||
| @ -126,6 +136,7 @@ | ||||
| #define USB_LPM_CTRL		0xd41a | ||||
| #define USB_BMU_RESET		0xd4b0 | ||||
| #define USB_U1U2_TIMER		0xd4da | ||||
| #define USB_FW_TASK		0xd4e8	/* RTL8153B */ | ||||
| #define USB_UPS_CTRL		0xd800 | ||||
| #define USB_POWER_CUT		0xd80a | ||||
| #define USB_MISC_0		0xd81a | ||||
| @ -133,18 +144,19 @@ | ||||
| #define USB_AFE_CTRL2		0xd824 | ||||
| #define USB_UPS_CFG		0xd842 | ||||
| #define USB_UPS_FLAGS		0xd848 | ||||
| #define USB_WDT1_CTRL		0xe404 | ||||
| #define USB_WDT11_CTRL		0xe43c | ||||
| #define USB_BP_BA		0xfc26 | ||||
| #define USB_BP_0		0xfc28 | ||||
| #define USB_BP_1		0xfc2a | ||||
| #define USB_BP_2		0xfc2c | ||||
| #define USB_BP_3		0xfc2e | ||||
| #define USB_BP_4		0xfc30 | ||||
| #define USB_BP_5		0xfc32 | ||||
| #define USB_BP_6		0xfc34 | ||||
| #define USB_BP_7		0xfc36 | ||||
| #define USB_BP_EN		0xfc38 | ||||
| #define USB_BP_8		0xfc38 | ||||
| #define USB_BP_BA		PLA_BP_BA | ||||
| #define USB_BP_0		PLA_BP_0 | ||||
| #define USB_BP_1		PLA_BP_1 | ||||
| #define USB_BP_2		PLA_BP_2 | ||||
| #define USB_BP_3		PLA_BP_3 | ||||
| #define USB_BP_4		PLA_BP_4 | ||||
| #define USB_BP_5		PLA_BP_5 | ||||
| #define USB_BP_6		PLA_BP_6 | ||||
| #define USB_BP_7		PLA_BP_7 | ||||
| #define USB_BP_EN		PLA_BP_EN	/* RTL8153A */ | ||||
| #define USB_BP_8		0xfc38		/* RTL8153B */ | ||||
| #define USB_BP_9		0xfc3a | ||||
| #define USB_BP_10		0xfc3c | ||||
| #define USB_BP_11		0xfc3e | ||||
| @ -346,7 +358,12 @@ | ||||
| /* PLA_INDICATE_FALG */ | ||||
| #define UPCOMING_RUNTIME_D3	BIT(0) | ||||
| 
 | ||||
| /* PLA_MACDBG_PRE and PLA_MACDBG_POST */ | ||||
| #define DEBUG_OE		BIT(0) | ||||
| #define DEBUG_LTSSM		0x0082 | ||||
| 
 | ||||
| /* PLA_EXTRA_STATUS */ | ||||
| #define U3P3_CHECK_EN		BIT(7)	/* RTL_VER_05 only */ | ||||
| #define LINK_CHANGE_FLAG	BIT(8) | ||||
| 
 | ||||
| /* USB_USB2PHY */ | ||||
| @ -368,6 +385,12 @@ | ||||
| #define STAT_SPEED_HIGH		0x0000 | ||||
| #define STAT_SPEED_FULL		0x0002 | ||||
| 
 | ||||
| /* USB_FW_FIX_EN0 */ | ||||
| #define FW_FIX_SUSPEND		BIT(14) | ||||
| 
 | ||||
| /* USB_FW_FIX_EN1 */ | ||||
| #define FW_IP_RESET_EN		BIT(9) | ||||
| 
 | ||||
| /* USB_LPM_CONFIG */ | ||||
| #define LPM_U1U2_EN		BIT(0) | ||||
| 
 | ||||
| @ -392,12 +415,24 @@ | ||||
| #define OWN_UPDATE		BIT(0) | ||||
| #define OWN_CLEAR		BIT(1) | ||||
| 
 | ||||
| /* USB_FW_TASK */ | ||||
| #define FC_PATCH_TASK		BIT(1) | ||||
| 
 | ||||
| /* USB_UPS_CTRL */ | ||||
| #define POWER_CUT		0x0100 | ||||
| 
 | ||||
| /* USB_PM_CTRL_STATUS */ | ||||
| #define RESUME_INDICATE		0x0001 | ||||
| 
 | ||||
| /* USB_CSTMR */ | ||||
| #define FORCE_SUPER		BIT(0) | ||||
| 
 | ||||
| /* USB_FW_CTRL */ | ||||
| #define FLOW_CTRL_PATCH_OPT	BIT(1) | ||||
| 
 | ||||
| /* USB_FC_TIMER */ | ||||
| #define CTRL_TIMER_EN		BIT(15) | ||||
| 
 | ||||
| /* USB_USB_CTRL */ | ||||
| #define RX_AGG_DISABLE		0x0010 | ||||
| #define RX_ZERO_EN		0x0080 | ||||
| @ -419,6 +454,9 @@ | ||||
| #define COALESCE_HIGH		250000U | ||||
| #define COALESCE_SLOW		524280U | ||||
| 
 | ||||
| /* USB_WDT1_CTRL */ | ||||
| #define WTD1_EN			BIT(0) | ||||
| 
 | ||||
| /* USB_WDT11_CTRL */ | ||||
| #define TIMER11_EN		0x0001 | ||||
| 
 | ||||
| @ -570,6 +608,8 @@ enum spd_duplex { | ||||
| #define EFUSE			0xcfdb | ||||
| #define PASS_THRU_MASK		0x1 | ||||
| 
 | ||||
| #define BP4_SUPER_ONLY		0x1578	/* RTL_VER_04 only */ | ||||
| 
 | ||||
| enum rtl_register_content { | ||||
| 	_1000bps	= 0x10, | ||||
| 	_100bps		= 0x08, | ||||
| @ -766,6 +806,19 @@ struct r8152 { | ||||
| 		u32 ctap_short_off:1; | ||||
| 	} ups_info; | ||||
| 
 | ||||
| #define RTL_VER_SIZE		32 | ||||
| 
 | ||||
| 	struct rtl_fw { | ||||
| 		const char *fw_name; | ||||
| 		const struct firmware *fw; | ||||
| 
 | ||||
| 		char version[RTL_VER_SIZE]; | ||||
| 		int (*pre_fw)(struct r8152 *tp); | ||||
| 		int (*post_fw)(struct r8152 *tp); | ||||
| 
 | ||||
| 		bool retry; | ||||
| 	} rtl_fw; | ||||
| 
 | ||||
| 	atomic_t rx_count; | ||||
| 
 | ||||
| 	bool eee_en; | ||||
| @ -788,6 +841,76 @@ struct r8152 { | ||||
| 	u8 autoneg; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct fw_block - block type and total length | ||||
|  * @type: type of the current block, such as RTL_FW_END, RTL_FW_PLA, | ||||
|  *	RTL_FW_USB and so on. | ||||
|  * @length: total length of the current block. | ||||
|  */ | ||||
| struct fw_block { | ||||
| 	__le32 type; | ||||
| 	__le32 length; | ||||
| } __packed; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct fw_header - header of the firmware file | ||||
|  * @checksum: checksum of sha256 which is calculated from the whole file | ||||
|  *	except the checksum field of the file. That is, calculate sha256 | ||||
|  *	from the version field to the end of the file. | ||||
|  * @version: version of this firmware. | ||||
|  * @blocks: the first firmware block of the file | ||||
|  */ | ||||
| struct fw_header { | ||||
| 	u8 checksum[32]; | ||||
| 	char version[RTL_VER_SIZE]; | ||||
| 	struct fw_block blocks[0]; | ||||
| } __packed; | ||||
| 
 | ||||
| /**
 | ||||
|  * struct fw_type_1 - a firmware block used by RTL_FW_PLA and RTL_FW_USB. | ||||
|  *	The layout of the firmware block is: | ||||
|  *	<struct fw_type_1> + <info> + <firmware data>. | ||||
|  * @fw_offset: offset of the firmware binary data. The start address of | ||||
|  *	the data would be the address of struct fw_type_1 + @fw_offset. | ||||
|  * @fw_reg: the register to load the firmware. Depends on chip. | ||||
|  * @bp_ba_addr: the register to write break point base address. Depends on | ||||
|  *	chip. | ||||
|  * @bp_ba_value: break point base address. Depends on chip. | ||||
|  * @bp_en_addr: the register to write break point enabled mask. Depends | ||||
|  *	on chip. | ||||
|  * @bp_en_value: break point enabled mask. Depends on the firmware. | ||||
|  * @bp_start: the start register of break points. Depends on chip. | ||||
|  * @bp_num: the break point number which needs to be set for this firmware. | ||||
|  *	Depends on the firmware. | ||||
|  * @bp: break points. Depends on firmware. | ||||
|  * @fw_ver_reg: the register to store the fw version. | ||||
|  * @fw_ver_data: the firmware version of the current type. | ||||
|  * @info: additional information for debugging, and is followed by the | ||||
|  *	binary data of firmware. | ||||
|  */ | ||||
| struct fw_type_1 { | ||||
| 	struct fw_block blk_hdr; | ||||
| 	__le16 fw_offset; | ||||
| 	__le16 fw_reg; | ||||
| 	__le16 bp_ba_addr; | ||||
| 	__le16 bp_ba_value; | ||||
| 	__le16 bp_en_addr; | ||||
| 	__le16 bp_en_value; | ||||
| 	__le16 bp_start; | ||||
| 	__le16 bp_num; | ||||
| 	__le16 bp[16]; /* any value determined by firmware */ | ||||
| 	__le32 reserved; | ||||
| 	__le16 fw_ver_reg; | ||||
| 	u8 fw_ver_data; | ||||
| 	char info[0]; | ||||
| } __packed; | ||||
| 
 | ||||
| enum rtl_fw_type { | ||||
| 	RTL_FW_END = 0, | ||||
| 	RTL_FW_PLA, | ||||
| 	RTL_FW_USB, | ||||
| }; | ||||
| 
 | ||||
| enum rtl_version { | ||||
| 	RTL_VER_UNKNOWN = 0, | ||||
| 	RTL_VER_01, | ||||
| @ -3226,6 +3349,435 @@ static void rtl_reset_bmu(struct r8152 *tp) | ||||
| 	ocp_write_byte(tp, MCU_TYPE_USB, USB_BMU_RESET, ocp_data); | ||||
| } | ||||
| 
 | ||||
| /* Clear the bp to stop the firmware before loading a new one */ | ||||
| static void rtl_clear_bp(struct r8152 *tp, u16 type) | ||||
| { | ||||
| 	switch (tp->version) { | ||||
| 	case RTL_VER_01: | ||||
| 	case RTL_VER_02: | ||||
| 	case RTL_VER_07: | ||||
| 		break; | ||||
| 	case RTL_VER_03: | ||||
| 	case RTL_VER_04: | ||||
| 	case RTL_VER_05: | ||||
| 	case RTL_VER_06: | ||||
| 		ocp_write_byte(tp, type, PLA_BP_EN, 0); | ||||
| 		break; | ||||
| 	case RTL_VER_08: | ||||
| 	case RTL_VER_09: | ||||
| 	default: | ||||
| 		if (type == MCU_TYPE_USB) { | ||||
| 			ocp_write_byte(tp, MCU_TYPE_USB, USB_BP2_EN, 0); | ||||
| 
 | ||||
| 			ocp_write_word(tp, MCU_TYPE_USB, USB_BP_8, 0); | ||||
| 			ocp_write_word(tp, MCU_TYPE_USB, USB_BP_9, 0); | ||||
| 			ocp_write_word(tp, MCU_TYPE_USB, USB_BP_10, 0); | ||||
| 			ocp_write_word(tp, MCU_TYPE_USB, USB_BP_11, 0); | ||||
| 			ocp_write_word(tp, MCU_TYPE_USB, USB_BP_12, 0); | ||||
| 			ocp_write_word(tp, MCU_TYPE_USB, USB_BP_13, 0); | ||||
| 			ocp_write_word(tp, MCU_TYPE_USB, USB_BP_14, 0); | ||||
| 			ocp_write_word(tp, MCU_TYPE_USB, USB_BP_15, 0); | ||||
| 		} else { | ||||
| 			ocp_write_byte(tp, MCU_TYPE_PLA, PLA_BP_EN, 0); | ||||
| 		} | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	ocp_write_word(tp, type, PLA_BP_0, 0); | ||||
| 	ocp_write_word(tp, type, PLA_BP_1, 0); | ||||
| 	ocp_write_word(tp, type, PLA_BP_2, 0); | ||||
| 	ocp_write_word(tp, type, PLA_BP_3, 0); | ||||
| 	ocp_write_word(tp, type, PLA_BP_4, 0); | ||||
| 	ocp_write_word(tp, type, PLA_BP_5, 0); | ||||
| 	ocp_write_word(tp, type, PLA_BP_6, 0); | ||||
| 	ocp_write_word(tp, type, PLA_BP_7, 0); | ||||
| 
 | ||||
| 	/* wait 3 ms to make sure the firmware is stopped */ | ||||
| 	usleep_range(3000, 6000); | ||||
| 	ocp_write_word(tp, type, PLA_BP_BA, 0); | ||||
| } | ||||
| 
 | ||||
| static bool rtl8152_is_fw_type1_ok(struct r8152 *tp, struct fw_type_1 *type1) | ||||
| { | ||||
| 	u16 fw_reg, bp_ba_addr, bp_en_addr, bp_start; | ||||
| 	bool rc = false; | ||||
| 	u32 length, type; | ||||
| 	int i, max_bp; | ||||
| 
 | ||||
| 	type = __le32_to_cpu(type1->blk_hdr.type); | ||||
| 	if (type == RTL_FW_PLA) { | ||||
| 		switch (tp->version) { | ||||
| 		case RTL_VER_01: | ||||
| 		case RTL_VER_02: | ||||
| 		case RTL_VER_07: | ||||
| 			fw_reg = 0xf800; | ||||
| 			bp_ba_addr = PLA_BP_BA; | ||||
| 			bp_en_addr = 0; | ||||
| 			bp_start = PLA_BP_0; | ||||
| 			max_bp = 8; | ||||
| 			break; | ||||
| 		case RTL_VER_03: | ||||
| 		case RTL_VER_04: | ||||
| 		case RTL_VER_05: | ||||
| 		case RTL_VER_06: | ||||
| 		case RTL_VER_08: | ||||
| 		case RTL_VER_09: | ||||
| 			fw_reg = 0xf800; | ||||
| 			bp_ba_addr = PLA_BP_BA; | ||||
| 			bp_en_addr = PLA_BP_EN; | ||||
| 			bp_start = PLA_BP_0; | ||||
| 			max_bp = 8; | ||||
| 			break; | ||||
| 		default: | ||||
| 			goto out; | ||||
| 		} | ||||
| 	} else if (type == RTL_FW_USB) { | ||||
| 		switch (tp->version) { | ||||
| 		case RTL_VER_03: | ||||
| 		case RTL_VER_04: | ||||
| 		case RTL_VER_05: | ||||
| 		case RTL_VER_06: | ||||
| 			fw_reg = 0xf800; | ||||
| 			bp_ba_addr = USB_BP_BA; | ||||
| 			bp_en_addr = USB_BP_EN; | ||||
| 			bp_start = USB_BP_0; | ||||
| 			max_bp = 8; | ||||
| 			break; | ||||
| 		case RTL_VER_08: | ||||
| 		case RTL_VER_09: | ||||
| 			fw_reg = 0xe600; | ||||
| 			bp_ba_addr = USB_BP_BA; | ||||
| 			bp_en_addr = USB_BP2_EN; | ||||
| 			bp_start = USB_BP_0; | ||||
| 			max_bp = 16; | ||||
| 			break; | ||||
| 		case RTL_VER_01: | ||||
| 		case RTL_VER_02: | ||||
| 		case RTL_VER_07: | ||||
| 		default: | ||||
| 			goto out; | ||||
| 		} | ||||
| 	} else { | ||||
| 		goto out; | ||||
| 	} | ||||
| 
 | ||||
| 	length = __le32_to_cpu(type1->blk_hdr.length); | ||||
| 	if (length < __le16_to_cpu(type1->fw_offset)) { | ||||
| 		dev_err(&tp->intf->dev, "invalid fw_offset\n"); | ||||
| 		goto out; | ||||
| 	} | ||||
| 
 | ||||
| 	length -= __le16_to_cpu(type1->fw_offset); | ||||
| 	if (length < 4 || (length & 3)) { | ||||
| 		dev_err(&tp->intf->dev, "invalid block length\n"); | ||||
| 		goto out; | ||||
| 	} | ||||
| 
 | ||||
| 	if (__le16_to_cpu(type1->fw_reg) != fw_reg) { | ||||
| 		dev_err(&tp->intf->dev, "invalid register to load firmware\n"); | ||||
| 		goto out; | ||||
| 	} | ||||
| 
 | ||||
| 	if (__le16_to_cpu(type1->bp_ba_addr) != bp_ba_addr) { | ||||
| 		dev_err(&tp->intf->dev, "invalid base address register\n"); | ||||
| 		goto out; | ||||
| 	} | ||||
| 
 | ||||
| 	if (__le16_to_cpu(type1->bp_en_addr) != bp_en_addr) { | ||||
| 		dev_err(&tp->intf->dev, "invalid enabled mask register\n"); | ||||
| 		goto out; | ||||
| 	} | ||||
| 
 | ||||
| 	if (__le16_to_cpu(type1->bp_start) != bp_start) { | ||||
| 		dev_err(&tp->intf->dev, | ||||
| 			"invalid start register of break point\n"); | ||||
| 		goto out; | ||||
| 	} | ||||
| 
 | ||||
| 	if (__le16_to_cpu(type1->bp_num) > max_bp) { | ||||
| 		dev_err(&tp->intf->dev, "invalid break point number\n"); | ||||
| 		goto out; | ||||
| 	} | ||||
| 
 | ||||
| 	for (i = __le16_to_cpu(type1->bp_num); i < max_bp; i++) { | ||||
| 		if (type1->bp[i]) { | ||||
| 			dev_err(&tp->intf->dev, "unused bp%u is not zero\n", i); | ||||
| 			goto out; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	rc = true; | ||||
| out: | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| /* Verify the checksum for the firmware file. It is calculated from the version
 | ||||
|  * field to the end of the file. Compare the result with the checksum field to | ||||
|  * make sure the file is correct. | ||||
|  */ | ||||
| static long rtl8152_fw_verify_checksum(struct r8152 *tp, | ||||
| 				       struct fw_header *fw_hdr, size_t size) | ||||
| { | ||||
| 	unsigned char checksum[sizeof(fw_hdr->checksum)]; | ||||
| 	struct crypto_shash *alg; | ||||
| 	struct shash_desc *sdesc; | ||||
| 	size_t len; | ||||
| 	long rc; | ||||
| 
 | ||||
| 	alg = crypto_alloc_shash("sha256", 0, 0); | ||||
| 	if (IS_ERR(alg)) { | ||||
| 		rc = PTR_ERR(alg); | ||||
| 		goto out; | ||||
| 	} | ||||
| 
 | ||||
| 	if (crypto_shash_digestsize(alg) != sizeof(fw_hdr->checksum)) { | ||||
| 		rc = -EFAULT; | ||||
| 		dev_err(&tp->intf->dev, "digestsize incorrect (%u)\n", | ||||
| 			crypto_shash_digestsize(alg)); | ||||
| 		goto free_shash; | ||||
| 	} | ||||
| 
 | ||||
| 	len = sizeof(*sdesc) + crypto_shash_descsize(alg); | ||||
| 	sdesc = kmalloc(len, GFP_KERNEL); | ||||
| 	if (!sdesc) { | ||||
| 		rc = -ENOMEM; | ||||
| 		goto free_shash; | ||||
| 	} | ||||
| 	sdesc->tfm = alg; | ||||
| 
 | ||||
| 	len = size - sizeof(fw_hdr->checksum); | ||||
| 	rc = crypto_shash_digest(sdesc, fw_hdr->version, len, checksum); | ||||
| 	kfree(sdesc); | ||||
| 	if (rc) | ||||
| 		goto free_shash; | ||||
| 
 | ||||
| 	if (memcmp(fw_hdr->checksum, checksum, sizeof(fw_hdr->checksum))) { | ||||
| 		dev_err(&tp->intf->dev, "checksum fail\n"); | ||||
| 		rc = -EFAULT; | ||||
| 	} | ||||
| 
 | ||||
| free_shash: | ||||
| 	crypto_free_shash(alg); | ||||
| out: | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| static long rtl8152_check_firmware(struct r8152 *tp, struct rtl_fw *rtl_fw) | ||||
| { | ||||
| 	const struct firmware *fw = rtl_fw->fw; | ||||
| 	struct fw_header *fw_hdr = (struct fw_header *)fw->data; | ||||
| 	struct fw_type_1 *pla = NULL, *usb = NULL; | ||||
| 	long ret = -EFAULT; | ||||
| 	int i; | ||||
| 
 | ||||
| 	if (fw->size < sizeof(*fw_hdr)) { | ||||
| 		dev_err(&tp->intf->dev, "file too small\n"); | ||||
| 		goto fail; | ||||
| 	} | ||||
| 
 | ||||
| 	ret = rtl8152_fw_verify_checksum(tp, fw_hdr, fw->size); | ||||
| 	if (ret) | ||||
| 		goto fail; | ||||
| 
 | ||||
| 	ret = -EFAULT; | ||||
| 
 | ||||
| 	for (i = sizeof(*fw_hdr); i < fw->size;) { | ||||
| 		struct fw_block *block = (struct fw_block *)&fw->data[i]; | ||||
| 		u32 type; | ||||
| 
 | ||||
| 		if ((i + sizeof(*block)) > fw->size) | ||||
| 			goto fail; | ||||
| 
 | ||||
| 		type = __le32_to_cpu(block->type); | ||||
| 		switch (type) { | ||||
| 		case RTL_FW_END: | ||||
| 			if (__le32_to_cpu(block->length) != sizeof(*block)) | ||||
| 				goto fail; | ||||
| 			goto success; | ||||
| 		case RTL_FW_PLA: | ||||
| 			if (pla) { | ||||
| 				dev_err(&tp->intf->dev, | ||||
| 					"multiple PLA firmware encountered"); | ||||
| 				goto fail; | ||||
| 			} | ||||
| 
 | ||||
| 			pla = (struct fw_type_1 *)block; | ||||
| 			if (!rtl8152_is_fw_type1_ok(tp, pla)) { | ||||
| 				dev_err(&tp->intf->dev, | ||||
| 					"load PLA firmware failed\n"); | ||||
| 				goto fail; | ||||
| 			} | ||||
| 			break; | ||||
| 		case RTL_FW_USB: | ||||
| 			if (usb) { | ||||
| 				dev_err(&tp->intf->dev, | ||||
| 					"multiple USB firmware encountered"); | ||||
| 				goto fail; | ||||
| 			} | ||||
| 
 | ||||
| 			usb = (struct fw_type_1 *)block; | ||||
| 			if (!rtl8152_is_fw_type1_ok(tp, usb)) { | ||||
| 				dev_err(&tp->intf->dev, | ||||
| 					"load USB firmware failed\n"); | ||||
| 				goto fail; | ||||
| 			} | ||||
| 			break; | ||||
| 		default: | ||||
| 			dev_warn(&tp->intf->dev, "Unknown type %u is found\n", | ||||
| 				 type); | ||||
| 			break; | ||||
| 		} | ||||
| 
 | ||||
| 		/* next block */ | ||||
| 		i += ALIGN(__le32_to_cpu(block->length), 8); | ||||
| 	} | ||||
| 
 | ||||
| success: | ||||
| 	return 0; | ||||
| fail: | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| static void rtl8152_fw_type_1_apply(struct r8152 *tp, struct fw_type_1 *type1) | ||||
| { | ||||
| 	u16 bp_en_addr, bp_index, type, bp_num, fw_ver_reg; | ||||
| 	u32 length; | ||||
| 	u8 *data; | ||||
| 	int i; | ||||
| 
 | ||||
| 	switch (__le32_to_cpu(type1->blk_hdr.type)) { | ||||
| 	case RTL_FW_PLA: | ||||
| 		type = MCU_TYPE_PLA; | ||||
| 		break; | ||||
| 	case RTL_FW_USB: | ||||
| 		type = MCU_TYPE_USB; | ||||
| 		break; | ||||
| 	default: | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	rtl_clear_bp(tp, type); | ||||
| 
 | ||||
| 	/* Enable backup/restore of MACDBG. This is required after clearing PLA
 | ||||
| 	 * break points and before applying the PLA firmware. | ||||
| 	 */ | ||||
| 	if (tp->version == RTL_VER_04 && type == MCU_TYPE_PLA && | ||||
| 	    !(ocp_read_word(tp, MCU_TYPE_PLA, PLA_MACDBG_POST) & DEBUG_OE)) { | ||||
| 		ocp_write_word(tp, MCU_TYPE_PLA, PLA_MACDBG_PRE, DEBUG_LTSSM); | ||||
| 		ocp_write_word(tp, MCU_TYPE_PLA, PLA_MACDBG_POST, DEBUG_LTSSM); | ||||
| 	} | ||||
| 
 | ||||
| 	length = __le32_to_cpu(type1->blk_hdr.length); | ||||
| 	length -= __le16_to_cpu(type1->fw_offset); | ||||
| 
 | ||||
| 	data = (u8 *)type1; | ||||
| 	data += __le16_to_cpu(type1->fw_offset); | ||||
| 
 | ||||
| 	generic_ocp_write(tp, __le16_to_cpu(type1->fw_reg), 0xff, length, data, | ||||
| 			  type); | ||||
| 
 | ||||
| 	ocp_write_word(tp, type, __le16_to_cpu(type1->bp_ba_addr), | ||||
| 		       __le16_to_cpu(type1->bp_ba_value)); | ||||
| 
 | ||||
| 	bp_index = __le16_to_cpu(type1->bp_start); | ||||
| 	bp_num = __le16_to_cpu(type1->bp_num); | ||||
| 	for (i = 0; i < bp_num; i++) { | ||||
| 		ocp_write_word(tp, type, bp_index, __le16_to_cpu(type1->bp[i])); | ||||
| 		bp_index += 2; | ||||
| 	} | ||||
| 
 | ||||
| 	bp_en_addr = __le16_to_cpu(type1->bp_en_addr); | ||||
| 	if (bp_en_addr) | ||||
| 		ocp_write_word(tp, type, bp_en_addr, | ||||
| 			       __le16_to_cpu(type1->bp_en_value)); | ||||
| 
 | ||||
| 	fw_ver_reg = __le16_to_cpu(type1->fw_ver_reg); | ||||
| 	if (fw_ver_reg) | ||||
| 		ocp_write_byte(tp, MCU_TYPE_USB, fw_ver_reg, | ||||
| 			       type1->fw_ver_data); | ||||
| 
 | ||||
| 	dev_dbg(&tp->intf->dev, "successfully applied %s\n", type1->info); | ||||
| } | ||||
| 
 | ||||
| static void rtl8152_apply_firmware(struct r8152 *tp) | ||||
| { | ||||
| 	struct rtl_fw *rtl_fw = &tp->rtl_fw; | ||||
| 	const struct firmware *fw = rtl_fw->fw; | ||||
| 	struct fw_header *fw_hdr = (struct fw_header *)fw->data; | ||||
| 	int i; | ||||
| 
 | ||||
| 	if (IS_ERR_OR_NULL(rtl_fw->fw)) | ||||
| 		return; | ||||
| 
 | ||||
| 	if (rtl_fw->pre_fw) | ||||
| 		rtl_fw->pre_fw(tp); | ||||
| 
 | ||||
| 	for (i = offsetof(struct fw_header, blocks); i < fw->size;) { | ||||
| 		struct fw_block *block = (struct fw_block *)&fw->data[i]; | ||||
| 
 | ||||
| 		switch (__le32_to_cpu(block->type)) { | ||||
| 		case RTL_FW_END: | ||||
| 			goto post_fw; | ||||
| 		case RTL_FW_PLA: | ||||
| 		case RTL_FW_USB: | ||||
| 			rtl8152_fw_type_1_apply(tp, (struct fw_type_1 *)block); | ||||
| 			break; | ||||
| 		default: | ||||
| 			break; | ||||
| 		} | ||||
| 
 | ||||
| 		i += ALIGN(__le32_to_cpu(block->length), 8); | ||||
| 	} | ||||
| 
 | ||||
| post_fw: | ||||
| 	if (rtl_fw->post_fw) | ||||
| 		rtl_fw->post_fw(tp); | ||||
| 
 | ||||
| 	strscpy(rtl_fw->version, fw_hdr->version, RTL_VER_SIZE); | ||||
| 	dev_info(&tp->intf->dev, "load %s successfully\n", rtl_fw->version); | ||||
| } | ||||
| 
 | ||||
| static void rtl8152_release_firmware(struct r8152 *tp) | ||||
| { | ||||
| 	struct rtl_fw *rtl_fw = &tp->rtl_fw; | ||||
| 
 | ||||
| 	if (!IS_ERR_OR_NULL(rtl_fw->fw)) { | ||||
| 		release_firmware(rtl_fw->fw); | ||||
| 		rtl_fw->fw = NULL; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static int rtl8152_request_firmware(struct r8152 *tp) | ||||
| { | ||||
| 	struct rtl_fw *rtl_fw = &tp->rtl_fw; | ||||
| 	long rc; | ||||
| 
 | ||||
| 	if (rtl_fw->fw || !rtl_fw->fw_name) { | ||||
| 		dev_info(&tp->intf->dev, "skip request firmware\n"); | ||||
| 		rc = 0; | ||||
| 		goto result; | ||||
| 	} | ||||
| 
 | ||||
| 	rc = request_firmware(&rtl_fw->fw, rtl_fw->fw_name, &tp->intf->dev); | ||||
| 	if (rc < 0) | ||||
| 		goto result; | ||||
| 
 | ||||
| 	rc = rtl8152_check_firmware(tp, rtl_fw); | ||||
| 	if (rc < 0) | ||||
| 		release_firmware(rtl_fw->fw); | ||||
| 
 | ||||
| result: | ||||
| 	if (rc) { | ||||
| 		rtl_fw->fw = ERR_PTR(rc); | ||||
| 
 | ||||
| 		dev_warn(&tp->intf->dev, | ||||
| 			 "unable to load firmware patch %s (%ld)\n", | ||||
| 			 rtl_fw->fw_name, rc); | ||||
| 	} | ||||
| 
 | ||||
| 	return rc; | ||||
| } | ||||
| 
 | ||||
| static void r8152_aldps_en(struct r8152 *tp, bool enable) | ||||
| { | ||||
| 	if (enable) { | ||||
| @ -3370,6 +3922,7 @@ static void rtl8152_disable(struct r8152 *tp) | ||||
| 
 | ||||
| static void r8152b_hw_phy_cfg(struct r8152 *tp) | ||||
| { | ||||
| 	rtl8152_apply_firmware(tp); | ||||
| 	rtl_eee_enable(tp, tp->eee_en); | ||||
| 	r8152_aldps_en(tp, true); | ||||
| 	r8152b_enable_fc(tp); | ||||
| @ -3524,6 +4077,126 @@ static int r8153_patch_request(struct r8152 *tp, bool request) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static int r8153_pre_firmware_1(struct r8152 *tp) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	/* Wait till the WTD timer is ready. It would take at most 104 ms. */ | ||||
| 	for (i = 0; i < 104; i++) { | ||||
| 		u32 ocp_data = ocp_read_byte(tp, MCU_TYPE_USB, USB_WDT1_CTRL); | ||||
| 
 | ||||
| 		if (!(ocp_data & WTD1_EN)) | ||||
| 			break; | ||||
| 		usleep_range(1000, 2000); | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int r8153_post_firmware_1(struct r8152 *tp) | ||||
| { | ||||
| 	/* set USB_BP_4 to support USB_SPEED_SUPER only */ | ||||
| 	if (ocp_read_byte(tp, MCU_TYPE_USB, USB_CSTMR) & FORCE_SUPER) | ||||
| 		ocp_write_word(tp, MCU_TYPE_USB, USB_BP_4, BP4_SUPER_ONLY); | ||||
| 
 | ||||
| 	/* reset UPHY timer to 36 ms */ | ||||
| 	ocp_write_word(tp, MCU_TYPE_PLA, PLA_UPHY_TIMER, 36000 / 16); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int r8153_pre_firmware_2(struct r8152 *tp) | ||||
| { | ||||
| 	u32 ocp_data; | ||||
| 
 | ||||
| 	r8153_pre_firmware_1(tp); | ||||
| 
 | ||||
| 	ocp_data = ocp_read_word(tp, MCU_TYPE_USB, USB_FW_FIX_EN0); | ||||
| 	ocp_data &= ~FW_FIX_SUSPEND; | ||||
| 	ocp_write_word(tp, MCU_TYPE_USB, USB_FW_FIX_EN0, ocp_data); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int r8153_post_firmware_2(struct r8152 *tp) | ||||
| { | ||||
| 	u32 ocp_data; | ||||
| 
 | ||||
| 	/* enable bp0 if support USB_SPEED_SUPER only */ | ||||
| 	if (ocp_read_byte(tp, MCU_TYPE_USB, USB_CSTMR) & FORCE_SUPER) { | ||||
| 		ocp_data = ocp_read_word(tp, MCU_TYPE_PLA, PLA_BP_EN); | ||||
| 		ocp_data |= BIT(0); | ||||
| 		ocp_write_word(tp, MCU_TYPE_PLA, PLA_BP_EN, ocp_data); | ||||
| 	} | ||||
| 
 | ||||
| 	/* reset UPHY timer to 36 ms */ | ||||
| 	ocp_write_word(tp, MCU_TYPE_PLA, PLA_UPHY_TIMER, 36000 / 16); | ||||
| 
 | ||||
| 	/* enable U3P3 check, set the counter to 4 */ | ||||
| 	ocp_write_word(tp, MCU_TYPE_PLA, PLA_EXTRA_STATUS, U3P3_CHECK_EN | 4); | ||||
| 
 | ||||
| 	ocp_data = ocp_read_word(tp, MCU_TYPE_USB, USB_FW_FIX_EN0); | ||||
| 	ocp_data |= FW_FIX_SUSPEND; | ||||
| 	ocp_write_word(tp, MCU_TYPE_USB, USB_FW_FIX_EN0, ocp_data); | ||||
| 
 | ||||
| 	ocp_data = ocp_read_byte(tp, MCU_TYPE_USB, USB_USB2PHY); | ||||
| 	ocp_data |= USB2PHY_L1 | USB2PHY_SUSPEND; | ||||
| 	ocp_write_byte(tp, MCU_TYPE_USB, USB_USB2PHY, ocp_data); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int r8153_post_firmware_3(struct r8152 *tp) | ||||
| { | ||||
| 	u32 ocp_data; | ||||
| 
 | ||||
| 	ocp_data = ocp_read_byte(tp, MCU_TYPE_USB, USB_USB2PHY); | ||||
| 	ocp_data |= USB2PHY_L1 | USB2PHY_SUSPEND; | ||||
| 	ocp_write_byte(tp, MCU_TYPE_USB, USB_USB2PHY, ocp_data); | ||||
| 
 | ||||
| 	ocp_data = ocp_read_word(tp, MCU_TYPE_USB, USB_FW_FIX_EN1); | ||||
| 	ocp_data |= FW_IP_RESET_EN; | ||||
| 	ocp_write_word(tp, MCU_TYPE_USB, USB_FW_FIX_EN1, ocp_data); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int r8153b_pre_firmware_1(struct r8152 *tp) | ||||
| { | ||||
| 	/* enable fc timer and set timer to 1 second. */ | ||||
| 	ocp_write_word(tp, MCU_TYPE_USB, USB_FC_TIMER, | ||||
| 		       CTRL_TIMER_EN | (1000 / 8)); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int r8153b_post_firmware_1(struct r8152 *tp) | ||||
| { | ||||
| 	u32 ocp_data; | ||||
| 
 | ||||
| 	/* enable bp0 for RTL8153-BND */ | ||||
| 	ocp_data = ocp_read_byte(tp, MCU_TYPE_USB, USB_MISC_1); | ||||
| 	if (ocp_data & BND_MASK) { | ||||
| 		ocp_data = ocp_read_word(tp, MCU_TYPE_PLA, PLA_BP_EN); | ||||
| 		ocp_data |= BIT(0); | ||||
| 		ocp_write_word(tp, MCU_TYPE_PLA, PLA_BP_EN, ocp_data); | ||||
| 	} | ||||
| 
 | ||||
| 	ocp_data = ocp_read_word(tp, MCU_TYPE_USB, USB_FW_CTRL); | ||||
| 	ocp_data |= FLOW_CTRL_PATCH_OPT; | ||||
| 	ocp_write_word(tp, MCU_TYPE_USB, USB_FW_CTRL, ocp_data); | ||||
| 
 | ||||
| 	ocp_data = ocp_read_word(tp, MCU_TYPE_USB, USB_FW_TASK); | ||||
| 	ocp_data |= FC_PATCH_TASK; | ||||
| 	ocp_write_word(tp, MCU_TYPE_USB, USB_FW_TASK, ocp_data); | ||||
| 
 | ||||
| 	ocp_data = ocp_read_word(tp, MCU_TYPE_USB, USB_FW_FIX_EN1); | ||||
| 	ocp_data |= FW_IP_RESET_EN; | ||||
| 	ocp_write_word(tp, MCU_TYPE_USB, USB_FW_FIX_EN1, ocp_data); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static void r8153_aldps_en(struct r8152 *tp, bool enable) | ||||
| { | ||||
| 	u16 data; | ||||
| @ -3558,6 +4231,8 @@ static void r8153_hw_phy_cfg(struct r8152 *tp) | ||||
| 	/* disable EEE before updating the PHY parameters */ | ||||
| 	rtl_eee_enable(tp, false); | ||||
| 
 | ||||
| 	rtl8152_apply_firmware(tp); | ||||
| 
 | ||||
| 	if (tp->version == RTL_VER_03) { | ||||
| 		data = ocp_reg_read(tp, OCP_EEE_CFG); | ||||
| 		data &= ~CTAP_SHORT_EN; | ||||
| @ -3630,6 +4305,8 @@ static void r8153b_hw_phy_cfg(struct r8152 *tp) | ||||
| 	/* disable EEE before updating the PHY parameters */ | ||||
| 	rtl_eee_enable(tp, false); | ||||
| 
 | ||||
| 	rtl8152_apply_firmware(tp); | ||||
| 
 | ||||
| 	r8153b_green_en(tp, test_bit(GREEN_ETHERNET, &tp->flags)); | ||||
| 
 | ||||
| 	data = sram_read(tp, SRAM_GREEN_CFG); | ||||
| @ -4156,11 +4833,22 @@ static void rtl_hw_phy_work_func_t(struct work_struct *work) | ||||
| 
 | ||||
| 	mutex_lock(&tp->control); | ||||
| 
 | ||||
| 	if (rtl8152_request_firmware(tp) == -ENODEV && tp->rtl_fw.retry) { | ||||
| 		tp->rtl_fw.retry = false; | ||||
| 		tp->rtl_fw.fw = NULL; | ||||
| 
 | ||||
| 		/* Delay execution in case request_firmware() is not ready yet.
 | ||||
| 		 */ | ||||
| 		queue_delayed_work(system_long_wq, &tp->hw_phy_work, HZ * 10); | ||||
| 		goto ignore_once; | ||||
| 	} | ||||
| 
 | ||||
| 	tp->rtl_ops.hw_phy_cfg(tp); | ||||
| 
 | ||||
| 	rtl8152_set_speed(tp, tp->autoneg, tp->speed, tp->duplex, | ||||
| 			  tp->advertising); | ||||
| 
 | ||||
| ignore_once: | ||||
| 	mutex_unlock(&tp->control); | ||||
| 
 | ||||
| 	usb_autopm_put_interface(tp->intf); | ||||
| @ -4198,6 +4886,11 @@ static int rtl8152_open(struct net_device *netdev) | ||||
| 	struct r8152 *tp = netdev_priv(netdev); | ||||
| 	int res = 0; | ||||
| 
 | ||||
| 	if (work_busy(&tp->hw_phy_work.work) & WORK_BUSY_PENDING) { | ||||
| 		cancel_delayed_work_sync(&tp->hw_phy_work); | ||||
| 		rtl_hw_phy_work_func_t(&tp->hw_phy_work.work); | ||||
| 	} | ||||
| 
 | ||||
| 	res = alloc_all_mem(tp); | ||||
| 	if (res) | ||||
| 		goto out; | ||||
| @ -4844,6 +5537,9 @@ static void rtl8152_get_drvinfo(struct net_device *netdev, | ||||
| 	strlcpy(info->driver, MODULENAME, sizeof(info->driver)); | ||||
| 	strlcpy(info->version, DRIVER_VERSION, sizeof(info->version)); | ||||
| 	usb_make_path(tp->udev, info->bus_info, sizeof(info->bus_info)); | ||||
| 	if (!IS_ERR_OR_NULL(tp->rtl_fw.fw)) | ||||
| 		strlcpy(info->fw_version, tp->rtl_fw.version, | ||||
| 			sizeof(info->fw_version)); | ||||
| } | ||||
| 
 | ||||
| static | ||||
| @ -5468,6 +6164,47 @@ static int rtl_ops_init(struct r8152 *tp) | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| #define FIRMWARE_8153A_2	"rtl_nic/rtl8153a-2.fw" | ||||
| #define FIRMWARE_8153A_3	"rtl_nic/rtl8153a-3.fw" | ||||
| #define FIRMWARE_8153A_4	"rtl_nic/rtl8153a-4.fw" | ||||
| #define FIRMWARE_8153B_2	"rtl_nic/rtl8153b-2.fw" | ||||
| 
 | ||||
| MODULE_FIRMWARE(FIRMWARE_8153A_2); | ||||
| MODULE_FIRMWARE(FIRMWARE_8153A_3); | ||||
| MODULE_FIRMWARE(FIRMWARE_8153A_4); | ||||
| MODULE_FIRMWARE(FIRMWARE_8153B_2); | ||||
| 
 | ||||
| static int rtl_fw_init(struct r8152 *tp) | ||||
| { | ||||
| 	struct rtl_fw *rtl_fw = &tp->rtl_fw; | ||||
| 
 | ||||
| 	switch (tp->version) { | ||||
| 	case RTL_VER_04: | ||||
| 		rtl_fw->fw_name		= FIRMWARE_8153A_2; | ||||
| 		rtl_fw->pre_fw		= r8153_pre_firmware_1; | ||||
| 		rtl_fw->post_fw		= r8153_post_firmware_1; | ||||
| 		break; | ||||
| 	case RTL_VER_05: | ||||
| 		rtl_fw->fw_name		= FIRMWARE_8153A_3; | ||||
| 		rtl_fw->pre_fw		= r8153_pre_firmware_2; | ||||
| 		rtl_fw->post_fw		= r8153_post_firmware_2; | ||||
| 		break; | ||||
| 	case RTL_VER_06: | ||||
| 		rtl_fw->fw_name		= FIRMWARE_8153A_4; | ||||
| 		rtl_fw->post_fw		= r8153_post_firmware_3; | ||||
| 		break; | ||||
| 	case RTL_VER_09: | ||||
| 		rtl_fw->fw_name		= FIRMWARE_8153B_2; | ||||
| 		rtl_fw->pre_fw		= r8153b_pre_firmware_1; | ||||
| 		rtl_fw->post_fw		= r8153b_post_firmware_1; | ||||
| 		break; | ||||
| 	default: | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static u8 rtl_get_version(struct usb_interface *intf) | ||||
| { | ||||
| 	struct usb_device *udev = interface_to_usbdev(intf); | ||||
| @ -5575,6 +6312,8 @@ static int rtl8152_probe(struct usb_interface *intf, | ||||
| 	if (ret) | ||||
| 		goto out; | ||||
| 
 | ||||
| 	rtl_fw_init(tp); | ||||
| 
 | ||||
| 	mutex_init(&tp->control); | ||||
| 	INIT_DELAYED_WORK(&tp->schedule, rtl_work_func_t); | ||||
| 	INIT_DELAYED_WORK(&tp->hw_phy_work, rtl_hw_phy_work_func_t); | ||||
| @ -5646,6 +6385,10 @@ static int rtl8152_probe(struct usb_interface *intf, | ||||
| 	intf->needs_remote_wakeup = 1; | ||||
| 
 | ||||
| 	tp->rtl_ops.init(tp); | ||||
| #if IS_BUILTIN(CONFIG_USB_RTL8152) | ||||
| 	/* Retry in case request_firmware() is not ready yet. */ | ||||
| 	tp->rtl_fw.retry = true; | ||||
| #endif | ||||
| 	queue_delayed_work(system_long_wq, &tp->hw_phy_work, 0); | ||||
| 	set_ethernet_addr(tp); | ||||
| 
 | ||||
| @ -5691,6 +6434,7 @@ static void rtl8152_disconnect(struct usb_interface *intf) | ||||
| 		tasklet_kill(&tp->tx_tl); | ||||
| 		cancel_delayed_work_sync(&tp->hw_phy_work); | ||||
| 		tp->rtl_ops.unload(tp); | ||||
| 		rtl8152_release_firmware(tp); | ||||
| 		free_netdev(tp->netdev); | ||||
| 	} | ||||
| } | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Hayes Wang
						Hayes Wang