link.c 9.5 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3
/*
 *   fs/cifs/link.c
 *
S
Steve French 已提交
4
 *   Copyright (C) International Business Machines  Corp., 2002,2008
L
Linus Torvalds 已提交
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50
 *   Author(s): Steve French (sfrench@us.ibm.com)
 *
 *   This library is free software; you can redistribute it and/or modify
 *   it under the terms of the GNU Lesser General Public License as published
 *   by the Free Software Foundation; either version 2.1 of the License, or
 *   (at your option) any later version.
 *
 *   This library is distributed in the hope that it will be useful,
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See
 *   the GNU Lesser General Public License for more details.
 *
 *   You should have received a copy of the GNU Lesser General Public License
 *   along with this library; if not, write to the Free Software
 *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
 */
#include <linux/fs.h>
#include <linux/stat.h>
#include <linux/namei.h>
#include "cifsfs.h"
#include "cifspdu.h"
#include "cifsglob.h"
#include "cifsproto.h"
#include "cifs_debug.h"
#include "cifs_fs_sb.h"

int
cifs_hardlink(struct dentry *old_file, struct inode *inode,
	      struct dentry *direntry)
{
	int rc = -EACCES;
	int xid;
	char *fromName = NULL;
	char *toName = NULL;
	struct cifs_sb_info *cifs_sb_target;
	struct cifsTconInfo *pTcon;
	struct cifsInodeInfo *cifsInode;

	xid = GetXid();

	cifs_sb_target = CIFS_SB(inode->i_sb);
	pTcon = cifs_sb_target->tcon;

/* No need to check for cross device links since server will do that
   BB note DFS case in future though (when we may have to check) */

51 52
	fromName = build_path_from_dentry(old_file);
	toName = build_path_from_dentry(direntry);
S
Steve French 已提交
53
	if ((fromName == NULL) || (toName == NULL)) {
L
Linus Torvalds 已提交
54 55 56 57
		rc = -ENOMEM;
		goto cifs_hl_exit;
	}

58 59
/*	if (cifs_sb_target->tcon->ses->capabilities & CAP_UNIX)*/
	if (pTcon->unix_ext)
L
Linus Torvalds 已提交
60
		rc = CIFSUnixCreateHardLink(xid, pTcon, fromName, toName,
S
Steve French 已提交
61
					    cifs_sb_target->local_nls,
62 63
					    cifs_sb_target->mnt_cifs_flags &
						CIFS_MOUNT_MAP_SPECIAL_CHR);
L
Linus Torvalds 已提交
64 65
	else {
		rc = CIFSCreateHardLink(xid, pTcon, fromName, toName,
S
Steve French 已提交
66
					cifs_sb_target->local_nls,
67 68
					cifs_sb_target->mnt_cifs_flags &
						CIFS_MOUNT_MAP_SPECIAL_CHR);
S
Steve French 已提交
69 70
		if ((rc == -EIO) || (rc == -EINVAL))
			rc = -EOPNOTSUPP;
L
Linus Torvalds 已提交
71 72
	}

73 74 75 76
	d_drop(direntry);	/* force new lookup from server of target */

	/* if source file is cached (oplocked) revalidate will not go to server
	   until the file is closed or oplock broken so update nlinks locally */
S
Steve French 已提交
77
	if (old_file->d_inode) {
78
		cifsInode = CIFS_I(old_file->d_inode);
S
Steve French 已提交
79
		if (rc == 0) {
80
			old_file->d_inode->i_nlink++;
81 82
/* BB should we make this contingent on superblock flag NOATIME? */
/*			old_file->d_inode->i_ctime = CURRENT_TIME;*/
83 84 85 86 87
			/* parent dir timestamps will update from srv
			within a second, would it really be worth it
			to set the parent dir cifs inode time to zero
			to force revalidate (faster) for it too? */
		}
S
Steve French 已提交
88
		/* if not oplocked will force revalidate to get info
89 90 91
		   on source file from srv */
		cifsInode->time = 0;

S
Steve French 已提交
92
		/* Will update parent dir timestamps from srv within a second.
93 94
		   Would it really be worth it to set the parent dir (cifs
		   inode) time field to zero to force revalidate on parent
S
Steve French 已提交
95
		   directory faster ie
96
			CIFS_I(inode)->time = 0;  */
L
Linus Torvalds 已提交
97 98 99
	}

cifs_hl_exit:
J
Jesper Juhl 已提交
100 101
	kfree(fromName);
	kfree(toName);
L
Linus Torvalds 已提交
102 103 104 105
	FreeXid(xid);
	return rc;
}

106
void *
L
Linus Torvalds 已提交
107 108 109 110 111 112
cifs_follow_link(struct dentry *direntry, struct nameidata *nd)
{
	struct inode *inode = direntry->d_inode;
	int rc = -EACCES;
	int xid;
	char *full_path = NULL;
S
Steve French 已提交
113
	char *target_path = ERR_PTR(-ENOMEM);
L
Linus Torvalds 已提交
114 115 116 117 118
	struct cifs_sb_info *cifs_sb;
	struct cifsTconInfo *pTcon;

	xid = GetXid();

119
	full_path = build_path_from_dentry(direntry);
L
Linus Torvalds 已提交
120 121 122 123 124

	if (!full_path)
		goto out_no_free;

	cFYI(1, ("Full path: %s inode = 0x%p", full_path, inode));
125 126
	cifs_sb = CIFS_SB(inode->i_sb);
	pTcon = cifs_sb->tcon;
L
Linus Torvalds 已提交
127 128 129 130 131 132
	target_path = kmalloc(PATH_MAX, GFP_KERNEL);
	if (!target_path) {
		target_path = ERR_PTR(-ENOMEM);
		goto out;
	}

133 134 135 136 137 138
	/* We could change this to:
		if (pTcon->unix_ext)
	   but there does not seem any point in refusing to
	   get symlink info if we can, even if unix extensions
	   turned off for this mount */

L
Linus Torvalds 已提交
139 140 141 142 143 144
	if (pTcon->ses->capabilities & CAP_UNIX)
		rc = CIFSSMBUnixQuerySymLink(xid, pTcon, full_path,
					     target_path,
					     PATH_MAX-1,
					     cifs_sb->local_nls);
	else {
145
		/* BB add read reparse point symlink code here */
L
Linus Torvalds 已提交
146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165
		/* rc = CIFSSMBQueryReparseLinkInfo */
		/* BB Add code to Query ReparsePoint info */
		/* BB Add MAC style xsymlink check here if enabled */
	}

	if (rc == 0) {

/* BB Add special case check for Samba DFS symlinks */

		target_path[PATH_MAX-1] = 0;
	} else {
		kfree(target_path);
		target_path = ERR_PTR(rc);
	}

out:
	kfree(full_path);
out_no_free:
	FreeXid(xid);
	nd_set_link(nd, target_path);
166
	return NULL;	/* No cookie */
L
Linus Torvalds 已提交
167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183
}

int
cifs_symlink(struct inode *inode, struct dentry *direntry, const char *symname)
{
	int rc = -EOPNOTSUPP;
	int xid;
	struct cifs_sb_info *cifs_sb;
	struct cifsTconInfo *pTcon;
	char *full_path = NULL;
	struct inode *newinode = NULL;

	xid = GetXid();

	cifs_sb = CIFS_SB(inode->i_sb);
	pTcon = cifs_sb->tcon;

184
	full_path = build_path_from_dentry(direntry);
L
Linus Torvalds 已提交
185

S
Steve French 已提交
186
	if (full_path == NULL) {
L
Linus Torvalds 已提交
187 188 189 190
		FreeXid(xid);
		return -ENOMEM;
	}

191
	cFYI(1, ("Full path: %s", full_path));
L
Linus Torvalds 已提交
192 193 194
	cFYI(1, ("symname is %s", symname));

	/* BB what if DFS and this volume is on different share? BB */
195
	if (pTcon->unix_ext)
L
Linus Torvalds 已提交
196 197 198
		rc = CIFSUnixCreateSymLink(xid, pTcon, full_path, symname,
					   cifs_sb->local_nls);
	/* else
S
Steve French 已提交
199 200
	   rc = CIFSCreateReparseSymLink(xid, pTcon, fromName, toName,
					cifs_sb_target->local_nls); */
L
Linus Torvalds 已提交
201 202

	if (rc == 0) {
203
		if (pTcon->unix_ext)
L
Linus Torvalds 已提交
204
			rc = cifs_get_inode_info_unix(&newinode, full_path,
S
Steve French 已提交
205
						      inode->i_sb, xid);
L
Linus Torvalds 已提交
206 207
		else
			rc = cifs_get_inode_info(&newinode, full_path, NULL,
208
						 inode->i_sb, xid, NULL);
L
Linus Torvalds 已提交
209 210

		if (rc != 0) {
211
			cFYI(1, ("Create symlink ok, getinodeinfo fail rc = %d",
L
Linus Torvalds 已提交
212 213
			      rc));
		} else {
214 215 216 217
			if (pTcon->nocase)
				direntry->d_op = &cifs_ci_dentry_ops;
			else
				direntry->d_op = &cifs_dentry_ops;
L
Linus Torvalds 已提交
218 219 220 221
			d_instantiate(direntry, newinode);
		}
	}

J
Jesper Juhl 已提交
222
	kfree(full_path);
L
Linus Torvalds 已提交
223 224 225 226 227 228 229 230 231 232 233 234 235 236
	FreeXid(xid);
	return rc;
}

int
cifs_readlink(struct dentry *direntry, char __user *pBuffer, int buflen)
{
	struct inode *inode = direntry->d_inode;
	int rc = -EACCES;
	int xid;
	int oplock = FALSE;
	struct cifs_sb_info *cifs_sb;
	struct cifsTconInfo *pTcon;
	char *full_path = NULL;
S
Steve French 已提交
237 238
	char *tmp_path = NULL;
	char *tmpbuffer;
L
Linus Torvalds 已提交
239 240 241 242 243 244 245
	int len;
	__u16 fid;

	xid = GetXid();
	cifs_sb = CIFS_SB(inode->i_sb);
	pTcon = cifs_sb->tcon;

S
Steve French 已提交
246
/* BB would it be safe against deadlock to grab this sem
L
Linus Torvalds 已提交
247
      even though rename itself grabs the sem and calls lookup? */
248
/*       mutex_lock(&inode->i_sb->s_vfs_rename_mutex);*/
249
	full_path = build_path_from_dentry(direntry);
250
/*       mutex_unlock(&inode->i_sb->s_vfs_rename_mutex);*/
L
Linus Torvalds 已提交
251

S
Steve French 已提交
252
	if (full_path == NULL) {
L
Linus Torvalds 已提交
253 254 255 256 257 258 259
		FreeXid(xid);
		return -ENOMEM;
	}

	cFYI(1,
	     ("Full path: %s inode = 0x%p pBuffer = 0x%p buflen = %d",
	      full_path, inode, pBuffer, buflen));
S
Steve French 已提交
260
	if (buflen > PATH_MAX)
L
Linus Torvalds 已提交
261 262 263
		len = PATH_MAX;
	else
		len = buflen;
S
Steve French 已提交
264 265
	tmpbuffer = kmalloc(len, GFP_KERNEL);
	if (tmpbuffer == NULL) {
J
Jesper Juhl 已提交
266
		kfree(full_path);
L
Linus Torvalds 已提交
267 268 269 270
		FreeXid(xid);
		return -ENOMEM;
	}

S
Steve French 已提交
271 272
/* BB add read reparse point symlink code and
	Unix extensions symlink code here BB */
273
/* We could disable this based on pTcon->unix_ext flag instead ... but why? */
L
Linus Torvalds 已提交
274 275 276 277 278
	if (cifs_sb->tcon->ses->capabilities & CAP_UNIX)
		rc = CIFSSMBUnixQuerySymLink(xid, pTcon, full_path,
				tmpbuffer,
				len - 1,
				cifs_sb->local_nls);
279
	else if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL) {
S
Steve French 已提交
280
		cERROR(1, ("SFU style symlinks not implemented yet"));
281 282
		/* add open and read as in fs/cifs/inode.c */
	} else {
L
Linus Torvalds 已提交
283
		rc = CIFSSMBOpen(xid, pTcon, full_path, FILE_OPEN, GENERIC_READ,
S
Steve French 已提交
284 285 286
				OPEN_REPARSE_POINT, &fid, &oplock, NULL,
				cifs_sb->local_nls,
				cifs_sb->mnt_cifs_flags &
287
					CIFS_MOUNT_MAP_SPECIAL_CHR);
S
Steve French 已提交
288
		if (!rc) {
L
Linus Torvalds 已提交
289 290
			rc = CIFSSMBQueryReparseLinkInfo(xid, pTcon, full_path,
				tmpbuffer,
S
Steve French 已提交
291
				len - 1,
L
Linus Torvalds 已提交
292 293
				fid,
				cifs_sb->local_nls);
S
Steve French 已提交
294
			if (CIFSSMBClose(xid, pTcon, fid)) {
295 296
				cFYI(1, ("Error closing junction point "
					 "(open for ioctl)"));
L
Linus Torvalds 已提交
297
			}
S
Steve French 已提交
298
			/* BB unwind this long, nested function, or remove BB */
S
Steve French 已提交
299
			if (rc == -EIO) {
L
Linus Torvalds 已提交
300
				/* Query if DFS Junction */
S
Steve French 已提交
301 302
				unsigned int num_referrals = 0;
				struct dfs_info3_param *refs = NULL;
L
Linus Torvalds 已提交
303 304 305 306
				tmp_path =
					kmalloc(MAX_TREE_SIZE + MAX_PATHCONF + 1,
						GFP_KERNEL);
				if (tmp_path) {
S
Steve French 已提交
307 308 309 310 311 312
					strncpy(tmp_path, pTcon->treeName,
						MAX_TREE_SIZE);
					strncat(tmp_path, full_path,
						MAX_PATHCONF);
					rc = get_dfs_path(xid, pTcon->ses,
						tmp_path,
313
						cifs_sb->local_nls,
S
Steve French 已提交
314
						&num_referrals, &refs,
315 316
						cifs_sb->mnt_cifs_flags &
						    CIFS_MOUNT_MAP_SPECIAL_CHR);
S
Steve French 已提交
317 318 319
					cFYI(1, ("Get DFS for %s rc = %d ",
						tmp_path, rc));
					if ((num_referrals == 0) && (rc == 0))
L
Linus Torvalds 已提交
320 321
						rc = -EACCES;
					else {
S
Steve French 已提交
322 323
						cFYI(1, ("num referral: %d",
							num_referrals));
S
Steve French 已提交
324
						if (refs && refs->path_name) {
325
							strncpy(tmpbuffer,
S
Steve French 已提交
326
								refs->path_name,
327
								len-1);
L
Linus Torvalds 已提交
328 329
						}
					}
S
Steve French 已提交
330
					kfree(refs);
L
Linus Torvalds 已提交
331 332
					kfree(tmp_path);
}
S
Steve French 已提交
333 334 335
				/* BB add code like else decode referrals
				then memcpy to tmpbuffer and free referrals
				string array BB */
L
Linus Torvalds 已提交
336 337 338 339 340 341 342 343 344 345 346 347 348 349
			}
		}
	}
	/* BB Anything else to do to handle recursive links? */
	/* BB Should we be using page ops here? */

	/* BB null terminate returned string in pBuffer? BB */
	if (rc == 0) {
		rc = vfs_readlink(direntry, pBuffer, len, tmpbuffer);
		cFYI(1,
		     ("vfs_readlink called from cifs_readlink returned %d",
		      rc));
	}

J
Jesper Juhl 已提交
350 351
	kfree(tmpbuffer);
	kfree(full_path);
L
Linus Torvalds 已提交
352 353 354 355
	FreeXid(xid);
	return rc;
}

356
void cifs_put_link(struct dentry *direntry, struct nameidata *nd, void *cookie)
L
Linus Torvalds 已提交
357 358 359 360 361
{
	char *p = nd_get_link(nd);
	if (!IS_ERR(p))
		kfree(p);
}