diff options
-rw-r--r-- | ufs/sizes.c | 92 |
1 files changed, 55 insertions, 37 deletions
diff --git a/ufs/sizes.c b/ufs/sizes.c index 4237ea56..ad14f5cc 100644 --- a/ufs/sizes.c +++ b/ufs/sizes.c @@ -373,8 +373,9 @@ offer_data (struct node *np, /* Logical block LBN of node NP has been extended with ffs_realloccg. It used to be allocated at OLD_PBN and is now at NEW_PBN. The old size was OLD_SIZE; it is now NEW_SIZE bytes long. Arrange for the data - on disk to be kept consistent, and free the old block if it has moved. */ -void + on disk to be kept consistent, and free the old block if it has moved. + Return one iff we've actually moved data around on disk. */ +int block_extended (struct node *np, daddr_t lbn, daddr_t old_pbn, @@ -393,42 +394,54 @@ block_extended (struct node *np, if (old_pbn != new_pbn) { - /* Fetch the old data for the part that has moved and offer it - to the kernel, to make sure it's paged in before - we deallocate the old block. */ - for (off = 0; off < round_page (old_size); off += vm_page_size) - { - - /* There's *got* to be a better way to do this... */ - - /* Force it out to disk */ - assert (np->dn->fileinfo); - pager_sync_some (np->dn->fileinfo->p, lbn * sblock->fs_bsize + off, - vm_page_size, 1); - - /* Read it back in */ - diskfs_device_read_sync (fsbtodb (sblock, old_pbn) + off / DEV_BSIZE, - (void *) &buf, vm_page_size); - /* If this page is the last one, then zero the excess first */ - if (off + vm_page_size > old_size) - bzero ((void *)(buf + old_size - off), - vm_page_size - (old_size - off)); - - /* And make sure it's in core. */ - offer_data (np, lbn * sblock->fs_bsize + off, vm_page_size, buf); - - /* And now, make sure it's on disk. (Why? Because a previous - write of this file, maybe even one from before we started - running may have been fsynced. We can't cause data already - on disk to be lossy at any time in the future while we move it. */ - pager_sync_some (np->dn->fileinfo->p, lbn * sblock->fs_bsize + off, - vm_page_size, 1); - } + vm_object_t mapobj; + error_t err; + vm_address_t mapaddr; + volatile int *pokeaddr; + + /* Map in this part of the file */ + mapobj = diskfs_get_filemap (np, VM_PROT_WRITE | VM_PROT_READ); + err = vm_map (mach_task_self (), &mapaddr, round_page (old_size), 0, 1, + mapobj, lbn * sblock->fs_bsize, 0, + VM_PROT_READ|VM_PROT_WRITE, VM_PROT_READ|VM_PROT_WRITE, 0); + assert_perror (err); - /* And deallocate the old block */ + /* Allow these pageins to occur even though we're holding the lock */ + spin_lock (&unlocked_pagein_lock); + np->dn->fileinfo->allow_unlocked_pagein = lbn * sblock->fs_bsize; + np->dn->fileinfo->unlocked_pagein_length = round_page (old_size); + spin_unlock (&unlocked_pagein_lock); + + /* Make sure all waiting pageins see this change. */ + mutex_lock (&np->allocptrlock->master); + condition_broadcast (&np->allocptrlock->wakeup); + mutex_unlock (&np->allocptrlock->master); + + /* Force the pages in core and make sure they are dirty */ + for (pokeaddr = (char *)mapaddr; + pokeaddr < mapaddr + round_page (old_size); + pokeaddr += vm_page_size / sizeof (*pokeaddr)) + *pokeaddr = *pokeaddr; + + /* Turn off the special pagein permission */ + spin_lock (&unlocked_pagein_lock); + np->dn->fileinfo->allow_unlocked_pagein = 0; + np->dn->fileinfo->unlocked_pagein_length = 0; + spin_unlock (&unlocked_pagein_lock); + + /* Undo mapping */ + mach_port_deallocate (mach_task_self (), mapobj); + vm_deallocate (mach_task_self (), mapaddr, round_page (old_size); + + /* Now it's OK to free the old block */ ffs_blkfree (np, old_pbn, old_size); + + /* Tell caller that we've moved data */ + return 1; } -} + else + return 0; +} /* Implement the diskfs_grow callback; see <hurd/diskfs.h> for the @@ -443,6 +456,7 @@ diskfs_grow (struct node *np, error_t err; struct dinode *di = dino (np->dn->number); mach_port_t pagerpt; + int needsync = 0; /* Zero an sblock->fs_bsize piece of disk starting at BNO, synchronously. We do this on newly allocated indirect @@ -496,7 +510,8 @@ diskfs_grow (struct node *np, old_pbn = read_disk_entry (di->di_db[olbn]); - block_extended (np, olbn, old_pbn, bno, osize, sblock->fs_bsize); + need_sync = block_extended (np, olbn, old_pbn, bno, + osize, sblock->fs_bsize); write_disk_entry (di->di_db[olbn], bno); record_poke (di, sizeof (struct dinode)); @@ -522,7 +537,7 @@ diskfs_grow (struct node *np, if (err) goto out; - block_extended (np, lbn, old_pbn, bno, osize, size); + need_sync = block_extended (np, lbn, old_pbn, bno, osize, size); write_disk_entry (di->di_db[lbn], bno); record_poke (di, sizeof (struct dinode)); @@ -653,6 +668,9 @@ diskfs_grow (struct node *np, rwlock_writer_unlock (&np->dn->allocptrlock); + if (need_sync) + diskfs_file_update (np, 1); + return err; } |