patch-2_6_7-vs1_9_1_12
[linux-2.6.git] / fs / cifs / fcntl.c
1 /*
2  *   fs/cifs/fcntl.c
3  *
4  *   vfs operations that deal with the file control API
5  * 
6  *   Copyright (C) International Business Machines  Corp., 2003,2004
7  *   Author(s): Steve French (sfrench@us.ibm.com)
8  *
9  *   This library is free software; you can redistribute it and/or modify
10  *   it under the terms of the GNU Lesser General Public License as published
11  *   by the Free Software Foundation; either version 2.1 of the License, or
12  *   (at your option) any later version.
13  *
14  *   This library is distributed in the hope that it will be useful,
15  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
16  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See
17  *   the GNU Lesser General Public License for more details.
18  *
19  *   You should have received a copy of the GNU Lesser General Public License
20  *   along with this library; if not, write to the Free Software
21  *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22  */
23 #include <linux/fs.h>
24 #include <linux/stat.h>
25 #include <linux/fcntl.h>
26 #include "cifsglob.h"
27 #include "cifsproto.h"
28 #include "cifs_unicode.h"
29 #include "cifs_debug.h"
30
31 int cifs_directory_notify(unsigned long arg, struct file * file)
32 {
33         int xid;
34         int rc = -EINVAL;
35         int oplock = FALSE;
36         struct cifs_sb_info *cifs_sb;
37         struct cifsTconInfo *pTcon;
38         char *full_path = NULL;
39         __u32 filter = FILE_NOTIFY_CHANGE_NAME | FILE_NOTIFY_CHANGE_ATTRIBUTES;
40     __u16 netfid;
41
42         xid = GetXid();
43         cifs_sb = CIFS_SB(file->f_dentry->d_sb);
44         pTcon = cifs_sb->tcon;
45
46         down(&file->f_dentry->d_sb->s_vfs_rename_sem);
47         full_path = build_path_from_dentry(file->f_dentry);
48         up(&file->f_dentry->d_sb->s_vfs_rename_sem);
49
50         if(full_path == NULL) {
51                 rc = -ENOMEM;
52         } else {
53                 cFYI(1,("cifs dir notify on file %s",full_path));
54                 rc = CIFSSMBOpen(xid, pTcon, full_path, FILE_OPEN, 
55                         GENERIC_READ | SYNCHRONIZE, 0 /* create options */,
56                         &netfid, &oplock,NULL, cifs_sb->local_nls);
57                 /* BB fixme - add this handle to a notify handle list */
58                 if(rc) {
59                         cFYI(1,("Could not open directory for notify"));
60                 } else {
61                         rc = CIFSSMBNotify(xid, pTcon, 1 /* subdirs */, netfid, 
62                                 filter, cifs_sb->local_nls);
63                         /* BB add code to close file eventually (at unmount
64                         it would close automatically but may be a way
65                         to do it easily when inode freed or when
66                         notify info is cleared/changed */
67                 }
68         }
69         
70         FreeXid(xid);
71         return rc;
72 }
73
74
75 long cifs_fcntl(int file_desc, unsigned int command, unsigned long arg,
76                                 struct file * file)
77 {
78         /* Few few file control functions need to be specially mapped. So far
79         only:
80                 F_NOTIFY (for directory change notification)
81         And eventually:
82                 F_GETLEASE
83                 F_SETLEASE 
84         need to be mapped here. The others either already are mapped downstream
85         or do not need to go to the server (client only sideeffects):
86                 F_DUPFD:
87                 F_GETFD:
88                 F_SETFD:
89                 F_GETFL:
90                 F_SETFL:
91                 F_GETLK:
92                 F_SETLK:
93                 F_SETLKW:
94                 F_GETOWN:
95                 F_SETOWN:
96                 F_GETSIG:
97                 F_SETSIG:
98         */
99         long rc = 0;
100
101         cFYI(1,("cifs_fcntl: command %d with arg %lx",command,arg)); /* BB removeme BB */
102
103         switch (command) {
104         case F_NOTIFY:
105                 /* let the local call have a chance to fail first */
106                 rc = generic_file_fcntl(file_desc,command,arg,file);
107                 if(rc)
108                         return rc;
109                 else {
110                         /* local call succeeded try to do remote notify to
111                         pick up changes from other clients to server file */
112                         cifs_directory_notify(arg, file);
113                         /* BB add case to long and return rc from above */
114                         return rc;
115                 }
116                 break;
117         default:
118                 break;
119         }
120         return generic_file_fcntl(file_desc,command,arg,file);
121 }
122