NetBSD/sys/dev/nand/nand_bbt.c

252 lines
6.4 KiB
C

/* $NetBSD: nand_bbt.c,v 1.8 2018/02/08 07:48:19 mrg Exp $ */
/*-
* Copyright (c) 2011 Department of Software Engineering,
* University of Szeged, Hungary
* Copyright (c) 2011 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.
*/
/*
* Implementation of Bad Block Tables (BBTs).
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: nand_bbt.c,v 1.8 2018/02/08 07:48:19 mrg Exp $");
#include <sys/param.h>
#include <sys/kmem.h>
#include "nand.h"
#include "nand_bbt.h"
void
nand_bbt_init(device_t self)
{
struct nand_softc *sc = device_private(self);
struct nand_chip *chip = &sc->sc_chip;
struct nand_bbt *bbt = &sc->sc_bbt;
bbt->nbbt_size = chip->nc_size / chip->nc_block_size / 4;
bbt->nbbt_bitmap = kmem_alloc(bbt->nbbt_size, KM_SLEEP);
memset(bbt->nbbt_bitmap, 0xff, bbt->nbbt_size);
}
void
nand_bbt_detach(device_t self)
{
struct nand_softc *sc = device_private(self);
struct nand_bbt *bbt = &sc->sc_bbt;
kmem_free(bbt->nbbt_bitmap, bbt->nbbt_size);
}
void
nand_bbt_scan(device_t self)
{
struct nand_softc *sc = device_private(self);
struct nand_chip *chip = &sc->sc_chip;
flash_off_t i, blocks, addr;
blocks = chip->nc_size / chip->nc_block_size;
aprint_normal_dev(self, "scanning for bad blocks\n");
addr = 0;
for (i = 0; i < blocks; i++) {
if (nand_isfactorybad(self, addr)) {
nand_bbt_block_markfactorybad(self, i);
} else if (nand_iswornoutbad(self, addr)) {
nand_bbt_block_markbad(self, i);
}
addr += chip->nc_block_size;
}
}
bool
nand_bbt_update(device_t self)
{
return true;
}
static bool
nand_bbt_page_has_bbt(device_t self, flash_off_t addr) {
struct nand_softc *sc = device_private(self);
struct nand_chip *chip = &sc->sc_chip;
uint8_t *oob = chip->nc_oob_cache;
nand_read_oob(self, addr, oob);
if (oob[NAND_BBT_OFFSET] == 'B' &&
oob[NAND_BBT_OFFSET + 1] == 'b' &&
oob[NAND_BBT_OFFSET + 2] == 't') {
return true;
} else {
return false;
}
}
static bool
nand_bbt_get_bbt_from_page(device_t self, flash_off_t addr)
{
struct nand_softc *sc = device_private(self);
struct nand_chip *chip = &sc->sc_chip;
struct nand_bbt *bbt = &sc->sc_bbt;
uint8_t *bbtp, *buf = chip->nc_page_cache;
size_t left, bbt_pages, i;
bbt_pages = bbt->nbbt_size / chip->nc_page_size;
if (bbt->nbbt_size % chip->nc_page_size)
bbt_pages++;
if (nand_isbad(self, addr)) {
return false;
}
if (nand_bbt_page_has_bbt(self, addr)) {
bbtp = bbt->nbbt_bitmap;
left = bbt->nbbt_size;
for (i = 0; i < bbt_pages; i++) {
nand_read_page(self, addr, buf);
if (i == bbt_pages - 1) {
KASSERT(left <= chip->nc_page_size);
memcpy(bbtp, buf, left);
} else {
memcpy(bbtp, buf, chip->nc_page_size);
}
bbtp += chip->nc_page_size;
left -= chip->nc_page_size;
addr += chip->nc_page_size;
}
return true;
} else {
return false;
}
}
bool
nand_bbt_load(device_t self)
{
struct nand_softc *sc = device_private(self);
struct nand_chip *chip = &sc->sc_chip;
flash_off_t blockaddr;
int n;
blockaddr = chip->nc_size - chip->nc_block_size;
/* XXX currently we check the last 4 blocks */
for (n = 0; n < 4; n++) {
if (nand_bbt_get_bbt_from_page(self, blockaddr)) {
break;
} else {
blockaddr -= chip->nc_block_size;
}
}
return true;
}
void
nand_bbt_block_markbad(device_t self, flash_off_t block)
{
if (nand_bbt_block_isbad(self, block)) {
aprint_error_dev(self,
"trying to mark block bad already marked in bbt\n");
}
/* XXX check if this is the correct marker */
nand_bbt_block_mark(self, block, NAND_BBT_MARKER_WORNOUT_BAD);
}
void
nand_bbt_block_markfactorybad(device_t self, flash_off_t block)
{
if (nand_bbt_block_isbad(self, block)) {
aprint_error_dev(self,
"trying to mark block factory bad already"
" marked in bbt\n");
}
nand_bbt_block_mark(self, block, NAND_BBT_MARKER_FACTORY_BAD);
}
void
nand_bbt_block_mark(device_t self, flash_off_t block, uint8_t marker)
{
struct nand_softc *sc = device_private(self);
struct nand_chip *chip = &sc->sc_chip;
struct nand_bbt *bbt = &sc->sc_bbt;
uint8_t clean;
__USE(chip);
KASSERT(block < chip->nc_size / chip->nc_block_size);
clean = (0xfc << ((block % 4) * 2));
marker = (marker << ((block % 4) * 2));
/* set byte containing the 2 bit marker for this block */
bbt->nbbt_bitmap[block / 4] &= clean;
bbt->nbbt_bitmap[block / 4] |= marker;
}
bool
nand_bbt_block_isbad(device_t self, flash_off_t block)
{
struct nand_softc *sc = device_private(self);
struct nand_chip *chip = &sc->sc_chip;
struct nand_bbt *bbt = &sc->sc_bbt;
uint8_t byte, marker;
bool result;
__USE(chip);
KASSERT(block < chip->nc_size / chip->nc_block_size);
/* get byte containing the 2 bit marker for this block */
byte = bbt->nbbt_bitmap[block / 4];
/* extract the 2 bit marker from the byte */
marker = (byte >> ((block % 4) * 2)) & 0x03;
switch (marker) {
case NAND_BBT_MARKER_FACTORY_BAD:
case NAND_BBT_MARKER_WORNOUT_BAD:
case NAND_BBT_MARKER_RESERVED:
result = true;
break;
case NAND_BBT_MARKER_GOOD:
result = false;
break;
default:
panic("error in marker extraction");
}
return result;
}