riscv64: fix linking with binutils 2.40
[uclibc-ng.git] / libc / sysdeps / linux / arm / mmap64.S
blobc9f2bd2f7e90679193040f7080f50e36ffb82699
1 /* Copyright (C) 2000 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
4    The GNU C Library is free software; you can redistribute it and/or
5    modify it under the terms of the GNU Lesser General Public
6    License as published by the Free Software Foundation; either
7    version 2.1 of the License, or (at your option) any later version.
9    The GNU C Library is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12    Lesser General Public License for more details.
14    You should have received a copy of the GNU Lesser General Public
15    License along with the GNU C Library; if not, see
16    <http://www.gnu.org/licenses/>.  */
18 #include <_lfs_64.h>
19 #define _ERRNO_H
20 #include <bits/errno.h>
21 #include <sys/syscall.h>
22 #include <bits/arm_asm.h>
23 #include <bits/arm_bx.h>
25 #ifdef __NR_mmap2
27 /* The mmap2 system call takes six arguments, all in registers.  */
28 .text
29 .global mmap64
30 .type mmap64,%function
31 .align 2
33 #ifdef __ARM_EABI__
34 #if defined(THUMB1_ONLY)
35 .thumb_func
36 mmap64:
37 #ifdef __ARMEB__
38 /* Offsets are after pushing 3 words.  */
39 # define LOW_OFFSET  12 + 8 + 4
40 # define HIGH_OFFSET 12 + 8 + 0
41 #else
42 # define LOW_OFFSET  12 + 8 + 0
43 # define HIGH_OFFSET 12 + 8 + 4
44 #endif
45         push    {r4, r5, r6}
46         ldr     r6, [sp, $LOW_OFFSET]
47         ldr     r5, [sp, $HIGH_OFFSET]
48         lsl     r4, r6, #20             @ check that offset is page-aligned
49         bne     .Linval
50         lsr     r4, r5, #12             @ check for overflow
51         bne     .Linval
52         @ compose page offset
53         lsr     r6, r6, #12
54         lsl     r5, r5, #20
55         orr     r5, r5, r6
56         ldr     r4, [sp, #8]            @ load fd
57         DO_CALL (mmap2)
58         ldr     r1, =0xfffff000
59         cmp     r0, r1
60         bcs     .Lerror
61         bx      lr
62 .Linval:
63         ldr     r0, =-EINVAL
64         pop     {r4, r5, r6}
65 .Lerror:
66         push    {r3, lr}
67         bl      __syscall_error
68         POP_RET
69 .pool
70 #else /* !THUMB1_ONLY */
71 mmap64:
72 #ifdef __ARMEB__
73 # define LOW_OFFSET      8 + 4
74 /* The initial + 4 is for the stack postdecrement.  */
75 # define HIGH_OFFSET 4 + 8 + 0
76 #else
77 # define LOW_OFFSET      8 + 0
78 # define HIGH_OFFSET 4 + 8 + 4
79 #endif
80         ldr     ip, [sp, $LOW_OFFSET]
81         str     r5, [sp, #-4]!
82         ldr     r5, [sp, $HIGH_OFFSET]
83         str     r4, [sp, #-4]!
84         movs    r4, ip, lsl $20         @ check that offset is page-aligned
85         mov     ip, ip, lsr $12
86         IT(t, eq)
87         moveqs  r4, r5, lsr $12         @ check for overflow
88         bne     .Linval
89         ldr     r4, [sp, $8]            @ load fd
90         orr     r5, ip, r5, lsl $20     @ compose page offset
91         DO_CALL (mmap2)
92         cmn     r0, $4096
93         ldmfd   sp!, {r4, r5}
94         IT(t, cc)
95         BXC(cc, lr)
96         b       __syscall_error
97 .Linval:
98         mov     r0, $-EINVAL
99         ldmfd   sp!, {r4, r5}
100         b       __syscall_error
101 #endif
102 #else /* !__ARM_EABI__ */
103 mmap64:
104         stmfd   sp!, {r4, r5, lr}
105         ldr     r5, [sp, $16]
106         ldr     r4, [sp, $12]
107         movs    ip, r5, lsl $20         @ check that offset is page-aligned
108         bne     .Linval
109         ldr     ip, [sp, $20]
110         mov     r5, r5, lsr $12
111         orr     r5, r5, ip, lsl $20     @ compose page offset
112         movs    ip, ip, lsr $12
113         bne     .Linval                 @ check for overflow
114         mov     ip, r0
115         DO_CALL (mmap2)
116         cmn     r0, $4096
117         ldmccfd sp!, {r4, r5, pc}
118         cmn     r0, $ENOSYS
119         ldmnefd sp!, {r4, r5, lr}
120         bne     __error
121         /* The current kernel does not support mmap2.  Fall back to plain
122            mmap if the offset is small enough.  */
123         ldr     r5, [sp, $20]
124         mov     r0, ip                  @ first arg was clobbered
125         teq     r5, $0
126         ldmeqfd sp!, {r4, r5, lr}
127         beq     HIDDEN_JUMPTARGET(mmap)
128 .Linval:
129         mov     r0, $-EINVAL
130         ldmfd   sp!, {r4, r5, lr}
131         b       __error
133 __error:
134         b       __syscall_error
135 #endif
136 .size mmap64,.-mmap64
138 #endif