|  | /* | 
|  | *   fs/cifs/cache.c - CIFS filesystem cache index structure definitions | 
|  | * | 
|  | *   Copyright (c) 2010 Novell, Inc. | 
|  | *   Authors(s): Suresh Jayaraman (sjayaraman@suse.de> | 
|  | * | 
|  | *   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 "fscache.h" | 
|  | #include "cifs_debug.h" | 
|  |  | 
|  | /* | 
|  | * CIFS filesystem definition for FS-Cache | 
|  | */ | 
|  | struct fscache_netfs cifs_fscache_netfs = { | 
|  | .name = "cifs", | 
|  | .version = 0, | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * Register CIFS for caching with FS-Cache | 
|  | */ | 
|  | int cifs_fscache_register(void) | 
|  | { | 
|  | return fscache_register_netfs(&cifs_fscache_netfs); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Unregister CIFS for caching | 
|  | */ | 
|  | void cifs_fscache_unregister(void) | 
|  | { | 
|  | fscache_unregister_netfs(&cifs_fscache_netfs); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Server object for FS-Cache | 
|  | */ | 
|  | const struct fscache_cookie_def cifs_fscache_server_index_def = { | 
|  | .name = "CIFS.server", | 
|  | .type = FSCACHE_COOKIE_TYPE_INDEX, | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * Auxiliary data attached to CIFS superblock within the cache | 
|  | */ | 
|  | struct cifs_fscache_super_auxdata { | 
|  | u64	resource_id;		/* unique server resource id */ | 
|  | }; | 
|  |  | 
|  | char *extract_sharename(const char *treename) | 
|  | { | 
|  | const char *src; | 
|  | char *delim, *dst; | 
|  | int len; | 
|  |  | 
|  | /* skip double chars at the beginning */ | 
|  | src = treename + 2; | 
|  |  | 
|  | /* share name is always preceded by '\\' now */ | 
|  | delim = strchr(src, '\\'); | 
|  | if (!delim) | 
|  | return ERR_PTR(-EINVAL); | 
|  | delim++; | 
|  | len = strlen(delim); | 
|  |  | 
|  | /* caller has to free the memory */ | 
|  | dst = kstrndup(delim, len, GFP_KERNEL); | 
|  | if (!dst) | 
|  | return ERR_PTR(-ENOMEM); | 
|  |  | 
|  | return dst; | 
|  | } | 
|  |  | 
|  | static enum | 
|  | fscache_checkaux cifs_fscache_super_check_aux(void *cookie_netfs_data, | 
|  | const void *data, | 
|  | uint16_t datalen, | 
|  | loff_t object_size) | 
|  | { | 
|  | struct cifs_fscache_super_auxdata auxdata; | 
|  | const struct cifs_tcon *tcon = cookie_netfs_data; | 
|  |  | 
|  | if (datalen != sizeof(auxdata)) | 
|  | return FSCACHE_CHECKAUX_OBSOLETE; | 
|  |  | 
|  | memset(&auxdata, 0, sizeof(auxdata)); | 
|  | auxdata.resource_id = tcon->resource_id; | 
|  |  | 
|  | if (memcmp(data, &auxdata, datalen) != 0) | 
|  | return FSCACHE_CHECKAUX_OBSOLETE; | 
|  |  | 
|  | return FSCACHE_CHECKAUX_OKAY; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Superblock object for FS-Cache | 
|  | */ | 
|  | const struct fscache_cookie_def cifs_fscache_super_index_def = { | 
|  | .name = "CIFS.super", | 
|  | .type = FSCACHE_COOKIE_TYPE_INDEX, | 
|  | .check_aux = cifs_fscache_super_check_aux, | 
|  | }; | 
|  |  | 
|  | static enum | 
|  | fscache_checkaux cifs_fscache_inode_check_aux(void *cookie_netfs_data, | 
|  | const void *data, | 
|  | uint16_t datalen, | 
|  | loff_t object_size) | 
|  | { | 
|  | struct cifs_fscache_inode_auxdata auxdata; | 
|  | struct cifsInodeInfo *cifsi = cookie_netfs_data; | 
|  |  | 
|  | if (datalen != sizeof(auxdata)) | 
|  | return FSCACHE_CHECKAUX_OBSOLETE; | 
|  |  | 
|  | memset(&auxdata, 0, sizeof(auxdata)); | 
|  | auxdata.eof = cifsi->server_eof; | 
|  | auxdata.last_write_time_sec = cifsi->vfs_inode.i_mtime.tv_sec; | 
|  | auxdata.last_change_time_sec = cifsi->vfs_inode.i_ctime.tv_sec; | 
|  | auxdata.last_write_time_nsec = cifsi->vfs_inode.i_mtime.tv_nsec; | 
|  | auxdata.last_change_time_nsec = cifsi->vfs_inode.i_ctime.tv_nsec; | 
|  |  | 
|  | if (memcmp(data, &auxdata, datalen) != 0) | 
|  | return FSCACHE_CHECKAUX_OBSOLETE; | 
|  |  | 
|  | return FSCACHE_CHECKAUX_OKAY; | 
|  | } | 
|  |  | 
|  | const struct fscache_cookie_def cifs_fscache_inode_object_def = { | 
|  | .name		= "CIFS.uniqueid", | 
|  | .type		= FSCACHE_COOKIE_TYPE_DATAFILE, | 
|  | .check_aux	= cifs_fscache_inode_check_aux, | 
|  | }; |