[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <4D377CA5.5090501@bluewatersys.com>
Date: Thu, 20 Jan 2011 13:07:01 +1300
From: Ryan Mallon <ryan@...ewatersys.com>
To: Charles Manning <cdhmanning@...il.com>
CC: linux-kernel@...r.kernel.org, linux-fsdevel@...r.kernel.org,
akpm@...ux-foundation.org
Subject: Re: [PATCH 05/10] Add yaffs2 file system: Tags processing code
On 01/14/2011 04:06 PM, Charles Manning wrote:
> Signed-off-by: Charles Manning <cdhmanning@...il.com>
A few comments below.
~Ryan
> ---
> fs/yaffs2/yaffs_packedtags1.c | 53 ++++++
> fs/yaffs2/yaffs_packedtags1.h | 39 ++++
> fs/yaffs2/yaffs_packedtags2.c | 187 +++++++++++++++++++
> fs/yaffs2/yaffs_packedtags2.h | 47 +++++
> fs/yaffs2/yaffs_tagscompat.c | 413 +++++++++++++++++++++++++++++++++++++++++
> fs/yaffs2/yaffs_tagscompat.h | 36 ++++
> 6 files changed, 775 insertions(+), 0 deletions(-)
> create mode 100644 fs/yaffs2/yaffs_packedtags1.c
> create mode 100644 fs/yaffs2/yaffs_packedtags1.h
> create mode 100644 fs/yaffs2/yaffs_packedtags2.c
> create mode 100644 fs/yaffs2/yaffs_packedtags2.h
> create mode 100644 fs/yaffs2/yaffs_tagscompat.c
> create mode 100644 fs/yaffs2/yaffs_tagscompat.h
>
> diff --git a/fs/yaffs2/yaffs_packedtags1.c b/fs/yaffs2/yaffs_packedtags1.c
> new file mode 100644
> index 0000000..873f065
> --- /dev/null
> +++ b/fs/yaffs2/yaffs_packedtags1.c
> @@ -0,0 +1,53 @@
> +/*
> + * YAFFS: Yet Another Flash File System. A NAND-flash specific file system.
> + *
> + * Copyright (C) 2002-2010 Aleph One Ltd.
> + * for Toby Churchill Ltd and Brightstar Engineering
> + *
> + * Created by Charles Manning <charles@...ph1.co.uk>
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 as
> + * published by the Free Software Foundation.
> + */
> +
> +#include "yaffs_packedtags1.h"
> +#include "yportenv.h"
> +
> +void yaffs_pack_tags1(struct yaffs_packed_tags1 *pt,
> + const struct yaffs_ext_tags *t)
> +{
> + pt->chunk_id = t->chunk_id;
> + pt->serial_number = t->serial_number;
> + pt->n_bytes = t->n_bytes;
> + pt->obj_id = t->obj_id;
> + pt->ecc = 0;
> + pt->deleted = (t->is_deleted) ? 0 : 1;
> + pt->unused_stuff = 0;
> + pt->should_be_ff = 0xFFFFFFFF;
> +}
> +
> +void yaffs_unpack_tags1(struct yaffs_ext_tags *t,
> + const struct yaffs_packed_tags1 *pt)
> +{
> + static const u8 all_ff[12] = {
> + 0xff, 0xff, 0xff, 0xff,
> + 0xff, 0xff, 0xff, 0xff,
> + 0xff, 0xff, 0xff, 0xff
> + };
> +
> + if (memcmp(all_ff, pt, sizeof(struct yaffs_packed_tags1))) {
> + t->block_bad = 0;
> + if (pt->should_be_ff != 0xFFFFFFFF)
You have a combination of upper and lower letters for hex constants. You
should pick one and be consistent.
> + t->block_bad = 1;
> + t->chunk_used = 1;
> + t->obj_id = pt->obj_id;
> + t->chunk_id = pt->chunk_id;
> + t->n_bytes = pt->n_bytes;
> + t->ecc_result = YAFFS_ECC_RESULT_NO_ERROR;
> + t->is_deleted = (pt->deleted) ? 0 : 1;
> + t->serial_number = pt->serial_number;
> + } else {
> + memset(t, 0, sizeof(struct yaffs_ext_tags));
> + }
> +}
> diff --git a/fs/yaffs2/yaffs_packedtags1.h b/fs/yaffs2/yaffs_packedtags1.h
> new file mode 100644
> index 0000000..d6861ff
> --- /dev/null
> +++ b/fs/yaffs2/yaffs_packedtags1.h
> @@ -0,0 +1,39 @@
> +/*
> + * YAFFS: Yet another Flash File System . A NAND-flash specific file system.
> + *
> + * Copyright (C) 2002-2010 Aleph One Ltd.
> + * for Toby Churchill Ltd and Brightstar Engineering
> + *
> + * Created by Charles Manning <charles@...ph1.co.uk>
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU Lesser General Public License version 2.1 as
> + * published by the Free Software Foundation.
> + *
> + * Note: Only YAFFS headers are LGPL, YAFFS C code is covered by GPL.
> + */
> +
> +/* This is used to pack YAFFS1 tags, not YAFFS2 tags. */
> +
> +#ifndef __YAFFS_PACKEDTAGS1_H__
> +#define __YAFFS_PACKEDTAGS1_H__
> +
> +#include "yaffs_guts.h"
> +
> +struct yaffs_packed_tags1 {
> + unsigned chunk_id:20;
> + unsigned serial_number:2;
> + unsigned n_bytes:10;
> + unsigned obj_id:18;
> + unsigned ecc:12;
> + unsigned deleted:1;
> + unsigned unused_stuff:1;
> + unsigned should_be_ff;
> +
> +};
> +
> +void yaffs_pack_tags1(struct yaffs_packed_tags1 *pt,
> + const struct yaffs_ext_tags *t);
> +void yaffs_unpack_tags1(struct yaffs_ext_tags *t,
> + const struct yaffs_packed_tags1 *pt);
> +#endif
> diff --git a/fs/yaffs2/yaffs_packedtags2.c b/fs/yaffs2/yaffs_packedtags2.c
> new file mode 100644
> index 0000000..2bb31b0
> --- /dev/null
> +++ b/fs/yaffs2/yaffs_packedtags2.c
> @@ -0,0 +1,187 @@
> +/*
> + * YAFFS: Yet Another Flash File System. A NAND-flash specific file system.
> + *
> + * Copyright (C) 2002-2010 Aleph One Ltd.
> + * for Toby Churchill Ltd and Brightstar Engineering
> + *
> + * Created by Charles Manning <charles@...ph1.co.uk>
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 as
> + * published by the Free Software Foundation.
> + */
> +
> +#include "yaffs_packedtags2.h"
> +#include "yportenv.h"
> +#include "yaffs_trace.h"
> +#include "yaffs_tagsvalidity.h"
> +
> +/* This code packs a set of extended tags into a binary structure for
> + * NAND storage
> + */
> +
> +/* Some of the information is "extra" struff which can be packed in to
> + * speed scanning
> + * This is defined by having the EXTRA_HEADER_INFO_FLAG set.
> + */
> +
> +/* Extra flags applied to chunk_id */
> +
> +#define EXTRA_HEADER_INFO_FLAG 0x80000000
> +#define EXTRA_SHRINK_FLAG 0x40000000
> +#define EXTRA_SHADOWS_FLAG 0x20000000
> +#define EXTRA_SPARE_FLAGS 0x10000000
> +
> +#define ALL_EXTRA_FLAGS 0xF0000000
> +
> +/* Also, the top 4 bits of the object Id are set to the object type. */
> +#define EXTRA_OBJECT_TYPE_SHIFT (28)
> +#define EXTRA_OBJECT_TYPE_MASK ((0x0F) << EXTRA_OBJECT_TYPE_SHIFT)
> +
> +static void yaffs_dump_packed_tags2_tags_only(
> + const struct yaffs_packed_tags2_tags_only *ptt)
> +{
> + yaffs_trace(YAFFS_TRACE_MTD,
> + "packed tags obj %d chunk %d byte %d seq %d",
> + ptt->obj_id, ptt->chunk_id, ptt->n_bytes, ptt->seq_number);
> +}
> +
> +static void yaffs_dump_packed_tags2(const struct yaffs_packed_tags2 *pt)
> +{
> + yaffs_dump_packed_tags2_tags_only(&pt->t);
> +}
> +
> +static void yaffs_dump_tags2(const struct yaffs_ext_tags *t)
> +{
> + yaffs_trace(YAFFS_TRACE_MTD,
> + "ext.tags eccres %d blkbad %d chused %d obj %d chunk%d byte %d del %d ser %d seq %d",
> + t->ecc_result, t->block_bad, t->chunk_used, t->obj_id,
> + t->chunk_id, t->n_bytes, t->is_deleted, t->serial_number,
> + t->seq_number);
> +
> +}
> +
> +void yaffs_pack_tags2_tags_only(struct yaffs_packed_tags2_tags_only *ptt,
> + const struct yaffs_ext_tags *t)
> +{
> + ptt->chunk_id = t->chunk_id;
> + ptt->seq_number = t->seq_number;
> + ptt->n_bytes = t->n_bytes;
> + ptt->obj_id = t->obj_id;
> +
> + if (t->chunk_id == 0 && t->extra_available) {
> + /* Store the extra header info instead */
> + /* We save the parent object in the chunk_id */
> + ptt->chunk_id = EXTRA_HEADER_INFO_FLAG | t->extra_parent_id;
> + if (t->extra_is_shrink)
> + ptt->chunk_id |= EXTRA_SHRINK_FLAG;
> + if (t->extra_shadows)
> + ptt->chunk_id |= EXTRA_SHADOWS_FLAG;
> +
> + ptt->obj_id &= ~EXTRA_OBJECT_TYPE_MASK;
> + ptt->obj_id |= (t->extra_obj_type << EXTRA_OBJECT_TYPE_SHIFT);
> +
> + if (t->extra_obj_type == YAFFS_OBJECT_TYPE_HARDLINK)
> + ptt->n_bytes = t->extra_equiv_id;
> + else if (t->extra_obj_type == YAFFS_OBJECT_TYPE_FILE)
> + ptt->n_bytes = t->extra_length;
> + else
> + ptt->n_bytes = 0;
> + }
> +
> + yaffs_dump_packed_tags2_tags_only(ptt);
> + yaffs_dump_tags2(t);
> +}
> +
> +void yaffs_pack_tags2(struct yaffs_packed_tags2 *pt,
> + const struct yaffs_ext_tags *t, int tags_ecc)
> +{
> + yaffs_pack_tags2_tags_only(&pt->t, t);
> +
> + if (tags_ecc)
> + yaffs_ecc_calc_other((unsigned char *)&pt->t,
> + sizeof(struct
> + yaffs_packed_tags2_tags_only),
> + &pt->ecc);
> +}
> +
> +void yaffs_unpack_tags2_tags_only(struct yaffs_ext_tags *t,
> + struct yaffs_packed_tags2_tags_only *ptt)
> +{
> + memset(t, 0, sizeof(struct yaffs_ext_tags));
> + yaffs_init_tags(t);
> +
> + if (ptt->seq_number != 0xFFFFFFFF) {
> + t->block_bad = 0;
> + t->chunk_used = 1;
> + t->obj_id = ptt->obj_id;
> + t->chunk_id = ptt->chunk_id;
> + t->n_bytes = ptt->n_bytes;
> + t->is_deleted = 0;
> + t->serial_number = 0;
> + t->seq_number = ptt->seq_number;
> +
> + /* Do extra header info stuff */
> + if (ptt->chunk_id & EXTRA_HEADER_INFO_FLAG) {
> + t->chunk_id = 0;
> + t->n_bytes = 0;
> +
> + t->extra_available = 1;
> + t->extra_parent_id =
> + ptt->chunk_id & (~(ALL_EXTRA_FLAGS));
> + t->extra_is_shrink =
> + (ptt->chunk_id & EXTRA_SHRINK_FLAG) ? 1 : 0;
> + t->extra_shadows =
> + (ptt->chunk_id & EXTRA_SHADOWS_FLAG) ? 1 : 0;
> + t->extra_obj_type =
> + ptt->obj_id >> EXTRA_OBJECT_TYPE_SHIFT;
> + t->obj_id &= ~EXTRA_OBJECT_TYPE_MASK;
> +
> + if (t->extra_obj_type == YAFFS_OBJECT_TYPE_HARDLINK)
> + t->extra_equiv_id = ptt->n_bytes;
> + else
> + t->extra_length = ptt->n_bytes;
> + }
> + }
> + yaffs_dump_packed_tags2_tags_only(ptt);
> + yaffs_dump_tags2(t);
> +}
> +
> +void yaffs_unpack_tags2(struct yaffs_ext_tags *t, struct yaffs_packed_tags2 *pt,
> + int tags_ecc)
> +{
> + enum yaffs_ecc_result ecc_result = YAFFS_ECC_RESULT_NO_ERROR;
> +
> + if (pt->t.seq_number != 0xFFFFFFFF && tags_ecc) {
> + /* Chunk is in use and we need to do ECC */
> +
> + struct yaffs_ecc_other ecc;
> + int result;
> + yaffs_ecc_calc_other((unsigned char *)&pt->t,
> + sizeof(struct yaffs_packed_tags2_tags_only),
> + &ecc);
> + result =
> + yaffs_ecc_correct_other((unsigned char *)&pt->t,
> + sizeof(struct yaffs_packed_tags2_tags_only),
> + &pt->ecc, &ecc);
> + switch (result) {
> + case 0:
> + ecc_result = YAFFS_ECC_RESULT_NO_ERROR;
> + break;
> + case 1:
> + ecc_result = YAFFS_ECC_RESULT_FIXED;
> + break;
> + case -1:
> + ecc_result = YAFFS_ECC_RESULT_UNFIXED;
> + break;
> + default:
> + ecc_result = YAFFS_ECC_RESULT_UNKNOWN;
> + }
> + }
> + yaffs_unpack_tags2_tags_only(t, &pt->t);
> +
> + t->ecc_result = ecc_result;
> +
> + yaffs_dump_packed_tags2(pt);
> + yaffs_dump_tags2(t);
> +}
> diff --git a/fs/yaffs2/yaffs_packedtags2.h b/fs/yaffs2/yaffs_packedtags2.h
> new file mode 100644
> index 0000000..f329669
> --- /dev/null
> +++ b/fs/yaffs2/yaffs_packedtags2.h
> @@ -0,0 +1,47 @@
> +/*
> + * YAFFS: Yet another Flash File System . A NAND-flash specific file system.
> + *
> + * Copyright (C) 2002-2010 Aleph One Ltd.
> + * for Toby Churchill Ltd and Brightstar Engineering
> + *
> + * Created by Charles Manning <charles@...ph1.co.uk>
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU Lesser General Public License version 2.1 as
> + * published by the Free Software Foundation.
> + *
> + * Note: Only YAFFS headers are LGPL, YAFFS C code is covered by GPL.
> + */
> +
> +/* This is used to pack YAFFS2 tags, not YAFFS1tags. */
> +
> +#ifndef __YAFFS_PACKEDTAGS2_H__
> +#define __YAFFS_PACKEDTAGS2_H__
> +
> +#include "yaffs_guts.h"
> +#include "yaffs_ecc.h"
> +
> +struct yaffs_packed_tags2_tags_only {
> + unsigned seq_number;
> + unsigned obj_id;
> + unsigned chunk_id;
> + unsigned n_bytes;
> +};
> +
> +struct yaffs_packed_tags2 {
> + struct yaffs_packed_tags2_tags_only t;
> + struct yaffs_ecc_other ecc;
> +};
> +
> +/* Full packed tags with ECC, used for oob tags */
> +void yaffs_pack_tags2(struct yaffs_packed_tags2 *pt,
> + const struct yaffs_ext_tags *t, int tags_ecc);
> +void yaffs_unpack_tags2(struct yaffs_ext_tags *t, struct yaffs_packed_tags2 *pt,
> + int tags_ecc);
> +
> +/* Only the tags part (no ECC for use with inband tags */
> +void yaffs_pack_tags2_tags_only(struct yaffs_packed_tags2_tags_only *pt,
> + const struct yaffs_ext_tags *t);
> +void yaffs_unpack_tags2_tags_only(struct yaffs_ext_tags *t,
> + struct yaffs_packed_tags2_tags_only *pt);
> +#endif
> diff --git a/fs/yaffs2/yaffs_tagscompat.c b/fs/yaffs2/yaffs_tagscompat.c
> new file mode 100644
> index 0000000..35d33c9
> --- /dev/null
> +++ b/fs/yaffs2/yaffs_tagscompat.c
> @@ -0,0 +1,413 @@
> +/*
> + * YAFFS: Yet Another Flash File System. A NAND-flash specific file system.
> + *
> + * Copyright (C) 2002-2010 Aleph One Ltd.
> + * for Toby Churchill Ltd and Brightstar Engineering
> + *
> + * Created by Charles Manning <charles@...ph1.co.uk>
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 as
> + * published by the Free Software Foundation.
> + */
> +
> +#include "yaffs_guts.h"
> +#include "yaffs_tagscompat.h"
> +#include "yaffs_ecc.h"
> +#include "yaffs_getblockinfo.h"
> +#include "yaffs_trace.h"
> +
> +static void yaffs_handle_rd_data_error(struct yaffs_dev *dev, int nand_chunk);
> +
> +
> +/********** Tags ECC calculations *********/
> +
> +void yaffs_calc_ecc(const u8 *data, struct yaffs_spare *spare)
> +{
> + yaffs_ecc_cacl(data, spare->ecc1);
> + yaffs_ecc_cacl(&data[256], spare->ecc2);
> +}
> +
> +void yaffs_calc_tags_ecc(struct yaffs_tags *tags)
> +{
> + /* Calculate an ecc */
> + unsigned char *b = ((union yaffs_tags_union *)tags)->as_bytes;
> + unsigned i, j;
> + unsigned ecc = 0;
> + unsigned bit = 0;
> +
> + tags->ecc = 0;
> +
> + for (i = 0; i < 8; i++) {
> + for (j = 1; j & 0xff; j <<= 1) {
> + bit++;
> + if (b[i] & j)
> + ecc ^= bit;
> + }
> + }
> + tags->ecc = ecc;
> +}
> +
> +int yaffs_check_tags_ecc(struct yaffs_tags *tags)
> +{
> + unsigned ecc = tags->ecc;
> +
> + yaffs_calc_tags_ecc(tags);
> +
> + ecc ^= tags->ecc;
> +
> + if (ecc && ecc <= 64) {
> + /* TODO: Handle the failure better. Retire? */
> + unsigned char *b = ((union yaffs_tags_union *)tags)->as_bytes;
> +
> + ecc--;
> +
> + b[ecc / 8] ^= (1 << (ecc & 7));
> +
> + /* Now recvalc the ecc */
> + yaffs_calc_tags_ecc(tags);
> +
> + return 1; /* recovered error */
> + } else if (ecc) {
> + /* Wierd ecc failure value */
> + /* TODO Need to do somethiong here */
Maybe print a warning at least?
> + return -1; /* unrecovered error */
> + }
> + return 0;
> +}
> +
> +/********** Tags **********/
> +
> +static void yaffs_load_tags_to_spare(struct yaffs_spare *spare_ptr,
> + struct yaffs_tags *tags_ptr)
> +{
> + union yaffs_tags_union *tu = (union yaffs_tags_union *)tags_ptr;
> +
> + yaffs_calc_tags_ecc(tags_ptr);
> +
> + spare_ptr->tb0 = tu->as_bytes[0];
> + spare_ptr->tb1 = tu->as_bytes[1];
> + spare_ptr->tb2 = tu->as_bytes[2];
> + spare_ptr->tb3 = tu->as_bytes[3];
> + spare_ptr->tb4 = tu->as_bytes[4];
> + spare_ptr->tb5 = tu->as_bytes[5];
> + spare_ptr->tb6 = tu->as_bytes[6];
> + spare_ptr->tb7 = tu->as_bytes[7];
> +}
> +
> +static void yaffs_get_tags_from_spare(struct yaffs_dev *dev,
> + struct yaffs_spare *spare_ptr,
> + struct yaffs_tags *tags_ptr)
> +{
> + union yaffs_tags_union *tu = (union yaffs_tags_union *)tags_ptr;
> + int result;
> +
> + tu->as_bytes[0] = spare_ptr->tb0;
> + tu->as_bytes[1] = spare_ptr->tb1;
> + tu->as_bytes[2] = spare_ptr->tb2;
> + tu->as_bytes[3] = spare_ptr->tb3;
> + tu->as_bytes[4] = spare_ptr->tb4;
> + tu->as_bytes[5] = spare_ptr->tb5;
> + tu->as_bytes[6] = spare_ptr->tb6;
> + tu->as_bytes[7] = spare_ptr->tb7;
> +
> + result = yaffs_check_tags_ecc(tags_ptr);
> + if (result > 0)
> + dev->n_tags_ecc_fixed++;
> + else if (result < 0)
> + dev->n_tags_ecc_unfixed++;
> +}
> +
> +static void yaffs_spare_init(struct yaffs_spare *spare)
> +{
> + memset(spare, 0xFF, sizeof(struct yaffs_spare));
> +}
This could be replaced by having a global:
static const struct yaffs_spare_ff = {
.tb0 = 0xff;
.tb1 = 0xff;
...
};
> +
> +static int yaffs_wr_nand(struct yaffs_dev *dev,
> + int nand_chunk, const u8 *data,
> + struct yaffs_spare *spare)
> +{
> + if (nand_chunk < dev->param.start_block * dev->param.chunks_per_block) {
> + yaffs_trace(YAFFS_TRACE_ERROR,
> + "**>> yaffs chunk %d is not valid",
> + nand_chunk);
> + return YAFFS_FAIL;
> + }
> +
> + return dev->param.write_chunk_fn(dev, nand_chunk, data, spare);
> +}
> +
> +static int yaffs_rd_chunk_nand(struct yaffs_dev *dev,
> + int nand_chunk,
> + u8 *data,
> + struct yaffs_spare *spare,
> + enum yaffs_ecc_result *ecc_result,
> + int correct_errors)
> +{
> + int ret_val;
> + struct yaffs_spare local_spare;
> +
> + if (!spare) {
> + /* If we don't have a real spare, then we use a local one. */
> + /* Need this for the calculation of the ecc */
> + spare = &local_spare;
Is this safe? Can we get left pointing at something which isn't there
anymore?
> + }
> +
> + if (!dev->param.use_nand_ecc) {
> + ret_val =
> + dev->param.read_chunk_fn(dev, nand_chunk, data, spare);
> + if (data && correct_errors) {
> + /* Do ECC correction */
> + /* Todo handle any errors */
> + int ecc_result1, ecc_result2;
> + u8 calc_ecc[3];
> +
> + yaffs_ecc_cacl(data, calc_ecc);
> + ecc_result1 =
> + yaffs_ecc_correct(data, spare->ecc1, calc_ecc);
> + yaffs_ecc_cacl(&data[256], calc_ecc);
> + ecc_result2 =
> + yaffs_ecc_correct(&data[256], spare->ecc2,
> + calc_ecc);
> +
> + if (ecc_result1 > 0) {
> + yaffs_trace(YAFFS_TRACE_ERROR,
> + "**>>yaffs ecc error fix performed on chunk %d:0",
> + nand_chunk);
> + dev->n_ecc_fixed++;
> + } else if (ecc_result1 < 0) {
> + yaffs_trace(YAFFS_TRACE_ERROR,
> + "**>>yaffs ecc error unfixed on chunk %d:0",
> + nand_chunk);
> + dev->n_ecc_unfixed++;
> + }
> +
> + if (ecc_result2 > 0) {
> + yaffs_trace(YAFFS_TRACE_ERROR,
> + "**>>yaffs ecc error fix performed on chunk %d:1",
> + nand_chunk);
> + dev->n_ecc_fixed++;
> + } else if (ecc_result2 < 0) {
> + yaffs_trace(YAFFS_TRACE_ERROR,
> + "**>>yaffs ecc error unfixed on chunk %d:1",
> + nand_chunk);
> + dev->n_ecc_unfixed++;
> + }
> +
> + if (ecc_result1 || ecc_result2) {
> + /* We had a data problem on this page */
> + yaffs_handle_rd_data_error(dev, nand_chunk);
> + }
> +
> + if (ecc_result1 < 0 || ecc_result2 < 0)
> + *ecc_result = YAFFS_ECC_RESULT_UNFIXED;
> + else if (ecc_result1 > 0 || ecc_result2 > 0)
> + *ecc_result = YAFFS_ECC_RESULT_FIXED;
> + else
> + *ecc_result = YAFFS_ECC_RESULT_NO_ERROR;
> + }
> + } else {
> + /* Must allocate enough memory for spare+2*sizeof(int) */
> + /* for ecc results from device. */
> + struct yaffs_nand_spare nspare;
> +
> + memset(&nspare, 0, sizeof(nspare));
> +
> + ret_val = dev->param.read_chunk_fn(dev, nand_chunk, data,
> + (struct yaffs_spare *)
> + &nspare);
> + memcpy(spare, &nspare, sizeof(struct yaffs_spare));
> + if (data && correct_errors) {
> + if (nspare.eccres1 > 0) {
> + yaffs_trace(YAFFS_TRACE_ERROR,
> + "**>>mtd ecc error fix performed on chunk %d:0",
> + nand_chunk);
> + } else if (nspare.eccres1 < 0) {
> + yaffs_trace(YAFFS_TRACE_ERROR,
> + "**>>mtd ecc error unfixed on chunk %d:0",
> + nand_chunk);
> + }
> +
> + if (nspare.eccres2 > 0) {
> + yaffs_trace(YAFFS_TRACE_ERROR,
> + "**>>mtd ecc error fix performed on chunk %d:1",
> + nand_chunk);
> + } else if (nspare.eccres2 < 0) {
> + yaffs_trace(YAFFS_TRACE_ERROR,
> + "**>>mtd ecc error unfixed on chunk %d:1",
> + nand_chunk);
> + }
> +
> + if (nspare.eccres1 || nspare.eccres2) {
> + /* We had a data problem on this page */
> + yaffs_handle_rd_data_error(dev, nand_chunk);
> + }
> +
> + if (nspare.eccres1 < 0 || nspare.eccres2 < 0)
> + *ecc_result = YAFFS_ECC_RESULT_UNFIXED;
> + else if (nspare.eccres1 > 0 || nspare.eccres2 > 0)
> + *ecc_result = YAFFS_ECC_RESULT_FIXED;
> + else
> + *ecc_result = YAFFS_ECC_RESULT_NO_ERROR;
> +
> + }
> + }
> + return ret_val;
> +}
> +
> +/*
> + * Functions for robustisizing
> + */
> +
> +static void yaffs_handle_rd_data_error(struct yaffs_dev *dev, int nand_chunk)
> +{
> + int flash_block = nand_chunk / dev->param.chunks_per_block;
> +
> + /* Mark the block for retirement */
> + yaffs_get_block_info(dev, flash_block + dev->block_offset)->
> + needs_retiring = 1;
> + yaffs_trace(YAFFS_TRACE_ERROR | YAFFS_TRACE_BAD_BLOCKS,
> + "**>>Block %d marked for retirement",
> + flash_block);
> +
> + /* TODO:
> + * Just do a garbage collection on the affected block
> + * then retire the block
> + * NB recursion
> + */
> +}
> +
> +int yaffs_tags_compat_wr(struct yaffs_dev *dev,
> + int nand_chunk,
> + const u8 *data, const struct yaffs_ext_tags *ext_tags)
> +{
> + struct yaffs_spare spare;
> + struct yaffs_tags tags;
> +
> + yaffs_spare_init(&spare);
> +
> + if (ext_tags->is_deleted)
> + spare.page_status = 0;
> + else {
> + tags.obj_id = ext_tags->obj_id;
> + tags.chunk_id = ext_tags->chunk_id;
> +
> + tags.n_bytes_lsb = ext_tags->n_bytes & 0x3ff;
> +
> + if (dev->data_bytes_per_chunk >= 1024)
> + tags.n_bytes_msb = (ext_tags->n_bytes >> 10) & 3;
> + else
> + tags.n_bytes_msb = 3;
> +
> + tags.serial_number = ext_tags->serial_number;
> +
> + if (!dev->param.use_nand_ecc && data)
> + yaffs_calc_ecc(data, &spare);
> +
> + yaffs_load_tags_to_spare(&spare, &tags);
> +
> + }
> +
> + return yaffs_wr_nand(dev, nand_chunk, data, &spare);
> +}
> +
> +int yaffs_tags_compat_rd(struct yaffs_dev *dev,
> + int nand_chunk,
> + u8 *data, struct yaffs_ext_tags *ext_tags)
> +{
> + struct yaffs_spare spare;
> + struct yaffs_tags tags;
> + enum yaffs_ecc_result ecc_result = YAFFS_ECC_RESULT_UNKNOWN;
> + static struct yaffs_spare spare_ff;
> + static int init;
> +
> + if (!init) {
> + memset(&spare_ff, 0xFF, sizeof(spare_ff));
> + init = 1;
Don't do this. Just have a const initialised struct.
> + }
> +
> + if (yaffs_rd_chunk_nand(dev, nand_chunk,
> + data, &spare, &ecc_result, 1)) {
> + /* ext_tags may be NULL */
> + if (ext_tags) {
> +
> + int deleted =
> + (hweight8(spare.page_status) < 7) ? 1 : 0;
> +
> + ext_tags->is_deleted = deleted;
> + ext_tags->ecc_result = ecc_result;
> + ext_tags->block_bad = 0; /* We're reading it */
> + /* therefore it is not a bad block */
> + ext_tags->chunk_used =
> + (memcmp(&spare_ff, &spare, sizeof(spare_ff)) !=
> + 0) ? 1 : 0;
> +
> + if (ext_tags->chunk_used) {
> + yaffs_get_tags_from_spare(dev, &spare, &tags);
> +
> + ext_tags->obj_id = tags.obj_id;
> + ext_tags->chunk_id = tags.chunk_id;
> + ext_tags->n_bytes = tags.n_bytes_lsb;
> +
> + if (dev->data_bytes_per_chunk >= 1024)
> + ext_tags->n_bytes |=
> + (((unsigned)tags.
> + n_bytes_msb) << 10);
> +
> + ext_tags->serial_number = tags.serial_number;
> + }
> + }
> +
> + return YAFFS_OK;
> + } else {
> + return YAFFS_FAIL;
> + }
> +}
> +
> +int yaffs_tags_compat_mark_bad(struct yaffs_dev *dev, int flash_block)
> +{
> + struct yaffs_spare spare;
> +
> + memset(&spare, 0xff, sizeof(struct yaffs_spare));
> +
> + spare.block_status = 'Y';
> +
> + yaffs_wr_nand(dev, flash_block * dev->param.chunks_per_block, NULL,
> + &spare);
> + yaffs_wr_nand(dev, flash_block * dev->param.chunks_per_block + 1,
> + NULL, &spare);
> +
> + return YAFFS_OK;
> +}
> +
> +int yaffs_tags_compat_query_block(struct yaffs_dev *dev,
> + int block_no,
> + enum yaffs_block_state *state,
> + u32 *seq_number)
> +{
> + struct yaffs_spare spare0, spare1;
> + static struct yaffs_spare spare_ff;
> + static int init;
> + enum yaffs_ecc_result dummy;
> +
> + if (!init) {
> + memset(&spare_ff, 0xFF, sizeof(spare_ff));
> + init = 1;
> + }
> +
> + *seq_number = 0;
> +
> + yaffs_rd_chunk_nand(dev, block_no * dev->param.chunks_per_block, NULL,
> + &spare0, &dummy, 1);
> + yaffs_rd_chunk_nand(dev, block_no * dev->param.chunks_per_block + 1,
> + NULL, &spare1, &dummy, 1);
> +
> + if (hweight8(spare0.block_status & spare1.block_status) < 7)
> + *state = YAFFS_BLOCK_STATE_DEAD;
> + else if (memcmp(&spare_ff, &spare0, sizeof(spare_ff)) == 0)
> + *state = YAFFS_BLOCK_STATE_EMPTY;
> + else
> + *state = YAFFS_BLOCK_STATE_NEEDS_SCANNING;
> +
> + return YAFFS_OK;
> +}
> diff --git a/fs/yaffs2/yaffs_tagscompat.h b/fs/yaffs2/yaffs_tagscompat.h
> new file mode 100644
> index 0000000..ae15e0b
> --- /dev/null
> +++ b/fs/yaffs2/yaffs_tagscompat.h
> @@ -0,0 +1,36 @@
> +/*
> + * YAFFS: Yet another Flash File System . A NAND-flash specific file system.
> + *
> + * Copyright (C) 2002-2010 Aleph One Ltd.
> + * for Toby Churchill Ltd and Brightstar Engineering
> + *
> + * Created by Charles Manning <charles@...ph1.co.uk>
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU Lesser General Public License version 2.1 as
> + * published by the Free Software Foundation.
> + *
> + * Note: Only YAFFS headers are LGPL, YAFFS C code is covered by GPL.
> + */
> +
> +#ifndef __YAFFS_TAGSCOMPAT_H__
> +#define __YAFFS_TAGSCOMPAT_H__
> +
> +#include "yaffs_guts.h"
> +int yaffs_tags_compat_wr(struct yaffs_dev *dev,
> + int nand_chunk,
> + const u8 *data, const struct yaffs_ext_tags *tags);
> +int yaffs_tags_compat_rd(struct yaffs_dev *dev,
> + int nand_chunk,
> + u8 *data, struct yaffs_ext_tags *tags);
> +int yaffs_tags_compat_mark_bad(struct yaffs_dev *dev, int block_no);
> +int yaffs_tags_compat_query_block(struct yaffs_dev *dev,
> + int block_no,
> + enum yaffs_block_state *state,
> + u32 *seq_number);
> +
> +void yaffs_calc_tags_ecc(struct yaffs_tags *tags);
> +int yaffs_check_tags_ecc(struct yaffs_tags *tags);
> +int yaffs_count_bits(u8 byte);
> +
> +#endif
--
Bluewater Systems Ltd - ARM Technology Solution Centre
Ryan Mallon 5 Amuri Park, 404 Barbadoes St
ryan@...ewatersys.com PO Box 13 889, Christchurch 8013
http://www.bluewatersys.com New Zealand
Phone: +64 3 3779127 Freecall: Australia 1800 148 751
Fax: +64 3 3779135 USA 1800 261 2934
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
Powered by blists - more mailing lists