Blame view

fs/ceph/export.c 5.57 KB
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
1
2
3
  #include "ceph_debug.h"
  
  #include <linux/exportfs.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
4
  #include <linux/slab.h>
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
  #include <asm/unaligned.h>
  
  #include "super.h"
  
  /*
   * NFS export support
   *
   * NFS re-export of a ceph mount is, at present, only semireliable.
   * The basic issue is that the Ceph architectures doesn't lend itself
   * well to generating filehandles that will remain valid forever.
   *
   * So, we do our best.  If you're lucky, your inode will be in the
   * client's cache.  If it's not, and you have a connectable fh, then
   * the MDS server may be able to find it for you.  Otherwise, you get
   * ESTALE.
   *
   * There are ways to this more reliable, but in the non-connectable fh
   * case, we won't every work perfectly, and in the connectable case,
   * some changes are needed on the MDS side to work better.
   */
  
  /*
   * Basic fh
   */
  struct ceph_nfs_fh {
  	u64 ino;
  } __attribute__ ((packed));
  
  /*
   * Larger 'connectable' fh that includes parent ino and name hash.
   * Use this whenever possible, as it works more reliably.
   */
  struct ceph_nfs_confh {
  	u64 ino, parent_ino;
  	u32 parent_name_hash;
  } __attribute__ ((packed));
  
  static int ceph_encode_fh(struct dentry *dentry, u32 *rawfh, int *max_len,
  			  int connectable)
  {
92923dcbf   Aneesh Kumar K.V   ceph: Fix return ...
45
  	int type;
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
46
47
48
49
  	struct ceph_nfs_fh *fh = (void *)rawfh;
  	struct ceph_nfs_confh *cfh = (void *)rawfh;
  	struct dentry *parent = dentry->d_parent;
  	struct inode *inode = dentry->d_inode;
92923dcbf   Aneesh Kumar K.V   ceph: Fix return ...
50
51
  	int connected_handle_length = sizeof(*cfh)/4;
  	int handle_length = sizeof(*fh)/4;
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
52
53
54
55
  
  	/* don't re-export snaps */
  	if (ceph_snap(inode) != CEPH_NOSNAP)
  		return -EINVAL;
92923dcbf   Aneesh Kumar K.V   ceph: Fix return ...
56
  	if (*max_len >= connected_handle_length) {
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
57
58
59
60
61
  		dout("encode_fh %p connectable
  ", dentry);
  		cfh->ino = ceph_ino(dentry->d_inode);
  		cfh->parent_ino = ceph_ino(parent->d_inode);
  		cfh->parent_name_hash = parent->d_name.hash;
92923dcbf   Aneesh Kumar K.V   ceph: Fix return ...
62
  		*max_len = connected_handle_length;
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
63
  		type = 2;
92923dcbf   Aneesh Kumar K.V   ceph: Fix return ...
64
  	} else if (*max_len >= handle_length) {
bba0cd0e3   Aneesh Kumar K.V   ceph: Update max_...
65
66
  		if (connectable) {
  			*max_len = connected_handle_length;
92923dcbf   Aneesh Kumar K.V   ceph: Fix return ...
67
  			return 255;
bba0cd0e3   Aneesh Kumar K.V   ceph: Update max_...
68
  		}
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
69
70
71
  		dout("encode_fh %p
  ", dentry);
  		fh->ino = ceph_ino(dentry->d_inode);
92923dcbf   Aneesh Kumar K.V   ceph: Fix return ...
72
  		*max_len = handle_length;
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
73
74
  		type = 1;
  	} else {
bba0cd0e3   Aneesh Kumar K.V   ceph: Update max_...
75
  		*max_len = handle_length;
92923dcbf   Aneesh Kumar K.V   ceph: Fix return ...
76
  		return 255;
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
  	}
  	return type;
  }
  
  /*
   * convert regular fh to dentry
   *
   * FIXME: we should try harder by querying the mds for the ino.
   */
  static struct dentry *__fh_to_dentry(struct super_block *sb,
  				     struct ceph_nfs_fh *fh)
  {
  	struct inode *inode;
  	struct dentry *dentry;
  	struct ceph_vino vino;
  	int err;
  
  	dout("__fh_to_dentry %llx
  ", fh->ino);
  	vino.ino = fh->ino;
  	vino.snap = CEPH_NOSNAP;
  	inode = ceph_find_inode(sb, vino);
  	if (!inode)
  		return ERR_PTR(-ESTALE);
  
  	dentry = d_obtain_alias(inode);
0d509c949   Dan Carpenter   ceph: d_obtain_al...
103
  	if (IS_ERR(dentry)) {
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
104
105
106
107
  		pr_err("fh_to_dentry %llx -- inode %p but ENOMEM
  ",
  		       fh->ino, inode);
  		iput(inode);
0d509c949   Dan Carpenter   ceph: d_obtain_al...
108
  		return dentry;
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
  	}
  	err = ceph_init_dentry(dentry);
  
  	if (err < 0) {
  		iput(inode);
  		return ERR_PTR(err);
  	}
  	dout("__fh_to_dentry %llx %p dentry %p
  ", fh->ino, inode, dentry);
  	return dentry;
  }
  
  /*
   * convert connectable fh to dentry
   */
  static struct dentry *__cfh_to_dentry(struct super_block *sb,
  				      struct ceph_nfs_confh *cfh)
  {
640ef79d2   Cheng Renquan   ceph: use ceph_sb...
127
  	struct ceph_mds_client *mdsc = &ceph_sb_to_client(sb)->mdsc;
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
  	struct inode *inode;
  	struct dentry *dentry;
  	struct ceph_vino vino;
  	int err;
  
  	dout("__cfh_to_dentry %llx (%llx/%x)
  ",
  	     cfh->ino, cfh->parent_ino, cfh->parent_name_hash);
  
  	vino.ino = cfh->ino;
  	vino.snap = CEPH_NOSNAP;
  	inode = ceph_find_inode(sb, vino);
  	if (!inode) {
  		struct ceph_mds_request *req;
  
  		req = ceph_mdsc_create_request(mdsc, CEPH_MDS_OP_LOOKUPHASH,
  					       USE_ANY_MDS);
  		if (IS_ERR(req))
7e34bc524   Julia Lawall   fs/ceph: Use ERR_...
146
  			return ERR_CAST(req);
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
  
  		req->r_ino1 = vino;
  		req->r_ino2.ino = cfh->parent_ino;
  		req->r_ino2.snap = CEPH_NOSNAP;
  		req->r_path2 = kmalloc(16, GFP_NOFS);
  		snprintf(req->r_path2, 16, "%d", cfh->parent_name_hash);
  		req->r_num_caps = 1;
  		err = ceph_mdsc_do_request(mdsc, NULL, req);
  		ceph_mdsc_put_request(req);
  		inode = ceph_find_inode(sb, vino);
  		if (!inode)
  			return ERR_PTR(err ? err : -ESTALE);
  	}
  
  	dentry = d_obtain_alias(inode);
0d509c949   Dan Carpenter   ceph: d_obtain_al...
162
  	if (IS_ERR(dentry)) {
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
163
164
165
166
  		pr_err("cfh_to_dentry %llx -- inode %p but ENOMEM
  ",
  		       cfh->ino, inode);
  		iput(inode);
0d509c949   Dan Carpenter   ceph: d_obtain_al...
167
  		return dentry;
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
  	}
  	err = ceph_init_dentry(dentry);
  	if (err < 0) {
  		iput(inode);
  		return ERR_PTR(err);
  	}
  	dout("__cfh_to_dentry %llx %p dentry %p
  ", cfh->ino, inode, dentry);
  	return dentry;
  }
  
  static struct dentry *ceph_fh_to_dentry(struct super_block *sb, struct fid *fid,
  					int fh_len, int fh_type)
  {
  	if (fh_type == 1)
  		return __fh_to_dentry(sb, (struct ceph_nfs_fh *)fid->raw);
  	else
  		return __cfh_to_dentry(sb, (struct ceph_nfs_confh *)fid->raw);
  }
  
  /*
   * get parent, if possible.
   *
   * FIXME: we could do better by querying the mds to discover the
   * parent.
   */
  static struct dentry *ceph_fh_to_parent(struct super_block *sb,
  					 struct fid *fid,
  					int fh_len, int fh_type)
  {
  	struct ceph_nfs_confh *cfh = (void *)fid->raw;
  	struct ceph_vino vino;
  	struct inode *inode;
  	struct dentry *dentry;
  	int err;
  
  	if (fh_type == 1)
  		return ERR_PTR(-ESTALE);
  
  	pr_debug("fh_to_parent %llx/%d
  ", cfh->parent_ino,
  		 cfh->parent_name_hash);
  
  	vino.ino = cfh->ino;
  	vino.snap = CEPH_NOSNAP;
  	inode = ceph_find_inode(sb, vino);
  	if (!inode)
  		return ERR_PTR(-ESTALE);
  
  	dentry = d_obtain_alias(inode);
0d509c949   Dan Carpenter   ceph: d_obtain_al...
218
  	if (IS_ERR(dentry)) {
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
219
220
221
222
  		pr_err("fh_to_parent %llx -- inode %p but ENOMEM
  ",
  		       cfh->ino, inode);
  		iput(inode);
0d509c949   Dan Carpenter   ceph: d_obtain_al...
223
  		return dentry;
a8e63b7d5   Sage Weil   ceph: nfs re-expo...
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
  	}
  	err = ceph_init_dentry(dentry);
  	if (err < 0) {
  		iput(inode);
  		return ERR_PTR(err);
  	}
  	dout("fh_to_parent %llx %p dentry %p
  ", cfh->ino, inode, dentry);
  	return dentry;
  }
  
  const struct export_operations ceph_export_ops = {
  	.encode_fh = ceph_encode_fh,
  	.fh_to_dentry = ceph_fh_to_dentry,
  	.fh_to_parent = ceph_fh_to_parent,
  };