1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * fs/cifs/cache.c - CIFS filesystem cache index structure definitions
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Copyright (c) 2010 Novell, Inc.
5*4882a593Smuzhiyun * Authors(s): Suresh Jayaraman (sjayaraman@suse.de>
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * This library is free software; you can redistribute it and/or modify
8*4882a593Smuzhiyun * it under the terms of the GNU Lesser General Public License as published
9*4882a593Smuzhiyun * by the Free Software Foundation; either version 2.1 of the License, or
10*4882a593Smuzhiyun * (at your option) any later version.
11*4882a593Smuzhiyun *
12*4882a593Smuzhiyun * This library is distributed in the hope that it will be useful,
13*4882a593Smuzhiyun * but WITHOUT ANY WARRANTY; without even the implied warranty of
14*4882a593Smuzhiyun * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
15*4882a593Smuzhiyun * the GNU Lesser General Public License for more details.
16*4882a593Smuzhiyun *
17*4882a593Smuzhiyun * You should have received a copy of the GNU Lesser General Public License
18*4882a593Smuzhiyun * along with this library; if not, write to the Free Software
19*4882a593Smuzhiyun * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20*4882a593Smuzhiyun */
21*4882a593Smuzhiyun #include "fscache.h"
22*4882a593Smuzhiyun #include "cifs_debug.h"
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun /*
25*4882a593Smuzhiyun * CIFS filesystem definition for FS-Cache
26*4882a593Smuzhiyun */
27*4882a593Smuzhiyun struct fscache_netfs cifs_fscache_netfs = {
28*4882a593Smuzhiyun .name = "cifs",
29*4882a593Smuzhiyun .version = 0,
30*4882a593Smuzhiyun };
31*4882a593Smuzhiyun
32*4882a593Smuzhiyun /*
33*4882a593Smuzhiyun * Register CIFS for caching with FS-Cache
34*4882a593Smuzhiyun */
cifs_fscache_register(void)35*4882a593Smuzhiyun int cifs_fscache_register(void)
36*4882a593Smuzhiyun {
37*4882a593Smuzhiyun return fscache_register_netfs(&cifs_fscache_netfs);
38*4882a593Smuzhiyun }
39*4882a593Smuzhiyun
40*4882a593Smuzhiyun /*
41*4882a593Smuzhiyun * Unregister CIFS for caching
42*4882a593Smuzhiyun */
cifs_fscache_unregister(void)43*4882a593Smuzhiyun void cifs_fscache_unregister(void)
44*4882a593Smuzhiyun {
45*4882a593Smuzhiyun fscache_unregister_netfs(&cifs_fscache_netfs);
46*4882a593Smuzhiyun }
47*4882a593Smuzhiyun
48*4882a593Smuzhiyun /*
49*4882a593Smuzhiyun * Server object for FS-Cache
50*4882a593Smuzhiyun */
51*4882a593Smuzhiyun const struct fscache_cookie_def cifs_fscache_server_index_def = {
52*4882a593Smuzhiyun .name = "CIFS.server",
53*4882a593Smuzhiyun .type = FSCACHE_COOKIE_TYPE_INDEX,
54*4882a593Smuzhiyun };
55*4882a593Smuzhiyun
extract_sharename(const char * treename)56*4882a593Smuzhiyun char *extract_sharename(const char *treename)
57*4882a593Smuzhiyun {
58*4882a593Smuzhiyun const char *src;
59*4882a593Smuzhiyun char *delim, *dst;
60*4882a593Smuzhiyun int len;
61*4882a593Smuzhiyun
62*4882a593Smuzhiyun /* skip double chars at the beginning */
63*4882a593Smuzhiyun src = treename + 2;
64*4882a593Smuzhiyun
65*4882a593Smuzhiyun /* share name is always preceded by '\\' now */
66*4882a593Smuzhiyun delim = strchr(src, '\\');
67*4882a593Smuzhiyun if (!delim)
68*4882a593Smuzhiyun return ERR_PTR(-EINVAL);
69*4882a593Smuzhiyun delim++;
70*4882a593Smuzhiyun len = strlen(delim);
71*4882a593Smuzhiyun
72*4882a593Smuzhiyun /* caller has to free the memory */
73*4882a593Smuzhiyun dst = kstrndup(delim, len, GFP_KERNEL);
74*4882a593Smuzhiyun if (!dst)
75*4882a593Smuzhiyun return ERR_PTR(-ENOMEM);
76*4882a593Smuzhiyun
77*4882a593Smuzhiyun return dst;
78*4882a593Smuzhiyun }
79*4882a593Smuzhiyun
80*4882a593Smuzhiyun static enum
cifs_fscache_super_check_aux(void * cookie_netfs_data,const void * data,uint16_t datalen,loff_t object_size)81*4882a593Smuzhiyun fscache_checkaux cifs_fscache_super_check_aux(void *cookie_netfs_data,
82*4882a593Smuzhiyun const void *data,
83*4882a593Smuzhiyun uint16_t datalen,
84*4882a593Smuzhiyun loff_t object_size)
85*4882a593Smuzhiyun {
86*4882a593Smuzhiyun struct cifs_fscache_super_auxdata auxdata;
87*4882a593Smuzhiyun const struct cifs_tcon *tcon = cookie_netfs_data;
88*4882a593Smuzhiyun
89*4882a593Smuzhiyun if (datalen != sizeof(auxdata))
90*4882a593Smuzhiyun return FSCACHE_CHECKAUX_OBSOLETE;
91*4882a593Smuzhiyun
92*4882a593Smuzhiyun memset(&auxdata, 0, sizeof(auxdata));
93*4882a593Smuzhiyun auxdata.resource_id = tcon->resource_id;
94*4882a593Smuzhiyun auxdata.vol_create_time = tcon->vol_create_time;
95*4882a593Smuzhiyun auxdata.vol_serial_number = tcon->vol_serial_number;
96*4882a593Smuzhiyun
97*4882a593Smuzhiyun if (memcmp(data, &auxdata, datalen) != 0)
98*4882a593Smuzhiyun return FSCACHE_CHECKAUX_OBSOLETE;
99*4882a593Smuzhiyun
100*4882a593Smuzhiyun return FSCACHE_CHECKAUX_OKAY;
101*4882a593Smuzhiyun }
102*4882a593Smuzhiyun
103*4882a593Smuzhiyun /*
104*4882a593Smuzhiyun * Superblock object for FS-Cache
105*4882a593Smuzhiyun */
106*4882a593Smuzhiyun const struct fscache_cookie_def cifs_fscache_super_index_def = {
107*4882a593Smuzhiyun .name = "CIFS.super",
108*4882a593Smuzhiyun .type = FSCACHE_COOKIE_TYPE_INDEX,
109*4882a593Smuzhiyun .check_aux = cifs_fscache_super_check_aux,
110*4882a593Smuzhiyun };
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun static enum
cifs_fscache_inode_check_aux(void * cookie_netfs_data,const void * data,uint16_t datalen,loff_t object_size)113*4882a593Smuzhiyun fscache_checkaux cifs_fscache_inode_check_aux(void *cookie_netfs_data,
114*4882a593Smuzhiyun const void *data,
115*4882a593Smuzhiyun uint16_t datalen,
116*4882a593Smuzhiyun loff_t object_size)
117*4882a593Smuzhiyun {
118*4882a593Smuzhiyun struct cifs_fscache_inode_auxdata auxdata;
119*4882a593Smuzhiyun struct cifsInodeInfo *cifsi = cookie_netfs_data;
120*4882a593Smuzhiyun
121*4882a593Smuzhiyun if (datalen != sizeof(auxdata))
122*4882a593Smuzhiyun return FSCACHE_CHECKAUX_OBSOLETE;
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun memset(&auxdata, 0, sizeof(auxdata));
125*4882a593Smuzhiyun auxdata.eof = cifsi->server_eof;
126*4882a593Smuzhiyun auxdata.last_write_time_sec = cifsi->vfs_inode.i_mtime.tv_sec;
127*4882a593Smuzhiyun auxdata.last_change_time_sec = cifsi->vfs_inode.i_ctime.tv_sec;
128*4882a593Smuzhiyun auxdata.last_write_time_nsec = cifsi->vfs_inode.i_mtime.tv_nsec;
129*4882a593Smuzhiyun auxdata.last_change_time_nsec = cifsi->vfs_inode.i_ctime.tv_nsec;
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun if (memcmp(data, &auxdata, datalen) != 0)
132*4882a593Smuzhiyun return FSCACHE_CHECKAUX_OBSOLETE;
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun return FSCACHE_CHECKAUX_OKAY;
135*4882a593Smuzhiyun }
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun const struct fscache_cookie_def cifs_fscache_inode_object_def = {
138*4882a593Smuzhiyun .name = "CIFS.uniqueid",
139*4882a593Smuzhiyun .type = FSCACHE_COOKIE_TYPE_DATAFILE,
140*4882a593Smuzhiyun .check_aux = cifs_fscache_inode_check_aux,
141*4882a593Smuzhiyun };
142