Commit 24f97b6a authored by Heiner Kallweit's avatar Heiner Kallweit Committed by Jakub Kicinski

tg3: improve PCI VPD access

When working on the PCI VPD code I also tested with a Broadcom BCM95719
card. tg3 uses internal NVRAM access with this card, so I forced it to
PCI VPD mode for testing. PCI VPD access fails
(i + PCI_VPD_LRDT_TAG_SIZE + j > len) because only TG3_NVM_VPD_LEN (256)
bytes are read, but PCI VPD has 400 bytes on this card.

So add a constant TG3_NVM_PCI_VPD_MAX_LEN that defines the maximum
PCI VPD size. The actual VPD size is returned by pci_read_vpd().
In addition it's not worth looping over pci_read_vpd(). If we miss the
125ms timeout per VPD dword read then definitely something is wrong,
and if the tg3 module loading is killed then there's also not much
benefit in retrying the VPD read.
Signed-off-by: default avatarHeiner Kallweit <hkallweit1@gmail.com>
Reviewed-by: default avatarMichael Chan <michael.chan@broadcom.com>
Link: https://lore.kernel.org/r/cb9e9113-0861-3904-87e0-d4c4ab3c8860@gmail.comSigned-off-by: default avatarJakub Kicinski <kuba@kernel.org>
parent a61e4b60
...@@ -12826,11 +12826,13 @@ static __be32 *tg3_vpd_readblock(struct tg3 *tp, u32 *vpdlen) ...@@ -12826,11 +12826,13 @@ static __be32 *tg3_vpd_readblock(struct tg3 *tp, u32 *vpdlen)
offset = tg3_nvram_logical_addr(tp, offset); offset = tg3_nvram_logical_addr(tp, offset);
} }
}
if (!offset || !len) { if (!offset || !len) {
offset = TG3_NVM_VPD_OFF; offset = TG3_NVM_VPD_OFF;
len = TG3_NVM_VPD_LEN; len = TG3_NVM_VPD_LEN;
}
} else {
len = TG3_NVM_PCI_VPD_MAX_LEN;
} }
buf = kmalloc(len, GFP_KERNEL); buf = kmalloc(len, GFP_KERNEL);
...@@ -12846,26 +12848,16 @@ static __be32 *tg3_vpd_readblock(struct tg3 *tp, u32 *vpdlen) ...@@ -12846,26 +12848,16 @@ static __be32 *tg3_vpd_readblock(struct tg3 *tp, u32 *vpdlen)
if (tg3_nvram_read_be32(tp, offset + i, &buf[i/4])) if (tg3_nvram_read_be32(tp, offset + i, &buf[i/4]))
goto error; goto error;
} }
*vpdlen = len;
} else { } else {
u8 *ptr;
ssize_t cnt; ssize_t cnt;
unsigned int pos = 0;
cnt = pci_read_vpd(tp->pdev, 0, len, (u8 *)buf);
ptr = (u8 *)&buf[0]; if (cnt < 0)
for (i = 0; pos < len && i < 3; i++, pos += cnt, ptr += cnt) {
cnt = pci_read_vpd(tp->pdev, pos,
len - pos, ptr);
if (cnt == -ETIMEDOUT || cnt == -EINTR)
cnt = 0;
else if (cnt < 0)
goto error;
}
if (pos != len)
goto error; goto error;
*vpdlen = cnt;
} }
*vpdlen = len;
return buf; return buf;
error: error:
......
...@@ -2101,6 +2101,7 @@ ...@@ -2101,6 +2101,7 @@
/* Hardware Legacy NVRAM layout */ /* Hardware Legacy NVRAM layout */
#define TG3_NVM_VPD_OFF 0x100 #define TG3_NVM_VPD_OFF 0x100
#define TG3_NVM_VPD_LEN 256 #define TG3_NVM_VPD_LEN 256
#define TG3_NVM_PCI_VPD_MAX_LEN 512
/* Hardware Selfboot NVRAM layout */ /* Hardware Selfboot NVRAM layout */
#define TG3_NVM_HWSB_CFG1 0x00000004 #define TG3_NVM_HWSB_CFG1 0x00000004
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment