root/arch/loongarch/include/asm/linkage.h
/* SPDX-License-Identifier: GPL-2.0 */
#ifndef __ASM_LINKAGE_H
#define __ASM_LINKAGE_H

#define __ALIGN         .align 2
#define __ALIGN_STR     __stringify(__ALIGN)

#define SYM_FUNC_START(name)                            \
        SYM_START(name, SYM_L_GLOBAL, SYM_A_ALIGN)      \
        .cfi_startproc;

#define SYM_FUNC_START_NOALIGN(name)                    \
        SYM_START(name, SYM_L_GLOBAL, SYM_A_NONE)       \
        .cfi_startproc;

#define SYM_FUNC_START_LOCAL(name)                      \
        SYM_START(name, SYM_L_LOCAL, SYM_A_ALIGN)       \
        .cfi_startproc;

#define SYM_FUNC_START_LOCAL_NOALIGN(name)              \
        SYM_START(name, SYM_L_LOCAL, SYM_A_NONE)        \
        .cfi_startproc;

#define SYM_FUNC_START_WEAK(name)                       \
        SYM_START(name, SYM_L_WEAK, SYM_A_ALIGN)        \
        .cfi_startproc;

#define SYM_FUNC_START_WEAK_NOALIGN(name)               \
        SYM_START(name, SYM_L_WEAK, SYM_A_NONE)         \
        .cfi_startproc;

#define SYM_FUNC_END(name)                              \
        .cfi_endproc;                                   \
        SYM_END(name, SYM_T_FUNC)

#define SYM_CODE_START(name)                            \
        SYM_START(name, SYM_L_GLOBAL, SYM_A_ALIGN)      \
        .cfi_startproc;

#define SYM_CODE_END(name)                              \
        .cfi_endproc;                                   \
        SYM_END(name, SYM_T_NONE)

/*
 * This is for the signal handler trampoline, which is used as the return
 * address of the signal handlers in userspace instead of called normally.
 * The long standing libgcc bug https://gcc.gnu.org/PR124050 requires a
 * nop between .cfi_startproc and the actual address of the trampoline, so
 * we cannot simply use SYM_FUNC_START.
 *
 * This wrapper also contains all the .cfi_* directives for recovering
 * the content of the GPRs and the "return address" (where the rt_sigreturn
 * syscall will jump to), assuming there is a struct rt_sigframe (where
 * a struct sigcontext containing those information we need to recover) at
 * $sp.  The "DWARF for the LoongArch(TM) Architecture" manual states
 * column 0 is for $zero, but it does not make too much sense to
 * save/restore the hardware zero register.  Repurpose this column here
 * for the return address (here it's not the content of $ra we cannot use
 * the default column 3).
 */
#define SYM_SIGFUNC_START(name)                         \
        .cfi_startproc;                                 \
        .cfi_signal_frame;                              \
        .cfi_def_cfa 3, RT_SIGFRAME_SC;                 \
        .cfi_return_column 0;                           \
        .cfi_offset 0, SC_PC;                           \
                                                        \
        .irp num, 1,  2,  3,  4,  5,  6,  7,  8,        \
                  9,  10, 11, 12, 13, 14, 15, 16,       \
                  17, 18, 19, 20, 21, 22, 23, 24,       \
                  25, 26, 27, 28, 29, 30, 31;           \
        .cfi_offset \num, SC_REGS + \num * SZREG;       \
        .endr;                                          \
                                                        \
        nop;                                            \
        SYM_START(name, SYM_L_GLOBAL, SYM_A_ALIGN)

#define SYM_SIGFUNC_END(name) SYM_FUNC_END(name)

#endif