/* * Copyright (C) 2016-2017 Andes Technology, Inc. * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball. */ #ifndef _LINUX_NDS32_SYSDEP_H #define _LINUX_NDS32_SYSDEP_H 1 #include <common/sysdep.h> #ifdef __ASSEMBLER__ /* Define an entry point visible from C. */ # ifdef PIC # define ENTRY(name) \ .pic \ .align 2; \ .globl C_SYMBOL_NAME(name); \ .func C_SYMBOL_NAME(name); \ .type C_SYMBOL_NAME(name), @function; \ C_SYMBOL_NAME(name): \ cfi_startproc; # else # define ENTRY(name) \ .align 2; \ .globl C_SYMBOL_NAME(name); \ .func C_SYMBOL_NAME(name); \ .type C_SYMBOL_NAME(name), @function; \ C_SYMBOL_NAME(name): \ cfi_startproc; # endif #undef END #define END(name) \ cfi_endproc; \ .endfunc; \ .size C_SYMBOL_NAME(name), .-C_SYMBOL_NAME(name) /* If compiled for profiling, call `mcount' at the start of each function. */ # ifdef HAVE_ELF #undef NO_UNDERSCORES #define NO_UNDERSCORES # endif # ifdef NO_UNDERSCORES #define syscall_error __syscall_error # endif #define SYS_ify(syscall_name) (__NR_##syscall_name) #define __do_syscall(syscall_name) \ syscall SYS_ify(syscall_name); # ifdef PIC # define PSEUDO(name, syscall_name, args) \ .pic; \ .align 2; \ 99: mfusr $r15, $PC; \ sethi $r1, hi20(_GLOBAL_OFFSET_TABLE_ + 4); \ ori $r1, $r1, lo12(_GLOBAL_OFFSET_TABLE_ + 8); \ add $r1, $r15, $r1; \ sethi $r15, hi20(SYSCALL_ERROR@PLT); \ ori $r15, $r15, lo12(SYSCALL_ERROR@PLT); \ add $r15, $r15, $r1; \ jr $r15; \ nop; \ ENTRY(name); \ __do_syscall(syscall_name); \ bgez $r0, 2f; \ sltsi $r1, $r0, -4096; \ beqz $r1, 99b; \ 2: # else # define PSEUDO(name, syscall_name, args) \ .align 2; \ 99: j SYSCALL_ERROR; \ nop; \ ENTRY(name); \ __do_syscall(syscall_name); \ bgez $r0, 2f; \ sltsi $r1, $r0, -4096; \ beqz $r1, 99b; \ 2: # endif #define PSEUDO_NOERRNO(name, syscall_name, args) \ ENTRY(name); \ __do_syscall(syscall_name); #undef PSEUDO_END #define PSEUDO_END(sym) \ SYSCALL_ERROR_HANDLER \ END(sym) #undef PSEUDO_END_ERRVAL #define PSEUDO_END_ERRVAL(sym) END(sym) #define PSEUDO_ERRVAL(name, syscall_name, args) PSEUDO_NOERRNO(name, syscall_name, args) #define ret_ERRVAL ret #define ret_NOERRNO ret #if defined NOT_IN_libc #define SYSCALL_ERROR __local_syscall_error #ifdef PIC #define SYSCALL_ERROR_HANDLER \ __local_syscall_error: \ pushm $gp, $lp; \ cfi_adjust_cfa_offset(8) \ cfi_rel_offset(gp, 0) \ cfi_rel_offset(lp, 4) \ mfusr $r15, $PC; \ sethi $gp, hi20(_GLOBAL_OFFSET_TABLE_+4); \ ori $gp, $gp, lo12(_GLOBAL_OFFSET_TABLE_+8); \ add $gp, $gp, $r15; \ neg $r0, $r0; \ push $r0; \ cfi_adjust_cfa_offset(4) \ cfi_rel_offset(r0, 0) \ addi $sp, $sp, -4; \ bal C_SYMBOL_NAME(__errno_location@PLT); \ addi $sp, $sp, 4; \ pop $r1; \ cfi_adjust_cfa_offset(-4); \ cfi_restore(r1); \ swi $r1, [$r0]; \ li $r0, -1; \ popm $gp, $lp; \ cfi_adjust_cfa_offset(-8); \ cfi_restore(lp); \ cfi_restore(gp); \ 1: ret; #else #define SYSCALL_ERROR_HANDLER \ __local_syscall_error: \ push $lp; \ cfi_adjust_cfa_offset(4) \ cfi_rel_offset(lp, 0) \ neg $r0, $r0; \ push $r0; \ cfi_adjust_cfa_offset(4) \ cfi_rel_offset(r0, 0) \ addi $sp, $sp, -4; \ bal C_SYMBOL_NAME(__errno_location); \ addi $sp, $sp, 4; \ pop $r1; \ cfi_adjust_cfa_offset(-4); \ cfi_restore(r1); \ swi $r1, [$r0]; \ li $r0, -1; \ pop $lp; \ cfi_adjust_cfa_offset(-4); \ cfi_restore(lp); \ ret; #endif #else #define SYSCALL_ERROR_HANDLER #define SYSCALL_ERROR __syscall_error #endif #endif /* __ASSEMBLER__ */ #endif //_LINUX_NDS32_SYSDEP_H