4  *   Copyright (c) International Business Machines  Corp., 2003
 
   5  *   Author(s): Steve French (sfrench@us.ibm.com)
 
   7  *   This library is free software; you can redistribute it and/or modify
 
   8  *   it under the terms of the GNU Lesser General Public License as published
 
   9  *   by the Free Software Foundation; either version 2.1 of the License, or
 
  10  *   (at your option) any later version.
 
  12  *   This library is distributed in the hope that it will be useful,
 
  13  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
 
  14  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See
 
  15  *   the GNU Lesser General Public License for more details.
 
  17  *   You should have received a copy of the GNU Lesser General Public License
 
  18  *   along with this library; if not, write to the Free Software
 
  19  *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
 
  23 #include <linux/posix_acl_xattr.h>
 
  27 #include "cifsproto.h"
 
  28 #include "cifs_debug.h"
 
  30 #define MAX_EA_VALUE_SIZE 65535
 
  31 #define CIFS_XATTR_DOS_ATTRIB "user.DosAttrib"
 
  32 #define CIFS_XATTR_USER_PREFIX "user."
 
  33 #define CIFS_XATTR_SYSTEM_PREFIX "system."
 
  34 #define CIFS_XATTR_OS2_PREFIX "os2."
 
  35 #define CIFS_XATTR_SECURITY_PREFIX ".security"
 
  36 #define CIFS_XATTR_TRUSTED_PREFIX "trusted."
 
  37 #define XATTR_TRUSTED_PREFIX_LEN  8
 
  38 #define XATTR_SECURITY_PREFIX_LEN 9
 
  39 /* BB need to add server (Samba e.g) support for security and trusted prefix */
 
  43 int cifs_removexattr(struct dentry * direntry, const char * ea_name)
 
  46 #ifdef CONFIG_CIFS_XATTR
 
  48         struct cifs_sb_info *cifs_sb;
 
  49         struct cifsTconInfo *pTcon;
 
  50         struct super_block * sb;
 
  55         if(direntry->d_inode == NULL)
 
  57         sb = direntry->d_inode->i_sb;
 
  62         cifs_sb = CIFS_SB(sb);
 
  63         pTcon = cifs_sb->tcon;
 
  65         down(&sb->s_vfs_rename_sem);
 
  66         full_path = build_path_from_dentry(direntry);
 
  67         up(&sb->s_vfs_rename_sem);
 
  68         if(full_path == NULL) {
 
  73                 cFYI(1,("Null xattr names not supported"));
 
  74         } else if(strncmp(ea_name,CIFS_XATTR_USER_PREFIX,5)
 
  75                 && (strncmp(ea_name,CIFS_XATTR_OS2_PREFIX,4))) {
 
  76                 cFYI(1,("illegal xattr namespace %s (only user namespace supported)",ea_name));
 
  77                 /* BB what if no namespace prefix? */
 
  78                 /* Should we just pass them to server, except for
 
  79                 system and perhaps security prefixes? */
 
  81                 if(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_XATTR)
 
  84                 ea_name+=5; /* skip past user. prefix */
 
  85                 rc = CIFSSMBSetEA(xid,pTcon,full_path,ea_name,NULL,
 
  86                         (__u16)0, cifs_sb->local_nls,
 
  87                         cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
 
  96 int cifs_setxattr(struct dentry * direntry, const char * ea_name,
 
  97         const void * ea_value, size_t value_size, int flags)
 
 100 #ifdef CONFIG_CIFS_XATTR
 
 102         struct cifs_sb_info *cifs_sb;
 
 103         struct cifsTconInfo *pTcon;
 
 104         struct super_block * sb;
 
 109         if(direntry->d_inode == NULL)
 
 111         sb = direntry->d_inode->i_sb;
 
 116         cifs_sb = CIFS_SB(sb);
 
 117         pTcon = cifs_sb->tcon;
 
 119         down(&sb->s_vfs_rename_sem);
 
 120         full_path = build_path_from_dentry(direntry);
 
 121         up(&sb->s_vfs_rename_sem);
 
 122         if(full_path == NULL) {
 
 126         /* return dos attributes as pseudo xattr */
 
 127         /* return alt name if available as pseudo attr */
 
 129         /* if proc/fs/cifs/streamstoxattr is set then
 
 130                 search server for EAs or streams to 
 
 132         if(value_size > MAX_EA_VALUE_SIZE) {
 
 133                 cFYI(1,("size of EA value too large"));
 
 139         if(ea_name == NULL) {
 
 140                 cFYI(1,("Null xattr names not supported"));
 
 141         } else if(strncmp(ea_name,CIFS_XATTR_USER_PREFIX,5) == 0) {
 
 142                 if(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_XATTR)
 
 144                 if(strncmp(ea_name,CIFS_XATTR_DOS_ATTRIB,14) == 0) {
 
 145                         cFYI(1,("attempt to set cifs inode metadata"));
 
 147                 ea_name += 5; /* skip past user. prefix */
 
 148                 rc = CIFSSMBSetEA(xid,pTcon,full_path,ea_name,ea_value,
 
 149                         (__u16)value_size, cifs_sb->local_nls,
 
 150                         cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
 
 151         } else if(strncmp(ea_name, CIFS_XATTR_OS2_PREFIX,4) == 0) {
 
 152                 if(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_XATTR)
 
 155                 ea_name += 4; /* skip past os2. prefix */
 
 156                 rc = CIFSSMBSetEA(xid,pTcon,full_path,ea_name,ea_value,
 
 157                         (__u16)value_size, cifs_sb->local_nls,
 
 158                         cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
 
 161                 temp = strncmp(ea_name,POSIX_ACL_XATTR_ACCESS,
 
 162                         strlen(POSIX_ACL_XATTR_ACCESS));
 
 164 #ifdef CONFIG_CIFS_POSIX
 
 165                         if(sb->s_flags & MS_POSIXACL)
 
 166                                 rc = CIFSSMBSetPosixACL(xid, pTcon,full_path,
 
 167                                         ea_value, (const int)value_size, 
 
 168                                         ACL_TYPE_ACCESS,cifs_sb->local_nls,
 
 169                                         cifs_sb->mnt_cifs_flags & 
 
 170                                                 CIFS_MOUNT_MAP_SPECIAL_CHR);
 
 171                         cFYI(1,("set POSIX ACL rc %d",rc));
 
 173                         cFYI(1,("set POSIX ACL not supported"));
 
 175                 } else if(strncmp(ea_name,POSIX_ACL_XATTR_DEFAULT,strlen(POSIX_ACL_XATTR_DEFAULT)) == 0) {
 
 176 #ifdef CONFIG_CIFS_POSIX
 
 177                         if(sb->s_flags & MS_POSIXACL)
 
 178                                 rc = CIFSSMBSetPosixACL(xid, pTcon,full_path,
 
 179                                         ea_value, (const int)value_size, 
 
 180                                         ACL_TYPE_DEFAULT, cifs_sb->local_nls,
 
 181                                         cifs_sb->mnt_cifs_flags & 
 
 182                                                 CIFS_MOUNT_MAP_SPECIAL_CHR);
 
 183                         cFYI(1,("set POSIX default ACL rc %d",rc));
 
 185                         cFYI(1,("set default POSIX ACL not supported"));
 
 188                         cFYI(1,("illegal xattr request %s (only user namespace supported)",ea_name));
 
 189                   /* BB what if no namespace prefix? */
 
 190                   /* Should we just pass them to server, except for 
 
 191                   system and perhaps security prefixes? */
 
 202 ssize_t cifs_getxattr(struct dentry * direntry, const char * ea_name,
 
 203          void * ea_value, size_t buf_size)
 
 205         ssize_t rc = -EOPNOTSUPP;
 
 206 #ifdef CONFIG_CIFS_XATTR
 
 208         struct cifs_sb_info *cifs_sb;
 
 209         struct cifsTconInfo *pTcon;
 
 210         struct super_block * sb;
 
 215         if(direntry->d_inode == NULL)
 
 217         sb = direntry->d_inode->i_sb;
 
 223         cifs_sb = CIFS_SB(sb);
 
 224         pTcon = cifs_sb->tcon;
 
 226         down(&sb->s_vfs_rename_sem);
 
 227         full_path = build_path_from_dentry(direntry);
 
 228         up(&sb->s_vfs_rename_sem);
 
 229         if(full_path == NULL) {
 
 233         /* return dos attributes as pseudo xattr */
 
 234         /* return alt name if available as pseudo attr */
 
 235         if(ea_name == NULL) {
 
 236                 cFYI(1,("Null xattr names not supported"));
 
 237         } else if(strncmp(ea_name,CIFS_XATTR_USER_PREFIX,5) == 0) {
 
 238                 if(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_XATTR)
 
 241                 if(strncmp(ea_name,CIFS_XATTR_DOS_ATTRIB,14) == 0) {
 
 242                         cFYI(1,("attempt to query cifs inode metadata"));
 
 243                         /* revalidate/getattr then populate from inode */
 
 244                 } /* BB add else when above is implemented */
 
 245                 ea_name += 5; /* skip past user. prefix */
 
 246                 rc = CIFSSMBQueryEA(xid,pTcon,full_path,ea_name,ea_value,
 
 247                         buf_size, cifs_sb->local_nls,
 
 248                         cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
 
 249         } else if(strncmp(ea_name, CIFS_XATTR_OS2_PREFIX,4) == 0) {
 
 250                 if(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_XATTR)
 
 253                 ea_name += 4; /* skip past os2. prefix */
 
 254                 rc = CIFSSMBQueryEA(xid,pTcon,full_path,ea_name,ea_value,
 
 255                         buf_size, cifs_sb->local_nls,
 
 256                         cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
 
 257         } else if(strncmp(ea_name,POSIX_ACL_XATTR_ACCESS,
 
 258                           strlen(POSIX_ACL_XATTR_ACCESS)) == 0) {
 
 259 #ifdef CONFIG_CIFS_POSIX
 
 260                 if(sb->s_flags & MS_POSIXACL)
 
 261                         rc = CIFSSMBGetPosixACL(xid, pTcon, full_path,
 
 262                                 ea_value, buf_size, ACL_TYPE_ACCESS, 
 
 264                                 cifs_sb->mnt_cifs_flags & 
 
 265                                         CIFS_MOUNT_MAP_SPECIAL_CHR);
 
 266 /*              else if(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) {
 
 269                         rc = CIFSSMBOpen(xid, pTcon, full_path,
 
 270                                          FILE_OPEN, GENERIC_READ, 0, &fid,
 
 271                                          &oplock, NULL, cifs_sb->local_nls,
 
 272                                          cifs_sb->mnt_cifs_flags &
 
 273                                          CIFS_MOUNT_MAP_SPECIAL_CHR);
 
 275                                 rc = CIFSSMBGetCIFSACL(xid, pTcon, fid,
 
 278                                 CIFSSMBClose(xid, pTcon, fid)
 
 280                 } */  /* BB enable after fixing up return data */
 
 283                 cFYI(1,("query POSIX ACL not supported yet"));
 
 284 #endif /* CONFIG_CIFS_POSIX */
 
 285         } else if(strncmp(ea_name,POSIX_ACL_XATTR_DEFAULT,
 
 286                           strlen(POSIX_ACL_XATTR_DEFAULT)) == 0) {
 
 287 #ifdef CONFIG_CIFS_POSIX
 
 288                 if(sb->s_flags & MS_POSIXACL)
 
 289                         rc = CIFSSMBGetPosixACL(xid, pTcon, full_path,
 
 290                                 ea_value, buf_size, ACL_TYPE_DEFAULT, 
 
 292                                 cifs_sb->mnt_cifs_flags & 
 
 293                                         CIFS_MOUNT_MAP_SPECIAL_CHR);
 
 295                 cFYI(1,("query POSIX default ACL not supported yet"));
 
 297         } else if(strncmp(ea_name,
 
 298                   CIFS_XATTR_TRUSTED_PREFIX,XATTR_TRUSTED_PREFIX_LEN) == 0) {
 
 299                 cFYI(1,("Trusted xattr namespace not supported yet"));
 
 300         } else if(strncmp(ea_name,
 
 301                   CIFS_XATTR_SECURITY_PREFIX,XATTR_SECURITY_PREFIX_LEN) == 0) {
 
 302                 cFYI(1,("Security xattr namespace not supported yet"));
 
 304                 cFYI(1,("illegal xattr name request %s (only user namespace supported)",ea_name));
 
 307         /* We could add an additional check for streams ie 
 
 308             if proc/fs/cifs/streamstoxattr is set then
 
 309                 search server for EAs or streams to 
 
 322 ssize_t cifs_listxattr(struct dentry * direntry, char * data, size_t buf_size)
 
 324         ssize_t rc = -EOPNOTSUPP;
 
 325 #ifdef CONFIG_CIFS_XATTR
 
 327         struct cifs_sb_info *cifs_sb;
 
 328         struct cifsTconInfo *pTcon;
 
 329         struct super_block * sb;
 
 334         if(direntry->d_inode == NULL)
 
 336         sb = direntry->d_inode->i_sb;
 
 341         cifs_sb = CIFS_SB(sb);
 
 342         pTcon = cifs_sb->tcon;
 
 344         down(&sb->s_vfs_rename_sem);
 
 345         full_path = build_path_from_dentry(direntry);
 
 346         up(&sb->s_vfs_rename_sem);
 
 347         if(full_path == NULL) {
 
 351         /* return dos attributes as pseudo xattr */
 
 352         /* return alt name if available as pseudo attr */
 
 354         /* if proc/fs/cifs/streamstoxattr is set then
 
 355                 search server for EAs or streams to 
 
 357         rc = CIFSSMBQAllEAs(xid,pTcon,full_path,data,buf_size,
 
 359                                 cifs_sb->mnt_cifs_flags & 
 
 360                                         CIFS_MOUNT_MAP_SPECIAL_CHR);