xref: /openbmc/linux/fs/smb/client/reparse.c (revision 36db6e8484ed455bbb320d89a119378897ae991c)
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