1db0f1c07SPaulo Alcantara // SPDX-License-Identifier: GPL-2.0-only
2db0f1c07SPaulo Alcantara /*
3db0f1c07SPaulo Alcantara * Copyright (c) 2024 Paulo Alcantara <pc@manguebit.com>
4db0f1c07SPaulo Alcantara */
5db0f1c07SPaulo Alcantara
6db0f1c07SPaulo Alcantara #include <linux/fs.h>
7db0f1c07SPaulo Alcantara #include <linux/stat.h>
8db0f1c07SPaulo Alcantara #include <linux/slab.h>
9db0f1c07SPaulo Alcantara #include "cifsglob.h"
10db0f1c07SPaulo Alcantara #include "smb2proto.h"
11db0f1c07SPaulo Alcantara #include "cifsproto.h"
12db0f1c07SPaulo Alcantara #include "cifs_unicode.h"
13db0f1c07SPaulo Alcantara #include "cifs_debug.h"
14df0a8a19SPaulo Alcantara #include "fs_context.h"
15db0f1c07SPaulo Alcantara #include "reparse.h"
16db0f1c07SPaulo Alcantara
17201430d2SPali Rohár static int detect_directory_symlink_target(struct cifs_sb_info *cifs_sb,
18201430d2SPali Rohár const unsigned int xid,
19201430d2SPali Rohár const char *full_path,
20201430d2SPali Rohár const char *symname,
21201430d2SPali Rohár bool *directory);
22201430d2SPali Rohár
smb2_create_reparse_symlink(const unsigned int xid,struct inode * inode,struct dentry * dentry,struct cifs_tcon * tcon,const char * full_path,const char * symname)23db0f1c07SPaulo Alcantara int smb2_create_reparse_symlink(const unsigned int xid, struct inode *inode,
24db0f1c07SPaulo Alcantara struct dentry *dentry, struct cifs_tcon *tcon,
25db0f1c07SPaulo Alcantara const char *full_path, const char *symname)
26db0f1c07SPaulo Alcantara {
27db0f1c07SPaulo Alcantara struct reparse_symlink_data_buffer *buf = NULL;
28db0f1c07SPaulo Alcantara struct cifs_open_info_data data;
29db0f1c07SPaulo Alcantara struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
30db0f1c07SPaulo Alcantara struct inode *new;
31db0f1c07SPaulo Alcantara struct kvec iov;
32db0f1c07SPaulo Alcantara __le16 *path;
33201430d2SPali Rohár bool directory;
34db0f1c07SPaulo Alcantara char *sym, sep = CIFS_DIR_SEP(cifs_sb);
35db0f1c07SPaulo Alcantara u16 len, plen;
36db0f1c07SPaulo Alcantara int rc = 0;
37db0f1c07SPaulo Alcantara
38921d9090SPaulo Alcantara if (strlen(symname) > REPARSE_SYM_PATH_MAX)
39921d9090SPaulo Alcantara return -ENAMETOOLONG;
40921d9090SPaulo Alcantara
41db0f1c07SPaulo Alcantara sym = kstrdup(symname, GFP_KERNEL);
42db0f1c07SPaulo Alcantara if (!sym)
43db0f1c07SPaulo Alcantara return -ENOMEM;
44db0f1c07SPaulo Alcantara
45db0f1c07SPaulo Alcantara data = (struct cifs_open_info_data) {
46db0f1c07SPaulo Alcantara .reparse_point = true,
47db0f1c07SPaulo Alcantara .reparse = { .tag = IO_REPARSE_TAG_SYMLINK, },
48db0f1c07SPaulo Alcantara .symlink_target = sym,
49db0f1c07SPaulo Alcantara };
50db0f1c07SPaulo Alcantara
51db0f1c07SPaulo Alcantara convert_delimiter(sym, sep);
52db0f1c07SPaulo Alcantara path = cifs_convert_path_to_utf16(sym, cifs_sb);
53db0f1c07SPaulo Alcantara if (!path) {
54db0f1c07SPaulo Alcantara rc = -ENOMEM;
55db0f1c07SPaulo Alcantara goto out;
56db0f1c07SPaulo Alcantara }
57db0f1c07SPaulo Alcantara
58201430d2SPali Rohár /*
59201430d2SPali Rohár * SMB distinguish between symlink to directory and symlink to file.
60201430d2SPali Rohár * They cannot be exchanged (symlink of file type which points to
61201430d2SPali Rohár * directory cannot be resolved and vice-versa). Try to detect if
62201430d2SPali Rohár * the symlink target could be a directory or not. When detection
63201430d2SPali Rohár * fails then treat symlink as a file (non-directory) symlink.
64201430d2SPali Rohár */
65201430d2SPali Rohár directory = false;
66201430d2SPali Rohár rc = detect_directory_symlink_target(cifs_sb, xid, full_path, symname, &directory);
67201430d2SPali Rohár if (rc < 0)
68201430d2SPali Rohár goto out;
69201430d2SPali Rohár
70921d9090SPaulo Alcantara plen = 2 * UniStrnlen((wchar_t *)path, REPARSE_SYM_PATH_MAX);
71db0f1c07SPaulo Alcantara len = sizeof(*buf) + plen * 2;
72db0f1c07SPaulo Alcantara buf = kzalloc(len, GFP_KERNEL);
73db0f1c07SPaulo Alcantara if (!buf) {
74db0f1c07SPaulo Alcantara rc = -ENOMEM;
75db0f1c07SPaulo Alcantara goto out;
76db0f1c07SPaulo Alcantara }
77db0f1c07SPaulo Alcantara
78db0f1c07SPaulo Alcantara buf->ReparseTag = cpu_to_le32(IO_REPARSE_TAG_SYMLINK);
79db0f1c07SPaulo Alcantara buf->ReparseDataLength = cpu_to_le16(len - sizeof(struct reparse_data_buffer));
80db0f1c07SPaulo Alcantara buf->SubstituteNameOffset = cpu_to_le16(plen);
81db0f1c07SPaulo Alcantara buf->SubstituteNameLength = cpu_to_le16(plen);
82db0f1c07SPaulo Alcantara memcpy(&buf->PathBuffer[plen], path, plen);
83db0f1c07SPaulo Alcantara buf->PrintNameOffset = 0;
84db0f1c07SPaulo Alcantara buf->PrintNameLength = cpu_to_le16(plen);
85db0f1c07SPaulo Alcantara memcpy(buf->PathBuffer, path, plen);
86db0f1c07SPaulo Alcantara buf->Flags = cpu_to_le32(*symname != '/' ? SYMLINK_FLAG_RELATIVE : 0);
87db0f1c07SPaulo Alcantara if (*sym != sep)
88db0f1c07SPaulo Alcantara buf->Flags = cpu_to_le32(SYMLINK_FLAG_RELATIVE);
89db0f1c07SPaulo Alcantara
90db0f1c07SPaulo Alcantara convert_delimiter(sym, '/');
91db0f1c07SPaulo Alcantara iov.iov_base = buf;
92db0f1c07SPaulo Alcantara iov.iov_len = len;
93db0f1c07SPaulo Alcantara new = smb2_get_reparse_inode(&data, inode->i_sb, xid,
94201430d2SPali Rohár tcon, full_path, directory,
95201430d2SPali Rohár &iov, NULL);
96db0f1c07SPaulo Alcantara if (!IS_ERR(new))
97db0f1c07SPaulo Alcantara d_instantiate(dentry, new);
98db0f1c07SPaulo Alcantara else
99db0f1c07SPaulo Alcantara rc = PTR_ERR(new);
100db0f1c07SPaulo Alcantara out:
101db0f1c07SPaulo Alcantara kfree(path);
102db0f1c07SPaulo Alcantara cifs_free_open_info(&data);
103db0f1c07SPaulo Alcantara kfree(buf);
104db0f1c07SPaulo Alcantara return rc;
105db0f1c07SPaulo Alcantara }
106db0f1c07SPaulo Alcantara
detect_directory_symlink_target(struct cifs_sb_info * cifs_sb,const unsigned int xid,const char * full_path,const char * symname,bool * directory)107201430d2SPali Rohár static int detect_directory_symlink_target(struct cifs_sb_info *cifs_sb,
108201430d2SPali Rohár const unsigned int xid,
109201430d2SPali Rohár const char *full_path,
110201430d2SPali Rohár const char *symname,
111201430d2SPali Rohár bool *directory)
112201430d2SPali Rohár {
113201430d2SPali Rohár char sep = CIFS_DIR_SEP(cifs_sb);
114201430d2SPali Rohár struct cifs_open_parms oparms;
115201430d2SPali Rohár struct tcon_link *tlink;
116201430d2SPali Rohár struct cifs_tcon *tcon;
117201430d2SPali Rohár const char *basename;
118201430d2SPali Rohár struct cifs_fid fid;
119201430d2SPali Rohár char *resolved_path;
120201430d2SPali Rohár int full_path_len;
121201430d2SPali Rohár int basename_len;
122201430d2SPali Rohár int symname_len;
123201430d2SPali Rohár char *path_sep;
124201430d2SPali Rohár __u32 oplock;
125201430d2SPali Rohár int open_rc;
126201430d2SPali Rohár
127201430d2SPali Rohár /*
128201430d2SPali Rohár * First do some simple check. If the original Linux symlink target ends
129201430d2SPali Rohár * with slash, or last path component is dot or dot-dot then it is for
130201430d2SPali Rohár * sure symlink to the directory.
131201430d2SPali Rohár */
132201430d2SPali Rohár basename = kbasename(symname);
133201430d2SPali Rohár basename_len = strlen(basename);
134201430d2SPali Rohár if (basename_len == 0 || /* symname ends with slash */
135201430d2SPali Rohár (basename_len == 1 && basename[0] == '.') || /* last component is "." */
136201430d2SPali Rohár (basename_len == 2 && basename[0] == '.' && basename[1] == '.')) { /* or ".." */
137201430d2SPali Rohár *directory = true;
138201430d2SPali Rohár return 0;
139201430d2SPali Rohár }
140201430d2SPali Rohár
141201430d2SPali Rohár /*
142201430d2SPali Rohár * For absolute symlinks it is not possible to determinate
143201430d2SPali Rohár * if it should point to directory or file.
144201430d2SPali Rohár */
145201430d2SPali Rohár if (symname[0] == '/') {
146201430d2SPali Rohár cifs_dbg(FYI,
147201430d2SPali Rohár "%s: cannot determinate if the symlink target path '%s' "
148201430d2SPali Rohár "is directory or not, creating '%s' as file symlink\n",
149201430d2SPali Rohár __func__, symname, full_path);
150201430d2SPali Rohár return 0;
151201430d2SPali Rohár }
152201430d2SPali Rohár
153201430d2SPali Rohár /*
154201430d2SPali Rohár * If it was not detected as directory yet and the symlink is relative
155201430d2SPali Rohár * then try to resolve the path on the SMB server, check if the path
156201430d2SPali Rohár * exists and determinate if it is a directory or not.
157201430d2SPali Rohár */
158201430d2SPali Rohár
159201430d2SPali Rohár full_path_len = strlen(full_path);
160201430d2SPali Rohár symname_len = strlen(symname);
161201430d2SPali Rohár
162201430d2SPali Rohár tlink = cifs_sb_tlink(cifs_sb);
163201430d2SPali Rohár if (IS_ERR(tlink))
164201430d2SPali Rohár return PTR_ERR(tlink);
165201430d2SPali Rohár
166201430d2SPali Rohár resolved_path = kzalloc(full_path_len + symname_len + 1, GFP_KERNEL);
167201430d2SPali Rohár if (!resolved_path) {
168201430d2SPali Rohár cifs_put_tlink(tlink);
169201430d2SPali Rohár return -ENOMEM;
170201430d2SPali Rohár }
171201430d2SPali Rohár
172201430d2SPali Rohár /*
173201430d2SPali Rohár * Compose the resolved SMB symlink path from the SMB full path
174201430d2SPali Rohár * and Linux target symlink path.
175201430d2SPali Rohár */
176201430d2SPali Rohár memcpy(resolved_path, full_path, full_path_len+1);
177201430d2SPali Rohár path_sep = strrchr(resolved_path, sep);
178201430d2SPali Rohár if (path_sep)
179201430d2SPali Rohár path_sep++;
180201430d2SPali Rohár else
181201430d2SPali Rohár path_sep = resolved_path;
182201430d2SPali Rohár memcpy(path_sep, symname, symname_len+1);
183201430d2SPali Rohár if (sep == '\\')
184201430d2SPali Rohár convert_delimiter(path_sep, sep);
185201430d2SPali Rohár
186201430d2SPali Rohár tcon = tlink_tcon(tlink);
187201430d2SPali Rohár oparms = CIFS_OPARMS(cifs_sb, tcon, resolved_path,
188201430d2SPali Rohár FILE_READ_ATTRIBUTES, FILE_OPEN, 0, ACL_NO_MODE);
189201430d2SPali Rohár oparms.fid = &fid;
190201430d2SPali Rohár
191201430d2SPali Rohár /* Try to open as a directory (NOT_FILE) */
192201430d2SPali Rohár oplock = 0;
193201430d2SPali Rohár oparms.create_options = cifs_create_options(cifs_sb,
194201430d2SPali Rohár CREATE_NOT_FILE | OPEN_REPARSE_POINT);
195201430d2SPali Rohár open_rc = tcon->ses->server->ops->open(xid, &oparms, &oplock, NULL);
196201430d2SPali Rohár if (open_rc == 0) {
197201430d2SPali Rohár /* Successful open means that the target path is definitely a directory. */
198201430d2SPali Rohár *directory = true;
199201430d2SPali Rohár tcon->ses->server->ops->close(xid, tcon, &fid);
200201430d2SPali Rohár } else if (open_rc == -ENOTDIR) {
201201430d2SPali Rohár /* -ENOTDIR means that the target path is definitely a file. */
202201430d2SPali Rohár *directory = false;
203201430d2SPali Rohár } else if (open_rc == -ENOENT) {
204201430d2SPali Rohár /* -ENOENT means that the target path does not exist. */
205201430d2SPali Rohár cifs_dbg(FYI,
206201430d2SPali Rohár "%s: symlink target path '%s' does not exist, "
207201430d2SPali Rohár "creating '%s' as file symlink\n",
208201430d2SPali Rohár __func__, symname, full_path);
209201430d2SPali Rohár } else {
210201430d2SPali Rohár /* Try to open as a file (NOT_DIR) */
211201430d2SPali Rohár oplock = 0;
212201430d2SPali Rohár oparms.create_options = cifs_create_options(cifs_sb,
213201430d2SPali Rohár CREATE_NOT_DIR | OPEN_REPARSE_POINT);
214201430d2SPali Rohár open_rc = tcon->ses->server->ops->open(xid, &oparms, &oplock, NULL);
215201430d2SPali Rohár if (open_rc == 0) {
216201430d2SPali Rohár /* Successful open means that the target path is definitely a file. */
217201430d2SPali Rohár *directory = false;
218201430d2SPali Rohár tcon->ses->server->ops->close(xid, tcon, &fid);
219201430d2SPali Rohár } else if (open_rc == -EISDIR) {
220201430d2SPali Rohár /* -EISDIR means that the target path is definitely a directory. */
221201430d2SPali Rohár *directory = true;
222201430d2SPali Rohár } else {
223201430d2SPali Rohár /*
224201430d2SPali Rohár * This code branch is called when we do not have a permission to
225201430d2SPali Rohár * open the resolved_path or some other client/process denied
226201430d2SPali Rohár * opening the resolved_path.
227201430d2SPali Rohár *
228201430d2SPali Rohár * TODO: Try to use ops->query_dir_first on the parent directory
229201430d2SPali Rohár * of resolved_path, search for basename of resolved_path and
230201430d2SPali Rohár * check if the ATTR_DIRECTORY is set in fi.Attributes. In some
231201430d2SPali Rohár * case this could work also when opening of the path is denied.
232201430d2SPali Rohár */
233201430d2SPali Rohár cifs_dbg(FYI,
234201430d2SPali Rohár "%s: cannot determinate if the symlink target path '%s' "
235201430d2SPali Rohár "is directory or not, creating '%s' as file symlink\n",
236201430d2SPali Rohár __func__, symname, full_path);
237201430d2SPali Rohár }
238201430d2SPali Rohár }
239201430d2SPali Rohár
240201430d2SPali Rohár kfree(resolved_path);
241201430d2SPali Rohár cifs_put_tlink(tlink);
242201430d2SPali Rohár return 0;
243201430d2SPali Rohár }
244201430d2SPali Rohár
nfs_set_reparse_buf(struct reparse_posix_data * buf,mode_t mode,dev_t dev,struct kvec * iov)245db0f1c07SPaulo Alcantara static int nfs_set_reparse_buf(struct reparse_posix_data *buf,
246db0f1c07SPaulo Alcantara mode_t mode, dev_t dev,
247db0f1c07SPaulo Alcantara struct kvec *iov)
248db0f1c07SPaulo Alcantara {
249db0f1c07SPaulo Alcantara u64 type;
250db0f1c07SPaulo Alcantara u16 len, dlen;
251db0f1c07SPaulo Alcantara
252db0f1c07SPaulo Alcantara len = sizeof(*buf);
253db0f1c07SPaulo Alcantara
254db0f1c07SPaulo Alcantara switch ((type = reparse_mode_nfs_type(mode))) {
255db0f1c07SPaulo Alcantara case NFS_SPECFILE_BLK:
256db0f1c07SPaulo Alcantara case NFS_SPECFILE_CHR:
257db0f1c07SPaulo Alcantara dlen = sizeof(__le64);
258db0f1c07SPaulo Alcantara break;
259db0f1c07SPaulo Alcantara case NFS_SPECFILE_FIFO:
260db0f1c07SPaulo Alcantara case NFS_SPECFILE_SOCK:
261db0f1c07SPaulo Alcantara dlen = 0;
262db0f1c07SPaulo Alcantara break;
263db0f1c07SPaulo Alcantara default:
264db0f1c07SPaulo Alcantara return -EOPNOTSUPP;
265db0f1c07SPaulo Alcantara }
266db0f1c07SPaulo Alcantara
267db0f1c07SPaulo Alcantara buf->ReparseTag = cpu_to_le32(IO_REPARSE_TAG_NFS);
268db0f1c07SPaulo Alcantara buf->Reserved = 0;
269db0f1c07SPaulo Alcantara buf->InodeType = cpu_to_le64(type);
270db0f1c07SPaulo Alcantara buf->ReparseDataLength = cpu_to_le16(len + dlen -
271db0f1c07SPaulo Alcantara sizeof(struct reparse_data_buffer));
2727013af5bSPaulo Alcantara *(__le64 *)buf->DataBuffer = cpu_to_le64(((u64)MINOR(dev) << 32) |
2737013af5bSPaulo Alcantara MAJOR(dev));
274db0f1c07SPaulo Alcantara iov->iov_base = buf;
275db0f1c07SPaulo Alcantara iov->iov_len = len + dlen;
276db0f1c07SPaulo Alcantara return 0;
277db0f1c07SPaulo Alcantara }
278db0f1c07SPaulo Alcantara
mknod_nfs(unsigned int xid,struct inode * inode,struct dentry * dentry,struct cifs_tcon * tcon,const char * full_path,umode_t mode,dev_t dev)279df0a8a19SPaulo Alcantara static int mknod_nfs(unsigned int xid, struct inode *inode,
280db0f1c07SPaulo Alcantara struct dentry *dentry, struct cifs_tcon *tcon,
281db0f1c07SPaulo Alcantara const char *full_path, umode_t mode, dev_t dev)
282db0f1c07SPaulo Alcantara {
283db0f1c07SPaulo Alcantara struct cifs_open_info_data data;
284db0f1c07SPaulo Alcantara struct reparse_posix_data *p;
285db0f1c07SPaulo Alcantara struct inode *new;
286db0f1c07SPaulo Alcantara struct kvec iov;
287db0f1c07SPaulo Alcantara __u8 buf[sizeof(*p) + sizeof(__le64)];
288db0f1c07SPaulo Alcantara int rc;
289db0f1c07SPaulo Alcantara
290db0f1c07SPaulo Alcantara p = (struct reparse_posix_data *)buf;
291db0f1c07SPaulo Alcantara rc = nfs_set_reparse_buf(p, mode, dev, &iov);
292db0f1c07SPaulo Alcantara if (rc)
293db0f1c07SPaulo Alcantara return rc;
294db0f1c07SPaulo Alcantara
295db0f1c07SPaulo Alcantara data = (struct cifs_open_info_data) {
296db0f1c07SPaulo Alcantara .reparse_point = true,
297db0f1c07SPaulo Alcantara .reparse = { .tag = IO_REPARSE_TAG_NFS, .posix = p, },
298db0f1c07SPaulo Alcantara };
299db0f1c07SPaulo Alcantara
300db0f1c07SPaulo Alcantara new = smb2_get_reparse_inode(&data, inode->i_sb, xid,
301201430d2SPali Rohár tcon, full_path, false, &iov, NULL);
302db0f1c07SPaulo Alcantara if (!IS_ERR(new))
303db0f1c07SPaulo Alcantara d_instantiate(dentry, new);
304db0f1c07SPaulo Alcantara else
305db0f1c07SPaulo Alcantara rc = PTR_ERR(new);
306db0f1c07SPaulo Alcantara cifs_free_open_info(&data);
307db0f1c07SPaulo Alcantara return rc;
308db0f1c07SPaulo Alcantara }
309db0f1c07SPaulo Alcantara
wsl_set_reparse_buf(struct reparse_data_buffer * buf,mode_t mode,struct kvec * iov)310df0a8a19SPaulo Alcantara static int wsl_set_reparse_buf(struct reparse_data_buffer *buf,
311df0a8a19SPaulo Alcantara mode_t mode, struct kvec *iov)
312df0a8a19SPaulo Alcantara {
313df0a8a19SPaulo Alcantara u32 tag;
314df0a8a19SPaulo Alcantara
315df0a8a19SPaulo Alcantara switch ((tag = reparse_mode_wsl_tag(mode))) {
316df0a8a19SPaulo Alcantara case IO_REPARSE_TAG_LX_BLK:
317df0a8a19SPaulo Alcantara case IO_REPARSE_TAG_LX_CHR:
318df0a8a19SPaulo Alcantara case IO_REPARSE_TAG_LX_FIFO:
319df0a8a19SPaulo Alcantara case IO_REPARSE_TAG_AF_UNIX:
320df0a8a19SPaulo Alcantara break;
321df0a8a19SPaulo Alcantara default:
322df0a8a19SPaulo Alcantara return -EOPNOTSUPP;
323df0a8a19SPaulo Alcantara }
324df0a8a19SPaulo Alcantara
325df0a8a19SPaulo Alcantara buf->ReparseTag = cpu_to_le32(tag);
326df0a8a19SPaulo Alcantara buf->Reserved = 0;
327df0a8a19SPaulo Alcantara buf->ReparseDataLength = 0;
328df0a8a19SPaulo Alcantara iov->iov_base = buf;
329df0a8a19SPaulo Alcantara iov->iov_len = sizeof(*buf);
330df0a8a19SPaulo Alcantara return 0;
331df0a8a19SPaulo Alcantara }
332df0a8a19SPaulo Alcantara
ea_create_context(u32 dlen,size_t * cc_len)333df0a8a19SPaulo Alcantara static struct smb2_create_ea_ctx *ea_create_context(u32 dlen, size_t *cc_len)
334df0a8a19SPaulo Alcantara {
335df0a8a19SPaulo Alcantara struct smb2_create_ea_ctx *cc;
336df0a8a19SPaulo Alcantara
337df0a8a19SPaulo Alcantara *cc_len = round_up(sizeof(*cc) + dlen, 8);
338df0a8a19SPaulo Alcantara cc = kzalloc(*cc_len, GFP_KERNEL);
339df0a8a19SPaulo Alcantara if (!cc)
340df0a8a19SPaulo Alcantara return ERR_PTR(-ENOMEM);
341df0a8a19SPaulo Alcantara
342df0a8a19SPaulo Alcantara cc->ctx.NameOffset = cpu_to_le16(offsetof(struct smb2_create_ea_ctx,
343df0a8a19SPaulo Alcantara name));
344df0a8a19SPaulo Alcantara cc->ctx.NameLength = cpu_to_le16(4);
345df0a8a19SPaulo Alcantara memcpy(cc->name, SMB2_CREATE_EA_BUFFER, strlen(SMB2_CREATE_EA_BUFFER));
346df0a8a19SPaulo Alcantara cc->ctx.DataOffset = cpu_to_le16(offsetof(struct smb2_create_ea_ctx, ea));
347df0a8a19SPaulo Alcantara cc->ctx.DataLength = cpu_to_le32(dlen);
348df0a8a19SPaulo Alcantara return cc;
349df0a8a19SPaulo Alcantara }
350df0a8a19SPaulo Alcantara
351df0a8a19SPaulo Alcantara struct wsl_xattr {
352df0a8a19SPaulo Alcantara const char *name;
353df0a8a19SPaulo Alcantara __le64 value;
354df0a8a19SPaulo Alcantara u16 size;
355df0a8a19SPaulo Alcantara u32 next;
356df0a8a19SPaulo Alcantara };
357df0a8a19SPaulo Alcantara
wsl_set_xattrs(struct inode * inode,umode_t _mode,dev_t _dev,struct kvec * iov)358df0a8a19SPaulo Alcantara static int wsl_set_xattrs(struct inode *inode, umode_t _mode,
359df0a8a19SPaulo Alcantara dev_t _dev, struct kvec *iov)
360df0a8a19SPaulo Alcantara {
361df0a8a19SPaulo Alcantara struct smb2_file_full_ea_info *ea;
362df0a8a19SPaulo Alcantara struct smb2_create_ea_ctx *cc;
363df0a8a19SPaulo Alcantara struct smb3_fs_context *ctx = CIFS_SB(inode->i_sb)->ctx;
364df0a8a19SPaulo Alcantara __le64 uid = cpu_to_le64(from_kuid(current_user_ns(), ctx->linux_uid));
365df0a8a19SPaulo Alcantara __le64 gid = cpu_to_le64(from_kgid(current_user_ns(), ctx->linux_gid));
366df0a8a19SPaulo Alcantara __le64 dev = cpu_to_le64(((u64)MINOR(_dev) << 32) | MAJOR(_dev));
367df0a8a19SPaulo Alcantara __le64 mode = cpu_to_le64(_mode);
368df0a8a19SPaulo Alcantara struct wsl_xattr xattrs[] = {
3697449d736SSteve French { .name = SMB2_WSL_XATTR_UID, .value = uid, .size = SMB2_WSL_XATTR_UID_SIZE, },
3707449d736SSteve French { .name = SMB2_WSL_XATTR_GID, .value = gid, .size = SMB2_WSL_XATTR_GID_SIZE, },
3717449d736SSteve French { .name = SMB2_WSL_XATTR_MODE, .value = mode, .size = SMB2_WSL_XATTR_MODE_SIZE, },
3727449d736SSteve French { .name = SMB2_WSL_XATTR_DEV, .value = dev, .size = SMB2_WSL_XATTR_DEV_SIZE, },
373df0a8a19SPaulo Alcantara };
374df0a8a19SPaulo Alcantara size_t cc_len;
375df0a8a19SPaulo Alcantara u32 dlen = 0, next = 0;
376df0a8a19SPaulo Alcantara int i, num_xattrs;
3777449d736SSteve French u8 name_size = SMB2_WSL_XATTR_NAME_LEN + 1;
378df0a8a19SPaulo Alcantara
379df0a8a19SPaulo Alcantara memset(iov, 0, sizeof(*iov));
380df0a8a19SPaulo Alcantara
381df0a8a19SPaulo Alcantara /* Exclude $LXDEV xattr for sockets and fifos */
382df0a8a19SPaulo Alcantara if (S_ISSOCK(_mode) || S_ISFIFO(_mode))
383df0a8a19SPaulo Alcantara num_xattrs = ARRAY_SIZE(xattrs) - 1;
384df0a8a19SPaulo Alcantara else
385df0a8a19SPaulo Alcantara num_xattrs = ARRAY_SIZE(xattrs);
386df0a8a19SPaulo Alcantara
387df0a8a19SPaulo Alcantara for (i = 0; i < num_xattrs; i++) {
388df0a8a19SPaulo Alcantara xattrs[i].next = ALIGN(sizeof(*ea) + name_size +
389df0a8a19SPaulo Alcantara xattrs[i].size, 4);
390df0a8a19SPaulo Alcantara dlen += xattrs[i].next;
391df0a8a19SPaulo Alcantara }
392df0a8a19SPaulo Alcantara
393df0a8a19SPaulo Alcantara cc = ea_create_context(dlen, &cc_len);
394b93ac66fSDan Carpenter if (IS_ERR(cc))
395df0a8a19SPaulo Alcantara return PTR_ERR(cc);
396df0a8a19SPaulo Alcantara
397df0a8a19SPaulo Alcantara ea = &cc->ea;
398df0a8a19SPaulo Alcantara for (i = 0; i < num_xattrs; i++) {
399df0a8a19SPaulo Alcantara ea = (void *)((u8 *)ea + next);
400df0a8a19SPaulo Alcantara next = xattrs[i].next;
401df0a8a19SPaulo Alcantara ea->next_entry_offset = cpu_to_le32(next);
402df0a8a19SPaulo Alcantara
403df0a8a19SPaulo Alcantara ea->ea_name_length = name_size - 1;
404df0a8a19SPaulo Alcantara ea->ea_value_length = cpu_to_le16(xattrs[i].size);
405df0a8a19SPaulo Alcantara memcpy(ea->ea_data, xattrs[i].name, name_size);
406df0a8a19SPaulo Alcantara memcpy(&ea->ea_data[name_size],
407df0a8a19SPaulo Alcantara &xattrs[i].value, xattrs[i].size);
408df0a8a19SPaulo Alcantara }
409df0a8a19SPaulo Alcantara ea->next_entry_offset = 0;
410df0a8a19SPaulo Alcantara
411df0a8a19SPaulo Alcantara iov->iov_base = cc;
412df0a8a19SPaulo Alcantara iov->iov_len = cc_len;
413df0a8a19SPaulo Alcantara return 0;
414df0a8a19SPaulo Alcantara }
415df0a8a19SPaulo Alcantara
mknod_wsl(unsigned int xid,struct inode * inode,struct dentry * dentry,struct cifs_tcon * tcon,const char * full_path,umode_t mode,dev_t dev)416df0a8a19SPaulo Alcantara static int mknod_wsl(unsigned int xid, struct inode *inode,
417df0a8a19SPaulo Alcantara struct dentry *dentry, struct cifs_tcon *tcon,
418df0a8a19SPaulo Alcantara const char *full_path, umode_t mode, dev_t dev)
419df0a8a19SPaulo Alcantara {
420df0a8a19SPaulo Alcantara struct cifs_open_info_data data;
421df0a8a19SPaulo Alcantara struct reparse_data_buffer buf;
42297db4160SPaulo Alcantara struct smb2_create_ea_ctx *cc;
423df0a8a19SPaulo Alcantara struct inode *new;
42497db4160SPaulo Alcantara unsigned int len;
425df0a8a19SPaulo Alcantara struct kvec reparse_iov, xattr_iov;
426df0a8a19SPaulo Alcantara int rc;
427df0a8a19SPaulo Alcantara
428df0a8a19SPaulo Alcantara rc = wsl_set_reparse_buf(&buf, mode, &reparse_iov);
429df0a8a19SPaulo Alcantara if (rc)
430df0a8a19SPaulo Alcantara return rc;
431df0a8a19SPaulo Alcantara
432df0a8a19SPaulo Alcantara rc = wsl_set_xattrs(inode, mode, dev, &xattr_iov);
433df0a8a19SPaulo Alcantara if (rc)
434df0a8a19SPaulo Alcantara return rc;
435df0a8a19SPaulo Alcantara
436df0a8a19SPaulo Alcantara data = (struct cifs_open_info_data) {
437df0a8a19SPaulo Alcantara .reparse_point = true,
438df0a8a19SPaulo Alcantara .reparse = { .tag = le32_to_cpu(buf.ReparseTag), .buf = &buf, },
439df0a8a19SPaulo Alcantara };
440df0a8a19SPaulo Alcantara
44197db4160SPaulo Alcantara cc = xattr_iov.iov_base;
44297db4160SPaulo Alcantara len = le32_to_cpu(cc->ctx.DataLength);
44397db4160SPaulo Alcantara memcpy(data.wsl.eas, &cc->ea, len);
44497db4160SPaulo Alcantara data.wsl.eas_len = len;
44597db4160SPaulo Alcantara
446df0a8a19SPaulo Alcantara new = smb2_get_reparse_inode(&data, inode->i_sb,
447201430d2SPali Rohár xid, tcon, full_path, false,
448df0a8a19SPaulo Alcantara &reparse_iov, &xattr_iov);
449df0a8a19SPaulo Alcantara if (!IS_ERR(new))
450df0a8a19SPaulo Alcantara d_instantiate(dentry, new);
451df0a8a19SPaulo Alcantara else
452df0a8a19SPaulo Alcantara rc = PTR_ERR(new);
453df0a8a19SPaulo Alcantara cifs_free_open_info(&data);
454df0a8a19SPaulo Alcantara kfree(xattr_iov.iov_base);
455df0a8a19SPaulo Alcantara return rc;
456df0a8a19SPaulo Alcantara }
457df0a8a19SPaulo Alcantara
smb2_mknod_reparse(unsigned int xid,struct inode * inode,struct dentry * dentry,struct cifs_tcon * tcon,const char * full_path,umode_t mode,dev_t dev)458df0a8a19SPaulo Alcantara int smb2_mknod_reparse(unsigned int xid, struct inode *inode,
459df0a8a19SPaulo Alcantara struct dentry *dentry, struct cifs_tcon *tcon,
460df0a8a19SPaulo Alcantara const char *full_path, umode_t mode, dev_t dev)
461df0a8a19SPaulo Alcantara {
462df0a8a19SPaulo Alcantara struct smb3_fs_context *ctx = CIFS_SB(inode->i_sb)->ctx;
463df0a8a19SPaulo Alcantara int rc = -EOPNOTSUPP;
464df0a8a19SPaulo Alcantara
465df0a8a19SPaulo Alcantara switch (ctx->reparse_type) {
466df0a8a19SPaulo Alcantara case CIFS_REPARSE_TYPE_NFS:
467df0a8a19SPaulo Alcantara rc = mknod_nfs(xid, inode, dentry, tcon, full_path, mode, dev);
468df0a8a19SPaulo Alcantara break;
469df0a8a19SPaulo Alcantara case CIFS_REPARSE_TYPE_WSL:
470df0a8a19SPaulo Alcantara rc = mknod_wsl(xid, inode, dentry, tcon, full_path, mode, dev);
471df0a8a19SPaulo Alcantara break;
472df0a8a19SPaulo Alcantara }
473df0a8a19SPaulo Alcantara return rc;
474df0a8a19SPaulo Alcantara }
475df0a8a19SPaulo Alcantara
476db0f1c07SPaulo Alcantara /* See MS-FSCC 2.1.2.6 for the 'NFS' style reparse tags */
parse_reparse_posix(struct reparse_posix_data * buf,struct cifs_sb_info * cifs_sb,struct cifs_open_info_data * data)477db0f1c07SPaulo Alcantara static int parse_reparse_posix(struct reparse_posix_data *buf,
478db0f1c07SPaulo Alcantara struct cifs_sb_info *cifs_sb,
479db0f1c07SPaulo Alcantara struct cifs_open_info_data *data)
480db0f1c07SPaulo Alcantara {
481db0f1c07SPaulo Alcantara unsigned int len;
482db0f1c07SPaulo Alcantara u64 type;
483db0f1c07SPaulo Alcantara
484c6db81c5SPali Rohár len = le16_to_cpu(buf->ReparseDataLength);
485c6db81c5SPali Rohár if (len < sizeof(buf->InodeType)) {
486c6db81c5SPali Rohár cifs_dbg(VFS, "srv returned malformed nfs buffer\n");
487c6db81c5SPali Rohár return -EIO;
488c6db81c5SPali Rohár }
489c6db81c5SPali Rohár
490c6db81c5SPali Rohár len -= sizeof(buf->InodeType);
491c6db81c5SPali Rohár
492db0f1c07SPaulo Alcantara switch ((type = le64_to_cpu(buf->InodeType))) {
493db0f1c07SPaulo Alcantara case NFS_SPECFILE_LNK:
4946d52db31SPali Rohár if (len == 0 || (len % 2)) {
4956d52db31SPali Rohár cifs_dbg(VFS, "srv returned malformed nfs symlink buffer\n");
4966d52db31SPali Rohár return -EIO;
4976d52db31SPali Rohár }
4986d52db31SPali Rohár /*
4996d52db31SPali Rohár * Check that buffer does not contain UTF-16 null codepoint
5006d52db31SPali Rohár * because Linux cannot process symlink with null byte.
5016d52db31SPali Rohár */
5026d52db31SPali Rohár if (UniStrnlen((wchar_t *)buf->DataBuffer, len/2) != len/2) {
5036d52db31SPali Rohár cifs_dbg(VFS, "srv returned null byte in nfs symlink target location\n");
5046d52db31SPali Rohár return -EIO;
5056d52db31SPali Rohár }
506db0f1c07SPaulo Alcantara data->symlink_target = cifs_strndup_from_utf16(buf->DataBuffer,
507db0f1c07SPaulo Alcantara len, true,
508db0f1c07SPaulo Alcantara cifs_sb->local_nls);
509db0f1c07SPaulo Alcantara if (!data->symlink_target)
510db0f1c07SPaulo Alcantara return -ENOMEM;
511db0f1c07SPaulo Alcantara cifs_dbg(FYI, "%s: target path: %s\n",
512db0f1c07SPaulo Alcantara __func__, data->symlink_target);
513db0f1c07SPaulo Alcantara break;
514db0f1c07SPaulo Alcantara case NFS_SPECFILE_CHR:
515db0f1c07SPaulo Alcantara case NFS_SPECFILE_BLK:
5166d52db31SPali Rohár /* DataBuffer for block and char devices contains two 32-bit numbers */
5176d52db31SPali Rohár if (len != 8) {
5186d52db31SPali Rohár cifs_dbg(VFS, "srv returned malformed nfs buffer for type: 0x%llx\n", type);
5196d52db31SPali Rohár return -EIO;
5206d52db31SPali Rohár }
5216d52db31SPali Rohár break;
522db0f1c07SPaulo Alcantara case NFS_SPECFILE_FIFO:
523db0f1c07SPaulo Alcantara case NFS_SPECFILE_SOCK:
5246d52db31SPali Rohár /* DataBuffer for fifos and sockets is empty */
5256d52db31SPali Rohár if (len != 0) {
5266d52db31SPali Rohár cifs_dbg(VFS, "srv returned malformed nfs buffer for type: 0x%llx\n", type);
5276d52db31SPali Rohár return -EIO;
5286d52db31SPali Rohár }
529db0f1c07SPaulo Alcantara break;
530db0f1c07SPaulo Alcantara default:
531db0f1c07SPaulo Alcantara cifs_dbg(VFS, "%s: unhandled inode type: 0x%llx\n",
532db0f1c07SPaulo Alcantara __func__, type);
533db0f1c07SPaulo Alcantara return -EOPNOTSUPP;
534db0f1c07SPaulo Alcantara }
535db0f1c07SPaulo Alcantara return 0;
536db0f1c07SPaulo Alcantara }
537db0f1c07SPaulo Alcantara
smb2_parse_native_symlink(char ** target,const char * buf,unsigned int len,bool unicode,bool relative,const char * full_path,struct cifs_sb_info * cifs_sb)538430afd3eSPali Rohár int smb2_parse_native_symlink(char **target, const char *buf, unsigned int len,
539430afd3eSPali Rohár bool unicode, bool relative,
540430afd3eSPali Rohár const char *full_path,
541430afd3eSPali Rohár struct cifs_sb_info *cifs_sb)
542430afd3eSPali Rohár {
543430afd3eSPali Rohár char sep = CIFS_DIR_SEP(cifs_sb);
544430afd3eSPali Rohár char *linux_target = NULL;
545430afd3eSPali Rohár char *smb_target = NULL;
546430afd3eSPali Rohár int levels;
547430afd3eSPali Rohár int rc;
548430afd3eSPali Rohár int i;
549430afd3eSPali Rohár
550430afd3eSPali Rohár smb_target = cifs_strndup_from_utf16(buf, len, unicode, cifs_sb->local_nls);
551430afd3eSPali Rohár if (!smb_target) {
552430afd3eSPali Rohár rc = -ENOMEM;
553430afd3eSPali Rohár goto out;
554430afd3eSPali Rohár }
555430afd3eSPali Rohár
556430afd3eSPali Rohár if (smb_target[0] == sep && relative) {
557430afd3eSPali Rohár /*
558430afd3eSPali Rohár * This is a relative SMB symlink from the top of the share,
559430afd3eSPali Rohár * which is the top level directory of the Linux mount point.
560430afd3eSPali Rohár * Linux does not support such relative symlinks, so convert
561430afd3eSPali Rohár * it to the relative symlink from the current directory.
562430afd3eSPali Rohár * full_path is the SMB path to the symlink (from which is
563430afd3eSPali Rohár * extracted current directory) and smb_target is the SMB path
564430afd3eSPali Rohár * where symlink points, therefore full_path must always be on
565430afd3eSPali Rohár * the SMB share.
566430afd3eSPali Rohár */
567430afd3eSPali Rohár int smb_target_len = strlen(smb_target)+1;
568430afd3eSPali Rohár levels = 0;
569430afd3eSPali Rohár for (i = 1; full_path[i]; i++) { /* i=1 to skip leading sep */
570430afd3eSPali Rohár if (full_path[i] == sep)
571430afd3eSPali Rohár levels++;
572430afd3eSPali Rohár }
573430afd3eSPali Rohár linux_target = kmalloc(levels*3 + smb_target_len, GFP_KERNEL);
574430afd3eSPali Rohár if (!linux_target) {
575430afd3eSPali Rohár rc = -ENOMEM;
576430afd3eSPali Rohár goto out;
577430afd3eSPali Rohár }
578430afd3eSPali Rohár for (i = 0; i < levels; i++) {
579430afd3eSPali Rohár linux_target[i*3 + 0] = '.';
580430afd3eSPali Rohár linux_target[i*3 + 1] = '.';
581430afd3eSPali Rohár linux_target[i*3 + 2] = sep;
582430afd3eSPali Rohár }
583430afd3eSPali Rohár memcpy(linux_target + levels*3, smb_target+1, smb_target_len); /* +1 to skip leading sep */
584430afd3eSPali Rohár } else {
585430afd3eSPali Rohár linux_target = smb_target;
586430afd3eSPali Rohár smb_target = NULL;
587430afd3eSPali Rohár }
588430afd3eSPali Rohár
589430afd3eSPali Rohár if (sep == '\\')
590430afd3eSPali Rohár convert_delimiter(linux_target, '/');
591430afd3eSPali Rohár
592430afd3eSPali Rohár rc = 0;
593430afd3eSPali Rohár *target = linux_target;
594430afd3eSPali Rohár
595430afd3eSPali Rohár cifs_dbg(FYI, "%s: symlink target: %s\n", __func__, *target);
596430afd3eSPali Rohár
597430afd3eSPali Rohár out:
598430afd3eSPali Rohár if (rc != 0)
599430afd3eSPali Rohár kfree(linux_target);
600430afd3eSPali Rohár kfree(smb_target);
601430afd3eSPali Rohár return rc;
602430afd3eSPali Rohár }
603430afd3eSPali Rohár
parse_reparse_symlink(struct reparse_symlink_data_buffer * sym,u32 plen,bool unicode,struct cifs_sb_info * cifs_sb,const char * full_path,struct cifs_open_info_data * data)604db0f1c07SPaulo Alcantara static int parse_reparse_symlink(struct reparse_symlink_data_buffer *sym,
605db0f1c07SPaulo Alcantara u32 plen, bool unicode,
606db0f1c07SPaulo Alcantara struct cifs_sb_info *cifs_sb,
607430afd3eSPali Rohár const char *full_path,
608db0f1c07SPaulo Alcantara struct cifs_open_info_data *data)
609db0f1c07SPaulo Alcantara {
610db0f1c07SPaulo Alcantara unsigned int len;
611db0f1c07SPaulo Alcantara unsigned int offs;
612db0f1c07SPaulo Alcantara
613db0f1c07SPaulo Alcantara /* We handle Symbolic Link reparse tag here. See: MS-FSCC 2.1.2.4 */
614db0f1c07SPaulo Alcantara
615db0f1c07SPaulo Alcantara offs = le16_to_cpu(sym->SubstituteNameOffset);
616db0f1c07SPaulo Alcantara len = le16_to_cpu(sym->SubstituteNameLength);
617db0f1c07SPaulo Alcantara if (offs + 20 > plen || offs + len + 20 > plen) {
618db0f1c07SPaulo Alcantara cifs_dbg(VFS, "srv returned malformed symlink buffer\n");
619db0f1c07SPaulo Alcantara return -EIO;
620db0f1c07SPaulo Alcantara }
621db0f1c07SPaulo Alcantara
622430afd3eSPali Rohár return smb2_parse_native_symlink(&data->symlink_target,
623430afd3eSPali Rohár sym->PathBuffer + offs,
624430afd3eSPali Rohár len,
625430afd3eSPali Rohár unicode,
626430afd3eSPali Rohár le32_to_cpu(sym->Flags) & SYMLINK_FLAG_RELATIVE,
627430afd3eSPali Rohár full_path,
628430afd3eSPali Rohár cifs_sb);
629db0f1c07SPaulo Alcantara }
630db0f1c07SPaulo Alcantara
parse_reparse_point(struct reparse_data_buffer * buf,u32 plen,struct cifs_sb_info * cifs_sb,const char * full_path,bool unicode,struct cifs_open_info_data * data)631db0f1c07SPaulo Alcantara int parse_reparse_point(struct reparse_data_buffer *buf,
632db0f1c07SPaulo Alcantara u32 plen, struct cifs_sb_info *cifs_sb,
633430afd3eSPali Rohár const char *full_path,
634db0f1c07SPaulo Alcantara bool unicode, struct cifs_open_info_data *data)
635db0f1c07SPaulo Alcantara {
6360fa4b4aaSPaulo Alcantara struct cifs_tcon *tcon = cifs_sb_master_tcon(cifs_sb);
6370fa4b4aaSPaulo Alcantara
638db0f1c07SPaulo Alcantara data->reparse.buf = buf;
639db0f1c07SPaulo Alcantara
640db0f1c07SPaulo Alcantara /* See MS-FSCC 2.1.2 */
641db0f1c07SPaulo Alcantara switch (le32_to_cpu(buf->ReparseTag)) {
642db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_NFS:
643db0f1c07SPaulo Alcantara return parse_reparse_posix((struct reparse_posix_data *)buf,
644db0f1c07SPaulo Alcantara cifs_sb, data);
645db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_SYMLINK:
646db0f1c07SPaulo Alcantara return parse_reparse_symlink(
647db0f1c07SPaulo Alcantara (struct reparse_symlink_data_buffer *)buf,
648430afd3eSPali Rohár plen, unicode, cifs_sb, full_path, data);
649db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_LX_SYMLINK:
650db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_AF_UNIX:
651db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_LX_FIFO:
652db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_LX_CHR:
653db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_LX_BLK:
654474c0871SPali Rohár if (le16_to_cpu(buf->ReparseDataLength) != 0) {
655474c0871SPali Rohár cifs_dbg(VFS, "srv returned malformed buffer for reparse point: 0x%08x\n",
656474c0871SPali Rohár le32_to_cpu(buf->ReparseTag));
657474c0871SPali Rohár return -EIO;
658474c0871SPali Rohár }
659*10cbae84SPali Rohár return 0;
660db0f1c07SPaulo Alcantara default:
6610fa4b4aaSPaulo Alcantara cifs_tcon_dbg(VFS | ONCE, "unhandled reparse tag: 0x%08x\n",
6620fa4b4aaSPaulo Alcantara le32_to_cpu(buf->ReparseTag));
663*10cbae84SPali Rohár return -EOPNOTSUPP;
664db0f1c07SPaulo Alcantara }
665db0f1c07SPaulo Alcantara }
666db0f1c07SPaulo Alcantara
667db0f1c07SPaulo Alcantara int smb2_parse_reparse_point(struct cifs_sb_info *cifs_sb,
668430afd3eSPali Rohár const char *full_path,
669db0f1c07SPaulo Alcantara struct kvec *rsp_iov,
670db0f1c07SPaulo Alcantara struct cifs_open_info_data *data)
671db0f1c07SPaulo Alcantara {
672db0f1c07SPaulo Alcantara struct reparse_data_buffer *buf;
673db0f1c07SPaulo Alcantara struct smb2_ioctl_rsp *io = rsp_iov->iov_base;
674db0f1c07SPaulo Alcantara u32 plen = le32_to_cpu(io->OutputCount);
675db0f1c07SPaulo Alcantara
676db0f1c07SPaulo Alcantara buf = (struct reparse_data_buffer *)((u8 *)io +
wsl_to_fattr(struct cifs_open_info_data * data,struct cifs_sb_info * cifs_sb,u32 tag,struct cifs_fattr * fattr)677db0f1c07SPaulo Alcantara le32_to_cpu(io->OutputOffset));
678430afd3eSPali Rohár return parse_reparse_point(buf, plen, cifs_sb, full_path, true, data);
679db0f1c07SPaulo Alcantara }
680db0f1c07SPaulo Alcantara
68132cc06a6SPali Rohár static bool wsl_to_fattr(struct cifs_open_info_data *data,
68297db4160SPaulo Alcantara struct cifs_sb_info *cifs_sb,
68397db4160SPaulo Alcantara u32 tag, struct cifs_fattr *fattr)
68497db4160SPaulo Alcantara {
68597db4160SPaulo Alcantara struct smb2_file_full_ea_info *ea;
68632cc06a6SPali Rohár bool have_xattr_dev = false;
68797db4160SPaulo Alcantara u32 next = 0;
68897db4160SPaulo Alcantara
68997db4160SPaulo Alcantara switch (tag) {
69097db4160SPaulo Alcantara case IO_REPARSE_TAG_LX_SYMLINK:
69197db4160SPaulo Alcantara fattr->cf_mode |= S_IFLNK;
69297db4160SPaulo Alcantara break;
69397db4160SPaulo Alcantara case IO_REPARSE_TAG_LX_FIFO:
69497db4160SPaulo Alcantara fattr->cf_mode |= S_IFIFO;
69597db4160SPaulo Alcantara break;
69697db4160SPaulo Alcantara case IO_REPARSE_TAG_AF_UNIX:
69797db4160SPaulo Alcantara fattr->cf_mode |= S_IFSOCK;
69897db4160SPaulo Alcantara break;
69997db4160SPaulo Alcantara case IO_REPARSE_TAG_LX_CHR:
70097db4160SPaulo Alcantara fattr->cf_mode |= S_IFCHR;
70197db4160SPaulo Alcantara break;
70297db4160SPaulo Alcantara case IO_REPARSE_TAG_LX_BLK:
70397db4160SPaulo Alcantara fattr->cf_mode |= S_IFBLK;
70497db4160SPaulo Alcantara break;
70597db4160SPaulo Alcantara }
70697db4160SPaulo Alcantara
70797db4160SPaulo Alcantara if (!data->wsl.eas_len)
70897db4160SPaulo Alcantara goto out;
70997db4160SPaulo Alcantara
71097db4160SPaulo Alcantara ea = (struct smb2_file_full_ea_info *)data->wsl.eas;
71197db4160SPaulo Alcantara do {
71297db4160SPaulo Alcantara const char *name;
71397db4160SPaulo Alcantara void *v;
71497db4160SPaulo Alcantara u8 nlen;
71597db4160SPaulo Alcantara
71697db4160SPaulo Alcantara ea = (void *)((u8 *)ea + next);
71797db4160SPaulo Alcantara next = le32_to_cpu(ea->next_entry_offset);
71897db4160SPaulo Alcantara if (!le16_to_cpu(ea->ea_value_length))
71997db4160SPaulo Alcantara continue;
72097db4160SPaulo Alcantara
72197db4160SPaulo Alcantara name = ea->ea_data;
72297db4160SPaulo Alcantara nlen = ea->ea_name_length;
72397db4160SPaulo Alcantara v = (void *)((u8 *)ea->ea_data + ea->ea_name_length + 1);
72497db4160SPaulo Alcantara
72597db4160SPaulo Alcantara if (!strncmp(name, SMB2_WSL_XATTR_UID, nlen))
72697db4160SPaulo Alcantara fattr->cf_uid = wsl_make_kuid(cifs_sb, v);
72797db4160SPaulo Alcantara else if (!strncmp(name, SMB2_WSL_XATTR_GID, nlen))
72897db4160SPaulo Alcantara fattr->cf_gid = wsl_make_kgid(cifs_sb, v);
72932cc06a6SPali Rohár else if (!strncmp(name, SMB2_WSL_XATTR_MODE, nlen)) {
73032cc06a6SPali Rohár /* File type in reparse point tag and in xattr mode must match. */
73132cc06a6SPali Rohár if (S_DT(fattr->cf_mode) != S_DT(le32_to_cpu(*(__le32 *)v)))
73232cc06a6SPali Rohár return false;
73397db4160SPaulo Alcantara fattr->cf_mode = (umode_t)le32_to_cpu(*(__le32 *)v);
73432cc06a6SPali Rohár } else if (!strncmp(name, SMB2_WSL_XATTR_DEV, nlen)) {
735bbc258dcSPaulo Alcantara fattr->cf_rdev = reparse_mkdev(v);
73632cc06a6SPali Rohár have_xattr_dev = true;
73732cc06a6SPali Rohár }
73897db4160SPaulo Alcantara } while (next);
73997db4160SPaulo Alcantara out:
74032cc06a6SPali Rohár
74132cc06a6SPali Rohár /* Major and minor numbers for char and block devices are mandatory. */
74232cc06a6SPali Rohár if (!have_xattr_dev && (tag == IO_REPARSE_TAG_LX_CHR || tag == IO_REPARSE_TAG_LX_BLK))
74332cc06a6SPali Rohár return false;
74432cc06a6SPali Rohár
posix_reparse_to_fattr(struct cifs_sb_info * cifs_sb,struct cifs_fattr * fattr,struct cifs_open_info_data * data)74597db4160SPaulo Alcantara fattr->cf_dtype = S_DT(fattr->cf_mode);
74632cc06a6SPali Rohár return true;
74797db4160SPaulo Alcantara }
74897db4160SPaulo Alcantara
749136fca78SRalph Boehme static bool posix_reparse_to_fattr(struct cifs_sb_info *cifs_sb,
750db0f1c07SPaulo Alcantara struct cifs_fattr *fattr,
751db0f1c07SPaulo Alcantara struct cifs_open_info_data *data)
752db0f1c07SPaulo Alcantara {
753db0f1c07SPaulo Alcantara struct reparse_posix_data *buf = data->reparse.posix;
754db0f1c07SPaulo Alcantara
755136fca78SRalph Boehme
756136fca78SRalph Boehme if (buf == NULL)
757136fca78SRalph Boehme return true;
758136fca78SRalph Boehme
759136fca78SRalph Boehme if (le16_to_cpu(buf->ReparseDataLength) < sizeof(buf->InodeType)) {
760136fca78SRalph Boehme WARN_ON_ONCE(1);
761c6db81c5SPali Rohár return false;
762136fca78SRalph Boehme }
763136fca78SRalph Boehme
764db0f1c07SPaulo Alcantara switch (le64_to_cpu(buf->InodeType)) {
765db0f1c07SPaulo Alcantara case NFS_SPECFILE_CHR:
766136fca78SRalph Boehme if (le16_to_cpu(buf->ReparseDataLength) != sizeof(buf->InodeType) + 8) {
767136fca78SRalph Boehme WARN_ON_ONCE(1);
768c6db81c5SPali Rohár return false;
769136fca78SRalph Boehme }
770db0f1c07SPaulo Alcantara fattr->cf_mode |= S_IFCHR;
771bbc258dcSPaulo Alcantara fattr->cf_rdev = reparse_mkdev(buf->DataBuffer);
772db0f1c07SPaulo Alcantara break;
773db0f1c07SPaulo Alcantara case NFS_SPECFILE_BLK:
774136fca78SRalph Boehme if (le16_to_cpu(buf->ReparseDataLength) != sizeof(buf->InodeType) + 8) {
775136fca78SRalph Boehme WARN_ON_ONCE(1);
776c6db81c5SPali Rohár return false;
777136fca78SRalph Boehme }
778db0f1c07SPaulo Alcantara fattr->cf_mode |= S_IFBLK;
779bbc258dcSPaulo Alcantara fattr->cf_rdev = reparse_mkdev(buf->DataBuffer);
780db0f1c07SPaulo Alcantara break;
781db0f1c07SPaulo Alcantara case NFS_SPECFILE_FIFO:
782db0f1c07SPaulo Alcantara fattr->cf_mode |= S_IFIFO;
783db0f1c07SPaulo Alcantara break;
784db0f1c07SPaulo Alcantara case NFS_SPECFILE_SOCK:
785db0f1c07SPaulo Alcantara fattr->cf_mode |= S_IFSOCK;
786db0f1c07SPaulo Alcantara break;
787db0f1c07SPaulo Alcantara case NFS_SPECFILE_LNK:
788db0f1c07SPaulo Alcantara fattr->cf_mode |= S_IFLNK;
789db0f1c07SPaulo Alcantara break;
790db0f1c07SPaulo Alcantara default:
791db0f1c07SPaulo Alcantara WARN_ON_ONCE(1);
792db0f1c07SPaulo Alcantara return false;
cifs_reparse_point_to_fattr(struct cifs_sb_info * cifs_sb,struct cifs_fattr * fattr,struct cifs_open_info_data * data)793db0f1c07SPaulo Alcantara }
794136fca78SRalph Boehme return true;
795db0f1c07SPaulo Alcantara }
796db0f1c07SPaulo Alcantara
797136fca78SRalph Boehme bool cifs_reparse_point_to_fattr(struct cifs_sb_info *cifs_sb,
798136fca78SRalph Boehme struct cifs_fattr *fattr,
799136fca78SRalph Boehme struct cifs_open_info_data *data)
800136fca78SRalph Boehme {
801136fca78SRalph Boehme u32 tag = data->reparse.tag;
802136fca78SRalph Boehme bool ok;
803136fca78SRalph Boehme
804db0f1c07SPaulo Alcantara switch (tag) {
8050397a48cSPaulo Alcantara case IO_REPARSE_TAG_INTERNAL:
8060397a48cSPaulo Alcantara if (!(fattr->cf_cifsattrs & ATTR_DIRECTORY))
8070397a48cSPaulo Alcantara return false;
8080397a48cSPaulo Alcantara fallthrough;
80958252401SPaulo Alcantara case IO_REPARSE_TAG_DFS:
81058252401SPaulo Alcantara case IO_REPARSE_TAG_DFSR:
81158252401SPaulo Alcantara case IO_REPARSE_TAG_MOUNT_POINT:
81258252401SPaulo Alcantara /* See cifs_create_junction_fattr() */
81358252401SPaulo Alcantara fattr->cf_mode = S_IFDIR | 0711;
81458252401SPaulo Alcantara break;
815db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_LX_SYMLINK:
816db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_LX_FIFO:
817db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_AF_UNIX:
818db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_LX_CHR:
819db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_LX_BLK:
82032cc06a6SPali Rohár ok = wsl_to_fattr(data, cifs_sb, tag, fattr);
82132cc06a6SPali Rohár if (!ok)
82232cc06a6SPali Rohár return false;
823db0f1c07SPaulo Alcantara break;
824136fca78SRalph Boehme case IO_REPARSE_TAG_NFS:
825136fca78SRalph Boehme ok = posix_reparse_to_fattr(cifs_sb, fattr, data);
826136fca78SRalph Boehme if (!ok)
827136fca78SRalph Boehme return false;
828136fca78SRalph Boehme break;
829db0f1c07SPaulo Alcantara case 0: /* SMB1 symlink */
830db0f1c07SPaulo Alcantara case IO_REPARSE_TAG_SYMLINK:
831db0f1c07SPaulo Alcantara fattr->cf_mode |= S_IFLNK;
832db0f1c07SPaulo Alcantara break;
833db0f1c07SPaulo Alcantara default:
834db0f1c07SPaulo Alcantara return false;
835db0f1c07SPaulo Alcantara }
836136fca78SRalph Boehme
83758252401SPaulo Alcantara fattr->cf_dtype = S_DT(fattr->cf_mode);
838db0f1c07SPaulo Alcantara return true;
839db0f1c07SPaulo Alcantara }
840