536 lines
14 KiB
C
536 lines
14 KiB
C
/* $NetBSD: omap2_nand.c,v 1.2 2019/11/03 13:45:57 jmcneill Exp $ */
|
|
|
|
/*-
|
|
* Copyright (c) 2010 Department of Software Engineering,
|
|
* University of Szeged, Hungary
|
|
* Copyright (c) 2010 Adam Hoka <ahoka@NetBSD.org>
|
|
* All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
|
* by the Department of Software Engineering, University of Szeged, Hungary
|
|
*
|
|
* 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 ``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 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.
|
|
*/
|
|
|
|
/* Device driver for the NAND controller found in Texas Instruments OMAP2
|
|
* and later SOCs.
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
__KERNEL_RCSID(0, "$NetBSD: omap2_nand.c,v 1.2 2019/11/03 13:45:57 jmcneill Exp $");
|
|
|
|
/* TODO move to opt_* */
|
|
#undef OMAP2_NAND_HARDWARE_ECC
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/cdefs.h>
|
|
#include <sys/device.h>
|
|
|
|
#include <sys/bus.h>
|
|
|
|
#include <arm/ti/omap2_gpmcreg.h>
|
|
|
|
#include <dev/nand/nand.h>
|
|
#include <dev/nand/onfi.h>
|
|
|
|
#include <dev/fdt/fdtvar.h>
|
|
|
|
extern struct flash_interface nand_flash_if;
|
|
extern int flash_print(void *, const char *);
|
|
|
|
/* GPMC_STATUS */
|
|
#define WAIT0 __BIT(8) /* active low */
|
|
|
|
/* GPMC_ECC_CONTROL */
|
|
#define ECCCLEAR __BIT(8)
|
|
#define ECCPOINTER __BITS(3,0)
|
|
|
|
/* GPMC_ECC_CONFIG */
|
|
#define ECCALGORITHM __BIT(16)
|
|
#define ECCCS __BITS(3,1)
|
|
#define ECC16B __BIT(7)
|
|
#define ECCENABLE __BIT(0)
|
|
/* GPMC_ECC_SIZE_CONFIG */
|
|
#define ECCSIZE1 __BITS(29,22)
|
|
|
|
/* GPMC_CONFIG1_i */
|
|
#define DEVICETYPE __BITS(11,10)
|
|
#define DEVICESIZE __BITS(13,12)
|
|
|
|
#define MASKEDINT(mask, integer) ((integer) << (ffs(mask) - 1) & mask)
|
|
|
|
/* NAND status register */
|
|
#define NAND_WP_BIT __BIT(4)
|
|
|
|
static int omap2_nand_match(device_t, cfdata_t, void *);
|
|
static void omap2_nand_attach(device_t, device_t, void *);
|
|
|
|
static void omap2_nand_command(device_t self, uint8_t command);
|
|
static void omap2_nand_address(device_t self, uint8_t address);
|
|
static void omap2_nand_busy(device_t self);
|
|
static void omap2_nand_read_1(device_t self, uint8_t *data);
|
|
static void omap2_nand_write_1(device_t self, uint8_t data);
|
|
static void omap2_nand_read_2(device_t self, uint16_t *data);
|
|
static void omap2_nand_write_2(device_t self, uint16_t data);
|
|
bool omap2_nand_isbusy(device_t self);
|
|
static void omap2_nand_read_buf_1(device_t self, void *buf, size_t len);
|
|
static void omap2_nand_read_buf_2(device_t self, void *buf, size_t len);
|
|
static void omap2_nand_write_buf_1(device_t self, const void *buf, size_t len);
|
|
static void omap2_nand_write_buf_2(device_t self, const void *buf, size_t len);
|
|
|
|
#ifdef OMAP2_NAND_HARDWARE_ECC
|
|
static int omap2_nand_ecc_init(device_t self);
|
|
static int omap2_nand_ecc_prepare(device_t self, int mode);
|
|
static int omap2_nand_ecc_compute(device_t self, const uint8_t *data, uint8_t *ecc);
|
|
static int omap2_nand_ecc_correct(device_t self, uint8_t *data, const uint8_t *oldecc,
|
|
const uint8_t *calcecc);
|
|
#endif
|
|
|
|
struct omap2_nand_softc {
|
|
device_t sc_dev;
|
|
device_t sc_nanddev;
|
|
|
|
int sc_cs;
|
|
int sc_buswidth; /* 0: 8bit, 1: 16bit */
|
|
|
|
struct nand_interface sc_nand_if;
|
|
|
|
bus_space_tag_t sc_iot;
|
|
bus_space_handle_t sc_ioh;
|
|
bus_space_handle_t sc_gpmc_ioh;
|
|
|
|
bus_size_t sc_cmd_reg;
|
|
bus_size_t sc_addr_reg;
|
|
bus_size_t sc_data_reg;
|
|
};
|
|
|
|
static const char * compatible[] = {
|
|
"ti,omap2-nand",
|
|
"ti,omap2-onenand",
|
|
NULL
|
|
};
|
|
|
|
CFATTACH_DECL_NEW(omapnand, sizeof(struct omap2_nand_softc), omap2_nand_match,
|
|
omap2_nand_attach, NULL, NULL);
|
|
|
|
static inline uint32_t
|
|
gpmc_register_read(struct omap2_nand_softc *sc, bus_size_t reg)
|
|
{
|
|
return bus_space_read_4(sc->sc_iot, sc->sc_gpmc_ioh, reg);
|
|
}
|
|
|
|
static inline void
|
|
gpmc_register_write(struct omap2_nand_softc *sc, bus_size_t reg, const uint32_t data)
|
|
{
|
|
bus_space_write_4(sc->sc_iot, sc->sc_gpmc_ioh, reg, data);
|
|
}
|
|
|
|
static void
|
|
omap2_nand_command(device_t self, uint8_t command)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, sc->sc_cmd_reg, command);
|
|
};
|
|
|
|
static void
|
|
omap2_nand_address(device_t self, uint8_t address)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, sc->sc_addr_reg, address);
|
|
};
|
|
|
|
bool
|
|
omap2_nand_isbusy(device_t self)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
uint8_t status;
|
|
|
|
DELAY(1); /* just to be sure we are not early */
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh,
|
|
sc->sc_cmd_reg, ONFI_READ_STATUS);
|
|
|
|
DELAY(1);
|
|
|
|
status = bus_space_read_1(sc->sc_iot,
|
|
sc->sc_ioh, sc->sc_data_reg);
|
|
|
|
return !(status & ONFI_STATUS_RDY);
|
|
};
|
|
|
|
static int
|
|
omap2_nand_match(device_t parent, cfdata_t match, void *aux)
|
|
{
|
|
struct fdt_attach_args * const faa = aux;
|
|
|
|
return of_match_compatible(faa->faa_phandle, compatible);
|
|
}
|
|
|
|
static void
|
|
omap2_nand_attach(device_t parent, device_t self, void *aux)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
struct fdt_attach_args * const faa = aux;
|
|
const int phandle = faa->faa_phandle;
|
|
struct flash_attach_args flash;
|
|
bus_addr_t addr, part_addr;
|
|
bus_size_t size, part_size;
|
|
const u_int *prop;
|
|
uint32_t val;
|
|
int len, child;
|
|
|
|
if (fdtbus_get_reg(OF_parent(phandle), 0, &addr, &size) != 0) {
|
|
aprint_error(": couldn't get registers\n");
|
|
return;
|
|
}
|
|
|
|
sc->sc_iot = faa->faa_bst;
|
|
sc->sc_dev = self;
|
|
|
|
prop = fdtbus_get_prop(phandle, "reg", &len);
|
|
if (prop == NULL || len < 4) {
|
|
aprint_error(": couldn't read reg property\n");
|
|
return;
|
|
}
|
|
|
|
sc->sc_cs = be32toh(prop[0]);
|
|
|
|
/* map i/o space */
|
|
if (bus_space_map(sc->sc_iot, addr, size, 0, &sc->sc_gpmc_ioh) != 0) {
|
|
aprint_error(": couldn't map registers\n");
|
|
return;
|
|
}
|
|
if (bus_space_subregion(sc->sc_iot, sc->sc_gpmc_ioh, GPMC_CS_CONFIG(sc->sc_cs), 0x30, &sc->sc_ioh) != 0) {
|
|
aprint_error(": couldn't map cs registers\n");
|
|
return;
|
|
}
|
|
|
|
aprint_naive("\n");
|
|
aprint_normal(": CS%d\n", sc->sc_cs);
|
|
|
|
sc->sc_cmd_reg = GPMC_NAND_COMMAND_0 - GPMC_CONFIG1_0;
|
|
sc->sc_addr_reg = GPMC_NAND_ADDRESS_0 - GPMC_CONFIG1_0;
|
|
sc->sc_data_reg = GPMC_NAND_DATA_0 - GPMC_CONFIG1_0;
|
|
|
|
/* turn off write protection if enabled */
|
|
val = gpmc_register_read(sc, GPMC_CONFIG);
|
|
val |= NAND_WP_BIT;
|
|
gpmc_register_write(sc, GPMC_CONFIG, val);
|
|
|
|
/*
|
|
* do the reset dance for NAND
|
|
*/
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh,
|
|
sc->sc_cmd_reg, ONFI_RESET);
|
|
|
|
omap2_nand_busy(self);
|
|
|
|
/* read GPMC_CONFIG1_i to get buswidth */
|
|
val = bus_space_read_4(sc->sc_iot, sc->sc_ioh, GPMC_CONFIG1_i);
|
|
|
|
if ((val & DEVICESIZE) == MASKEDINT(DEVICESIZE, 0x01)) {
|
|
/* 16bit */
|
|
sc->sc_buswidth = 1;
|
|
} else if ((val & DEVICESIZE) == MASKEDINT(DEVICESIZE, 0x00)) {
|
|
/* 8bit */
|
|
sc->sc_buswidth = 0;
|
|
} else {
|
|
panic("invalid buswidth reported by config1");
|
|
}
|
|
|
|
nand_init_interface(&sc->sc_nand_if);
|
|
|
|
sc->sc_nand_if.command = &omap2_nand_command;
|
|
sc->sc_nand_if.address = &omap2_nand_address;
|
|
sc->sc_nand_if.read_buf_1 = &omap2_nand_read_buf_1;
|
|
sc->sc_nand_if.read_buf_2 = &omap2_nand_read_buf_2;
|
|
sc->sc_nand_if.read_1 = &omap2_nand_read_1;
|
|
sc->sc_nand_if.read_2 = &omap2_nand_read_2;
|
|
sc->sc_nand_if.write_buf_1 = &omap2_nand_write_buf_1;
|
|
sc->sc_nand_if.write_buf_2 = &omap2_nand_write_buf_2;
|
|
sc->sc_nand_if.write_1 = &omap2_nand_write_1;
|
|
sc->sc_nand_if.write_2 = &omap2_nand_write_2;
|
|
sc->sc_nand_if.busy = &omap2_nand_busy;
|
|
|
|
#ifdef OMAP2_NAND_HARDWARE_ECC
|
|
omap2_nand_ecc_init(self);
|
|
sc->sc_nand_if.ecc_compute = &omap2_nand_ecc_compute;
|
|
sc->sc_nand_if.ecc_correct = &omap2_nand_ecc_correct;
|
|
sc->sc_nand_if.ecc_prepare = &omap2_nand_ecc_prepare;
|
|
sc->sc_nand_if.ecc.necc_code_size = 3;
|
|
sc->sc_nand_if.ecc.necc_block_size = 512;
|
|
sc->sc_nand_if.ecc.necc_type = NAND_ECC_TYPE_HW;
|
|
#else
|
|
sc->sc_nand_if.ecc.necc_code_size = 3;
|
|
sc->sc_nand_if.ecc.necc_block_size = 256;
|
|
#endif /* OMAP2_NAND_HARDWARE_ECC */
|
|
|
|
if (!pmf_device_register1(sc->sc_dev, NULL, NULL, NULL))
|
|
aprint_error_dev(sc->sc_dev,
|
|
"couldn't establish power handler\n");
|
|
|
|
sc->sc_nanddev = nand_attach_mi(&sc->sc_nand_if, sc->sc_dev);
|
|
if (sc->sc_nanddev == NULL)
|
|
return;
|
|
|
|
for (child = OF_child(phandle); child; child = OF_peer(child)) {
|
|
if (!fdtbus_status_okay(child))
|
|
continue;
|
|
|
|
if (fdtbus_get_reg(child, 0, &part_addr, &part_size) != 0) {
|
|
aprint_error_dev(self, "couldn't parse partition %s\n",
|
|
fdtbus_get_string(child, "name"));
|
|
continue;
|
|
}
|
|
|
|
memset(&flash, 0, sizeof(flash));
|
|
flash.flash_if = &nand_flash_if;
|
|
flash.partinfo.part_offset = part_addr;
|
|
flash.partinfo.part_size = part_size;
|
|
flash.partinfo.part_flags = 0;
|
|
flash.partinfo.part_name = fdtbus_get_string(child, "label");
|
|
if (flash.partinfo.part_name == NULL)
|
|
flash.partinfo.part_name = fdtbus_get_string(child, "name");
|
|
|
|
config_found_ia(sc->sc_nanddev, "flashbus", &flash, flash_print);
|
|
}
|
|
}
|
|
|
|
static void
|
|
omap2_nand_busy(device_t self)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
while (!(gpmc_register_read(sc, GPMC_STATUS) & WAIT0)) {
|
|
DELAY(1);
|
|
}
|
|
}
|
|
|
|
static void
|
|
omap2_nand_read_1(device_t self, uint8_t *data)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
*data = bus_space_read_1(sc->sc_iot, sc->sc_ioh, sc->sc_data_reg);
|
|
}
|
|
|
|
static void
|
|
omap2_nand_write_1(device_t self, uint8_t data)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh, sc->sc_data_reg, data);
|
|
}
|
|
|
|
static void
|
|
omap2_nand_read_2(device_t self, uint16_t *data)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
*data = bus_space_read_2(sc->sc_iot, sc->sc_ioh, sc->sc_data_reg);
|
|
}
|
|
|
|
static void
|
|
omap2_nand_write_2(device_t self, uint16_t data)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
bus_space_write_2(sc->sc_iot, sc->sc_ioh, sc->sc_data_reg, data);
|
|
}
|
|
|
|
static void
|
|
omap2_nand_read_buf_1(device_t self, void *buf, size_t len)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
KASSERT(buf != NULL);
|
|
KASSERT(len >= 1);
|
|
|
|
bus_space_read_multi_1(sc->sc_iot, sc->sc_ioh,
|
|
sc->sc_data_reg, buf, len);
|
|
}
|
|
|
|
static void
|
|
omap2_nand_read_buf_2(device_t self, void *buf, size_t len)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
KASSERT(buf != NULL);
|
|
KASSERT(len >= 2);
|
|
KASSERT(!(len & 0x01));
|
|
|
|
bus_space_read_multi_2(sc->sc_iot, sc->sc_ioh,
|
|
sc->sc_data_reg, buf, len / 2);
|
|
}
|
|
|
|
static void
|
|
omap2_nand_write_buf_1(device_t self, const void *buf, size_t len)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
KASSERT(buf != NULL);
|
|
KASSERT(len >= 1);
|
|
|
|
bus_space_write_multi_1(sc->sc_iot, sc->sc_ioh,
|
|
sc->sc_data_reg, buf, len);
|
|
}
|
|
|
|
static void
|
|
omap2_nand_write_buf_2(device_t self, const void *buf, size_t len)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
|
|
KASSERT(buf != NULL);
|
|
KASSERT(len >= 2);
|
|
KASSERT(!(len & 0x01));
|
|
|
|
bus_space_write_multi_2(sc->sc_iot, sc->sc_ioh,
|
|
sc->sc_data_reg, buf, len / 2);
|
|
}
|
|
|
|
#ifdef OMAP2_NAND_HARDWARE_ECC
|
|
static uint32_t
|
|
convert_ecc(const uint8_t *ecc)
|
|
{
|
|
return ecc[0] | (ecc[1] << 16) | ((ecc[2] & 0xf0) << 20) |
|
|
((ecc[2] & 0x0f) << 8);
|
|
}
|
|
|
|
static int
|
|
omap2_nand_ecc_init(device_t self)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
uint32_t val;
|
|
|
|
val = gpmc_register_read(sc, GPMC_ECC_CONTROL);
|
|
/* clear ecc, select ecc register 1 */
|
|
val &= ~ECCPOINTER;
|
|
val |= ECCCLEAR | MASKEDINT(ECCPOINTER, 1);
|
|
gpmc_register_write(sc, GPMC_ECC_CONTROL, val);
|
|
|
|
/* XXX too many MAGIC */
|
|
/* set ecc size to 512, set all regs to eccsize1*/
|
|
val = gpmc_register_read(sc, GPMC_ECC_SIZE_CONFIG);
|
|
val &= ~ECCSIZE1;
|
|
val |= MASKEDINT(ECCSIZE1, 512) | 0x0f;
|
|
gpmc_register_write(sc, GPMC_ECC_CONTROL, val);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
omap2_nand_ecc_compute(device_t self, const uint8_t *data, uint8_t *ecc)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
uint32_t val;
|
|
|
|
/* read ecc result register */
|
|
val = gpmc_register_read(sc, GPMC_ECC1_RESULT);
|
|
|
|
ecc[0] = val & 0xff;
|
|
ecc[1] = (val >> 16) & 0xff;
|
|
ecc[2] = ((val >> 8) & 0x0f) | ((val >> 20) & 0xf0);
|
|
|
|
/* disable ecc engine */
|
|
val = gpmc_register_read(sc, GPMC_ECC_CONFIG);
|
|
val &= ~ECCENABLE;
|
|
gpmc_register_write(sc, GPMC_ECC_CONFIG, val);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
omap2_nand_ecc_prepare(device_t self, int mode)
|
|
{
|
|
struct omap2_nand_softc *sc = device_private(self);
|
|
uint32_t val;
|
|
|
|
/* same for read/write */
|
|
switch (mode) {
|
|
case NAND_ECC_READ:
|
|
case NAND_ECC_WRITE:
|
|
val = gpmc_register_read(sc, GPMC_ECC_CONTROL);
|
|
/* clear ecc, select ecc register 1 */
|
|
val &= ~ECCPOINTER;
|
|
val |= ECCCLEAR | MASKEDINT(ECCPOINTER, 1);
|
|
gpmc_register_write(sc, GPMC_ECC_CONTROL, val);
|
|
|
|
val = gpmc_register_read(sc, GPMC_ECC_CONFIG);
|
|
val &= ~ECCCS;
|
|
val |= ECCENABLE | MASKEDINT(ECCCS, sc->sc_cs);
|
|
if (sc->sc_buswidth == 1)
|
|
val |= ECC16B;
|
|
else
|
|
val &= ~ECC16B;
|
|
gpmc_register_write(sc, GPMC_ECC_CONFIG, val);
|
|
|
|
break;
|
|
default:
|
|
aprint_error_dev(self, "invalid i/o mode for ecc prepare\n");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
omap2_nand_ecc_correct(device_t self, uint8_t *data, const uint8_t *oldecc,
|
|
const uint8_t *calcecc)
|
|
{
|
|
uint32_t oecc, cecc, xor;
|
|
uint16_t parity, offset;
|
|
uint8_t bit;
|
|
|
|
oecc = convert_ecc(oldecc);
|
|
cecc = convert_ecc(calcecc);
|
|
|
|
/* get the difference */
|
|
xor = oecc ^ cecc;
|
|
|
|
/* the data was correct if all bits are zero */
|
|
if (xor == 0x00)
|
|
return NAND_ECC_OK;
|
|
|
|
switch (popcount32(xor)) {
|
|
case 12:
|
|
/* single byte error */
|
|
parity = xor >> 16;
|
|
bit = (parity & 0x07);
|
|
offset = (parity >> 3) & 0x01ff;
|
|
/* correct bit */
|
|
data[offset] ^= (0x01 << bit);
|
|
return NAND_ECC_CORRECTED;
|
|
case 1:
|
|
return NAND_ECC_INVALID;
|
|
default:
|
|
/* erased page! */
|
|
if ((oecc == 0x0fff0fff) && (cecc == 0x00000000))
|
|
return NAND_ECC_OK;
|
|
|
|
return NAND_ECC_TWOBIT;
|
|
}
|
|
}
|
|
#endif /* !OMAP2_NAND_HARDWARE_ECC */
|