/*- * Copyright (c) 2012 Robert N. M. Watson * Copyright (c) 2012 SRI International * All rights reserved. * * This software was developed by SRI International and the University of * Cambridge Computer Laboratory under DARPA/AFRL contract (FA8750-10-C-0237) * ("CTSRD"), as part of the DARPA CRASH research programme. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include __FBSDID("$FreeBSD$"); #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* Read Mode */ #define ISF_CMD_RA 0xFF /* Read Array mode */ #define ISF_CMD_RSR 0x70 /* Read Status Register mode */ #define ISF_CMD_RDI 0x90 /* Read Device ID/Config Reg mode */ #define ISF_CMD_RQ 0x98 /* Read Query mode */ #define ISF_CMD_CSR 0x50 /* Clear Status Register */ /* Write Mode */ #define ISF_CMD_WPS 0x40 /* Word Program Setup */ #define ISF_CMD_BPS 0xE8 /* Buffered Program Setup */ #define ISF_CMD_BPC 0xD0 /* Buffered Program Confirm */ /* Erase Mode */ #define ISF_CMD_BES 0x20 /* Block Erase Setup */ #define ISF_CMD_BEC 0xD0 /* Block Erase Confirm */ /* Block Locking/Unlocking */ #define ISF_CMD_LBS 0x60 /* Lock Block Setup */ #define ISF_CMD_LB 0x01 /* Lock Block */ #define ISF_CMD_UB 0xD0 /* Unlock Block */ /* * Read Device Identifier registers. * * NOTE: ISF_RDIR_BLC is relative to the block base address. */ #define ISF_REG_MC 0x00 /* Manufacture Code */ #define ISF_REG_ID 0x01 /* Device ID Code */ #define ISF_REG_BLC 0x02 /* Block Lock Configuration */ #define ISF_REG_RCR 0x05 /* Read Configuration Register */ /* * Protection Registers */ #define ISF_REG_L0 0x80 /* Lock Register 0 */ #define ISF_REG_FPP 0x81 /* 64-bit Factory Protection Register */ #define ISF_REG_UPP 0x85 /* 64-bit User Protection Register */ #define ISF_REG_L1 0x89 /* Lock Register 1 */ #define ISF_REG_PP1 0x8A /* 128-bit Protection Register 1 */ #define ISF_REG_PP2 0x92 /* 128-bit Protection Register 2 */ #define ISF_REG_PP3 0x9A /* 128-bit Protection Register 3 */ #define ISF_REG_PP4 0xA2 /* 128-bit Protection Register 4 */ #define ISF_REG_PP5 0xAA /* 128-bit Protection Register 5 */ #define ISF_REG_PP6 0xB2 /* 128-bit Protection Register 6 */ #define ISF_REG_PP7 0xBA /* 128-bit Protection Register 7 */ #define ISF_REG_PP8 0xC2 /* 128-bit Protection Register 8 */ #define ISF_REG_PP9 0xCA /* 128-bit Protection Register 9 */ #define ISF_REG_PP10 0xD2 /* 128-bit Protection Register 10 */ #define ISF_REG_PP11 0xDA /* 128-bit Protection Register 11 */ #define ISF_REG_PP12 0xE2 /* 128-bit Protection Register 12 */ #define ISF_REG_PP13 0xEA /* 128-bit Protection Register 13 */ #define ISF_REG_PP14 0xF2 /* 128-bit Protection Register 14 */ #define ISF_REG_PP15 0xFA /* 128-bit Protection Register 15 */ #define ISF_REG_PP16 0x102 /* 128-bit Protection Register 16 */ #define ISF_SR_BWS (1 << 0) /* BEFP Status */ #define ISF_SR_BLS (1 << 1) /* Block-Locked Status */ #define ISF_SR_PSS (1 << 2) /* Program Suspend Status */ #define ISF_SR_VPPS (1 << 3) /* Vpp Status */ #define ISF_SR_PS (1 << 4) /* Program Status */ #define ISF_SR_ES (1 << 5) /* Erase Status */ #define ISF_SR_ESS (1 << 6) /* Erase Suspend Status */ #define ISF_SR_DWS (1 << 7) /* Device Write Status */ #define ISF_SR_FSC_MASK (ISF_SR_VPPS | ISF_SR_PS | ISF_SR_BLS) #define ISF_BUFFER_PROGRAM MALLOC_DEFINE(M_ISF, "isf_data", "Intel StrateFlash driver"); static int isf_debug = 0; static struct isf_chips { uint16_t chip_id; size_t chip_size; const char *chip_desc; } chip_ids[] = { { 0x8817, 0x0800000, "64-Mbit Top Parameter" }, { 0x881A, 0x0800000, "64-Mbit Bottom Parameter" }, { 0x8818, 0x1000000, "128-Mbit Top Parameter" }, { 0x881B, 0x1000000, "128-Mbit Bottom Parameter" }, { 0x8919, 0x2000000, "256-Mbit Top Parameter" }, { 0x891C, 0x2000000, "256-Mbit Bottom Parameter" }, { 0x8961, 0x2000000, "512-Mbit package (half)" }, { 0x0000, 0x0000000, NULL } }; static void isf_task(void *arg); /* * Device driver for the Intel StrataFlash NOR flash device. This * implementation is known to work with 256Mb instances of the device, but may * also work with other 64/128/512Mb parts without much work. Multiple * device instances should be used when multiple parts are in the same * physical package, due to variable block size support in the StrataFlash * part. */ static uint16_t isf_read_reg(struct isf_softc *sc, uint16_t reg) { if (isf_debug) device_printf(sc->isf_dev, "isf_read_reg(0x%02x)\n", reg); return (le16toh(bus_read_2(sc->isf_res, reg * 2))); } static uint64_t isf_read_reg64(struct isf_softc *sc, uint16_t reg) { uint64_t val; uint16_t *val16 = (uint16_t *)&val; if (isf_debug) device_printf(sc->isf_dev, "isf_read_reg64(0x%02x)\n", reg); val16[0] = bus_read_2(sc->isf_res, reg * 2); val16[1] = bus_read_2(sc->isf_res, (reg+1) * 2); val16[2] = bus_read_2(sc->isf_res, (reg+2) * 2); val16[3] = bus_read_2(sc->isf_res, (reg+3) * 2); return(le64toh(val)); } static uint16_t isf_read_off(struct isf_softc *sc, off_t off) { KASSERT(off >= 0, ("%s: negative offset\n", __func__)); KASSERT(off < sc->isf_disk->d_mediasize, ("%s: offset out side address space 0x%08jx \n", __func__, (intmax_t)off)); if (isf_debug) device_printf(sc->isf_dev, "isf_read_off(0x%08jx)\n", (intmax_t)off); return (le16toh(bus_read_2(sc->isf_res, off))); } static void isf_write_cmd(struct isf_softc *sc, off_t off, uint16_t cmd) { if (isf_debug) device_printf(sc->isf_dev, "isf_write_cmd(0x%08jx, 0x%02x)\n", off, cmd); bus_write_2(sc->isf_res, off, htole16(cmd)); } static uint16_t isf_read_status(struct isf_softc *sc, off_t off) { isf_write_cmd(sc, off/2, ISF_CMD_RSR); return isf_read_off(sc, off); } static void isf_clear_status(struct isf_softc *sc) { isf_write_cmd(sc, 0, ISF_CMD_CSR); } static int isf_full_status_check(struct isf_softc *sc, off_t off) { int error = 0; uint16_t status; status = isf_read_status(sc, off); if (status & ISF_SR_VPPS) { device_printf(sc->isf_dev, "Vpp Range Error\n"); error = EIO; } else if (status & ISF_SR_PS) { device_printf(sc->isf_dev, "Program Error\n"); error = EIO; } else if (status & ISF_SR_BLS) { device_printf(sc->isf_dev, "Device Protect Error\n"); error = EIO; } isf_clear_status(sc); return(error); } static int isf_full_erase_status_check(struct isf_softc *sc, off_t off) { int error = 0; uint16_t status; status = isf_read_status(sc, off); if (status & ISF_SR_VPPS) { device_printf(sc->isf_dev, "Vpp Range Error\n"); error = EIO; } else if (status & (ISF_SR_PS|ISF_SR_ES)) { device_printf(sc->isf_dev, "Command Sequence Error\n"); error = EIO; } else if (status & ISF_SR_ES) { device_printf(sc->isf_dev, "Block Erase Error\n"); error = EIO; } else if (status & ISF_SR_BLS) { device_printf(sc->isf_dev, "Block Locked Error\n"); error = EIO; } isf_clear_status(sc); return(error); } static void isf_unlock_block(struct isf_softc *sc, off_t off) { isf_write_cmd(sc, off, ISF_CMD_LBS); isf_write_cmd(sc, off, ISF_CMD_UB); isf_write_cmd(sc, off, ISF_CMD_RA); } static void isf_lock_block(struct isf_softc *sc, off_t off) { isf_write_cmd(sc, off, ISF_CMD_LBS); isf_write_cmd(sc, off, ISF_CMD_LB); isf_write_cmd(sc, off, ISF_CMD_RA); } static void isf_read(struct isf_softc *sc, off_t off, void *data, size_t len) { KASSERT((uintptr_t)data % 2 == 0, ("%s: unaligned data %p", __func__, data)); KASSERT((len <= ISF_SECTORSIZE) && (len % 2 == 0), ("%s: invalid length %ju", __func__, len)); KASSERT(off % ISF_SECTORSIZE == 0, ("%s: invalid offset %ju\n", __func__, off)); /* * It is not permitted to read blocks that are in the process of * being erased, but we know they will be all 1's after the * erase so just report that value if asked about a block that * is being erased. */ if (sc->isf_bstate[off / ISF_ERASE_BLOCK] == BS_ERASING) memset(data, 0xFF, len); else bus_read_region_2(sc->isf_res, off, (uint16_t *)data, len / 2); } static int isf_write(struct isf_softc *sc, off_t off, void *data, size_t len) { int cycles, error = 0; uint16_t *dp; uint16_t status; off_t coff; KASSERT((uintptr_t)data % 2 == 0, ("%s: unaligned data %p", __func__, data)); KASSERT((len <= ISF_SECTORSIZE) && (len % 2 == 0), ("%s: invalid length %ju", __func__, len)); KASSERT(off % ISF_SECTORSIZE == 0, ("%s: invalid offset %ju\n", __func__, off)); KASSERT(!sc->isf_erasing, ("%s: trying to write while erasing\n", __func__)); KASSERT(sc->isf_bstate[off / ISF_ERASE_BLOCK] != BS_ERASING, ("%s: block being erased at %ju\n", __func__, off)); isf_unlock_block(sc, off); #ifdef ISF_BUFFER_PROGRAM for (dp = data, coff = off; dp - (uint16_t *)data < len / 2; dp += 32, coff += 64) { isf_clear_status(sc); isf_write_cmd(sc, coff, ISF_CMD_BPS); cycles = 0xFFFF; while ( !(isf_read_off(sc, coff) & ISF_SR_DWS) ) { if (cycles-- == 0) { device_printf(sc->isf_dev, "timeout waiting" " for write to start at 0x08%jx\n", (intmax_t)coff); return (EIO); } isf_write_cmd(sc, coff, ISF_CMD_BPS); } /* When writing N blocks, send N-1 as the count */ isf_write_cmd(sc, coff, 31); bus_write_region_2(sc->isf_res, coff, dp, 32); isf_write_cmd(sc, coff, ISF_CMD_BPC); status = isf_read_off(sc, coff); cycles = 0xFFFFF; while ( !(status & ISF_SR_DWS) ) { if (cycles-- == 0) { device_printf(sc->isf_dev, "timeout waiting" " for write to complete at 0x08%jx\n", (intmax_t)coff); error = EIO; break; } status = isf_read_off(sc, coff); } isf_full_status_check(sc, off); isf_write_cmd(sc, coff, ISF_CMD_RA); } #else for (dp = data, coff = off; dp - (uint16_t *)data < len / 2; dp++, coff += 2) { isf_write_cmd(sc, coff, ISF_CMD_WPS); bus_write_2(sc->isf_res, coff, *dp); status = isf_read_off(sc, coff); cycles=0xFFFFF; while ( !(status & ISF_SR_DWS) ) { if (cycles-- == 0) { device_printf(sc->isf_dev, "timeout waiting" " for write to complete at 0x08%jx\n", (intmax_t)coff); error = EIO; break; } status = isf_read_off(sc, coff); } } isf_full_status_check(sc, off); isf_write_cmd(sc, coff, ISF_CMD_RA); #endif isf_lock_block(sc, off); return error; } static void isf_erase_at(struct isf_softc *sc, off_t off) { int cycles; uint16_t status; isf_unlock_block(sc, off); isf_clear_status(sc); isf_write_cmd(sc, off, ISF_CMD_BES); isf_write_cmd(sc, off, ISF_CMD_BEC); cycles=0xFFFFFF; status = isf_read_off(sc, off); while ( !(status & ISF_SR_DWS) ) { #ifdef NOTYET ISF_SLEEP(sc, sc, hz); #endif if (cycles-- == 0) { device_printf(sc->isf_dev, "Giving up on erase\n"); break; } status = isf_read_off(sc, off); } isf_full_erase_status_check(sc, off); isf_lock_block(sc, off); isf_write_cmd(sc, off, ISF_CMD_RA); } static void isf_erase_range(struct isf_softc *sc, off_t blk_off, size_t size) { off_t off; off_t ms = sc->isf_disk->d_mediasize; KASSERT(blk_off % ISF_ERASE_BLOCK == 0, ("%s: invalid offset %ju\n", __func__, blk_off)); ISF_LOCK_ASSERT(sc); for (off = blk_off; off < blk_off + size; off += ISF_ERASE_BLOCK) { sc->isf_bstate[off / ISF_ERASE_BLOCK] = BS_ERASING; /* * The first or last 128K is four blocks depending which * part this is. For now, just assume both are and * erase four times. */ if (off == 0 || ms - off == ISF_ERASE_BLOCK) { isf_erase_at(sc, off); isf_erase_at(sc, off + 0x08000); isf_erase_at(sc, off + 0x10000); isf_erase_at(sc, off + 0x18000); } else isf_erase_at(sc, off); sc->isf_bstate[off / ISF_ERASE_BLOCK] = BS_STEADY; } } /* * disk(9) methods. */ static int isf_disk_ioctl(struct disk *disk, u_long cmd, void *data, int fflag, struct thread *td) { int error = 0; struct isf_softc *sc = disk->d_drv1; struct isf_range *ir; switch (cmd) { case ISF_ERASE: ir = data; if (ir->ir_off % ISF_ERASE_BLOCK != 0 || ir->ir_off >= disk->d_mediasize || ir->ir_size == 0 || ir->ir_size % ISF_ERASE_BLOCK != 0 || ir->ir_off + ir->ir_size > disk->d_mediasize) { error = EINVAL; break; } ISF_LOCK(sc); if (sc->isf_erasing) { ISF_UNLOCK(sc); error = EBUSY; break; } sc->isf_erasing = 1; isf_erase_range(sc, ir->ir_off, ir->ir_size); sc->isf_erasing = 0; ISF_UNLOCK(sc); break; default: error = EINVAL; } return (error); } static void isf_disk_strategy(struct bio *bp) { struct isf_softc *sc = bp->bio_disk->d_drv1; /* * We advertise a block size and maximum I/O size up the stack; catch * any attempts to not follow the rules. */ KASSERT(bp->bio_bcount == ISF_SECTORSIZE, ("%s: I/O size not %d", __func__, ISF_SECTORSIZE)); ISF_LOCK(sc); bioq_disksort(&sc->isf_bioq, bp); ISF_WAKEUP(sc); ISF_UNLOCK(sc); } static void isf_task(void *arg) { struct isf_softc *sc = arg; struct bio *bp; int ss = sc->isf_disk->d_sectorsize; int error, i; for (;;) { ISF_LOCK(sc); do { bp = bioq_first(&sc->isf_bioq); if (bp == NULL) { if (sc->isf_doomed) kproc_exit(0); else ISF_SLEEP(sc, sc, 0); } } while (bp == NULL); bioq_remove(&sc->isf_bioq, bp); error = 0; switch (bp->bio_cmd) { case BIO_READ: isf_read(sc, bp->bio_pblkno * ss, bp->bio_data, bp->bio_bcount); break; case BIO_WRITE: /* * In principle one could suspend the in-progress * erase, process any pending writes to other * blocks and then proceed, but that seems * overly complex for the likely usage modes. */ if (sc->isf_erasing) { error = EBUSY; break; } /* * Read in the block we want to write and check that * we're only setting bits to 0. If an erase would * be required return an I/O error. */ isf_read(sc, bp->bio_pblkno * ss, sc->isf_rbuf, bp->bio_bcount); for (i = 0; i < bp->bio_bcount / 2; i++) if ((sc->isf_rbuf[i] & ((uint16_t *)bp->bio_data)[i]) != ((uint16_t *)bp->bio_data)[i]) { device_printf(sc->isf_dev, "write" " requires erase at 0x%08jx\n", bp->bio_pblkno * ss); error = EIO; break; } if (error != 0) break; error = isf_write(sc, bp->bio_pblkno * ss, bp->bio_data, bp->bio_bcount); break; default: panic("%s: unsupported I/O operation %d", __func__, bp->bio_cmd); } if (error == 0) biodone(bp); else biofinish(bp, NULL, error); ISF_UNLOCK(sc); } } static void isf_dump_info(struct isf_softc *sc) { int i; int32_t reg; isf_write_cmd(sc, 0, ISF_CMD_RDI); device_printf(sc->isf_dev, "manufacturer code: 0x%04x\n", isf_read_reg(sc, ISF_REG_MC)); device_printf(sc->isf_dev, "device id code: 0x%04x\n", isf_read_reg(sc, ISF_REG_ID)); device_printf(sc->isf_dev, "read config register: 0x%04x\n", isf_read_reg(sc, ISF_REG_RCR)); device_printf(sc->isf_dev, "lock register 0: 0x%04x\n", isf_read_reg(sc, ISF_REG_L0)); device_printf(sc->isf_dev, "lock register 1: 0x%04x\n", isf_read_reg(sc, ISF_REG_L1)); device_printf(sc->isf_dev, "factory PPR: 0x%016jx\n", (uintmax_t)isf_read_reg64(sc, ISF_REG_FPP)); device_printf(sc->isf_dev, "user PPR (64-bit): 0x%016jx\n", (uintmax_t)isf_read_reg64(sc, ISF_REG_UPP)); for (reg = ISF_REG_PP1, i = 1; reg <= ISF_REG_PP16; reg += 8, i++) { /* XXX: big-endian ordering of uint64_t's */ device_printf(sc->isf_dev, "user PPR [%02d]: 0x%016jx%016jx\n", i, (uintmax_t)isf_read_reg64(sc, reg+4), (uintmax_t)isf_read_reg64(sc, reg)); } isf_write_cmd(sc, 0, ISF_CMD_RA); } static void isf_disk_insert(struct isf_softc *sc, off_t mediasize) { struct disk *disk; sc->isf_doomed = 0; sc->isf_erasing = 0; sc->isf_bstate = malloc(sizeof(*sc->isf_bstate) * (mediasize / ISF_ERASE_BLOCK), M_ISF, M_ZERO | M_WAITOK); kproc_create(&isf_task, sc, &sc->isf_proc, 0, 0, "isf"); disk = disk_alloc(); disk->d_drv1 = sc; disk->d_name = "isf"; disk->d_unit = sc->isf_unit; disk->d_strategy = isf_disk_strategy; disk->d_ioctl = isf_disk_ioctl; disk->d_sectorsize = ISF_SECTORSIZE; disk->d_mediasize = mediasize; disk->d_maxsize = ISF_SECTORSIZE; sc->isf_disk = disk; if (bootverbose) isf_dump_info(sc); disk_create(disk, DISK_VERSION); device_printf(sc->isf_dev, "%juM flash device\n", (uintmax_t)disk->d_mediasize / (1024 * 1024)); } static void isf_disk_remove(struct isf_softc *sc) { struct disk *disk; ISF_LOCK_ASSERT(sc); KASSERT(sc->isf_disk != NULL, ("%s: isf_disk NULL", __func__)); sc->isf_doomed = 1; ISF_WAKEUP(sc); ISF_SLEEP(sc, sc->isf_proc, 0); /* * XXXRW: Is it OK to call disk_destroy() under the mutex, or should * we be deferring that to the calling context once it is released? */ disk = sc->isf_disk; disk_gone(disk); disk_destroy(disk); sc->isf_disk = NULL; free(sc->isf_bstate, M_ISF); device_printf(sc->isf_dev, "flash device removed\n"); } int isf_attach(struct isf_softc *sc) { uint16_t id; u_long start, size; struct isf_chips *cp = chip_ids; start = rman_get_start(sc->isf_res); if (start % 2 != 0) { device_printf(sc->isf_dev, "Unsupported flash start alignment %lu\n", start); return (ENXIO); } isf_write_cmd(sc, 0, ISF_CMD_RDI); id = isf_read_reg(sc, ISF_REG_ID); while (cp->chip_id != id) cp++; if (cp->chip_desc == NULL) { device_printf(sc->isf_dev, "Unsupported device ID 0x%04x\n", id); return (ENXIO); } isf_write_cmd(sc, 0, ISF_CMD_RA); size = rman_get_size(sc->isf_res); if (size != cp->chip_size) { device_printf(sc->isf_dev, "Unsupported flash size %lu\n", size); return (ENXIO); } bioq_init(&sc->isf_bioq); ISF_LOCK_INIT(sc); sc->isf_disk = NULL; isf_disk_insert(sc, size); return(0); } void isf_detach(struct isf_softc *sc) { /* * Simulate a disk removal if one is present to deal with any pending * or queued I/O. This will occur as a result of a device driver * detach -- the Intel StrataFlash has no notion of removal itself. * * XXXRW: Is the locking here right? */ ISF_LOCK(sc); isf_disk_remove(sc); bioq_flush(&sc->isf_bioq, NULL, ENXIO); KASSERT(bioq_first(&sc->isf_bioq) == NULL, ("%s: non-empty bioq", __func__)); ISF_UNLOCK(sc); ISF_LOCK_DESTROY(sc); }