xfs_vnode.c 3.6 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2 3
 * Copyright (c) 2000-2003,2005 Silicon Graphics, Inc.
 * All Rights Reserved.
L
Linus Torvalds 已提交
4
 *
5 6
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
L
Linus Torvalds 已提交
7 8
 * published by the Free Software Foundation.
 *
9 10 11 12
 * This program is distributed in the hope that it would be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
L
Linus Torvalds 已提交
13
 *
14 15 16
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write the Free Software Foundation,
 * Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
L
Linus Torvalds 已提交
17 18
 */
#include "xfs.h"
19 20 21
#include "xfs_vnodeops.h"
#include "xfs_bmap_btree.h"
#include "xfs_inode.h"
L
Linus Torvalds 已提交
22

23 24 25 26 27 28 29 30 31 32 33
/*
 * And this gunk is needed for xfs_mount.h"
 */
#include "xfs_log.h"
#include "xfs_trans.h"
#include "xfs_sb.h"
#include "xfs_dmapi.h"
#include "xfs_inum.h"
#include "xfs_ag.h"
#include "xfs_mount.h"

L
Linus Torvalds 已提交
34 35 36 37 38 39 40

/*
 * Dedicated vnode inactive/reclaim sync semaphores.
 * Prime number of hash buckets since address is used as the key.
 */
#define NVSYNC                  37
#define vptosync(v)             (&vsync[((unsigned long)v) % NVSYNC])
41
static wait_queue_head_t vsync[NVSYNC];
L
Linus Torvalds 已提交
42

43
void __init
L
Linus Torvalds 已提交
44 45
vn_init(void)
{
46
	int i;
L
Linus Torvalds 已提交
47

48 49 50 51 52 53
	for (i = 0; i < NVSYNC; i++)
		init_waitqueue_head(&vsync[i]);
}

void
vn_iowait(
54
	xfs_inode_t	*ip)
55
{
56
	wait_queue_head_t *wq = vptosync(ip);
57

58
	wait_event(*wq, (atomic_read(&ip->i_iocount) == 0));
59 60 61 62
}

void
vn_iowake(
63
	xfs_inode_t	*ip)
64
{
65 66
	if (atomic_dec_and_test(&ip->i_iocount))
		wake_up(vptosync(ip));
L
Linus Torvalds 已提交
67 68
}

69 70 71 72 73 74 75
/*
 * Volume managers supporting multiple paths can send back ENODEV when the
 * final path disappears.  In this case continuing to fill the page cache
 * with dirty data which cannot be written out is evil, so prevent that.
 */
void
vn_ioerror(
76
	xfs_inode_t	*ip,
77 78 79 80 81
	int		error,
	char		*f,
	int		l)
{
	if (unlikely(error == -ENODEV))
82
		xfs_do_force_shutdown(ip->i_mount, SHUTDOWN_DEVICE_REQ, f, l);
83 84
}

85

L
Linus Torvalds 已提交
86 87 88
/*
 * Add a reference to a referenced vnode.
 */
89
bhv_vnode_t *
L
Linus Torvalds 已提交
90
vn_hold(
91
	bhv_vnode_t	*vp)
L
Linus Torvalds 已提交
92 93 94 95 96
{
	struct inode	*inode;

	XFS_STATS_INC(vn_hold);

97
	inode = igrab(vn_to_inode(vp));
L
Linus Torvalds 已提交
98 99 100 101 102
	ASSERT(inode);

	return vp;
}

103
#ifdef	XFS_INODE_TRACE
L
Linus Torvalds 已提交
104

105 106 107 108 109 110 111 112 113 114 115 116 117 118 119
/*
 * Reference count of Linux inode if present, -1 if the xfs_inode
 * has no associated Linux inode.
 */
static inline int xfs_icount(struct xfs_inode *ip)
{
	bhv_vnode_t *vp = XFS_ITOV_NULL(ip);

	if (vp)
		return vn_count(vp);
	return -1;
}

#define KTRACE_ENTER(ip, vk, s, line, ra)			\
	ktrace_enter(	(ip)->i_trace,				\
L
Linus Torvalds 已提交
120 121 122
/*  0 */		(void *)(__psint_t)(vk),		\
/*  1 */		(void *)(s),				\
/*  2 */		(void *)(__psint_t) line,		\
123
/*  3 */		(void *)(__psint_t)xfs_icount(ip),	\
L
Linus Torvalds 已提交
124
/*  4 */		(void *)(ra),				\
125
/*  5 */		NULL,					\
L
Linus Torvalds 已提交
126 127 128
/*  6 */		(void *)(__psint_t)current_cpu(),	\
/*  7 */		(void *)(__psint_t)current_pid(),	\
/*  8 */		(void *)__return_address,		\
129
/*  9 */		NULL, NULL, NULL, NULL, NULL, NULL, NULL)
L
Linus Torvalds 已提交
130 131 132 133 134

/*
 * Vnode tracing code.
 */
void
135
_xfs_itrace_entry(xfs_inode_t *ip, const char *func, inst_t *ra)
L
Linus Torvalds 已提交
136
{
137
	KTRACE_ENTER(ip, INODE_KTRACE_ENTRY, func, 0, ra);
L
Linus Torvalds 已提交
138 139 140
}

void
141
_xfs_itrace_exit(xfs_inode_t *ip, const char *func, inst_t *ra)
L
Linus Torvalds 已提交
142
{
143
	KTRACE_ENTER(ip, INODE_KTRACE_EXIT, func, 0, ra);
L
Linus Torvalds 已提交
144 145 146
}

void
147
xfs_itrace_hold(xfs_inode_t *ip, char *file, int line, inst_t *ra)
L
Linus Torvalds 已提交
148
{
149
	KTRACE_ENTER(ip, INODE_KTRACE_HOLD, file, line, ra);
L
Linus Torvalds 已提交
150 151 152
}

void
153
_xfs_itrace_ref(xfs_inode_t *ip, char *file, int line, inst_t *ra)
L
Linus Torvalds 已提交
154
{
155
	KTRACE_ENTER(ip, INODE_KTRACE_REF, file, line, ra);
L
Linus Torvalds 已提交
156 157 158
}

void
159
xfs_itrace_rele(xfs_inode_t *ip, char *file, int line, inst_t *ra)
L
Linus Torvalds 已提交
160
{
161
	KTRACE_ENTER(ip, INODE_KTRACE_RELE, file, line, ra);
L
Linus Torvalds 已提交
162
}
163
#endif	/* XFS_INODE_TRACE */