Merge tag 'upstream-4.9-rc1' of git://git.infradead.org/linux-ubifs
[cascardo/linux.git] / fs / ubifs / journal.c
index 0b9da5b..91bc76d 100644 (file)
@@ -907,6 +907,147 @@ int ubifs_jnl_delete_inode(struct ubifs_info *c, const struct inode *inode)
        return err;
 }
 
+/**
+ * ubifs_jnl_xrename - cross rename two directory entries.
+ * @c: UBIFS file-system description object
+ * @fst_dir: parent inode of 1st directory entry to exchange
+ * @fst_dentry: 1st directory entry to exchange
+ * @snd_dir: parent inode of 2nd directory entry to exchange
+ * @snd_dentry: 2nd directory entry to exchange
+ * @sync: non-zero if the write-buffer has to be synchronized
+ *
+ * This function implements the cross rename operation which may involve
+ * writing 2 inodes and 2 directory entries. It marks the written inodes as clean
+ * and returns zero on success. In case of failure, a negative error code is
+ * returned.
+ */
+int ubifs_jnl_xrename(struct ubifs_info *c, const struct inode *fst_dir,
+                     const struct dentry *fst_dentry,
+                     const struct inode *snd_dir,
+                     const struct dentry *snd_dentry, int sync)
+{
+       union ubifs_key key;
+       struct ubifs_dent_node *dent1, *dent2;
+       int err, dlen1, dlen2, lnum, offs, len, plen = UBIFS_INO_NODE_SZ;
+       int aligned_dlen1, aligned_dlen2;
+       int twoparents = (fst_dir != snd_dir);
+       const struct inode *fst_inode = d_inode(fst_dentry);
+       const struct inode *snd_inode = d_inode(snd_dentry);
+       void *p;
+
+       dbg_jnl("dent '%pd' in dir ino %lu between dent '%pd' in dir ino %lu",
+               fst_dentry, fst_dir->i_ino, snd_dentry, snd_dir->i_ino);
+
+       ubifs_assert(ubifs_inode(fst_dir)->data_len == 0);
+       ubifs_assert(ubifs_inode(snd_dir)->data_len == 0);
+       ubifs_assert(mutex_is_locked(&ubifs_inode(fst_dir)->ui_mutex));
+       ubifs_assert(mutex_is_locked(&ubifs_inode(snd_dir)->ui_mutex));
+
+       dlen1 = UBIFS_DENT_NODE_SZ + snd_dentry->d_name.len + 1;
+       dlen2 = UBIFS_DENT_NODE_SZ + fst_dentry->d_name.len + 1;
+       aligned_dlen1 = ALIGN(dlen1, 8);
+       aligned_dlen2 = ALIGN(dlen2, 8);
+
+       len = aligned_dlen1 + aligned_dlen2 + ALIGN(plen, 8);
+       if (twoparents)
+               len += plen;
+
+       dent1 = kmalloc(len, GFP_NOFS);
+       if (!dent1)
+               return -ENOMEM;
+
+       /* Make reservation before allocating sequence numbers */
+       err = make_reservation(c, BASEHD, len);
+       if (err)
+               goto out_free;
+
+       /* Make new dent for 1st entry */
+       dent1->ch.node_type = UBIFS_DENT_NODE;
+       dent_key_init_flash(c, &dent1->key, snd_dir->i_ino, &snd_dentry->d_name);
+       dent1->inum = cpu_to_le64(fst_inode->i_ino);
+       dent1->type = get_dent_type(fst_inode->i_mode);
+       dent1->nlen = cpu_to_le16(snd_dentry->d_name.len);
+       memcpy(dent1->name, snd_dentry->d_name.name, snd_dentry->d_name.len);
+       dent1->name[snd_dentry->d_name.len] = '\0';
+       zero_dent_node_unused(dent1);
+       ubifs_prep_grp_node(c, dent1, dlen1, 0);
+
+       /* Make new dent for 2nd entry */
+       dent2 = (void *)dent1 + aligned_dlen1;
+       dent2->ch.node_type = UBIFS_DENT_NODE;
+       dent_key_init_flash(c, &dent2->key, fst_dir->i_ino, &fst_dentry->d_name);
+       dent2->inum = cpu_to_le64(snd_inode->i_ino);
+       dent2->type = get_dent_type(snd_inode->i_mode);
+       dent2->nlen = cpu_to_le16(fst_dentry->d_name.len);
+       memcpy(dent2->name, fst_dentry->d_name.name, fst_dentry->d_name.len);
+       dent2->name[fst_dentry->d_name.len] = '\0';
+       zero_dent_node_unused(dent2);
+       ubifs_prep_grp_node(c, dent2, dlen2, 0);
+
+       p = (void *)dent2 + aligned_dlen2;
+       if (!twoparents)
+               pack_inode(c, p, fst_dir, 1);
+       else {
+               pack_inode(c, p, fst_dir, 0);
+               p += ALIGN(plen, 8);
+               pack_inode(c, p, snd_dir, 1);
+       }
+
+       err = write_head(c, BASEHD, dent1, len, &lnum, &offs, sync);
+       if (err)
+               goto out_release;
+       if (!sync) {
+               struct ubifs_wbuf *wbuf = &c->jheads[BASEHD].wbuf;
+
+               ubifs_wbuf_add_ino_nolock(wbuf, fst_dir->i_ino);
+               ubifs_wbuf_add_ino_nolock(wbuf, snd_dir->i_ino);
+       }
+       release_head(c, BASEHD);
+
+       dent_key_init(c, &key, snd_dir->i_ino, &snd_dentry->d_name);
+       err = ubifs_tnc_add_nm(c, &key, lnum, offs, dlen1, &snd_dentry->d_name);
+       if (err)
+               goto out_ro;
+
+       offs += aligned_dlen1;
+       dent_key_init(c, &key, fst_dir->i_ino, &fst_dentry->d_name);
+       err = ubifs_tnc_add_nm(c, &key, lnum, offs, dlen2, &fst_dentry->d_name);
+       if (err)
+               goto out_ro;
+
+       offs += aligned_dlen2;
+
+       ino_key_init(c, &key, fst_dir->i_ino);
+       err = ubifs_tnc_add(c, &key, lnum, offs, plen);
+       if (err)
+               goto out_ro;
+
+       if (twoparents) {
+               offs += ALIGN(plen, 8);
+               ino_key_init(c, &key, snd_dir->i_ino);
+               err = ubifs_tnc_add(c, &key, lnum, offs, plen);
+               if (err)
+                       goto out_ro;
+       }
+
+       finish_reservation(c);
+
+       mark_inode_clean(c, ubifs_inode(fst_dir));
+       if (twoparents)
+               mark_inode_clean(c, ubifs_inode(snd_dir));
+       kfree(dent1);
+       return 0;
+
+out_release:
+       release_head(c, BASEHD);
+out_ro:
+       ubifs_ro_mode(c, err);
+       finish_reservation(c);
+out_free:
+       kfree(dent1);
+       return err;
+}
+
 /**
  * ubifs_jnl_rename - rename a directory entry.
  * @c: UBIFS file-system description object
@@ -917,14 +1058,15 @@ int ubifs_jnl_delete_inode(struct ubifs_info *c, const struct inode *inode)
  * @sync: non-zero if the write-buffer has to be synchronized
  *
  * This function implements the re-name operation which may involve writing up
- * to 3 inodes and 2 directory entries. It marks the written inodes as clean
+ * to 4 inodes and 2 directory entries. It marks the written inodes as clean
  * and returns zero on success. In case of failure, a negative error code is
  * returned.
  */
 int ubifs_jnl_rename(struct ubifs_info *c, const struct inode *old_dir,
                     const struct dentry *old_dentry,
                     const struct inode *new_dir,
-                    const struct dentry *new_dentry, int sync)
+                    const struct dentry *new_dentry,
+                    const struct inode *whiteout, int sync)
 {
        void *p;
        union ubifs_key key;
@@ -958,7 +1100,7 @@ int ubifs_jnl_rename(struct ubifs_info *c, const struct inode *old_dir,
        aligned_dlen1 = ALIGN(dlen1, 8);
        aligned_dlen2 = ALIGN(dlen2, 8);
        len = aligned_dlen1 + aligned_dlen2 + ALIGN(ilen, 8) + ALIGN(plen, 8);
-       if (old_dir != new_dir)
+       if (move)
                len += plen;
        dent = kmalloc(len, GFP_NOFS);
        if (!dent)
@@ -980,13 +1122,19 @@ int ubifs_jnl_rename(struct ubifs_info *c, const struct inode *old_dir,
        zero_dent_node_unused(dent);
        ubifs_prep_grp_node(c, dent, dlen1, 0);
 
-       /* Make deletion dent */
        dent2 = (void *)dent + aligned_dlen1;
        dent2->ch.node_type = UBIFS_DENT_NODE;
        dent_key_init_flash(c, &dent2->key, old_dir->i_ino,
                            &old_dentry->d_name);
-       dent2->inum = 0;
-       dent2->type = DT_UNKNOWN;
+
+       if (whiteout) {
+               dent2->inum = cpu_to_le64(whiteout->i_ino);
+               dent2->type = get_dent_type(whiteout->i_mode);
+       } else {
+               /* Make deletion dent */
+               dent2->inum = 0;
+               dent2->type = DT_UNKNOWN;
+       }
        dent2->nlen = cpu_to_le16(old_dentry->d_name.len);
        memcpy(dent2->name, old_dentry->d_name.name, old_dentry->d_name.len);
        dent2->name[old_dentry->d_name.len] = '\0';
@@ -1035,16 +1183,26 @@ int ubifs_jnl_rename(struct ubifs_info *c, const struct inode *old_dir,
        if (err)
                goto out_ro;
 
-       err = ubifs_add_dirt(c, lnum, dlen2);
-       if (err)
-               goto out_ro;
+       offs += aligned_dlen1;
+       if (whiteout) {
+               dent_key_init(c, &key, old_dir->i_ino, &old_dentry->d_name);
+               err = ubifs_tnc_add_nm(c, &key, lnum, offs, dlen2, &old_dentry->d_name);
+               if (err)
+                       goto out_ro;
 
-       dent_key_init(c, &key, old_dir->i_ino, &old_dentry->d_name);
-       err = ubifs_tnc_remove_nm(c, &key, &old_dentry->d_name);
-       if (err)
-               goto out_ro;
+               ubifs_delete_orphan(c, whiteout->i_ino);
+       } else {
+               err = ubifs_add_dirt(c, lnum, dlen2);
+               if (err)
+                       goto out_ro;
+
+               dent_key_init(c, &key, old_dir->i_ino, &old_dentry->d_name);
+               err = ubifs_tnc_remove_nm(c, &key, &old_dentry->d_name);
+               if (err)
+                       goto out_ro;
+       }
 
-       offs += aligned_dlen1 + aligned_dlen2;
+       offs += aligned_dlen2;
        if (new_inode) {
                ino_key_init(c, &key, new_inode->i_ino);
                err = ubifs_tnc_add(c, &key, lnum, offs, ilen);
@@ -1058,7 +1216,7 @@ int ubifs_jnl_rename(struct ubifs_info *c, const struct inode *old_dir,
        if (err)
                goto out_ro;
 
-       if (old_dir != new_dir) {
+       if (move) {
                offs += ALIGN(plen, 8);
                ino_key_init(c, &key, new_dir->i_ino);
                err = ubifs_tnc_add(c, &key, lnum, offs, plen);