mirror of
https://git.musl-libc.org/git/musl
synced 2025-01-23 14:42:07 +03:00
5db475f0b9
on all mips variants, Linux did (and maybe still does) have some syscall return paths that wrongly return both the error flag in r7 and a negated error code in r2. in particular this happened for at least some causes of ENOSYS. add an extra check to only negate the error code if it's positive to begin with. bug report and concept for patch by Andreas Dröscher.
129 lines
3.3 KiB
C
129 lines
3.3 KiB
C
#define __SYSCALL_LL_E(x) (x)
|
|
#define __SYSCALL_LL_O(x) (x)
|
|
|
|
#define SYSCALL_RLIM_INFINITY (-1UL/2)
|
|
|
|
#if __mips_isa_rev >= 6
|
|
#define SYSCALL_CLOBBERLIST \
|
|
"$1", "$3", "$10", "$11", "$12", "$13", \
|
|
"$14", "$15", "$24", "$25", "memory"
|
|
#else
|
|
#define SYSCALL_CLOBBERLIST \
|
|
"$1", "$3", "$10", "$11", "$12", "$13", \
|
|
"$14", "$15", "$24", "$25", "hi", "lo", "memory"
|
|
#endif
|
|
|
|
static inline long __syscall0(long n)
|
|
{
|
|
register long r7 __asm__("$7");
|
|
register long r2 __asm__("$2");
|
|
__asm__ __volatile__ (
|
|
"daddu $2,$0,%2 ; syscall"
|
|
: "=&r"(r2), "=r"(r7)
|
|
: "ir"(n), "0"(r2)
|
|
: SYSCALL_CLOBBERLIST);
|
|
return r7 && r2>0 ? -r2 : r2;
|
|
}
|
|
|
|
static inline long __syscall1(long n, long a)
|
|
{
|
|
register long r4 __asm__("$4") = a;
|
|
register long r7 __asm__("$7");
|
|
register long r2 __asm__("$2");
|
|
__asm__ __volatile__ (
|
|
"daddu $2,$0,%2 ; syscall"
|
|
: "=&r"(r2), "=r"(r7)
|
|
: "ir"(n), "0"(r2), "r"(r4)
|
|
: SYSCALL_CLOBBERLIST);
|
|
return r7 && r2>0 ? -r2 : r2;
|
|
}
|
|
|
|
static inline long __syscall2(long n, long a, long b)
|
|
{
|
|
register long r4 __asm__("$4") = a;
|
|
register long r5 __asm__("$5") = b;
|
|
register long r7 __asm__("$7");
|
|
register long r2 __asm__("$2");
|
|
|
|
__asm__ __volatile__ (
|
|
"daddu $2,$0,%2 ; syscall"
|
|
: "=&r"(r2), "=r"(r7)
|
|
: "ir"(n), "0"(r2), "r"(r4), "r"(r5)
|
|
: SYSCALL_CLOBBERLIST);
|
|
return r7 && r2>0 ? -r2 : r2;
|
|
}
|
|
|
|
static inline long __syscall3(long n, long a, long b, long c)
|
|
{
|
|
register long r4 __asm__("$4") = a;
|
|
register long r5 __asm__("$5") = b;
|
|
register long r6 __asm__("$6") = c;
|
|
register long r7 __asm__("$7");
|
|
register long r2 __asm__("$2");
|
|
|
|
__asm__ __volatile__ (
|
|
"daddu $2,$0,%2 ; syscall"
|
|
: "=&r"(r2), "=r"(r7)
|
|
: "ir"(n), "0"(r2), "r"(r4), "r"(r5), "r"(r6)
|
|
: SYSCALL_CLOBBERLIST);
|
|
return r7 && r2>0 ? -r2 : r2;
|
|
}
|
|
|
|
static inline long __syscall4(long n, long a, long b, long c, long d)
|
|
{
|
|
register long r4 __asm__("$4") = a;
|
|
register long r5 __asm__("$5") = b;
|
|
register long r6 __asm__("$6") = c;
|
|
register long r7 __asm__("$7") = d;
|
|
register long r2 __asm__("$2");
|
|
|
|
__asm__ __volatile__ (
|
|
"daddu $2,$0,%2 ; syscall"
|
|
: "=&r"(r2), "+r"(r7)
|
|
: "ir"(n), "0"(r2), "r"(r4), "r"(r5), "r"(r6)
|
|
: SYSCALL_CLOBBERLIST);
|
|
return r7 && r2>0 ? -r2 : r2;
|
|
}
|
|
|
|
static inline long __syscall5(long n, long a, long b, long c, long d, long e)
|
|
{
|
|
register long r4 __asm__("$4") = a;
|
|
register long r5 __asm__("$5") = b;
|
|
register long r6 __asm__("$6") = c;
|
|
register long r7 __asm__("$7") = d;
|
|
register long r8 __asm__("$8") = e;
|
|
register long r2 __asm__("$2");
|
|
|
|
__asm__ __volatile__ (
|
|
"daddu $2,$0,%2 ; syscall"
|
|
: "=&r"(r2), "+r"(r7)
|
|
: "ir"(n), "0"(r2), "r"(r4), "r"(r5), "r"(r6), "r"(r8)
|
|
: SYSCALL_CLOBBERLIST);
|
|
return r7 && r2>0 ? -r2 : r2;
|
|
}
|
|
|
|
static inline long __syscall6(long n, long a, long b, long c, long d, long e, long f)
|
|
{
|
|
register long r4 __asm__("$4") = a;
|
|
register long r5 __asm__("$5") = b;
|
|
register long r6 __asm__("$6") = c;
|
|
register long r7 __asm__("$7") = d;
|
|
register long r8 __asm__("$8") = e;
|
|
register long r9 __asm__("$9") = f;
|
|
register long r2 __asm__("$2");
|
|
|
|
__asm__ __volatile__ (
|
|
"daddu $2,$0,%2 ; syscall"
|
|
: "=&r"(r2), "+r"(r7)
|
|
: "ir"(n), "0"(r2), "r"(r4), "r"(r5), "r"(r6), "r"(r8), "r"(r9)
|
|
: SYSCALL_CLOBBERLIST);
|
|
return r7 && r2>0 ? -r2 : r2;
|
|
}
|
|
|
|
#define VDSO_USEFUL
|
|
#define VDSO_CGT_SYM "__vdso_clock_gettime"
|
|
#define VDSO_CGT_VER "LINUX_2.6"
|
|
|
|
#define SO_SNDTIMEO_OLD 0x1005
|
|
#define SO_RCVTIMEO_OLD 0x1006
|