📄 yaffsfs.c
字号:
} yaffsfs_Unlock(); return retval;}int yaffs_chmod(const YCHAR *path, mode_t mode){ yaffs_Object *obj; int retVal = -1; yaffsfs_Lock(); obj = yaffsfs_FindObject(NULL,path,0); if(obj) { retVal = yaffsfs_DoChMod(obj,mode); } else { // todo error not found yaffsfs_SetError(-ENOENT); } yaffsfs_Unlock(); return retVal;}int yaffs_fchmod(int fd, mode_t mode){ yaffs_Object *obj; int retVal = -1; yaffsfs_Lock(); obj = yaffsfs_GetHandleObject(fd); if(obj) { retVal = yaffsfs_DoChMod(obj,mode); } else { // bad handle yaffsfs_SetError(-EBADF); } yaffsfs_Unlock(); return retVal;}int yaffs_mkdir(const YCHAR *path, mode_t mode){ yaffs_Object *parent = NULL; yaffs_Object *dir = NULL; YCHAR *name; int retVal= -1; yaffsfs_Lock(); parent = yaffsfs_FindDirectory(NULL,path,&name,0); if(parent) dir = yaffs_MknodDirectory(parent,name,mode,0,0); if(dir) { retVal = 0; } else { if(!parent){ yaffsfs_SetError(-ENOENT); // missing path } else if (yaffs_FindObjectByName(parent,name)){ yaffsfs_SetError(-EEXIST); // the name already exists } else yaffsfs_SetError(-ENOSPC); // just assume no space retVal = -1; } yaffsfs_Unlock(); return retVal;}int yaffs_mount(const YCHAR *path){ int retVal=-1; int result=YAFFS_FAIL; yaffs_Device *dev=NULL; YCHAR *dummy; T(YAFFS_TRACE_ALWAYS,(TSTR("yaffs: Mounting %s" TENDSTR),path)); yaffsfs_Lock(); dev = yaffsfs_FindDevice(path,&dummy); if(dev) { if(!dev->isMounted) { result = yaffs_GutsInitialise(dev); if(result == YAFFS_FAIL) { // todo error - mount failed yaffsfs_SetError(-ENOMEM); } retVal = result ? 0 : -1; } else { //todo error - already mounted. yaffsfs_SetError(-EBUSY); } } else { // todo error - no device yaffsfs_SetError(-ENODEV); } yaffsfs_Unlock(); return retVal;}int yaffs_sync(const YCHAR *path){ int retVal=-1; yaffs_Device *dev=NULL; YCHAR *dummy; yaffsfs_Lock(); dev = yaffsfs_FindDevice(path,&dummy); if(dev) { if(dev->isMounted) { yaffs_FlushEntireDeviceCache(dev); yaffs_CheckpointSave(dev); } else { //todo error - not mounted. yaffsfs_SetError(-EINVAL); } } else { // todo error - no device yaffsfs_SetError(-ENODEV); } yaffsfs_Unlock(); return retVal; }int yaffs_unmount(const YCHAR *path){ int retVal=-1; yaffs_Device *dev=NULL; YCHAR *dummy; yaffsfs_Lock(); dev = yaffsfs_FindDevice(path,&dummy); if(dev) { if(dev->isMounted) { int i; int inUse; yaffs_FlushEntireDeviceCache(dev); yaffs_CheckpointSave(dev); for(i = inUse = 0; i < YAFFSFS_N_HANDLES && !inUse; i++) { if(yaffsfs_handle[i].inUse && yaffsfs_handle[i].obj->myDev == dev) { inUse = 1; // the device is in use, can't unmount } } if(!inUse) { yaffs_Deinitialise(dev); retVal = 0; } else { // todo error can't unmount as files are open yaffsfs_SetError(-EBUSY); } } else { //todo error - not mounted. yaffsfs_SetError(-EINVAL); } } else { // todo error - no device yaffsfs_SetError(-ENODEV); } yaffsfs_Unlock(); return retVal;}loff_t yaffs_freespace(const YCHAR *path){ loff_t retVal=-1; yaffs_Device *dev=NULL; YCHAR *dummy; yaffsfs_Lock(); dev = yaffsfs_FindDevice(path,&dummy); if(dev && dev->isMounted) { retVal = yaffs_GetNumberOfFreeChunks(dev); retVal *= dev->nDataBytesPerChunk; } else { yaffsfs_SetError(-EINVAL); } yaffsfs_Unlock(); return retVal;}loff_t yaffs_totalspace(const YCHAR *path){ loff_t retVal=-1; yaffs_Device *dev=NULL; YCHAR *dummy; yaffsfs_Lock(); dev = yaffsfs_FindDevice(path,&dummy); if(dev && dev->isMounted) { retVal = (dev->endBlock - dev->startBlock + 1) - dev->nReservedBlocks; retVal *= dev->nChunksPerBlock; retVal *= dev->nDataBytesPerChunk; } else { yaffsfs_SetError(-EINVAL); } yaffsfs_Unlock(); return retVal;}int yaffs_inodecount(const YCHAR *path){ loff_t retVal= -1; yaffs_Device *dev=NULL; YCHAR *dummy; yaffsfs_Lock(); dev = yaffsfs_FindDevice(path,&dummy); if(dev && dev->isMounted) { int nObjects = dev->nObjectsCreated - dev->nFreeObjects; if(nObjects > dev->nHardLinks) retVal = nObjects - dev->nHardLinks; } if(retVal < 0){ yaffsfs_SetError(-EINVAL); } yaffsfs_Unlock(); return retVal; }void yaffs_initialise(yaffsfs_DeviceConfiguration *cfgList){ yaffsfs_DeviceConfiguration *cfg; yaffsfs_configurationList = cfgList; yaffsfs_InitHandles(); cfg = yaffsfs_configurationList; while(cfg && cfg->prefix && cfg->dev) { cfg->dev->isMounted = 0; cfg->dev->removeObjectCallback = yaffsfs_RemoveObjectCallback; cfg++; }}//// Directory search stuff.//// Directory search context//// NB this is an opaque structure.typedef struct{ __u32 magic; yaffs_dirent de; /* directory entry being used by this dsc */ YCHAR name[NAME_MAX+1]; /* name of directory being searched */ yaffs_Object *dirObj; /* ptr to directory being searched */ yaffs_Object *nextReturn; /* obj to be returned by next readddir */ int offset; struct ylist_head others; } yaffsfs_DirectorySearchContext;static struct ylist_head search_contexts;static void yaffsfs_SetDirRewound(yaffsfs_DirectorySearchContext *dsc){ if(dsc && dsc->dirObj && dsc->dirObj->variantType == YAFFS_OBJECT_TYPE_DIRECTORY){ dsc->offset = 0; if( ylist_empty(&dsc->dirObj->variant.directoryVariant.children)){ dsc->nextReturn = NULL; } else { dsc->nextReturn = ylist_entry(dsc->dirObj->variant.directoryVariant.children.next, yaffs_Object,siblings); } } else { /* Hey someone isn't playing nice! */ }}static void yaffsfs_DirAdvance(yaffsfs_DirectorySearchContext *dsc){ if(dsc && dsc->dirObj && dsc->dirObj->variantType == YAFFS_OBJECT_TYPE_DIRECTORY){ if( dsc->nextReturn == NULL || ylist_empty(&dsc->dirObj->variant.directoryVariant.children)){ dsc->nextReturn = NULL; } else { struct ylist_head *next = dsc->nextReturn->siblings.next; if( next == &dsc->dirObj->variant.directoryVariant.children) dsc->nextReturn = NULL; /* end of list */ else dsc->nextReturn = ylist_entry(next,yaffs_Object,siblings); } } else { /* Hey someone isn't playing nice! */ }}static void yaffsfs_RemoveObjectCallback(yaffs_Object *obj){ struct ylist_head *i; yaffsfs_DirectorySearchContext *dsc; /* if search contexts not initilised then skip */ if(!search_contexts.next) return; /* Iterate through the directory search contexts. * If any are the one being removed, then advance the dsc to * the next one to prevent a hanging ptr. */ ylist_for_each(i, &search_contexts) { if (i) { dsc = ylist_entry(i, yaffsfs_DirectorySearchContext,others); if(dsc->nextReturn == obj) yaffsfs_DirAdvance(dsc); } }}yaffs_DIR *yaffs_opendir(const YCHAR *dirname){ yaffs_DIR *dir = NULL; yaffs_Object *obj = NULL; yaffsfs_DirectorySearchContext *dsc = NULL; yaffsfs_Lock(); obj = yaffsfs_FindObject(NULL,dirname,0); if(obj && obj->variantType == YAFFS_OBJECT_TYPE_DIRECTORY) { dsc = YMALLOC(sizeof(yaffsfs_DirectorySearchContext)); dir = (yaffs_DIR *)dsc; if(dsc) { memset(dsc,0,sizeof(yaffsfs_DirectorySearchContext)); dsc->magic = YAFFS_MAGIC; dsc->dirObj = obj; yaffs_strncpy(dsc->name,dirname,NAME_MAX); YINIT_LIST_HEAD(&dsc->others); if(!search_contexts.next) YINIT_LIST_HEAD(&search_contexts); ylist_add(&dsc->others,&search_contexts); yaffsfs_SetDirRewound(dsc); } } yaffsfs_Unlock(); return dir;}struct yaffs_dirent *yaffs_readdir(yaffs_DIR *dirp){ yaffsfs_DirectorySearchContext *dsc = (yaffsfs_DirectorySearchContext *)dirp; struct yaffs_dirent *retVal = NULL; yaffsfs_Lock(); if(dsc && dsc->magic == YAFFS_MAGIC){ yaffsfs_SetError(0); if(dsc->nextReturn){ dsc->de.d_ino = yaffs_GetEquivalentObject(dsc->nextReturn)->objectId; dsc->de.d_dont_use = (unsigned)dsc->nextReturn; dsc->de.d_off = dsc->offset++; yaffs_GetObjectName(dsc->nextReturn,dsc->de.d_name,NAME_MAX); if(yaffs_strlen(dsc->de.d_name) == 0) { // this should not happen! yaffs_strcpy(dsc->de.d_name,_Y("zz")); } dsc->de.d_reclen = sizeof(struct yaffs_dirent); retVal = &dsc->de; yaffsfs_DirAdvance(dsc); } else retVal = NULL; } else { yaffsfs_SetError(-EBADF); } yaffsfs_Unlock(); return retVal;}void yaffs_rewinddir(yaffs_DIR *dirp){ yaffsfs_DirectorySearchContext *dsc = (yaffsfs_DirectorySearchContext *)dirp; yaffsfs_Lock(); yaffsfs_SetDirRewound(dsc); yaffsfs_Unlock();}int yaffs_closedir(yaffs_DIR *dirp){ yaffsfs_DirectorySearchContext *dsc = (yaffsfs_DirectorySearchContext *)dirp; yaffsfs_Lock(); dsc->magic = 0; ylist_del(&dsc->others); /* unhook from list */ YFREE(dsc); yaffsfs_Unlock(); return 0;}// end of directory stuffint yaffs_symlink(const YCHAR *oldpath, const YCHAR *newpath){ yaffs_Object *parent = NULL; yaffs_Object *obj; YCHAR *name; int retVal= -1; int mode = 0; // ignore for now yaffsfs_Lock(); parent = yaffsfs_FindDirectory(NULL,newpath,&name,0); if(parent){ obj = yaffs_MknodSymLink(parent,name,mode,0,0,oldpath); if(obj) { retVal = 0; } else { yaffsfs_SetError(-ENOSPC); // just assume no space for now retVal = -1; } } else { yaffsfs_SetError(-EINVAL); retVal = -1; } yaffsfs_Unlock(); return retVal;}int yaffs_readlink(const YCHAR *path, YCHAR *buf, int bufsiz){ yaffs_Object *obj = NULL; int retVal; yaffsfs_Lock(); obj = yaffsfs_FindObject(NULL,path,0); if(!obj) { yaffsfs_SetError(-ENOENT); retVal = -1; } else if(obj->variantType != YAFFS_OBJECT_TYPE_SYMLINK) { yaffsfs_SetError(-EINVAL); retVal = -1; } else { YCHAR *alias = obj->variant.symLinkVariant.alias; memset(buf,0,bufsiz); yaffs_strncpy(buf,alias,bufsiz - 1); retVal = 0; } yaffsfs_Unlock(); return retVal;}int yaffs_link(const YCHAR *oldpath, const YCHAR *newpath){ // Creates a link called newpath to existing oldpath yaffs_Object *obj = NULL; yaffs_Object *target = NULL; int retVal = 0; yaffsfs_Lock(); obj = yaffsfs_FindObject(NULL,oldpath,0); target = yaffsfs_FindObject(NULL,newpath,0); if(!obj) { yaffsfs_SetError(-ENOENT); retVal = -1; } else if(target) { yaffsfs_SetError(-EEXIST); retVal = -1; } else { yaffs_Object *newdir = NULL; yaffs_Object *link = NULL; YCHAR *newname; newdir = yaffsfs_FindDirectory(NULL,newpath,&newname,0); if(!newdir) { yaffsfs_SetError(-ENOTDIR); retVal = -1; } else if(newdir->myDev != obj->myDev) { yaffsfs_SetError(-EXDEV); retVal = -1; } if(newdir && yaffs_strlen(newname) > 0) { link = yaffs_Link(newdir,newname,obj); if(link) retVal = 0; else { yaffsfs_SetError(-ENOSPC); retVal = -1; } } } yaffsfs_Unlock(); return retVal;}int yaffs_mknod(const YCHAR *pathname, mode_t mode, dev_t dev);int yaffs_DumpDevStruct(const YCHAR *path){#if 0 YCHAR *rest; yaffs_Object *obj = yaffsfs_FindRoot(path,&rest); if(obj) { yaffs_Device *dev = obj->myDev; printf("\n" "nPageWrites.......... %d\n" "nPageReads........... %d\n" "nBlockErasures....... %d\n" "nGCCopies............ %d\n" "garbageCollections... %d\n" "passiveGarbageColl'ns %d\n" "\n", dev->nPageWrites, dev->nPageReads, dev->nBlockErasures, dev->nGCCopies, dev->garbageCollections, dev->passiveGarbageCollections ); }#endif return 0;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -