357ffb3c46
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@17369 a1c6a512-1295-4272-9138-f99709370657
793 lines
20 KiB
C
793 lines
20 KiB
C
/***************************************************************************
|
|
* __________ __ ___.
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
* \/ \/ \/ \/ \/
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2002 by Björn Stenberg
|
|
*
|
|
* All files in this archive are subject to the GNU General Public License.
|
|
* See the file COPYING in the source tree root for full license agreement.
|
|
*
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
* KIND, either express or implied.
|
|
*
|
|
****************************************************************************/
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <stdbool.h>
|
|
#include "file.h"
|
|
#include "fat.h"
|
|
#include "dir_uncached.h"
|
|
#include "debug.h"
|
|
#include "dircache.h"
|
|
#include "system.h"
|
|
|
|
/*
|
|
These functions provide a roughly POSIX-compatible file IO API.
|
|
|
|
Since the fat32 driver only manages sectors, we maintain a one-sector
|
|
cache for each open file. This way we can provide byte access without
|
|
having to re-read the sector each time.
|
|
The penalty is the RAM used for the cache and slightly more complex code.
|
|
*/
|
|
|
|
struct filedesc {
|
|
unsigned char cache[SECTOR_SIZE];
|
|
int cacheoffset; /* invariant: 0 <= cacheoffset <= SECTOR_SIZE */
|
|
long fileoffset;
|
|
long size;
|
|
int attr;
|
|
struct fat_file fatfile;
|
|
bool busy;
|
|
bool write;
|
|
bool dirty;
|
|
bool trunc;
|
|
};
|
|
|
|
static struct filedesc openfiles[MAX_OPEN_FILES];
|
|
|
|
static int flush_cache(int fd);
|
|
|
|
int creat(const char *pathname)
|
|
{
|
|
return open(pathname, O_WRONLY|O_CREAT|O_TRUNC);
|
|
}
|
|
|
|
static int open_internal(const char* pathname, int flags, bool use_cache)
|
|
{
|
|
DIR_UNCACHED* dir;
|
|
struct dirent_uncached* entry;
|
|
int fd;
|
|
char pathnamecopy[MAX_PATH];
|
|
char* name;
|
|
struct filedesc* file = NULL;
|
|
int rc;
|
|
#ifndef HAVE_DIRCACHE
|
|
(void)use_cache;
|
|
#endif
|
|
|
|
LDEBUGF("open(\"%s\",%d)\n",pathname,flags);
|
|
|
|
if ( pathname[0] != '/' ) {
|
|
DEBUGF("'%s' is not an absolute path.\n",pathname);
|
|
DEBUGF("Only absolute pathnames supported at the moment\n");
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
|
|
/* find a free file descriptor */
|
|
for ( fd=0; fd<MAX_OPEN_FILES; fd++ )
|
|
if ( !openfiles[fd].busy )
|
|
break;
|
|
|
|
if ( fd == MAX_OPEN_FILES ) {
|
|
DEBUGF("Too many files open\n");
|
|
errno = EMFILE;
|
|
return -2;
|
|
}
|
|
|
|
file = &openfiles[fd];
|
|
memset(file, 0, sizeof(struct filedesc));
|
|
|
|
if (flags & (O_RDWR | O_WRONLY)) {
|
|
file->write = true;
|
|
|
|
if (flags & O_TRUNC)
|
|
file->trunc = true;
|
|
}
|
|
file->busy = true;
|
|
|
|
#ifdef HAVE_DIRCACHE
|
|
if (dircache_is_enabled() && !file->write && use_cache)
|
|
{
|
|
const struct dircache_entry *ce;
|
|
# ifdef HAVE_MULTIVOLUME
|
|
int volume = strip_volume(pathname, pathnamecopy);
|
|
# endif
|
|
|
|
ce = dircache_get_entry_ptr(pathname);
|
|
if (!ce)
|
|
{
|
|
errno = ENOENT;
|
|
file->busy = false;
|
|
return -7;
|
|
}
|
|
|
|
fat_open(IF_MV2(volume,)
|
|
ce->startcluster,
|
|
&(file->fatfile),
|
|
NULL);
|
|
file->size = ce->size;
|
|
file->attr = ce->attribute;
|
|
file->cacheoffset = -1;
|
|
file->fileoffset = 0;
|
|
|
|
return fd;
|
|
}
|
|
#endif
|
|
|
|
strncpy(pathnamecopy,pathname,sizeof(pathnamecopy));
|
|
pathnamecopy[sizeof(pathnamecopy)-1] = 0;
|
|
|
|
/* locate filename */
|
|
name=strrchr(pathnamecopy+1,'/');
|
|
if ( name ) {
|
|
*name = 0;
|
|
dir = opendir_uncached(pathnamecopy);
|
|
*name = '/';
|
|
name++;
|
|
}
|
|
else {
|
|
dir = opendir_uncached("/");
|
|
name = pathnamecopy+1;
|
|
}
|
|
if (!dir) {
|
|
DEBUGF("Failed opening dir\n");
|
|
errno = EIO;
|
|
file->busy = false;
|
|
return -4;
|
|
}
|
|
|
|
if(name[0] == 0) {
|
|
DEBUGF("Empty file name\n");
|
|
errno = EINVAL;
|
|
file->busy = false;
|
|
closedir_uncached(dir);
|
|
return -5;
|
|
}
|
|
|
|
/* scan dir for name */
|
|
while ((entry = readdir_uncached(dir))) {
|
|
if ( !strcasecmp(name, entry->d_name) ) {
|
|
fat_open(IF_MV2(dir->fatdir.file.volume,)
|
|
entry->startcluster,
|
|
&(file->fatfile),
|
|
&(dir->fatdir));
|
|
file->size = file->trunc ? 0 : entry->size;
|
|
file->attr = entry->attribute;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ( !entry ) {
|
|
LDEBUGF("Didn't find file %s\n",name);
|
|
if ( file->write && (flags & O_CREAT) ) {
|
|
rc = fat_create_file(name,
|
|
&(file->fatfile),
|
|
&(dir->fatdir));
|
|
if (rc < 0) {
|
|
DEBUGF("Couldn't create %s in %s\n",name,pathnamecopy);
|
|
errno = EIO;
|
|
file->busy = false;
|
|
closedir_uncached(dir);
|
|
return rc * 10 - 6;
|
|
}
|
|
#ifdef HAVE_DIRCACHE
|
|
dircache_add_file(pathname, file->fatfile.firstcluster);
|
|
#endif
|
|
file->size = 0;
|
|
file->attr = 0;
|
|
}
|
|
else {
|
|
DEBUGF("Couldn't find %s in %s\n",name,pathnamecopy);
|
|
errno = ENOENT;
|
|
file->busy = false;
|
|
closedir_uncached(dir);
|
|
return -7;
|
|
}
|
|
} else {
|
|
if(file->write && (file->attr & FAT_ATTR_DIRECTORY)) {
|
|
errno = EISDIR;
|
|
file->busy = false;
|
|
closedir_uncached(dir);
|
|
return -8;
|
|
}
|
|
}
|
|
closedir_uncached(dir);
|
|
|
|
file->cacheoffset = -1;
|
|
file->fileoffset = 0;
|
|
|
|
if (file->write && (flags & O_APPEND)) {
|
|
rc = lseek(fd,0,SEEK_END);
|
|
if (rc < 0 )
|
|
return rc * 10 - 9;
|
|
}
|
|
|
|
#ifdef HAVE_DIRCACHE
|
|
if (file->write)
|
|
dircache_bind(fd, pathname);
|
|
#endif
|
|
|
|
return fd;
|
|
}
|
|
|
|
int open(const char* pathname, int flags)
|
|
{
|
|
/* By default, use the dircache if available. */
|
|
return open_internal(pathname, flags, true);
|
|
}
|
|
|
|
int close(int fd)
|
|
{
|
|
struct filedesc* file = &openfiles[fd];
|
|
int rc = 0;
|
|
|
|
LDEBUGF("close(%d)\n", fd);
|
|
|
|
if (fd < 0 || fd > MAX_OPEN_FILES-1) {
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
if (!file->busy) {
|
|
errno = EBADF;
|
|
return -2;
|
|
}
|
|
if (file->write) {
|
|
rc = fsync(fd);
|
|
if (rc < 0)
|
|
return rc * 10 - 3;
|
|
#ifdef HAVE_DIRCACHE
|
|
dircache_update_filesize(fd, file->size, file->fatfile.firstcluster);
|
|
dircache_update_filetime(fd);
|
|
#endif
|
|
}
|
|
|
|
file->busy = false;
|
|
return 0;
|
|
}
|
|
|
|
int fsync(int fd)
|
|
{
|
|
struct filedesc* file = &openfiles[fd];
|
|
int rc = 0;
|
|
|
|
LDEBUGF("fsync(%d)\n", fd);
|
|
|
|
if (fd < 0 || fd > MAX_OPEN_FILES-1) {
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
if (!file->busy) {
|
|
errno = EBADF;
|
|
return -2;
|
|
}
|
|
if (file->write) {
|
|
/* flush sector cache */
|
|
if ( file->dirty ) {
|
|
rc = flush_cache(fd);
|
|
if (rc < 0)
|
|
{
|
|
/* when failing, try to close the file anyway */
|
|
fat_closewrite(&(file->fatfile), file->size, file->attr);
|
|
return rc * 10 - 3;
|
|
}
|
|
}
|
|
|
|
/* truncate? */
|
|
if (file->trunc) {
|
|
rc = ftruncate(fd, file->size);
|
|
if (rc < 0)
|
|
{
|
|
/* when failing, try to close the file anyway */
|
|
fat_closewrite(&(file->fatfile), file->size, file->attr);
|
|
return rc * 10 - 4;
|
|
}
|
|
}
|
|
|
|
/* tie up all loose ends */
|
|
rc = fat_closewrite(&(file->fatfile), file->size, file->attr);
|
|
if (rc < 0)
|
|
return rc * 10 - 5;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int remove(const char* name)
|
|
{
|
|
int rc;
|
|
struct filedesc* file;
|
|
/* Can't use dircache now, because we need to access the fat structures. */
|
|
int fd = open_internal(name, O_WRONLY, false);
|
|
if ( fd < 0 )
|
|
return fd * 10 - 1;
|
|
|
|
file = &openfiles[fd];
|
|
#ifdef HAVE_DIRCACHE
|
|
dircache_remove(name);
|
|
#endif
|
|
rc = fat_remove(&(file->fatfile));
|
|
if ( rc < 0 ) {
|
|
DEBUGF("Failed removing file: %d\n", rc);
|
|
errno = EIO;
|
|
return rc * 10 - 3;
|
|
}
|
|
|
|
file->size = 0;
|
|
|
|
rc = close(fd);
|
|
if (rc<0)
|
|
return rc * 10 - 4;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int rename(const char* path, const char* newpath)
|
|
{
|
|
int rc, fd;
|
|
DIR_UNCACHED* dir;
|
|
char* nameptr;
|
|
char* dirptr;
|
|
struct filedesc* file;
|
|
char newpath2[MAX_PATH];
|
|
|
|
/* verify new path does not already exist */
|
|
/* If it is a directory, errno == EISDIR if the name exists */
|
|
fd = open(newpath, O_RDONLY);
|
|
if ( fd >= 0 || errno == EISDIR) {
|
|
close(fd);
|
|
errno = EBUSY;
|
|
return -1;
|
|
}
|
|
close(fd);
|
|
|
|
fd = open_internal(path, O_RDONLY, false);
|
|
if ( fd < 0 ) {
|
|
errno = EIO;
|
|
return fd * 10 - 2;
|
|
}
|
|
|
|
/* extract new file name */
|
|
nameptr = strrchr(newpath,'/');
|
|
if (nameptr)
|
|
nameptr++;
|
|
else
|
|
return - 3;
|
|
|
|
/* Extract new path */
|
|
strcpy(newpath2, newpath);
|
|
|
|
dirptr = strrchr(newpath2,'/');
|
|
if(dirptr)
|
|
*dirptr = 0;
|
|
else
|
|
return - 4;
|
|
|
|
dirptr = newpath2;
|
|
|
|
if(strlen(dirptr) == 0) {
|
|
dirptr = "/";
|
|
}
|
|
|
|
dir = opendir_uncached(dirptr);
|
|
if(!dir)
|
|
return - 5;
|
|
|
|
file = &openfiles[fd];
|
|
|
|
rc = fat_rename(&file->fatfile, &dir->fatdir, nameptr,
|
|
file->size, file->attr);
|
|
#ifdef HAVE_MULTIVOLUME
|
|
if ( rc == -1) {
|
|
DEBUGF("Failed renaming file across volumnes: %d\n", rc);
|
|
errno = EXDEV;
|
|
return -6;
|
|
}
|
|
#endif
|
|
if ( rc < 0 ) {
|
|
DEBUGF("Failed renaming file: %d\n", rc);
|
|
errno = EIO;
|
|
return rc * 10 - 7;
|
|
}
|
|
|
|
#ifdef HAVE_DIRCACHE
|
|
dircache_rename(path, newpath);
|
|
#endif
|
|
|
|
rc = close(fd);
|
|
if (rc<0) {
|
|
errno = EIO;
|
|
return rc * 10 - 8;
|
|
}
|
|
|
|
rc = closedir_uncached(dir);
|
|
if (rc<0) {
|
|
errno = EIO;
|
|
return rc * 10 - 9;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ftruncate(int fd, off_t size)
|
|
{
|
|
int rc, sector;
|
|
struct filedesc* file = &openfiles[fd];
|
|
|
|
sector = size / SECTOR_SIZE;
|
|
if (size % SECTOR_SIZE)
|
|
sector++;
|
|
|
|
rc = fat_seek(&(file->fatfile), sector);
|
|
if (rc < 0) {
|
|
errno = EIO;
|
|
return rc * 10 - 1;
|
|
}
|
|
|
|
rc = fat_truncate(&(file->fatfile));
|
|
if (rc < 0) {
|
|
errno = EIO;
|
|
return rc * 10 - 2;
|
|
}
|
|
|
|
file->size = size;
|
|
#ifdef HAVE_DIRCACHE
|
|
dircache_update_filesize(fd, size, file->fatfile.firstcluster);
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int flush_cache(int fd)
|
|
{
|
|
int rc;
|
|
struct filedesc* file = &openfiles[fd];
|
|
long sector = file->fileoffset / SECTOR_SIZE;
|
|
|
|
DEBUGF("Flushing dirty sector cache\n");
|
|
|
|
/* make sure we are on correct sector */
|
|
rc = fat_seek(&(file->fatfile), sector);
|
|
if ( rc < 0 )
|
|
return rc * 10 - 3;
|
|
|
|
rc = fat_readwrite(&(file->fatfile), 1, file->cache, true );
|
|
|
|
if ( rc < 0 ) {
|
|
if(file->fatfile.eof)
|
|
errno = ENOSPC;
|
|
|
|
return rc * 10 - 2;
|
|
}
|
|
|
|
file->dirty = false;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int readwrite(int fd, void* buf, long count, bool write)
|
|
{
|
|
long sectors;
|
|
long nread=0;
|
|
struct filedesc* file = &openfiles[fd];
|
|
int rc;
|
|
|
|
if (fd < 0 || fd > MAX_OPEN_FILES-1) {
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
if ( !file->busy ) {
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
|
|
LDEBUGF( "readwrite(%d,%lx,%ld,%s)\n",
|
|
fd,(long)buf,count,write?"write":"read");
|
|
|
|
/* attempt to read past EOF? */
|
|
if (!write && count > file->size - file->fileoffset)
|
|
count = file->size - file->fileoffset;
|
|
|
|
/* any head bytes? */
|
|
if ( file->cacheoffset != -1 ) {
|
|
int offs = file->cacheoffset;
|
|
int headbytes = MIN(count, SECTOR_SIZE - offs);
|
|
|
|
if (write) {
|
|
memcpy( file->cache + offs, buf, headbytes );
|
|
file->dirty = true;
|
|
}
|
|
else {
|
|
memcpy( buf, file->cache + offs, headbytes );
|
|
}
|
|
|
|
if (offs + headbytes == SECTOR_SIZE) {
|
|
if (file->dirty) {
|
|
rc = flush_cache(fd);
|
|
if ( rc < 0 ) {
|
|
errno = EIO;
|
|
return rc * 10 - 2;
|
|
}
|
|
}
|
|
file->cacheoffset = -1;
|
|
}
|
|
else {
|
|
file->cacheoffset += headbytes;
|
|
}
|
|
|
|
nread = headbytes;
|
|
count -= headbytes;
|
|
}
|
|
|
|
/* If the buffer has been modified, either it has been flushed already
|
|
* (if (offs+headbytes == SECTOR_SIZE)...) or does not need to be (no
|
|
* more data to follow in this call). Do NOT flush here. */
|
|
|
|
/* read/write whole sectors right into/from the supplied buffer */
|
|
sectors = count / SECTOR_SIZE;
|
|
if ( sectors ) {
|
|
rc = fat_readwrite(&(file->fatfile), sectors,
|
|
(unsigned char*)buf+nread, write );
|
|
if ( rc < 0 ) {
|
|
DEBUGF("Failed read/writing %ld sectors\n",sectors);
|
|
errno = EIO;
|
|
if(write && file->fatfile.eof) {
|
|
DEBUGF("No space left on device\n");
|
|
errno = ENOSPC;
|
|
} else {
|
|
file->fileoffset += nread;
|
|
}
|
|
file->cacheoffset = -1;
|
|
/* adjust file size to length written */
|
|
if ( write && file->fileoffset > file->size )
|
|
{
|
|
file->size = file->fileoffset;
|
|
#ifdef HAVE_DIRCACHE
|
|
dircache_update_filesize(fd, file->size, file->fatfile.firstcluster);
|
|
#endif
|
|
}
|
|
return nread ? nread : rc * 10 - 4;
|
|
}
|
|
else {
|
|
if ( rc > 0 ) {
|
|
nread += rc * SECTOR_SIZE;
|
|
count -= sectors * SECTOR_SIZE;
|
|
|
|
/* if eof, skip tail bytes */
|
|
if ( rc < sectors )
|
|
count = 0;
|
|
}
|
|
else {
|
|
/* eof */
|
|
count=0;
|
|
}
|
|
|
|
file->cacheoffset = -1;
|
|
}
|
|
}
|
|
|
|
/* any tail bytes? */
|
|
if ( count ) {
|
|
if (write) {
|
|
if ( file->fileoffset + nread < file->size ) {
|
|
/* sector is only partially filled. copy-back from disk */
|
|
LDEBUGF("Copy-back tail cache\n");
|
|
rc = fat_readwrite(&(file->fatfile), 1, file->cache, false );
|
|
if ( rc < 0 ) {
|
|
DEBUGF("Failed writing\n");
|
|
errno = EIO;
|
|
file->fileoffset += nread;
|
|
file->cacheoffset = -1;
|
|
/* adjust file size to length written */
|
|
if ( file->fileoffset > file->size )
|
|
{
|
|
file->size = file->fileoffset;
|
|
#ifdef HAVE_DIRCACHE
|
|
dircache_update_filesize(fd, file->size, file->fatfile.firstcluster);
|
|
#endif
|
|
}
|
|
return nread ? nread : rc * 10 - 5;
|
|
}
|
|
/* seek back one sector to put file position right */
|
|
rc = fat_seek(&(file->fatfile),
|
|
(file->fileoffset + nread) /
|
|
SECTOR_SIZE);
|
|
if ( rc < 0 ) {
|
|
DEBUGF("fat_seek() failed\n");
|
|
errno = EIO;
|
|
file->fileoffset += nread;
|
|
file->cacheoffset = -1;
|
|
/* adjust file size to length written */
|
|
if ( file->fileoffset > file->size )
|
|
{
|
|
file->size = file->fileoffset;
|
|
#ifdef HAVE_DIRCACHE
|
|
dircache_update_filesize(fd, file->size, file->fatfile.firstcluster);
|
|
#endif
|
|
}
|
|
return nread ? nread : rc * 10 - 6;
|
|
}
|
|
}
|
|
memcpy( file->cache, (unsigned char*)buf + nread, count );
|
|
file->dirty = true;
|
|
}
|
|
else {
|
|
rc = fat_readwrite(&(file->fatfile), 1, &(file->cache),false);
|
|
if (rc < 1 ) {
|
|
DEBUGF("Failed caching sector\n");
|
|
errno = EIO;
|
|
file->fileoffset += nread;
|
|
file->cacheoffset = -1;
|
|
return nread ? nread : rc * 10 - 7;
|
|
}
|
|
memcpy( (unsigned char*)buf + nread, file->cache, count );
|
|
}
|
|
|
|
nread += count;
|
|
file->cacheoffset = count;
|
|
}
|
|
|
|
file->fileoffset += nread;
|
|
LDEBUGF("fileoffset: %ld\n", file->fileoffset);
|
|
|
|
/* adjust file size to length written */
|
|
if ( write && file->fileoffset > file->size )
|
|
{
|
|
file->size = file->fileoffset;
|
|
#ifdef HAVE_DIRCACHE
|
|
dircache_update_filesize(fd, file->size, file->fatfile.firstcluster);
|
|
#endif
|
|
}
|
|
|
|
return nread;
|
|
}
|
|
|
|
ssize_t write(int fd, const void* buf, size_t count)
|
|
{
|
|
if (!openfiles[fd].write) {
|
|
errno = EACCES;
|
|
return -1;
|
|
}
|
|
return readwrite(fd, (void *)buf, count, true);
|
|
}
|
|
|
|
ssize_t read(int fd, void* buf, size_t count)
|
|
{
|
|
return readwrite(fd, buf, count, false);
|
|
}
|
|
|
|
|
|
off_t lseek(int fd, off_t offset, int whence)
|
|
{
|
|
off_t pos;
|
|
long newsector;
|
|
long oldsector;
|
|
int sectoroffset;
|
|
int rc;
|
|
struct filedesc* file = &openfiles[fd];
|
|
|
|
LDEBUGF("lseek(%d,%ld,%d)\n",fd,offset,whence);
|
|
|
|
if (fd < 0 || fd > MAX_OPEN_FILES-1) {
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
if ( !file->busy ) {
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
|
|
switch ( whence ) {
|
|
case SEEK_SET:
|
|
pos = offset;
|
|
break;
|
|
|
|
case SEEK_CUR:
|
|
pos = file->fileoffset + offset;
|
|
break;
|
|
|
|
case SEEK_END:
|
|
pos = file->size + offset;
|
|
break;
|
|
|
|
default:
|
|
errno = EINVAL;
|
|
return -2;
|
|
}
|
|
if ((pos < 0) || (pos > file->size)) {
|
|
errno = EINVAL;
|
|
return -3;
|
|
}
|
|
|
|
/* new sector? */
|
|
newsector = pos / SECTOR_SIZE;
|
|
oldsector = file->fileoffset / SECTOR_SIZE;
|
|
sectoroffset = pos % SECTOR_SIZE;
|
|
|
|
if ( (newsector != oldsector) ||
|
|
((file->cacheoffset==-1) && sectoroffset) ) {
|
|
|
|
if ( newsector != oldsector ) {
|
|
if (file->dirty) {
|
|
rc = flush_cache(fd);
|
|
if (rc < 0)
|
|
return rc * 10 - 5;
|
|
}
|
|
|
|
rc = fat_seek(&(file->fatfile), newsector);
|
|
if ( rc < 0 ) {
|
|
errno = EIO;
|
|
return rc * 10 - 4;
|
|
}
|
|
}
|
|
if ( sectoroffset ) {
|
|
rc = fat_readwrite(&(file->fatfile), 1,
|
|
&(file->cache),false);
|
|
if ( rc < 0 ) {
|
|
errno = EIO;
|
|
return rc * 10 - 6;
|
|
}
|
|
file->cacheoffset = sectoroffset;
|
|
}
|
|
else
|
|
file->cacheoffset = -1;
|
|
}
|
|
else
|
|
if ( file->cacheoffset != -1 )
|
|
file->cacheoffset = sectoroffset;
|
|
|
|
file->fileoffset = pos;
|
|
|
|
return pos;
|
|
}
|
|
|
|
off_t filesize(int fd)
|
|
{
|
|
struct filedesc* file = &openfiles[fd];
|
|
|
|
if (fd < 0 || fd > MAX_OPEN_FILES-1) {
|
|
errno = EINVAL;
|
|
return -1;
|
|
}
|
|
if ( !file->busy ) {
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
|
|
return file->size;
|
|
}
|
|
|
|
|
|
#ifdef HAVE_HOTSWAP
|
|
/* release all file handles on a given volume "by force", to avoid leaks */
|
|
int release_files(int volume)
|
|
{
|
|
struct filedesc* pfile = openfiles;
|
|
int fd;
|
|
int closed = 0;
|
|
for ( fd=0; fd<MAX_OPEN_FILES; fd++, pfile++)
|
|
{
|
|
if (pfile->fatfile.volume == volume)
|
|
{
|
|
pfile->busy = false; /* mark as available, no further action */
|
|
closed++;
|
|
}
|
|
}
|
|
return closed; /* return how many we did */
|
|
}
|
|
#endif /* #ifdef HAVE_HOTSWAP */
|
|
|