diff options
author | mckusick <mckusick@FreeBSD.org> | 2000-11-14 09:00:25 +0000 |
---|---|---|
committer | mckusick <mckusick@FreeBSD.org> | 2000-11-14 09:00:25 +0000 |
commit | 56cb617a9d071d8ad21ab908ea9224e718dae618 (patch) | |
tree | e98b9c55b484605511e513813f8c63cfae8575ae /sys/ufs | |
parent | 363c7c2287703c2a9abc892764f54fefc22b28ad (diff) | |
download | FreeBSD-src-56cb617a9d071d8ad21ab908ea9224e718dae618.zip FreeBSD-src-56cb617a9d071d8ad21ab908ea9224e718dae618.tar.gz |
When deleting a file, the ordering of events imposed by soft updates
is to first write the deleted directory entry to disk, second write
the zero'ed inode to disk, and finally to release the freed blocks
and the inode back to the cylinder-group map. As this ordering
requires two disk writes to occur which are normally spaced about
30 seconds apart (except when memory is under duress), it takes
about a minute from the time that a file is deleted until its inode
and data blocks show up in the cylinder-group map for reallocation.
If a file has had only a brief lifetime (less than 30 seconds from
creation to deletion), neither its inode nor its directory entry
may have been written to disk. If its directory entry has not been
written to disk, then we need not wait for that directory block to
be written as the on-disk directory block does not reference the
inode. Similarly, if the allocated inode has never been written to
disk, we do not have to wait for it to be written back either as
its on-disk representation is still zero'ed out. Thus, in the case
of a short lived file, we can simply release the blocks and inode
to the cylinder-group map immediately. As the inode and its blocks
are released immediately, they are immediately available for other
uses. If they are not released for a minute, then other inodes and
blocks must be allocated for short lived files, cluttering up the
vnode and buffer caches. The previous code was a bit too aggressive
in trying to release the blocks and inode back to the cylinder-group
map resulting in their being made available when in fact the inode
on disk had not yet been zero'ed. This patch takes a more conservative
approach to doing the release which avoids doing the release prematurely.
Diffstat (limited to 'sys/ufs')
-rw-r--r-- | sys/ufs/ffs/ffs_softdep.c | 33 |
1 files changed, 18 insertions, 15 deletions
diff --git a/sys/ufs/ffs/ffs_softdep.c b/sys/ufs/ffs/ffs_softdep.c index a1525dc..98ad959 100644 --- a/sys/ufs/ffs/ffs_softdep.c +++ b/sys/ufs/ffs/ffs_softdep.c @@ -1662,6 +1662,16 @@ softdep_setup_freeblocks(ip, length) if ((inodedep->id_state & IOSTARTED) != 0) panic("softdep_setup_freeblocks: inode busy"); /* + * Add the freeblks structure to the list of operations that + * must await the zero'ed inode being written to disk. If we + * still have a bitmap dependency (delay == 0), then the inode + * has never been written to disk, so we can process the + * freeblks below once we have deleted the dependencies. + */ + delay = (inodedep->id_state & DEPCOMPLETE); + if (delay) + WORKLIST_INSERT(&inodedep->id_bufwait, &freeblks->fb_list); + /* * Because the file length has been truncated to zero, any * pending block allocation dependency structures associated * with this inode are obsolete and can simply be de-allocated. @@ -1694,23 +1704,16 @@ softdep_setup_freeblocks(ip, length) brelse(bp); ACQUIRE_LOCK(&lk); } + if (inodedep_lookup(fs, ip->i_number, 0, &inodedep) != 0) + (void) free_inodedep(inodedep); + FREE_LOCK(&lk); /* - * Add the freeblks structure to the list of operations that - * must await the zero'ed inode being written to disk. If we - * still have a bitmap dependency, then the inode has never been - * written to disk, so we can process the freeblks immediately. - * If the inodedep does not exist, then the zero'ed inode has - * been written and we can also proceed. + * If the inode has never been written to disk (delay == 0), + * then we can process the freeblks now that we have deleted + * the dependencies. */ - if (inodedep_lookup(fs, ip->i_number, 0, &inodedep) == 0 || - free_inodedep(inodedep) || - (inodedep->id_state & DEPCOMPLETE) == 0) { - FREE_LOCK(&lk); + if (!delay) handle_workitem_freeblocks(freeblks); - } else { - WORKLIST_INSERT(&inodedep->id_bufwait, &freeblks->fb_list); - FREE_LOCK(&lk); - } } /* @@ -2017,7 +2020,7 @@ handle_workitem_freeblocks(freeblks) #ifdef DIAGNOSTIC if (freeblks->fb_chkcnt != blocksreleased) - panic("handle_workitem_freeblocks: block count"); + printf("handle_workitem_freeblocks: block count"); if (allerror) softdep_error("handle_workitem_freeblks", allerror); #endif /* DIAGNOSTIC */ |