mirror of
https://github.com/dynup/kpatch
synced 2025-04-01 22:48:08 +00:00
create-diff-object: Remove the multi_pfe flag.
In ARM64, every function section should have its own pfe section. It is a bug in GCC 11/12 which will only generate a single pfe section for all functions. The bug has been fixed in GCC 13.1. As the create-diff-object is generating the pfe sections on its own, we should also fix this bug, instead of try to repeat the bug. -- Adjusted whitespace in Zimao's proposed code. Signed-off-by: Pete Swain <swine@google.com>
This commit is contained in:
parent
d7a3c475d5
commit
dc2864904f
kpatch-build
@ -70,7 +70,6 @@ enum subsection {
|
|||||||
enum loglevel loglevel = NORMAL;
|
enum loglevel loglevel = NORMAL;
|
||||||
|
|
||||||
bool KLP_ARCH;
|
bool KLP_ARCH;
|
||||||
bool multi_pfe;
|
|
||||||
|
|
||||||
int jump_label_errors, static_call_errors;
|
int jump_label_errors, static_call_errors;
|
||||||
|
|
||||||
@ -3773,114 +3772,68 @@ static void kpatch_create_callbacks_objname_rela(struct kpatch_elf *kelf, char *
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
static void kpatch_set_pfe_link(struct kpatch_elf *kelf)
|
||||||
* Allocate the mcount/patchable_function_entry sections which must be done
|
|
||||||
* before the patched object is torn down so that the section flags can be
|
|
||||||
* copied.
|
|
||||||
*/
|
|
||||||
static void kpatch_alloc_mcount_sections(struct kpatch_elf *kelf, struct kpatch_elf *kelfout)
|
|
||||||
{
|
{
|
||||||
int nr;
|
struct section* sec;
|
||||||
struct symbol *sym;
|
struct rela *rela;
|
||||||
int text_idx = 0;
|
|
||||||
|
|
||||||
nr = 0;
|
list_for_each_entry(sec, &kelf->sections, list) {
|
||||||
list_for_each_entry(sym, &kelfout->symbols, list) {
|
if (strcmp(sec->name, "__patchable_function_entries")) {
|
||||||
if (sym->type == STT_FUNC && sym->status != SAME &&
|
continue;
|
||||||
sym->has_func_profiling) {
|
|
||||||
text_idx = sym->sec->index;
|
|
||||||
nr++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* create text/rela section pair */
|
|
||||||
switch(kelf->arch) {
|
|
||||||
case AARCH64: {
|
|
||||||
struct section *sec;
|
|
||||||
int entries = multi_pfe ? 1 : nr;
|
|
||||||
int copies = multi_pfe ? nr : 1;
|
|
||||||
int flags = 0, rflags = 0;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Depending on the compiler the __patchable_function_entries section
|
|
||||||
* can be ordered or not, copy this flag to the section we created to
|
|
||||||
* avoid:
|
|
||||||
* ld: __patchable_function_entries has both ordered [...] and unordered [...] sections
|
|
||||||
*/
|
|
||||||
sec = find_section_by_name(&kelf->sections, "__patchable_function_entries");
|
|
||||||
if (sec) {
|
|
||||||
flags = (sec->sh.sh_flags & (SHF_LINK_ORDER|SHF_WRITE));
|
|
||||||
if (sec->rela)
|
|
||||||
rflags = (sec->rela->sh.sh_flags & (SHF_LINK_ORDER|SHF_WRITE));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (nr = 0; nr < copies; nr++) {
|
if (!sec->rela) {
|
||||||
sec = create_section_pair(kelfout,
|
continue;
|
||||||
"__patchable_function_entries",
|
}
|
||||||
sizeof(void *), entries);
|
list_for_each_entry(rela, &sec->rela->relas, list) {
|
||||||
|
rela->sym->sec->pfe = sec;
|
||||||
sec->sh.sh_flags |= flags;
|
|
||||||
if (sec->rela)
|
|
||||||
sec->rela->sh.sh_flags |= rflags;
|
|
||||||
if (multi_pfe)
|
|
||||||
sec->sh.sh_link = 0;
|
|
||||||
else
|
|
||||||
sec->sh.sh_link = text_idx;
|
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PPC64:
|
|
||||||
case X86_64:
|
|
||||||
case S390:
|
|
||||||
create_section_pair(kelfout, "__mcount_loc", sizeof(void *), nr);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
ERROR("unsupported arch\n");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Populate the mcount sections allocated by kpatch_alloc_mcount_sections()
|
|
||||||
* previously.
|
|
||||||
* This function basically reimplements the functionality of the Linux
|
* This function basically reimplements the functionality of the Linux
|
||||||
* recordmcount script, so that patched functions can be recognized by ftrace.
|
* recordmcount script, so that patched functions can be recognized by ftrace.
|
||||||
*
|
*
|
||||||
* TODO: Eventually we can modify recordmount so that it recognizes our bundled
|
* TODO: Eventually we can modify recordmount so that it recognizes our bundled
|
||||||
* sections as valid and does this work for us.
|
* sections as valid and does this work for us.
|
||||||
*/
|
*/
|
||||||
static void kpatch_populate_mcount_sections(struct kpatch_elf *kelf)
|
static void kpatch_create_mcount_sections(struct kpatch_elf *kelf)
|
||||||
{
|
{
|
||||||
int nr, index;
|
int nr, index;
|
||||||
struct section *sec, *relasec;
|
struct section *relasec;
|
||||||
struct symbol *sym;
|
struct symbol *sym;
|
||||||
struct rela *rela, *mcount_rela;
|
struct rela *rela, *mcount_rela;
|
||||||
void **funcs;
|
void **funcs;
|
||||||
|
bool pfe_per_function = false;
|
||||||
|
|
||||||
switch(kelf->arch) {
|
nr = 0;
|
||||||
|
list_for_each_entry(sym, &kelf->symbols, list)
|
||||||
|
if (sym->type == STT_FUNC && sym->status != SAME &&
|
||||||
|
sym->has_func_profiling)
|
||||||
|
nr++;
|
||||||
|
|
||||||
|
switch (kelf->arch) {
|
||||||
case AARCH64:
|
case AARCH64:
|
||||||
if (multi_pfe)
|
/* For aarch64, we will create separate __patchable_function_entries sections for each symbols. */
|
||||||
sec = NULL;
|
pfe_per_function = true;
|
||||||
else
|
relasec = NULL;
|
||||||
sec = find_section_by_name(&kelf->sections,
|
|
||||||
"__patchable_function_entries");
|
|
||||||
break;
|
break;
|
||||||
case PPC64:
|
case PPC64:
|
||||||
case X86_64:
|
case X86_64:
|
||||||
case S390:
|
case S390:
|
||||||
sec = find_section_by_name(&kelf->sections, "__mcount_loc");
|
{
|
||||||
|
struct section *sec;
|
||||||
|
|
||||||
|
/* create text/rela section pair */
|
||||||
|
sec = create_section_pair(kelf, "__mcount_loc", sizeof(void*), nr);
|
||||||
|
relasec = sec->rela;
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
ERROR("unsupported arch\n");
|
ERROR("unsupported arch\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (multi_pfe) {
|
|
||||||
relasec = NULL;
|
|
||||||
nr = 0;
|
|
||||||
} else {
|
|
||||||
relasec = sec->rela;
|
|
||||||
nr = (int) (sec->data->d_size / sizeof(void *));
|
|
||||||
}
|
|
||||||
|
|
||||||
/* populate sections */
|
/* populate sections */
|
||||||
index = 0;
|
index = 0;
|
||||||
list_for_each_entry(sym, &kelf->symbols, list) {
|
list_for_each_entry(sym, &kelf->symbols, list) {
|
||||||
@ -3897,6 +3850,7 @@ static void kpatch_populate_mcount_sections(struct kpatch_elf *kelf)
|
|||||||
|
|
||||||
switch(kelf->arch) {
|
switch(kelf->arch) {
|
||||||
case AARCH64: {
|
case AARCH64: {
|
||||||
|
struct section *sec;
|
||||||
unsigned char *insn;
|
unsigned char *insn;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
@ -3921,6 +3875,14 @@ static void kpatch_populate_mcount_sections(struct kpatch_elf *kelf)
|
|||||||
ERROR("%s: unexpected instruction in patch section of function\n", sym->name);
|
ERROR("%s: unexpected instruction in patch section of function\n", sym->name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Allocate __patchable_function_entries for symbol */
|
||||||
|
sec = create_section_pair(kelf, "__patchable_function_entries", sizeof(void *), 1);
|
||||||
|
sec->sh.sh_flags |= SHF_WRITE | SHF_LINK_ORDER;
|
||||||
|
/* We will reset this sh_link in the reindex function. */
|
||||||
|
sec->sh.sh_link = 0;
|
||||||
|
|
||||||
|
relasec = sec->rela;
|
||||||
|
sym->sec->pfe = sec;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case PPC64: {
|
case PPC64: {
|
||||||
@ -3991,18 +3953,6 @@ static void kpatch_populate_mcount_sections(struct kpatch_elf *kelf)
|
|||||||
ERROR("unsupported arch");
|
ERROR("unsupported arch");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (multi_pfe) {
|
|
||||||
sec = find_nth_section_by_name(&kelf->sections, nr, "__patchable_function_entries");
|
|
||||||
if (!sec)
|
|
||||||
ERROR("cannot retrieve pre-allocated __pfe #%d\n", nr);
|
|
||||||
|
|
||||||
relasec = sec->rela;
|
|
||||||
sym->sec->pfe = sec;
|
|
||||||
sec->sh.sh_link = sec->index;
|
|
||||||
|
|
||||||
nr++;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* 'rela' points to the mcount/fentry call.
|
* 'rela' points to the mcount/fentry call.
|
||||||
*
|
*
|
||||||
@ -4013,9 +3963,8 @@ static void kpatch_populate_mcount_sections(struct kpatch_elf *kelf)
|
|||||||
mcount_rela->type = absolute_rela_type(kelf);
|
mcount_rela->type = absolute_rela_type(kelf);
|
||||||
mcount_rela->addend = insn_offset - sym->sym.st_value;
|
mcount_rela->addend = insn_offset - sym->sym.st_value;
|
||||||
|
|
||||||
if (multi_pfe) {
|
if (pfe_per_function) {
|
||||||
mcount_rela->offset = 0;
|
mcount_rela->offset = 0;
|
||||||
sec = NULL;
|
|
||||||
} else {
|
} else {
|
||||||
mcount_rela->offset = (unsigned int) (index * sizeof(*funcs));
|
mcount_rela->offset = (unsigned int) (index * sizeof(*funcs));
|
||||||
}
|
}
|
||||||
@ -4185,19 +4134,21 @@ static void kpatch_find_func_profiling_calls(struct kpatch_elf *kelf)
|
|||||||
switch(kelf->arch) {
|
switch(kelf->arch) {
|
||||||
case AARCH64: {
|
case AARCH64: {
|
||||||
struct section *sec;
|
struct section *sec;
|
||||||
|
|
||||||
list_for_each_entry(sec, &kelf->sections, list) {
|
list_for_each_entry(sec, &kelf->sections, list) {
|
||||||
if (strcmp(sec->name, "__patchable_function_entries"))
|
if (strcmp(sec->name, "__patchable_function_entries")) {
|
||||||
continue;
|
continue;
|
||||||
if (multi_pfe && sym->sec->pfe != sec)
|
}
|
||||||
|
if (sym->sec->pfe != sec) {
|
||||||
continue;
|
continue;
|
||||||
if (!sec->rela)
|
}
|
||||||
|
if (!sec->rela) {
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
list_for_each_entry(rela, &sec->rela->relas, list) {
|
list_for_each_entry(rela, &sec->rela->relas, list) {
|
||||||
if (rela->sym->sec && sym->sec == rela->sym->sec) {
|
if (rela->sym->sec && sym->sec == rela->sym->sec) {
|
||||||
sym->has_func_profiling = 1;
|
sym->has_func_profiling = 1;
|
||||||
goto next_symbol;
|
goto next_symbol;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -4287,12 +4238,6 @@ static error_t parse_opt (int key, char *arg, struct argp_state *state)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool has_multi_pfe(struct kpatch_elf *kelf)
|
|
||||||
{
|
|
||||||
return !!find_nth_section_by_name(&kelf->sections, 1,
|
|
||||||
"__patchable_function_entries");
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct argp argp = { options, parse_opt, args_doc, NULL };
|
static struct argp argp = { options, parse_opt, args_doc, NULL };
|
||||||
|
|
||||||
int main(int argc, char *argv[])
|
int main(int argc, char *argv[])
|
||||||
@ -4326,7 +4271,10 @@ int main(int argc, char *argv[])
|
|||||||
|
|
||||||
kelf_orig = kpatch_elf_open(orig_obj);
|
kelf_orig = kpatch_elf_open(orig_obj);
|
||||||
kelf_patched = kpatch_elf_open(patched_obj);
|
kelf_patched = kpatch_elf_open(patched_obj);
|
||||||
multi_pfe = has_multi_pfe(kelf_orig) || has_multi_pfe(kelf_patched);
|
|
||||||
|
kpatch_set_pfe_link(kelf_orig);
|
||||||
|
kpatch_set_pfe_link(kelf_patched);
|
||||||
|
|
||||||
kpatch_find_func_profiling_calls(kelf_orig);
|
kpatch_find_func_profiling_calls(kelf_orig);
|
||||||
kpatch_find_func_profiling_calls(kelf_patched);
|
kpatch_find_func_profiling_calls(kelf_patched);
|
||||||
|
|
||||||
@ -4388,9 +4336,6 @@ int main(int argc, char *argv[])
|
|||||||
/* this is destructive to kelf_patched */
|
/* this is destructive to kelf_patched */
|
||||||
kpatch_migrate_included_elements(kelf_patched, &kelf_out);
|
kpatch_migrate_included_elements(kelf_patched, &kelf_out);
|
||||||
|
|
||||||
/* this must be done before kelf_patched is torn down */
|
|
||||||
kpatch_alloc_mcount_sections(kelf_patched, kelf_out);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Teardown kelf_patched since we shouldn't access sections or symbols
|
* Teardown kelf_patched since we shouldn't access sections or symbols
|
||||||
* through it anymore. Don't free however, since our section and symbol
|
* through it anymore. Don't free however, since our section and symbol
|
||||||
@ -4409,7 +4354,7 @@ int main(int argc, char *argv[])
|
|||||||
kpatch_create_callbacks_objname_rela(kelf_out, parent_name);
|
kpatch_create_callbacks_objname_rela(kelf_out, parent_name);
|
||||||
kpatch_build_strings_section_data(kelf_out);
|
kpatch_build_strings_section_data(kelf_out);
|
||||||
|
|
||||||
kpatch_populate_mcount_sections(kelf_out);
|
kpatch_create_mcount_sections(kelf_out);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* At this point, the set of output sections and symbols is
|
* At this point, the set of output sections and symbols is
|
||||||
|
@ -685,6 +685,7 @@ void kpatch_dump_kelf(struct kpatch_elf *kelf)
|
|||||||
if (sec->rela)
|
if (sec->rela)
|
||||||
printf(", rela-> %s", sec->rela->name);
|
printf(", rela-> %s", sec->rela->name);
|
||||||
}
|
}
|
||||||
|
printf(", pfe-> [%d]", (sec->pfe) == NULL ? -1 : (int)sec->pfe->index);
|
||||||
next:
|
next:
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
@ -694,8 +695,10 @@ next:
|
|||||||
printf("sym %02d, type %d, bind %d, ndx %02d, name %s (%s)",
|
printf("sym %02d, type %d, bind %d, ndx %02d, name %s (%s)",
|
||||||
sym->index, sym->type, sym->bind, sym->sym.st_shndx,
|
sym->index, sym->type, sym->bind, sym->sym.st_shndx,
|
||||||
sym->name, status_str(sym->status));
|
sym->name, status_str(sym->status));
|
||||||
if (sym->sec && (sym->type == STT_FUNC || sym->type == STT_OBJECT))
|
if (sym->sec && (sym->type == STT_FUNC || sym->type == STT_OBJECT)) {
|
||||||
printf(" -> %s", sym->sec->name);
|
printf(" -> %s", sym->sec->name);
|
||||||
|
printf(", profiling: %d", sym->has_func_profiling);
|
||||||
|
}
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -964,6 +967,7 @@ struct section *create_section_pair(struct kpatch_elf *kelf, char *name,
|
|||||||
relasec->sh.sh_type = SHT_RELA;
|
relasec->sh.sh_type = SHT_RELA;
|
||||||
relasec->sh.sh_entsize = sizeof(GElf_Rela);
|
relasec->sh.sh_entsize = sizeof(GElf_Rela);
|
||||||
relasec->sh.sh_addralign = 8;
|
relasec->sh.sh_addralign = 8;
|
||||||
|
relasec->sh.sh_flags = SHF_INFO_LINK;
|
||||||
|
|
||||||
/* set text rela section pointer */
|
/* set text rela section pointer */
|
||||||
sec->rela = relasec;
|
sec->rela = relasec;
|
||||||
|
Loading…
Reference in New Issue
Block a user