1 // SPDX-License-Identifier: GPL-2.0
2 #include <linux/ceph/ceph_debug.h>
4 #include <linux/exportfs.h>
5 #include <linux/slab.h>
6 #include <asm/unaligned.h>
9 #include "mds_client.h"
16 } __attribute__ ((packed
));
19 * Larger fh that includes parent ino.
21 struct ceph_nfs_confh
{
23 } __attribute__ ((packed
));
25 static int ceph_encode_fh(struct inode
*inode
, u32
*rawfh
, int *max_len
,
26 struct inode
*parent_inode
)
29 struct ceph_nfs_fh
*fh
= (void *)rawfh
;
30 struct ceph_nfs_confh
*cfh
= (void *)rawfh
;
31 int connected_handle_length
= sizeof(*cfh
)/4;
32 int handle_length
= sizeof(*fh
)/4;
34 /* don't re-export snaps */
35 if (ceph_snap(inode
) != CEPH_NOSNAP
)
38 if (parent_inode
&& (*max_len
< connected_handle_length
)) {
39 *max_len
= connected_handle_length
;
40 return FILEID_INVALID
;
41 } else if (*max_len
< handle_length
) {
42 *max_len
= handle_length
;
43 return FILEID_INVALID
;
47 dout("encode_fh %llx with parent %llx\n",
48 ceph_ino(inode
), ceph_ino(parent_inode
));
49 cfh
->ino
= ceph_ino(inode
);
50 cfh
->parent_ino
= ceph_ino(parent_inode
);
51 *max_len
= connected_handle_length
;
52 type
= FILEID_INO32_GEN_PARENT
;
54 dout("encode_fh %llx\n", ceph_ino(inode
));
55 fh
->ino
= ceph_ino(inode
);
56 *max_len
= handle_length
;
57 type
= FILEID_INO32_GEN
;
62 static struct dentry
*__fh_to_dentry(struct super_block
*sb
, u64 ino
)
64 struct ceph_mds_client
*mdsc
= ceph_sb_to_client(sb
)->mdsc
;
66 struct ceph_vino vino
;
70 vino
.snap
= CEPH_NOSNAP
;
71 inode
= ceph_find_inode(sb
, vino
);
73 struct ceph_mds_request
*req
;
76 req
= ceph_mdsc_create_request(mdsc
, CEPH_MDS_OP_LOOKUPINO
,
81 mask
= CEPH_STAT_CAP_INODE
;
82 if (ceph_security_xattr_wanted(d_inode(sb
->s_root
)))
83 mask
|= CEPH_CAP_XATTR_SHARED
;
84 req
->r_args
.getattr
.mask
= cpu_to_le32(mask
);
88 err
= ceph_mdsc_do_request(mdsc
, NULL
, req
);
89 inode
= req
->r_target_inode
;
92 ceph_mdsc_put_request(req
);
94 return ERR_PTR(-ESTALE
);
95 if (inode
->i_nlink
== 0) {
97 return ERR_PTR(-ESTALE
);
101 return d_obtain_alias(inode
);
105 * convert regular fh to dentry
107 static struct dentry
*ceph_fh_to_dentry(struct super_block
*sb
,
109 int fh_len
, int fh_type
)
111 struct ceph_nfs_fh
*fh
= (void *)fid
->raw
;
113 if (fh_type
!= FILEID_INO32_GEN
&&
114 fh_type
!= FILEID_INO32_GEN_PARENT
)
116 if (fh_len
< sizeof(*fh
) / 4)
119 dout("fh_to_dentry %llx\n", fh
->ino
);
120 return __fh_to_dentry(sb
, fh
->ino
);
123 static struct dentry
*__get_parent(struct super_block
*sb
,
124 struct dentry
*child
, u64 ino
)
126 struct ceph_mds_client
*mdsc
= ceph_sb_to_client(sb
)->mdsc
;
127 struct ceph_mds_request
*req
;
132 req
= ceph_mdsc_create_request(mdsc
, CEPH_MDS_OP_LOOKUPPARENT
,
135 return ERR_CAST(req
);
138 req
->r_inode
= d_inode(child
);
139 ihold(d_inode(child
));
141 req
->r_ino1
= (struct ceph_vino
) {
147 mask
= CEPH_STAT_CAP_INODE
;
148 if (ceph_security_xattr_wanted(d_inode(sb
->s_root
)))
149 mask
|= CEPH_CAP_XATTR_SHARED
;
150 req
->r_args
.getattr
.mask
= cpu_to_le32(mask
);
153 err
= ceph_mdsc_do_request(mdsc
, NULL
, req
);
155 ceph_mdsc_put_request(req
);
159 inode
= req
->r_target_inode
;
162 ceph_mdsc_put_request(req
);
164 return ERR_PTR(-ENOENT
);
166 return d_obtain_alias(inode
);
169 static struct dentry
*ceph_get_parent(struct dentry
*child
)
171 /* don't re-export snaps */
172 if (ceph_snap(d_inode(child
)) != CEPH_NOSNAP
)
173 return ERR_PTR(-EINVAL
);
175 dout("get_parent %p ino %llx.%llx\n",
176 child
, ceph_vinop(d_inode(child
)));
177 return __get_parent(child
->d_sb
, child
, 0);
181 * convert regular fh to parent
183 static struct dentry
*ceph_fh_to_parent(struct super_block
*sb
,
185 int fh_len
, int fh_type
)
187 struct ceph_nfs_confh
*cfh
= (void *)fid
->raw
;
188 struct dentry
*dentry
;
190 if (fh_type
!= FILEID_INO32_GEN_PARENT
)
192 if (fh_len
< sizeof(*cfh
) / 4)
195 dout("fh_to_parent %llx\n", cfh
->parent_ino
);
196 dentry
= __get_parent(sb
, NULL
, cfh
->ino
);
197 if (unlikely(dentry
== ERR_PTR(-ENOENT
)))
198 dentry
= __fh_to_dentry(sb
, cfh
->parent_ino
);
202 static int ceph_get_name(struct dentry
*parent
, char *name
,
203 struct dentry
*child
)
205 struct ceph_mds_client
*mdsc
;
206 struct ceph_mds_request
*req
;
209 mdsc
= ceph_inode_to_client(d_inode(child
))->mdsc
;
210 req
= ceph_mdsc_create_request(mdsc
, CEPH_MDS_OP_LOOKUPNAME
,
215 inode_lock(d_inode(parent
));
217 req
->r_inode
= d_inode(child
);
218 ihold(d_inode(child
));
219 req
->r_ino2
= ceph_vino(d_inode(parent
));
220 req
->r_parent
= d_inode(parent
);
221 set_bit(CEPH_MDS_R_PARENT_LOCKED
, &req
->r_req_flags
);
223 err
= ceph_mdsc_do_request(mdsc
, NULL
, req
);
225 inode_unlock(d_inode(parent
));
228 struct ceph_mds_reply_info_parsed
*rinfo
= &req
->r_reply_info
;
229 memcpy(name
, rinfo
->dname
, rinfo
->dname_len
);
230 name
[rinfo
->dname_len
] = 0;
231 dout("get_name %p ino %llx.%llx name %s\n",
232 child
, ceph_vinop(d_inode(child
)), name
);
234 dout("get_name %p ino %llx.%llx err %d\n",
235 child
, ceph_vinop(d_inode(child
)), err
);
238 ceph_mdsc_put_request(req
);
242 const struct export_operations ceph_export_ops
= {
243 .encode_fh
= ceph_encode_fh
,
244 .fh_to_dentry
= ceph_fh_to_dentry
,
245 .fh_to_parent
= ceph_fh_to_parent
,
246 .get_parent
= ceph_get_parent
,
247 .get_name
= ceph_get_name
,