Eliminate f_back field.

extend() now uses f_cluster instead of f_back, while map_cluster makes
sure it's not set to LONG_LAST_CLUSTER (but to the cluster before it)
when calling extend().
extend() and first_fat() now return the new cluster number or
LONG_LAST_CLUSTER, just like find_fat_free()
Hopefully I didn't break anything... Initial testing was successful.


git-svn-id: https://svn.code.sf.net/p/freedos/svn/kernel/trunk@816 6ac86273-5f31-0410-b378-82cca8765d1b
This commit is contained in:
Bart Oldeman 2004-03-22 22:33:05 +00:00
parent 7e91d9cf9d
commit ce290a033f
3 changed files with 45 additions and 46 deletions

View File

@ -55,7 +55,6 @@ struct f_node {
struct dpb FAR *f_dpb; /* the block device for file */ struct dpb FAR *f_dpb; /* the block device for file */
ULONG f_offset; /* byte offset for next op */ ULONG f_offset; /* byte offset for next op */
CLUSTER f_back; /* the cluster we were at */
CLUSTER f_cluster_offset; /* relative cluster number within file */ CLUSTER f_cluster_offset; /* relative cluster number within file */
CLUSTER f_cluster; /* the cluster we are at */ CLUSTER f_cluster; /* the cluster we are at */
}; };

View File

@ -332,7 +332,6 @@ BOOL dir_write(REG f_node_ptr fnp)
/* Do a "seek" to the directory position */ /* Do a "seek" to the directory position */
/* and convert the fnode to a directory fnode. */ /* and convert the fnode to a directory fnode. */
fnp->f_offset = fnp->f_diroff * (ULONG)DIRENT_SIZE; fnp->f_offset = fnp->f_diroff * (ULONG)DIRENT_SIZE;
fnp->f_back = LONG_LAST_CLUSTER;
fnp->f_cluster = fnp->f_dirstart; fnp->f_cluster = fnp->f_dirstart;
fnp->f_cluster_offset = 0; fnp->f_cluster_offset = 0;

View File

@ -50,10 +50,9 @@ STATIC void copy_file_changes(f_node_ptr src, f_node_ptr dst);
BOOL find_free(f_node_ptr); BOOL find_free(f_node_ptr);
CLUSTER find_fat_free(f_node_ptr); CLUSTER find_fat_free(f_node_ptr);
VOID wipe_out(f_node_ptr); VOID wipe_out(f_node_ptr);
BOOL last_link(f_node_ptr); CLUSTER extend(f_node_ptr);
BOOL extend(f_node_ptr);
COUNT extend_dir(f_node_ptr); COUNT extend_dir(f_node_ptr);
BOOL first_fat(f_node_ptr); CLUSTER first_fat(f_node_ptr);
COUNT map_cluster(f_node_ptr, COUNT); COUNT map_cluster(f_node_ptr, COUNT);
STATIC VOID shrink_file(f_node_ptr fnp); STATIC VOID shrink_file(f_node_ptr fnp);
@ -262,7 +261,6 @@ long dos_open(char *path, unsigned flags, unsigned attrib)
/* Now change to file */ /* Now change to file */
fnp->f_offset = 0l; fnp->f_offset = 0l;
fnp->f_back = LONG_LAST_CLUSTER;
if (status != S_OPENED) if (status != S_OPENED)
{ {
fnp->f_cluster = FREE; fnp->f_cluster = FREE;
@ -1141,7 +1139,6 @@ COUNT dos_mkdir(BYTE * dir)
/* Set the fnode to the desired mode */ /* Set the fnode to the desired mode */
fnp->f_mode = WRONLY; fnp->f_mode = WRONLY;
fnp->f_back = LONG_LAST_CLUSTER;
init_direntry(&fnp->f_dir, D_DIR, free_fat); init_direntry(&fnp->f_dir, D_DIR, free_fat);
@ -1152,7 +1149,6 @@ COUNT dos_mkdir(BYTE * dir)
fnp->f_offset = 0l; fnp->f_offset = 0l;
/* Mark the cluster in the FAT as used */ /* Mark the cluster in the FAT as used */
fnp->f_cluster = free_fat;
dpbp = fnp->f_dpb; dpbp = fnp->f_dpb;
link_fat(dpbp, free_fat, LONG_LAST_CLUSTER); link_fat(dpbp, free_fat, LONG_LAST_CLUSTER);
@ -1227,12 +1223,7 @@ COUNT dos_mkdir(BYTE * dir)
return SUCCESS; return SUCCESS;
} }
BOOL last_link(f_node_ptr fnp) STATIC CLUSTER extend(f_node_ptr fnp)
{
return (fnp->f_cluster == LONG_LAST_CLUSTER);
}
STATIC BOOL extend(f_node_ptr fnp)
{ {
CLUSTER free_fat; CLUSTER free_fat;
@ -1245,24 +1236,24 @@ STATIC BOOL extend(f_node_ptr fnp)
/* No empty clusters, disk is FULL! Translate into a useful */ /* No empty clusters, disk is FULL! Translate into a useful */
/* error message. */ /* error message. */
if (free_fat == LONG_LAST_CLUSTER) if (free_fat == LONG_LAST_CLUSTER)
return FALSE; return free_fat;
/* Now that we've found a free FAT entry, mark it as the last */ /* Now that we've found a free FAT entry, mark it as the last */
/* entry and save. */ /* entry and save. */
link_fat(fnp->f_dpb, fnp->f_back, free_fat); link_fat(fnp->f_dpb, fnp->f_cluster, free_fat);
fnp->f_cluster = free_fat;
link_fat(fnp->f_dpb, free_fat, LONG_LAST_CLUSTER); link_fat(fnp->f_dpb, free_fat, LONG_LAST_CLUSTER);
/* Mark the directory so that the entry is updated */ /* Mark the directory so that the entry is updated */
fnp->f_flags.f_dmod = TRUE; fnp->f_flags.f_dmod = TRUE;
return TRUE; return free_fat;
} }
STATIC COUNT extend_dir(f_node_ptr fnp) STATIC COUNT extend_dir(f_node_ptr fnp)
{ {
REG COUNT idx; REG COUNT idx;
if (!extend(fnp)) CLUSTER cluster = extend(fnp);
if (cluster == LONG_LAST_CLUSTER)
{ {
dir_close(fnp); dir_close(fnp);
return DE_HNDLDSKFULL; return DE_HNDLDSKFULL;
@ -1274,7 +1265,7 @@ STATIC COUNT extend_dir(f_node_ptr fnp)
REG struct buffer FAR *bp; REG struct buffer FAR *bp;
/* as we are overwriting it completely, don't read first */ /* as we are overwriting it completely, don't read first */
bp = getblockOver(clus2phys(fnp->f_cluster, fnp->f_dpb) + idx, bp = getblockOver(clus2phys(cluster, fnp->f_dpb) + idx,
fnp->f_dpb->dpb_unit); fnp->f_dpb->dpb_unit);
#ifdef DISPLAY_GETBLOCK #ifdef DISPLAY_GETBLOCK
printf("DIR (extend_dir)\n"); printf("DIR (extend_dir)\n");
@ -1306,7 +1297,7 @@ STATIC COUNT extend_dir(f_node_ptr fnp)
} }
/* JPP: finds the next free cluster in the FAT */ /* JPP: finds the next free cluster in the FAT */
STATIC BOOL first_fat(f_node_ptr fnp) STATIC CLUSTER first_fat(f_node_ptr fnp)
{ {
CLUSTER free_fat; CLUSTER free_fat;
@ -1316,19 +1307,18 @@ STATIC BOOL first_fat(f_node_ptr fnp)
/* No empty clusters, disk is FULL! Translate into a useful */ /* No empty clusters, disk is FULL! Translate into a useful */
/* error message. */ /* error message. */
if (free_fat == LONG_LAST_CLUSTER) if (free_fat == LONG_LAST_CLUSTER)
return FALSE; return free_fat;
/* Now that we've found a free FAT entry, mark it as the last */ /* Now that we've found a free FAT entry, mark it as the last */
/* entry and save it. */ /* entry and save it. */
/* BUG!! this caused wrong allocation, if file was created, /* BUG!! this caused wrong allocation, if file was created,
then seeked, then written */ then seeked, then written */
fnp->f_cluster = free_fat;
setdstart(fnp->f_dpb, &fnp->f_dir, free_fat); setdstart(fnp->f_dpb, &fnp->f_dir, free_fat);
link_fat(fnp->f_dpb, free_fat, LONG_LAST_CLUSTER); link_fat(fnp->f_dpb, free_fat, LONG_LAST_CLUSTER);
/* Mark the directory so that the entry is updated */ /* Mark the directory so that the entry is updated */
fnp->f_flags.f_dmod = TRUE; fnp->f_flags.f_dmod = TRUE;
return TRUE; return free_fat;
} }
/* Description. /* Description.
@ -1352,8 +1342,7 @@ STATIC BOOL first_fat(f_node_ptr fnp)
COUNT map_cluster(REG f_node_ptr fnp, COUNT mode) COUNT map_cluster(REG f_node_ptr fnp, COUNT mode)
{ {
CLUSTER relcluster = (CLUSTER)((fnp->f_offset / fnp->f_dpb->dpb_secsize) >> CLUSTER relcluster, cluster;
fnp->f_dpb->dpb_shftcnt);
#ifdef DISPLAY_GETBLOCK #ifdef DISPLAY_GETBLOCK
printf("map_cluster: current %lu, offset %lu, diff=%lu ", printf("map_cluster: current %lu, offset %lu, diff=%lu ",
@ -1366,12 +1355,21 @@ COUNT map_cluster(REG f_node_ptr fnp, COUNT mode)
if ((mode == XFR_WRITE) && checkdstart(fnp->f_dpb, &fnp->f_dir, FREE)) if ((mode == XFR_WRITE) && checkdstart(fnp->f_dpb, &fnp->f_dir, FREE))
{ {
/* If there are no more free fat entries, then we are full! */ /* If there are no more free fat entries, then we are full! */
if (!first_fat(fnp)) cluster = first_fat(fnp);
if (cluster == LONG_LAST_CLUSTER)
{ {
return DE_HNDLDSKFULL; return DE_HNDLDSKFULL;
} }
fnp->f_cluster = cluster;
} }
/* If this is a read but the file still has zero bytes return */
/* immediately.... */
if ((mode == XFR_READ) && (fnp->f_cluster == FREE))
return DE_SEEK;
relcluster = (CLUSTER)((fnp->f_offset / fnp->f_dpb->dpb_secsize) >>
fnp->f_dpb->dpb_shftcnt);
if (relcluster < fnp->f_cluster_offset) if (relcluster < fnp->f_cluster_offset)
{ {
/* Set internal index and cluster size. */ /* Set internal index and cluster size. */
@ -1387,27 +1385,30 @@ COUNT map_cluster(REG f_node_ptr fnp, COUNT mode)
/* up to the relative cluster position where the index falls */ /* up to the relative cluster position where the index falls */
/* within the cluster. */ /* within the cluster. */
FOREVER 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, */ /* If this is a read and the next is a LAST_CLUSTER, */
/* then we are going to read past EOF, return zero read */ /* then we are going to read past EOF, return zero read */
if ((mode == XFR_READ) && (last_link(fnp) || fnp->f_cluster == FREE)) /* or expand the list if we're going to write and have run into */
return DE_SEEK;
/* expand the list if we're going to write and have run into */
/* the last cluster marker. */ /* the last cluster marker. */
if ((mode == XFR_WRITE) && last_link(fnp) && !extend(fnp)) if (cluster == LONG_LAST_CLUSTER)
return DE_HNDLDSKFULL; {
if (mode == XFR_READ)
if (fnp->f_cluster_offset == relcluster)
break;
fnp->f_back = fnp->f_cluster;
/* get next cluster in the chain */
fnp->f_cluster = next_cluster(fnp->f_dpb, fnp->f_cluster);
fnp->f_cluster_offset++;
if (fnp->f_cluster == 1)
return DE_SEEK; 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 #ifdef DISPLAY_GETBLOCK