summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--ufs/inode.c126
1 files changed, 69 insertions, 57 deletions
diff --git a/ufs/inode.c b/ufs/inode.c
index 67037450..d5c17e99 100644
--- a/ufs/inode.c
+++ b/ufs/inode.c
@@ -72,14 +72,7 @@ iget (ino_t inum, struct node **npp)
dn->number = inum;
dn->dirents = 0;
- rwlock_init (&dn->dinlock);
- rwlock_init (&dn->sinlock);
- rwlock_init (&dn->datalock);
- dn->dinloc = 0;
- dn->sinloc = 0;
- dn->dinloclen = 0;
- dn->sinloclen = 0;
- dn->sininfo = 0;
+ rwlock_init (&dn->allocptrlock);
dn->fileinfo = 0;
np = diskfs_make_node (dn);
@@ -92,6 +85,7 @@ iget (ino_t inum, struct node **npp)
spin_unlock (&diskfs_node_refcnt_lock);
err = read_disknode (np);
+
if (lblkno (sblock, np->dn_stat.st_size) < NDADDR)
np->allocsize = fragroundup (sblock, np->dn_stat.st_size);
else
@@ -146,31 +140,63 @@ diskfs_node_norefs (struct node *np)
np->dn->hnext->dn->hprevp = np->dn->hprevp;
if (np->dn->dirents)
free (np->dn->dirents);
- if (np->dn->sininfo || np->dn->fileinfo || np->dn->dinloc
- || np->dn->sinloc || np->dn->dinloclen || np->dn->sinloclen)
- {
- printf ("I=%d\n", np->dn->number);
- printf ("Hard %d\tSoft %d\n", np->references, np->light_references);
- fflush (stdout);
- }
- assert (!np->dn->sininfo);
assert (!np->dn->fileinfo);
- assert (!np->dn->dinloc);
- assert (!np->dn->sinloc);
- assert (!np->dn->dinloclen);
- assert (!np->dn->sinloclen);
free (np->dn);
free (np);
}
-/* The last hard referencs to a node has gone away; arrange to have
+/* The last hard reference to a node has gone away; arrange to have
all the weak references dropped that can be. */
void
-diskfs_lost_hardrefs (struct node *np)
+diskfs_try_dropping_softrefs (struct node *np)
{
drop_pager_softrefs (np);
}
+/* The last hard reference to a node has gone away. */
+void
+diskfs_lost_hardrefs (struct node *np)
+{
+ /* Check and see if there is a pager which has only
+ one reference (ours). If so, then drop that reference,
+ breaking the cycle. The complexity in this routine
+ is all due to this cycle. */
+
+ if (np->dn->fileinfo)
+ {
+ spin_lock (&_libports_portrefcntlock);
+ if (np->fileinfo->p->pi.refcnt == 1)
+ {
+ struct pager *p;
+
+ /* The only way to get a new reference to the pager
+ in this state is to call diskfs_get_filemap; this
+ can't happen as long as we hold NP locked. So
+ we can safely unlock _libports_portrefcntlock for
+ the following call. */
+ spin_unlock (&_libports_portrefcntlock);
+
+ /* Right now the node is locked with no hard refs;
+ this is an anomolous situation. Before messing with
+ the reference count on the file pager, we have to
+ give ourselves a reference back so that we are really
+ allowed to hold the lock. Then we can do the
+ unreference. */
+ p = np->fileinfo->p;
+ np->fileinfo = 0;
+ diskfs_nref (np);
+ pager_unreference (p);
+
+ assert (np->references == 1 && np->light_references == 0);
+
+ /* This will do the real deallocate. Whew. */
+ diskfs_nput (np);
+ }
+ else
+ spin_unlock (&_libports_portrefcntlock);
+ }
+}
+
/* A new hard reference to a node has been created; it's now OK to
have unused weak references. */
void
@@ -184,7 +210,7 @@ static error_t
read_disknode (struct node *np)
{
struct stat *st = &np->dn_stat;
- struct dinode *di = &dinodes[np->dn->number];
+ struct dinode *di = dino (np->dn->number);
error_t err;
volatile long long pid = getpid ();
@@ -192,6 +218,8 @@ read_disknode (struct node *np)
if (err)
return err;
+ np->istranslated = !! di->translator;
+
st->st_fstype = FSTYPE_UFS;
st->st_fsid = pid;
st->st_ino = np->dn->number;
@@ -241,7 +269,7 @@ static void
write_node (struct node *np)
{
struct stat *st = &np->dn_stat;
- struct dinode *di = &dinodes[np->dn->number];
+ struct dinode *di = dino (np->dn->number);
error_t err;
assert (!np->dn_set_ctime && !np->dn_set_atime && !np->dn_set_mtime);
@@ -329,7 +357,7 @@ create_symlink_hook (struct node *np, char *target)
if (err)
return err;
- bcopy (target, dinodes[np->dn->number].di_shortlink, len);
+ bcopy (target, (dino (np->dn->number))->di_shortlink, len);
np->dn_stat.st_size = len;
np->dn_set_ctime = 1;
np->dn_set_mtime = 1;
@@ -356,7 +384,7 @@ read_symlink_hook (struct node *np,
if (err)
return err;
- bcopy (dinodes[np->dn->number].di_shortlink, buf, np->dn_stat.st_size);
+ bcopy ((dino (np->dn->number))->di_shortlink, buf, np->dn_stat.st_size);
np->dn_set_atime = 1;
diskfs_end_catch_exception ();
@@ -387,10 +415,9 @@ void
diskfs_write_disknode (struct node *np, int wait)
{
write_node (np);
- sync_dinode (np, wait);
+ sync_dinode (np->dn->number, wait);
}
-
/* Implement the diskfs_set_statfs callback from the diskfs library;
see <hurd/diskfs.h> for the interface description. */
error_t
@@ -430,7 +457,7 @@ diskfs_set_translator (struct node *np, char *name, u_int namelen,
if (err)
return err;
- blkno = dinodes[np->dn->number].di_trans;
+ blkno = (dino (np->dn->number))->di_trans;
if (namelen && !blkno)
{
@@ -441,27 +468,31 @@ diskfs_set_translator (struct node *np, char *name, u_int namelen,
diskfs_end_catch_exception ();
return err;
}
- dinodes[np->dn->number].di_trans = blkno;
+ (dino (np->dn->number))->di_trans = blkno;
np->dn_set_ctime = 1;
}
else if (!namelen && blkno)
{
/* Clear block for translator going away. */
ffs_blkfree (np, blkno, sblock->fs_bsize);
- dinodes[np->dn->number].di_trans = 0;
+ (dino (np->dn->number))->di_trans = 0;
+ np->istranslated = 0;
np->dn_set_ctime = 1;
}
- diskfs_end_catch_exception ();
-
if (namelen)
{
bcopy (&namelen, buf, sizeof (u_int));
bcopy (name, buf + sizeof (u_int), namelen);
- err = dev_write_sync (fsbtodb (sblock, blkno),
- (vm_address_t)buf, sblock->fs_bsize);
+
+ bcopy (buf, fsaddr (sblock, blkno), sblock->fs_bsize);
+ sync_disk_blocks (blkno, sblock->fs_bsize, 1);
+
+ np->istranslated = 1;
np->dn_set_ctime = 1;
}
+
+ diskfs_end_catch_exception ();
return err;
}
@@ -478,15 +509,11 @@ diskfs_get_translator (struct node *np, char **namep, u_int *namelen)
err = diskfs_catch_exception ();
if (err)
return err;
- blkno = dinodes[np->dn->number].di_trans;
- diskfs_end_catch_exception ();
+ blkno = (dino (np->dn->number))->di_trans;
assert (blkno);
-
- err = dev_read_sync (fsbtodb (sblock, blkno), (vm_address_t *)&buf,
- sblock->fs_bsize);
- if (err)
- return err;
+ bcopy (fsaddr (sblock, blkno), buf, sblock->fs_bsize);
+ diskfs_end_catch_exception ();
datalen = *(u_int *)buf;
if (datalen > *namelen)
@@ -496,21 +523,6 @@ diskfs_get_translator (struct node *np, char **namep, u_int *namelen)
return 0;
}
-/* Implement the diskfs_node_translated callback from the diskfs library.
- See <hurd/diskfs.h> for the interface description. */
-int
-diskfs_node_translated (struct node *np)
-{
- int ret;
-
- if (diskfs_catch_exception ())
- return 0;
-
- ret = !! dinodes[np->dn->number].di_trans;
- diskfs_end_catch_exception ();
- return ret;
-}
-
/* Called when all hard ports have gone away. */
void
diskfs_shutdown_soft_ports ()