diff options
author | Damien Bergamini <damien@cvs.openbsd.org> | 2010-04-30 16:06:47 +0000 |
---|---|---|
committer | Damien Bergamini <damien@cvs.openbsd.org> | 2010-04-30 16:06:47 +0000 |
commit | 34644739c1a40c2f7194a8d776349a74ee4da7da (patch) | |
tree | c3dd186dfd7dd33942fb6711b3bfa6747e06c8c2 /sys/dev | |
parent | 8b114d886ea016b76c7e9ecddecd7506884a7c06 (diff) |
add support for firmware images in "type-length-value" format.
Diffstat (limited to 'sys/dev')
-rw-r--r-- | sys/dev/pci/if_iwn.c | 178 | ||||
-rw-r--r-- | sys/dev/pci/if_iwnreg.h | 25 | ||||
-rw-r--r-- | sys/dev/pci/if_iwnvar.h | 3 |
3 files changed, 162 insertions, 44 deletions
diff --git a/sys/dev/pci/if_iwn.c b/sys/dev/pci/if_iwn.c index d9247671e84..554fa39e2f0 100644 --- a/sys/dev/pci/if_iwn.c +++ b/sys/dev/pci/if_iwn.c @@ -1,4 +1,4 @@ -/* $OpenBSD: if_iwn.c,v 1.89 2010/04/20 22:05:43 tedu Exp $ */ +/* $OpenBSD: if_iwn.c,v 1.90 2010/04/30 16:06:46 damien Exp $ */ /*- * Copyright (c) 2007-2010 Damien Bergamini <damien.bergamini@free.fr> @@ -237,6 +237,10 @@ int iwn4965_load_firmware(struct iwn_softc *); int iwn5000_load_firmware_section(struct iwn_softc *, uint32_t, const uint8_t *, int); int iwn5000_load_firmware(struct iwn_softc *); +int iwn_read_firmware_leg(struct iwn_softc *, + struct iwn_fw_info *); +int iwn_read_firmware_tlv(struct iwn_softc *, + struct iwn_fw_info *); int iwn_read_firmware(struct iwn_softc *); int iwn_clock_wait(struct iwn_softc *); int iwn_apm_init(struct iwn_softc *); @@ -5155,69 +5159,46 @@ iwn5000_load_firmware(struct iwn_softc *sc) return 0; } +/* + * Extract text and data sections from a legacy firmware image. + */ int -iwn_read_firmware(struct iwn_softc *sc) +iwn_read_firmware_leg(struct iwn_softc *sc, struct iwn_fw_info *fw) { - const struct iwn_hal *hal = sc->sc_hal; - struct iwn_fw_info *fw = &sc->fw; const uint32_t *ptr; + size_t hdrlen = 24; uint32_t rev; - size_t size; - int error; - /* Read firmware image from filesystem. */ - if ((error = loadfirmware(sc->fwname, &fw->data, &size)) != 0) { - printf("%s: error, %d, could not read firmware %s\n", - sc->sc_dev.dv_xname, error, sc->fwname); - return error; - } - if (size < 28) { - printf("%s: truncated firmware header: %d bytes\n", - sc->sc_dev.dv_xname, size); - free(fw->data, M_DEVBUF); - return EINVAL; - } - - /* Process firmware header. */ ptr = (const uint32_t *)fw->data; rev = letoh32(*ptr++); + /* Check firmware API version. */ if (IWN_FW_API(rev) <= 1) { printf("%s: bad firmware, need API version >=2\n", sc->sc_dev.dv_xname); - free(fw->data, M_DEVBUF); return EINVAL; } if (IWN_FW_API(rev) >= 3) { /* Skip build number (version 2 header). */ - size -= 4; + hdrlen += 4; ptr++; } + if (fw->size < hdrlen) { + printf("%s: firmware too short: %d bytes\n", + sc->sc_dev.dv_xname, fw->size); + return EINVAL; + } fw->main.textsz = letoh32(*ptr++); fw->main.datasz = letoh32(*ptr++); fw->init.textsz = letoh32(*ptr++); fw->init.datasz = letoh32(*ptr++); fw->boot.textsz = letoh32(*ptr++); - size -= 24; - - /* Sanity-check firmware header. */ - if (fw->main.textsz > hal->fw_text_maxsz || - fw->main.datasz > hal->fw_data_maxsz || - fw->init.textsz > hal->fw_text_maxsz || - fw->init.datasz > hal->fw_data_maxsz || - fw->boot.textsz > IWN_FW_BOOT_TEXT_MAXSZ || - (fw->boot.textsz & 3) != 0) { - printf("%s: invalid firmware header\n", sc->sc_dev.dv_xname); - free(fw->data, M_DEVBUF); - return EINVAL; - } /* Check that all firmware sections fit. */ - if (fw->main.textsz + fw->main.datasz + fw->init.textsz + - fw->init.datasz + fw->boot.textsz > size) { - printf("%s: firmware file too short: %d bytes\n", - sc->sc_dev.dv_xname, size); - free(fw->data, M_DEVBUF); + if (fw->size < hdrlen + fw->main.textsz + fw->main.datasz + + fw->init.textsz + fw->init.datasz + fw->boot.textsz) { + printf("%s: firmware too short: %d bytes\n", + sc->sc_dev.dv_xname, fw->size); return EINVAL; } @@ -5227,7 +5208,124 @@ iwn_read_firmware(struct iwn_softc *sc) fw->init.text = fw->main.data + fw->main.datasz; fw->init.data = fw->init.text + fw->init.textsz; fw->boot.text = fw->init.data + fw->init.datasz; + return 0; +} + +/* + * Extract text and data sections from a TLV firmware image. + */ +int +iwn_read_firmware_tlv(struct iwn_softc *sc, struct iwn_fw_info *fw) +{ + const struct iwn_fw_tlv_hdr *hdr; + const uint32_t *ptr, *end; + uint32_t type, len; + + if (fw->size < sizeof (*hdr)) { + printf("%s: firmware too short: %d bytes\n", + sc->sc_dev.dv_xname, fw->size); + return EINVAL; + } + hdr = (const struct iwn_fw_tlv_hdr *)fw->data; + if (hdr->signature != htole32(IWN_FW_SIGNATURE)) { + printf("%s: bad firmware signature 0x%08x\n", + sc->sc_dev.dv_xname, letoh32(hdr->signature)); + return EINVAL; + } + DPRINTF(("FW: \"%.64s\", build 0x%x\n", hdr->descr, + letoh32(hdr->build))); + + ptr = (const uint32_t *)(hdr + 1); + end = (const uint32_t *)(fw->data + fw->size); + + /* Parse type-length-value fields. */ + while (ptr + 2 <= end) { + type = letoh32(*ptr++); + len = letoh32(*ptr++); + if (ptr + (len + 3) / 4 > end) { + printf("%s: firmware too short: %d bytes\n", + sc->sc_dev.dv_xname, fw->size); + return EINVAL; + } + switch (type) { + case IWN_FW_TLV_MAIN_TEXT: + fw->main.text = (const uint8_t *)ptr; + fw->main.textsz = len; + break; + case IWN_FW_TLV_MAIN_DATA: + fw->main.data = (const uint8_t *)ptr; + fw->main.datasz = len; + break; + case IWN_FW_TLV_INIT_TEXT: + fw->init.text = (const uint8_t *)ptr; + fw->init.textsz = len; + break; + case IWN_FW_TLV_INIT_DATA: + fw->init.data = (const uint8_t *)ptr; + fw->init.datasz = len; + break; + case IWN_FW_TLV_BOOT_TEXT: + fw->boot.text = (const uint8_t *)ptr; + fw->boot.textsz = len; + break; + default: + DPRINTF(("TLV type %d not handled\n", type)); + break; + } + /* TLV fields are 32-bit aligned. */ + ptr += (len + 3) / 4; + } + return 0; +} + +int +iwn_read_firmware(struct iwn_softc *sc) +{ + const struct iwn_hal *hal = sc->sc_hal; + struct iwn_fw_info *fw = &sc->fw; + int error; + + memset(fw, 0, sizeof (*fw)); + + /* Read firmware image from filesystem. */ + if ((error = loadfirmware(sc->fwname, &fw->data, &fw->size)) != 0) { + printf("%s: error, %d, could not read firmware %s\n", + sc->sc_dev.dv_xname, error, sc->fwname); + return error; + } + if (fw->size < sizeof (uint32_t)) { + printf("%s: firmware too short: %d bytes\n", + sc->sc_dev.dv_xname, fw->size); + free(fw->data, M_DEVBUF); + return EINVAL; + } + + /* Retrieve text and data sections. */ + if (*(const uint32_t *)fw->data != 0) /* Legacy image. */ + error = iwn_read_firmware_leg(sc, fw); + else + error = iwn_read_firmware_tlv(sc, fw); + if (error != 0) { + printf("%s: could not read firmware sections\n", + sc->sc_dev.dv_xname); + free(fw->data, M_DEVBUF); + return error; + } + + /* Make sure text and data sections fit in hardware memory. */ + if (fw->main.textsz > hal->fw_text_maxsz || + fw->main.datasz > hal->fw_data_maxsz || + fw->init.textsz > hal->fw_text_maxsz || + fw->init.datasz > hal->fw_data_maxsz || + fw->boot.textsz > IWN_FW_BOOT_TEXT_MAXSZ || + (fw->boot.textsz & 3) != 0) { + printf("%s: firmware sections too large\n", + sc->sc_dev.dv_xname); + free(fw->data, M_DEVBUF); + return EINVAL; + } + /* We can proceed with loading the firmware. */ return 0; } diff --git a/sys/dev/pci/if_iwnreg.h b/sys/dev/pci/if_iwnreg.h index 0296f3a782e..a2b01109d0d 100644 --- a/sys/dev/pci/if_iwnreg.h +++ b/sys/dev/pci/if_iwnreg.h @@ -1,4 +1,4 @@ -/* $OpenBSD: if_iwnreg.h,v 1.38 2010/04/10 08:37:36 damien Exp $ */ +/* $OpenBSD: if_iwnreg.h,v 1.39 2010/04/30 16:06:46 damien Exp $ */ /*- * Copyright (c) 2007, 2008 @@ -1248,6 +1248,27 @@ struct iwn_fw_dump { uint32_t time[2]; } __packed; +/* TLV firmware header. */ +struct iwn_fw_tlv_hdr { + uint32_t zero; /* Always 0, to differentiate from legacy. */ + uint32_t signature; +#define IWN_FW_SIGNATURE 0x0a4c5749 /* "IWL\n" */ + + uint8_t descr[64]; + uint32_t rev; +#define IWN_FW_API(x) (((x) >> 8) & 0xff) + + uint32_t build; +} __packed; + +/* Firmware TLV fields types. */ +#define IWN_FW_TLV_MAIN_TEXT 1 +#define IWN_FW_TLV_MAIN_DATA 2 +#define IWN_FW_TLV_INIT_TEXT 3 +#define IWN_FW_TLV_INIT_DATA 4 +#define IWN_FW_TLV_BOOT_TEXT 5 +#define IWN_FW_TLV_PBREQ_MAXLEN 6 + #define IWN4965_FW_TEXT_MAXSZ ( 96 * 1024) #define IWN4965_FW_DATA_MAXSZ ( 40 * 1024) #define IWN5000_FW_TEXT_MAXSZ (256 * 1024) @@ -1256,8 +1277,6 @@ struct iwn_fw_dump { #define IWN4965_FWSZ (IWN4965_FW_TEXT_MAXSZ + IWN4965_FW_DATA_MAXSZ) #define IWN5000_FWSZ IWN5000_FW_TEXT_MAXSZ -#define IWN_FW_API(x) (((x) >> 8) & 0xff) - /* * Offsets into EEPROM. */ diff --git a/sys/dev/pci/if_iwnvar.h b/sys/dev/pci/if_iwnvar.h index 58e64224584..827406c2784 100644 --- a/sys/dev/pci/if_iwnvar.h +++ b/sys/dev/pci/if_iwnvar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: if_iwnvar.h,v 1.17 2010/02/17 18:23:00 damien Exp $ */ +/* $OpenBSD: if_iwnvar.h,v 1.18 2010/04/30 16:06:46 damien Exp $ */ /*- * Copyright (c) 2007, 2008 @@ -150,6 +150,7 @@ struct iwn_fw_part { struct iwn_fw_info { u_char *data; + size_t size; struct iwn_fw_part init; struct iwn_fw_part main; struct iwn_fw_part boot; |