blob: 5ff25e02521557c152c757c7271d82da9329e64e [file] [log] [blame]
Pavel Shilovskyf0df7372012-09-18 16:20:26 -07001/*
2 * fs/cifs/smb2file.c
3 *
4 * Copyright (C) International Business Machines Corp., 2002, 2011
5 * Author(s): Steve French (sfrench@us.ibm.com),
6 * Pavel Shilovsky ((pshilovsky@samba.org) 2012
7 *
8 * This library is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU Lesser General Public License as published
10 * by the Free Software Foundation; either version 2.1 of the License, or
11 * (at your option) any later version.
12 *
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
16 * the GNU Lesser General Public License for more details.
17 *
18 * You should have received a copy of the GNU Lesser General Public License
19 * along with this library; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21 */
22#include <linux/fs.h>
23#include <linux/stat.h>
24#include <linux/slab.h>
25#include <linux/pagemap.h>
26#include <asm/div64.h>
27#include "cifsfs.h"
28#include "cifspdu.h"
29#include "cifsglob.h"
30#include "cifsproto.h"
31#include "cifs_debug.h"
32#include "cifs_fs_sb.h"
33#include "cifs_unicode.h"
34#include "fscache.h"
35#include "smb2proto.h"
36
Pavel Shilovsky2e44b282012-09-18 16:20:33 -070037void
38smb2_set_oplock_level(struct cifsInodeInfo *cinode, __u32 oplock)
39{
40 oplock &= 0xFF;
41 if (oplock == SMB2_OPLOCK_LEVEL_EXCLUSIVE) {
42 cinode->clientCanCacheAll = true;
43 cinode->clientCanCacheRead = true;
44 cFYI(1, "Exclusive Oplock granted on inode %p",
45 &cinode->vfs_inode);
46 } else if (oplock == SMB2_OPLOCK_LEVEL_II) {
47 cinode->clientCanCacheAll = false;
48 cinode->clientCanCacheRead = true;
49 cFYI(1, "Level II Oplock granted on inode %p",
50 &cinode->vfs_inode);
51 } else {
52 cinode->clientCanCacheAll = false;
53 cinode->clientCanCacheRead = false;
54 }
55}
56
Pavel Shilovskyf0df7372012-09-18 16:20:26 -070057int
58smb2_open_file(const unsigned int xid, struct cifs_tcon *tcon, const char *path,
59 int disposition, int desired_access, int create_options,
60 struct cifs_fid *fid, __u32 *oplock, FILE_ALL_INFO *buf,
61 struct cifs_sb_info *cifs_sb)
62{
63 int rc;
64 __le16 *smb2_path;
65 struct smb2_file_all_info *smb2_data = NULL;
66
67 smb2_path = cifs_convert_path_to_utf16(path, cifs_sb);
68 if (smb2_path == NULL) {
69 rc = -ENOMEM;
70 goto out;
71 }
72
73 smb2_data = kzalloc(sizeof(struct smb2_file_all_info) + MAX_NAME * 2,
74 GFP_KERNEL);
75 if (smb2_data == NULL) {
76 rc = -ENOMEM;
77 goto out;
78 }
79
80 desired_access |= FILE_READ_ATTRIBUTES;
Pavel Shilovsky2e44b282012-09-18 16:20:33 -070081 *oplock = SMB2_OPLOCK_LEVEL_EXCLUSIVE;
Pavel Shilovskyf0df7372012-09-18 16:20:26 -070082
83 rc = SMB2_open(xid, tcon, smb2_path, &fid->persistent_fid,
84 &fid->volatile_fid, desired_access, disposition,
Pavel Shilovsky2e44b282012-09-18 16:20:33 -070085 0, 0, (__u8 *)oplock, smb2_data);
Pavel Shilovskyf0df7372012-09-18 16:20:26 -070086 if (rc)
87 goto out;
88
89 if (buf) {
90 /* open response does not have IndexNumber field - get it */
91 rc = SMB2_get_srv_num(xid, tcon, fid->persistent_fid,
92 fid->volatile_fid,
93 &smb2_data->IndexNumber);
94 if (rc) {
95 /* let get_inode_info disable server inode numbers */
96 smb2_data->IndexNumber = 0;
97 rc = 0;
98 }
99 move_smb2_info_to_cifs(buf, smb2_data);
100 }
101
102out:
Pavel Shilovskyf0df7372012-09-18 16:20:26 -0700103 kfree(smb2_data);
104 kfree(smb2_path);
105 return rc;
106}