Received: by 2002:a05:6358:d09b:b0:dc:cd0c:909e with SMTP id jc27csp3621387rwb; Fri, 16 Dec 2022 18:45:33 -0800 (PST) X-Google-Smtp-Source: AA0mqf5xZpiLS70c7udUmcbO0wdeSMQEsMPiVTOUwRGh0PAnZODjQsRAwhoa0PoP0MunrhkyxAtj X-Received: by 2002:a17:902:a711:b0:189:747e:97cc with SMTP id w17-20020a170902a71100b00189747e97ccmr32499031plq.26.1671245132931; Fri, 16 Dec 2022 18:45:32 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1671245132; cv=none; d=google.com; s=arc-20160816; b=GLv2GknYhKOfhbz9bOWOizs1kvYbhsv3wtyZX8ktKd2XwSpxjyT/A78/vYxNR+7jma JDkx0aMzOxErMcwYspYlC4x3oAZIu7/OY26q1sqYjOo8NdActckGUtCYg2atrICPz0iH cUjQUES2gwbRN5n96uVQBqP9b6C2JIsxMskQhJ9gyd3UkWSoT9sA+fELa5ZTwQEjfS5Y wpBA86NHfhivjyThIKW5w3ydpgbsEeAx0gfWnifogwgnz8j60rLRuDCBwm65loYGIM6S 31X6mBdz7ChINw4i4/GsT6SKloa15Eusm+dwWsoYJLIXzK5uAM3+OcvPdtNiLRkXLa70 OqbQ== 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=Oh2GgNG7uPS5EhSBbHiWxqMPGLm7aSbv2vUhchOyXKs=; b=z6jj6AfUOJPFc6jkJynz3nmiZB+deqQ5HG9tk8VkYWG9njXshSi4JZu1OITjj6yNOL Mj9R9wsB65R9h8GybFGeWCMaj+YHDinJ83eGR4IC+SzrHmh/A1bK3ZG3pzbhKZ4LxvfG REY0LuDvpQBvvI2cCqAcyb5Bse2WfSh69h4pqxmnI+8XXOLPBp/aOkJ2opHC6EGxO/gD xjd6By3oTfvpS8Ff8SyKkQ4o8V0iRBUkbtI98t5WX1iVkKNf0mmsz+tNHBQ43P+NvgNV bzTW1lWfeGCTfmum4puf7x2DNAwJeTiNWxnHZGqOY8fq+voY3dtqUJTL3Yzqb1x9HWag I7rw== 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 y12-20020a170902b48c00b00189788d1dedsi3878329plr.96.2022.12.16.18.45.23; Fri, 16 Dec 2022 18:45:32 -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 S230034AbiLQBrz (ORCPT + 68 others); Fri, 16 Dec 2022 20:47:55 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60354 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S230010AbiLQBrx (ORCPT ); Fri, 16 Dec 2022 20:47:53 -0500 Received: from loongson.cn (mail.loongson.cn [114.242.206.163]) by lindbergh.monkeyblade.net (Postfix) with ESMTP id D29A8D6F for ; Fri, 16 Dec 2022 17:47:49 -0800 (PST) Received: from loongson.cn (unknown [111.9.175.10]) by gateway (Coremail) with SMTP id _____8DxOenEH51joUgGAA--.11077S3; Sat, 17 Dec 2022 09:47:48 +0800 (CST) Received: from [10.136.12.26] (unknown [111.9.175.10]) by localhost.localdomain (Coremail) with SMTP id AQAAf8AxHuTBH51jHxECAA--.6930S3; Sat, 17 Dec 2022 09:47:47 +0800 (CST) Subject: Re: [PATCH 4/6] LoongArch: Strip guess_unwinder out from prologue_unwinder To: Qing Zhang , Huacai Chen , WANG Xuerui Cc: loongarch@lists.linux.dev, linux-kernel@vger.kernel.org, Steven Rostedt , Masami Hiramatsu , Mark Rutland References: <20221215040141.18610-1-hejinyang@loongson.cn> <20221215040141.18610-5-hejinyang@loongson.cn> <36c3560a-0a2d-4c3f-ecb6-0e7e71746917@loongson.cn> <6fdb3dc9-2b34-ce53-1ff8-6b23c1eaaa82@loongson.cn> From: Jinyang He Message-ID: Date: Sat, 17 Dec 2022 09:47:45 +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: <6fdb3dc9-2b34-ce53-1ff8-6b23c1eaaa82@loongson.cn> Content-Type: text/plain; charset=utf-8; format=flowed Content-Transfer-Encoding: 8bit Content-Language: en-US X-CM-TRANSID: AQAAf8AxHuTBH51jHxECAA--.6930S3 X-CM-SenderInfo: pkhmx0p1dqwqxorr0wxvrqhubq/ X-Coremail-Antispam: 1Uk129KBjvAXoW3ZF13Kr18Wr4Dtr1DGFy3Arb_yoW8WFWxto W7KF13Wr4rXrWUK3yUA34UJFy5tw4UJwnrArW5tr13Gr4Iya43Z3yUJa45tayxKr1rKr4U Gryjqr1FvFWxXr1fn29KB7ZKAUJUUUUr529EdanIXcx71UUUUU7KY7ZEXasCq-sGcSsGvf J3Ic02F40EFcxC0VAKzVAqx4xG6I80ebIjqfuFe4nvWSU5nxnvy29KBjDU0xBIdaVrnRJU UUBFb4IE77IF4wAFF20E14v26r1j6r4UM7CY07I20VC2zVCF04k26cxKx2IYs7xG6rWj6s 0DM7CIcVAFz4kK6r1Y6r17M28lY4IEw2IIxxk0rwA2F7IY1VAKz4vEj48ve4kI8wA2z4x0 Y4vE2Ix0cI8IcVAFwI0_Jr0_JF4l84ACjcxK6xIIjxv20xvEc7CjxVAFwI0_Jr0_Gr1l84 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=-1.9 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 2022-12-16 10:14, Qing Zhang wrote: > Hi, Jinyang > > On 2022/12/16 上午9:40, Jinyang He wrote: >> On 2022-12-15 17:15, Qing Zhang wrote: >> >>> Hi, Jinyang >>> >>> On 2022/12/15 下午12:01, 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     |  22 ++++ >>>>   arch/loongarch/kernel/Makefile          |   3 +- >>>>   arch/loongarch/kernel/unwind.c          |  52 +++++++++ >>>>   arch/loongarch/kernel/unwind_guess.c    |  41 ++----- >>>>   arch/loongarch/kernel/unwind_prologue.c | 135 >>>> +++++++++--------------- >>>>   5 files changed, 135 insertions(+), 118 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..a16aff1d086a 100644 >>>> --- a/arch/loongarch/include/asm/unwind.h >>>> +++ b/arch/loongarch/include/asm/unwind.h >>>> @@ -18,6 +18,8 @@ enum unwinder_type { >>>>       UNWINDER_PROLOGUE, >>>>   }; >>>>   +struct unwinder_ops; >>>> + >>>>   struct unwind_state { >>>>       char type; /* UNWINDER_XXX */ >>>>       struct stack_info stack_info; >>>> @@ -25,8 +27,22 @@ struct unwind_state { >>>>       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 +65,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 >>>>     obj-$(CONFIG_PERF_EVENTS)    += perf_event.o perf_regs.o >>>> diff --git a/arch/loongarch/kernel/unwind.c >>>> b/arch/loongarch/kernel/unwind.c >>>> new file mode 100644 >>>> index 000000000000..568c6fe707d1 >>>> --- /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) { >>>> +        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); > >  here. :) The PC from branch 'task == current' is the RA of current frame. And the PC from branch 'regs' is regs->csr_era, which may be RA of frame where to get the regs->csr_era. They all may traced by function_graph. Ftrace_graph_ret_addr() returns orignal addr and not destroys result, if the addr was not traced, so calling it for deal with above cases. Thanks, Jinyang > >>>> + 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 8ce32c37c587..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) { >>>> -        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_add(state, state->pc, state->sp); >>> >>> Do we need to unwind_graph_add again here? unwinder_guess and >>> unwind_prologue have already been done. >> >> Hi, Qing >> >> >> Sorry I don't what meanning here, as here is a delete line. >> > Sorry, It's the unwind_start up here. >> >>> >>>> -    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 d464c533c64f..9677e13c4b4c 100644 >>>> --- a/arch/loongarch/kernel/unwind_prologue.c >>>> +++ b/arch/loongarch/kernel/unwind_prologue.c >>>> @@ -9,6 +9,8 @@ >>>>   #include >>>>   #include >>>>   +static const struct unwinder_ops *guard_unwinder = &unwinder_guess; >>>> + >>>>   static inline void unwind_state_fixup(struct unwind_state *state) >>>>   { >>>>   #ifdef CONFIG_DYNAMIC_FTRACE >>>> @@ -19,31 +21,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 +79,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 +100,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 +109,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) { >>>> -        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 guard unwinder. >>>> +     */ >>>> +    if (!__kernel_text_address(state->pc)) { >>>> +        unwind_register_unwinder(state, guard_unwinder); >>> >>> Just add a comment here. Instead of using guard_unwinder, can we still >>> use guess_unwinder? >> >> Yes, and I'll use '&guess_unwinder' in next version. And I'll >> drop unwind type in next version, too. >> >> >> Thanks, >> >> Jinyang >>