/* * YAFFS: Yet Another Flash File System. A NAND-flash specific file system. * * Copyright (C) 2002-2007 Aleph One Ltd. * for Toby Churchill Ltd and Brightstar Engineering * * Created by Charles Manning * * 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. */ /* * yaffscfg.c The configuration for the "direct" use of yaffs. * * This file is intended to be modified to your requirements. * There is no need to redistribute this file. */ /* XXX U-BOOT XXX */ #include #include #include "nand.h" #include "yaffscfg.h" #include "yaffsfs.h" #include "yaffs_packedtags2.h" #include "yaffs_mtdif.h" #include "yaffs_mtdif2.h" #if 0 #include #else #include "malloc.h" #endif unsigned yaffs_traceMask = 0x0; /* Disable logging */ static int yaffs_errno = 0; void yaffsfs_SetError(int err) { //Do whatever to set error yaffs_errno = err; } int yaffsfs_GetError(void) { return yaffs_errno; } void yaffsfs_Lock(void) { } void yaffsfs_Unlock(void) { } __u32 yaffsfs_CurrentTime(void) { return 0; } void *yaffs_malloc(size_t size) { return malloc(size); } void yaffs_free(void *ptr) { free(ptr); } void yaffsfs_LocalInitialisation(void) { // Define locking semaphore. } // Configuration for: // /ram 2MB ramdisk // /boot 2MB boot disk (flash) // /flash 14MB flash disk (flash) // NB Though /boot and /flash occupy the same physical device they // are still disticnt "yaffs_Devices. You may think of these as "partitions" // using non-overlapping areas in the same device. // #include "yaffs_ramdisk.h" #include "yaffs_flashif.h" static int isMounted = 0; #define MOUNT_POINT "/flash" extern nand_info_t nand_info[]; /* XXX U-BOOT XXX */ #if 0 static yaffs_Device ramDev; static yaffs_Device bootDev; static yaffs_Device flashDev; #endif static yaffsfs_DeviceConfiguration yaffsfs_config[] = { /* XXX U-BOOT XXX */ #if 0 { "/ram", &ramDev}, { "/boot", &bootDev}, { "/flash", &flashDev}, #else { MOUNT_POINT, 0}, #endif {(void *)0,(void *)0} }; int yaffs_StartUp(void) { struct mtd_info *mtd = &nand_info[0]; int yaffsVersion = 2; int nBlocks; yaffs_Device *flashDev = calloc(1, sizeof(yaffs_Device)); yaffsfs_config[0].dev = flashDev; /* store the mtd device for later use */ flashDev->genericDevice = mtd; // Stuff to configure YAFFS // Stuff to initialise anything special (eg lock semaphore). yaffsfs_LocalInitialisation(); // Set up devices /* XXX U-BOOT XXX */ #if 0 // /ram ramDev.nBytesPerChunk = 512; ramDev.nChunksPerBlock = 32; ramDev.nReservedBlocks = 2; // Set this smaller for RAM ramDev.startBlock = 1; // Can't use block 0 ramDev.endBlock = 127; // Last block in 2MB. ramDev.useNANDECC = 1; ramDev.nShortOpCaches = 0; // Disable caching on this device. ramDev.genericDevice = (void *) 0; // Used to identify the device in fstat. ramDev.writeChunkWithTagsToNAND = yramdisk_WriteChunkWithTagsToNAND; ramDev.readChunkWithTagsFromNAND = yramdisk_ReadChunkWithTagsFromNAND; ramDev.eraseBlockInNAND = yramdisk_EraseBlockInNAND; ramDev.initialiseNAND = yramdisk_InitialiseNAND; // /boot bootDev.nBytesPerChunk = 612; bootDev.nChunksPerBlock = 32; bootDev.nReservedBlocks = 5; bootDev.startBlock = 1; // Can't use block 0 bootDev.endBlock = 127; // Last block in 2MB. bootDev.useNANDECC = 0; // use YAFFS's ECC bootDev.nShortOpCaches = 10; // Use caches bootDev.genericDevice = (void *) 1; // Used to identify the device in fstat. bootDev.writeChunkToNAND = yflash_WriteChunkToNAND; bootDev.readChunkFromNAND = yflash_ReadChunkFromNAND; bootDev.eraseBlockInNAND = yflash_EraseBlockInNAND; bootDev.initialiseNAND = yflash_InitialiseNAND; #endif // /flash flashDev->nReservedBlocks = 5; // flashDev->nShortOpCaches = (options.no_cache) ? 0 : 10; flashDev->nShortOpCaches = 10; // Use caches flashDev->useNANDECC = 0; // do not use YAFFS's ECC if (yaffsVersion == 2) { flashDev->writeChunkWithTagsToNAND = nandmtd2_WriteChunkWithTagsToNAND; flashDev->readChunkWithTagsFromNAND = nandmtd2_ReadChunkWithTagsFromNAND; flashDev->markNANDBlockBad = nandmtd2_MarkNANDBlockBad; flashDev->queryNANDBlock = nandmtd2_QueryNANDBlock; flashDev->spareBuffer = YMALLOC(mtd->oobsize); flashDev->isYaffs2 = 1; #if (LINUX_VERSION_CODE > KERNEL_VERSION(2,6,17)) flashDev->nDataBytesPerChunk = mtd->writesize; flashDev->nChunksPerBlock = mtd->erasesize / mtd->writesize; #else flashDev->nDataBytesPerChunk = mtd->oobblock; flashDev->nChunksPerBlock = mtd->erasesize / mtd->oobblock; #endif nBlocks = mtd->size / mtd->erasesize; flashDev->nCheckpointReservedBlocks = 10; flashDev->startBlock = 0; flashDev->endBlock = nBlocks - 1; } else { flashDev->writeChunkToNAND = nandmtd_WriteChunkToNAND; flashDev->readChunkFromNAND = nandmtd_ReadChunkFromNAND; flashDev->isYaffs2 = 0; nBlocks = mtd->size / (YAFFS_CHUNKS_PER_BLOCK * YAFFS_BYTES_PER_CHUNK); flashDev->startBlock = 320; flashDev->endBlock = nBlocks - 1; flashDev->nChunksPerBlock = YAFFS_CHUNKS_PER_BLOCK; flashDev->nDataBytesPerChunk = YAFFS_BYTES_PER_CHUNK; } /* ... and common functions */ flashDev->eraseBlockInNAND = nandmtd_EraseBlockInNAND; flashDev->initialiseNAND = nandmtd_InitialiseNAND; yaffs_initialise(yaffsfs_config); return 0; } void make_a_file(char *yaffsName,char bval,int sizeOfFile) { int outh; int i; unsigned char buffer[100]; outh = yaffs_open(yaffsName, O_CREAT | O_RDWR | O_TRUNC, S_IREAD | S_IWRITE); if (outh < 0) { printf("Error opening file: %d\n", outh); return; } memset(buffer,bval,100); do{ i = sizeOfFile; if(i > 100) i = 100; sizeOfFile -= i; yaffs_write(outh,buffer,i); } while (sizeOfFile > 0); yaffs_close(outh); } void read_a_file(char *fn) { int h; int i = 0; unsigned char b; h = yaffs_open(fn, O_RDWR,0); if(h<0) { printf("File not found\n"); return; } while(yaffs_read(h,&b,1)> 0) { printf("%02x ",b); i++; if(i > 32) { printf("\n"); i = 0;; } } printf("\n"); yaffs_close(h); } void cmd_yaffs_mount(char *mp) { yaffs_StartUp(); int retval = yaffs_mount(mp); if( retval != -1) isMounted = 1; else printf("Error mounting %s, return value: %d\n", mp, yaffsfs_GetError()); } static void checkMount(void) { if( !isMounted ) { cmd_yaffs_mount(MOUNT_POINT); } } void cmd_yaffs_umount(char *mp) { checkMount(); if( yaffs_unmount(mp) == -1) printf("Error umounting %s, return value: %d\n", mp, yaffsfs_GetError()); } void cmd_yaffs_write_file(char *yaffsName,char bval,int sizeOfFile) { checkMount(); make_a_file(yaffsName,bval,sizeOfFile); } void cmd_yaffs_read_file(char *fn) { checkMount(); read_a_file(fn); } void cmd_yaffs_mread_file(char *fn, char *addr) { int h; struct yaffs_stat s; checkMount(); yaffs_stat(fn,&s); printf ("Copy %s to 0x%p... ", fn, addr); h = yaffs_open(fn, O_RDWR,0); if(h<0) { printf("File not found\n"); return; } yaffs_read(h,addr,(int)s.st_size); printf("\t[DONE]\n"); yaffs_close(h); } void cmd_yaffs_mwrite_file(char *fn, char *addr, int size) { int outh; checkMount(); outh = yaffs_open(fn, O_CREAT | O_RDWR | O_TRUNC, S_IREAD | S_IWRITE); if (outh < 0) { printf("Error opening file: %d\n", outh); } yaffs_write(outh,addr,size); yaffs_close(outh); } void cmd_yaffs_ls(const char *mountpt, int longlist) { int i; yaffs_DIR *d; yaffs_dirent *de; struct yaffs_stat stat; char tempstr[255]; checkMount(); d = yaffs_opendir(mountpt); if(!d) { printf("opendir failed\n"); } else { for(i = 0; (de = yaffs_readdir(d)) != NULL; i++) { if (longlist) { sprintf(tempstr, "%s/%s", mountpt, de->d_name); yaffs_stat(tempstr, &stat); printf("%-25s\t%7ld\n",de->d_name, stat.st_size); } else { printf("%s\n",de->d_name); } } } } void cmd_yaffs_mkdir(const char *dir) { checkMount(); int retval = yaffs_mkdir(dir, 0); if ( retval < 0) printf("yaffs_mkdir returning error: %d\n", retval); } void cmd_yaffs_rmdir(const char *dir) { checkMount(); int retval = yaffs_rmdir(dir); if ( retval < 0) printf("yaffs_rmdir returning error: %d\n", retval); } void cmd_yaffs_rm(const char *path) { checkMount(); int retval = yaffs_unlink(path); if ( retval < 0) printf("yaffs_unlink returning error: %d\n", retval); } void cmd_yaffs_mv(const char *oldPath, const char *newPath) { checkMount(); int retval = yaffs_rename(newPath, oldPath); if ( retval < 0) printf("yaffs_unlink returning error: %d\n", retval); }