Received: by 2002:a05:6358:16cc:b0:ea:6187:17c9 with SMTP id r12csp11593934rwl; Tue, 3 Jan 2023 01:29:53 -0800 (PST) X-Google-Smtp-Source: AMrXdXtKK/qXQZnnUtMABCVAnGnwMWeR+E7nqehgQIkz8toXEEXxa+AjJrrFoX2FxGOFa20S1p8n X-Received: by 2002:a17:90a:1a5e:b0:226:6a38:d2c2 with SMTP id 30-20020a17090a1a5e00b002266a38d2c2mr8720917pjl.10.1672738193665; Tue, 03 Jan 2023 01:29:53 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1672738193; cv=none; d=google.com; s=arc-20160816; b=dHlicJGKI/VDyR9TJ+PHcxqhG8KcPNaqXBl3CMqytDFQX9Sbl5VNmKDU8t9wLu2ahE qG8bSZOdQ90h1TXBVR47cN1mivjYYbINKcXSPa5rAkvd3uCXlUinRi8wQ6Ay9DyumuH3 hJ5OzvwNV7tixW6yiLG0WV0k7cYyzC0EJOQIKOuxCljlmSejNd7dJiKq/zoA9ESA6BuB ehDjyKJSy+Gxar0OjxG3iQPtGOfsFzYTEdfLRA3o+ZXAHN21m7y9mXxJyMbpTcxDayZ1 0My+27obtWpW7mQPjO43m0guCxe0vSMVaqHRpia/BArkCKNL4PUYp0lRpBnCZEXS831T 0uUQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-language:content-transfer-encoding :in-reply-to:mime-version:user-agent:date:message-id:from:references :cc:to:subject; bh=3ekLOmEq8IgMcNWNVe53PeQ+p1Wh6GYnrXlQ+MgAMKo=; b=z1iG7lc1JpanNp1T3l1KZcuBu0gCY8ax0mv2GzGLXT0MnPMfWqGGxILCiKpOllaZgI yUrD60KN1n+csrJGR1vmKe9Z9g1Ft7CUre3IdkAhx+4WC+QiVKCCwtq3sUMxa6OJ6Gbh CT40Li0YWpU82bFJpxI85KRvVuV4f6oiSmatlDVyZBDwsoGCKNJIWa1jV6ctFa/CHtA8 Wjrc5ASCre2+et3aKEwoi9xlRuLUqwQZHeeBQYruxfVtntTJsWKNo+Gj5B86XY1lyMge EpGmLUqKvPbVIeHdMXMyP9sSAHVbYugxFutp0wkdS12Vsg1Xxk/dUOGkVa7W94QkdM27 IpSw== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id me11-20020a17090b17cb00b0022656fb77f0si9501165pjb.55.2023.01.03.01.29.45; Tue, 03 Jan 2023 01:29:53 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S236943AbjACIlP (ORCPT + 61 others); Tue, 3 Jan 2023 03:41:15 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:56122 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S237097AbjACIlI (ORCPT ); Tue, 3 Jan 2023 03:41:08 -0500 Received: from loongson.cn (mail.loongson.cn [114.242.206.163]) by lindbergh.monkeyblade.net (Postfix) with ESMTP id 38FFE2C2 for ; Tue, 3 Jan 2023 00:41:01 -0800 (PST) Received: from loongson.cn (unknown [111.9.175.10]) by gateway (Coremail) with SMTP id _____8CxI_Ab6rNjxxYKAA--.22421S3; Tue, 03 Jan 2023 16:40:59 +0800 (CST) Received: from [10.136.12.26] (unknown [111.9.175.10]) by localhost.localdomain (Coremail) with SMTP id AQAAf8Axfb4Y6rNjn8sSAA--.33045S3; Tue, 03 Jan 2023 16:40:58 +0800 (CST) Subject: Re: [PATCH v2 4/6] LoongArch: Strip guess_unwinder out from prologue_unwinder To: Huacai Chen Cc: WANG Xuerui , Qing Zhang , loongarch@lists.linux.dev, linux-kernel@vger.kernel.org, Steven Rostedt , Masami Hiramatsu , Mark Rutland References: <20221229061516.31671-1-hejinyang@loongson.cn> <20221229061516.31671-5-hejinyang@loongson.cn> From: Jinyang He Message-ID: Date: Tue, 3 Jan 2023 16:40:56 +0800 User-Agent: Mozilla/5.0 (X11; Linux loongarch64; rv:68.0) Gecko/20100101 Thunderbird/68.7.0 MIME-Version: 1.0 In-Reply-To: Content-Type: text/plain; charset=utf-8; format=flowed Content-Transfer-Encoding: 8bit Content-Language: en-US X-CM-TRANSID: AQAAf8Axfb4Y6rNjn8sSAA--.33045S3 X-CM-SenderInfo: pkhmx0p1dqwqxorr0wxvrqhubq/ X-Coremail-Antispam: 1Uk129KBjvAXoW3tw48uw1xAFy5Gr17JrWrXwb_yoW8Wr1UXo W3tF1Sgr4xJ345J3yjy34UtFyYgr4jka1UAw4ftr13WF42y343ur4jqas8XFyIq3W5KrWU Cry2gF1rZan7Zwn3n29KB7ZKAUJUUUUr529EdanIXcx71UUUUU7KY7ZEXasCq-sGcSsGvf J3Ic02F40EFcxC0VAKzVAqx4xG6I80ebIjqfuFe4nvWSU5nxnvy29KBjDU0xBIdaVrnRJU UUBFb4IE77IF4wAFF20E14v26r1j6r4UM7CY07I20VC2zVCF04k26cxKx2IYs7xG6rWj6s 0DM7CIcVAFz4kK6r1Y6r17M28lY4IEw2IIxxk0rwA2F7IY1VAKz4vEj48ve4kI8wA2z4x0 Y4vE2Ix0cI8IcVAFwI0_Gr0_Xr1l84ACjcxK6xIIjxv20xvEc7CjxVAFwI0_Gr0_Cr1l84 ACjcxK6I8E87Iv67AKxVW8Jr0_Cr1UM28EF7xvwVC2z280aVCY1x0267AKxVW8Jr0_Cr1U M2kKe7AKxVWUXVWUAwAS0I0E0xvYzxvE52x082IY62kv0487Mc804VCY07AIYIkI8VC2zV CFFI0UMc02F40EFcxC0VAKzVAqx4xG6I80ewAv7VC0I7IYx2IY67AKxVWUXVWUAwAv7VC2 z280aVAFwI0_Jr0_Gr1lOx8S6xCaFVCjc4AY6r1j6r4UM4x0Y48IcVAKI48JMxk0xIA0c2 IEe2xFo4CEbIxvr21l42xK82IYc2Ij64vIr41l4I8I3I0E4IkC6x0Yz7v_Jr0_Gr1l4IxY O2xFxVAFwI0_Jrv_JF1lx2IqxVAqx4xG67AKxVWUJVWUGwC20s026x8GjcxK67AKxVWUGV WUWwC2zVAF1VAY17CE14v26r1q6r43MIIYrxkI7VAKI48JMIIF0xvE2Ix0cI8IcVAFwI0_ Jr0_JF4lIxAIcVC0I7IYx2IY6xkF7I0E14v26r1j6r4UMIIF0xvE42xK8VAvwI8IcIk0rV WUJVWUCwCI42IY6I8E87Iv67AKxVWUJVW8JwCI42IY6I8E87Iv6xkF7I0E14v26r1j6r4U YxBIdaVFxhVjvjDU0xZFpf9x07jepB-UUUUU= X-Spam-Status: No, score=-5.0 required=5.0 tests=BAYES_00,NICE_REPLY_A, SPF_HELO_PASS,SPF_PASS autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 2023-01-01 17:56, Huacai Chen wrote: > Hi, Jinyang, > > On Thu, Dec 29, 2022 at 2:15 PM Jinyang He wrote: >> The prolugue unwinder rely on symbol info. When PC is not in kernel >> text address, it cannot find relative symbol info and it will be broken. >> The guess unwinder will be used in this case. And the guess unwinder >> codes in prolugue unwinder is redundant. Strip it out and set the >> unwinder info in unwind_state. >> >> Signed-off-by: Jinyang He >> --- >> arch/loongarch/include/asm/unwind.h | 26 ++++- >> arch/loongarch/kernel/Makefile | 3 +- >> arch/loongarch/kernel/traps.c | 3 - >> arch/loongarch/kernel/unwind.c | 52 +++++++++ >> arch/loongarch/kernel/unwind_guess.c | 41 ++------ >> arch/loongarch/kernel/unwind_prologue.c | 133 +++++++++--------------- >> 6 files changed, 132 insertions(+), 126 deletions(-) >> create mode 100644 arch/loongarch/kernel/unwind.c >> >> diff --git a/arch/loongarch/include/asm/unwind.h b/arch/loongarch/include/asm/unwind.h >> index 6ece48f0ff77..4a55fd7b77ad 100644 >> --- a/arch/loongarch/include/asm/unwind.h >> +++ b/arch/loongarch/include/asm/unwind.h >> @@ -13,20 +13,30 @@ >> #include >> #include >> >> -enum unwinder_type { >> - UNWINDER_GUESS, >> - UNWINDER_PROLOGUE, >> -}; >> +struct unwinder_ops; >> >> struct unwind_state { >> - char type; /* UNWINDER_XXX */ >> struct stack_info stack_info; >> struct task_struct *task; >> bool first, error, is_ftrace; >> int graph_idx; >> unsigned long sp, pc, ra; >> + const struct unwinder_ops *ops; >> +}; >> + >> +struct unwinder_ops { >> + void (*unwind_start)(struct unwind_state *state, >> + struct task_struct *task, struct pt_regs *regs); >> + bool (*unwind_next_frame)(struct unwind_state *state); >> + unsigned long (*unwind_get_return_address)(struct unwind_state *state); >> }; >> >> +extern const struct unwinder_ops *default_unwinder; >> +extern const struct unwinder_ops unwinder_guess; >> +#ifdef CONFIG_UNWINDER_PROLOGUE >> +extern const struct unwinder_ops unwinder_prologue; >> +#endif >> + >> void unwind_start(struct unwind_state *state, >> struct task_struct *task, struct pt_regs *regs); >> bool unwind_next_frame(struct unwind_state *state); >> @@ -49,4 +59,10 @@ static inline unsigned long unwind_graph_addr(struct unwind_state *state, >> return ftrace_graph_ret_addr(state->task, &state->graph_idx, >> pc, (unsigned long *)(cfa - GRAPH_FAKE_OFFSET)); >> } >> + >> +static inline void unwind_register_unwinder(struct unwind_state *state, >> + const struct unwinder_ops *unwinder) >> +{ >> + state->ops = unwinder; >> +} >> #endif /* _ASM_UNWIND_H */ >> diff --git a/arch/loongarch/kernel/Makefile b/arch/loongarch/kernel/Makefile >> index 7ca65195f7f8..cb6029ea3ea9 100644 >> --- a/arch/loongarch/kernel/Makefile >> +++ b/arch/loongarch/kernel/Makefile >> @@ -8,7 +8,7 @@ extra-y := vmlinux.lds >> obj-y += head.o cpu-probe.o cacheinfo.o env.o setup.o entry.o genex.o \ >> traps.o irq.o idle.o process.o dma.o mem.o io.o reset.o switch.o \ >> elf.o syscall.o signal.o time.o topology.o inst.o ptrace.o vdso.o \ >> - alternative.o unaligned.o >> + alternative.o unaligned.o unwind.o unwind_guess.o >> >> obj-$(CONFIG_ACPI) += acpi.o >> obj-$(CONFIG_EFI) += efi.o >> @@ -42,7 +42,6 @@ obj-$(CONFIG_MAGIC_SYSRQ) += sysrq.o >> obj-$(CONFIG_KEXEC) += machine_kexec.o relocate_kernel.o >> obj-$(CONFIG_CRASH_DUMP) += crash_dump.o >> >> -obj-$(CONFIG_UNWINDER_GUESS) += unwind_guess.o >> obj-$(CONFIG_UNWINDER_PROLOGUE) += unwind_prologue.o > We do have CONFIG_UNWINDER_GUESS and I don't like to build guess > unwinder unconditionally, maybe we can refactor it to make the > unwinder still user selectable. In order to provide backtrace info to the user, prologue unwinder will use guess unwinder when its first pc is not a kernel address, such as the first PC is in ebpf generated codes. This patch is intended to extract guess unwinder codes to avoid redundant codes. And if we provide a new unwinder in the future like orc unwinder which take effect after orc_init is set, we can use another backup unwinder like guess unwinder in case we need backtrace before orc_init being set. If you mind this, could we just do as follows, obj-$(CONFIG_UNWINDER_GUESS)   += unwind_guess.o obj-$(CONFIG_UNWINDER_PROLOGUE) += unwind_prologue.o unwind_guess.o Or copy the guess unwinder codes to unwind_prologue.c and set another name for guess unwinder. Thanks, Jinyang > Huacai >> obj-$(CONFIG_PERF_EVENTS) += perf_event.o perf_regs.o >> diff --git a/arch/loongarch/kernel/traps.c b/arch/loongarch/kernel/traps.c >> index 3adc44832205..1ea14f6c18d3 100644 >> --- a/arch/loongarch/kernel/traps.c >> +++ b/arch/loongarch/kernel/traps.c >> @@ -72,9 +72,6 @@ static void show_backtrace(struct task_struct *task, const struct pt_regs *regs, >> if (!task) >> task = current; >> >> - if (user_mode(regs)) >> - state.type = UNWINDER_GUESS; >> - >> printk("%sCall Trace:", loglvl); >> for (unwind_start(&state, task, pregs); >> !unwind_done(&state); unwind_next_frame(&state)) { >> diff --git a/arch/loongarch/kernel/unwind.c b/arch/loongarch/kernel/unwind.c >> new file mode 100644 >> index 000000000000..24d2cf99bfb6 >> --- /dev/null >> +++ b/arch/loongarch/kernel/unwind.c >> @@ -0,0 +1,52 @@ >> +// SPDX-License-Identifier: GPL-2.0 >> +/* >> + * Copyright (C) 2022 Loongson Technology Corporation Limited >> + */ >> +#include >> + >> +#if defined(CONFIG_UNWINDER_GUESS) >> +const struct unwinder_ops *default_unwinder = &unwinder_guess; >> +#elif defined(CONFIG_UNWINDER_PROLOGUE) >> +const struct unwinder_ops *default_unwinder = &unwinder_prologue; >> +#endif >> + >> +unsigned long unwind_get_return_address(struct unwind_state *state) >> +{ >> + if (!state->ops || unwind_done(state)) >> + return 0; >> + return state->ops->unwind_get_return_address(state); >> +} >> +EXPORT_SYMBOL_GPL(unwind_get_return_address); >> + >> +void unwind_start(struct unwind_state *state, struct task_struct *task, >> + struct pt_regs *regs) >> +{ >> + memset(state, 0, sizeof(*state)); >> + unwind_register_unwinder(state, default_unwinder); >> + if (regs) { >> + state->sp = regs->regs[3]; >> + state->pc = regs->csr_era; >> + state->ra = regs->regs[1]; >> + } else if (task == current || task == NULL) { >> + state->sp = (unsigned long)__builtin_frame_address(0); >> + state->pc = (unsigned long)__builtin_return_address(0); >> + state->ra = 0; >> + } else { >> + state->sp = thread_saved_fp(task); >> + state->pc = thread_saved_ra(task); >> + state->ra = 0; >> + } >> + state->task = task; >> + get_stack_info(state->sp, state->task, &state->stack_info); >> + state->pc = unwind_graph_addr(state, state->pc, state->sp); >> + state->ops->unwind_start(state, task, regs); >> +} >> +EXPORT_SYMBOL_GPL(unwind_start); >> + >> +bool unwind_next_frame(struct unwind_state *state) >> +{ >> + if (!state->ops || unwind_done(state)) >> + return false; >> + return state->ops->unwind_next_frame(state); >> +} >> +EXPORT_SYMBOL_GPL(unwind_next_frame); >> diff --git a/arch/loongarch/kernel/unwind_guess.c b/arch/loongarch/kernel/unwind_guess.c >> index 935d24f8c95c..b7ca2b88ac63 100644 >> --- a/arch/loongarch/kernel/unwind_guess.c >> +++ b/arch/loongarch/kernel/unwind_guess.c >> @@ -7,51 +7,23 @@ >> >> #include >> >> -unsigned long unwind_get_return_address(struct unwind_state *state) >> +static unsigned long get_return_address(struct unwind_state *state) >> { >> - if (unwind_done(state)) >> - return 0; >> return state->pc; >> } >> -EXPORT_SYMBOL_GPL(unwind_get_return_address); >> >> -void unwind_start(struct unwind_state *state, struct task_struct *task, >> +static void start(struct unwind_state *state, struct task_struct *task, >> struct pt_regs *regs) >> { >> - memset(state, 0, sizeof(*state)); >> - >> - if (regs) { >> - state->sp = regs->regs[3]; >> - state->pc = regs->csr_era; >> - } else if (task == current || task == NULL) { >> - state->sp = (unsigned long)__builtin_frame_address(0); >> - state->pc = (unsigned long)__builtin_return_address(0); >> - } else { >> - state->sp = thread_saved_fp(task); >> - state->pc = thread_saved_ra(task); >> - } >> - >> - state->task = task; >> - state->first = true; >> - state->pc = unwind_graph_addr(state, state->pc, state->sp); >> - get_stack_info(state->sp, state->task, &state->stack_info); >> - >> if (!unwind_done(state) && !__kernel_text_address(state->pc)) >> unwind_next_frame(state); >> } >> -EXPORT_SYMBOL_GPL(unwind_start); >> >> -bool unwind_next_frame(struct unwind_state *state) >> +static bool next_frame(struct unwind_state *state) >> { >> struct stack_info *info = &state->stack_info; >> unsigned long addr; >> >> - if (unwind_done(state)) >> - return false; >> - >> - if (state->first) >> - state->first = false; >> - >> do { >> for (state->sp += sizeof(unsigned long); >> state->sp < info->end; >> @@ -68,4 +40,9 @@ bool unwind_next_frame(struct unwind_state *state) >> >> return false; >> } >> -EXPORT_SYMBOL_GPL(unwind_next_frame); >> + >> +const struct unwinder_ops unwinder_guess = { >> + .unwind_start = start, >> + .unwind_next_frame = next_frame, >> + .unwind_get_return_address = get_return_address, >> +}; >> diff --git a/arch/loongarch/kernel/unwind_prologue.c b/arch/loongarch/kernel/unwind_prologue.c >> index f7994ed05f04..beb57ea24da2 100644 >> --- a/arch/loongarch/kernel/unwind_prologue.c >> +++ b/arch/loongarch/kernel/unwind_prologue.c >> @@ -19,31 +19,19 @@ static inline void unwind_state_fixup(struct unwind_state *state) >> #endif >> } >> >> -unsigned long unwind_get_return_address(struct unwind_state *state) >> +static unsigned long get_return_address(struct unwind_state *state) >> { >> - if (unwind_done(state)) >> - return 0; >> return state->pc; >> } >> -EXPORT_SYMBOL_GPL(unwind_get_return_address); >> - >> -static bool unwind_by_guess(struct unwind_state *state) >> -{ >> - struct stack_info *info = &state->stack_info; >> - unsigned long addr; >> - >> - for (state->sp += sizeof(unsigned long); >> - state->sp < info->end; >> - state->sp += sizeof(unsigned long)) { >> - addr = *(unsigned long *)(state->sp); >> - state->pc = unwind_graph_addr(state, addr, state->sp + 8); >> - if (__kernel_text_address(state->pc)) >> - return true; >> - } >> - >> - return false; >> -} >> >> +/* >> + * LoongArch function prologue like follows, >> + * [others instructions not use stack var] >> + * addi.d sp, sp, -imm >> + * st.d xx, sp, offset <- save callee saved regs and >> + * st.d yy, sp, offset save ra if function is nest. >> + * [others instructions] >> + */ >> static bool unwind_by_prologue(struct unwind_state *state) >> { >> long frame_ra = -1; >> @@ -89,6 +77,10 @@ static bool unwind_by_prologue(struct unwind_state *state) >> ip++; >> } >> >> + /* >> + * Not find stack alloc action, PC may be in a leaf function. Only the >> + * first being true is reasonable, otherwise indicate analysis is broken. >> + */ >> if (!frame_size) { >> if (state->first) >> goto first; >> @@ -106,6 +98,7 @@ static bool unwind_by_prologue(struct unwind_state *state) >> ip++; >> } >> >> + /* Not find save $ra action, PC may be in a leaf function, too. */ >> if (frame_ra < 0) { >> if (state->first) { >> state->sp = state->sp + frame_size; >> @@ -114,96 +107,63 @@ static bool unwind_by_prologue(struct unwind_state *state) >> return false; >> } >> >> - if (state->first) >> - state->first = false; >> - >> state->pc = *(unsigned long *)(state->sp + frame_ra); >> state->sp = state->sp + frame_size; >> goto out; >> >> first: >> - state->first = false; >> - if (state->pc == state->ra) >> - return false; >> - >> state->pc = state->ra; >> >> out: >> + state->first = false; >> unwind_state_fixup(state); >> return !!__kernel_text_address(state->pc); >> } >> >> -void unwind_start(struct unwind_state *state, struct task_struct *task, >> +static void start(struct unwind_state *state, struct task_struct *task, >> struct pt_regs *regs) >> { >> - memset(state, 0, sizeof(*state)); >> - state->type = UNWINDER_PROLOGUE; >> - >> - if (regs) { >> - state->sp = regs->regs[3]; >> - state->pc = regs->csr_era; >> - state->ra = regs->regs[1]; >> - if (!__kernel_text_address(state->pc)) >> - state->type = UNWINDER_GUESS; >> - } else if (task == current || task == NULL) { >> - state->sp = (unsigned long)__builtin_frame_address(0); >> - state->pc = (unsigned long)__builtin_return_address(0); >> - state->ra = 0; >> - } else { >> - state->sp = thread_saved_fp(task); >> - state->pc = thread_saved_ra(task); >> - state->ra = 0; >> - } >> - >> - state->task = task; >> state->first = true; >> - state->pc = unwind_graph_addr(state, state->pc, state->sp); >> - get_stack_info(state->sp, state->task, &state->stack_info); >> >> - if (!unwind_done(state) && !__kernel_text_address(state->pc)) >> - unwind_next_frame(state); >> + /* >> + * The current PC is not kernel text address, we cannot find its >> + * relative symbol. Thus, prologue analysis will be broken. Luckly, >> + * we can use the guess unwinder. >> + */ >> + if (!__kernel_text_address(state->pc)) { >> + unwind_register_unwinder(state, &unwinder_guess); >> + if (!unwind_done(state)) >> + unwind_next_frame(state); >> + } >> } >> -EXPORT_SYMBOL_GPL(unwind_start); >> >> -bool unwind_next_frame(struct unwind_state *state) >> +static bool next_frame(struct unwind_state *state) >> { >> struct stack_info *info = &state->stack_info; >> struct pt_regs *regs; >> unsigned long pc; >> >> - if (unwind_done(state)) >> - return false; >> - >> do { >> - switch (state->type) { >> - case UNWINDER_GUESS: >> - state->first = false; >> - if (unwind_by_guess(state)) >> - return true; >> - break; >> - >> - case UNWINDER_PROLOGUE: >> - if (unwind_by_prologue(state)) { >> - state->pc = unwind_graph_addr(state, state->pc, state->sp); >> - return true; >> - } >> + if (unwind_by_prologue(state)) { >> + state->pc = unwind_graph_addr(state, state->pc, state->sp); >> + return true; >> + } >> >> - if (info->type == STACK_TYPE_IRQ && >> - info->end == state->sp) { >> - regs = (struct pt_regs *)info->next_sp; >> - pc = regs->csr_era; >> + if (info->type == STACK_TYPE_IRQ && >> + info->end == state->sp) { >> + regs = (struct pt_regs *)info->next_sp; >> + pc = regs->csr_era; >> >> - if (user_mode(regs) || !__kernel_text_address(pc)) >> - return false; >> + if (user_mode(regs) || !__kernel_text_address(pc)) >> + return false; >> >> - state->first = true; >> - state->ra = regs->regs[1]; >> - state->sp = regs->regs[3]; >> - state->pc = pc; >> - get_stack_info(state->sp, state->task, info); >> + state->first = true; >> + state->ra = regs->regs[1]; >> + state->sp = regs->regs[3]; >> + state->pc = pc; >> + get_stack_info(state->sp, state->task, info); >> >> - return true; >> - } >> + return true; >> } >> >> state->sp = info->next_sp; >> @@ -212,4 +172,9 @@ bool unwind_next_frame(struct unwind_state *state) >> >> return false; >> } >> -EXPORT_SYMBOL_GPL(unwind_next_frame); >> + >> +const struct unwinder_ops unwinder_prologue = { >> + .unwind_start = start, >> + .unwind_next_frame = next_frame, >> + .unwind_get_return_address = get_return_address, >> +}; >> -- >> 2.34.3 >> >>