return (flag & O_TRUNC);
  }
  
 -/*
 - * Note that the low bits of the passed in "open_flag"
 - * are not the same as in the local variable "flag". See
 - * open_to_namei_flags() for more details.
 - */
 -struct file *do_filp_open(int dfd, const char *pathname,
 -              int open_flag, int mode, int acc_mode)
 +static struct file *finish_open(struct nameidata *nd,
 +                              int open_flag, int acc_mode)
  {
        struct file *filp;
 -      struct nameidata nd;
 -      int error;
 -      struct path path;
 -      struct dentry *dir;
 -      int count = 0;
        int will_truncate;
 -      int flag = open_to_namei_flags(open_flag);
 -      int force_reval = 0;
 +      int error;
  
-               if (acc_mode & MAY_WRITE)
-                       vfs_dq_init(nd->path.dentry->d_inode);
- 
 +      will_truncate = open_will_truncate(open_flag, nd->path.dentry->d_inode);
 +      if (will_truncate) {
 +              error = mnt_want_write(nd->path.mnt);
 +              if (error)
 +                      goto exit;
 +      }
 +      error = may_open(&nd->path, acc_mode, open_flag);
 +      if (error) {
 +              if (will_truncate)
 +                      mnt_drop_write(nd->path.mnt);
 +              goto exit;
 +      }
 +      filp = nameidata_to_filp(nd);
 +      if (!IS_ERR(filp)) {
 +              error = ima_file_check(filp, acc_mode);
 +              if (error) {
 +                      fput(filp);
 +                      filp = ERR_PTR(error);
 +              }
 +      }
 +      if (!IS_ERR(filp)) {
 +              if (will_truncate) {
 +                      error = handle_truncate(&nd->path);
 +                      if (error) {
 +                              fput(filp);
 +                              filp = ERR_PTR(error);
 +                      }
 +              }
 +      }
        /*
 -       * O_SYNC is implemented as __O_SYNC|O_DSYNC.  As many places only
 -       * check for O_DSYNC if the need any syncing at all we enforce it's
 -       * always set instead of having to deal with possibly weird behaviour
 -       * for malicious applications setting only __O_SYNC.
 +       * It is now safe to drop the mnt write
 +       * because the filp has had a write taken
 +       * on its behalf.
         */
 -      if (open_flag & __O_SYNC)
 -              open_flag |= O_DSYNC;
 -
 -      if (!acc_mode)
 -              acc_mode = MAY_OPEN | ACC_MODE(open_flag);
 +      if (will_truncate)
 +              mnt_drop_write(nd->path.mnt);
 +      return filp;
  
 -      /* O_TRUNC implies we need access checks for write permissions */
 -      if (flag & O_TRUNC)
 -              acc_mode |= MAY_WRITE;
 +exit:
 +      if (!IS_ERR(nd->intent.open.file))
 +              release_open_intent(nd);
 +      path_put(&nd->path);
 +      return ERR_PTR(error);
 +}
  
 -      /* Allow the LSM permission hook to distinguish append 
 -         access from general write access. */
 -      if (flag & O_APPEND)
 -              acc_mode |= MAY_APPEND;
 +static struct file *do_last(struct nameidata *nd, struct path *path,
 +                          int open_flag, int acc_mode,
 +                          int mode, const char *pathname,
 +                          int *want_dir)
 +{
 +      struct dentry *dir = nd->path.dentry;
 +      struct file *filp;
 +      int error = -EISDIR;
  
 -      /*
 -       * The simplest case - just a plain lookup.
 -       */
 -      if (!(flag & O_CREAT)) {
 -              filp = get_empty_filp();
 -
 -              if (filp == NULL)
 -                      return ERR_PTR(-ENFILE);
 -              nd.intent.open.file = filp;
 -              filp->f_flags = open_flag;
 -              nd.intent.open.flags = flag;
 -              nd.intent.open.create_mode = 0;
 -              error = do_path_lookup(dfd, pathname,
 -                                      lookup_flags(flag)|LOOKUP_OPEN, &nd);
 -              if (IS_ERR(nd.intent.open.file)) {
 -                      if (error == 0) {
 -                              error = PTR_ERR(nd.intent.open.file);
 -                              path_put(&nd.path);
 +      switch (nd->last_type) {
 +      case LAST_DOTDOT:
 +              follow_dotdot(nd);
 +              dir = nd->path.dentry;
 +              if (nd->path.mnt->mnt_sb->s_type->fs_flags & FS_REVAL_DOT) {
 +                      if (!dir->d_op->d_revalidate(dir, nd)) {
 +                              error = -ESTALE;
 +                              goto exit;
                        }
 -              } else if (error)
 -                      release_open_intent(&nd);
 -              if (error)
 -                      return ERR_PTR(error);
 +              }
 +              /* fallthrough */
 +      case LAST_DOT:
 +      case LAST_ROOT:
 +              if (open_flag & O_CREAT)
 +                      goto exit;
 +              /* fallthrough */
 +      case LAST_BIND:
 +              audit_inode(pathname, dir);
                goto ok;
        }