Added a new call vfs_new_vnode() which can be used when a new vnode is created
but doesn't have to be read from disk for efficiency reasons. Fixed a bug in get_vnode() in case of fs->get_vnode() returns an error of some kind. fs_mount() no longer hides the return code from the fs->mount() function anymore (for now). Added some comments and ToDo items. Changed some return error values to the B_xxx style (will probably need to define some new error codes for OpenBeOS). Changed some return types to status_t/void where appropriate. git-svn-id: file:///srv/svn/repos/haiku/trunk/current@2786 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
896aefa8ec
commit
43604406d8
@ -495,7 +495,7 @@ create_new_vnode(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int
|
static status_t
|
||||||
dec_vnode_ref_count(struct vnode *vnode, bool reenter)
|
dec_vnode_ref_count(struct vnode *vnode, bool reenter)
|
||||||
{
|
{
|
||||||
int err;
|
int err;
|
||||||
@ -542,12 +542,11 @@ dec_vnode_ref_count(struct vnode *vnode, bool reenter)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int
|
static void
|
||||||
inc_vnode_ref_count(struct vnode *vnode)
|
inc_vnode_ref_count(struct vnode *vnode)
|
||||||
{
|
{
|
||||||
atomic_add(&vnode->ref_count, 1);
|
atomic_add(&vnode->ref_count, 1);
|
||||||
PRINT(("inc_vnode_ref_count: vnode %p, ref now %ld\n", vnode, vnode->ref_count));
|
PRINT(("inc_vnode_ref_count: vnode %p, ref now %ld\n", vnode, vnode->ref_count));
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -563,7 +562,7 @@ lookup_vnode(mount_id mountID, vnode_id vnodeID)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int
|
static status_t
|
||||||
get_vnode(mount_id mountID, vnode_id vnodeID, struct vnode **_vnode, int reenter)
|
get_vnode(mount_id mountID, vnode_id vnodeID, struct vnode **_vnode, int reenter)
|
||||||
{
|
{
|
||||||
struct vnode *vnode;
|
struct vnode *vnode;
|
||||||
@ -575,13 +574,13 @@ get_vnode(mount_id mountID, vnode_id vnodeID, struct vnode **_vnode, int reenter
|
|||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
vnode = lookup_vnode(mountID, vnodeID);
|
vnode = lookup_vnode(mountID, vnodeID);
|
||||||
if (vnode) {
|
if (vnode && vnode->busy) {
|
||||||
if (vnode->busy) {
|
// ToDo: this is an endless loop if the vnode is not
|
||||||
mutex_unlock(&gVnodeMutex);
|
// becoming unbusy anymore (for whatever reason)
|
||||||
snooze(10000); // 10 ms
|
mutex_unlock(&gVnodeMutex);
|
||||||
mutex_lock(&gVnodeMutex);
|
snooze(10000); // 10 ms
|
||||||
continue;
|
mutex_lock(&gVnodeMutex);
|
||||||
}
|
continue;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -599,13 +598,13 @@ get_vnode(mount_id mountID, vnode_id vnodeID, struct vnode **_vnode, int reenter
|
|||||||
}
|
}
|
||||||
vnode->mount_id = mountID;
|
vnode->mount_id = mountID;
|
||||||
vnode->id = vnodeID;
|
vnode->id = vnodeID;
|
||||||
|
|
||||||
mutex_lock(&gMountMutex);
|
mutex_lock(&gMountMutex);
|
||||||
|
|
||||||
vnode->mount = find_mount(mountID);
|
vnode->mount = find_mount(mountID);
|
||||||
if (!vnode->mount) {
|
if (!vnode->mount) {
|
||||||
mutex_unlock(&gMountMutex);
|
mutex_unlock(&gMountMutex);
|
||||||
err = ERR_INVALID_HANDLE;
|
err = B_ENTRY_NOT_FOUND;
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
vnode->busy = true;
|
vnode->busy = true;
|
||||||
@ -616,10 +615,10 @@ get_vnode(mount_id mountID, vnode_id vnodeID, struct vnode **_vnode, int reenter
|
|||||||
mutex_unlock(&gMountMutex);
|
mutex_unlock(&gMountMutex);
|
||||||
|
|
||||||
err = FS_CALL(vnode, get_vnode)(vnode->mount->cookie, vnodeID, &vnode->private_node, reenter);
|
err = FS_CALL(vnode, get_vnode)(vnode->mount->cookie, vnodeID, &vnode->private_node, reenter);
|
||||||
if (err < 0 && vnode->private_node == NULL) {
|
if (err < 0 || vnode->private_node == NULL) {
|
||||||
remove_vnode_from_mount_list(vnode, vnode->mount);
|
remove_vnode_from_mount_list(vnode, vnode->mount);
|
||||||
if (vnode->private_node == NULL)
|
if (vnode->private_node == NULL)
|
||||||
err = EINVAL;
|
err = B_BAD_VALUE;
|
||||||
}
|
}
|
||||||
mutex_lock(&gVnodeMutex);
|
mutex_lock(&gVnodeMutex);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
@ -758,10 +757,11 @@ vnode_path_to_vnode(struct vnode *vnode, char *path, bool traverseLeafLink, stru
|
|||||||
mutex_unlock(&gVnodeMutex);
|
mutex_unlock(&gVnodeMutex);
|
||||||
|
|
||||||
if (!nextVnode) {
|
if (!nextVnode) {
|
||||||
// pretty screwed up here
|
// pretty screwed up here - the file system found the vnode, but the hash
|
||||||
|
// lookup failed, so our internal structures are messed up
|
||||||
panic("path_to_vnode: could not lookup vnode (mountid 0x%lx vnid 0x%Lx)\n", vnode->mount_id, vnodeID);
|
panic("path_to_vnode: could not lookup vnode (mountid 0x%lx vnid 0x%Lx)\n", vnode->mount_id, vnodeID);
|
||||||
put_vnode(vnode);
|
put_vnode(vnode);
|
||||||
return ERR_VFS_PATH_NOT_FOUND;
|
return B_ENTRY_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the new node is a symbolic link, resolve it (if we've been told to do it)
|
// If the new node is a symbolic link, resolve it (if we've been told to do it)
|
||||||
@ -1188,7 +1188,60 @@ get_new_fd(int type, struct vnode *vnode, fs_cookie cookie, int openMode, bool k
|
|||||||
// Functions the VFS exports for other parts of the kernel
|
// Functions the VFS exports for other parts of the kernel
|
||||||
|
|
||||||
|
|
||||||
int
|
status_t
|
||||||
|
vfs_new_vnode(mount_id mountID, vnode_id vnodeID, fs_vnode privateNode)
|
||||||
|
{
|
||||||
|
struct vnode *vnode;
|
||||||
|
status_t status = B_OK;
|
||||||
|
|
||||||
|
if (privateNode == NULL)
|
||||||
|
return B_BAD_VALUE;
|
||||||
|
|
||||||
|
mutex_lock(&gVnodeMutex);
|
||||||
|
|
||||||
|
// file system integrity check:
|
||||||
|
// test if the vnode already exists and bail out if this is the case!
|
||||||
|
|
||||||
|
// ToDo: the R5 implementation obviously checks for a different cookie
|
||||||
|
// and doesn't panic if they are equal
|
||||||
|
|
||||||
|
vnode = lookup_vnode(mountID, vnodeID);
|
||||||
|
if (vnode != NULL)
|
||||||
|
panic("vnode %ld:%Ld already exists (node = %p, vnode->node = %p)!", mountID, vnodeID, privateNode, vnode->private_node);
|
||||||
|
|
||||||
|
vnode = create_new_vnode();
|
||||||
|
if (!vnode) {
|
||||||
|
status = B_NO_MEMORY;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
|
vnode->mount_id = mountID;
|
||||||
|
vnode->id = vnodeID;
|
||||||
|
vnode->private_node = privateNode;
|
||||||
|
|
||||||
|
// add the vnode to the mount structure
|
||||||
|
mutex_lock(&gMountMutex);
|
||||||
|
vnode->mount = find_mount(mountID);
|
||||||
|
if (!vnode->mount) {
|
||||||
|
mutex_unlock(&gMountMutex);
|
||||||
|
status = B_ENTRY_NOT_FOUND;
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
hash_insert(gVnodeTable, vnode);
|
||||||
|
|
||||||
|
add_vnode_to_mount_list(vnode, vnode->mount);
|
||||||
|
mutex_unlock(&gMountMutex);
|
||||||
|
|
||||||
|
vnode->ref_count = 1;
|
||||||
|
|
||||||
|
err:
|
||||||
|
mutex_unlock(&gVnodeMutex);
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
status_t
|
||||||
vfs_get_vnode(mount_id mountID, vnode_id vnodeID, fs_vnode *_fsNode)
|
vfs_get_vnode(mount_id mountID, vnode_id vnodeID, fs_vnode *_fsNode)
|
||||||
{
|
{
|
||||||
struct vnode *vnode;
|
struct vnode *vnode;
|
||||||
@ -1202,7 +1255,7 @@ vfs_get_vnode(mount_id mountID, vnode_id vnodeID, fs_vnode *_fsNode)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int
|
status_t
|
||||||
vfs_put_vnode(mount_id mountID, vnode_id vnodeID)
|
vfs_put_vnode(mount_id mountID, vnode_id vnodeID)
|
||||||
{
|
{
|
||||||
struct vnode *vnode;
|
struct vnode *vnode;
|
||||||
@ -1234,7 +1287,7 @@ vfs_vnode_release_ref(void *vnode)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int
|
status_t
|
||||||
vfs_remove_vnode(mount_id mountID, vnode_id vnodeID)
|
vfs_remove_vnode(mount_id mountID, vnode_id vnodeID)
|
||||||
{
|
{
|
||||||
struct vnode *vnode;
|
struct vnode *vnode;
|
||||||
@ -1276,7 +1329,7 @@ vfs_get_vnode_from_fd(int fd, bool kernel, void **vnode)
|
|||||||
*vnode = get_vnode_from_fd(ioctx, fd);
|
*vnode = get_vnode_from_fd(ioctx, fd);
|
||||||
|
|
||||||
if (*vnode == NULL)
|
if (*vnode == NULL)
|
||||||
return ERR_INVALID_HANDLE;
|
return B_FILE_ERROR;
|
||||||
|
|
||||||
return B_NO_ERROR;
|
return B_NO_ERROR;
|
||||||
}
|
}
|
||||||
@ -2275,7 +2328,7 @@ common_sync(int fd, bool kernel)
|
|||||||
|
|
||||||
descriptor = get_fd_and_vnode(fd, &vnode, kernel);
|
descriptor = get_fd_and_vnode(fd, &vnode, kernel);
|
||||||
if (descriptor == NULL)
|
if (descriptor == NULL)
|
||||||
return ERR_INVALID_HANDLE;
|
return B_FILE_ERROR;
|
||||||
|
|
||||||
if (FS_CALL(vnode, fsync) != NULL)
|
if (FS_CALL(vnode, fsync) != NULL)
|
||||||
status = FS_CALL(vnode, fsync)(vnode->mount->cookie, vnode->private_node);
|
status = FS_CALL(vnode, fsync)(vnode->mount->cookie, vnode->private_node);
|
||||||
@ -3109,7 +3162,7 @@ fs_mount(char *path, const char *device, const char *fsName, void *args, bool ke
|
|||||||
|
|
||||||
mount->fs = get_file_system(fsName);
|
mount->fs = get_file_system(fsName);
|
||||||
if (mount->fs == NULL) {
|
if (mount->fs == NULL) {
|
||||||
err = ERR_VFS_INVALID_FS;
|
err = ENODEV;
|
||||||
goto err2;
|
goto err2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3120,13 +3173,14 @@ fs_mount(char *path, const char *device, const char *fsName, void *args, bool ke
|
|||||||
if (!gRoot) {
|
if (!gRoot) {
|
||||||
// we haven't mounted anything yet
|
// we haven't mounted anything yet
|
||||||
if (strcmp(path, "/") != 0) {
|
if (strcmp(path, "/") != 0) {
|
||||||
err = ERR_VFS_GENERAL;
|
err = B_ERROR;
|
||||||
goto err3;
|
goto err3;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = FS_MOUNT_CALL(mount, mount)(mount->id, device, NULL, &mount->cookie, &root_id);
|
err = FS_MOUNT_CALL(mount, mount)(mount->id, device, NULL, &mount->cookie, &root_id);
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
err = ERR_VFS_GENERAL;
|
// ToDo: why should we hide the error code from the file system here?
|
||||||
|
//err = ERR_VFS_GENERAL;
|
||||||
goto err3;
|
goto err3;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3137,7 +3191,7 @@ fs_mount(char *path, const char *device, const char *fsName, void *args, bool ke
|
|||||||
goto err2;
|
goto err2;
|
||||||
|
|
||||||
if (!covered_vnode) {
|
if (!covered_vnode) {
|
||||||
err = ERR_VFS_GENERAL;
|
err = B_ERROR;
|
||||||
goto err2;
|
goto err2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3411,7 +3465,7 @@ set_cwd(int fd, char *path, bool kernel)
|
|||||||
|
|
||||||
if (!S_ISDIR(stat.st_mode)) {
|
if (!S_ISDIR(stat.st_mode)) {
|
||||||
// nope, can't cwd to here
|
// nope, can't cwd to here
|
||||||
rc = ERR_VFS_WRONG_STREAM_TYPE;
|
rc = B_NOT_A_DIRECTORY;
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3838,7 +3892,7 @@ user_open_entry_ref(dev_t device, ino_t inode, const char *userName, int omode)
|
|||||||
int status;
|
int status;
|
||||||
|
|
||||||
if (!CHECK_USER_ADDRESS(userName))
|
if (!CHECK_USER_ADDRESS(userName))
|
||||||
return ERR_VM_BAD_USER_MEMORY;
|
return B_BAD_ADDRESS;
|
||||||
|
|
||||||
status = user_strlcpy(name, userName, sizeof(name) - 1);
|
status = user_strlcpy(name, userName, sizeof(name) - 1);
|
||||||
if (status < B_OK)
|
if (status < B_OK)
|
||||||
@ -3855,7 +3909,7 @@ user_open(const char *userPath, int omode)
|
|||||||
int status;
|
int status;
|
||||||
|
|
||||||
if (!CHECK_USER_ADDRESS(userPath))
|
if (!CHECK_USER_ADDRESS(userPath))
|
||||||
return ERR_VM_BAD_USER_MEMORY;
|
return B_BAD_ADDRESS;
|
||||||
|
|
||||||
status = user_strlcpy(path, userPath, SYS_MAX_PATH_LEN);
|
status = user_strlcpy(path, userPath, SYS_MAX_PATH_LEN);
|
||||||
if (status < 0)
|
if (status < 0)
|
||||||
|
Loading…
Reference in New Issue
Block a user