summaryrefslogtreecommitdiff
path: root/libc/sysdeps/linux/nds32/sysdep.S
blob: 0ab325f268ff023316fc2781434ef800a119e5ae (plain)
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
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
/*
 * Copyright (C) 2016 Andes Technology, Inc.
 * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball.
 */

/* Copyright (C) 1991-2003 Free Software Foundation, Inc.

   The GNU C Library is free software; you can redistribute it and/or
   modify it under the terms of the GNU Lesser General Public
   License as published by the Free Software Foundation; either
   version 2.1 of the License, or (at your option) any later version.

   The GNU C Library is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
   Lesser General Public License for more details.

   You should have received a copy of the GNU Lesser General Public
   License along with the GNU C Library; if not, write to the Free
   Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
   02111-1307 USA.  */
 
#include <sysdep.h>
#define _ERRNO_H
#include <bits/errno.h>

.text

.globl __syscall_error

ENTRY (__syscall_error)
	subri $r0, $r0, #0

#define __syscall_error __syscall_error_1

#undef syscall_error
#ifdef NO_UNDERSCORES
__syscall_error:
#else
syscall_error:
#endif

#ifdef PIC
	/* set GP register */
	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
#endif

#if defined (EWOULDBLOCK_sys) && EWOULDBLOCK_sys != EAGAIN
	/* We translate the system's EWOULDBLOCK error into EAGAIN.
	   The GNU C library always defines EWOULDBLOCK==EAGAIN.
	   EWOULDBLOCK_sys is the original number.  */
	   push	$t0
           cfi_adjust_cfa_offset(4)
	   li	$t0, EWOULDBLOCK_sys
	   bne	$r0, $t0, 1f
	   pop	$t0
	   cfi_adjust_cfa_offset(-4)
	   li	$r0, EAGAIN
1:
#endif

#ifdef _LIBC_REENTRANT
	push	$lp
        cfi_adjust_cfa_offset(4)
        cfi_rel_offset(lp, 0)
	push	$r0
        cfi_adjust_cfa_offset(4)
        cfi_rel_offset(r0, 0)

#ifdef PIC
	bal	C_SYMBOL_NAME(__errno_location@PLT)
#else
	bal	C_SYMBOL_NAME(__errno_location)
#endif
	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)
#ifdef PIC
	/* restore GP register */
	popm	$gp, $lp
        cfi_adjust_cfa_offset(-8)
        cfi_restore(lp)
        cfi_restore(gp)
#endif
2:
	ret
#else
#ifndef	PIC
	l.w	$r1, .L1
	swi	$r0, [$r1]
	li	$r0, -1
	ret

.L1:	.long C_SYMBOL_NAME(errno)
#else
	s.w	$r0, errno@GOTOFF
	li	$r0, -1
	
	/* restore GP register */
	popm	$gp, $lp
        cfi_adjust_cfa_offset(-8)
        cfi_restore(lp)
        cfi_restore(gp)
2:
	ret

#endif
#endif

#undef __syscall_error
END (__syscall_error)