mirror of https://github.com/FDOS/kernel.git
2281 lines
66 KiB
C
2281 lines
66 KiB
C
/****************************************************************/
|
|
/* */
|
|
/* fatfs.c */
|
|
/* DOS-C */
|
|
/* */
|
|
/* FAT File System I/O Functions */
|
|
/* */
|
|
/* Copyright (c) 1995,1998 */
|
|
/* Pasquale J. Villani */
|
|
/* All Rights Reserved */
|
|
/* */
|
|
/* This file is part of DOS-C. */
|
|
/* */
|
|
/* DOS-C is free software; you can redistribute it and/or */
|
|
/* modify it under the terms of the GNU General Public License */
|
|
/* as published by the Free Software Foundation; either version */
|
|
/* 2, or (at your option) any later version. */
|
|
/* */
|
|
/* DOS-C is distributed in the hope that it will be useful, but */
|
|
/* WITHOUT ANY WARRANTY; without even the implied warranty of */
|
|
/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See */
|
|
/* the GNU General Public License for more details. */
|
|
/* */
|
|
/* You should have received a copy of the GNU General Public */
|
|
/* License along with DOS-C; see the file COPYING. If not, */
|
|
/* write to the Free Software Foundation, 675 Mass Ave, */
|
|
/* Cambridge, MA 02139, USA. */
|
|
/****************************************************************/
|
|
|
|
#include "portab.h"
|
|
#include "globals.h"
|
|
|
|
#ifdef VERSION_STRINGS
|
|
BYTE *RcsId = "$Id$";
|
|
#endif
|
|
|
|
/* */
|
|
/* function prototypes */
|
|
/* */
|
|
f_node_ptr xlt_fd(COUNT);
|
|
COUNT xlt_fnp(f_node_ptr);
|
|
STATIC void save_far_f_node(f_node_ptr fnp);
|
|
STATIC f_node_ptr get_near_f_node(void);
|
|
STATIC f_node_ptr split_path(char *, char *);
|
|
BOOL find_fname(f_node_ptr, char *, int);
|
|
/* /// Added - Ron Cemer */
|
|
STATIC void merge_file_changes(f_node_ptr fnp, int collect);
|
|
/* /// Added - Ron Cemer */
|
|
STATIC int is_same_file(f_node_ptr fnp1, f_node_ptr fnp2);
|
|
/* /// Added - Ron Cemer */
|
|
STATIC void copy_file_changes(f_node_ptr src, f_node_ptr dst);
|
|
BOOL find_free(f_node_ptr);
|
|
STATIC int alloc_find_free(f_node_ptr fnp, char *path, char *fcbname);
|
|
VOID wipe_out(f_node_ptr);
|
|
CLUSTER extend(f_node_ptr);
|
|
COUNT extend_dir(f_node_ptr);
|
|
CLUSTER first_fat(f_node_ptr);
|
|
COUNT map_cluster(f_node_ptr, COUNT);
|
|
STATIC VOID shrink_file(f_node_ptr fnp);
|
|
|
|
#ifdef WITHFAT32
|
|
CLUSTER getdstart(struct dpb FAR *dpbp, struct dirent *dentry)
|
|
{
|
|
if (!ISFAT32(dpbp))
|
|
return dentry->dir_start;
|
|
return (((CLUSTER)dentry->dir_start_high << 16) | dentry->dir_start);
|
|
}
|
|
|
|
void setdstart(struct dpb FAR *dpbp, struct dirent *dentry, CLUSTER value)
|
|
{
|
|
if (!ISFAT32(dpbp))
|
|
{
|
|
dentry->dir_start = (UWORD)value;
|
|
return;
|
|
}
|
|
dentry->dir_start = (UWORD)value;
|
|
dentry->dir_start_high = (UWORD)(value >> 16);
|
|
}
|
|
|
|
BOOL checkdstart(struct dpb FAR *dpbp, struct dirent *dentry, CLUSTER value)
|
|
{
|
|
if (!ISFAT32(dpbp))
|
|
return dentry->dir_start == (UWORD)value;
|
|
return (dentry->dir_start == (UWORD)value &&
|
|
dentry->dir_start_high == (UWORD)(value >> 16));
|
|
}
|
|
#endif
|
|
|
|
ULONG clus2phys(CLUSTER cl_no, struct dpb FAR * dpbp)
|
|
{
|
|
CLUSTER data =
|
|
#ifdef WITHFAT32
|
|
ISFAT32(dpbp) ? dpbp->dpb_xdata :
|
|
#endif
|
|
dpbp->dpb_data;
|
|
return ((ULONG) (cl_no - 2) << dpbp->dpb_shftcnt) + data;
|
|
}
|
|
|
|
struct dpb FAR *get_dpb(COUNT dsk)
|
|
{
|
|
register struct cds FAR *cdsp = get_cds(dsk);
|
|
|
|
if (cdsp == NULL || cdsp->cdsFlags & CDSNETWDRV)
|
|
return NULL;
|
|
return cdsp->cdsDpb;
|
|
}
|
|
|
|
/* initialize all direntry fields except for the name */
|
|
STATIC void init_direntry(struct dirent *dentry, unsigned attrib,
|
|
CLUSTER cluster)
|
|
{
|
|
struct dostime dt;
|
|
|
|
dentry->dir_size = 0l;
|
|
#ifdef WITHFAT32
|
|
dentry->dir_start_high = (UWORD)(cluster >> 16);
|
|
#else
|
|
dentry->dir_start_high = 0;
|
|
#endif
|
|
dentry->dir_start = (UWORD)cluster;
|
|
dentry->dir_attrib = attrib;
|
|
dentry->dir_case = 0;
|
|
DosGetTime(&dt);
|
|
dentry->dir_crtimems = dt.hundredth;
|
|
if (dt.second & 1)
|
|
dentry->dir_crtimems += 100;
|
|
dentry->dir_time = dentry->dir_crtime =
|
|
TM_ENCODE(dt.hour, dt.minute, dt.second >> 1);
|
|
dentry->dir_date = dentry->dir_crdate = dentry->dir_accdate = dos_getdate();
|
|
}
|
|
|
|
/************************************************************************/
|
|
/* */
|
|
/* Internal file handlers - open, create, read, write, close, etc. */
|
|
/* */
|
|
/************************************************************************/
|
|
|
|
/* Open a file given the path. Flags is 0 for read, 1 for write and 2 */
|
|
/* for update. */
|
|
/* Returns an long where the high word is a status code and the low */
|
|
/* word is an integer file descriptor or a negative error code */
|
|
/* see DosOpenSft(), dosfns.c for an explanation of the flags bits */
|
|
/* directory opens are allowed here; these are not allowed by DosOpenSft*/
|
|
|
|
long dos_open(char *path, unsigned flags, unsigned attrib)
|
|
{
|
|
REG f_node_ptr fnp;
|
|
char fcbname[FNAME_SIZE + FEXT_SIZE];
|
|
int status = S_OPENED;
|
|
|
|
/* First test the flags to see if the user has passed a valid */
|
|
/* file mode... */
|
|
if ((flags & O_ACCMODE) > 2)
|
|
return DE_INVLDACC;
|
|
|
|
/* first split the passed dir into comopnents (i.e. - path to */
|
|
/* new directory and name of new directory. */
|
|
if ((fnp = split_path(path, fcbname)) == NULL)
|
|
return DE_PATHNOTFND;
|
|
|
|
/* Check that we don't have a duplicate name, so if we */
|
|
/* find one, truncate it (O_CREAT). */
|
|
if (find_fname(fnp, fcbname, D_ALL | attrib))
|
|
{
|
|
if (flags & O_TRUNC)
|
|
{
|
|
/* The only permissable attribute is archive, */
|
|
/* check for any other bit set. If it is, give */
|
|
/* an access error. */
|
|
if ((fnp->f_dir.dir_attrib & (D_RDONLY | D_DIR | D_VOLID))
|
|
|| (fnp->f_dir.dir_attrib & ~D_ARCHIVE & ~attrib))
|
|
{
|
|
dir_close(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
|
|
/* Release the existing files FAT and set the */
|
|
/* length to zero, effectively truncating the */
|
|
/* file to zero. */
|
|
wipe_out(fnp);
|
|
status = S_REPLACED;
|
|
}
|
|
else if (flags & O_OPEN)
|
|
{
|
|
/* force r/o open for FCB if the file is read-only */
|
|
if ((flags & O_FCB) && (fnp->f_dir.dir_attrib & D_RDONLY))
|
|
flags = (flags & ~3) | O_RDONLY;
|
|
|
|
/* Check permissions. -- JPP */
|
|
if ((fnp->f_dir.dir_attrib & D_RDONLY) &&
|
|
((flags & O_ACCMODE) != O_RDONLY))
|
|
{
|
|
dir_close(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
dir_close(fnp);
|
|
return DE_FILEEXISTS;
|
|
}
|
|
}
|
|
else if (flags & O_CREAT)
|
|
{
|
|
int ret = alloc_find_free(fnp, path, fcbname);
|
|
if (ret != SUCCESS)
|
|
return ret;
|
|
|
|
/* put the fnode's name into the directory. */
|
|
memcpy(fnp->f_dir.dir_name, fcbname, FNAME_SIZE + FEXT_SIZE);
|
|
status = S_CREATED;
|
|
}
|
|
else
|
|
{
|
|
/* open: If we can't find the file, just return a not */
|
|
/* found error. */
|
|
dir_close(fnp);
|
|
return DE_FILENOTFND;
|
|
}
|
|
|
|
/* Set the fnode to the desired mode */
|
|
/* Updating the directory entry first. */
|
|
fnp->f_mode = flags & O_ACCMODE;
|
|
|
|
if (status != S_OPENED)
|
|
{
|
|
init_direntry(&fnp->f_dir, attrib, FREE);
|
|
fnp->f_flags = F_DMOD | F_DDIR;
|
|
if (!dir_write(fnp))
|
|
{
|
|
release_f_node(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
}
|
|
|
|
/* Now change to file */
|
|
fnp->f_offset = 0l;
|
|
|
|
if (status != S_OPENED)
|
|
{
|
|
fnp->f_cluster = FREE;
|
|
setdstart(fnp->f_dpb, &fnp->f_dir, FREE);
|
|
fnp->f_cluster_offset = 0;
|
|
}
|
|
|
|
fnp->f_flags = 0;
|
|
if (status != S_OPENED)
|
|
fnp->f_flags = F_DMOD;
|
|
|
|
merge_file_changes(fnp, status == S_OPENED); /* /// Added - Ron Cemer */
|
|
/* /// Moved from above. - Ron Cemer */
|
|
fnp->f_cluster = getdstart(fnp->f_dpb, &fnp->f_dir);
|
|
fnp->f_cluster_offset = 0;
|
|
|
|
save_far_f_node(fnp);
|
|
return xlt_fnp(fnp) | ((long)status << 16);
|
|
}
|
|
|
|
BOOL fcmp_wild(const char * s1, const char * s2, unsigned n)
|
|
{
|
|
while (n--)
|
|
{
|
|
if (*s1 == '?')
|
|
{
|
|
++s1, ++s2;
|
|
continue;
|
|
}
|
|
if (*s1++ != *s2++)
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
COUNT dos_close(COUNT fd)
|
|
{
|
|
f_node_ptr fnp;
|
|
|
|
/* Translate the fd into a useful pointer */
|
|
fnp = xlt_fd(fd);
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
if (fnp == (f_node_ptr) 0)
|
|
return DE_INVLDHNDL;
|
|
|
|
if (fnp->f_flags & F_DMOD)
|
|
{
|
|
if (!(fnp->f_flags & F_DDATE))
|
|
{
|
|
fnp->f_dir.dir_time = dos_gettime();
|
|
fnp->f_dir.dir_date = dos_getdate();
|
|
}
|
|
|
|
merge_file_changes(fnp, FALSE); /* /// Added - Ron Cemer */
|
|
}
|
|
fnp->f_flags |= F_DDIR;
|
|
|
|
dir_close(fnp);
|
|
return SUCCESS;
|
|
}
|
|
|
|
COUNT dos_commit(COUNT fd)
|
|
{
|
|
f_node_ptr fnp, fnp2;
|
|
|
|
/* Translate the fd into a useful pointer */
|
|
fnp = xlt_fd(fd);
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
if (fnp == (f_node_ptr) 0)
|
|
return DE_INVLDHNDL;
|
|
fnp2 = get_f_node();
|
|
if (fnp2 == (f_node_ptr) 0)
|
|
{
|
|
release_near_f_node(fnp);
|
|
return DE_INVLDHNDL;
|
|
}
|
|
|
|
/* a copy of the fnode is closed meaning that the directory info
|
|
is updated etc, but we keep our old info */
|
|
memcpy(fnp2, fnp, sizeof(*fnp));
|
|
save_far_f_node(fnp2);
|
|
release_near_f_node(fnp);
|
|
return dos_close(xlt_fnp(fnp2));
|
|
}
|
|
|
|
/* */
|
|
/* split a path into it's component directory and file name */
|
|
/* */
|
|
f_node_ptr split_path(char * path, char * fcbname)
|
|
{
|
|
REG f_node_ptr fnp;
|
|
|
|
/* Start off by parsing out the components. */
|
|
int dirlength = ParseDosName(path, fcbname, FALSE);
|
|
|
|
if (dirlength < SUCCESS)
|
|
return (f_node_ptr) 0;
|
|
|
|
/* 11/29/99 jt
|
|
* Networking and Cdroms. You can put in here a return.
|
|
* Maybe a return of 0xDEADBEEF or something for Split or Dir_open.
|
|
* Just to let upper level Fdos know its a sft, CDS function.
|
|
* Right now for Networking there is no support for Rename, MkDir
|
|
* RmDir & Delete.
|
|
|
|
<insert code here or in dir_open. I would but it in Dir_open.
|
|
Do the redirection in Network.c>
|
|
|
|
*/
|
|
#ifdef DEBUG
|
|
if (get_cds(path[0]-'A')->cdsFlags & CDSNETWDRV)
|
|
{
|
|
printf("split path called for redirected file: `%s'\n",
|
|
fcbname);
|
|
return (f_node_ptr) 0;
|
|
}
|
|
#endif
|
|
|
|
/* Translate the path into a useful pointer */
|
|
{
|
|
char tmp = path[dirlength];
|
|
path[dirlength] = '\0';
|
|
fnp = dir_open(path);
|
|
path[dirlength] = tmp;
|
|
}
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit... */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
if (fnp == (f_node_ptr) 0 || fnp->f_count <= 0)
|
|
{
|
|
dir_close(fnp);
|
|
return (f_node_ptr) 0;
|
|
}
|
|
|
|
return fnp;
|
|
}
|
|
|
|
/* checks whether directory part of path exists */
|
|
BOOL dir_exists(char * path)
|
|
{
|
|
REG f_node_ptr fnp;
|
|
char fcbname[FNAME_SIZE + FEXT_SIZE];
|
|
|
|
if ((fnp = split_path(path, fcbname)) == NULL)
|
|
return FALSE;
|
|
|
|
dir_close(fnp);
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL fcbmatch(const char *fcbname1, const char *fcbname2)
|
|
{
|
|
return memcmp(fcbname1, fcbname2, FNAME_SIZE + FEXT_SIZE) == 0;
|
|
}
|
|
|
|
STATIC BOOL find_fname(f_node_ptr fnp, char *fcbname, int attr)
|
|
{
|
|
while (dir_read(fnp) == 1)
|
|
{
|
|
if (fcbmatch(fnp->f_dir.dir_name, fcbname)
|
|
&& (fnp->f_dir.dir_attrib & ~(D_RDONLY | D_ARCHIVE | attr)) == 0)
|
|
{
|
|
return TRUE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
/* Description.
|
|
* Remove entries with D_LFN attribute preceeding the directory entry
|
|
* pointed by fnp, fnode isn't modified (I hope).
|
|
* Return value.
|
|
* SUCCESS - completed successfully.
|
|
* DE_BLKINVLD - error occured, fnode is released.
|
|
* input: fnp with valid non-LFN directory entry, not equal to '..' or
|
|
* '.'
|
|
*/
|
|
COUNT remove_lfn_entries(f_node_ptr fnp)
|
|
{
|
|
unsigned original_diroff = fnp->f_diroff;
|
|
|
|
while (TRUE)
|
|
{
|
|
if (fnp->f_diroff == 0)
|
|
break;
|
|
fnp->f_diroff -= 2;
|
|
/* it cannot / should not get below 0 because of '.' and '..'
|
|
* except for root directories... but then dir_read() makes it 0
|
|
* again */
|
|
if (dir_read(fnp) <= 0) {
|
|
dir_close(fnp);
|
|
return DE_BLKINVLD;
|
|
}
|
|
if (fnp->f_dir.dir_attrib != D_LFN)
|
|
break;
|
|
fnp->f_dir.dir_name[0] = DELETED;
|
|
fnp->f_flags |= F_DMOD;
|
|
if (!dir_write(fnp)) return DE_BLKINVLD;
|
|
}
|
|
fnp->f_diroff = original_diroff - 1;
|
|
if (dir_read(fnp) <= 0) {
|
|
dir_close(fnp);
|
|
return DE_BLKINVLD;
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* /// Added - Ron Cemer */
|
|
/* If more than one f_node has a file open, and a write
|
|
occurs, this function must be called to propagate the
|
|
results of that write to the other f_nodes which have
|
|
that file open. Note that this function only has an
|
|
effect if SHARE is installed. This is for compatibility
|
|
reasons, since DOS without SHARE does not share changes
|
|
between two or more open instances of the same file
|
|
unless these instances were generated by dup() or dup2(). */
|
|
STATIC void merge_file_changes(f_node_ptr fnp, int collect)
|
|
{
|
|
f_node_ptr fnp2;
|
|
int i, fd;
|
|
|
|
if (!IsShareInstalled())
|
|
return;
|
|
|
|
fd = xlt_fnp(fnp);
|
|
fnp2 = get_near_f_node();
|
|
for (i = 0; i < f_nodes_cnt; i++)
|
|
{
|
|
fmemcpy(fnp2, &f_nodes[i], sizeof(*fnp2));
|
|
if ((fnp != (f_node_ptr) 0)
|
|
&& (i != fd)
|
|
&& (fnp->f_count > 0) && (is_same_file(fnp, fnp2)))
|
|
{
|
|
if (collect)
|
|
{
|
|
/* We're collecting file changes from any other
|
|
f_node which refers to this file. */
|
|
if (fnp2->f_mode != RDONLY)
|
|
{
|
|
copy_file_changes(fnp2, fnp);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* We just made changes to this file, so we are
|
|
distributing these changes to the other f_nodes
|
|
which refer to this file. */
|
|
copy_file_changes(fnp, fnp2);
|
|
fmemcpy(&f_nodes[i], fnp2, sizeof(*fnp2));
|
|
}
|
|
}
|
|
}
|
|
release_near_f_node(fnp2);
|
|
}
|
|
|
|
/* /// Added - Ron Cemer */
|
|
STATIC int is_same_file(f_node_ptr fnp1, f_node_ptr fnp2)
|
|
{
|
|
return
|
|
(fnp1->f_dpb->dpb_unit == fnp2->f_dpb->dpb_unit)
|
|
&& (fnp1->f_dpb->dpb_subunit == fnp2->f_dpb->dpb_subunit)
|
|
&& (fcbmatch(fnp1->f_dir.dir_name, fnp2->f_dir.dir_name))
|
|
&& ((fnp1->f_dir.dir_attrib & D_VOLID) == 0)
|
|
&& ((fnp2->f_dir.dir_attrib & D_VOLID) == 0)
|
|
&& (fnp1->f_diroff == fnp2->f_diroff)
|
|
&& (fnp1->f_dirstart == fnp2->f_dirstart)
|
|
&& (fnp1->f_dpb == fnp2->f_dpb);
|
|
}
|
|
|
|
/* /// Added - Ron Cemer */
|
|
STATIC void copy_file_changes(f_node_ptr src, f_node_ptr dst)
|
|
{
|
|
dst->f_dir.dir_start = src->f_dir.dir_start;
|
|
#ifdef WITHFAT32
|
|
dst->f_dir.dir_start_high = src->f_dir.dir_start_high;
|
|
#endif
|
|
dst->f_dir.dir_size = src->f_dir.dir_size;
|
|
dst->f_dir.dir_date = src->f_dir.dir_date;
|
|
dst->f_dir.dir_time = src->f_dir.dir_time;
|
|
}
|
|
|
|
STATIC COUNT delete_dir_entry(f_node_ptr fnp)
|
|
{
|
|
COUNT rc;
|
|
|
|
/* Ok, so we can delete. Start out by */
|
|
/* clobbering all FAT entries for this file */
|
|
/* (or, in English, truncate the FAT). */
|
|
if ((rc = remove_lfn_entries(fnp)) < 0)
|
|
return rc;
|
|
|
|
wipe_out(fnp);
|
|
*(fnp->f_dir.dir_name) = DELETED;
|
|
|
|
/* The directory has been modified, so set the */
|
|
/* bit before closing it, allowing it to be */
|
|
/* updated */
|
|
fnp->f_flags |= F_DMOD;
|
|
dir_close(fnp);
|
|
|
|
/* SUCCESSful completion, return it */
|
|
return SUCCESS;
|
|
}
|
|
|
|
COUNT dos_delete(BYTE * path, int attrib)
|
|
{
|
|
REG f_node_ptr fnp;
|
|
char fcbname[FNAME_SIZE + FEXT_SIZE];
|
|
|
|
/* first split the passed dir into components (i.e. - */
|
|
/* path to new directory and name of new directory */
|
|
if ((fnp = split_path(path, fcbname)) == NULL)
|
|
{
|
|
return DE_PATHNOTFND;
|
|
}
|
|
|
|
/* Check that we don't have a duplicate name, so if we */
|
|
/* find one, it's an error. */
|
|
if (find_fname(fnp, fcbname, attrib))
|
|
{
|
|
/* Do not delete directories or r/o files */
|
|
/* lfn entries and volume labels are only found */
|
|
/* by find_fname() if attrib is set to a */
|
|
/* special value */
|
|
if (fnp->f_dir.dir_attrib & (D_RDONLY | D_DIR))
|
|
{
|
|
dir_close(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
|
|
return delete_dir_entry(fnp);
|
|
}
|
|
else
|
|
{
|
|
/* No such file, return the error */
|
|
dir_close(fnp);
|
|
return DE_FILENOTFND;
|
|
}
|
|
}
|
|
|
|
COUNT dos_rmdir(BYTE * path)
|
|
{
|
|
REG f_node_ptr fnp;
|
|
REG f_node_ptr fnp1;
|
|
BOOL found;
|
|
char fcbname[FNAME_SIZE + FEXT_SIZE];
|
|
|
|
/* first split the passed dir into comopnents (i.e. - */
|
|
/* path to new directory and name of new directory */
|
|
if ((fnp = split_path(path, fcbname)) == NULL)
|
|
{
|
|
return DE_PATHNOTFND;
|
|
}
|
|
|
|
/* Check that we're not trying to remove the root! */
|
|
if ((path[0] == '\\') && (path[1] == NULL))
|
|
{
|
|
dir_close(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
|
|
/* Check that we don't have a duplicate name, so if we */
|
|
/* find one, it's an error. */
|
|
if (find_fname(fnp, fcbname, D_ALL))
|
|
{
|
|
/* The only permissable attribute is directory, */
|
|
/* check for any other bit set. If it is, give */
|
|
/* an access error. */
|
|
/* if (fnp->f_dir.dir_attrib & ~D_DIR) */
|
|
|
|
/* directories may have attributes, too. at least my WinNT disk
|
|
has many 'archive' directories
|
|
we still don't allow RDONLY directories to be deleted TE */
|
|
|
|
/* if (fnp->f_dir.dir_attrib & ~(D_DIR |D_HIDDEN|D_ARCHIVE|D_SYSTEM))
|
|
{
|
|
dir_close(fnp);
|
|
return DE_ACCESS;
|
|
} */
|
|
|
|
/* Check that the directory is empty. Only the */
|
|
/* "." and ".." are permissable. */
|
|
fnp->f_flags &= ~F_DMOD;
|
|
fnp1 = dir_open(path);
|
|
if (fnp1 == NULL)
|
|
{
|
|
dir_close(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
|
|
dir_read(fnp1);
|
|
if (fnp1->f_dir.dir_name[0] != '.')
|
|
{
|
|
dir_close(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
|
|
dir_read(fnp1);
|
|
if (fnp1->f_dir.dir_name[0] != '.')
|
|
{
|
|
dir_close(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
|
|
/* Now search through the directory and make certain */
|
|
/* that there are no entries. */
|
|
found = FALSE;
|
|
while (dir_read(fnp1) == 1)
|
|
{
|
|
if (fnp1->f_dir.dir_name[0] == '\0')
|
|
break;
|
|
if (fnp1->f_dir.dir_name[0] == DELETED
|
|
|| fnp1->f_dir.dir_attrib == D_LFN)
|
|
continue;
|
|
else
|
|
{
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
dir_close(fnp1);
|
|
/* If anything was found, exit with an error. */
|
|
if (found)
|
|
{
|
|
dir_close(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
return delete_dir_entry(fnp);
|
|
}
|
|
else
|
|
{
|
|
/* No such file, return the error */
|
|
dir_close(fnp);
|
|
return DE_FILENOTFND;
|
|
}
|
|
}
|
|
|
|
COUNT dos_rename(BYTE * path1, BYTE * path2, int attrib)
|
|
{
|
|
REG f_node_ptr fnp1;
|
|
REG f_node_ptr fnp2;
|
|
COUNT ret;
|
|
char fcbname[FNAME_SIZE + FEXT_SIZE];
|
|
|
|
/* first split the passed target into compnents (i.e. - path to */
|
|
/* new file name and name of new file name */
|
|
if ((fnp2 = split_path(path2, fcbname)) == NULL)
|
|
{
|
|
return DE_PATHNOTFND;
|
|
}
|
|
|
|
/* Check that we don't have a duplicate name, so if we find */
|
|
/* one, it's an error. */
|
|
if (find_fname(fnp2, fcbname, attrib))
|
|
{
|
|
dir_close(fnp2);
|
|
return DE_ACCESS;
|
|
}
|
|
|
|
/* next split the passed source into compnents (i.e. - path to */
|
|
/* old file name and name of old file name */
|
|
if ((fnp1 = split_path(path1, fcbname)) == NULL)
|
|
{
|
|
dir_close(fnp2);
|
|
return DE_PATHNOTFND;
|
|
}
|
|
|
|
if (!find_fname(fnp1, fcbname, attrib))
|
|
{
|
|
/* No such file, return the error */
|
|
dir_close(fnp1);
|
|
dir_close(fnp2);
|
|
return DE_FILENOTFND;
|
|
}
|
|
|
|
ret = alloc_find_free(fnp2, path2, fcbname);
|
|
if (ret != SUCCESS) {
|
|
dir_close(fnp1);
|
|
return ret;
|
|
}
|
|
|
|
if ((ret = remove_lfn_entries(fnp1)) < 0)
|
|
return ret;
|
|
|
|
/* put the fnode's name into the directory. */
|
|
memcpy(fnp2->f_dir.dir_name, fcbname, FNAME_SIZE + FEXT_SIZE);
|
|
|
|
/* Set the fnode to the desired mode */
|
|
fnp2->f_dir.dir_size = fnp1->f_dir.dir_size;
|
|
fnp2->f_dir.dir_start = fnp1->f_dir.dir_start;
|
|
#ifdef WITHFAT32
|
|
fnp2->f_dir.dir_start_high = fnp1->f_dir.dir_start_high;
|
|
#endif
|
|
fnp2->f_dir.dir_attrib = fnp1->f_dir.dir_attrib;
|
|
fnp2->f_dir.dir_time = fnp1->f_dir.dir_time;
|
|
fnp2->f_dir.dir_date = fnp1->f_dir.dir_date;
|
|
|
|
/* The directory has been modified, so set the bit before */
|
|
/* closing it, allowing it to be updated. */
|
|
fnp1->f_flags = fnp2->f_flags = F_DMOD | F_DDIR;
|
|
|
|
/* Ok, so we can delete this one. Save the file info. */
|
|
*(fnp1->f_dir.dir_name) = DELETED;
|
|
|
|
dir_close(fnp1);
|
|
dir_close(fnp2);
|
|
|
|
/* SUCCESSful completion, return it */
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* */
|
|
/* wipe out all FAT entries starting from st for create, delete, etc. */
|
|
/* */
|
|
STATIC VOID wipe_out_clusters(struct dpb FAR * dpbp, CLUSTER st)
|
|
{
|
|
REG CLUSTER next;
|
|
|
|
/* Loop from start until either a FREE entry is */
|
|
/* encountered (due to a fractured file system) of the */
|
|
/* last cluster is encountered. */
|
|
while (st != LONG_LAST_CLUSTER)
|
|
{
|
|
/* get the next cluster pointed to */
|
|
next = next_cluster(dpbp, st);
|
|
|
|
/* just exit if a damaged file system exists */
|
|
if (next == FREE || next == 1)
|
|
return;
|
|
|
|
/* zap the FAT pointed to */
|
|
link_fat(dpbp, st, FREE);
|
|
|
|
/* and the start of free space pointer */
|
|
#ifdef WITHFAT32
|
|
if (ISFAT32(dpbp))
|
|
{
|
|
if ((dpbp->dpb_xcluster == UNKNCLUSTER) || (dpbp->dpb_xcluster > st))
|
|
dpbp->dpb_xcluster = st;
|
|
}
|
|
else
|
|
#endif
|
|
if ((dpbp->dpb_cluster == UNKNCLUSTER) || (dpbp->dpb_cluster > st))
|
|
dpbp->dpb_cluster = (UWORD)st;
|
|
|
|
/* and just follow the linked list */
|
|
st = next;
|
|
}
|
|
#ifdef WITHFAT32
|
|
if (ISFAT32(dpbp))
|
|
write_fsinfo(dpbp);
|
|
#endif
|
|
}
|
|
|
|
/* */
|
|
/* wipe out all FAT entries for create, delete, etc. */
|
|
/* */
|
|
STATIC VOID wipe_out(f_node_ptr fnp)
|
|
{
|
|
/* if already free or not valid file, just exit */
|
|
if ((fnp == NULL) || checkdstart(fnp->f_dpb, &fnp->f_dir, FREE))
|
|
return;
|
|
|
|
wipe_out_clusters(fnp->f_dpb, getdstart(fnp->f_dpb, &fnp->f_dir));
|
|
}
|
|
|
|
STATIC BOOL find_free(f_node_ptr fnp)
|
|
{
|
|
COUNT rc;
|
|
|
|
while ((rc = dir_read(fnp)) == 1)
|
|
if (fnp->f_dir.dir_name[0] == DELETED)
|
|
return TRUE;
|
|
return rc >= 0;
|
|
}
|
|
|
|
/* alloc_find_free: resets the directory by a close followed by */
|
|
/* an open. Then finds a spare directory entry and if not */
|
|
/* available, tries to extend the directory. */
|
|
STATIC int alloc_find_free(f_node_ptr fnp, char *path, char *fcbname)
|
|
{
|
|
fnp->f_flags &= ~F_DMOD;
|
|
dir_close(fnp);
|
|
fnp = split_path(path, fcbname);
|
|
|
|
/* Get a free f_node pointer so that we can use */
|
|
/* it in building the new file. */
|
|
/* Note that if we're in the root and we don't */
|
|
/* find an empty slot, we need to abort. */
|
|
if (find_free(fnp) == 0)
|
|
{
|
|
if (fnp->f_dirstart == 0)
|
|
{
|
|
fnp->f_flags &= ~F_DMOD;
|
|
dir_close(fnp);
|
|
return DE_TOOMANY;
|
|
}
|
|
else
|
|
{
|
|
/* Otherwise just expand the directory */
|
|
int ret;
|
|
|
|
if ((ret = extend_dir(fnp)) != SUCCESS)
|
|
/* fnp already closed in extend_dir */
|
|
return ret;
|
|
}
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
|
|
/* */
|
|
/* dos_getdate for the file date */
|
|
/* */
|
|
date dos_getdate(void)
|
|
{
|
|
struct dosdate dd;
|
|
|
|
/* First - get the system date set by either the user */
|
|
/* on start-up or the CMOS clock */
|
|
DosGetDate(&dd);
|
|
return DT_ENCODE(dd.month, dd.monthday, dd.year - EPOCH_YEAR);
|
|
}
|
|
|
|
/* */
|
|
/* dos_gettime for the file time */
|
|
/* */
|
|
time dos_gettime(void)
|
|
{
|
|
struct dostime dt;
|
|
|
|
/* First - get the system time set by either the user */
|
|
/* on start-up or the CMOS clock */
|
|
DosGetTime(&dt);
|
|
return TM_ENCODE(dt.hour, dt.minute, dt.second >> 1);
|
|
}
|
|
|
|
/* */
|
|
/* dos_getftime for the file time */
|
|
/* */
|
|
COUNT dos_getftime(COUNT fd, date FAR * dp, time FAR * tp)
|
|
{
|
|
f_node_ptr fnp;
|
|
|
|
/* Translate the fd into an fnode pointer, since all internal */
|
|
/* operations are achieved through fnodes. */
|
|
fnp = xlt_fd(fd);
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
if (fnp == (f_node_ptr) 0)
|
|
return DE_INVLDHNDL;
|
|
|
|
/* Get the date and time from the fnode and return */
|
|
*dp = fnp->f_dir.dir_date;
|
|
*tp = fnp->f_dir.dir_time;
|
|
|
|
release_near_f_node(fnp);
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* */
|
|
/* dos_setftime for the file time */
|
|
/* */
|
|
COUNT dos_setftime(COUNT fd, date dp, time tp)
|
|
{
|
|
f_node_ptr fnp;
|
|
|
|
/* Translate the fd into an fnode pointer, since all internal */
|
|
/* operations are achieved through fnodes. */
|
|
fnp = xlt_fd(fd);
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
if (fnp == (f_node_ptr) 0)
|
|
return DE_INVLDHNDL;
|
|
|
|
/* Set the date and time from the fnode and return */
|
|
fnp->f_dir.dir_date = dp;
|
|
fnp->f_dir.dir_time = tp;
|
|
/* mark file as modified and set this date upon closing */
|
|
fnp->f_flags |= F_DMOD | F_DDATE;
|
|
|
|
save_far_f_node(fnp);
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* */
|
|
/* dos_getfsize for the file time */
|
|
/* */
|
|
ULONG dos_getfsize(COUNT fd)
|
|
{
|
|
f_node_ptr fnp;
|
|
|
|
/* Translate the fd into an fnode pointer, since all internal */
|
|
/* operations are achieved through fnodes. */
|
|
fnp = xlt_fd(fd);
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
if (fnp == (f_node_ptr) 0)
|
|
return (ULONG)-1l;
|
|
|
|
/* Return the file size */
|
|
release_near_f_node(fnp);
|
|
return fnp->f_dir.dir_size;
|
|
}
|
|
|
|
/* */
|
|
/* dos_setfsize for the file time */
|
|
/* */
|
|
BOOL dos_setfsize(COUNT fd, LONG size)
|
|
{
|
|
f_node_ptr fnp;
|
|
|
|
/* Translate the fd into an fnode pointer, since all internal */
|
|
/* operations are achieved through fnodes. */
|
|
fnp = xlt_fd(fd);
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
if (fnp == (f_node_ptr) 0)
|
|
return FALSE;
|
|
|
|
/* Change the file size */
|
|
fnp->f_dir.dir_size = size;
|
|
|
|
merge_file_changes(fnp, FALSE); /* /// Added - Ron Cemer */
|
|
save_far_f_node(fnp);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* */
|
|
/* Find free cluster in disk FAT table */
|
|
/* */
|
|
STATIC CLUSTER find_fat_free(f_node_ptr fnp)
|
|
{
|
|
REG CLUSTER idx, size;
|
|
struct dpb FAR *dpbp = fnp->f_dpb;
|
|
|
|
#ifdef DISPLAY_GETBLOCK
|
|
printf("[find_fat_free]\n");
|
|
#endif
|
|
|
|
/* Start from optimized lookup point for start of FAT */
|
|
idx = 2;
|
|
size = dpbp->dpb_size;
|
|
|
|
#ifdef WITHFAT32
|
|
if (ISFAT32(dpbp))
|
|
{
|
|
if (dpbp->dpb_xcluster != UNKNCLUSTER)
|
|
idx = dpbp->dpb_xcluster;
|
|
size = dpbp->dpb_xsize;
|
|
}
|
|
else
|
|
#endif
|
|
if (dpbp->dpb_cluster != UNKNCLUSTER)
|
|
idx = dpbp->dpb_cluster;
|
|
|
|
/* Search the FAT table looking for the first free */
|
|
/* entry. */
|
|
for (; idx <= size; idx++)
|
|
{
|
|
if (next_cluster(dpbp, idx) == FREE)
|
|
break;
|
|
}
|
|
|
|
/* No empty clusters, disk is FULL! */
|
|
#ifdef WITHFAT32
|
|
if (ISFAT32(dpbp))
|
|
{
|
|
if (idx > dpbp->dpb_xsize)
|
|
{
|
|
dpbp->dpb_xcluster = UNKNCLUSTER;
|
|
write_fsinfo(dpbp);
|
|
return LONG_LAST_CLUSTER;
|
|
}
|
|
if (dpbp->dpb_xnfreeclst != XUNKNCLSTFREE)
|
|
dpbp->dpb_xnfreeclst--; /* TE: moved from link_fat() */
|
|
|
|
/* return the free entry */
|
|
dpbp->dpb_xcluster = idx;
|
|
write_fsinfo(dpbp);
|
|
return idx;
|
|
}
|
|
#endif
|
|
|
|
if ((UWORD)idx > dpbp->dpb_size)
|
|
{
|
|
dpbp->dpb_cluster = UNKNCLUSTER;
|
|
return LONG_LAST_CLUSTER;
|
|
}
|
|
|
|
if (dpbp->dpb_nfreeclst != UNKNCLSTFREE)
|
|
dpbp->dpb_nfreeclst--; /* TE: moved from link_fat() */
|
|
|
|
/* return the free entry */
|
|
dpbp->dpb_cluster = (UWORD)idx;
|
|
return idx;
|
|
}
|
|
|
|
/* */
|
|
/* create a directory - returns success or a negative error */
|
|
/* number */
|
|
/* */
|
|
COUNT dos_mkdir(BYTE * dir)
|
|
{
|
|
REG f_node_ptr fnp;
|
|
REG COUNT idx;
|
|
struct buffer FAR *bp;
|
|
struct dpb FAR *dpbp;
|
|
CLUSTER free_fat, parent;
|
|
COUNT ret;
|
|
char fcbname[FNAME_SIZE + FEXT_SIZE];
|
|
|
|
/* first split the passed dir into components (i.e. - */
|
|
/* path to new directory and name of new directory */
|
|
if ((fnp = split_path(dir, fcbname)) == NULL)
|
|
{
|
|
return DE_PATHNOTFND;
|
|
}
|
|
|
|
/* check that the resulting combined path does not exceed
|
|
the 67 MAX_CDSPATH limit. this leads to problems:
|
|
A) you can't CD to this directory later
|
|
B) you can't create files in this subdirectory
|
|
C) the created dir will not be found later, so you
|
|
can create an unlimited amount of same dirs. this space
|
|
is lost forever
|
|
*/
|
|
if (strlen(dir) >= MAX_CDSPATH) /* dir is already output of "truename" */
|
|
{
|
|
dir_close(fnp);
|
|
return DE_PATHNOTFND;
|
|
}
|
|
|
|
/* Check that we don't have a duplicate name, so if we */
|
|
/* find one, it's an error. */
|
|
if (find_fname(fnp, fcbname, D_ALL))
|
|
{
|
|
dir_close(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
|
|
parent = fnp->f_dirstart;
|
|
|
|
ret = alloc_find_free(fnp, dir, fcbname);
|
|
if (ret != SUCCESS)
|
|
return ret;
|
|
|
|
/* get an empty cluster, so that we make it into a */
|
|
/* directory. */
|
|
/* TE this has to be done (and failed) BEFORE the dir entry */
|
|
/* is changed */
|
|
free_fat = find_fat_free(fnp);
|
|
|
|
/* No empty clusters, disk is FULL! Translate into a */
|
|
/* useful error message. */
|
|
if (free_fat == LONG_LAST_CLUSTER)
|
|
{
|
|
dir_close(fnp);
|
|
return DE_HNDLDSKFULL;
|
|
}
|
|
|
|
/* put the fnode's name into the directory. */
|
|
memcpy(fnp->f_dir.dir_name, fcbname, FNAME_SIZE + FEXT_SIZE);
|
|
|
|
/* Set the fnode to the desired mode */
|
|
fnp->f_mode = WRONLY;
|
|
|
|
init_direntry(&fnp->f_dir, D_DIR, free_fat);
|
|
|
|
fnp->f_flags = F_DMOD | F_DDIR;
|
|
|
|
fnp->f_offset = 0l;
|
|
|
|
/* Mark the cluster in the FAT as used */
|
|
dpbp = fnp->f_dpb;
|
|
link_fat(dpbp, free_fat, LONG_LAST_CLUSTER);
|
|
|
|
/* Craft the new directory. Note that if we're in a new */
|
|
/* directory just under the root, ".." pointer is 0. */
|
|
/* as we are overwriting it completely, don't read first */
|
|
bp = getblockOver(clus2phys(free_fat, dpbp), dpbp->dpb_unit);
|
|
#ifdef DISPLAY_GETBLOCK
|
|
printf("FAT (dos_mkdir)\n");
|
|
#endif
|
|
if (bp == NULL)
|
|
{
|
|
dir_close(fnp);
|
|
return DE_BLKINVLD;
|
|
}
|
|
|
|
/* Create the "." entry */
|
|
DirEntBuffer.dir_name[0] = '.';
|
|
memset(DirEntBuffer.dir_name + 1, ' ', FNAME_SIZE + FEXT_SIZE - 1);
|
|
init_direntry(&DirEntBuffer, D_DIR, free_fat);
|
|
|
|
/* And put it out */
|
|
putdirent(&DirEntBuffer, bp->b_buffer);
|
|
|
|
/* create the ".." entry */
|
|
DirEntBuffer.dir_name[1] = '.';
|
|
#ifdef WITHFAT32
|
|
if (ISFAT32(dpbp) && parent == dpbp->dpb_xrootclst)
|
|
{
|
|
parent = 0;
|
|
}
|
|
#endif
|
|
setdstart(dpbp, &DirEntBuffer, parent);
|
|
|
|
/* and put it out */
|
|
putdirent(&DirEntBuffer, &bp->b_buffer[DIRENT_SIZE]);
|
|
|
|
/* fill the rest of the block with zeros */
|
|
fmemset(&bp->b_buffer[2 * DIRENT_SIZE], 0, BUFFERSIZE - 2 * DIRENT_SIZE);
|
|
|
|
/* Mark the block to be written out */
|
|
bp->b_flag |= BFR_DIRTY | BFR_VALID;
|
|
|
|
/* clear out the rest of the blocks in the cluster */
|
|
for (idx = 1; idx <= dpbp->dpb_clsmask; idx++)
|
|
{
|
|
|
|
/* as we are overwriting it completely, don't read first */
|
|
bp = getblockOver(clus2phys(getdstart(dpbp, &fnp->f_dir), dpbp) + idx,
|
|
dpbp->dpb_unit);
|
|
#ifdef DISPLAY_GETBLOCK
|
|
printf("DIR (dos_mkdir)\n");
|
|
#endif
|
|
if (bp == NULL)
|
|
{
|
|
dir_close(fnp);
|
|
return DE_BLKINVLD;
|
|
}
|
|
fmemset(bp->b_buffer, 0, BUFFERSIZE);
|
|
bp->b_flag |= BFR_DIRTY | BFR_VALID;
|
|
bp->b_flag |= BFR_UNCACHE; /* need not be cached */
|
|
}
|
|
|
|
/* flush the drive buffers so that all info is written */
|
|
/* hazard: no error checking! */
|
|
flush_buffers(dpbp->dpb_unit);
|
|
|
|
/* Close the directory so that the entry is updated */
|
|
fnp->f_flags |= F_DMOD;
|
|
dir_close(fnp);
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
STATIC CLUSTER extend(f_node_ptr fnp)
|
|
{
|
|
CLUSTER free_fat;
|
|
|
|
/* get an empty cluster, so that we use it to extend the file. */
|
|
free_fat = find_fat_free(fnp);
|
|
|
|
/* No empty clusters, disk is FULL! Translate into a useful */
|
|
/* error message. */
|
|
if (free_fat == LONG_LAST_CLUSTER)
|
|
return free_fat;
|
|
|
|
/* Now that we've found a free FAT entry, mark it as the last */
|
|
/* entry and save. */
|
|
if (fnp->f_cluster == FREE)
|
|
setdstart(fnp->f_dpb, &fnp->f_dir, free_fat);
|
|
else
|
|
link_fat(fnp->f_dpb, fnp->f_cluster, free_fat);
|
|
link_fat(fnp->f_dpb, free_fat, LONG_LAST_CLUSTER);
|
|
|
|
/* Mark the directory so that the entry is updated */
|
|
fnp->f_flags |= F_DMOD;
|
|
return free_fat;
|
|
}
|
|
|
|
STATIC COUNT extend_dir(f_node_ptr fnp)
|
|
{
|
|
REG COUNT idx;
|
|
|
|
CLUSTER cluster = extend(fnp);
|
|
if (cluster == LONG_LAST_CLUSTER)
|
|
{
|
|
dir_close(fnp);
|
|
return DE_HNDLDSKFULL;
|
|
}
|
|
|
|
/* clear out the blocks in the cluster */
|
|
for (idx = 0; idx <= fnp->f_dpb->dpb_clsmask; idx++)
|
|
{
|
|
REG struct buffer FAR *bp;
|
|
|
|
/* as we are overwriting it completely, don't read first */
|
|
bp = getblockOver(clus2phys(cluster, fnp->f_dpb) + idx,
|
|
fnp->f_dpb->dpb_unit);
|
|
#ifdef DISPLAY_GETBLOCK
|
|
printf("DIR (extend_dir)\n");
|
|
#endif
|
|
if (bp == NULL)
|
|
{
|
|
dir_close(fnp);
|
|
return DE_BLKINVLD;
|
|
}
|
|
fmemset(bp->b_buffer, 0, BUFFERSIZE);
|
|
bp->b_flag |= BFR_DIRTY | BFR_VALID;
|
|
|
|
if (idx != 0)
|
|
bp->b_flag |= BFR_UNCACHE; /* needs not be cached */
|
|
}
|
|
|
|
if (!find_free(fnp))
|
|
{
|
|
dir_close(fnp);
|
|
return DE_HNDLDSKFULL;
|
|
}
|
|
|
|
/* flush the drive buffers so that all info is written */
|
|
/* hazard: no error checking! */
|
|
flush_buffers(fnp->f_dpb->dpb_unit);
|
|
|
|
return SUCCESS;
|
|
|
|
}
|
|
|
|
/* Description.
|
|
* Finds the cluster which contains byte at the fnp->f_offset offset and
|
|
* stores its number to the fnp->f_cluster. The search begins from the start of
|
|
* a file or a directory depending whether fnp->f_ddir is FALSE or TRUE
|
|
* and continues through the FAT chain until the target cluster is found.
|
|
* The mode can have only XFR_READ or XFR_WRITE values.
|
|
* In the XFR_WRITE mode map_cluster extends the FAT chain by creating
|
|
* new clusters upon necessity.
|
|
* Return value.
|
|
* DE_HNDLDSKFULL - [XFR_WRITE mode only] unable to find free cluster
|
|
* for extending the FAT chain, the disk is full.
|
|
* The fnode is released from memory.
|
|
* DE_SEEK - [XFR_READ mode only] byte at f_offset lies outside of
|
|
* the FAT chain. The fnode is not released.
|
|
* Notes.
|
|
* If we are moving forward, then use the relative cluster number offset
|
|
* that we are at now (f_cluster_offset) to start, instead of starting
|
|
* at the beginning. */
|
|
|
|
COUNT map_cluster(REG f_node_ptr fnp, COUNT mode)
|
|
{
|
|
CLUSTER relcluster, cluster;
|
|
|
|
#ifdef DISPLAY_GETBLOCK
|
|
printf("map_cluster: current %lu, offset %lu, diff=%lu ",
|
|
(ULONG)fnp->f_cluster_offset, fnp->f_offset,
|
|
fnp->f_offset - fnp->f_cluster_offset);
|
|
#endif
|
|
|
|
if (fnp->f_cluster == FREE)
|
|
{
|
|
/* If this is a read but the file still has zero bytes return */
|
|
/* immediately.... */
|
|
if (mode == XFR_READ)
|
|
return DE_SEEK;
|
|
|
|
/* If someone did a seek, but no writes have occured, we will */
|
|
/* need to initialize the fnode. */
|
|
/* (mode == XFR_WRITE) */
|
|
/* If there are no more free fat entries, then we are full! */
|
|
cluster = extend(fnp);
|
|
if (cluster == LONG_LAST_CLUSTER)
|
|
{
|
|
return DE_HNDLDSKFULL;
|
|
}
|
|
fnp->f_cluster = cluster;
|
|
}
|
|
|
|
relcluster = (CLUSTER)((fnp->f_offset / fnp->f_dpb->dpb_secsize) >>
|
|
fnp->f_dpb->dpb_shftcnt);
|
|
if (relcluster < fnp->f_cluster_offset)
|
|
{
|
|
/* Set internal index and cluster size. */
|
|
fnp->f_cluster = (fnp->f_flags & F_DDIR) ? fnp->f_dirstart :
|
|
getdstart(fnp->f_dpb, &fnp->f_dir);
|
|
fnp->f_cluster_offset = 0;
|
|
}
|
|
|
|
/* Now begin the linear search. The relative cluster is */
|
|
/* maintained as part of the set of physical indices. It is */
|
|
/* also the highest order index and is mapped directly into */
|
|
/* physical cluster. Our search is performed by pacing an index */
|
|
/* up to the relative cluster position where the index falls */
|
|
/* within the cluster. */
|
|
|
|
while (fnp->f_cluster_offset != relcluster)
|
|
{
|
|
/* get next cluster in the chain */
|
|
cluster = next_cluster(fnp->f_dpb, fnp->f_cluster);
|
|
if (cluster == 1)
|
|
return DE_SEEK;
|
|
|
|
/* If this is a read and the next is a LAST_CLUSTER, */
|
|
/* then we are going to read past EOF, return zero read */
|
|
/* or expand the list if we're going to write and have run into */
|
|
/* the last cluster marker. */
|
|
if (cluster == LONG_LAST_CLUSTER)
|
|
{
|
|
if (mode == XFR_READ)
|
|
return DE_SEEK;
|
|
|
|
/* mode == XFR_WRITE */
|
|
cluster = extend(fnp);
|
|
if (cluster == LONG_LAST_CLUSTER)
|
|
return DE_HNDLDSKFULL;
|
|
}
|
|
|
|
fnp->f_cluster = cluster;
|
|
fnp->f_cluster_offset++;
|
|
}
|
|
|
|
#ifdef DISPLAY_GETBLOCK
|
|
printf("done.\n");
|
|
#endif
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* extends a file from f_dir.dir_size to f_offset */
|
|
/* Proper OS's write zeros in between, but DOS just adds */
|
|
/* garbage sectors, and lets the caller do the zero filling */
|
|
/* if you prefer you can have this enabled using */
|
|
/* #define WRITEZEROS 1 */
|
|
/* but because we want to be compatible, we don't do this by */
|
|
/* default */
|
|
STATIC COUNT dos_extend(f_node_ptr fnp)
|
|
{
|
|
#ifdef WRITEZEROS
|
|
struct buffer FAR *bp;
|
|
UCOUNT xfr_cnt = 0;
|
|
/* The variable secsize will be used later. */
|
|
UWORD secsize = fnp->f_dpb->dpb_secsize;
|
|
ULONG count;
|
|
unsigned sector, boff;
|
|
#endif
|
|
|
|
if (fnp->f_offset <= fnp->f_dir.dir_size)
|
|
return SUCCESS;
|
|
|
|
#ifdef WRITEZEROS
|
|
count = fnp->f_offset - fnp->f_dir.dir_size;
|
|
fnp->f_offset = fnp->f_dir.dir_size;
|
|
while (count > 0)
|
|
#endif
|
|
{
|
|
if (map_cluster(fnp, XFR_WRITE) != SUCCESS)
|
|
return DE_HNDLDSKFULL;
|
|
|
|
#ifdef WRITEZEROS
|
|
/* Compute the block within the cluster and the offset */
|
|
/* within the block. */
|
|
sector = (UBYTE)(fnp->f_offset / secsize) & fnp->f_dpb->dpb_clsmask;
|
|
boff = (UWORD)(fnp->f_offset % secsize);
|
|
|
|
#ifdef DSK_DEBUG
|
|
printf("write %d links; dir offset %ld, cluster %d\n",
|
|
fnp->f_count, fnp->f_diroff, fnp->f_cluster);
|
|
#endif
|
|
|
|
xfr_cnt = count < (ULONG) secsize - boff ?
|
|
(UWORD) count : secsize - boff;
|
|
|
|
/* get a buffer to store the block in */
|
|
if ((boff == 0) && (xfr_cnt == secsize))
|
|
{
|
|
bp = getblockOver(clus2phys(fnp->f_cluster, fnp->f_dpb) +
|
|
sector, fnp->f_dpb->dpb_unit);
|
|
|
|
}
|
|
else
|
|
{
|
|
bp = getblock(clus2phys(fnp->f_cluster, fnp->f_dpb) + sector,
|
|
fnp->f_dpb->dpb_unit);
|
|
}
|
|
if (bp == NULL)
|
|
{
|
|
return DE_BLKINVLD;
|
|
}
|
|
|
|
/* set a block to zero */
|
|
fmemset((BYTE FAR *) & bp->b_buffer[boff], 0, xfr_cnt);
|
|
bp->b_flag |= BFR_DIRTY | BFR_VALID;
|
|
|
|
if (xfr_cnt == sizeof(bp->b_buffer)) /* probably not used later */
|
|
{
|
|
bp->b_flag |= BFR_UNCACHE;
|
|
}
|
|
|
|
/* update pointers and counters */
|
|
count -= xfr_cnt;
|
|
fnp->f_offset += xfr_cnt;
|
|
#endif
|
|
fnp->f_dir.dir_size = fnp->f_offset;
|
|
merge_file_changes(fnp, FALSE); /* /// Added - Ron Cemer */
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
/*
|
|
comments read optimization for large reads: read total clusters in one piece
|
|
|
|
running a program like
|
|
|
|
while (1) {
|
|
read(fd, header, sizeof(header)); // small read
|
|
read(fd, buffer, header.size); // where size is large, up to 63K
|
|
// with average ~32K
|
|
}
|
|
|
|
FreeDOS 2025 is really slow.
|
|
on a P200 with modern 30GB harddisk, doing above for a 14.5 MB file
|
|
|
|
MSDOS 6.22 clustersize 8K ~2.5 sec (accumulates over clusters, reads for 63 sectors seen),
|
|
IBM PCDOS 7.0 8K ~4.3
|
|
IBM PCDOS 7.0 16K ~2.8
|
|
FreeDOS ke2025 ~17.5
|
|
|
|
with the read optimization (ke2025a),
|
|
|
|
clustersize 8K ~6.5 sec
|
|
clustersize 16K ~4.2 sec
|
|
|
|
it was verified with IBM feature tool,
|
|
that the drive read ahead cache (says it) is on. still this huge difference ;-)
|
|
|
|
|
|
it's coded pretty conservative to avoid all special cases,
|
|
so it shouldn't break anything :-)
|
|
|
|
possible further optimization:
|
|
|
|
collect read across clusters (if file is not fragmented).
|
|
MSDOS does this (as readcounts up to 63 sectors where seen)
|
|
specially important for diskettes, where clustersize is 1 sector
|
|
|
|
the same should be done for writes as well
|
|
|
|
the time to compile the complete kernel (on some P200) is
|
|
reduced from 67 to 56 seconds - in an otherwise identical configuration.
|
|
|
|
it's not clear if this improvement shows up elsewhere, but it shouldn't harm either
|
|
|
|
|
|
TE 10/18/01 14:00
|
|
|
|
collect read across clusters (if file is not fragmented) done.
|
|
|
|
seems still to work :-))
|
|
|
|
no large performance gains visible, but should now work _much_
|
|
better for the people, that complain about slow floppy access
|
|
|
|
the
|
|
fnp->f_offset +to_xfer < fnp->f_dir.dir_size && avoid EOF problems
|
|
|
|
condition can probably _carefully_ be dropped
|
|
|
|
|
|
TE 10/18/01 19:00
|
|
|
|
*/
|
|
|
|
/* Read/write block from disk */
|
|
/* checking for valid access was already done by the functions in
|
|
dosfns.c */
|
|
long rwblock(COUNT fd, VOID FAR * buffer, UCOUNT count, int mode)
|
|
{
|
|
REG f_node_ptr fnp;
|
|
REG struct buffer FAR *bp;
|
|
UCOUNT xfr_cnt = 0;
|
|
UCOUNT ret_cnt = 0;
|
|
unsigned secsize;
|
|
unsigned to_xfer = count;
|
|
ULONG currentblock;
|
|
|
|
#if 0 /*DSK_DEBUG*/
|
|
if (bDumpRdWrParms)
|
|
{
|
|
printf("rwblock:fd %02x buffer %04x:%04x count %x\n",
|
|
fd, FP_SEG(buffer), FP_OFF(buffer), count);
|
|
}
|
|
#endif
|
|
/* Translate the fd into an fnode pointer, since all internal */
|
|
/* operations are achieved through fnodes. */
|
|
fnp = xlt_fd(fd);
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
if (fnp == (f_node_ptr) 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (mode==XFR_WRITE)
|
|
{
|
|
fnp->f_dir.dir_attrib |= D_ARCHIVE;
|
|
fnp->f_flags |= F_DMOD; /* mark file as modified */
|
|
fnp->f_flags &= ~F_DDATE; /* set date not valid any more */
|
|
|
|
if (dos_extend(fnp) != SUCCESS)
|
|
{
|
|
save_far_f_node(fnp);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* Test that we are really about to do a data transfer. If the */
|
|
/* count is zero and the mode is XFR_READ, just exit. (Any */
|
|
/* read with a count of zero is a nop). */
|
|
/* */
|
|
/* A write (mode is XFR_WRITE) is a special case. It sets the */
|
|
/* file length to the current length (truncates it). */
|
|
/* */
|
|
/* NOTE: doing this up front saves a lot of headaches later. */
|
|
|
|
if (count == 0)
|
|
{
|
|
/* NOTE: doing this up front made a lot of headaches later :-( TE */
|
|
/* FAT allocation has to be extended if necessary TE */
|
|
/* Now done in dos_extend BO */
|
|
/* remove all the following allocated clusters in shrink_file */
|
|
if (mode == XFR_WRITE)
|
|
{
|
|
fnp->f_dir.dir_size = fnp->f_offset;
|
|
shrink_file(fnp);
|
|
}
|
|
save_far_f_node(fnp);
|
|
return 0;
|
|
}
|
|
|
|
/* The variable secsize will be used later. */
|
|
secsize = fnp->f_dpb->dpb_secsize;
|
|
|
|
/* Adjust the far pointer from user space to supervisor space */
|
|
buffer = adjust_far(buffer);
|
|
|
|
/* Do the data transfer. Use block transfer methods so that we */
|
|
/* can utilize memory management in future DOS-C versions. */
|
|
while (ret_cnt < count)
|
|
{
|
|
unsigned sector, boff;
|
|
|
|
/* Do an EOF test and return whatever was transferred */
|
|
/* but only for regular files. */
|
|
if (mode == XFR_READ && !(fnp->f_flags & F_DDIR) && (fnp->f_offset >= fnp->f_dir.dir_size))
|
|
{
|
|
save_far_f_node(fnp);
|
|
return ret_cnt;
|
|
}
|
|
|
|
/* Position the file to the fnode's pointer position. This is */
|
|
/* done by updating the fnode's cluster, block (sector) and */
|
|
/* byte offset so that read or write becomes a simple data move */
|
|
/* into or out of the block data buffer. */
|
|
|
|
/* The more difficult scenario is the (more common) */
|
|
/* file offset case. Here, we need to take the fnode's */
|
|
/* offset pointer (f_offset) and translate it into a */
|
|
/* relative cluster position, cluster block (sector) */
|
|
/* offset (sector) and byte offset (boff). Once we */
|
|
/* have this information, we need to translate the */
|
|
/* relative cluster position into an absolute cluster */
|
|
/* position (f_cluster). This is unfortunate because it */
|
|
/* requires a linear search through the file's FAT */
|
|
/* entries. It made sense when DOS was originally */
|
|
/* designed as a simple floppy disk operating system */
|
|
/* where the FAT was contained in core, but now */
|
|
/* requires a search through the FAT blocks. */
|
|
/* */
|
|
/* The algorithm in this function takes advantage of */
|
|
/* the blockio block buffering scheme to simplify the */
|
|
/* task. */
|
|
#ifdef DISPLAY_GETBLOCK
|
|
printf("rwblock: ");
|
|
#endif
|
|
if (map_cluster(fnp, mode) != SUCCESS)
|
|
{
|
|
save_far_f_node(fnp);
|
|
return ret_cnt;
|
|
}
|
|
if (mode == XFR_WRITE)
|
|
{
|
|
merge_file_changes(fnp, FALSE); /* /// Added - Ron Cemer */
|
|
}
|
|
|
|
/* Compute the block within the cluster and the offset */
|
|
/* within the block. */
|
|
sector = (UBYTE)(fnp->f_offset / secsize) & fnp->f_dpb->dpb_clsmask;
|
|
boff = (UWORD)(fnp->f_offset % secsize);
|
|
|
|
currentblock = clus2phys(fnp->f_cluster, fnp->f_dpb) + sector;
|
|
|
|
/* see comments above */
|
|
|
|
if (!(fnp->f_flags & F_DDIR) && /* don't experiment with directories yet */
|
|
boff == 0) /* complete sectors only */
|
|
{
|
|
static ULONG startoffset;
|
|
UCOUNT sectors_to_xfer, sectors_wanted;
|
|
|
|
startoffset = fnp->f_offset;
|
|
sectors_wanted = to_xfer;
|
|
|
|
/* avoid EOF problems */
|
|
if (mode == XFR_READ && to_xfer > fnp->f_dir.dir_size - fnp->f_offset)
|
|
sectors_wanted = (UCOUNT)(fnp->f_dir.dir_size - fnp->f_offset);
|
|
|
|
sectors_wanted /= secsize;
|
|
|
|
if (sectors_wanted == 0)
|
|
goto normal_xfer;
|
|
|
|
sectors_to_xfer = fnp->f_dpb->dpb_clsmask + 1 - sector;
|
|
|
|
sectors_to_xfer = min(sectors_to_xfer, sectors_wanted);
|
|
|
|
fnp->f_offset += sectors_to_xfer * secsize;
|
|
|
|
while (sectors_to_xfer < sectors_wanted)
|
|
{
|
|
if (map_cluster(fnp, mode) != SUCCESS)
|
|
break;
|
|
|
|
if (clus2phys(fnp->f_cluster, fnp->f_dpb) !=
|
|
currentblock + sectors_to_xfer)
|
|
break;
|
|
|
|
sectors_to_xfer += fnp->f_dpb->dpb_clsmask + 1;
|
|
|
|
sectors_to_xfer = min(sectors_to_xfer, sectors_wanted);
|
|
|
|
fnp->f_offset = startoffset + sectors_to_xfer * secsize;
|
|
|
|
}
|
|
|
|
xfr_cnt = sectors_to_xfer * secsize;
|
|
|
|
/* avoid caching trouble */
|
|
|
|
DeleteBlockInBufferCache(currentblock,
|
|
currentblock + sectors_to_xfer - 1,
|
|
fnp->f_dpb->dpb_unit, mode);
|
|
|
|
if (dskxfer(fnp->f_dpb->dpb_unit,
|
|
currentblock,
|
|
(VOID FAR *) buffer, sectors_to_xfer,
|
|
mode == XFR_READ ? DSKREAD : DSKWRITE))
|
|
{
|
|
fnp->f_offset = startoffset;
|
|
save_far_f_node(fnp);
|
|
return DE_ACCESS;
|
|
}
|
|
|
|
goto update_pointers;
|
|
}
|
|
|
|
/* normal read: just the old, buffer = sector based read */
|
|
normal_xfer:
|
|
|
|
#ifdef DSK_DEBUG
|
|
printf("r/w %d links; dir offset %d, cluster %d, mode %x\n",
|
|
fnp->f_count, fnp->f_diroff, fnp->f_cluster, mode);
|
|
#endif
|
|
|
|
/* Get the block we need from cache */
|
|
bp = getblock(currentblock
|
|
/*clus2phys(fnp->f_cluster, fnp->f_dpb) + fnp->f_sector */
|
|
, fnp->f_dpb->dpb_unit);
|
|
|
|
#ifdef DISPLAY_GETBLOCK
|
|
printf("DATA (rwblock)\n");
|
|
#endif
|
|
if (bp == NULL) /* (struct buffer *)0 --> DS:0 !! */
|
|
{
|
|
save_far_f_node(fnp);
|
|
return ret_cnt;
|
|
}
|
|
|
|
/* transfer a block */
|
|
/* Transfer size as either a full block size, or the */
|
|
/* requested transfer size, whichever is smaller. */
|
|
/* Then compare to what is left, since we can transfer */
|
|
/* a maximum of what is left. */
|
|
xfr_cnt = min(to_xfer, secsize - boff);
|
|
if (!(fnp->f_flags & F_DDIR) && mode == XFR_READ)
|
|
xfr_cnt = (UWORD) min(xfr_cnt, fnp->f_dir.dir_size - fnp->f_offset);
|
|
|
|
/* transfer a block */
|
|
/* Transfer size as either a full block size, or the */
|
|
/* requested transfer size, whichever is smaller. */
|
|
/* Then compare to what is left, since we can transfer */
|
|
/* a maximum of what is left. */
|
|
if (mode == XFR_WRITE)
|
|
{
|
|
fmemcpy(&bp->b_buffer[boff], buffer, xfr_cnt);
|
|
bp->b_flag |= BFR_DIRTY | BFR_VALID;
|
|
}
|
|
else
|
|
{
|
|
fmemcpy(buffer, &bp->b_buffer[boff], xfr_cnt);
|
|
}
|
|
|
|
/* complete buffer transferred ?
|
|
probably not reused later
|
|
*/
|
|
if (xfr_cnt == sizeof(bp->b_buffer) ||
|
|
(mode == XFR_READ && fnp->f_offset + xfr_cnt == fnp->f_dir.dir_size))
|
|
{
|
|
bp->b_flag |= BFR_UNCACHE;
|
|
}
|
|
|
|
/* update pointers and counters */
|
|
fnp->f_offset += xfr_cnt;
|
|
|
|
update_pointers:
|
|
ret_cnt += xfr_cnt;
|
|
to_xfer -= xfr_cnt;
|
|
buffer = adjust_far((char FAR *)buffer + xfr_cnt);
|
|
if (mode == XFR_WRITE)
|
|
{
|
|
if (fnp->f_offset > fnp->f_dir.dir_size)
|
|
{
|
|
fnp->f_dir.dir_size = fnp->f_offset;
|
|
}
|
|
merge_file_changes(fnp, FALSE); /* /// Added - Ron Cemer */
|
|
}
|
|
}
|
|
save_far_f_node(fnp);
|
|
return ret_cnt;
|
|
}
|
|
|
|
/* Position the file pointer to the desired offset */
|
|
/* Returns a long current offset or a negative error code */
|
|
LONG dos_lseek(COUNT fd, LONG foffset, COUNT origin)
|
|
{
|
|
REG f_node_ptr fnp;
|
|
|
|
/* Translate the fd into a useful pointer */
|
|
|
|
fnp = xlt_fd(fd);
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
|
|
if (fnp == (f_node_ptr) 0)
|
|
return (LONG) DE_INVLDHNDL;
|
|
|
|
/* now do the actual lseek adjustment to the file poitner */
|
|
switch (origin)
|
|
{
|
|
/* offset from beginning of file */
|
|
case 0:
|
|
fnp->f_offset = (ULONG) foffset;
|
|
break;
|
|
|
|
/* offset from current location */
|
|
case 1:
|
|
fnp->f_offset += foffset;
|
|
break;
|
|
|
|
/* offset from eof */
|
|
case 2:
|
|
fnp->f_offset = fnp->f_dir.dir_size + foffset;
|
|
break;
|
|
|
|
/* default to an invalid function */
|
|
default:
|
|
release_near_f_node(fnp);
|
|
return (LONG) DE_INVLDFUNC;
|
|
}
|
|
save_far_f_node(fnp);
|
|
return fnp->f_offset;
|
|
}
|
|
|
|
/* returns the number of unused clusters */
|
|
CLUSTER dos_free(struct dpb FAR * dpbp)
|
|
{
|
|
/* There's an unwritten rule here. All fs */
|
|
/* cluster start at 2 and run to max_cluster+2 */
|
|
REG CLUSTER i;
|
|
REG CLUSTER cnt = 0;
|
|
CLUSTER max_cluster = dpbp->dpb_size;
|
|
|
|
#ifdef WITHFAT32
|
|
if (ISFAT32(dpbp))
|
|
{
|
|
if (dpbp->dpb_xnfreeclst != XUNKNCLSTFREE)
|
|
return dpbp->dpb_xnfreeclst;
|
|
max_cluster = dpbp->dpb_xsize;
|
|
}
|
|
else
|
|
#endif
|
|
if (dpbp->dpb_nfreeclst != UNKNCLSTFREE)
|
|
return dpbp->dpb_nfreeclst;
|
|
|
|
for (i = 2; i <= max_cluster; i++)
|
|
{
|
|
if (next_cluster(dpbp, i) == 0)
|
|
++cnt;
|
|
}
|
|
#ifdef WITHFAT32
|
|
if (ISFAT32(dpbp))
|
|
{
|
|
dpbp->dpb_xnfreeclst = cnt;
|
|
write_fsinfo(dpbp);
|
|
return cnt;
|
|
}
|
|
#endif
|
|
dpbp->dpb_nfreeclst = (UWORD)cnt;
|
|
return cnt;
|
|
}
|
|
|
|
#ifndef IPL
|
|
int dos_cd(char * PathName)
|
|
{
|
|
f_node_ptr fnp;
|
|
struct cds FAR *cdsp = get_cds(PathName[0] - 'A');
|
|
|
|
if ((media_check(cdsp->cdsDpb) < 0))
|
|
return DE_INVLDDRV;
|
|
|
|
/* now test for its existance. If it doesn't, return an error. */
|
|
if ((fnp = dir_open(PathName)) == NULL)
|
|
return DE_PATHNOTFND;
|
|
|
|
/* problem: RBIL table 01643 does not give a FAT32 field for the
|
|
CDS start cluster. But we are not using this field ourselves */
|
|
cdsp->cdsStrtClst = (UWORD)fnp->f_dirstart;
|
|
dir_close(fnp);
|
|
return SUCCESS;
|
|
}
|
|
#endif
|
|
|
|
/* try to allocate a near f_node */
|
|
/* (there are just two of them, in the SDA) */
|
|
|
|
f_node_ptr get_near_f_node(void)
|
|
{
|
|
f_node_ptr fnp = fnode;
|
|
|
|
if (fnp->f_count == 0)
|
|
fnp->f_count++;
|
|
else
|
|
{
|
|
fnp++;
|
|
if (fnp->f_count == 0)
|
|
fnp->f_count++;
|
|
else
|
|
{
|
|
fnp = (f_node_ptr) 0;
|
|
panic("more than two near fnodes requested at the same time!\n");
|
|
}
|
|
}
|
|
return fnp;
|
|
}
|
|
|
|
/* Try to allocate an f_node from the available files array */
|
|
|
|
f_node_ptr get_f_node(void)
|
|
{
|
|
REG int i;
|
|
f_node_ptr fnp = get_near_f_node();
|
|
|
|
if (fnp != (f_node_ptr)0)
|
|
{
|
|
for (i = 0; i < f_nodes_cnt; i++)
|
|
{
|
|
if (f_nodes[i].f_count == 0)
|
|
{
|
|
++f_nodes[i].f_count;
|
|
fnode_fd[fnp - fnode] = i;
|
|
return fnp;
|
|
}
|
|
}
|
|
release_near_f_node(fnp);
|
|
}
|
|
return (f_node_ptr) 0;
|
|
}
|
|
|
|
VOID release_f_node(f_node_ptr fnp)
|
|
{
|
|
struct f_node FAR *fp = &f_nodes[xlt_fnp(fnp)];
|
|
|
|
if (fp->f_count > 0)
|
|
--fp->f_count;
|
|
else
|
|
fp->f_count = 0;
|
|
release_near_f_node(fnp);
|
|
}
|
|
|
|
#ifndef IPL
|
|
VOID dos_setdta(BYTE FAR * newdta)
|
|
{
|
|
dta = newdta;
|
|
}
|
|
|
|
COUNT dos_getfattr_fd(COUNT fd)
|
|
{
|
|
f_node_ptr fnp = xlt_fd(fd);
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
if (fnp == (f_node_ptr) 0)
|
|
return DE_TOOMANY;
|
|
|
|
release_near_f_node(fnp);
|
|
return fnp->f_dir.dir_attrib;
|
|
}
|
|
|
|
COUNT dos_getfattr(BYTE * name)
|
|
{
|
|
COUNT result, fd;
|
|
|
|
fd = (short)dos_open(name, O_RDONLY | O_OPEN, 0);
|
|
if (fd < SUCCESS)
|
|
return fd;
|
|
|
|
result = dos_getfattr_fd(fd);
|
|
dos_close(fd);
|
|
return result;
|
|
}
|
|
|
|
COUNT dos_setfattr(BYTE * name, UWORD attrp)
|
|
{
|
|
COUNT fd;
|
|
f_node_ptr fnp;
|
|
|
|
/* JPP-If user tries to set VOLID or DIR bits, return error */
|
|
if ((attrp & (D_VOLID | D_DIR | 0xC0)) != 0)
|
|
return DE_ACCESS;
|
|
|
|
fd = (short)dos_open(name, O_RDONLY | O_OPEN, 0);
|
|
if (fd < SUCCESS)
|
|
return fd;
|
|
|
|
fnp = xlt_fd(fd);
|
|
|
|
/* Set the attribute from the fnode and return */
|
|
/* clear all attributes but DIR and VOLID */
|
|
fnp->f_dir.dir_attrib &= (D_VOLID | D_DIR); /* JPP */
|
|
|
|
/* set attributes that user requested */
|
|
fnp->f_dir.dir_attrib |= attrp; /* JPP */
|
|
fnp->f_flags |= F_DMOD | F_DDATE;
|
|
save_far_f_node(fnp);
|
|
dos_close(fd);
|
|
return SUCCESS;
|
|
}
|
|
#endif
|
|
|
|
#ifdef WITHFAT32
|
|
VOID bpb_to_dpb(bpb FAR * bpbp, REG struct dpb FAR * dpbp, BOOL extended)
|
|
#else
|
|
VOID bpb_to_dpb(bpb FAR * bpbp, REG struct dpb FAR * dpbp)
|
|
#endif
|
|
{
|
|
ULONG size;
|
|
REG UWORD shftcnt;
|
|
bpb sbpb;
|
|
|
|
fmemcpy(&sbpb, bpbp, sizeof(sbpb));
|
|
for (shftcnt = 0; (sbpb.bpb_nsector >> shftcnt) > 1; shftcnt++)
|
|
;
|
|
dpbp->dpb_shftcnt = shftcnt;
|
|
|
|
dpbp->dpb_mdb = sbpb.bpb_mdesc;
|
|
dpbp->dpb_secsize = sbpb.bpb_nbyte;
|
|
dpbp->dpb_clsmask = sbpb.bpb_nsector - 1;
|
|
dpbp->dpb_fatstrt = sbpb.bpb_nreserved;
|
|
dpbp->dpb_fats = sbpb.bpb_nfat;
|
|
dpbp->dpb_dirents = sbpb.bpb_ndirent;
|
|
size = sbpb.bpb_nsize == 0 ? sbpb.bpb_huge : (ULONG) sbpb.bpb_nsize;
|
|
dpbp->dpb_fatsize = sbpb.bpb_nfsect;
|
|
dpbp->dpb_dirstrt = dpbp->dpb_fatstrt + dpbp->dpb_fats * dpbp->dpb_fatsize;
|
|
dpbp->dpb_data = dpbp->dpb_dirstrt
|
|
+ (dpbp->dpb_dirents + dpbp->dpb_secsize/DIRENT_SIZE - 1) /
|
|
(dpbp->dpb_secsize/DIRENT_SIZE);
|
|
dpbp->dpb_size = (UWORD)((size - dpbp->dpb_data) >> shftcnt) + 1;
|
|
{ /* Make sure the number of FAT sectors is actually enough to hold that */
|
|
/* many clusters. Otherwise back the number of clusters down (LG & AB) */
|
|
unsigned fatsiz;
|
|
ULONG tmp = dpbp->dpb_fatsize * (ULONG)(dpbp->dpb_secsize / 2);/* entries/2 */
|
|
if (tmp >= 0x10000UL)
|
|
goto ckok;
|
|
fatsiz = (unsigned) tmp;
|
|
if (dpbp->dpb_size > FAT_MAGIC) {/* FAT16 */
|
|
if (fatsiz <= FAT_MAGIC) /* FAT12 - let it pass through rather */
|
|
goto ckok; /* than lose data correcting FAT type */
|
|
} else { /* FAT12 */
|
|
if (fatsiz >= 0x4000)
|
|
goto ckok;
|
|
fatsiz = fatsiz * 4 / 3;
|
|
}
|
|
if (dpbp->dpb_size >= fatsiz) /* FAT too short */
|
|
dpbp->dpb_size = fatsiz - 1; /* - 2 reserved entries + 1 */
|
|
ckok:;
|
|
}
|
|
dpbp->dpb_flags = 0;
|
|
dpbp->dpb_cluster = UNKNCLUSTER;
|
|
/* number of free clusters */
|
|
dpbp->dpb_nfreeclst = UNKNCLSTFREE;
|
|
|
|
#ifdef WITHFAT32
|
|
if (extended)
|
|
{
|
|
dpbp->dpb_xfatsize = sbpb.bpb_nfsect == 0 ? sbpb.bpb_xnfsect
|
|
: sbpb.bpb_nfsect;
|
|
dpbp->dpb_xcluster = UNKNCLUSTER;
|
|
dpbp->dpb_xnfreeclst = XUNKNCLSTFREE; /* number of free clusters */
|
|
|
|
dpbp->dpb_xflags = 0;
|
|
dpbp->dpb_xfsinfosec = 0xffff;
|
|
dpbp->dpb_xbackupsec = 0xffff;
|
|
dpbp->dpb_xrootclst = 0;
|
|
dpbp->dpb_xdata = dpbp->dpb_data;
|
|
dpbp->dpb_xsize = dpbp->dpb_size;
|
|
|
|
if (ISFAT32(dpbp))
|
|
{
|
|
dpbp->dpb_xflags = sbpb.bpb_xflags;
|
|
dpbp->dpb_xfsinfosec = sbpb.bpb_xfsinfosec;
|
|
dpbp->dpb_xbackupsec = sbpb.bpb_xbackupsec;
|
|
dpbp->dpb_dirents = 0;
|
|
dpbp->dpb_dirstrt = 0xffff;
|
|
dpbp->dpb_size = 0;
|
|
dpbp->dpb_xdata =
|
|
dpbp->dpb_fatstrt + dpbp->dpb_fats * dpbp->dpb_xfatsize;
|
|
dpbp->dpb_xsize = ((size - dpbp->dpb_xdata) >> shftcnt) + 1;
|
|
dpbp->dpb_xrootclst = sbpb.bpb_xrootclst;
|
|
read_fsinfo(dpbp);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
STATIC int rqblockio(unsigned char command, struct dpb FAR * dpbp)
|
|
{
|
|
retry:
|
|
MediaReqHdr.r_length = sizeof(request);
|
|
MediaReqHdr.r_unit = dpbp->dpb_subunit;
|
|
MediaReqHdr.r_command = command;
|
|
MediaReqHdr.r_mcmdesc = dpbp->dpb_mdb;
|
|
MediaReqHdr.r_status = 0;
|
|
execrh((request FAR *) & MediaReqHdr, dpbp->dpb_device);
|
|
if ((MediaReqHdr.r_status & S_ERROR) || !(MediaReqHdr.r_status & S_DONE))
|
|
{
|
|
FOREVER
|
|
{
|
|
switch (block_error(&MediaReqHdr, dpbp->dpb_unit, dpbp->dpb_device, 0))
|
|
{
|
|
case ABORT:
|
|
case FAIL:
|
|
return DE_INVLDDRV;
|
|
|
|
case RETRY:
|
|
goto retry;
|
|
|
|
case CONTINUE:
|
|
return SUCCESS;
|
|
}
|
|
}
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
COUNT media_check(REG struct dpb FAR * dpbp)
|
|
{
|
|
int ret;
|
|
if (dpbp == NULL)
|
|
return DE_INVLDDRV;
|
|
|
|
/* First test if anyone has changed the removable media */
|
|
ret = rqblockio(C_MEDIACHK, dpbp);
|
|
if (ret < SUCCESS)
|
|
return ret;
|
|
|
|
switch (MediaReqHdr.r_mcretcode | dpbp->dpb_flags)
|
|
{
|
|
case M_NOT_CHANGED:
|
|
/* It was definitely not changed, so ignore it */
|
|
return SUCCESS;
|
|
|
|
/* If it is forced or the media may have changed, */
|
|
/* rebuild the bpb */
|
|
case M_DONT_KNOW:
|
|
/* hazard: no error checking! */
|
|
flush_buffers(dpbp->dpb_unit);
|
|
|
|
/* If it definitely changed, don't know (falls through) */
|
|
/* or has been changed, rebuild the bpb. */
|
|
case M_CHANGED:
|
|
default:
|
|
setinvld(dpbp->dpb_unit);
|
|
ret = rqblockio(C_BLDBPB, dpbp);
|
|
if (ret < SUCCESS)
|
|
return ret;
|
|
#ifdef WITHFAT32
|
|
/* extend dpb only for internal or FAT32 devices */
|
|
bpb_to_dpb(MediaReqHdr.r_bpptr, dpbp,
|
|
ISFAT32(dpbp) || FP_SEG(dpbp) == FP_SEG(&os_major));
|
|
#else
|
|
bpb_to_dpb(MediaReqHdr.r_bpptr, dpbp);
|
|
#endif
|
|
return SUCCESS;
|
|
}
|
|
}
|
|
|
|
/* translate the f_node pointer into an fd */
|
|
COUNT xlt_fnp(f_node_ptr fnp)
|
|
{
|
|
return fnode_fd[fnp - fnode];
|
|
}
|
|
|
|
/* allocate a near fnode and copy the far fd fnode to it */
|
|
f_node_ptr xlt_fd(int fd)
|
|
{
|
|
f_node_ptr fnp = (f_node_ptr) 0;
|
|
|
|
/* If the fd was invalid because it was out of range or the */
|
|
/* requested file was not open, tell the caller and exit */
|
|
/* note: an invalid fd is indicated by a 0 return */
|
|
if (fd < f_nodes_cnt)
|
|
{
|
|
fnp = get_near_f_node();
|
|
if (fnp != (f_node_ptr)0)
|
|
{
|
|
fmemcpy(fnp, &f_nodes[fd], sizeof(*fnp));
|
|
if (fnp->f_count <= 0)
|
|
{
|
|
release_near_f_node(fnp);
|
|
fnp = (f_node_ptr) 0;
|
|
} else
|
|
fnode_fd[fnp - fnode] = fd;
|
|
}
|
|
}
|
|
return fnp;
|
|
}
|
|
|
|
/* copy a near fnode to the corresponding far one and release it */
|
|
STATIC void save_far_f_node(f_node_ptr fnp)
|
|
{
|
|
fmemcpy(&f_nodes[xlt_fnp(fnp)], fnp, sizeof(*fnp));
|
|
release_near_f_node(fnp);
|
|
}
|
|
|
|
/* TE
|
|
if the current filesize in FAT is larger then the dir_size
|
|
it's truncated here.
|
|
the BUG was:
|
|
copy COMMAND.COM xxx
|
|
echo >xxx
|
|
|
|
then, the dirsize of xxx was set to ~20, but the allocated
|
|
FAT entries not returned.
|
|
this code corrects this
|
|
|
|
Unfortunately, this code _nearly_ works, but fails one of the
|
|
Apps tested (VB ISAM); BO: confirmation???
|
|
*/
|
|
|
|
STATIC VOID shrink_file(f_node_ptr fnp)
|
|
{
|
|
|
|
ULONG lastoffset = fnp->f_offset; /* has to be saved */
|
|
CLUSTER next, st;
|
|
struct dpb FAR *dpbp = fnp->f_dpb;
|
|
|
|
fnp->f_offset = fnp->f_dir.dir_size; /* end of file */
|
|
|
|
if (fnp->f_offset)
|
|
fnp->f_offset--; /* last existing cluster */
|
|
|
|
if (map_cluster(fnp, XFR_READ) != SUCCESS) /* error, don't truncate */
|
|
goto done;
|
|
|
|
st = fnp->f_cluster;
|
|
|
|
next = next_cluster(dpbp, st);
|
|
|
|
if (next == 1) /* error */
|
|
goto done;
|
|
|
|
/* Loop from start until either a FREE entry is */
|
|
/* encountered (due to a damaged file system) or the */
|
|
/* last cluster is encountered. */
|
|
/* zap the FAT pointed to */
|
|
|
|
if (fnp->f_dir.dir_size == 0)
|
|
{
|
|
fnp->f_cluster = FREE;
|
|
setdstart(dpbp, &fnp->f_dir, FREE);
|
|
link_fat(dpbp, st, FREE);
|
|
}
|
|
else
|
|
{
|
|
if (next == LONG_LAST_CLUSTER) /* nothing to do */
|
|
goto done;
|
|
link_fat(dpbp, st, LONG_LAST_CLUSTER);
|
|
}
|
|
|
|
wipe_out_clusters(dpbp, next);
|
|
|
|
done:
|
|
fnp->f_offset = lastoffset; /* has to be restored */
|
|
|
|
}
|
|
|
|
/*
|
|
* TE 12 jun 2001 bugs corrected
|
|
* handles disk full (in a incompatible way :-( )
|
|
* allows use of last cluster
|
|
* prevents mkdir, if disk is full (was creating crosslinked dirs)
|
|
* bugs detected, but NOT corrected
|
|
* on disk full, MSDOS will NOT write any byte, simply return SUCCESS, 0 bytes
|
|
* FreeDOS will write all possible bytes, then close file(BUG)
|
|
*
|
|
* the dos_mkdir/extenddir (with getblock() instead of getblockOver) was a real
|
|
* performance killer on large drives. (~0.5 sec /dos_mkdir) TE
|
|
*/
|