From 2b92531df29bda1abea88235fbdf952c80d9d469 Mon Sep 17 00:00:00 2001 From: Seth Jennings <sjenning@redhat.com> Date: Thu, 15 May 2014 15:22:52 -0500 Subject: [PATCH] fix list corruption in special section handlers The kpatch_regenerate_* functions use a local list_head to construct the new list. While the local list_head is copied to the sec->relas after it is built, the neighboring nodes in the list are not updated, leading to list corruption. This commit uses list_replace() which updates the neighbor nodes properly. Regression introduced by PR #117 5d36dd1. Fixes #185. Signed-off-by: Seth Jennings <sjenning@redhat.com> --- kpatch-build/create-diff-object.c | 6 +++--- kpatch-build/list.h | 16 ++++++++++++++++ 2 files changed, 19 insertions(+), 3 deletions(-) diff --git a/kpatch-build/create-diff-object.c b/kpatch-build/create-diff-object.c index fbce2a0..cd58730 100644 --- a/kpatch-build/create-diff-object.c +++ b/kpatch-build/create-diff-object.c @@ -1029,7 +1029,7 @@ void kpatch_regenerate_bug_table_rela_section(struct kpatch_elf *kelf) } /* overwrite with new relas list */ - sec->relas = newrelas; + list_replace(&newrelas, &sec->relas); /* include both rela and text sections */ sec->include = 1; @@ -1076,7 +1076,7 @@ void kpatch_regenerate_smp_locks_sections(struct kpatch_elf *kelf) } /* overwrite with new relas list */ - sec->relas = newrelas; + list_replace(&newrelas, &sec->relas); /* include both rela and text sections */ sec->include = 1; @@ -1141,7 +1141,7 @@ void kpatch_regenerate_parainstructions_sections(struct kpatch_elf *kelf) } /* overwrite with new relas table */ - sec->relas = newrelas; + list_replace(&newrelas, &sec->relas); /* mark sections for inclusion */ sec->include = 1; diff --git a/kpatch-build/list.h b/kpatch-build/list.h index 7da8453..825fd95 100644 --- a/kpatch-build/list.h +++ b/kpatch-build/list.h @@ -142,6 +142,22 @@ static inline void list_del(struct list_head *entry) entry->prev = LIST_POISON2; } +/** + * list_replace - replace old entry by new one + * @old : the element to be replaced + * @new : the new element to insert + * + * If @old was empty, it will be overwritten. + */ +static inline void list_replace(struct list_head *old, + struct list_head *new) +{ + new->next = old->next; + new->next->prev = new; + new->prev = old->prev; + new->prev->next = new; +} + #define list_entry(ptr, type, member) \ container_of(ptr, type, member)