diff mbox

Btrfs: merge inode_list in __merge_refs

Message ID 1352410044-10547-1-git-send-email-ablock84@googlemail.com (mailing list archive)
State New, archived
Headers show

Commit Message

Alexander Block Nov. 8, 2012, 9:27 p.m. UTC
When __merge_refs merges two refs, it is also needed to merge the
inode_list of both refs. Otherwise we have missed backrefs and memory
leaks. This happens for example if two inodes share an extent and
both lie in the same leaf and thus also have the same parent.

Signed-off-by: Alexander Block <ablock84@googlemail.com>
Reviewed-by: Jan Schmidt <list.btrfs@jan-o-sch.net>
---
 fs/btrfs/backref.c |   13 +++++++++++--
 1 file changed, 11 insertions(+), 2 deletions(-)

Comments

Alexander Block Nov. 8, 2012, 9:35 p.m. UTC | #1
On Thu, Nov 8, 2012 at 10:27 PM, Alexander Block
<ablock84@googlemail.com> wrote:
> When __merge_refs merges two refs, it is also needed to merge the
> inode_list of both refs. Otherwise we have missed backrefs and memory
> leaks. This happens for example if two inodes share an extent and
> both lie in the same leaf and thus also have the same parent.
>
> Signed-off-by: Alexander Block <ablock84@googlemail.com>
> Reviewed-by: Jan Schmidt <list.btrfs@jan-o-sch.net>
> ---
>  fs/btrfs/backref.c |   13 +++++++++++--
>  1 file changed, 11 insertions(+), 2 deletions(-)
>
> diff --git a/fs/btrfs/backref.c b/fs/btrfs/backref.c
> index 208d8aa..aea6d2d 100644
> --- a/fs/btrfs/backref.c
> +++ b/fs/btrfs/backref.c
> @@ -461,6 +461,7 @@ static int __merge_refs(struct list_head *head, int mode)
>                      pos2 = n2, n2 = pos2->next) {
>                         struct __prelim_ref *ref2;
>                         struct __prelim_ref *xchg;
> +                       struct extent_inode_elem *eie;
>
>                         ref2 = list_entry(pos2, struct __prelim_ref, list);
>
> @@ -472,12 +473,20 @@ static int __merge_refs(struct list_head *head, int mode)
>                                         ref1 = ref2;
>                                         ref2 = xchg;
>                                 }
> -                               ref1->count += ref2->count;
>                         } else {
>                                 if (ref1->parent != ref2->parent)
>                                         continue;
> -                               ref1->count += ref2->count;
>                         }
> +
> +                       eie = ref1->inode_list;
> +                       while (eie && eie->next)
> +                               eie = eie->next;
> +                       if (eie)
> +                               eie->next = ref2->inode_list;
> +                       else
> +                               ref1->inode_list = ref2->inode_list;
> +                       ref1->count += ref2->count;
> +
>                         list_del(&ref2->list);
>                         kfree(ref2);
>                 }
> --
> 1.7.10.4
>

Used wrong CC for stable list. Corrected now.
--
To unsubscribe from this list: send the line "unsubscribe linux-btrfs" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Greg KH Nov. 9, 2012, 7:01 a.m. UTC | #2
On Thu, Nov 08, 2012 at 10:35:19PM +0100, Alexander Block wrote:
> 
> Used wrong CC for stable list. Corrected now.

<formletter>

This is not the correct way to submit patches for inclusion in the
stable kernel tree.  Please read Documentation/stable_kernel_rules.txt
for how to do this properly.

</formletter>
--
To unsubscribe from this list: send the line "unsubscribe linux-btrfs" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
diff mbox

Patch

diff --git a/fs/btrfs/backref.c b/fs/btrfs/backref.c
index 208d8aa..aea6d2d 100644
--- a/fs/btrfs/backref.c
+++ b/fs/btrfs/backref.c
@@ -461,6 +461,7 @@  static int __merge_refs(struct list_head *head, int mode)
 		     pos2 = n2, n2 = pos2->next) {
 			struct __prelim_ref *ref2;
 			struct __prelim_ref *xchg;
+			struct extent_inode_elem *eie;
 
 			ref2 = list_entry(pos2, struct __prelim_ref, list);
 
@@ -472,12 +473,20 @@  static int __merge_refs(struct list_head *head, int mode)
 					ref1 = ref2;
 					ref2 = xchg;
 				}
-				ref1->count += ref2->count;
 			} else {
 				if (ref1->parent != ref2->parent)
 					continue;
-				ref1->count += ref2->count;
 			}
+
+			eie = ref1->inode_list;
+			while (eie && eie->next)
+				eie = eie->next;
+			if (eie)
+				eie->next = ref2->inode_list;
+			else
+				ref1->inode_list = ref2->inode_list;
+			ref1->count += ref2->count;
+
 			list_del(&ref2->list);
 			kfree(ref2);
 		}