minix/lib/libpuffs/inode.c
Lionel Sambuc 6a94be7e63 Message types for VFS putnode
Change-Id: I0802ccaaaa6ee1b4eb96d62b08f9795c790ce39b
2014-07-28 17:05:28 +02:00

103 lines
2.7 KiB
C

/*
* Created (MFS based):
* June 2011 (Evgeniy Ivanov)
*/
#include "fs.h"
#include <string.h>
#include <assert.h>
#include <minix/vfsif.h>
#include "puffs.h"
#include "puffs_priv.h"
void release_node(struct puffs_usermount *pu, struct puffs_node *pn)
{
assert(pn->pn_count == 0);
/* Required if puffs_node_reclaim() decides to leave node in the list */
pn->pn_mountpoint = FALSE;
if (pu->pu_ops.puffs_node_reclaim) {
if (global_pu->pu_ops.puffs_node_reclaim(global_pu, pn) != 0)
lpuffs_debug("Warning: reclaim failed\n");
} else {
puffs_pn_put(pn);
}
}
/*===========================================================================*
* fs_putnode *
*===========================================================================*/
int fs_putnode(void)
{
/* Find the pnode specified by the request message and decrease its counter.
* Release unused pnode.
*/
struct puffs_node *pn;
int count = fs_m_in.m_vfs_fs_putnode.count;
ino_t inum = fs_m_in.m_vfs_fs_putnode.inode;
if ((pn = puffs_pn_nodewalk(global_pu, 0, &inum)) == NULL) {
/* XXX Probably removed from the list, see puffs_pn_remove() */
struct puffs_node *pn_cur, *pn_next;
pn_cur = LIST_FIRST(&global_pu->pu_pnode_removed_lst);
while (pn_cur) {
pn_next = LIST_NEXT(pn_cur, pn_entries);
if (pn_cur->pn_va.va_fileid == inum) {
pn = pn_cur;
break;
}
pn_cur = pn_next;
}
}
if (pn == NULL) {
lpuffs_debug("%s:%d putnode: pnode #%ld dev: %d not found\n", __FILE__,
__LINE__, inum, fs_dev);
panic("fs_putnode failed");
}
if (count <= 0) {
lpuffs_debug("%s:%d putnode: bad value for count: %d\n", __FILE__,
__LINE__, count);
panic("fs_putnode failed");
} else if (pn->pn_count == 0) {
/* FUSE fs might store in the list pnodes, which we hasn't
* open, this means we got put request for file,
* which wasn't opened by VFS.
*/
lpuffs_debug("%s:%d putnode: pn_count already zero\n", __FILE__,
__LINE__);
panic("fs_putnode failed");
} else if (count > pn->pn_count) {
struct puffs_node *pn_cur, *pn_next;
struct puffs_usermount *pu = global_pu;
ino_t ino = pn->pn_va.va_fileid;
pn_cur = LIST_FIRST(&pu->pu_pnodelst);
lpuffs_debug("inum count path polen hash\n");
while (pn_cur) {
pn_next = LIST_NEXT(pn_cur, pn_entries);
if (pn_cur->pn_va.va_fileid == ino) {
lpuffs_debug("%ld: %d %s %u %u\n", ino, pn_cur->pn_count,
pn_cur->pn_po.po_path,
pn_cur->pn_po.po_len,
pn_cur->pn_po.po_hash);
}
pn_cur = pn_next;
}
lpuffs_debug("%s:%d putnode: count too high: %d > %d\n", __FILE__,
__LINE__, count, pn->pn_count);
panic("fs_putnode failed");
}
pn->pn_count -= count;
if (pn->pn_count == 0)
release_node(global_pu, pn);
return(OK);
}