summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--ext2fs/pager.c236
1 files changed, 41 insertions, 195 deletions
diff --git a/ext2fs/pager.c b/ext2fs/pager.c
index 95b6da13..4ce2ba1d 100644
--- a/ext2fs/pager.c
+++ b/ext2fs/pager.c
@@ -1,5 +1,6 @@
/* Pager for ext2fs
- Copyright (C) 1994 Free Software Foundation
+
+ Copyright (C) 1994, 1995 Free Software Foundation, Inc.
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License as
@@ -15,9 +16,8 @@
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
-#include "ext2fs.h"
#include <strings.h>
-#include <stdio.h>
+#include "ext2fs.h"
spin_lock_t pagerlistlock = SPIN_LOCK_INITIALIZER;
struct user_pager_info *filepagerlist;
@@ -42,23 +42,18 @@ find_address (struct user_pager_info *upi,
int *disksize,
struct rwlock **nplock)
{
- error_t err;
-
assert (upi->type == DISK || upi->type == FILE_DATA);
if (upi->type == DISK)
{
- *disksize = __vm_page_size;
+ *disksize = vm_page_size;
*addr = offset / DEV_BSIZE;
*nplock = 0;
return 0;
}
else
{
- struct iblock_spec indirs[NIADDR + 1];
- struct node *np;
-
- np = upi->np;
+ struct node *np = upi->np;
rwlock_reader_lock (&np->dn->allocptrlock);
*nplock = &np->dn->allocptrlock;
@@ -69,24 +64,12 @@ find_address (struct user_pager_info *upi,
return EIO;
}
- if (offset + __vm_page_size > np->allocsize)
+ if (offset + vm_page_size > np->allocsize)
*disksize = np->allocsize - offset;
else
- *disksize = __vm_page_size;
-
- err = fetch_indir_spec (np, lblkno (sblock, offset), indirs);
- if (err)
- rwlock_reader_unlock (&np->dn->allocptrlock);
- else
- {
- if (indirs[0].bno)
- *addr = (fsbtodb (sblock, indirs[0].bno)
- + blkoff (sblock, offset) / DEV_BSIZE);
- else
- *addr = 0;
- }
-
- return err;
+ *disksize = vm_page_size;
+
+ return ext2_getblk(np, offset / block_size, 0, addr);
}
}
@@ -111,17 +94,13 @@ pager_read_page (struct user_pager_info *pager,
if (addr)
{
err = dev_read_sync (addr, (void *)buf, disksize);
- if (!err && disksize != __vm_page_size)
- bzero ((void *)(*buf + disksize), __vm_page_size - disksize);
+ if (!err && disksize != vm_page_size)
+ bzero ((void *)(*buf + disksize), vm_page_size - disksize);
*writelock = 0;
}
else
{
-#if 0
- printf ("Write-locked pagein Object %#x\tOffset %#x\n", pager, page);
- fflush (stdout);
-#endif
- vm_allocate (mach_task_self (), buf, __vm_page_size, 1);
+ vm_allocate (mach_task_self (), buf, vm_page_size, 1);
*writelock = 1;
}
@@ -151,11 +130,11 @@ pager_write_page (struct user_pager_info *pager,
err = dev_write_sync (addr, buf, disksize);
else
{
- printf ("Attempt to write unallocated disk\n.");
- printf ("Object %p\tOffset %#x\n", pager, page);
- fflush (stdout);
- err = 0; /* unallocated disk;
- error would be pointless */
+ ext2_error("pager_write_page",
+ "Attempt to write unallocated disk;"
+ " object = %p; offset = 0x%x", pager, page);
+ /* unallocated disk; error would be pointless */
+ err = 0;
}
if (nplock)
@@ -170,168 +149,36 @@ error_t
pager_unlock_page (struct user_pager_info *pager,
vm_offset_t address)
{
- struct node *np;
- error_t err;
- struct iblock_spec indirs[NIADDR + 1];
- daddr_t bno;
- struct disknode *dn;
- struct ext2_inode *di;
-
- /* Zero an sblock->fs_bsize piece of disk starting at BNO,
- synchronously. We do this on newly allocated indirect
- blocks before setting the pointer to them to ensure that an
- indirect block absolutely never points to garbage. */
- void zero_disk_block (int bno)
- {
- bzero (indir_block (bno), sblock->fs_bsize);
- sync_disk_blocks (bno, sblock->fs_bsize, 1);
- };
-
- /* Problem--where to get cred values for allocation here? */
-
-#if 0
- printf ("Unlock page request, Object %#x\tOffset %#x...", pager, address);
- fflush (stdout);
-#endif
-
if (pager->type == DISK)
return 0;
-
- np = pager->np;
- dn = np->dn;
- di = dino (dn->number);
-
- rwlock_writer_lock (&dn->allocptrlock);
-
- /* If this is the last block, we don't let it get unlocked. */
- if (address + __vm_page_size
- > blkroundup (sblock, np->allocsize) - sblock->fs_bsize)
- {
- printf ("attempt to unlock at last block denied\n");
- fflush (stdout);
- rwlock_writer_unlock (&dn->allocptrlock);
- return EIO;
- }
-
- err = fetch_indir_spec (np, lblkno (sblock, address), indirs);
- if (err)
- {
- rwlock_writer_unlock (&dn->allocptrlock);
- return EIO;
- }
-
- err = diskfs_catch_exception ();
- if (err)
+ else
{
- rwlock_writer_unlock (&dn->allocptrlock);
- return EIO;
- }
+ error_t err;
+ struct node *np = pager->np;
+ struct disknode *dn = np->dn;
+ int block_size_shift = EXT2_BLOCK_SIZE_BITS(sblock);
- /* See if we need a triple indirect block; fail if we do. */
- assert (indirs[0].offset == -1
- || indirs[1].offset == -1
- || indirs[2].offset == -1);
+ rwlock_writer_lock (&dn->allocptrlock);
- /* Check to see if this block is allocated. */
- if (indirs[0].bno == 0)
- {
- if (indirs[0].offset == -1)
- {
- err = ffs_alloc (np, lblkno (sblock, address),
- ffs_blkpref (np, lblkno (sblock, address),
- lblkno (sblock, address), di->di_db),
- sblock->fs_bsize, &bno, 0);
- if (err)
- goto out;
- assert (lblkno (sblock, address) < NDADDR);
- indirs[0].bno = di->di_db[lblkno (sblock, address)] = bno;
- record_poke (di, sizeof (struct ext2_inode));
- }
- else
+ /* If this is the last block, we don't let it get unlocked. */
+ if (address + vm_page_size
+ > ((np->allocsize >> block_size_shift) << block_size_shift))
{
- daddr_t *siblock;
-
- /* We need to set siblock to the single indirect block
- array; see if the single indirect block is allocated. */
- if (indirs[1].bno == 0)
- {
- if (indirs[1].offset == -1)
- {
- err = ffs_alloc (np, lblkno (sblock, address),
- ffs_blkpref (np, lblkno (sblock, address),
- INDIR_SINGLE, di->di_ib),
- sblock->fs_bsize, &bno, 0);
- if (err)
- goto out;
- zero_disk_block (bno);
- indirs[1].bno = di->di_ib[INDIR_SINGLE] = bno;
- record_poke (di, sizeof (struct ext2_inode));
- }
- else
- {
- daddr_t *diblock;
-
- /* We need to set diblock to the double indirect
- block array; see if the double indirect block is
- allocated. */
- if (indirs[2].bno == 0)
- {
- /* This assert because triple indirection is
- not supported. */
- assert (indirs[2].offset == -1);
-
- err = ffs_alloc (np, lblkno (sblock, address),
- ffs_blkpref (np, lblkno (sblock,
- address),
- INDIR_DOUBLE, di->di_ib),
- sblock->fs_bsize, &bno, 0);
- if (err)
- goto out;
- zero_disk_block (bno);
- indirs[2].bno = di->di_ib[INDIR_DOUBLE] = bno;
- record_poke (di, sizeof (struct ext2_inode));
- }
-
- diblock = indir_block (indirs[2].bno);
- mark_indir_dirty (np, indirs[2].bno);
-
- /* Now we can allocate the single indirect block */
-
- err = ffs_alloc (np, lblkno (sblock, address),
- ffs_blkpref (np, lblkno (sblock, address),
- indirs[1].offset, diblock),
- sblock->fs_bsize, &bno, 0);
- if (err)
- goto out;
- zero_disk_block (bno);
- indirs[1].bno = diblock[indirs[1].offset] = bno;
- record_poke (diblock, sblock->fs_bsize);
- }
- }
-
- siblock = indir_block (indirs[1].bno);
- mark_indir_dirty (np, indirs[1].bno);
-
- /* Now we can allocate the data block. */
-
- err = ffs_alloc (np, lblkno (sblock, address),
- ffs_blkpref (np, lblkno (sblock, address),
- indirs[0].offset, siblock),
- sblock->fs_bsize, &bno, 0);
- if (err)
- goto out;
-
- dev_write_sync (fsbtodb (sblock, bno), zeroblock, sblock->fs_bsize);
-
- indirs[0].bno = siblock[indirs[0].offset] = bno;
- record_poke (siblock, sblock->fs_bsize);
+ ext2_error ("pager_unlock_page",
+ "attempt to unlock at last block denied\n");
+ rwlock_writer_unlock (&dn->allocptrlock);
+ return EIO;
}
+
+ err = diskfs_catch_exception ();
+ if (!err)
+ err = ext2_getblk(np, address / block_size, 1, &buf);
+ diskfs_end_catch_exception ();
+
+ rwlock_writer_unlock (&dn->allocptrlock);
+
+ return err;
}
-
- out:
- diskfs_end_catch_exception ();
- rwlock_writer_unlock (&dn->allocptrlock);
- return err;
}
/* Implement the pager_report_extent callback from the pager library. See
@@ -387,8 +234,7 @@ create_disk_pager ()
/* This syncs a single file (NP) to disk. Wait for all I/O to complete
if WAIT is set. NP->lock must be held. */
void
-diskfs_file_update (struct node *np,
- int wait)
+diskfs_file_update (struct node *np, int wait)
{
struct dirty_indir *d, *tmp;
struct user_pager_info *upi;
@@ -407,7 +253,7 @@ diskfs_file_update (struct node *np,
for (d = np->dn->dirty; d; d = tmp)
{
- sync_disk_blocks (d->bno, sblock->fs_bsize, wait);
+ sync_disk_image (d->bno, block_size, wait);
tmp = d->next;
free (d);
}