97e069713b
systems a chance to know if they have locked already. This fixes a locking problem in BFS where one thread tried to acquire two read locks (where someone else trying to acquire a write lock would have caused a dead lock). git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@17108 a95241bf-73f2-0310-859d-f6bbb57e9c96
131 lines
2.7 KiB
C++
131 lines
2.7 KiB
C++
/*
|
|
* Copyright 2004-2006, Axel Dörfler, axeld@pinc-software.de.
|
|
* Distributed under the terms of the MIT License.
|
|
*/
|
|
|
|
|
|
#include "vnode_store.h"
|
|
|
|
#include <file_cache.h>
|
|
#include <vfs.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
|
|
static void
|
|
store_destroy(struct vm_store *store)
|
|
{
|
|
free(store);
|
|
}
|
|
|
|
|
|
static status_t
|
|
store_commit(struct vm_store *_store, off_t size)
|
|
{
|
|
vnode_store *store = (vnode_store *)_store;
|
|
|
|
store->vm.committed_size = size;
|
|
return B_OK;
|
|
}
|
|
|
|
|
|
static bool
|
|
store_has_page(struct vm_store *_store, off_t offset)
|
|
{
|
|
// We always pretend to have the page - even if it's beyond the size of
|
|
// the file. The read function will only cut down the size of the read,
|
|
// it won't fail because of that.
|
|
return true;
|
|
}
|
|
|
|
|
|
static status_t
|
|
store_read(struct vm_store *_store, off_t offset, const iovec *vecs, size_t count,
|
|
size_t *_numBytes, bool fsReenter)
|
|
{
|
|
vnode_store *store = (vnode_store *)_store;
|
|
size_t bytesUntouched = *_numBytes;
|
|
|
|
status_t status = vfs_read_pages(store->vnode, NULL, offset, vecs, count,
|
|
_numBytes, fsReenter);
|
|
|
|
bytesUntouched -= *_numBytes;
|
|
|
|
// if the request could be filled completely, or an error occured, we're done here
|
|
if (status < B_OK || bytesUntouched == 0)
|
|
return status;
|
|
|
|
// Clear out any leftovers that were not touched by the above read - we're
|
|
// doing this here so that not every file system/device has to implement
|
|
// this
|
|
for (int32 i = count; i-- > 0 && bytesUntouched != 0;) {
|
|
size_t length = min_c(bytesUntouched, vecs[i].iov_len);
|
|
|
|
// ToDo: will have to map the pages in later (when we switch to physical pages)
|
|
memset((void *)((addr_t)vecs[i].iov_base + vecs[i].iov_len - length), 0, length);
|
|
bytesUntouched -= length;
|
|
}
|
|
|
|
return B_OK;
|
|
}
|
|
|
|
|
|
static status_t
|
|
store_write(struct vm_store *_store, off_t offset, const iovec *vecs, size_t count,
|
|
size_t *_numBytes, bool fsReenter)
|
|
{
|
|
vnode_store *store = (vnode_store *)_store;
|
|
return vfs_write_pages(store->vnode, NULL, offset, vecs, count, _numBytes, fsReenter);
|
|
}
|
|
|
|
|
|
static void
|
|
store_acquire_ref(struct vm_store *_store)
|
|
{
|
|
vnode_store *store = (vnode_store *)_store;
|
|
vfs_acquire_vnode(store->vnode);
|
|
}
|
|
|
|
|
|
static void
|
|
store_release_ref(struct vm_store *_store)
|
|
{
|
|
vnode_store *store = (vnode_store *)_store;
|
|
vfs_put_vnode(store->vnode);
|
|
}
|
|
|
|
|
|
static vm_store_ops sStoreOps = {
|
|
&store_destroy,
|
|
&store_commit,
|
|
&store_has_page,
|
|
&store_read,
|
|
&store_write,
|
|
NULL, /* fault */
|
|
&store_acquire_ref,
|
|
&store_release_ref
|
|
};
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
extern "C" vm_store *
|
|
vm_create_vnode_store(void *vnode)
|
|
{
|
|
vnode_store *store = (vnode_store *)malloc(sizeof(struct vnode_store));
|
|
if (store == NULL)
|
|
return NULL;
|
|
|
|
store->vm.ops = &sStoreOps;
|
|
store->vm.cache = NULL;
|
|
store->vm.committed_size = 0;
|
|
|
|
store->vnode = vnode;
|
|
store->file_cache_ref = NULL;
|
|
|
|
return &store->vm;
|
|
}
|
|
|