[BACK]Return to fcntl.c CVS log [TXT][DIR] Up to [Development] / linux-2.6-xfs / fs / cifs

File: [Development] / linux-2.6-xfs / fs / cifs / fcntl.c (download)

Revision 1.2, Thu Jun 17 03:20:52 2004 UTC (13 years, 4 months ago) by nathans
Branch: MAIN
Changes since 1.1: +27 -2 lines

Merge up to 2.6.7, and upgrade kdb at the same time.

/*
 *   fs/cifs/fcntl.c
 *
 *   vfs operations that deal with the file control API
 * 
 *   Copyright (C) International Business Machines  Corp., 2003,2004
 *   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/fcntl.h>
#include "cifsglob.h"
#include "cifsproto.h"
#include "cifs_unicode.h"
#include "cifs_debug.h"

int cifs_directory_notify(unsigned long arg, struct file * file)
{
	int xid;
	int rc = -EINVAL;
	int oplock = FALSE;
	struct cifs_sb_info *cifs_sb;
	struct cifsTconInfo *pTcon;
	char *full_path = NULL;
	__u32 filter = FILE_NOTIFY_CHANGE_NAME | FILE_NOTIFY_CHANGE_ATTRIBUTES;
    __u16 netfid;

	xid = GetXid();
	cifs_sb = CIFS_SB(file->f_dentry->d_sb);
	pTcon = cifs_sb->tcon;

	down(&file->f_dentry->d_sb->s_vfs_rename_sem);
	full_path = build_path_from_dentry(file->f_dentry);
	up(&file->f_dentry->d_sb->s_vfs_rename_sem);

	if(full_path == NULL) {
		rc = -ENOMEM;
	} else {
		cFYI(1,("cifs dir notify on file %s",full_path));
		rc = CIFSSMBOpen(xid, pTcon, full_path, FILE_OPEN, 
			GENERIC_READ | SYNCHRONIZE, 0 /* create options */,
			&netfid, &oplock,NULL, cifs_sb->local_nls);
		/* BB fixme - add this handle to a notify handle list */
		if(rc) {
			cFYI(1,("Could not open directory for notify"));
		} else {
			rc = CIFSSMBNotify(xid, pTcon, 1 /* subdirs */, netfid, 
				filter, cifs_sb->local_nls);
			/* BB add code to close file eventually (at unmount
			it would close automatically but may be a way
			to do it easily when inode freed or when
			notify info is cleared/changed */
		}
	}
	
	FreeXid(xid);
	return rc;
}


long cifs_fcntl(int file_desc, unsigned int command, unsigned long arg,
				struct file * file)
{
	/* Few few file control functions need to be specially mapped. So far
	only:
		F_NOTIFY (for directory change notification)
	And eventually:
		F_GETLEASE
		F_SETLEASE 
	need to be mapped here. The others either already are mapped downstream
	or do not need to go to the server (client only sideeffects):
		F_DUPFD:
		F_GETFD:
		F_SETFD:
		F_GETFL:
		F_SETFL:
		F_GETLK:
		F_SETLK:
		F_SETLKW:
		F_GETOWN:
		F_SETOWN:
		F_GETSIG:
		F_SETSIG:
	*/
	long rc = 0;

	cFYI(1,("cifs_fcntl: command %d with arg %lx",command,arg)); /* BB removeme BB */

	switch (command) {
	case F_NOTIFY:
		/* let the local call have a chance to fail first */
		rc = generic_file_fcntl(file_desc,command,arg,file);
		if(rc)
			return rc;
		else {
			/* local call succeeded try to do remote notify to
			pick up changes from other clients to server file */
			cifs_directory_notify(arg, file);
			/* BB add case to long and return rc from above */
			return rc;
		}
		break;
	default:
		break;
	}
	return generic_file_fcntl(file_desc,command,arg,file);
}