cifsfs.c 28.1 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3
/*
 *   fs/cifs/cifsfs.c
 *
S
Steve French 已提交
4
 *   Copyright (C) International Business Machines  Corp., 2002,2007
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
 *   Author(s): Steve French (sfrench@us.ibm.com)
 *
 *   Common Internet FileSystem (CIFS) client
 *
 *   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
 */

/* Note that BB means BUGBUG (ie something to fix eventually) */

#include <linux/module.h>
#include <linux/fs.h>
#include <linux/mount.h>
#include <linux/slab.h>
#include <linux/init.h>
#include <linux/list.h>
#include <linux/seq_file.h>
#include <linux/vfs.h>
#include <linux/mempool.h>
35
#include <linux/delay.h>
36
#include <linux/kthread.h>
37
#include <linux/freezer.h>
L
Linus Torvalds 已提交
38 39 40 41 42 43 44 45
#include "cifsfs.h"
#include "cifspdu.h"
#define DECLARE_GLOBALS_HERE
#include "cifsglob.h"
#include "cifsproto.h"
#include "cifs_debug.h"
#include "cifs_fs_sb.h"
#include <linux/mm.h>
46
#include <linux/key-type.h>
L
Linus Torvalds 已提交
47 48 49 50
#define CIFS_MAGIC_NUMBER 0xFF534D42	/* the first four bytes of SMB PDUs */

#ifdef CONFIG_CIFS_QUOTA
static struct quotactl_ops cifs_quotactl_ops;
S
Steve French 已提交
51 52
#endif /* QUOTA */

L
Linus Torvalds 已提交
53 54 55 56 57 58 59 60
int cifsFYI = 0;
int cifsERROR = 1;
int traceSMB = 0;
unsigned int oplockEnabled = 1;
unsigned int experimEnabled = 0;
unsigned int linuxExtEnabled = 1;
unsigned int lookupCacheEnabled = 1;
unsigned int multiuser_mount = 0;
61 62
unsigned int extended_security = CIFSSEC_DEF;
/* unsigned int ntlmv2_support = 0; */
L
Linus Torvalds 已提交
63
unsigned int sign_CIFS_PDUs = 1;
S
Steve French 已提交
64 65
extern struct task_struct *oplockThread; /* remove sparse warning */
struct task_struct *oplockThread = NULL;
66
/* extern struct task_struct * dnotifyThread; remove sparse warning */
S
Steve French 已提交
67
static struct task_struct *dnotifyThread = NULL;
68
static const struct super_operations cifs_super_ops;
L
Linus Torvalds 已提交
69 70
unsigned int CIFSMaxBufSize = CIFS_MAX_MSGSIZE;
module_param(CIFSMaxBufSize, int, 0);
71 72
MODULE_PARM_DESC(CIFSMaxBufSize, "Network buffer size (not including header). "
				 "Default: 16384 Range: 8192 to 130048");
L
Linus Torvalds 已提交
73 74
unsigned int cifs_min_rcv = CIFS_MIN_RCV_POOL;
module_param(cifs_min_rcv, int, 0);
75 76
MODULE_PARM_DESC(cifs_min_rcv, "Network buffers in pool. Default: 4 Range: "
				"1 to 64");
L
Linus Torvalds 已提交
77 78
unsigned int cifs_min_small = 30;
module_param(cifs_min_small, int, 0);
79 80
MODULE_PARM_DESC(cifs_min_small, "Small network buffers in pool. Default: 30 "
				 "Range: 2 to 256");
L
Linus Torvalds 已提交
81 82
unsigned int cifs_max_pending = CIFS_MAX_REQ;
module_param(cifs_max_pending, int, 0);
83 84
MODULE_PARM_DESC(cifs_max_pending, "Simultaneous requests to server. "
				   "Default: 50 Range: 2 to 256");
L
Linus Torvalds 已提交
85 86 87 88 89

extern mempool_t *cifs_sm_req_poolp;
extern mempool_t *cifs_req_poolp;
extern mempool_t *cifs_mid_poolp;

90
extern struct kmem_cache *cifs_oplock_cachep;
L
Linus Torvalds 已提交
91 92 93 94 95 96 97 98

static int
cifs_read_super(struct super_block *sb, void *data,
		const char *devname, int silent)
{
	struct inode *inode;
	struct cifs_sb_info *cifs_sb;
	int rc = 0;
99

100 101
	/* BB should we make this contingent on mount parm? */
	sb->s_flags |= MS_NODIRATIME | MS_NOATIME;
S
Steve French 已提交
102
	sb->s_fs_info = kzalloc(sizeof(struct cifs_sb_info), GFP_KERNEL);
L
Linus Torvalds 已提交
103
	cifs_sb = CIFS_SB(sb);
104
	if (cifs_sb == NULL)
L
Linus Torvalds 已提交
105 106 107 108 109 110 111 112 113 114 115 116 117
		return -ENOMEM;

	rc = cifs_mount(sb, cifs_sb, data, devname);

	if (rc) {
		if (!silent)
			cERROR(1,
			       ("cifs_mount failed w/return code = %d", rc));
		goto out_mount_failed;
	}

	sb->s_magic = CIFS_MAGIC_NUMBER;
	sb->s_op = &cifs_super_ops;
118
/*	if (cifs_sb->tcon->ses->server->maxBuf > MAX_CIFS_HDR_SIZE + 512)
S
Steve French 已提交
119 120
	    sb->s_blocksize =
		cifs_sb->tcon->ses->server->maxBuf - MAX_CIFS_HDR_SIZE; */
L
Linus Torvalds 已提交
121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138
#ifdef CONFIG_CIFS_QUOTA
	sb->s_qcop = &cifs_quotactl_ops;
#endif
	sb->s_blocksize = CIFS_MAX_MSGSIZE;
	sb->s_blocksize_bits = 14;	/* default 2**14 = CIFS_MAX_MSGSIZE */
	inode = iget(sb, ROOT_I);

	if (!inode) {
		rc = -ENOMEM;
		goto out_no_root;
	}

	sb->s_root = d_alloc_root(inode);

	if (!sb->s_root) {
		rc = -ENOMEM;
		goto out_no_root;
	}
139

140 141 142 143 144 145
#ifdef CONFIG_CIFS_EXPERIMENTAL
	if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) {
		cFYI(1, ("export ops supported"));
		sb->s_export_op = &cifs_export_ops;
	}
#endif /* EXPERIMENTAL */
L
Linus Torvalds 已提交
146 147 148 149 150 151 152 153 154

	return 0;

out_no_root:
	cERROR(1, ("cifs_read_super: get root inode failed"));
	if (inode)
		iput(inode);

out_mount_failed:
155 156
	if (cifs_sb) {
		if (cifs_sb->local_nls)
157
			unload_nls(cifs_sb->local_nls);
L
Linus Torvalds 已提交
158 159 160 161 162 163 164 165 166 167 168 169 170
		kfree(cifs_sb);
	}
	return rc;
}

static void
cifs_put_super(struct super_block *sb)
{
	int rc = 0;
	struct cifs_sb_info *cifs_sb;

	cFYI(1, ("In cifs_put_super"));
	cifs_sb = CIFS_SB(sb);
171
	if (cifs_sb == NULL) {
S
Steve French 已提交
172
		cFYI(1, ("Empty cifs superblock info passed to unmount"));
L
Linus Torvalds 已提交
173 174
		return;
	}
S
Steve French 已提交
175
	rc = cifs_umount(sb, cifs_sb);
L
Linus Torvalds 已提交
176 177 178 179 180 181 182 183 184
	if (rc) {
		cERROR(1, ("cifs_umount failed with return code %d", rc));
	}
	unload_nls(cifs_sb->local_nls);
	kfree(cifs_sb);
	return;
}

static int
185
cifs_statfs(struct dentry *dentry, struct kstatfs *buf)
L
Linus Torvalds 已提交
186
{
187
	struct super_block *sb = dentry->d_sb;
S
Steve French 已提交
188
	int xid;
189
	int rc = -EOPNOTSUPP;
L
Linus Torvalds 已提交
190 191 192 193 194 195 196 197 198 199 200
	struct cifs_sb_info *cifs_sb;
	struct cifsTconInfo *pTcon;

	xid = GetXid();

	cifs_sb = CIFS_SB(sb);
	pTcon = cifs_sb->tcon;

	buf->f_type = CIFS_MAGIC_NUMBER;

	/* instead could get the real value via SMB_QUERY_FS_ATTRIBUTE_INFO */
S
Steve French 已提交
201
	buf->f_namelen = PATH_MAX; /* PATH_MAX may be too long - it would
202 203 204
				      presumably be total path, but note
				      that some servers (includinng Samba 3)
				      have a shorter maximum path */
L
Linus Torvalds 已提交
205 206 207 208
	buf->f_files = 0;	/* undefined */
	buf->f_ffree = 0;	/* unlimited */

/* BB we could add a second check for a QFS Unix capability bit */
209
/* BB FIXME check CIFS_POSIX_EXTENSIONS Unix cap first FIXME BB */
210 211
    if ((pTcon->ses->capabilities & CAP_UNIX) && (CIFS_POSIX_EXTENSIONS &
			le64_to_cpu(pTcon->fsUnixInfo.Capability)))
212
	    rc = CIFSSMBQFSPosixInfo(xid, pTcon, buf);
L
Linus Torvalds 已提交
213 214 215

    /* Only need to call the old QFSInfo if failed
    on newer one */
216 217
    if (rc)
	if (pTcon->ses->capabilities & CAP_NT_SMBS)
218
		rc = CIFSSMBQFSInfo(xid, pTcon, buf); /* not supported by OS2 */
L
Linus Torvalds 已提交
219

220 221 222
	/* Some old Windows servers also do not support level 103, retry with
	   older level one if old server failed the previous call or we
	   bypassed it because we detected that this was an older LANMAN sess */
223
	if (rc)
224
		rc = SMBOldQFSInfo(xid, pTcon, buf);
S
Steve French 已提交
225
	/* int f_type;
L
Linus Torvalds 已提交
226 227
	   __fsid_t f_fsid;
	   int f_namelen;  */
228
	/* BB get from info in tcon struct at mount time call to QFSAttrInfo */
L
Linus Torvalds 已提交
229
	FreeXid(xid);
230 231
	return 0;		/* always return success? what if volume is no
				   longer available? */
L
Linus Torvalds 已提交
232 233
}

234
static int cifs_permission(struct inode *inode, int mask, struct nameidata *nd)
L
Linus Torvalds 已提交
235 236 237 238 239
{
	struct cifs_sb_info *cifs_sb;

	cifs_sb = CIFS_SB(inode->i_sb);

240
	if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM)
L
Linus Torvalds 已提交
241
		return 0;
242
	else /* file mode might have been restricted at mount time
243
		on the client (above and beyond ACL on servers) for
L
Linus Torvalds 已提交
244
		servers which do not support setting and viewing mode bits,
245
		so allowing client to check permissions is useful */
L
Linus Torvalds 已提交
246 247 248
		return generic_permission(inode, mask, NULL);
}

249 250 251 252 253
static struct kmem_cache *cifs_inode_cachep;
static struct kmem_cache *cifs_req_cachep;
static struct kmem_cache *cifs_mid_cachep;
struct kmem_cache *cifs_oplock_cachep;
static struct kmem_cache *cifs_sm_req_cachep;
L
Linus Torvalds 已提交
254 255 256 257 258 259 260 261
mempool_t *cifs_sm_req_poolp;
mempool_t *cifs_req_poolp;
mempool_t *cifs_mid_poolp;

static struct inode *
cifs_alloc_inode(struct super_block *sb)
{
	struct cifsInodeInfo *cifs_inode;
262
	cifs_inode = kmem_cache_alloc(cifs_inode_cachep, GFP_KERNEL);
L
Linus Torvalds 已提交
263 264 265 266 267 268 269 270 271 272 273
	if (!cifs_inode)
		return NULL;
	cifs_inode->cifsAttrs = 0x20;	/* default */
	atomic_set(&cifs_inode->inUse, 0);
	cifs_inode->time = 0;
	/* Until the file is open and we have gotten oplock
	info back from the server, can not assume caching of
	file data or metadata */
	cifs_inode->clientCanCacheRead = FALSE;
	cifs_inode->clientCanCacheAll = FALSE;
	cifs_inode->vfs_inode.i_blkbits = 14;  /* 2**14 = CIFS_MAX_MSGSIZE */
274

275 276 277
	/* Can not set i_flags here - they get immediately overwritten
	   to zero by the VFS */
/*	cifs_inode->vfs_inode.i_flags = S_NOATIME | S_NOCMTIME;*/
L
Linus Torvalds 已提交
278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301
	INIT_LIST_HEAD(&cifs_inode->openFileList);
	return &cifs_inode->vfs_inode;
}

static void
cifs_destroy_inode(struct inode *inode)
{
	kmem_cache_free(cifs_inode_cachep, CIFS_I(inode));
}

/*
 * cifs_show_options() is for displaying mount options in /proc/mounts.
 * Not all settable options are displayed but most of the important
 * ones are.
 */
static int
cifs_show_options(struct seq_file *s, struct vfsmount *m)
{
	struct cifs_sb_info *cifs_sb;

	cifs_sb = CIFS_SB(m->mnt_sb);

	if (cifs_sb) {
		if (cifs_sb->tcon) {
302
/* BB add prepath to mount options displayed */
L
Linus Torvalds 已提交
303 304 305 306 307
			seq_printf(s, ",unc=%s", cifs_sb->tcon->treeName);
			if (cifs_sb->tcon->ses) {
				if (cifs_sb->tcon->ses->userName)
					seq_printf(s, ",username=%s",
					   cifs_sb->tcon->ses->userName);
308
				if (cifs_sb->tcon->ses->domainName)
L
Linus Torvalds 已提交
309 310 311
					seq_printf(s, ",domain=%s",
					   cifs_sb->tcon->ses->domainName);
			}
312 313 314 315 316 317
			if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_OVERR_UID) ||
			   !(cifs_sb->tcon->unix_ext))
				seq_printf(s, ",uid=%d", cifs_sb->mnt_uid);
			if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_OVERR_GID) ||
			   !(cifs_sb->tcon->unix_ext))
				seq_printf(s, ",gid=%d", cifs_sb->mnt_gid);
L
Linus Torvalds 已提交
318
		}
319 320
		if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_POSIX_PATHS)
			seq_printf(s, ",posixpaths");
321 322
		seq_printf(s, ",rsize=%d", cifs_sb->rsize);
		seq_printf(s, ",wsize=%d", cifs_sb->wsize);
L
Linus Torvalds 已提交
323 324 325 326 327
	}
	return 0;
}

#ifdef CONFIG_CIFS_QUOTA
328 329
int cifs_xquota_set(struct super_block *sb, int quota_type, qid_t qid,
		struct fs_disk_quota *pdquota)
L
Linus Torvalds 已提交
330 331 332 333 334
{
	int xid;
	int rc = 0;
	struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
	struct cifsTconInfo *pTcon;
335

336
	if (cifs_sb)
L
Linus Torvalds 已提交
337 338 339 340 341 342
		pTcon = cifs_sb->tcon;
	else
		return -EIO;


	xid = GetXid();
343
	if (pTcon) {
344
		cFYI(1, ("set type: 0x%x id: %d", quota_type, qid));
L
Linus Torvalds 已提交
345
	} else {
346
		rc = -EIO;
L
Linus Torvalds 已提交
347 348 349 350 351 352
	}

	FreeXid(xid);
	return rc;
}

353 354
int cifs_xquota_get(struct super_block *sb, int quota_type, qid_t qid,
		    struct fs_disk_quota *pdquota)
L
Linus Torvalds 已提交
355 356 357 358 359 360
{
	int xid;
	int rc = 0;
	struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
	struct cifsTconInfo *pTcon;

361
	if (cifs_sb)
L
Linus Torvalds 已提交
362 363 364 365 366
		pTcon = cifs_sb->tcon;
	else
		return -EIO;

	xid = GetXid();
367
	if (pTcon) {
368
		cFYI(1, ("set type: 0x%x id: %d", quota_type, qid));
L
Linus Torvalds 已提交
369 370 371 372 373 374 375 376
	} else {
		rc = -EIO;
	}

	FreeXid(xid);
	return rc;
}

377
int cifs_xstate_set(struct super_block *sb, unsigned int flags, int operation)
L
Linus Torvalds 已提交
378
{
379
	int xid;
L
Linus Torvalds 已提交
380 381 382 383
	int rc = 0;
	struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
	struct cifsTconInfo *pTcon;

384
	if (cifs_sb)
L
Linus Torvalds 已提交
385 386 387 388 389
		pTcon = cifs_sb->tcon;
	else
		return -EIO;

	xid = GetXid();
390
	if (pTcon) {
391
		cFYI(1, ("flags: 0x%x operation: 0x%x", flags, operation));
L
Linus Torvalds 已提交
392 393 394 395 396 397 398 399
	} else {
		rc = -EIO;
	}

	FreeXid(xid);
	return rc;
}

400
int cifs_xstate_get(struct super_block *sb, struct fs_quota_stat *qstats)
L
Linus Torvalds 已提交
401 402 403 404 405 406
{
	int xid;
	int rc = 0;
	struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
	struct cifsTconInfo *pTcon;

407
	if (cifs_sb) {
L
Linus Torvalds 已提交
408 409 410 411 412
		pTcon = cifs_sb->tcon;
	} else {
		return -EIO;
	}
	xid = GetXid();
413
	if (pTcon) {
414
		cFYI(1, ("pqstats %p", qstats));
L
Linus Torvalds 已提交
415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430
	} else {
		rc = -EIO;
	}

	FreeXid(xid);
	return rc;
}

static struct quotactl_ops cifs_quotactl_ops = {
	.set_xquota	= cifs_xquota_set,
	.get_xquota	= cifs_xquota_set,
	.set_xstate	= cifs_xstate_set,
	.get_xstate	= cifs_xstate_get,
};
#endif

431
static void cifs_umount_begin(struct vfsmount *vfsmnt, int flags)
432
{
433
	struct cifs_sb_info *cifs_sb;
434
	struct cifsTconInfo *tcon;
435

436 437 438
	if (!(flags & MNT_FORCE))
		return;
	cifs_sb = CIFS_SB(vfsmnt->mnt_sb);
439
	if (cifs_sb == NULL)
440 441 442
		return;

	tcon = cifs_sb->tcon;
443
	if (tcon == NULL)
444
		return;
445 446 447 448 449
	down(&tcon->tconSem);
	if (atomic_read(&tcon->useCount) == 1)
		tcon->tidStatus = CifsExiting;
	up(&tcon->tconSem);

450
	/* cancel_brl_requests(tcon); */ /* BB mark all brl mids as exiting */
S
Steve French 已提交
451
	/* cancel_notify_requests(tcon); */
452 453
	if (tcon->ses && tcon->ses->server) {
		cFYI(1, ("wake up tasks now - umount begin not complete"));
454
		wake_up_all(&tcon->ses->server->request_q);
455 456 457 458 459
		wake_up_all(&tcon->ses->server->response_q);
		msleep(1); /* yield */
		/* we have to kick the requests once more */
		wake_up_all(&tcon->ses->server->response_q);
		msleep(1);
460 461
	}
/* BB FIXME - finish add checks for tidStatus BB */
462 463 464 465

	return;
}

466 467 468 469 470 471 472 473
#ifdef CONFIG_CIFS_STATS2
static int cifs_show_stats(struct seq_file *s, struct vfsmount *mnt)
{
	/* BB FIXME */
	return 0;
}
#endif

L
Linus Torvalds 已提交
474 475 476 477 478 479
static int cifs_remount(struct super_block *sb, int *flags, char *data)
{
	*flags |= MS_NODIRATIME;
	return 0;
}

480
static const struct super_operations cifs_super_ops = {
L
Linus Torvalds 已提交
481 482 483 484 485
	.read_inode = cifs_read_inode,
	.put_super = cifs_put_super,
	.statfs = cifs_statfs,
	.alloc_inode = cifs_alloc_inode,
	.destroy_inode = cifs_destroy_inode,
486 487 488 489 490
/*	.drop_inode	    = generic_delete_inode,
	.delete_inode	= cifs_delete_inode,  */  /* Do not need above two
	functions unless later we add lazy close of inodes or unless the
	kernel forgets to call us with the same number of releases (closes)
	as opens */
L
Linus Torvalds 已提交
491
	.show_options = cifs_show_options,
S
Steve French 已提交
492
	.umount_begin   = cifs_umount_begin,
L
Linus Torvalds 已提交
493
	.remount_fs = cifs_remount,
494
#ifdef CONFIG_CIFS_STATS2
495
	.show_stats = cifs_show_stats,
496
#endif
L
Linus Torvalds 已提交
497 498
};

499
static int
L
Linus Torvalds 已提交
500
cifs_get_sb(struct file_system_type *fs_type,
501
	    int flags, const char *dev_name, void *data, struct vfsmount *mnt)
L
Linus Torvalds 已提交
502 503 504 505 506 507 508
{
	int rc;
	struct super_block *sb = sget(fs_type, NULL, set_anon_super, NULL);

	cFYI(1, ("Devname: %s flags: %d ", dev_name, flags));

	if (IS_ERR(sb))
509
		return PTR_ERR(sb);
L
Linus Torvalds 已提交
510 511 512

	sb->s_flags = flags;

513
	rc = cifs_read_super(sb, data, dev_name, flags & MS_SILENT ? 1 : 0);
L
Linus Torvalds 已提交
514 515 516
	if (rc) {
		up_write(&sb->s_umount);
		deactivate_super(sb);
517
		return rc;
L
Linus Torvalds 已提交
518 519
	}
	sb->s_flags |= MS_ACTIVE;
520
	return simple_set_mnt(mnt, sb);
L
Linus Torvalds 已提交
521 522
}

523 524
static ssize_t cifs_file_aio_write(struct kiocb *iocb, const struct iovec *iov,
				   unsigned long nr_segs, loff_t pos)
L
Linus Torvalds 已提交
525
{
526
	struct inode *inode = iocb->ki_filp->f_path.dentry->d_inode;
L
Linus Torvalds 已提交
527 528
	ssize_t written;

529
	written = generic_file_aio_write(iocb, iov, nr_segs, pos);
530 531
	if (!CIFS_I(inode)->clientCanCacheAll)
		filemap_fdatawrite(inode->i_mapping);
L
Linus Torvalds 已提交
532 533 534
	return written;
}

535 536 537
static loff_t cifs_llseek(struct file *file, loff_t offset, int origin)
{
	/* origin == SEEK_END => we must revalidate the cached file length */
538
	if (origin == SEEK_END) {
539 540 541 542 543
		int retval;

		/* some applications poll for the file length in this strange
		   way so we must seek to end on non-oplocked files by
		   setting the revalidate time to zero */
544
		CIFS_I(file->f_path.dentry->d_inode)->time = 0;
545 546

		retval = cifs_revalidate(file->f_path.dentry);
547 548 549 550 551 552
		if (retval < 0)
			return (loff_t)retval;
	}
	return remote_llseek(file, offset, origin);
}

L
Linus Torvalds 已提交
553 554 555 556 557 558 559
static struct file_system_type cifs_fs_type = {
	.owner = THIS_MODULE,
	.name = "cifs",
	.get_sb = cifs_get_sb,
	.kill_sb = kill_anon_super,
	/*  .fs_flags */
};
560
const struct inode_operations cifs_dir_inode_ops = {
L
Linus Torvalds 已提交
561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581
	.create = cifs_create,
	.lookup = cifs_lookup,
	.getattr = cifs_getattr,
	.unlink = cifs_unlink,
	.link = cifs_hardlink,
	.mkdir = cifs_mkdir,
	.rmdir = cifs_rmdir,
	.rename = cifs_rename,
	.permission = cifs_permission,
/*	revalidate:cifs_revalidate,   */
	.setattr = cifs_setattr,
	.symlink = cifs_symlink,
	.mknod   = cifs_mknod,
#ifdef CONFIG_CIFS_XATTR
	.setxattr = cifs_setxattr,
	.getxattr = cifs_getxattr,
	.listxattr = cifs_listxattr,
	.removexattr = cifs_removexattr,
#endif
};

582
const struct inode_operations cifs_file_inode_ops = {
L
Linus Torvalds 已提交
583 584 585 586 587 588 589 590 591 592
/*	revalidate:cifs_revalidate, */
	.setattr = cifs_setattr,
	.getattr = cifs_getattr, /* do we need this anymore? */
	.rename = cifs_rename,
	.permission = cifs_permission,
#ifdef CONFIG_CIFS_XATTR
	.setxattr = cifs_setxattr,
	.getxattr = cifs_getxattr,
	.listxattr = cifs_listxattr,
	.removexattr = cifs_removexattr,
593
#endif
L
Linus Torvalds 已提交
594 595
};

596
const struct inode_operations cifs_symlink_inode_ops = {
597
	.readlink = generic_readlink,
L
Linus Torvalds 已提交
598 599 600 601 602 603 604 605 606 607 608
	.follow_link = cifs_follow_link,
	.put_link = cifs_put_link,
	.permission = cifs_permission,
	/* BB add the following two eventually */
	/* revalidate: cifs_revalidate,
	   setattr:    cifs_notify_change, *//* BB do we need notify change */
#ifdef CONFIG_CIFS_XATTR
	.setxattr = cifs_setxattr,
	.getxattr = cifs_getxattr,
	.listxattr = cifs_listxattr,
	.removexattr = cifs_removexattr,
609
#endif
L
Linus Torvalds 已提交
610 611
};

612
const struct file_operations cifs_file_ops = {
613 614 615 616
	.read = do_sync_read,
	.write = do_sync_write,
	.aio_read = generic_file_aio_read,
	.aio_write = cifs_file_aio_write,
L
Linus Torvalds 已提交
617 618 619 620 621 622
	.open = cifs_open,
	.release = cifs_close,
	.lock = cifs_lock,
	.fsync = cifs_fsync,
	.flush = cifs_flush,
	.mmap  = cifs_file_mmap,
623
	.splice_read = generic_file_splice_read,
624
	.llseek = cifs_llseek,
625 626 627 628
#ifdef CONFIG_CIFS_POSIX
	.ioctl	= cifs_ioctl,
#endif /* CONFIG_CIFS_POSIX */

L
Linus Torvalds 已提交
629 630 631 632 633
#ifdef CONFIG_CIFS_EXPERIMENTAL
	.dir_notify = cifs_dir_notify,
#endif /* CONFIG_CIFS_EXPERIMENTAL */
};

634
const struct file_operations cifs_file_direct_ops = {
635
	/* no mmap, no aio, no readv -
L
Linus Torvalds 已提交
636 637 638 639 640 641 642 643
	   BB reevaluate whether they can be done with directio, no cache */
	.read = cifs_user_read,
	.write = cifs_user_write,
	.open = cifs_open,
	.release = cifs_close,
	.lock = cifs_lock,
	.fsync = cifs_fsync,
	.flush = cifs_flush,
644
	.splice_read = generic_file_splice_read,
645 646 647
#ifdef CONFIG_CIFS_POSIX
	.ioctl  = cifs_ioctl,
#endif /* CONFIG_CIFS_POSIX */
648
	.llseek = cifs_llseek,
L
Linus Torvalds 已提交
649 650 651 652
#ifdef CONFIG_CIFS_EXPERIMENTAL
	.dir_notify = cifs_dir_notify,
#endif /* CONFIG_CIFS_EXPERIMENTAL */
};
653
const struct file_operations cifs_file_nobrl_ops = {
654 655 656 657 658 659 660 661 662
	.read = do_sync_read,
	.write = do_sync_write,
	.aio_read = generic_file_aio_read,
	.aio_write = cifs_file_aio_write,
	.open = cifs_open,
	.release = cifs_close,
	.fsync = cifs_fsync,
	.flush = cifs_flush,
	.mmap  = cifs_file_mmap,
663
	.splice_read = generic_file_splice_read,
664
	.llseek = cifs_llseek,
665
#ifdef CONFIG_CIFS_POSIX
666
	.ioctl	= cifs_ioctl,
667 668 669
#endif /* CONFIG_CIFS_POSIX */

#ifdef CONFIG_CIFS_EXPERIMENTAL
670
	.dir_notify = cifs_dir_notify,
671 672 673
#endif /* CONFIG_CIFS_EXPERIMENTAL */
};

674
const struct file_operations cifs_file_direct_nobrl_ops = {
675
	/* no mmap, no aio, no readv -
676 677 678 679 680 681 682
	   BB reevaluate whether they can be done with directio, no cache */
	.read = cifs_user_read,
	.write = cifs_user_write,
	.open = cifs_open,
	.release = cifs_close,
	.fsync = cifs_fsync,
	.flush = cifs_flush,
683
	.splice_read = generic_file_splice_read,
684
#ifdef CONFIG_CIFS_POSIX
685
	.ioctl  = cifs_ioctl,
686
#endif /* CONFIG_CIFS_POSIX */
687
	.llseek = cifs_llseek,
688
#ifdef CONFIG_CIFS_EXPERIMENTAL
689
	.dir_notify = cifs_dir_notify,
690 691
#endif /* CONFIG_CIFS_EXPERIMENTAL */
};
L
Linus Torvalds 已提交
692

693
const struct file_operations cifs_dir_ops = {
L
Linus Torvalds 已提交
694 695 696 697 698 699
	.readdir = cifs_readdir,
	.release = cifs_closedir,
	.read    = generic_read_dir,
#ifdef CONFIG_CIFS_EXPERIMENTAL
	.dir_notify = cifs_dir_notify,
#endif /* CONFIG_CIFS_EXPERIMENTAL */
700
	.ioctl  = cifs_ioctl,
L
Linus Torvalds 已提交
701 702 703
};

static void
704
cifs_init_once(struct kmem_cache *cachep, void *inode)
L
Linus Torvalds 已提交
705 706 707
{
	struct cifsInodeInfo *cifsi = inode;

C
Christoph Lameter 已提交
708 709
	inode_init_once(&cifsi->vfs_inode);
	INIT_LIST_HEAD(&cifsi->lockList);
L
Linus Torvalds 已提交
710 711 712 713 714 715
}

static int
cifs_init_inodecache(void)
{
	cifs_inode_cachep = kmem_cache_create("cifs_inode_cache",
716
					      sizeof(struct cifsInodeInfo),
717 718
					      0, (SLAB_RECLAIM_ACCOUNT|
						SLAB_MEM_SPREAD),
719
					      cifs_init_once);
L
Linus Torvalds 已提交
720 721 722 723 724 725 726 727 728
	if (cifs_inode_cachep == NULL)
		return -ENOMEM;

	return 0;
}

static void
cifs_destroy_inodecache(void)
{
729
	kmem_cache_destroy(cifs_inode_cachep);
L
Linus Torvalds 已提交
730 731 732 733 734
}

static int
cifs_init_request_bufs(void)
{
735
	if (CIFSMaxBufSize < 8192) {
L
Linus Torvalds 已提交
736 737 738 739 740 741 742 743 744 745 746 747
	/* Buffer size can not be smaller than 2 * PATH_MAX since maximum
	Unicode path name has to fit in any SMB/CIFS path based frames */
		CIFSMaxBufSize = 8192;
	} else if (CIFSMaxBufSize > 1024*127) {
		CIFSMaxBufSize = 1024 * 127;
	} else {
		CIFSMaxBufSize &= 0x1FE00; /* Round size to even 512 byte mult*/
	}
/*	cERROR(1,("CIFSMaxBufSize %d 0x%x",CIFSMaxBufSize,CIFSMaxBufSize)); */
	cifs_req_cachep = kmem_cache_create("cifs_request",
					    CIFSMaxBufSize +
					    MAX_CIFS_HDR_SIZE, 0,
748
					    SLAB_HWCACHE_ALIGN, NULL);
L
Linus Torvalds 已提交
749 750 751
	if (cifs_req_cachep == NULL)
		return -ENOMEM;

752
	if (cifs_min_rcv < 1)
L
Linus Torvalds 已提交
753 754 755
		cifs_min_rcv = 1;
	else if (cifs_min_rcv > 64) {
		cifs_min_rcv = 64;
756
		cERROR(1, ("cifs_min_rcv set to maximum (64)"));
L
Linus Torvalds 已提交
757 758
	}

759 760
	cifs_req_poolp = mempool_create_slab_pool(cifs_min_rcv,
						  cifs_req_cachep);
L
Linus Torvalds 已提交
761

762
	if (cifs_req_poolp == NULL) {
L
Linus Torvalds 已提交
763 764 765
		kmem_cache_destroy(cifs_req_cachep);
		return -ENOMEM;
	}
766
	/* MAX_CIFS_SMALL_BUFFER_SIZE bytes is enough for most SMB responses and
L
Linus Torvalds 已提交
767 768
	almost all handle based requests (but not write response, nor is it
	sufficient for path based requests).  A smaller size would have
769
	been more efficient (compacting multiple slab items on one 4k page)
L
Linus Torvalds 已提交
770 771
	for the case in which debug was on, but this larger size allows
	more SMBs to use small buffer alloc and is still much more
S
Steve French 已提交
772
	efficient to alloc 1 per page off the slab compared to 17K (5page)
L
Linus Torvalds 已提交
773 774
	alloc of large cifs buffers even when page debugging is on */
	cifs_sm_req_cachep = kmem_cache_create("cifs_small_rq",
S
Steve French 已提交
775
			MAX_CIFS_SMALL_BUFFER_SIZE, 0, SLAB_HWCACHE_ALIGN,
776
			NULL);
L
Linus Torvalds 已提交
777 778 779
	if (cifs_sm_req_cachep == NULL) {
		mempool_destroy(cifs_req_poolp);
		kmem_cache_destroy(cifs_req_cachep);
S
Steve French 已提交
780
		return -ENOMEM;
L
Linus Torvalds 已提交
781 782
	}

783
	if (cifs_min_small < 2)
L
Linus Torvalds 已提交
784 785 786
		cifs_min_small = 2;
	else if (cifs_min_small > 256) {
		cifs_min_small = 256;
S
Steve French 已提交
787
		cFYI(1, ("cifs_min_small set to maximum (256)"));
L
Linus Torvalds 已提交
788 789
	}

790 791
	cifs_sm_req_poolp = mempool_create_slab_pool(cifs_min_small,
						     cifs_sm_req_cachep);
L
Linus Torvalds 已提交
792

793
	if (cifs_sm_req_poolp == NULL) {
L
Linus Torvalds 已提交
794 795 796 797 798 799 800 801 802 803 804 805 806
		mempool_destroy(cifs_req_poolp);
		kmem_cache_destroy(cifs_req_cachep);
		kmem_cache_destroy(cifs_sm_req_cachep);
		return -ENOMEM;
	}

	return 0;
}

static void
cifs_destroy_request_bufs(void)
{
	mempool_destroy(cifs_req_poolp);
807
	kmem_cache_destroy(cifs_req_cachep);
L
Linus Torvalds 已提交
808
	mempool_destroy(cifs_sm_req_poolp);
809
	kmem_cache_destroy(cifs_sm_req_cachep);
L
Linus Torvalds 已提交
810 811 812 813 814 815
}

static int
cifs_init_mids(void)
{
	cifs_mid_cachep = kmem_cache_create("cifs_mpx_ids",
816 817
					    sizeof(struct mid_q_entry), 0,
					    SLAB_HWCACHE_ALIGN, NULL);
L
Linus Torvalds 已提交
818 819 820
	if (cifs_mid_cachep == NULL)
		return -ENOMEM;

821 822
	/* 3 is a reasonable minimum number of simultaneous operations */
	cifs_mid_poolp = mempool_create_slab_pool(3, cifs_mid_cachep);
823
	if (cifs_mid_poolp == NULL) {
L
Linus Torvalds 已提交
824 825 826 827 828
		kmem_cache_destroy(cifs_mid_cachep);
		return -ENOMEM;
	}

	cifs_oplock_cachep = kmem_cache_create("cifs_oplock_structs",
829 830
					sizeof(struct oplock_q_entry), 0,
					SLAB_HWCACHE_ALIGN, NULL);
L
Linus Torvalds 已提交
831 832
	if (cifs_oplock_cachep == NULL) {
		mempool_destroy(cifs_mid_poolp);
833
		kmem_cache_destroy(cifs_mid_cachep);
L
Linus Torvalds 已提交
834 835 836 837 838 839 840 841 842 843
		return -ENOMEM;
	}

	return 0;
}

static void
cifs_destroy_mids(void)
{
	mempool_destroy(cifs_mid_poolp);
844 845
	kmem_cache_destroy(cifs_mid_cachep);
	kmem_cache_destroy(cifs_oplock_cachep);
L
Linus Torvalds 已提交
846 847
}

848
static int cifs_oplock_thread(void *dummyarg)
L
Linus Torvalds 已提交
849
{
S
Steve French 已提交
850
	struct oplock_q_entry *oplock_item;
L
Linus Torvalds 已提交
851
	struct cifsTconInfo *pTcon;
S
Steve French 已提交
852
	struct inode *inode;
L
Linus Torvalds 已提交
853 854 855
	__u16  netfid;
	int rc;

856
	set_freezable();
L
Linus Torvalds 已提交
857
	do {
S
Steve French 已提交
858
		if (try_to_freeze())
859
			continue;
860

L
Linus Torvalds 已提交
861
		spin_lock(&GlobalMid_Lock);
862
		if (list_empty(&GlobalOplock_Q)) {
L
Linus Torvalds 已提交
863 864 865 866
			spin_unlock(&GlobalMid_Lock);
			set_current_state(TASK_INTERRUPTIBLE);
			schedule_timeout(39*HZ);
		} else {
S
Steve French 已提交
867
			oplock_item = list_entry(GlobalOplock_Q.next,
L
Linus Torvalds 已提交
868
				struct oplock_q_entry, qhead);
869
			if (oplock_item) {
870
				cFYI(1, ("found oplock item to write out"));
L
Linus Torvalds 已提交
871 872 873 874 875 876
				pTcon = oplock_item->tcon;
				inode = oplock_item->pinode;
				netfid = oplock_item->netfid;
				spin_unlock(&GlobalMid_Lock);
				DeleteOplockQEntry(oplock_item);
				/* can not grab inode sem here since it would
S
Steve French 已提交
877
				deadlock when oplock received on delete
878
				since vfs_unlink holds the i_mutex across
L
Linus Torvalds 已提交
879
				the call */
880
				/* mutex_lock(&inode->i_mutex);*/
L
Linus Torvalds 已提交
881
				if (S_ISREG(inode->i_mode)) {
882 883
					rc =
					   filemap_fdatawrite(inode->i_mapping);
884 885
					if (CIFS_I(inode)->clientCanCacheRead
									 == 0) {
L
Linus Torvalds 已提交
886 887 888 889 890
						filemap_fdatawait(inode->i_mapping);
						invalidate_remote_inode(inode);
					}
				} else
					rc = 0;
891
				/* mutex_unlock(&inode->i_mutex);*/
L
Linus Torvalds 已提交
892 893
				if (rc)
					CIFS_I(inode)->write_behind_rc = rc;
S
Steve French 已提交
894 895 896 897 898 899 900 901
				cFYI(1, ("Oplock flush inode %p rc %d",
					inode, rc));

				/* releasing stale oplock after recent reconnect
				of smb session using a now incorrect file
				handle is not a data integrity issue but do
				not bother sending an oplock release if session
				to server still is disconnected since oplock
L
Linus Torvalds 已提交
902
				already released by the server in that case */
903
				if (pTcon->tidStatus != CifsNeedReconnect) {
L
Linus Torvalds 已提交
904
				    rc = CIFSSMBLock(0, pTcon, netfid,
S
Steve French 已提交
905
					    0 /* len */ , 0 /* offset */, 0,
L
Linus Torvalds 已提交
906 907
					    0, LOCKING_ANDX_OPLOCK_RELEASE,
					    0 /* wait flag */);
908
					cFYI(1, ("Oplock release rc = %d", rc));
L
Linus Torvalds 已提交
909 910 911
				}
			} else
				spin_unlock(&GlobalMid_Lock);
912 913
			set_current_state(TASK_INTERRUPTIBLE);
			schedule_timeout(1);  /* yield in case q were corrupt */
L
Linus Torvalds 已提交
914
		}
915 916 917
	} while (!kthread_should_stop());

	return 0;
L
Linus Torvalds 已提交
918 919
}

920
static int cifs_dnotify_thread(void *dummyarg)
921
{
922 923 924
	struct list_head *tmp;
	struct cifsSesInfo *ses;

925
	do {
926
		if (try_to_freeze())
S
Steve French 已提交
927
			continue;
928
		set_current_state(TASK_INTERRUPTIBLE);
929 930 931 932 933 934
		schedule_timeout(15*HZ);
		read_lock(&GlobalSMBSeslock);
		/* check if any stuck requests that need
		   to be woken up and wakeq so the
		   thread can wake up and error out */
		list_for_each(tmp, &GlobalSMBSessionList) {
S
Steve French 已提交
935
			ses = list_entry(tmp, struct cifsSesInfo,
936
				cifsSessionList);
S
Steve French 已提交
937
			if (ses && ses->server &&
938
			     atomic_read(&ses->server->inFlight))
939 940 941
				wake_up_all(&ses->server->response_q);
		}
		read_unlock(&GlobalSMBSeslock);
942 943 944
	} while (!kthread_should_stop());

	return 0;
L
Linus Torvalds 已提交
945 946 947 948 949 950 951 952 953
}

static int __init
init_cifs(void)
{
	int rc = 0;
#ifdef CONFIG_PROC_FS
	cifs_proc_init();
#endif
954
/*	INIT_LIST_HEAD(&GlobalServerList);*/	/* BB not implemented yet */
L
Linus Torvalds 已提交
955 956 957
	INIT_LIST_HEAD(&GlobalSMBSessionList);
	INIT_LIST_HEAD(&GlobalTreeConnectionList);
	INIT_LIST_HEAD(&GlobalOplock_Q);
958 959 960
#ifdef CONFIG_CIFS_EXPERIMENTAL
	INIT_LIST_HEAD(&GlobalDnotifyReqList);
	INIT_LIST_HEAD(&GlobalDnotifyRsp_Q);
S
Steve French 已提交
961
#endif
L
Linus Torvalds 已提交
962 963 964 965 966
/*
 *  Initialize Global counters
 */
	atomic_set(&sesInfoAllocCount, 0);
	atomic_set(&tconInfoAllocCount, 0);
S
Steve French 已提交
967
	atomic_set(&tcpSesAllocCount, 0);
L
Linus Torvalds 已提交
968 969 970 971
	atomic_set(&tcpSesReconnectCount, 0);
	atomic_set(&tconInfoReconnectCount, 0);

	atomic_set(&bufAllocCount, 0);
972 973 974 975 976 977
	atomic_set(&smBufAllocCount, 0);
#ifdef CONFIG_CIFS_STATS2
	atomic_set(&totBufAllocCount, 0);
	atomic_set(&totSmBufAllocCount, 0);
#endif /* CONFIG_CIFS_STATS2 */

L
Linus Torvalds 已提交
978 979 980 981
	atomic_set(&midCount, 0);
	GlobalCurrentXid = 0;
	GlobalTotalActiveXid = 0;
	GlobalMaxActiveXid = 0;
982
	memset(Local_System_Name, 0, 15);
L
Linus Torvalds 已提交
983 984 985
	rwlock_init(&GlobalSMBSeslock);
	spin_lock_init(&GlobalMid_Lock);

986
	if (cifs_max_pending < 2) {
L
Linus Torvalds 已提交
987
		cifs_max_pending = 2;
S
Steve French 已提交
988
		cFYI(1, ("cifs_max_pending set to min of 2"));
989
	} else if (cifs_max_pending > 256) {
L
Linus Torvalds 已提交
990
		cifs_max_pending = 256;
S
Steve French 已提交
991
		cFYI(1, ("cifs_max_pending set to max of 256"));
L
Linus Torvalds 已提交
992 993 994
	}

	rc = cifs_init_inodecache();
995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008
	if (rc)
		goto out_clean_proc;

	rc = cifs_init_mids();
	if (rc)
		goto out_destroy_inodecache;

	rc = cifs_init_request_bufs();
	if (rc)
		goto out_destroy_mids;

	rc = register_filesystem(&cifs_fs_type);
	if (rc)
		goto out_destroy_request_bufs;
1009 1010 1011 1012 1013
#ifdef CONFIG_CIFS_UPCALL
	rc = register_key_type(&cifs_spnego_key_type);
	if (rc)
		goto out_unregister_filesystem;
#endif
1014 1015 1016
	oplockThread = kthread_run(cifs_oplock_thread, NULL, "cifsoplockd");
	if (IS_ERR(oplockThread)) {
		rc = PTR_ERR(oplockThread);
S
Steve French 已提交
1017
		cERROR(1, ("error %d create oplock thread", rc));
1018
		goto out_unregister_key_type;
L
Linus Torvalds 已提交
1019
	}
1020 1021 1022 1023

	dnotifyThread = kthread_run(cifs_dnotify_thread, NULL, "cifsdnotifyd");
	if (IS_ERR(dnotifyThread)) {
		rc = PTR_ERR(dnotifyThread);
S
Steve French 已提交
1024
		cERROR(1, ("error %d create dnotify thread", rc));
1025 1026 1027 1028 1029 1030 1031
		goto out_stop_oplock_thread;
	}

	return 0;

 out_stop_oplock_thread:
	kthread_stop(oplockThread);
1032 1033 1034
 out_unregister_key_type:
#ifdef CONFIG_CIFS_UPCALL
	unregister_key_type(&cifs_spnego_key_type);
1035
 out_unregister_filesystem:
1036
#endif
1037 1038 1039 1040 1041 1042 1043 1044
	unregister_filesystem(&cifs_fs_type);
 out_destroy_request_bufs:
	cifs_destroy_request_bufs();
 out_destroy_mids:
	cifs_destroy_mids();
 out_destroy_inodecache:
	cifs_destroy_inodecache();
 out_clean_proc:
L
Linus Torvalds 已提交
1045 1046 1047 1048 1049 1050 1051 1052 1053
#ifdef CONFIG_PROC_FS
	cifs_proc_clean();
#endif
	return rc;
}

static void __exit
exit_cifs(void)
{
1054
	cFYI(0, ("exit_cifs"));
L
Linus Torvalds 已提交
1055 1056
#ifdef CONFIG_PROC_FS
	cifs_proc_clean();
1057 1058 1059
#endif
#ifdef CONFIG_CIFS_UPCALL
	unregister_key_type(&cifs_spnego_key_type);
L
Linus Torvalds 已提交
1060 1061 1062 1063 1064
#endif
	unregister_filesystem(&cifs_fs_type);
	cifs_destroy_inodecache();
	cifs_destroy_mids();
	cifs_destroy_request_bufs();
1065 1066
	kthread_stop(oplockThread);
	kthread_stop(dnotifyThread);
L
Linus Torvalds 已提交
1067 1068 1069
}

MODULE_AUTHOR("Steve French <sfrench@us.ibm.com>");
S
Steve French 已提交
1070
MODULE_LICENSE("GPL");	/* combination of LGPL + GPL source behaves as GPL */
L
Linus Torvalds 已提交
1071
MODULE_DESCRIPTION
1072 1073
    ("VFS to access servers complying with the SNIA CIFS Specification "
     "e.g. Samba and Windows");
L
Linus Torvalds 已提交
1074 1075 1076
MODULE_VERSION(CIFS_VERSION);
module_init(init_cifs)
module_exit(exit_cifs)