Quellcodebibliothek Statistik Leitseite products/sources/formale Sprachen/C/Linux/fs/smb/client/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 93 kB image not shown  

Quelle  inode.c   Sprache: C

 
// SPDX-License-Identifier: LGPL-2.1
/*
 *
 *   Copyright (C) International Business Machines  Corp., 2002,2010
 *   Author(s): Steve French (sfrench@us.ibm.com)
 *
 */

#include <linux/fs.h>
#include <linux/stat.h>
#include <linux/slab.h>
#include <linux/pagemap.h>
#include <linux/freezer.h>
#include <linux/sched/signal.h>
#include <linux/wait_bit.h>
#include <linux/fiemap.h>
#include <asm/div64.h>
#include "cifsfs.h"
#include "cifspdu.h"
#include "cifsglob.h"
#include "cifsproto.h"
#include "smb2proto.h"
#include "cifs_debug.h"
#include "cifs_fs_sb.h"
#include "cifs_unicode.h"
#include "fscache.h"
#include "fs_context.h"
#include "cifs_ioctl.h"
#include "cached_dir.h"
#include "reparse.h"

/*
 * Set parameters for the netfs library
 */

static void cifs_set_netfs_context(struct inode *inode)
{
 struct cifsInodeInfo *cifs_i = CIFS_I(inode);

 netfs_inode_init(&cifs_i->netfs, &cifs_req_ops, true);
}

static void cifs_set_ops(struct inode *inode)
{
 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
 struct netfs_inode *ictx = netfs_inode(inode);

 switch (inode->i_mode & S_IFMT) {
 case S_IFREG:
  inode->i_op = &cifs_file_inode_ops;
  if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DIRECT_IO) {
   set_bit(NETFS_ICTX_UNBUFFERED, &ictx->flags);
   if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_BRL)
    inode->i_fop = &cifs_file_direct_nobrl_ops;
   else
    inode->i_fop = &cifs_file_direct_ops;
  } else if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_STRICT_IO) {
   if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_BRL)
    inode->i_fop = &cifs_file_strict_nobrl_ops;
   else
    inode->i_fop = &cifs_file_strict_ops;
  } else if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_BRL)
   inode->i_fop = &cifs_file_nobrl_ops;
  else { /* not direct, send byte range locks */
   inode->i_fop = &cifs_file_ops;
  }

  /* check if server can support readahead */
  if (cifs_sb_master_tcon(cifs_sb)->ses->server->max_read <
    PAGE_SIZE + MAX_CIFS_HDR_SIZE)
   inode->i_data.a_ops = &cifs_addr_ops_smallbuf;
  else
   inode->i_data.a_ops = &cifs_addr_ops;
  mapping_set_large_folios(inode->i_mapping);
  break;
 case S_IFDIR:
  if (IS_AUTOMOUNT(inode)) {
   inode->i_op = &cifs_namespace_inode_operations;
  } else {
   inode->i_op = &cifs_dir_inode_ops;
   inode->i_fop = &cifs_dir_ops;
  }
  break;
 case S_IFLNK:
  inode->i_op = &cifs_symlink_inode_ops;
  break;
 default:
  init_special_inode(inode, inode->i_mode, inode->i_rdev);
  break;
 }
}

/* check inode attributes against fattr. If they don't match, tag the
 * inode for cache invalidation
 */

static void
cifs_revalidate_cache(struct inode *inode, struct cifs_fattr *fattr)
{
 struct cifs_fscache_inode_coherency_data cd;
 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
 struct timespec64 mtime;

 cifs_dbg(FYI, "%s: revalidating inode %llu\n",
   __func__, cifs_i->uniqueid);

 if (inode->i_state & I_NEW) {
  cifs_dbg(FYI, "%s: inode %llu is new\n",
    __func__, cifs_i->uniqueid);
  return;
 }

 /* don't bother with revalidation if we have an oplock */
 if (CIFS_CACHE_READ(cifs_i)) {
  cifs_dbg(FYI, "%s: inode %llu is oplocked\n",
    __func__, cifs_i->uniqueid);
  return;
 }

  /* revalidate if mtime or size have changed */
 fattr->cf_mtime = timestamp_truncate(fattr->cf_mtime, inode);
 mtime = inode_get_mtime(inode);
 if (timespec64_equal(&mtime, &fattr->cf_mtime) &&
     cifs_i->netfs.remote_i_size == fattr->cf_eof) {
  cifs_dbg(FYI, "%s: inode %llu is unchanged\n",
    __func__, cifs_i->uniqueid);
  return;
 }

 cifs_dbg(FYI, "%s: invalidating inode %llu mapping\n",
   __func__, cifs_i->uniqueid);
 set_bit(CIFS_INO_INVALID_MAPPING, &cifs_i->flags);
 /* Invalidate fscache cookie */
 cifs_fscache_fill_coherency(&cifs_i->netfs.inode, &cd);
 fscache_invalidate(cifs_inode_cookie(inode), &cd, i_size_read(inode), 0);
}

/*
 * copy nlink to the inode, unless it wasn't provided.  Provide
 * sane values if we don't have an existing one and none was provided
 */

static void
cifs_nlink_fattr_to_inode(struct inode *inode, struct cifs_fattr *fattr)
{
 /*
 * if we're in a situation where we can't trust what we
 * got from the server (readdir, some non-unix cases)
 * fake reasonable values
 */

 if (fattr->cf_flags & CIFS_FATTR_UNKNOWN_NLINK) {
  /* only provide fake values on a new inode */
  if (inode->i_state & I_NEW) {
   if (fattr->cf_cifsattrs & ATTR_DIRECTORY)
    set_nlink(inode, 2);
   else
    set_nlink(inode, 1);
  }
  return;
 }

 /* we trust the server, so update it */
 set_nlink(inode, fattr->cf_nlink);
}

/* populate an inode with info from a cifs_fattr struct */
int
cifs_fattr_to_inode(struct inode *inode, struct cifs_fattr *fattr,
      bool from_readdir)
{
 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);

 if (!(inode->i_state & I_NEW) &&
     unlikely(inode_wrong_type(inode, fattr->cf_mode))) {
  CIFS_I(inode)->time = 0; /* force reval */
  return -ESTALE;
 }
 if (inode->i_state & I_NEW)
  CIFS_I(inode)->netfs.zero_point = fattr->cf_eof;

 cifs_revalidate_cache(inode, fattr);

 spin_lock(&inode->i_lock);
 fattr->cf_mtime = timestamp_truncate(fattr->cf_mtime, inode);
 fattr->cf_atime = timestamp_truncate(fattr->cf_atime, inode);
 fattr->cf_ctime = timestamp_truncate(fattr->cf_ctime, inode);
 /* we do not want atime to be less than mtime, it broke some apps */
 if (timespec64_compare(&fattr->cf_atime, &fattr->cf_mtime) < 0)
  inode_set_atime_to_ts(inode, fattr->cf_mtime);
 else
  inode_set_atime_to_ts(inode, fattr->cf_atime);
 inode_set_mtime_to_ts(inode, fattr->cf_mtime);
 inode_set_ctime_to_ts(inode, fattr->cf_ctime);
 inode->i_rdev = fattr->cf_rdev;
 cifs_nlink_fattr_to_inode(inode, fattr);
 inode->i_uid = fattr->cf_uid;
 inode->i_gid = fattr->cf_gid;

 /* if dynperm is set, don't clobber existing mode */
 if (inode->i_state & I_NEW ||
     !(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM))
  inode->i_mode = fattr->cf_mode;

 cifs_i->cifsAttrs = fattr->cf_cifsattrs;
 cifs_i->reparse_tag = fattr->cf_cifstag;

 if (fattr->cf_flags & CIFS_FATTR_NEED_REVAL)
  cifs_i->time = 0;
 else
  cifs_i->time = jiffies;

 if (fattr->cf_flags & CIFS_FATTR_DELETE_PENDING)
  set_bit(CIFS_INO_DELETE_PENDING, &cifs_i->flags);
 else
  clear_bit(CIFS_INO_DELETE_PENDING, &cifs_i->flags);

 cifs_i->netfs.remote_i_size = fattr->cf_eof;
 /*
 * Can't safely change the file size here if the client is writing to
 * it due to potential races.
 */

 if (is_size_safe_to_change(cifs_i, fattr->cf_eof, from_readdir)) {
  i_size_write(inode, fattr->cf_eof);

  /*
 * i_blocks is not related to (i_size / i_blksize),
 * but instead 512 byte (2**9) size is required for
 * calculating num blocks.
 */

  inode->i_blocks = (512 - 1 + fattr->cf_bytes) >> 9;
 }

 if (S_ISLNK(fattr->cf_mode) && fattr->cf_symlink_target) {
  kfree(cifs_i->symlink_target);
  cifs_i->symlink_target = fattr->cf_symlink_target;
  fattr->cf_symlink_target = NULL;
 }
 spin_unlock(&inode->i_lock);

 if (fattr->cf_flags & CIFS_FATTR_JUNCTION)
  inode->i_flags |= S_AUTOMOUNT;
 if (inode->i_state & I_NEW) {
  cifs_set_netfs_context(inode);
  cifs_set_ops(inode);
 }
 return 0;
}

void
cifs_fill_uniqueid(struct super_block *sb, struct cifs_fattr *fattr)
{
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);

 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM)
  return;

 fattr->cf_uniqueid = iunique(sb, ROOT_I);
}

/* Fill a cifs_fattr struct with info from FILE_UNIX_BASIC_INFO. */
void
cifs_unix_basic_to_fattr(struct cifs_fattr *fattr, FILE_UNIX_BASIC_INFO *info,
    struct cifs_sb_info *cifs_sb)
{
 memset(fattr, 0, sizeof(*fattr));
 fattr->cf_uniqueid = le64_to_cpu(info->UniqueId);
 fattr->cf_bytes = le64_to_cpu(info->NumOfBytes);
 fattr->cf_eof = le64_to_cpu(info->EndOfFile);

 fattr->cf_atime = cifs_NTtimeToUnix(info->LastAccessTime);
 fattr->cf_mtime = cifs_NTtimeToUnix(info->LastModificationTime);
 fattr->cf_ctime = cifs_NTtimeToUnix(info->LastStatusChange);
 /* old POSIX extensions don't get create time */

 fattr->cf_mode = le64_to_cpu(info->Permissions);

 /*
 * Since we set the inode type below we need to mask off
 * to avoid strange results if bits set above.
 */

 fattr->cf_mode &= ~S_IFMT;
 switch (le32_to_cpu(info->Type)) {
 case UNIX_FILE:
  fattr->cf_mode |= S_IFREG;
  fattr->cf_dtype = DT_REG;
  break;
 case UNIX_SYMLINK:
  fattr->cf_mode |= S_IFLNK;
  fattr->cf_dtype = DT_LNK;
  break;
 case UNIX_DIR:
  fattr->cf_mode |= S_IFDIR;
  fattr->cf_dtype = DT_DIR;
  break;
 case UNIX_CHARDEV:
  fattr->cf_mode |= S_IFCHR;
  fattr->cf_dtype = DT_CHR;
  fattr->cf_rdev = MKDEV(le64_to_cpu(info->DevMajor),
           le64_to_cpu(info->DevMinor) & MINORMASK);
  break;
 case UNIX_BLOCKDEV:
  fattr->cf_mode |= S_IFBLK;
  fattr->cf_dtype = DT_BLK;
  fattr->cf_rdev = MKDEV(le64_to_cpu(info->DevMajor),
           le64_to_cpu(info->DevMinor) & MINORMASK);
  break;
 case UNIX_FIFO:
  fattr->cf_mode |= S_IFIFO;
  fattr->cf_dtype = DT_FIFO;
  break;
 case UNIX_SOCKET:
  fattr->cf_mode |= S_IFSOCK;
  fattr->cf_dtype = DT_SOCK;
  break;
 default:
  /* safest to call it a file if we do not know */
  fattr->cf_mode |= S_IFREG;
  fattr->cf_dtype = DT_REG;
  cifs_dbg(FYI, "unknown type %d\n", le32_to_cpu(info->Type));
  break;
 }

 fattr->cf_uid = cifs_sb->ctx->linux_uid;
 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_OVERR_UID)) {
  u64 id = le64_to_cpu(info->Uid);
  if (id < ((uid_t)-1)) {
   kuid_t uid = make_kuid(&init_user_ns, id);
   if (uid_valid(uid))
    fattr->cf_uid = uid;
  }
 }
 
 fattr->cf_gid = cifs_sb->ctx->linux_gid;
 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_OVERR_GID)) {
  u64 id = le64_to_cpu(info->Gid);
  if (id < ((gid_t)-1)) {
   kgid_t gid = make_kgid(&init_user_ns, id);
   if (gid_valid(gid))
    fattr->cf_gid = gid;
  }
 }

 fattr->cf_nlink = le64_to_cpu(info->Nlinks);
}

/*
 * Fill a cifs_fattr struct with fake inode info.
 *
 * Needed to setup cifs_fattr data for the directory which is the
 * junction to the new submount (ie to setup the fake directory
 * which represents a DFS referral or reparse mount point).
 */

static void cifs_create_junction_fattr(struct cifs_fattr *fattr,
           struct super_block *sb)
{
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);

 cifs_dbg(FYI, "%s: creating fake fattr\n", __func__);

 memset(fattr, 0, sizeof(*fattr));
 fattr->cf_mode = S_IFDIR | S_IXUGO | S_IRWXU;
 fattr->cf_uid = cifs_sb->ctx->linux_uid;
 fattr->cf_gid = cifs_sb->ctx->linux_gid;
 ktime_get_coarse_real_ts64(&fattr->cf_mtime);
 fattr->cf_atime = fattr->cf_ctime = fattr->cf_mtime;
 fattr->cf_nlink = 2;
 fattr->cf_flags = CIFS_FATTR_JUNCTION;
}

/* Update inode with final fattr data */
static int update_inode_info(struct super_block *sb,
        struct cifs_fattr *fattr,
        struct inode **inode)
{
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 int rc = 0;

 if (!*inode) {
  *inode = cifs_iget(sb, fattr);
  if (!*inode)
   rc = -ENOMEM;
  return rc;
 }
 /* We already have inode, update it.
 *
 * If file type or uniqueid is different, return error.
 */

 if (unlikely((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) &&
       CIFS_I(*inode)->uniqueid != fattr->cf_uniqueid)) {
  CIFS_I(*inode)->time = 0; /* force reval */
  return -ESTALE;
 }
 return cifs_fattr_to_inode(*inode, fattr, false);
}

#ifdef CONFIG_CIFS_ALLOW_INSECURE_LEGACY
static int
cifs_get_file_info_unix(struct file *filp)
{
 int rc;
 unsigned int xid;
 FILE_UNIX_BASIC_INFO find_data;
 struct cifs_fattr fattr = {};
 struct inode *inode = file_inode(filp);
 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
 struct cifsFileInfo *cfile = filp->private_data;
 struct cifs_tcon *tcon = tlink_tcon(cfile->tlink);

 xid = get_xid();

 if (cfile->symlink_target) {
  fattr.cf_symlink_target = kstrdup(cfile->symlink_target, GFP_KERNEL);
  if (!fattr.cf_symlink_target) {
   rc = -ENOMEM;
   goto cifs_gfiunix_out;
  }
 }

 rc = CIFSSMBUnixQFileInfo(xid, tcon, cfile->fid.netfid, &find_data);
 if (!rc) {
  cifs_unix_basic_to_fattr(&fattr, &find_data, cifs_sb);
 } else if (rc == -EREMOTE) {
  cifs_create_junction_fattr(&fattr, inode->i_sb);
 } else
  goto cifs_gfiunix_out;

 rc = cifs_fattr_to_inode(inode, &fattr, false);

cifs_gfiunix_out:
 free_xid(xid);
 return rc;
}

static int cifs_get_unix_fattr(const unsigned char *full_path,
          struct super_block *sb,
          struct cifs_fattr *fattr,
          struct inode **pinode,
          const unsigned int xid)
{
 struct TCP_Server_Info *server;
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 FILE_UNIX_BASIC_INFO find_data;
 struct cifs_tcon *tcon;
 struct tcon_link *tlink;
 int rc, tmprc;

 cifs_dbg(FYI, "Getting info on %s\n", full_path);

 tlink = cifs_sb_tlink(cifs_sb);
 if (IS_ERR(tlink))
  return PTR_ERR(tlink);
 tcon = tlink_tcon(tlink);
 server = tcon->ses->server;

 /* could have done a find first instead but this returns more info */
 rc = CIFSSMBUnixQPathInfo(xid, tcon, full_path, &find_data,
      cifs_sb->local_nls, cifs_remap(cifs_sb));
 cifs_dbg(FYI, "%s: query path info: rc = %d\n", __func__, rc);
 cifs_put_tlink(tlink);

 if (!rc) {
  cifs_unix_basic_to_fattr(fattr, &find_data, cifs_sb);
 } else if (rc == -EREMOTE) {
  cifs_create_junction_fattr(fattr, sb);
  rc = 0;
 } else {
  return rc;
 }

 if (!*pinode)
  cifs_fill_uniqueid(sb, fattr);

 /* check for Minshall+French symlinks */
 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MF_SYMLINKS) {
  tmprc = check_mf_symlink(xid, tcon, cifs_sb, fattr, full_path);
  cifs_dbg(FYI, "check_mf_symlink: %d\n", tmprc);
 }

 if (S_ISLNK(fattr->cf_mode) && !fattr->cf_symlink_target) {
  if (!server->ops->query_symlink)
   return -EOPNOTSUPP;
  rc = server->ops->query_symlink(xid, tcon,
      cifs_sb, full_path,
      &fattr->cf_symlink_target);
  cifs_dbg(FYI, "%s: query_symlink: %d\n", __func__, rc);
 }
 return rc;
}

int cifs_get_inode_info_unix(struct inode **pinode,
        const unsigned char *full_path,
        struct super_block *sb, unsigned int xid)
{
 struct cifs_fattr fattr = {};
 int rc;

 rc = cifs_get_unix_fattr(full_path, sb, &fattr, pinode, xid);
 if (rc)
  goto out;

 rc = update_inode_info(sb, &fattr, pinode);
out:
 kfree(fattr.cf_symlink_target);
 return rc;
}
#else
static inline int cifs_get_unix_fattr(const unsigned char *full_path,
          struct super_block *sb,
          struct cifs_fattr *fattr,
          struct inode **pinode,
          const unsigned int xid)
{
 return -EOPNOTSUPP;
}

int cifs_get_inode_info_unix(struct inode **pinode,
        const unsigned char *full_path,
        struct super_block *sb, unsigned int xid)
{
 return -EOPNOTSUPP;
}
#endif /* CONFIG_CIFS_ALLOW_INSECURE_LEGACY */

static int
cifs_sfu_type(struct cifs_fattr *fattr, const char *path,
       struct cifs_sb_info *cifs_sb, unsigned int xid)
{
 int rc;
 __u32 oplock;
 struct tcon_link *tlink;
 struct cifs_tcon *tcon;
 struct cifs_fid fid;
 struct cifs_open_parms oparms;
 struct cifs_io_parms io_parms = {0};
 char *symlink_buf_utf16;
 unsigned int symlink_len_utf16;
 char buf[24];
 unsigned int bytes_read;
 char *pbuf;
 int buf_type = CIFS_NO_BUFFER;

 pbuf = buf;

 fattr->cf_mode &= ~S_IFMT;

 if (fattr->cf_eof == 0) {
  cifs_dbg(FYI, "Fifo\n");
  fattr->cf_mode |= S_IFIFO;
  fattr->cf_dtype = DT_FIFO;
  return 0;
 } else if (fattr->cf_eof > 1 && fattr->cf_eof < 8) {
  fattr->cf_mode |= S_IFREG;
  fattr->cf_dtype = DT_REG;
  return -EINVAL;  /* EOPNOTSUPP? */
 }

 tlink = cifs_sb_tlink(cifs_sb);
 if (IS_ERR(tlink))
  return PTR_ERR(tlink);
 tcon = tlink_tcon(tlink);

 oparms = (struct cifs_open_parms) {
  .tcon = tcon,
  .cifs_sb = cifs_sb,
  .desired_access = GENERIC_READ,
  .create_options = cifs_create_options(cifs_sb, CREATE_NOT_DIR),
  .disposition = FILE_OPEN,
  .path = path,
  .fid = &fid,
 };

 if (tcon->ses->server->oplocks)
  oplock = REQ_OPLOCK;
 else
  oplock = 0;
 rc = tcon->ses->server->ops->open(xid, &oparms, &oplock, NULL);
 if (rc) {
  cifs_dbg(FYI, "check sfu type of %s, open rc = %d\n", path, rc);
  cifs_put_tlink(tlink);
  return rc;
 }

 /* Read header */
 io_parms.netfid = fid.netfid;
 io_parms.pid = current->tgid;
 io_parms.tcon = tcon;
 io_parms.offset = 0;
 io_parms.length = 24;

 rc = tcon->ses->server->ops->sync_read(xid, &fid, &io_parms,
     &bytes_read, &pbuf, &buf_type);
 if ((rc == 0) && (bytes_read >= 8)) {
  if (memcmp("IntxBLK\0", pbuf, 8) == 0) {
   cifs_dbg(FYI, "Block device\n");
   fattr->cf_mode |= S_IFBLK;
   fattr->cf_dtype = DT_BLK;
   if (bytes_read == 24) {
    /* we have enough to decode dev num */
    __u64 mjr; /* major */
    __u64 mnr; /* minor */
    mjr = le64_to_cpu(*(__le64 *)(pbuf+8));
    mnr = le64_to_cpu(*(__le64 *)(pbuf+16));
    fattr->cf_rdev = MKDEV(mjr, mnr);
   } else if (bytes_read == 16) {
    /*
 * Windows NFS server before Windows Server 2012
 * stores major and minor number in SFU-modified
 * style, just as 32-bit numbers. Recognize it.
 */

    __u32 mjr; /* major */
    __u32 mnr; /* minor */
    mjr = le32_to_cpu(*(__le32 *)(pbuf+8));
    mnr = le32_to_cpu(*(__le32 *)(pbuf+12));
    fattr->cf_rdev = MKDEV(mjr, mnr);
   }
  } else if (memcmp("IntxCHR\0", pbuf, 8) == 0) {
   cifs_dbg(FYI, "Char device\n");
   fattr->cf_mode |= S_IFCHR;
   fattr->cf_dtype = DT_CHR;
   if (bytes_read == 24) {
    /* we have enough to decode dev num */
    __u64 mjr; /* major */
    __u64 mnr; /* minor */
    mjr = le64_to_cpu(*(__le64 *)(pbuf+8));
    mnr = le64_to_cpu(*(__le64 *)(pbuf+16));
    fattr->cf_rdev = MKDEV(mjr, mnr);
   } else if (bytes_read == 16) {
    /*
 * Windows NFS server before Windows Server 2012
 * stores major and minor number in SFU-modified
 * style, just as 32-bit numbers. Recognize it.
 */

    __u32 mjr; /* major */
    __u32 mnr; /* minor */
    mjr = le32_to_cpu(*(__le32 *)(pbuf+8));
    mnr = le32_to_cpu(*(__le32 *)(pbuf+12));
    fattr->cf_rdev = MKDEV(mjr, mnr);
   }
  } else if (memcmp("LnxSOCK", pbuf, 8) == 0) {
   cifs_dbg(FYI, "Socket\n");
   fattr->cf_mode |= S_IFSOCK;
   fattr->cf_dtype = DT_SOCK;
  } else if (memcmp("IntxLNK\1", pbuf, 8) == 0) {
   cifs_dbg(FYI, "Symlink\n");
   fattr->cf_mode |= S_IFLNK;
   fattr->cf_dtype = DT_LNK;
   if ((fattr->cf_eof > 8) && (fattr->cf_eof % 2 == 0)) {
    symlink_buf_utf16 = kmalloc(fattr->cf_eof-8 + 1, GFP_KERNEL);
    if (symlink_buf_utf16) {
     io_parms.offset = 8;
     io_parms.length = fattr->cf_eof-8 + 1;
     buf_type = CIFS_NO_BUFFER;
     rc = tcon->ses->server->ops->sync_read(xid, &fid, &io_parms,
                &symlink_len_utf16,
                &symlink_buf_utf16,
                &buf_type);
     /*
 * Check that read buffer has valid length and does not
 * contain UTF-16 null codepoint (via UniStrnlen() call)
 * because Linux cannot process symlink with null byte.
 */

     if ((rc == 0) &&
         (symlink_len_utf16 > 0) &&
         (symlink_len_utf16 < fattr->cf_eof-8 + 1) &&
         (symlink_len_utf16 % 2 == 0) &&
         (UniStrnlen((wchar_t *)symlink_buf_utf16, symlink_len_utf16/2) == symlink_len_utf16/2)) {
      fattr->cf_symlink_target =
       cifs_strndup_from_utf16(symlink_buf_utf16,
          symlink_len_utf16,
          true,
          cifs_sb->local_nls);
      if (!fattr->cf_symlink_target)
       rc = -ENOMEM;
     }
     kfree(symlink_buf_utf16);
    } else {
     rc = -ENOMEM;
    }
   }
  } else if (memcmp("LnxFIFO", pbuf, 8) == 0) {
   cifs_dbg(FYI, "FIFO\n");
   fattr->cf_mode |= S_IFIFO;
   fattr->cf_dtype = DT_FIFO;
  } else {
   fattr->cf_mode |= S_IFREG; /* file? */
   fattr->cf_dtype = DT_REG;
   rc = -EOPNOTSUPP;
  }
 } else if ((rc == 0) && (bytes_read == 1) && (pbuf[0] == '\0')) {
  cifs_dbg(FYI, "Socket\n");
  fattr->cf_mode |= S_IFSOCK;
  fattr->cf_dtype = DT_SOCK;
 } else {
  fattr->cf_mode |= S_IFREG; /* then it is a file */
  fattr->cf_dtype = DT_REG;
  rc = -EOPNOTSUPP; /* or some unknown SFU type */
 }

 tcon->ses->server->ops->close(xid, tcon, &fid);
 cifs_put_tlink(tlink);
 return rc;
}

#define SFBITS_MASK (S_ISVTX | S_ISGID | S_ISUID)  /* SETFILEBITS valid bits */

/*
 * Fetch mode bits as provided by SFU.
 *
 * FIXME: Doesn't this clobber the type bit we got from cifs_sfu_type ?
 */

static int cifs_sfu_mode(struct cifs_fattr *fattr, const unsigned char *path,
    struct cifs_sb_info *cifs_sb, unsigned int xid)
{
#ifdef CONFIG_CIFS_XATTR
 ssize_t rc;
 char ea_value[4];
 __u32 mode;
 struct tcon_link *tlink;
 struct cifs_tcon *tcon;

 tlink = cifs_sb_tlink(cifs_sb);
 if (IS_ERR(tlink))
  return PTR_ERR(tlink);
 tcon = tlink_tcon(tlink);

 if (tcon->ses->server->ops->query_all_EAs == NULL) {
  cifs_put_tlink(tlink);
  return -EOPNOTSUPP;
 }

 rc = tcon->ses->server->ops->query_all_EAs(xid, tcon, path,
   "SETFILEBITS", ea_value, 4 /* size of buf */,
   cifs_sb);
 cifs_put_tlink(tlink);
 if (rc < 0)
  return (int)rc;
 else if (rc > 3) {
  mode = le32_to_cpu(*((__le32 *)ea_value));
  fattr->cf_mode &= ~SFBITS_MASK;
  cifs_dbg(FYI, "special bits 0%o org mode 0%o\n",
    mode, fattr->cf_mode);
  fattr->cf_mode = (mode & SFBITS_MASK) | fattr->cf_mode;
  cifs_dbg(FYI, "special mode bits 0%o\n", mode);
 }

 return 0;
#else
 return -EOPNOTSUPP;
#endif
}

#define POSIX_TYPE_FILE    0
#define POSIX_TYPE_DIR     1
#define POSIX_TYPE_SYMLINK 2
#define POSIX_TYPE_CHARDEV 3
#define POSIX_TYPE_BLKDEV  4
#define POSIX_TYPE_FIFO    5
#define POSIX_TYPE_SOCKET  6

#define POSIX_X_OTH      0000001
#define POSIX_W_OTH      0000002
#define POSIX_R_OTH      0000004
#define POSIX_X_GRP      0000010
#define POSIX_W_GRP      0000020
#define POSIX_R_GRP      0000040
#define POSIX_X_USR      0000100
#define POSIX_W_USR      0000200
#define POSIX_R_USR      0000400
#define POSIX_STICKY     0001000
#define POSIX_SET_GID    0002000
#define POSIX_SET_UID    0004000

#define POSIX_OTH_MASK      0000007
#define POSIX_GRP_MASK      0000070
#define POSIX_USR_MASK      0000700
#define POSIX_PERM_MASK     0000777
#define POSIX_FILETYPE_MASK 0070000

#define POSIX_FILETYPE_SHIFT 12

static u32 wire_perms_to_posix(u32 wire)
{
 u32 mode = 0;

 mode |= (wire & POSIX_X_OTH) ? S_IXOTH : 0;
 mode |= (wire & POSIX_W_OTH) ? S_IWOTH : 0;
 mode |= (wire & POSIX_R_OTH) ? S_IROTH : 0;
 mode |= (wire & POSIX_X_GRP) ? S_IXGRP : 0;
 mode |= (wire & POSIX_W_GRP) ? S_IWGRP : 0;
 mode |= (wire & POSIX_R_GRP) ? S_IRGRP : 0;
 mode |= (wire & POSIX_X_USR) ? S_IXUSR : 0;
 mode |= (wire & POSIX_W_USR) ? S_IWUSR : 0;
 mode |= (wire & POSIX_R_USR) ? S_IRUSR : 0;
 mode |= (wire & POSIX_STICKY) ? S_ISVTX : 0;
 mode |= (wire & POSIX_SET_GID) ? S_ISGID : 0;
 mode |= (wire & POSIX_SET_UID) ? S_ISUID : 0;

 return mode;
}

static u32 posix_filetypes[] = {
 S_IFREG,
 S_IFDIR,
 S_IFLNK,
 S_IFCHR,
 S_IFBLK,
 S_IFIFO,
 S_IFSOCK
};

static u32 wire_filetype_to_posix(u32 wire_type)
{
 if (wire_type >= ARRAY_SIZE(posix_filetypes)) {
  pr_warn("Unexpected type %u", wire_type);
  return 0;
 }
 return posix_filetypes[wire_type];
}

umode_t wire_mode_to_posix(u32 wire, bool is_dir)
{
 u32 wire_type;
 u32 mode;

 wire_type = (wire & POSIX_FILETYPE_MASK) >> POSIX_FILETYPE_SHIFT;
 /* older servers do not set POSIX file type in the mode field in the response */
 if ((wire_type == 0) && is_dir)
  mode = wire_perms_to_posix(wire) | S_IFDIR;
 else
  mode = (wire_perms_to_posix(wire) | wire_filetype_to_posix(wire_type));
 return (umode_t)mode;
}

/* Fill a cifs_fattr struct with info from POSIX info struct */
static void smb311_posix_info_to_fattr(struct cifs_fattr *fattr,
           struct cifs_open_info_data *data,
           struct super_block *sb)
{
 struct smb311_posix_qinfo *info = &data->posix_fi;
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 struct cifs_tcon *tcon = cifs_sb_master_tcon(cifs_sb);

 memset(fattr, 0, sizeof(*fattr));

 /* no fattr->flags to set */
 fattr->cf_cifsattrs = le32_to_cpu(info->DosAttributes);
 fattr->cf_uniqueid = le64_to_cpu(info->Inode);

 if (info->LastAccessTime)
  fattr->cf_atime = cifs_NTtimeToUnix(info->LastAccessTime);
 else
  ktime_get_coarse_real_ts64(&fattr->cf_atime);

 fattr->cf_ctime = cifs_NTtimeToUnix(info->ChangeTime);
 fattr->cf_mtime = cifs_NTtimeToUnix(info->LastWriteTime);

 if (data->adjust_tz) {
  fattr->cf_ctime.tv_sec += tcon->ses->server->timeAdj;
  fattr->cf_mtime.tv_sec += tcon->ses->server->timeAdj;
 }

 /*
 * The srv fs device id is overridden on network mount so setting
 * @fattr->cf_rdev isn't needed here.
 */

 fattr->cf_eof = le64_to_cpu(info->EndOfFile);
 fattr->cf_bytes = le64_to_cpu(info->AllocationSize);
 fattr->cf_createtime = le64_to_cpu(info->CreationTime);
 fattr->cf_nlink = le32_to_cpu(info->HardLinks);
 fattr->cf_mode = wire_mode_to_posix(le32_to_cpu(info->Mode),
         fattr->cf_cifsattrs & ATTR_DIRECTORY);

 if (cifs_open_data_reparse(data) &&
     cifs_reparse_point_to_fattr(cifs_sb, fattr, data))
  goto out_reparse;

 fattr->cf_dtype = S_DT(fattr->cf_mode);

out_reparse:
 if (S_ISLNK(fattr->cf_mode)) {
  if (likely(data->symlink_target))
   fattr->cf_eof = strnlen(data->symlink_target, PATH_MAX);
  fattr->cf_symlink_target = data->symlink_target;
  data->symlink_target = NULL;
 }
 sid_to_id(cifs_sb, &data->posix_owner, fattr, SIDOWNER);
 sid_to_id(cifs_sb, &data->posix_group, fattr, SIDGROUP);

 cifs_dbg(FYI, "POSIX query info: mode 0x%x uniqueid 0x%llx nlink %d\n",
  fattr->cf_mode, fattr->cf_uniqueid, fattr->cf_nlink);
}

static void cifs_open_info_to_fattr(struct cifs_fattr *fattr,
        struct cifs_open_info_data *data,
        struct super_block *sb)
{
 struct smb2_file_all_info *info = &data->fi;
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 struct cifs_tcon *tcon = cifs_sb_master_tcon(cifs_sb);

 memset(fattr, 0, sizeof(*fattr));
 fattr->cf_cifsattrs = le32_to_cpu(info->Attributes);
 if (info->DeletePending)
  fattr->cf_flags |= CIFS_FATTR_DELETE_PENDING;

 if (info->LastAccessTime)
  fattr->cf_atime = cifs_NTtimeToUnix(info->LastAccessTime);
 else
  ktime_get_coarse_real_ts64(&fattr->cf_atime);

 fattr->cf_ctime = cifs_NTtimeToUnix(info->ChangeTime);
 fattr->cf_mtime = cifs_NTtimeToUnix(info->LastWriteTime);

 if (data->adjust_tz) {
  fattr->cf_ctime.tv_sec += tcon->ses->server->timeAdj;
  fattr->cf_mtime.tv_sec += tcon->ses->server->timeAdj;
 }

 fattr->cf_eof = le64_to_cpu(info->EndOfFile);
 fattr->cf_bytes = le64_to_cpu(info->AllocationSize);
 fattr->cf_createtime = le64_to_cpu(info->CreationTime);
 fattr->cf_nlink = le32_to_cpu(info->NumberOfLinks);
 fattr->cf_uid = cifs_sb->ctx->linux_uid;
 fattr->cf_gid = cifs_sb->ctx->linux_gid;

 fattr->cf_mode = cifs_sb->ctx->file_mode;
 if (cifs_open_data_reparse(data) &&
     cifs_reparse_point_to_fattr(cifs_sb, fattr, data))
  goto out_reparse;

 if (fattr->cf_cifsattrs & ATTR_DIRECTORY) {
  fattr->cf_mode = S_IFDIR | cifs_sb->ctx->dir_mode;
  fattr->cf_dtype = DT_DIR;
  /*
 * Server can return wrong NumberOfLinks value for directories
 * when Unix extensions are disabled - fake it.
 */

  if (!tcon->unix_ext)
   fattr->cf_flags |= CIFS_FATTR_UNKNOWN_NLINK;
 } else {
  fattr->cf_mode = S_IFREG | cifs_sb->ctx->file_mode;
  fattr->cf_dtype = DT_REG;

  /*
 * Don't accept zero nlink from non-unix servers unless
 * delete is pending.  Instead mark it as unknown.
 */

  if ((fattr->cf_nlink < 1) && !tcon->unix_ext &&
      !info->DeletePending) {
   cifs_dbg(VFS, "bogus file nlink value %u\n",
     fattr->cf_nlink);
   fattr->cf_flags |= CIFS_FATTR_UNKNOWN_NLINK;
  }
 }

 /* clear write bits if ATTR_READONLY is set */
 if (fattr->cf_cifsattrs & ATTR_READONLY)
  fattr->cf_mode &= ~(S_IWUGO);

out_reparse:
 if (S_ISLNK(fattr->cf_mode)) {
  if (likely(data->symlink_target))
   fattr->cf_eof = strnlen(data->symlink_target, PATH_MAX);
  fattr->cf_symlink_target = data->symlink_target;
  data->symlink_target = NULL;
 }
}

static int
cifs_get_file_info(struct file *filp)
{
 int rc;
 unsigned int xid;
 struct cifs_open_info_data data = {};
 struct cifs_fattr fattr;
 struct inode *inode = file_inode(filp);
 struct cifsFileInfo *cfile = filp->private_data;
 struct cifs_tcon *tcon = tlink_tcon(cfile->tlink);
 struct TCP_Server_Info *server = tcon->ses->server;
 struct dentry *dentry = filp->f_path.dentry;
 void *page = alloc_dentry_path();
 const unsigned char *path;

 if (!server->ops->query_file_info) {
  free_dentry_path(page);
  return -ENOSYS;
 }

 xid = get_xid();
 rc = server->ops->query_file_info(xid, tcon, cfile, &data);
 switch (rc) {
 case 0:
  /* TODO: add support to query reparse tag */
  data.adjust_tz = false;
  if (data.symlink_target) {
   data.reparse_point = true;
   data.reparse.tag = IO_REPARSE_TAG_SYMLINK;
  }
  path = build_path_from_dentry(dentry, page);
  if (IS_ERR(path)) {
   rc = PTR_ERR(path);
   goto cgfi_exit;
  }
  cifs_open_info_to_fattr(&fattr, &data, inode->i_sb);
  if (fattr.cf_flags & CIFS_FATTR_DELETE_PENDING)
   cifs_mark_open_handles_for_deleted_file(inode, path);
  break;
 case -EREMOTE:
  cifs_create_junction_fattr(&fattr, inode->i_sb);
  break;
 case -EOPNOTSUPP:
 case -EINVAL:
  /*
 * FIXME: legacy server -- fall back to path-based call?
 * for now, just skip revalidating and mark inode for
 * immediate reval.
 */

  rc = 0;
  CIFS_I(inode)->time = 0;
  goto cgfi_exit;
 default:
  goto cgfi_exit;
 }

 /*
 * don't bother with SFU junk here -- just mark inode as needing
 * revalidation.
 */

 fattr.cf_uniqueid = CIFS_I(inode)->uniqueid;
 fattr.cf_flags |= CIFS_FATTR_NEED_REVAL;
 /* if filetype is different, return error */
 rc = cifs_fattr_to_inode(inode, &fattr, false);
cgfi_exit:
 cifs_free_open_info(&data);
 free_dentry_path(page);
 free_xid(xid);
 return rc;
}

/* Simple function to return a 64 bit hash of string.  Rarely called */
static __u64 simple_hashstr(const char *str)
{
 const __u64 hash_mult =  1125899906842597ULL; /* a big enough prime */
 __u64 hash = 0;

 while (*str)
  hash = (hash + (__u64) *str++) * hash_mult;

 return hash;
}

#ifdef CONFIG_CIFS_ALLOW_INSECURE_LEGACY
/**
 * cifs_backup_query_path_info - SMB1 fallback code to get ino
 *
 * Fallback code to get file metadata when we don't have access to
 * full_path (EACCES) and have backup creds.
 *
 * @xid: transaction id used to identify original request in logs
 * @tcon: information about the server share we have mounted
 * @sb: the superblock stores info such as disk space available
 * @full_path: name of the file we are getting the metadata for
 * @resp_buf: will be set to cifs resp buf and needs to be freed with
 *  cifs_buf_release() when done with @data
 * @data: will be set to search info result buffer
 */

static int
cifs_backup_query_path_info(int xid,
       struct cifs_tcon *tcon,
       struct super_block *sb,
       const char *full_path,
       void **resp_buf,
       FILE_ALL_INFO **data)
{
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 struct cifs_search_info info = {0};
 u16 flags;
 int rc;

 *resp_buf = NULL;
 info.endOfSearch = false;
 if (tcon->unix_ext)
  info.info_level = SMB_FIND_FILE_UNIX;
 else if ((tcon->ses->capabilities &
    tcon->ses->server->vals->cap_nt_find) == 0)
  info.info_level = SMB_FIND_FILE_INFO_STANDARD;
 else if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM)
  info.info_level = SMB_FIND_FILE_ID_FULL_DIR_INFO;
 else /* no srvino useful for fallback to some netapp */
  info.info_level = SMB_FIND_FILE_DIRECTORY_INFO;

 flags = CIFS_SEARCH_CLOSE_ALWAYS |
  CIFS_SEARCH_CLOSE_AT_END |
  CIFS_SEARCH_BACKUP_SEARCH;

 rc = CIFSFindFirst(xid, tcon, full_path,
      cifs_sb, NULL, flags, &info, false);
 if (rc)
  return rc;

 *resp_buf = (void *)info.ntwrk_buf_start;
 *data = (FILE_ALL_INFO *)info.srch_entries_start;
 return 0;
}
#endif /* CONFIG_CIFS_ALLOW_INSECURE_LEGACY */

static void cifs_set_fattr_ino(int xid, struct cifs_tcon *tcon, struct super_block *sb,
          struct inode **inode, const char *full_path,
          struct cifs_open_info_data *data, struct cifs_fattr *fattr)
{
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 struct TCP_Server_Info *server = tcon->ses->server;
 int rc;

 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM)) {
  if (*inode)
   fattr->cf_uniqueid = CIFS_I(*inode)->uniqueid;
  else
   fattr->cf_uniqueid = iunique(sb, ROOT_I);
  return;
 }

 /*
 * If we have an inode pass a NULL tcon to ensure we don't
 * make a round trip to the server. This only works for SMB2+.
 */

 rc = server->ops->get_srv_inum(xid, *inode ? NULL : tcon, cifs_sb, full_path,
           &fattr->cf_uniqueid, data);
 if (rc) {
  /*
 * If that fails reuse existing ino or generate one
 * and disable server ones
 */

  if (*inode)
   fattr->cf_uniqueid = CIFS_I(*inode)->uniqueid;
  else {
   fattr->cf_uniqueid = iunique(sb, ROOT_I);
   cifs_autodisable_serverino(cifs_sb);
  }
  return;
 }

 /* If no errors, check for zero root inode (invalid) */
 if (fattr->cf_uniqueid == 0 && strlen(full_path) == 0) {
  cifs_dbg(FYI, "Invalid (0) inodenum\n");
  if (*inode) {
   /* reuse */
   fattr->cf_uniqueid = CIFS_I(*inode)->uniqueid;
  } else {
   /* make an ino by hashing the UNC */
   fattr->cf_flags |= CIFS_FATTR_FAKE_ROOT_INO;
   fattr->cf_uniqueid = simple_hashstr(tcon->tree_name);
  }
 }
}

static inline bool is_inode_cache_good(struct inode *ino)
{
 return ino && CIFS_CACHE_READ(CIFS_I(ino)) && CIFS_I(ino)->time != 0;
}

static int reparse_info_to_fattr(struct cifs_open_info_data *data,
     struct super_block *sb,
     const unsigned int xid,
     struct cifs_tcon *tcon,
     const char *full_path,
     struct cifs_fattr *fattr)
{
 struct TCP_Server_Info *server = tcon->ses->server;
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 struct kvec rsp_iov, *iov = NULL;
 int rsp_buftype = CIFS_NO_BUFFER;
 u32 tag = data->reparse.tag;
 int rc = 0;

 if (!tag && server->ops->query_reparse_point) {
  rc = server->ops->query_reparse_point(xid, tcon, cifs_sb,
            full_path, &tag,
            &rsp_iov, &rsp_buftype);
  if (!rc)
   iov = &rsp_iov;
 } else if (data->reparse.io.buftype != CIFS_NO_BUFFER &&
     data->reparse.io.iov.iov_base) {
  iov = &data->reparse.io.iov;
 }

 rc = -EOPNOTSUPP;
 data->reparse.tag = tag;
 if (!data->reparse.tag) {
  if (server->ops->query_symlink) {
   rc = server->ops->query_symlink(xid, tcon,
       cifs_sb, full_path,
       &data->symlink_target);
  }
  if (rc == -EOPNOTSUPP)
   data->reparse.tag = IO_REPARSE_TAG_INTERNAL;
 }

 switch (data->reparse.tag) {
 case 0: /* SMB1 symlink */
  break;
 case IO_REPARSE_TAG_INTERNAL:
  rc = 0;
  if (le32_to_cpu(data->fi.Attributes) & ATTR_DIRECTORY) {
   cifs_create_junction_fattr(fattr, sb);
   goto out;
  }
  break;
 default:
  /* Check for cached reparse point data */
  if (data->symlink_target || data->reparse.buf) {
   rc = 0;
  } else if (iov && server->ops->get_reparse_point_buffer) {
   struct reparse_data_buffer *reparse_buf;
   u32 reparse_len;

   reparse_buf = server->ops->get_reparse_point_buffer(iov, &reparse_len);
   rc = parse_reparse_point(reparse_buf, reparse_len,
       cifs_sb, full_path, data);
   /*
 * If the reparse point was not handled but it is the
 * name surrogate which points to directory, then treat
 * is as a new mount point. Name surrogate reparse point
 * represents another named entity in the system.
 */

   if (rc == -EOPNOTSUPP &&
       IS_REPARSE_TAG_NAME_SURROGATE(data->reparse.tag) &&
       (le32_to_cpu(data->fi.Attributes) & ATTR_DIRECTORY)) {
    rc = 0;
    cifs_create_junction_fattr(fattr, sb);
    goto out;
   }
   /*
 * If the reparse point is unsupported by the Linux SMB
 * client then let it process by the SMB server. So mask
 * the -EOPNOTSUPP error code. This will allow Linux SMB
 * client to send SMB OPEN request to server. If server
 * does not support this reparse point too then server
 * will return error during open the path.
 */

   if (rc == -EOPNOTSUPP)
    rc = 0;
  }

  if (data->reparse.tag == IO_REPARSE_TAG_SYMLINK && !rc) {
   bool directory = le32_to_cpu(data->fi.Attributes) & ATTR_DIRECTORY;
   rc = smb2_fix_symlink_target_type(&data->symlink_target, directory, cifs_sb);
  }
  break;
 }

 if (tcon->posix_extensions)
  smb311_posix_info_to_fattr(fattr, data, sb);
 else
  cifs_open_info_to_fattr(fattr, data, sb);
out:
 fattr->cf_cifstag = data->reparse.tag;
 free_rsp_buf(rsp_buftype, rsp_iov.iov_base);
 return rc;
}

static int cifs_get_fattr(struct cifs_open_info_data *data,
     struct super_block *sb, int xid,
     const struct cifs_fid *fid,
     struct cifs_fattr *fattr,
     struct inode **inode,
     const char *full_path)
{
 struct cifs_open_info_data tmp_data = {};
 struct cifs_tcon *tcon;
 struct TCP_Server_Info *server;
 struct tcon_link *tlink;
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 void *smb1_backup_rsp_buf = NULL;
 int rc = 0;
 int tmprc = 0;

 tlink = cifs_sb_tlink(cifs_sb);
 if (IS_ERR(tlink))
  return PTR_ERR(tlink);
 tcon = tlink_tcon(tlink);
 server = tcon->ses->server;

 /*
 * 1. Fetch file metadata if not provided (data)
 */


 if (!data) {
  rc = server->ops->query_path_info(xid, tcon, cifs_sb,
        full_path, &tmp_data);
  data = &tmp_data;
 }

 /*
 * 2. Convert it to internal cifs metadata (fattr)
 */


 switch (rc) {
 case 0:
  /*
 * If the file is a reparse point, it is more complicated
 * since we have to check if its reparse tag matches a known
 * special file type e.g. symlink or fifo or char etc.
 */

  if (cifs_open_data_reparse(data)) {
   rc = reparse_info_to_fattr(data, sb, xid, tcon,
         full_path, fattr);
  } else {
   cifs_open_info_to_fattr(fattr, data, sb);
  }
  if (!rc && *inode &&
      (fattr->cf_flags & CIFS_FATTR_DELETE_PENDING))
   cifs_mark_open_handles_for_deleted_file(*inode, full_path);
  break;
 case -EREMOTE:
  /* DFS link, no metadata available on this server */
  cifs_create_junction_fattr(fattr, sb);
  rc = 0;
  break;
 case -EACCES:
#ifdef CONFIG_CIFS_ALLOW_INSECURE_LEGACY
  /*
 * perm errors, try again with backup flags if possible
 *
 * For SMB2 and later the backup intent flag
 * is already sent if needed on open and there
 * is no path based FindFirst operation to use
 * to retry with
 */

  if (backup_cred(cifs_sb) && is_smb1_server(server)) {
   /* for easier reading */
   FILE_ALL_INFO *fi;
   FILE_DIRECTORY_INFO *fdi;
   SEARCH_ID_FULL_DIR_INFO *si;

   rc = cifs_backup_query_path_info(xid, tcon, sb,
        full_path,
        &smb1_backup_rsp_buf,
        &fi);
   if (rc)
    goto out;

   move_cifs_info_to_smb2(&data->fi, fi);
   fdi = (FILE_DIRECTORY_INFO *)fi;
   si = (SEARCH_ID_FULL_DIR_INFO *)fi;

   cifs_dir_info_to_fattr(fattr, fdi, cifs_sb);
   fattr->cf_uniqueid = le64_to_cpu(si->UniqueId);
   /* uniqueid set, skip get inum step */
   goto handle_mnt_opt;
  } else {
   /* nothing we can do, bail out */
   goto out;
  }
#else
  goto out;
#endif /* CONFIG_CIFS_ALLOW_INSECURE_LEGACY */
  break;
 default:
  cifs_dbg(FYI, "%s: unhandled err rc %d\n", __func__, rc);
  goto out;
 }

 /*
 * 3. Get or update inode number (fattr->cf_uniqueid)
 */


 cifs_set_fattr_ino(xid, tcon, sb, inode, full_path, data, fattr);

 /*
 * 4. Tweak fattr based on mount options
 */

#ifdef CONFIG_CIFS_ALLOW_INSECURE_LEGACY
handle_mnt_opt:
#endif /* CONFIG_CIFS_ALLOW_INSECURE_LEGACY */
 /* query for SFU type info if supported and needed */
 if ((fattr->cf_cifsattrs & ATTR_SYSTEM) &&
     (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL)) {
  tmprc = cifs_sfu_type(fattr, full_path, cifs_sb, xid);
  if (tmprc)
   cifs_dbg(FYI, "cifs_sfu_type failed: %d\n", tmprc);
 }

 /* fill in 0777 bits from ACL */
 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MODE_FROM_SID) {
  rc = cifs_acl_to_fattr(cifs_sb, fattr, *inode,
           true, full_path, fid);
  if (rc == -EREMOTE)
   rc = 0;
  if (rc) {
   cifs_dbg(FYI, "%s: Get mode from SID failed. rc=%d\n",
     __func__, rc);
   goto out;
  }
 } else if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) {
  rc = cifs_acl_to_fattr(cifs_sb, fattr, *inode,
           false, full_path, fid);
  if (rc == -EREMOTE)
   rc = 0;
  if (rc) {
   cifs_dbg(FYI, "%s: Getting ACL failed with error: %d\n",
     __func__, rc);
   goto out;
  }
 } else if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL)
  /* fill in remaining high mode bits e.g. SUID, VTX */
  cifs_sfu_mode(fattr, full_path, cifs_sb, xid);
 else if (!(tcon->posix_extensions))
  /* clear write bits if ATTR_READONLY is set */
  if (fattr->cf_cifsattrs & ATTR_READONLY)
   fattr->cf_mode &= ~(S_IWUGO);


 /* check for Minshall+French symlinks */
 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MF_SYMLINKS) {
  tmprc = check_mf_symlink(xid, tcon, cifs_sb, fattr, full_path);
  cifs_dbg(FYI, "check_mf_symlink: %d\n", tmprc);
 }

out:
 cifs_buf_release(smb1_backup_rsp_buf);
 cifs_put_tlink(tlink);
 cifs_free_open_info(&tmp_data);
 return rc;
}

int cifs_get_inode_info(struct inode **inode,
   const char *full_path,
   struct cifs_open_info_data *data,
   struct super_block *sb, int xid,
   const struct cifs_fid *fid)
{
 struct cifs_fattr fattr = {};
 int rc;

 if (!data && is_inode_cache_good(*inode)) {
  cifs_dbg(FYI, "No need to revalidate cached inode sizes\n");
  return 0;
 }

 rc = cifs_get_fattr(data, sb, xid, fid, &fattr, inode, full_path);
 if (rc)
  goto out;

 rc = update_inode_info(sb, &fattr, inode);
out:
 kfree(fattr.cf_symlink_target);
 return rc;
}

static int smb311_posix_get_fattr(struct cifs_open_info_data *data,
      struct cifs_fattr *fattr,
      const char *full_path,
      struct super_block *sb,
      const unsigned int xid)
{
 struct cifs_open_info_data tmp_data = {};
 struct TCP_Server_Info *server;
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 struct cifs_tcon *tcon;
 struct tcon_link *tlink;
 int tmprc;
 int rc = 0;

 tlink = cifs_sb_tlink(cifs_sb);
 if (IS_ERR(tlink))
  return PTR_ERR(tlink);
 tcon = tlink_tcon(tlink);
 server = tcon->ses->server;

 /*
 * 1. Fetch file metadata if not provided (data)
 */

 if (!data) {
  rc = server->ops->query_path_info(xid, tcon, cifs_sb,
        full_path, &tmp_data);
  data = &tmp_data;
 }

 /*
 * 2. Convert it to internal cifs metadata (fattr)
 */


 switch (rc) {
 case 0:
  if (cifs_open_data_reparse(data)) {
   rc = reparse_info_to_fattr(data, sb, xid, tcon,
         full_path, fattr);
  } else {
   smb311_posix_info_to_fattr(fattr, data, sb);
  }
  break;
 case -EREMOTE:
  /* DFS link, no metadata available on this server */
  cifs_create_junction_fattr(fattr, sb);
  rc = 0;
  break;
 case -EACCES:
  /*
 * For SMB2 and later the backup intent flag
 * is already sent if needed on open and there
 * is no path based FindFirst operation to use
 * to retry with so nothing we can do, bail out
 */

  goto out;
 default:
  cifs_dbg(FYI, "%s: unhandled err rc %d\n", __func__, rc);
  goto out;
 }

 /*
 * 3. Tweak fattr based on mount options
 */

 /* check for Minshall+French symlinks */
 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MF_SYMLINKS) {
  tmprc = check_mf_symlink(xid, tcon, cifs_sb, fattr, full_path);
  cifs_dbg(FYI, "check_mf_symlink: %d\n", tmprc);
 }

out:
 cifs_put_tlink(tlink);
 cifs_free_open_info(data);
 return rc;
}

int smb311_posix_get_inode_info(struct inode **inode,
    const char *full_path,
    struct cifs_open_info_data *data,
    struct super_block *sb,
    const unsigned int xid)
{
 struct cifs_fattr fattr = {};
 int rc;

 if (!data && is_inode_cache_good(*inode)) {
  cifs_dbg(FYI, "No need to revalidate cached inode sizes\n");
  return 0;
 }

 rc = smb311_posix_get_fattr(data, &fattr, full_path, sb, xid);
 if (rc)
  goto out;

 rc = update_inode_info(sb, &fattr, inode);
 if (!rc && fattr.cf_flags & CIFS_FATTR_DELETE_PENDING)
  cifs_mark_open_handles_for_deleted_file(*inode, full_path);
out:
 kfree(fattr.cf_symlink_target);
 return rc;
}

static const struct inode_operations cifs_ipc_inode_ops = {
 .lookup = cifs_lookup,
};

static int
cifs_find_inode(struct inode *inode, void *opaque)
{
 struct cifs_fattr *fattr = opaque;

 /* [!] The compared values must be the same in struct cifs_fscache_inode_key. */

 /* don't match inode with different uniqueid */
 if (CIFS_I(inode)->uniqueid != fattr->cf_uniqueid)
  return 0;

 /* use createtime like an i_generation field */
 if (CIFS_I(inode)->createtime != fattr->cf_createtime)
  return 0;

 /* don't match inode of different type */
 if (inode_wrong_type(inode, fattr->cf_mode))
  return 0;

 /* if it's not a directory or has no dentries, then flag it */
 if (S_ISDIR(inode->i_mode) && !hlist_empty(&inode->i_dentry))
  fattr->cf_flags |= CIFS_FATTR_INO_COLLISION;

 return 1;
}

static int
cifs_init_inode(struct inode *inode, void *opaque)
{
 struct cifs_fattr *fattr = opaque;

 CIFS_I(inode)->uniqueid = fattr->cf_uniqueid;
 CIFS_I(inode)->createtime = fattr->cf_createtime;
 return 0;
}

/*
 * walk dentry list for an inode and report whether it has aliases that
 * are hashed. We use this to determine if a directory inode can actually
 * be used.
 */

static bool
inode_has_hashed_dentries(struct inode *inode)
{
 struct dentry *dentry;

 spin_lock(&inode->i_lock);
 hlist_for_each_entry(dentry, &inode->i_dentry, d_u.d_alias) {
  if (!d_unhashed(dentry) || IS_ROOT(dentry)) {
   spin_unlock(&inode->i_lock);
   return true;
  }
 }
 spin_unlock(&inode->i_lock);
 return false;
}

/* Given fattrs, get a corresponding inode */
struct inode *
cifs_iget(struct super_block *sb, struct cifs_fattr *fattr)
{
 unsigned long hash;
 struct inode *inode;

retry_iget5_locked:
 cifs_dbg(FYI, "looking for uniqueid=%llu\n", fattr->cf_uniqueid);

 /* hash down to 32-bits on 32-bit arch */
 hash = cifs_uniqueid_to_ino_t(fattr->cf_uniqueid);

 inode = iget5_locked(sb, hash, cifs_find_inode, cifs_init_inode, fattr);
 if (inode) {
  /* was there a potentially problematic inode collision? */
  if (fattr->cf_flags & CIFS_FATTR_INO_COLLISION) {
   fattr->cf_flags &= ~CIFS_FATTR_INO_COLLISION;

   if (inode_has_hashed_dentries(inode)) {
    cifs_autodisable_serverino(CIFS_SB(sb));
    iput(inode);
    fattr->cf_uniqueid = iunique(sb, ROOT_I);
    goto retry_iget5_locked;
   }
  }

  /* can't fail - see cifs_find_inode() */
  cifs_fattr_to_inode(inode, fattr, false);
  if (sb->s_flags & SB_NOATIME)
   inode->i_flags |= S_NOATIME | S_NOCMTIME;
  if (inode->i_state & I_NEW) {
   inode->i_ino = hash;
   cifs_fscache_get_inode_cookie(inode);
   unlock_new_inode(inode);
  }
 }

 return inode;
}

/* gets root inode */
struct inode *cifs_root_iget(struct super_block *sb)
{
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 struct cifs_fattr fattr = {};
 struct cifs_tcon *tcon = cifs_sb_master_tcon(cifs_sb);
 struct inode *inode = NULL;
 unsigned int xid;
 char *path = NULL;
 int len;
 int rc;

 if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_USE_PREFIX_PATH)
     && cifs_sb->prepath) {
  len = strlen(cifs_sb->prepath);
  path = kzalloc(len + 2 /* leading sep + null */, GFP_KERNEL);
  if (path == NULL)
   return ERR_PTR(-ENOMEM);
  path[0] = '/';
  memcpy(path+1, cifs_sb->prepath, len);
 } else {
  path = kstrdup("", GFP_KERNEL);
  if (path == NULL)
   return ERR_PTR(-ENOMEM);
 }

 xid = get_xid();
 if (tcon->unix_ext) {
  rc = cifs_get_unix_fattr(path, sb, &fattr, &inode, xid);
  /* some servers mistakenly claim POSIX support */
  if (rc != -EOPNOTSUPP)
   goto iget_root;
  cifs_dbg(VFS, "server does not support POSIX extensions\n");
  tcon->unix_ext = false;
 }

 convert_delimiter(path, CIFS_DIR_SEP(cifs_sb));
 if (tcon->posix_extensions)
  rc = smb311_posix_get_fattr(NULL, &fattr, path, sb, xid);
 else
  rc = cifs_get_fattr(NULL, sb, xid, NULL, &fattr, &inode, path);

iget_root:
 if (!rc) {
  if (fattr.cf_flags & CIFS_FATTR_JUNCTION) {
   fattr.cf_flags &= ~CIFS_FATTR_JUNCTION;
   cifs_autodisable_serverino(cifs_sb);
  }
  inode = cifs_iget(sb, &fattr);
 }

 if (!inode) {
  inode = ERR_PTR(rc);
  goto out;
 }

 if (!rc && fattr.cf_flags & CIFS_FATTR_DELETE_PENDING)
  cifs_mark_open_handles_for_deleted_file(inode, path);

 if (rc && tcon->pipe) {
  cifs_dbg(FYI, "ipc connection - fake read inode\n");
  spin_lock(&inode->i_lock);
  inode->i_mode |= S_IFDIR;
  set_nlink(inode, 2);
  inode->i_op = &cifs_ipc_inode_ops;
  inode->i_fop = &simple_dir_operations;
  inode->i_uid = cifs_sb->ctx->linux_uid;
  inode->i_gid = cifs_sb->ctx->linux_gid;
  spin_unlock(&inode->i_lock);
 } else if (rc) {
  iget_failed(inode);
  inode = ERR_PTR(rc);
 }

out:
 kfree(path);
 free_xid(xid);
 kfree(fattr.cf_symlink_target);
 return inode;
}

int
cifs_set_file_info(struct inode *inode, struct iattr *attrs, unsigned int xid,
     const char *full_path, __u32 dosattr)
{
 bool set_time = false;
 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
 struct TCP_Server_Info *server;
 FILE_BASIC_INFO info_buf;

 if (attrs == NULL)
  return -EINVAL;

 server = cifs_sb_master_tcon(cifs_sb)->ses->server;
 if (!server->ops->set_file_info)
  return -ENOSYS;

 info_buf.Pad = 0;

 if (attrs->ia_valid & ATTR_ATIME) {
  set_time = true;
  info_buf.LastAccessTime =
   cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_atime));
 } else
  info_buf.LastAccessTime = 0;

 if (attrs->ia_valid & ATTR_MTIME) {
  set_time = true;
  info_buf.LastWriteTime =
      cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_mtime));
 } else
  info_buf.LastWriteTime = 0;

 /*
 * Samba throws this field away, but windows may actually use it.
 * Do not set ctime unless other time stamps are changed explicitly
 * (i.e. by utimes()) since we would then have a mix of client and
 * server times.
 */

 if (set_time && (attrs->ia_valid & ATTR_CTIME)) {
  cifs_dbg(FYI, "CIFS - CTIME changed\n");
  info_buf.ChangeTime =
      cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_ctime));
 } else
  info_buf.ChangeTime = 0;

 info_buf.CreationTime = 0; /* don't change */
 info_buf.Attributes = cpu_to_le32(dosattr);

 return server->ops->set_file_info(inode, full_path, &info_buf, xid);
}

#ifdef CONFIG_CIFS_ALLOW_INSECURE_LEGACY
/*
 * Open the given file (if it isn't already), set the DELETE_ON_CLOSE bit
 * and rename it to a random name that hopefully won't conflict with
 * anything else.
 */

int
cifs_rename_pending_delete(const char *full_path, struct dentry *dentry,
      const unsigned int xid)
{
 int oplock = 0;
 int rc;
 struct cifs_fid fid;
 struct cifs_open_parms oparms;
 struct inode *inode = d_inode(dentry);
 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
 struct tcon_link *tlink;
 struct cifs_tcon *tcon;
 __u32 dosattr, origattr;
 FILE_BASIC_INFO *info_buf = NULL;

 tlink = cifs_sb_tlink(cifs_sb);
 if (IS_ERR(tlink))
  return PTR_ERR(tlink);
 tcon = tlink_tcon(tlink);

 /*
 * We cannot rename the file if the server doesn't support
 * CAP_INFOLEVEL_PASSTHRU
 */

 if (!(tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)) {
  rc = -EBUSY;
  goto out;
 }

 oparms = (struct cifs_open_parms) {
  .tcon = tcon,
  .cifs_sb = cifs_sb,
  .desired_access = DELETE | FILE_WRITE_ATTRIBUTES,
  .create_options = cifs_create_options(cifs_sb, CREATE_NOT_DIR),
  .disposition = FILE_OPEN,
  .path = full_path,
  .fid = &fid,
 };

 rc = CIFS_open(xid, &oparms, &oplock, NULL);
 if (rc != 0)
  goto out;

 origattr = cifsInode->cifsAttrs;
 if (origattr == 0)
  origattr |= ATTR_NORMAL;

 dosattr = origattr & ~ATTR_READONLY;
 if (dosattr == 0)
  dosattr |= ATTR_NORMAL;
 dosattr |= ATTR_HIDDEN;

 /* set ATTR_HIDDEN and clear ATTR_READONLY, but only if needed */
 if (dosattr != origattr) {
  info_buf = kzalloc(sizeof(*info_buf), GFP_KERNEL);
  if (info_buf == NULL) {
   rc = -ENOMEM;
   goto out_close;
  }
  info_buf->Attributes = cpu_to_le32(dosattr);
  rc = CIFSSMBSetFileInfo(xid, tcon, info_buf, fid.netfid,
     current->tgid);
  /* although we would like to mark the file hidden
     if that fails we will still try to rename it */

  if (!rc)
   cifsInode->cifsAttrs = dosattr;
  else
   dosattr = origattr; /* since not able to change them */
 }

 /* rename the file */
 rc = CIFSSMBRenameOpenFile(xid, tcon, fid.netfid, NULL,
       cifs_sb->local_nls,
       cifs_remap(cifs_sb));
 if (rc != 0) {
  rc = -EBUSY;
  goto undo_setattr;
 }

 /* try to set DELETE_ON_CLOSE */
 if (!test_bit(CIFS_INO_DELETE_PENDING, &cifsInode->flags)) {
  rc = CIFSSMBSetFileDisposition(xid, tcon, true, fid.netfid,
            current->tgid);
  /*
 * some samba versions return -ENOENT when we try to set the
 * file disposition here. Likely a samba bug, but work around
 * it for now. This means that some cifsXXX files may hang
 * around after they shouldn't.
 *
 * BB: remove this hack after more servers have the fix
 */

  if (rc == -ENOENT)
   rc = 0;
  else if (rc != 0) {
   rc = -EBUSY;
   goto undo_rename;
  }
  set_bit(CIFS_INO_DELETE_PENDING, &cifsInode->flags);
 }

out_close:
 CIFSSMBClose(xid, tcon, fid.netfid);
out:
 kfree(info_buf);
 cifs_put_tlink(tlink);
 return rc;

 /*
 * reset everything back to the original state. Don't bother
 * dealing with errors here since we can't do anything about
 * them anyway.
 */

undo_rename:
 CIFSSMBRenameOpenFile(xid, tcon, fid.netfid, dentry->d_name.name,
    cifs_sb->local_nls, cifs_remap(cifs_sb));
undo_setattr:
 if (dosattr != origattr) {
  info_buf->Attributes = cpu_to_le32(origattr);
  if (!CIFSSMBSetFileInfo(xid, tcon, info_buf, fid.netfid,
     current->tgid))
   cifsInode->cifsAttrs = origattr;
 }

 goto out_close;
}
#endif /* CONFIG_CIFS_ALLOW_INSECURE_LEGACY */

/* copied from fs/nfs/dir.c with small changes */
static void
cifs_drop_nlink(struct inode *inode)
{
 spin_lock(&inode->i_lock);
 if (inode->i_nlink > 0)
  drop_nlink(inode);
 spin_unlock(&inode->i_lock);
}

/*
 * If d_inode(dentry) is null (usually meaning the cached dentry
 * is a negative dentry) then we would attempt a standard SMB delete, but
 * if that fails we can not attempt the fall back mechanisms on EACCES
 * but will return the EACCES to the caller. Note that the VFS does not call
 * unlink on negative dentries currently.
 */

static int __cifs_unlink(struct inode *dir, struct dentry *dentry, bool sillyrename)
{
 int rc = 0;
 unsigned int xid;
 const char *full_path;
 void *page;
 struct inode *inode = d_inode(dentry);
 struct cifsInodeInfo *cifs_inode;
 struct super_block *sb = dir->i_sb;
 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
 struct tcon_link *tlink;
 struct cifs_tcon *tcon;
 __u32 dosattr = 0, origattr = 0;
 struct TCP_Server_Info *server;
 struct iattr *attrs = NULL;
 bool rehash = false;

 cifs_dbg(FYI, "cifs_unlink, dir=0x%p, dentry=0x%p\n", dir, dentry);

 if (unlikely(cifs_forced_shutdown(cifs_sb)))
  return -EIO;

 /* Unhash dentry in advance to prevent any concurrent opens */
 spin_lock(&dentry->d_lock);
 if (!d_unhashed(dentry)) {
  __d_drop(dentry);
  rehash = true;
 }
 spin_unlock(&dentry->d_lock);

 tlink = cifs_sb_tlink(cifs_sb);
 if (IS_ERR(tlink))
  return PTR_ERR(tlink);
 tcon = tlink_tcon(tlink);
 server = tcon->ses->server;

 xid = get_xid();
 page = alloc_dentry_path();

 if (tcon->nodelete) {
  rc = -EACCES;
  goto unlink_out;
 }

 /* Unlink can be called from rename so we can not take the
 * sb->s_vfs_rename_mutex here */

 full_path = build_path_from_dentry(dentry, page);
 if (IS_ERR(full_path)) {
  rc = PTR_ERR(full_path);
  goto unlink_out;
 }

 netfs_wait_for_outstanding_io(inode);
 cifs_close_deferred_file_under_dentry(tcon, dentry);
#ifdef CONFIG_CIFS_ALLOW_INSECURE_LEGACY
 if (cap_unix(tcon->ses) && (CIFS_UNIX_POSIX_PATH_OPS_CAP &
    le64_to_cpu(tcon->fsUnixInfo.Capability))) {
  rc = CIFSPOSIXDelFile(xid, tcon, full_path,
   SMB_POSIX_UNLINK_FILE_TARGET, cifs_sb->local_nls,
   cifs_remap(cifs_sb));
  cifs_dbg(FYI, "posix del rc %d\n", rc);
  if ((rc == 0) || (rc == -ENOENT))
   goto psx_del_no_retry;
 }
#endif /* CONFIG_CIFS_ALLOW_INSECURE_LEGACY */

retry_std_delete:
 if (!server->ops->unlink) {
  rc = -ENOSYS;
  goto psx_del_no_retry;
 }

 /* For SMB2+, if the file is open, we always perform a silly rename.
 *
 * We check for d_count() right after calling
 * cifs_close_deferred_file_under_dentry() to make sure that the
 * dentry's refcount gets dropped in case the file had any deferred
 * close.
 */

 if (!sillyrename && server->vals->protocol_id > SMB10_PROT_ID) {
  spin_lock(&dentry->d_lock);
  if (d_count(dentry) > 1)
   sillyrename = true;
  spin_unlock(&dentry->d_lock);
 }

 if (sillyrename)
  rc = -EBUSY;
 else
  rc = server->ops->unlink(xid, tcon, full_path, cifs_sb, dentry);

psx_del_no_retry:
 if (!rc) {
  if (inode) {
   cifs_mark_open_handles_for_deleted_file(inode, full_path);
   cifs_drop_nlink(inode);
  }
 } else if (rc == -ENOENT) {
  if (simple_positive(dentry))
   d_delete(dentry);
 } else if (rc == -EBUSY) {
  if (server->ops->rename_pending_delete) {
   rc = server->ops->rename_pending_delete(full_path,
        dentry, xid);
   if (rc == 0) {
    cifs_mark_open_handles_for_deleted_file(inode, full_path);
    cifs_drop_nlink(inode);
   }
  }
 } else if ((rc == -EACCES) && (dosattr == 0) && inode) {
  attrs = kzalloc(sizeof(*attrs), GFP_KERNEL);
  if (attrs == NULL) {
   rc = -ENOMEM;
   goto out_reval;
  }

  /* try to reset dos attributes */
  cifs_inode = CIFS_I(inode);
  origattr = cifs_inode->cifsAttrs;
  if (origattr == 0)
   origattr |= ATTR_NORMAL;
  dosattr = origattr & ~ATTR_READONLY;
  if (dosattr == 0)
   dosattr |= ATTR_NORMAL;
  dosattr |= ATTR_HIDDEN;

  rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
  if (rc != 0)
   goto out_reval;

  goto retry_std_delete;
 }

 /* undo the setattr if we errored out and it's needed */
 if (rc != 0 && dosattr != 0)
  cifs_set_file_info(inode, attrs, xid, full_path, origattr);

out_reval:
 if (inode) {
  cifs_inode = CIFS_I(inode);
  cifs_inode->time = 0; /* will force revalidate to get info
   when needed */

  inode_set_ctime_current(inode);
 }
 inode_set_mtime_to_ts(dir, inode_set_ctime_current(dir));
 cifs_inode = CIFS_I(dir);
 CIFS_I(dir)->time = 0; /* force revalidate of dir as well */
unlink_out:
 free_dentry_path(page);
 kfree(attrs);
 free_xid(xid);
 cifs_put_tlink(tlink);
 if (rehash)
  d_rehash(dentry);
 return rc;
}

int cifs_unlink(struct inode *dir, struct dentry *dentry)
{
 return __cifs_unlink(dir, dentry, false);
}

static int
cifs_mkdir_qinfo(struct inode *parent, struct dentry *dentry, umode_t mode,
   const char *full_path, struct cifs_sb_info *cifs_sb,
   struct cifs_tcon *tcon, const unsigned int xid)
{
 int rc = 0;
 struct inode *inode = NULL;

 if (tcon->posix_extensions) {
  rc = smb311_posix_get_inode_info(&inode, full_path,
       NULL, parent->i_sb, xid);
#ifdef CONFIG_CIFS_ALLOW_INSECURE_LEGACY
 } else if (tcon->unix_ext) {
  rc = cifs_get_inode_info_unix(&inode, full_path, parent->i_sb,
           xid);
#endif /* CONFIG_CIFS_ALLOW_INSECURE_LEGACY */
 } else {
  rc = cifs_get_inode_info(&inode, full_path, NULL, parent->i_sb,
      xid, NULL);
 }

 if (rc)
  return rc;

 if (!S_ISDIR(inode->i_mode)) {
  /*
 * mkdir succeeded, but another client has managed to remove the
 * sucker and replace it with non-directory.  Return success,
 * but don't leave the child in dcache.
 */

   iput(inode);
   d_drop(dentry);
   return 0;
 }
 /*
 * setting nlink not necessary except in cases where we failed to get it
 * from the server or was set bogus. Also, since this is a brand new
 * inode, no need to grab the i_lock before setting the i_nlink.
 */

 if (inode->i_nlink < 2)
  set_nlink(inode, 2);
 mode &= ~current_umask();
 /* must turn on setgid bit if parent dir has it */
 if (parent->i_mode & S_ISGID)
  mode |= S_ISGID;

#ifdef CONFIG_CIFS_ALLOW_INSECURE_LEGACY
 if (tcon->unix_ext) {
  struct cifs_unix_set_info_args args = {
   .mode = mode,
   .ctime = NO_CHANGE_64,
   .atime = NO_CHANGE_64,
   .mtime = NO_CHANGE_64,
   .device = 0,
  };
  if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID) {
   args.uid = current_fsuid();
   if (parent->i_mode & S_ISGID)
    args.gid = parent->i_gid;
   else
    args.gid = current_fsgid();
  } else {
   args.uid = INVALID_UID; /* no change */
   args.gid = INVALID_GID; /* no change */
  }
  CIFSSMBUnixSetPathInfo(xid, tcon, full_path, &args,
           cifs_sb->local_nls,
           cifs_remap(cifs_sb));
 } else {
#else
 {
#endif /* CONFIG_CIFS_ALLOW_INSECURE_LEGACY */
  struct TCP_Server_Info *server = tcon->ses->server;
  if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) &&
      (mode & S_IWUGO) == 0 && server->ops->mkdir_setinfo)
   server->ops->mkdir_setinfo(inode, full_path, cifs_sb,
         tcon, xid);
  if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)
   inode->i_mode = (mode | S_IFDIR);

  if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID) {
   inode->i_uid = current_fsuid();
   if (inode->i_mode & S_ISGID)
    inode->i_gid = parent->i_gid;
   else
    inode->i_gid = current_fsgid();
  }
 }
 d_instantiate(dentry, inode);
 return 0;
}

#ifdef CONFIG_CIFS_ALLOW_INSECURE_LEGACY
static int
cifs_posix_mkdir(struct inode *inode, struct dentry *dentry, umode_t mode,
   const char *full_path, struct cifs_sb_info *cifs_sb,
   struct cifs_tcon *tcon, const unsigned int xid)
{
 int rc = 0;
 u32 oplock = 0;
 FILE_UNIX_BASIC_INFO *info = NULL;
 struct inode *newinode = NULL;
 struct cifs_fattr fattr;

 info = kzalloc(sizeof(FILE_UNIX_BASIC_INFO), GFP_KERNEL);
 if (info == NULL) {
  rc = -ENOMEM;
  goto posix_mkdir_out;
 }

 mode &= ~current_umask();
 rc = CIFSPOSIXCreate(xid, tcon, SMB_O_DIRECTORY | SMB_O_CREAT, mode,
        NULL /* netfid */, info, &oplock, full_path,
        cifs_sb->local_nls, cifs_remap(cifs_sb));
 if (rc == -EOPNOTSUPP)
  goto posix_mkdir_out;
 else if (rc) {
  cifs_dbg(FYI, "posix mkdir returned 0x%x\n", rc);
  d_drop(dentry);
  goto posix_mkdir_out;
 }

 if (info->Type == cpu_to_le32(-1))
  /* no return info, go query for it */
  goto posix_mkdir_get_info;
 /*
 * BB check (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID ) to see if
 * need to set uid/gid.
 */


 cifs_unix_basic_to_fattr(&fattr, info, cifs_sb);
 cifs_fill_uniqueid(inode->i_sb, &fattr);
 newinode = cifs_iget(inode->i_sb, &fattr);
 if (!newinode)
  goto posix_mkdir_get_info;

 d_instantiate(dentry, newinode);

#ifdef CONFIG_CIFS_DEBUG2
 cifs_dbg(FYI, "instantiated dentry %p %pd to inode %p\n",
   dentry, dentry, newinode);

 if (newinode->i_nlink != 2)
  cifs_dbg(FYI, "unexpected number of links %d\n",
    newinode->i_nlink);
#endif

posix_mkdir_out:
 kfree(info);
 return rc;
posix_mkdir_get_info:
 rc = cifs_mkdir_qinfo(inode, dentry, mode, full_path, cifs_sb, tcon,
         xid);
 goto posix_mkdir_out;
}
#endif /* CONFIG_CIFS_ALLOW_INSECURE_LEGACY */

struct dentry *cifs_mkdir(struct mnt_idmap *idmap, struct inode *inode,
     struct dentry *direntry, umode_t mode)
{
 int rc = 0;
 unsigned int xid;
 struct cifs_sb_info *cifs_sb;
 struct tcon_link *tlink;
 struct cifs_tcon *tcon;
 struct TCP_Server_Info *server;
 const char *full_path;
 void *page;

 cifs_dbg(FYI, "In cifs_mkdir, mode = %04ho inode = 0x%p\n",
   mode, inode);

 cifs_sb = CIFS_SB(inode->i_sb);
 if (unlikely(cifs_forced_shutdown(cifs_sb)))
  return ERR_PTR(-EIO);
 tlink = cifs_sb_tlink(cifs_sb);
 if (IS_ERR(tlink))
  return ERR_CAST(tlink);
 tcon = tlink_tcon(tlink);

 xid = get_xid();

 page = alloc_dentry_path();
 full_path = build_path_from_dentry(direntry, page);
 if (IS_ERR(full_path)) {
  rc = PTR_ERR(full_path);
  goto mkdir_out;
 }

 server = tcon->ses->server;

 if ((server->ops->posix_mkdir) && (tcon->posix_extensions)) {
  rc = server->ops->posix_mkdir(xid, inode, mode, tcon, full_path,
           cifs_sb);
  d_drop(direntry); /* for time being always refresh inode info */
  goto mkdir_out;
 }

#ifdef CONFIG_CIFS_ALLOW_INSECURE_LEGACY
 if (cap_unix(tcon->ses) && (CIFS_UNIX_POSIX_PATH_OPS_CAP &
    le64_to_cpu(tcon->fsUnixInfo.Capability))) {
  rc = cifs_posix_mkdir(inode, direntry, mode, full_path, cifs_sb,
          tcon, xid);
  if (rc != -EOPNOTSUPP)
   goto mkdir_out;
 }
#endif /* CONFIG_CIFS_ALLOW_INSECURE_LEGACY */

 if (!server->ops->mkdir) {
  rc = -ENOSYS;
  goto mkdir_out;
 }

 /* BB add setting the equivalent of mode via CreateX w/ACLs */
 rc = server->ops->mkdir(xid, inode, mode, tcon, full_path, cifs_sb);
 if (rc) {
  cifs_dbg(FYI, "cifs_mkdir returned 0x%x\n", rc);
  d_drop(direntry);
  goto mkdir_out;
 }

 /* TODO: skip this for smb2/smb3 */
 rc = cifs_mkdir_qinfo(inode, direntry, mode, full_path, cifs_sb, tcon,
         xid);
mkdir_out:
 /*
 * Force revalidate to get parent dir info when needed since cached
 * attributes are invalid now.
 */

 CIFS_I(inode)->time = 0;
 free_dentry_path(page);
 free_xid(xid);
 cifs_put_tlink(tlink);
 return ERR_PTR(rc);
}

int cifs_rmdir(struct inode *inode, struct dentry *direntry)
{
 int rc = 0;
 unsigned int xid;
 struct cifs_sb_info *cifs_sb;
 struct tcon_link *tlink;
 struct cifs_tcon *tcon;
 struct TCP_Server_Info *server;
 const char *full_path;
 void *page = alloc_dentry_path();
 struct cifsInodeInfo *cifsInode;

 cifs_dbg(FYI, "cifs_rmdir, inode = 0x%p\n", inode);

 xid = get_xid();

 full_path = build_path_from_dentry(direntry, page);
 if (IS_ERR(full_path)) {
  rc = PTR_ERR(full_path);
  goto rmdir_exit;
 }

 cifs_sb = CIFS_SB(inode->i_sb);
 if (unlikely(cifs_forced_shutdown(cifs_sb))) {
  rc = -EIO;
  goto rmdir_exit;
 }

 tlink = cifs_sb_tlink(cifs_sb);
 if (IS_ERR(tlink)) {
  rc = PTR_ERR(tlink);
  goto rmdir_exit;
 }
 tcon = tlink_tcon(tlink);
 server = tcon->ses->server;

 if (!server->ops->rmdir) {
  rc = -ENOSYS;
  cifs_put_tlink(tlink);
  goto rmdir_exit;
 }

 if (tcon->nodelete) {
  rc = -EACCES;
  cifs_put_tlink(tlink);
  goto rmdir_exit;
 }

 rc = server->ops->rmdir(xid, tcon, full_path, cifs_sb);
 cifs_put_tlink(tlink);

 cifsInode = CIFS_I(d_inode(direntry));

 if (!rc) {
  set_bit(CIFS_INO_DELETE_PENDING, &cifsInode->flags);
  spin_lock(&d_inode(direntry)->i_lock);
  i_size_write(d_inode(direntry), 0);
  clear_nlink(d_inode(direntry));
  spin_unlock(&d_inode(direntry)->i_lock);
 }

--> --------------------

--> maximum size reached

--> --------------------

Messung V0.5
C=96 H=87 G=91

¤ Dauer der Verarbeitung: 0.22 Sekunden  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.