Split qsort from the main Yaffs Direct base
[yaffs2.git] / yaffs_mtdif_multi.c
index e61699d6e85f4475a2ddf87630c7161d0daec1a7..7c01461ab93de21de9e5e272ec5bcfd58504255c 100644 (file)
 #include "linux/kernel.h"
 #include "linux/version.h"
 #include "linux/types.h"
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(3, 14, 0))
+#include "uapi/linux/major.h"
+#endif
 
 #include "yaffs_trace.h"
 #include "yaffs_guts.h"
 #include "yaffs_linux.h"
 
-
 #if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 2, 0))
 #define MTD_OPS_AUTO_OOB MTD_OOB_AUTO
 #endif
 
 
+#if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 4, 0))
+#define mtd_erase(m, ei) (m)->erase(m, ei)
+#define mtd_write_oob(m, addr, pops) (m)->write_oob(m, addr, pops)
+#define mtd_read_oob(m, addr, pops) (m)->read_oob(m, addr, pops)
+#define mtd_block_isbad(m, offs) (m)->block_isbad(m, offs)
+#define mtd_block_markbad(m, offs) (m)->block_markbad(m, offs)
+#endif
+
+
+
 int nandmtd_erase_block(struct yaffs_dev *dev, int block_no)
 {
        struct mtd_info *mtd = yaffs_dev_to_mtd(dev);
@@ -50,7 +62,7 @@ int nandmtd_erase_block(struct yaffs_dev *dev, int block_no)
        ei.callback = NULL;
        ei.priv = (u_long) dev;
 
-       retval = mtd->erase(mtd, &ei);
+       retval = mtd_erase(mtd, &ei);
 
        if (retval == 0)
                return YAFFS_OK;
@@ -68,6 +80,20 @@ static       int yaffs_mtd_write(struct yaffs_dev *dev, int nand_chunk,
        struct mtd_oob_ops ops;
        int retval;
 
+       yaffs_trace(YAFFS_TRACE_MTD,
+                       "yaffs_mtd_write(%p, %d, %p, %d, %p, %d)\n",
+                       dev, nand_chunk, data, data_len, oob, oob_len);
+
+       if (!data || !data_len) {
+               data = NULL;
+               data_len = 0;
+       }
+
+       if (!oob || !oob_len) {
+               oob = NULL;
+               oob_len = 0;
+       }
+
        addr = ((loff_t) nand_chunk) * dev->param.total_bytes_per_chunk;
        memset(&ops, 0, sizeof(ops));
        ops.mode = MTD_OPS_AUTO_OOB;
@@ -76,7 +102,7 @@ static       int yaffs_mtd_write(struct yaffs_dev *dev, int nand_chunk,
        ops.datbuf = (u8 *)data;
        ops.oobbuf = (u8 *)oob;
 
-       retval = mtd->write_oob(mtd, addr, &ops);
+       retval = mtd_write_oob(mtd, addr, &ops);
        if (retval) {
                yaffs_trace(YAFFS_TRACE_MTD,
                        "write_oob failed, chunk %d, mtd error %d",
@@ -95,7 +121,7 @@ static int yaffs_mtd_read(struct yaffs_dev *dev, int nand_chunk,
        struct mtd_oob_ops ops;
        int retval;
 
-       addr = ((loff_t) nand_chunk) * dev->data_bytes_per_chunk;
+       addr = ((loff_t) nand_chunk) * dev->param.total_bytes_per_chunk;
        memset(&ops, 0, sizeof(ops));
        ops.mode = MTD_OPS_AUTO_OOB;
        ops.len = (data) ? data_len : 0;
@@ -112,7 +138,7 @@ static int yaffs_mtd_read(struct yaffs_dev *dev, int nand_chunk,
        /* Read page and oob using MTD.
         * Check status and determine ECC result.
         */
-       retval = mtd->read_oob(mtd, addr, &ops);
+       retval = mtd_read_oob(mtd, addr, &ops);
        if (retval)
                yaffs_trace(YAFFS_TRACE_MTD,
                        "read_oob failed, chunk %d, mtd error %d",
@@ -165,7 +191,7 @@ static      int yaffs_mtd_erase(struct yaffs_dev *dev, int block_no)
        ei.callback = NULL;
        ei.priv = (u_long) dev;
 
-       retval = mtd->erase(mtd, &ei);
+       retval = mtd_erase(mtd, &ei);
 
        if (retval == 0)
                return YAFFS_OK;
@@ -176,24 +202,24 @@ static    int yaffs_mtd_erase(struct yaffs_dev *dev, int block_no)
 static int yaffs_mtd_mark_bad(struct yaffs_dev *dev, int block_no)
 {
        struct mtd_info *mtd = yaffs_dev_to_mtd(dev);
-       int blocksize = dev->param.chunks_per_block * dev->data_bytes_per_chunk;
+       int blocksize = dev->param.chunks_per_block * dev->param.total_bytes_per_chunk;
        int retval;
 
        yaffs_trace(YAFFS_TRACE_BAD_BLOCKS, "marking block %d bad", block_no);
 
-       retval = mtd->block_markbad(mtd, (loff_t) blocksize * block_no);
+       retval = mtd_block_markbad(mtd, (loff_t) blocksize * block_no);
        return (retval) ? YAFFS_FAIL : YAFFS_OK;
 }
 
 static int yaffs_mtd_check_bad(struct yaffs_dev *dev, int block_no)
 {
        struct mtd_info *mtd = yaffs_dev_to_mtd(dev);
-       int blocksize = dev->param.chunks_per_block * dev->data_bytes_per_chunk;
+       int blocksize = dev->param.chunks_per_block * dev->param.total_bytes_per_chunk;
        int retval;
 
-       yaffs_trace(YAFFS_TRACE_BAD_BLOCKS, "checking block %d bad", block_no);
+       yaffs_trace(YAFFS_TRACE_MTD, "checking block %d bad", block_no);
 
-       retval = mtd->block_isbad(mtd, (loff_t) blocksize * block_no);
+       retval = mtd_block_isbad(mtd, (loff_t) blocksize * block_no);
        return (retval) ? YAFFS_FAIL : YAFFS_OK;
 }
 
@@ -208,7 +234,6 @@ static int yaffs_mtd_deinitialise(struct yaffs_dev *dev)
 }
 
 
-
 void yaffs_mtd_drv_install(struct yaffs_dev *dev)
 {
        struct yaffs_driver *drv = &dev->drv;
@@ -221,3 +246,65 @@ void yaffs_mtd_drv_install(struct yaffs_dev *dev)
        drv->drv_initialise_fn = yaffs_mtd_initialise;
        drv->drv_deinitialise_fn = yaffs_mtd_deinitialise;
 }
+
+
+struct mtd_info * yaffs_get_mtd_device(dev_t sdev)
+{
+       struct mtd_info *mtd;
+
+       mtd = yaffs_get_mtd_device(sdev);
+
+       /* Check it's an mtd device..... */
+       if (MAJOR(sdev) != MTD_BLOCK_MAJOR)
+               return NULL;    /* This isn't an mtd device */
+
+       /* Check it's NAND */
+       if (mtd->type != MTD_NANDFLASH) {
+               yaffs_trace(YAFFS_TRACE_ALWAYS,
+                       "yaffs: MTD device is not NAND it's type %d",
+                       mtd->type);
+               return NULL;
+       }
+
+       yaffs_trace(YAFFS_TRACE_OS, " %s %d", WRITE_SIZE_STR, WRITE_SIZE(mtd));
+       yaffs_trace(YAFFS_TRACE_OS, " oobsize %d", mtd->oobsize);
+       yaffs_trace(YAFFS_TRACE_OS, " erasesize %d", mtd->erasesize);
+#if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 29)
+       yaffs_trace(YAFFS_TRACE_OS, " size %u", mtd->size);
+#else
+       yaffs_trace(YAFFS_TRACE_OS, " size %lld", mtd->size);
+#endif
+
+       return mtd;
+}
+
+int yaffs_verify_mtd(struct mtd_info *mtd, int yaffs_version, int inband_tags)
+{
+       if (yaffs_version == 2) {
+               if ((WRITE_SIZE(mtd) < YAFFS_MIN_YAFFS2_CHUNK_SIZE ||
+                    mtd->oobsize < YAFFS_MIN_YAFFS2_SPARE_SIZE) &&
+                   !inband_tags) {
+                       yaffs_trace(YAFFS_TRACE_ALWAYS,
+                               "MTD device does not have the right page sizes"
+                       );
+                       return -1;
+               }
+       } else {
+               if (WRITE_SIZE(mtd) < YAFFS_BYTES_PER_CHUNK ||
+                   mtd->oobsize != YAFFS_BYTES_PER_SPARE) {
+                       yaffs_trace(YAFFS_TRACE_ALWAYS,
+                               "MTD device does not support have the right page sizes"
+                       );
+                       return -1;
+               }
+       }
+
+       return 0;
+}
+
+
+void yaffs_put_mtd_device(struct mtd_info *mtd)
+{
+       if(mtd)
+               put_mtd_device(mtd);
+}