104 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			104 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.REQ_COUNT;
 | |
|   ino_t inum = fs_m_in.REQ_INODE_NR;
 | |
| 
 | |
|   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);
 | |
| }
 | 
