summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/dev/pci/if_cas.c78
1 files changed, 63 insertions, 15 deletions
diff --git a/sys/dev/pci/if_cas.c b/sys/dev/pci/if_cas.c
index 2dc43009a11..6f7951a72a7 100644
--- a/sys/dev/pci/if_cas.c
+++ b/sys/dev/pci/if_cas.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_cas.c,v 1.6 2007/04/15 16:31:30 kettenis Exp $ */
+/* $OpenBSD: if_cas.c,v 1.7 2007/04/15 19:57:16 kettenis Exp $ */
/*
*
@@ -176,13 +176,14 @@ static const u_int8_t cas_promdat2[] = {
int
cas_pci_enaddr(struct cas_softc *sc, struct pci_attach_args *pa)
{
+ struct pci_vpd_largeres *res;
struct pci_vpd *vpd;
bus_space_handle_t romh;
bus_space_tag_t romt;
bus_size_t romsize;
- u_int8_t buf[32];
+ u_int8_t buf[32], *desc;
pcireg_t address, mask;
- int dataoff, vpdoff;
+ int dataoff, vpdoff, len;
int rv = -1;
address = pci_conf_read(pa->pa_pc, pa->pa_tag, PCI_ROM_REG);
@@ -215,21 +216,68 @@ cas_pci_enaddr(struct cas_softc *sc, struct pci_attach_args *pa)
if (vpdoff < 0x1c)
goto fail;
+next:
bus_space_read_region_1(romt, romh, vpdoff, buf, sizeof(buf));
-
- /*
- * The VPD is not in PCI 2.2 standard format. The length in
- * the resource header is big endian.
- */
- vpd = (struct pci_vpd *)(buf + 3);
- if (!PCI_VPDRES_ISLARGE(buf[0]) ||
- PCI_VPDRES_LARGE_NAME(buf[0]) != PCI_VPDRES_TYPE_VPD)
- goto fail;
- if (vpd->vpd_key0 != 'N' || vpd->vpd_key1 != 'A')
+ if (!PCI_VPDRES_ISLARGE(buf[0]))
goto fail;
- bcopy(buf + 6, sc->sc_arpcom.ac_enaddr, ETHER_ADDR_LEN);
- rv = 0;
+ res = (struct pci_vpd_largeres *)buf;
+ vpdoff += sizeof(*res);
+
+ len = ((res->vpdres_len_msb << 8) + res->vpdres_len_lsb);
+ switch(PCI_VPDRES_LARGE_NAME(res->vpdres_byte0)) {
+ case PCI_VPDRES_TYPE_IDENTIFIER_STRING:
+ /* Skip identifier string. */
+ vpdoff += len;
+ goto next;
+
+ case PCI_VPDRES_TYPE_VPD:
+ while (len > 0) {
+ bus_space_read_region_1(romt, romh, vpdoff,
+ buf, sizeof(buf));
+
+ vpd = (struct pci_vpd *)buf;
+ vpdoff += sizeof(*vpd) + vpd->vpd_len;
+ len -= sizeof(*vpd) + vpd->vpd_len;
+
+ /*
+ * We're looking for an "Enhanced" VPD...
+ */
+ if (vpd->vpd_key0 != 'Z')
+ continue;
+
+ desc = buf + sizeof(*vpd);
+
+ /*
+ * ...which is an instance property...
+ */
+ if (desc[0] != 'I')
+ continue;
+ desc += 3;
+
+ /*
+ * ...that's a byte array with the proper
+ * length for a MAC address...
+ */
+ if (desc[0] != 'B' || desc[1] != ETHER_ADDR_LEN)
+ continue;
+ desc += 2;
+
+ /*
+ * ...named "local-mac-address".
+ */
+ if (strcmp(desc, "local-mac-address") != 0)
+ continue;
+ desc += strlen("local-mac-address") + 1;
+
+ bcopy(desc, sc->sc_arpcom.ac_enaddr, ETHER_ADDR_LEN);
+ rv = 0;
+ }
+ break;
+
+ default:
+ goto fail;
+ }
fail:
if (romsize != 0)