From 940e6e683bf6b568cdc59ead07f17a888923326f Mon Sep 17 00:00:00 2001 From: Marcus Glocker <mglocker@cvs.openbsd.org> Date: Tue, 18 Sep 2007 17:35:39 +0000 Subject: Drag back the original DragonFlyBSD firmware loading routines, so we can load the same firmware revision as they do. Our previously used firmware images seem to contain the wrong revision. You need to bump your firmware package to version 1.2. Other than expected, loading the right firmware revision still doesn't fix a fatal chip error at initialization time. --- sys/dev/ic/bwi.c | 411 +++++++++++++++++++++++++++++++++------------------- sys/dev/ic/bwivar.h | 38 ++++- 2 files changed, 291 insertions(+), 158 deletions(-) (limited to 'sys/dev') diff --git a/sys/dev/ic/bwi.c b/sys/dev/ic/bwi.c index ac95dea2feb..fac9585d162 100644 --- a/sys/dev/ic/bwi.c +++ b/sys/dev/ic/bwi.c @@ -1,4 +1,4 @@ -/* $OpenBSD: bwi.c,v 1.38 2007/09/17 20:43:18 mglocker Exp $ */ +/* $OpenBSD: bwi.c,v 1.39 2007/09/18 17:35:38 mglocker Exp $ */ /* * Copyright (c) 2007 The DragonFly Project. All rights reserved. @@ -144,10 +144,14 @@ void bwi_mac_init_tpctl_11bg(struct bwi_mac *); void bwi_mac_detach(struct bwi_mac *); int bwi_get_firmware(const char *, const uint8_t *, size_t, size_t *, size_t *); +int bwi_fwimage_is_valid(struct bwi_softc *, uint8_t *, + size_t fw_len, char *, uint8_t); +int bwi_mac_fw_alloc(struct bwi_mac *); +void bwi_mac_fw_free(struct bwi_mac *); int bwi_mac_fw_load(struct bwi_mac *); int bwi_mac_gpio_init(struct bwi_mac *); int bwi_mac_gpio_fini(struct bwi_mac *); -int bwi_mac_fw_load_iv(struct bwi_mac *, uint8_t *, int); +int bwi_mac_fw_load_iv(struct bwi_mac *, uint8_t *, size_t); int bwi_mac_fw_init(struct bwi_mac *); void bwi_mac_opmode_init(struct bwi_mac *); void bwi_mac_hostflags_init(struct bwi_mac *); @@ -1007,6 +1011,10 @@ bwi_mac_init(struct bwi_mac *mac) /* * Load and initialize firmwares */ + error = bwi_mac_fw_alloc(mac); + if (error) + return (error); + error = bwi_mac_fw_load(mac); if (error) return (error); @@ -1498,7 +1506,7 @@ bwi_mac_init_tpctl_11bg(struct bwi_mac *mac) void bwi_mac_detach(struct bwi_mac *mac) { - + bwi_mac_fw_free(mac); } int @@ -1536,57 +1544,195 @@ bwi_get_firmware(const char *name, const uint8_t *ucode, size_t size_ucode, } int -bwi_mac_fw_load(struct bwi_mac *mac) +bwi_fwimage_is_valid(struct bwi_softc *sc, uint8_t *fw, size_t fw_len, + char *fw_name, uint8_t fw_type) +{ + const struct bwi_fwhdr *hdr; + + if (fw_len < sizeof(*hdr)) { + printf("invalid firmware (%s): invalid size %u\n", + fw_name, fw_len); + return (1); + } + + hdr = (const struct bwi_fwhdr *)fw; + + if (fw_type != BWI_FW_T_IV) { + /* + * Don't verify IV's size, it has different meaning + */ + if (betoh32(hdr->fw_size) != fw_len - sizeof(*hdr)) { + printf("invalid firmware (%s): size mismatch, " + "fw %u, real %u\n", + fw_name, betoh32(hdr->fw_size), + fw_len - sizeof(*hdr)); + return (1); + } + } + + if (hdr->fw_type != fw_type) { + printf("invalid firmware (%s): type mismatch, " + "fw \'%c\', target \'%c\'\n", + fw_name, hdr->fw_type, fw_type); + return (1); + } + + if (hdr->fw_gen != BWI_FW_GEN_1) { + printf("invalid firmware (%s): wrong generation, " + "fw %d, target %d\n", + fw_name, hdr->fw_gen, BWI_FW_GEN_1); + return (1); + } + + return (0); +} + +int +bwi_mac_fw_alloc(struct bwi_mac *mac) { struct bwi_softc *sc = mac->mac_sc; - char *name = "bwi-airforce"; - char filename[64]; - uint8_t *ucode; - uint16_t fw_rev; - uint32_t *fw; - size_t size_ucode, size_fw, size_pcm, off_fw, off_pcm; - int fw_len, i, error = 0; + char fwname[64]; + int idx, error; + + if (mac->mac_ucode == NULL) { + snprintf(fwname, sizeof(fwname), "ucode%d.fw", + mac->mac_rev >= 5 ? 5 : mac->mac_rev); + + error = loadfirmware(fwname, &mac->mac_ucode, + &mac->mac_ucode_size); + if (error != 0) { + printf("%s: error %d, could not read firmware %s!\n", + sc->sc_dev.dv_xname, error, fwname); + return (ENOMEM); + } + DPRINTF(1, "%s: loaded firmware file %s\n", + sc->sc_dev.dv_xname, fwname); - /* - * Load FW file - */ - if ((error = loadfirmware(name, &ucode, &size_ucode)) != 0) { - printf("%s: error %d, could not read firmware %s!\n", - sc->sc_dev.dv_xname, error, name); - return (EIO); + if (bwi_fwimage_is_valid(sc, mac->mac_ucode, + mac->mac_ucode_size, fwname, BWI_FW_T_UCODE)) + return (EINVAL); } - DPRINTF(1, "%s: successfully read %s\n", sc->sc_dev.dv_xname, name); - /* - * Get FW file offset - */ - snprintf(filename, sizeof(filename), "bwi_microcode%d.fw", - mac->mac_rev >= 5 ? 5 : mac->mac_rev); - if (bwi_get_firmware(filename, ucode, size_ucode, &size_fw, &off_fw)) { - printf("%s: get offset for firmware file %s failed!\n", - sc->sc_dev.dv_xname, filename); - error = ENOMEM; - goto out; - } + if (mac->mac_pcm == NULL) { + snprintf(fwname, sizeof(fwname), "pcm%d.fw", + mac->mac_rev < 5 ? 4 : 5); - /* - * Get PCM file offset - */ - snprintf(filename, sizeof(filename), "bwi_pcm%d.fw", - mac->mac_rev < 5 ? 4 : 5); - if (bwi_get_firmware(filename, ucode, size_ucode, &size_pcm, - &off_pcm)) { - printf("%s: get offset for firmware file %s failed!\n", - sc->sc_dev.dv_xname, filename); - error = ENOMEM; - goto out; + error = loadfirmware(fwname, &mac->mac_pcm, + &mac->mac_pcm_size); + if (error != 0) { + printf("%s: error %d, could not read firmware %s!\n", + sc->sc_dev.dv_xname, error, fwname); + return (ENOMEM); + } + DPRINTF(1, "%s: loaded firmware file %s\n", + sc->sc_dev.dv_xname, fwname); + + if (bwi_fwimage_is_valid(sc, mac->mac_pcm, + mac->mac_pcm_size, fwname, BWI_FW_T_PCM)) + return (EINVAL); + } + + if (mac->mac_iv == NULL) { + /* TODO: 11A */ + if (mac->mac_rev == 2 || mac->mac_rev == 4) { + idx = 2; + } else if (mac->mac_rev >= 5 && mac->mac_rev <= 10) { + idx = 5; + } else { + DPRINTF(1, "%s: no suitable IV for MAC rev %d\n", + sc->sc_dev.dv_xname, mac->mac_rev); + return (ENODEV); + } + + snprintf(fwname, sizeof(fwname), "b0g0initvals%d.fw", idx); + + error = loadfirmware(fwname, &mac->mac_iv, + &mac->mac_iv_size); + if (error != 0) { + printf("%s: error %d, could not read firmware %s!\n", + sc->sc_dev.dv_xname, error, fwname); + return (ENOMEM); + } + DPRINTF(1, "%s: loaded firmware file %s\n", + sc->sc_dev.dv_xname, fwname); + + if (bwi_fwimage_is_valid(sc, mac->mac_iv, + mac->mac_iv_size, fwname, BWI_FW_T_IV)) + return (EINVAL); + } + + if (mac->mac_iv_ext == NULL) { + /* TODO: 11A */ + if (mac->mac_rev == 2 || mac->mac_rev == 4 || + mac->mac_rev >= 11) { + /* No extended IV */ + goto back; + } else if (mac->mac_rev >= 5 && mac->mac_rev <= 10) { + idx = 5; + } else { + printf("%s: no suitable ExtIV for MAC rev %d\n", + sc->sc_dev.dv_xname, mac->mac_rev); + return (ENODEV); + } + + snprintf(fwname, sizeof(fwname), "b0g0bsinitvals%d.fw", idx); + + error = loadfirmware(fwname, &mac->mac_iv_ext, + &mac->mac_iv_ext_size); + if (error != 0) { + printf("%s: error %d, could not read firmware %s!\n", + sc->sc_dev.dv_xname, error, fwname); + return (ENOMEM); + } + DPRINTF(1, "%s: loaded firmware file %s\n", + sc->sc_dev.dv_xname, fwname); + + if (bwi_fwimage_is_valid(sc, mac->mac_iv_ext, + mac->mac_iv_ext_size, fwname, BWI_FW_T_IV)) + return (EINVAL); + } + +back: + return (0); +} + +void +bwi_mac_fw_free(struct bwi_mac *mac) +{ + if (mac->mac_ucode != NULL) { + free(mac->mac_ucode, M_DEVBUF); + mac->mac_ucode = NULL; } + if (mac->mac_pcm != NULL) { + free(mac->mac_pcm, M_DEVBUF); + mac->mac_pcm = NULL; + } + + if (mac->mac_iv != NULL) { + free(mac->mac_iv, M_DEVBUF); + mac->mac_iv = NULL; + } + + if (mac->mac_iv_ext != NULL) { + free(mac->mac_iv_ext, M_DEVBUF); + mac->mac_iv_ext = NULL; + } +} + +int +bwi_mac_fw_load(struct bwi_mac *mac) +{ + struct bwi_softc *sc = mac->mac_sc; + uint16_t fw_rev; + const uint32_t *fw; + int fw_len, i, error = 0; + /* * Load FW image */ - fw = (uint32_t *)(ucode + off_fw); - fw_len = size_fw / sizeof(uint32_t); + fw = (const uint32_t *)(mac->mac_ucode + BWI_FWHDR_SZ); + fw_len = (mac->mac_ucode_size - BWI_FWHDR_SZ) / sizeof(uint32_t); CSR_WRITE_4(sc, BWI_MOBJ_CTRL, BWI_MOBJ_CTRL_VAL(BWI_FW_UCODE_MOBJ | BWI_WR_MOBJ_AUTOINC, 0)); @@ -1598,8 +1744,8 @@ bwi_mac_fw_load(struct bwi_mac *mac) /* * Load PCM image */ - fw = (uint32_t *)(ucode + off_pcm); - fw_len = size_pcm / sizeof(uint32_t); + fw = (const uint32_t *)(mac->mac_pcm + BWI_FWHDR_SZ); + fw_len = (mac->mac_pcm_size - BWI_FWHDR_SZ) / sizeof(uint32_t); CSR_WRITE_4(sc, BWI_MOBJ_CTRL, BWI_MOBJ_CTRL_VAL(BWI_FW_PCM_MOBJ, 0x01ea)); @@ -1650,7 +1796,6 @@ bwi_mac_fw_load(struct bwi_mac *mac) MOBJ_READ_2(mac, BWI_COMM_MOBJ, BWI_COMM_MOBJ_FWPATCHLV)); out: - free(ucode, M_DEVBUF); return (error); } @@ -1707,138 +1852,104 @@ bwi_mac_gpio_fini(struct bwi_mac *mac) } int -bwi_mac_fw_load_iv(struct bwi_mac *mac, uint8_t *fw_image, int fw_len) +bwi_mac_fw_load_iv(struct bwi_mac *mac, uint8_t *fw, size_t fw_len) { struct bwi_softc *sc = mac->mac_sc; + const struct bwi_fwhdr *hdr; const struct bwi_fw_iv *iv; - uint16_t offset, size; - uint32_t val; - int iv_len, i, error = 0; + int n, i, iv_img_size; /* Get the number of IVs in the IV image */ - iv_len = fw_len / sizeof(struct bwi_fw_iv); - DPRINTF(1, "%s: IV count %d\n", sc->sc_dev.dv_xname, iv_len); + hdr = (const struct bwi_fwhdr *)fw; + n = betoh32(hdr->fw_iv_cnt); + DPRINTF(1, "%s: IV count %d\n", sc->sc_dev.dv_xname, n); + + /* Calculate the IV image size, for later sanity check */ + iv_img_size = fw_len - sizeof(*hdr); /* Locate the first IV */ - iv = (const struct bwi_fw_iv *)fw_image; + iv = (const struct bwi_fw_iv *)(fw + sizeof(*hdr)); + + for (i = 0; i < n; ++i) { + uint16_t iv_ofs, ofs; + int sz = 0; + + if (iv_img_size < sizeof(iv->iv_ofs)) { + printf("%s: invalid IV image, ofs\n", + sc->sc_dev.dv_xname); + return (EINVAL); + } + iv_img_size -= sizeof(iv->iv_ofs); + sz += sizeof(iv->iv_ofs); - for (i = 0; i < iv_len; i++, iv++) { - offset = betoh16(iv->offset); - size = betoh16(iv->size); - val = betoh32(iv->val); + iv_ofs = betoh16(iv->iv_ofs); - if (offset >= 0x1000) { - error = ENODEV; - goto error; + ofs = __SHIFTOUT(iv_ofs, BWI_FW_IV_OFS_MASK); + if (ofs >= 0x1000) { + printf("%s: invalid ofs (0x%04x) for %dth iv\n", + sc->sc_dev.dv_xname, ofs, i); + return (EINVAL); } - if (size == sizeof(uint16_t)) { - if (val & 0xffff0000) { - error = ENODEV; - goto error; + if (iv_ofs & BWI_FW_IV_IS_32BIT) { + uint32_t val32; + + if (iv_img_size < sizeof(iv->iv_val.val32)) { + printf("%s: invalid IV image, val32\n", + sc->sc_dev.dv_xname); + return (EINVAL); } - CSR_WRITE_2(sc, offset, (uint16_t)val); - } else if (size == sizeof(uint32_t)) - CSR_WRITE_4(sc, offset, val); - else { - error = ENODEV; - goto error; + iv_img_size -= sizeof(iv->iv_val.val32); + sz += sizeof(iv->iv_val.val32); + + val32 = betoh32(iv->iv_val.val32); + CSR_WRITE_4(sc, ofs, val32); + } else { + uint16_t val16; + + if (iv_img_size < sizeof(iv->iv_val.val16)) { + printf("%s: invalid IV image, val16\n", + sc->sc_dev.dv_xname); + return (EINVAL); + } + iv_img_size -= sizeof(iv->iv_val.val16); + sz += sizeof(iv->iv_val.val16); + + val16 = betoh16(iv->iv_val.val16); + CSR_WRITE_2(sc, ofs, val16); } + + iv = (const struct bwi_fw_iv *)((const uint8_t *)iv + sz); } - return (error); + if (iv_img_size != 0) { + printf("%s: invalid IV image, size left %d\n", + sc->sc_dev.dv_xname, iv_img_size); + return (EINVAL); + } -error: - printf("%s: bad IV format!\n", sc->sc_dev.dv_xname); - return (error); + return (0); } int bwi_mac_fw_init(struct bwi_mac *mac) { struct bwi_softc *sc = mac->mac_sc; - char *name = "bwi-airforce"; - char fwname[64]; - uint8_t *ucode; - size_t size_ucode, size_iv, off_iv; - int idx, error = 0; - - /* - * Load FW file - */ - if ((error = loadfirmware(name, &ucode, &size_ucode)) != 0) { - printf("%s: error %d, could not read firmware %s!\n", - sc->sc_dev.dv_xname, error, name); - return (EIO); - } - DPRINTF(1, "%s: successfully read %s\n", sc->sc_dev.dv_xname, name); - - /* - * Load IV - * - * TODO: 11A - */ - if (mac->mac_rev == 2 || mac->mac_rev == 4) - idx = 2; - else if (mac->mac_rev >= 5 && mac->mac_rev <= 10) - idx = 5; - else { - printf("%s: no suitable IV for MAC rev %d\n", - sc->sc_dev.dv_xname, mac->mac_rev); - error = ENODEV; - goto out; - } - snprintf(fwname, sizeof(fwname), "bwi_initval%02d.fw", idx); - - DPRINTF(1, "%s: IV image is %s\n", sc->sc_dev.dv_xname, fwname); - - if (bwi_get_firmware(fwname, ucode, size_ucode, &size_iv, &off_iv)) { - printf("%s: IV image %s not found!\n", - sc->sc_dev.dv_xname, fwname); - error = ENOMEM; - goto out; - } + int error; - error = bwi_mac_fw_load_iv(mac, (ucode + off_iv), size_iv); + error = bwi_mac_fw_load_iv(mac, mac->mac_iv, mac->mac_iv_size); if (error) { - printf("%s: load IV failed!\n", sc->sc_dev.dv_xname); - goto out; - } - - /* - * Load extended IV - * - * TODO: 11A - */ - if (mac->mac_rev == 2 || mac->mac_rev == 4 || mac->mac_rev >= 11) - /* No extended IV */ - goto out; - else if (mac->mac_rev >= 5 && mac->mac_rev <= 10) - idx = 5; - else { - printf("%s: no suitable extended IV for MAC rev %d\n", - sc->sc_dev.dv_xname, mac->mac_rev); - error = ENODEV; - goto out; + printf("%s: load IV failed\n", sc->sc_dev.dv_xname); + return (error); } - snprintf(fwname, sizeof(fwname), "bwi_initval%02d.fw", idx); - - DPRINTF(1, "%s: extended IV image is %s\n", - sc->sc_dev.dv_xname, fwname); - if (bwi_get_firmware(fwname, ucode, size_ucode, &size_iv, &off_iv)) { - printf("%s: extended IV image %s not found!\n", - sc->sc_dev.dv_xname, fwname); - error = ENOMEM; - goto out; + if (mac->mac_iv_ext != NULL) { + error = bwi_mac_fw_load_iv(mac, mac->mac_iv_ext, + mac->mac_iv_ext_size); + if (error) + printf("%s: load ExtIV failed\n", sc->sc_dev.dv_xname); } - error = bwi_mac_fw_load_iv(mac, (ucode + off_iv), size_iv); - if (error) - printf("%s: load extended IV failed!\n", sc->sc_dev.dv_xname); - -out: - free(ucode, M_DEVBUF); return (error); } diff --git a/sys/dev/ic/bwivar.h b/sys/dev/ic/bwivar.h index 1ad1dcd826d..18d055de580 100644 --- a/sys/dev/ic/bwivar.h +++ b/sys/dev/ic/bwivar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: bwivar.h,v 1.10 2007/09/16 19:02:37 mglocker Exp $ */ +/* $OpenBSD: bwivar.h,v 1.11 2007/09/18 17:35:38 mglocker Exp $ */ /* * Copyright (c) 2007 The DragonFly Project. All rights reserved. @@ -221,9 +221,25 @@ struct bwi_txstats_data { int stats_idx; }; +struct bwi_fwhdr { + /* Big endian */ + uint8_t fw_type; /* BWI_FW_T_ */ + uint8_t fw_gen; /* BWI_FW_GEN */ + uint8_t fw_pad[2]; + uint32_t fw_size; +#define fw_iv_cnt fw_size +} __packed; + +#define BWI_FWHDR_SZ sizeof(struct bwi_fwhdr) #define BWI_FW_VERSION3 3 #define BWI_FW_VERSION4 4 #define BWI_FW_VERSION3_REVMAX 0x128 +#define BWI_FW_T_UCODE 'u' +#define BWI_FW_T_PCM 'p' +#define BWI_FW_T_IV 'i' +#define BWI_FW_GEN_1 1 +#define BWI_FW_IV_OFS_MASK __BITS(14, 0) +#define BWI_FW_IV_IS_32BIT __BIT(15) struct fwheader { char filename[64]; @@ -233,9 +249,11 @@ struct fwheader { struct bwi_fw_iv { /* Big endian */ - uint16_t offset; - uint16_t size; - uint32_t val; + uint16_t iv_ofs; + union { + uint32_t val32; + uint16_t val16; + } iv_val; } __packed; enum bwi_clock_mode { @@ -377,10 +395,14 @@ struct bwi_mac { struct bwi_tpctl mac_tpctl; /* TX power control */ uint32_t mac_flags; /* BWI_MAC_F_ */ - struct fw_image *mac_ucode; - struct fw_image *mac_pcm; - struct fw_image *mac_iv; - struct fw_image *mac_iv_ext; + uint8_t *mac_ucode; + size_t mac_ucode_size; + uint8_t *mac_pcm; + size_t mac_pcm_size; + uint8_t *mac_iv; + size_t mac_iv_size; + uint8_t *mac_iv_ext; + size_t mac_iv_ext_size; }; #define BWI_MAC_F_BSWAP 0x1 -- cgit v1.2.3