summaryrefslogtreecommitdiff
path: root/ufs
diff options
context:
space:
mode:
Diffstat (limited to 'ufs')
-rw-r--r--ufs/pager.c71
1 files changed, 62 insertions, 9 deletions
diff --git a/ufs/pager.c b/ufs/pager.c
index d4bc87c5..0da2609e 100644
--- a/ufs/pager.c
+++ b/ufs/pager.c
@@ -22,6 +22,8 @@
spin_lock_t node2pagelock = SPIN_LOCK_INITIALIZER;
+spin_lock_t unlocked_pagein_lock = SPIN_LOCK_INITIALIZER;
+
#ifdef DONT_CACHE_MEMORY_OBJECTS
#define MAY_CACHE 0
#else
@@ -34,15 +36,18 @@ struct port_bucket *pager_bucket;
disk address (in disk block) in *ADDR. If *NPLOCK is set on
return, then release that mutex after I/O on the data has
completed. Set DISKSIZE to be the amount of valid data on disk.
- (If this is an unallocated block, then set *ADDR to zero.) */
+ (If this is an unallocated block, then set *ADDR to zero.)
+ ISREAD is non-zero iff this is for a pagein. */
static error_t
find_address (struct user_pager_info *upi,
vm_address_t offset,
daddr_t *addr,
int *disksize,
- struct rwlock **nplock)
+ struct rwlock **nplock,
+ int isread)
{
error_t err;
+ struct rwlock *lock;
assert (upi->type == DISK || upi->type == FILE_DATA);
@@ -60,12 +65,58 @@ find_address (struct user_pager_info *upi,
np = upi->np;
- rwlock_reader_lock (&np->dn->allocptrlock);
- *nplock = &np->dn->allocptrlock;
+ if (isread)
+ {
+ try_again:
+
+ /* If we should allow an unlocked pagein, do so. (This
+ still has a slight race; there could be a pageout in progress
+ which is blocked on NP->np->allocptrlock itself. In that
+ case the pagein that should proceed unimpeded is blocked
+ in the pager library waiting for the pageout to complete.
+ I think this is sufficiently rare to put it off for the time
+ being.) */
+
+ spin_lock (&unlocked_pagein_lock);
+ if (offset >= upi->allow_unlocked_pagein
+ && (offset + vm_page_size
+ <= upi->allow_unlocked_pagein + upi->unlocked_pagein_length))
+ {
+ spin_unlock (&unlocked_pagein_lock);
+ *nplock = 0;
+ goto have_lock;
+ }
+ spin_unlock (&unlocked_pagein_lock);
+
+ /* Block on the rwlock if necessary; but when we wake up,
+ don't acquire it; check again from the top.
+ This is mutated inline from rwlock.h. */
+ lock = &np->dn->allocptrlock;
+ mutex_lock (&lock->master);
+ if (lock->readers == -1 || lock->writers_waiting)
+ {
+ lock->readers_waiting++;
+ condition_wait (&lock->wakeup, &lock->master);
+ lock->readers_waiting--;
+ mutex_unlock (&lock->master);
+ goto try_again;
+ }
+ lock->readers++;
+ mutex_unlock (&lock->master);
+ *nplock = lock;
+ }
+ else
+ {
+ rwlock_reader_lock (&np->dn->allocptrlock);
+ *nplock = &np->dn->allocptrlock;
+ }
+ have_lock:
+
if (offset >= np->allocsize)
{
- rwlock_reader_unlock (&np->dn->allocptrlock);
+ if (*nplock)
+ rwlock_reader_unlock (*nplock);
return EIO;
}
@@ -75,8 +126,8 @@ find_address (struct user_pager_info *upi,
*disksize = __vm_page_size;
err = fetch_indir_spec (np, lblkno (sblock, offset), indirs);
- if (err)
- rwlock_reader_unlock (&np->dn->allocptrlock);
+ if (err && *nplock)
+ rwlock_reader_unlock (*nplock);
else
{
if (indirs[0].bno)
@@ -104,7 +155,7 @@ pager_read_page (struct user_pager_info *pager,
daddr_t addr;
int disksize;
- err = find_address (pager, page, &addr, &disksize, &nplock);
+ err = find_address (pager, page, &addr, &disksize, &nplock, 1);
if (err)
return err;
@@ -143,7 +194,7 @@ pager_write_page (struct user_pager_info *pager,
struct rwlock *nplock;
error_t err;
- err = find_address (pager, page, &addr, &disksize, &nplock);
+ err = find_address (pager, page, &addr, &disksize, &nplock, 0);
if (err)
return err;
@@ -490,6 +541,8 @@ diskfs_get_filemap (struct node *np, vm_prot_t prot)
upi->type = FILE_DATA;
upi->np = np;
upi->max_prot = prot;
+ upi->allow_unlocked_pagein = 0;
+ upi->unlocked_pagein_length = 0;
diskfs_nref_light (np);
upi->p = pager_create (upi, pager_bucket,
MAY_CACHE, MEMORY_OBJECT_COPY_DELAY);