only use getrlimit/setrlimit syscalls if they exist
[musl] / src / misc / setrlimit.c
index 8a1b8cc..5b713cf 100644 (file)
@@ -6,45 +6,48 @@
 #define MIN(a, b) ((a)<(b) ? (a) : (b))
 #define FIX(x) do{ if ((x)>=SYSCALL_RLIM_INFINITY) (x)=RLIM_INFINITY; }while(0)
 
-int __setrlimit(int resource, const struct rlimit *rlim)
-{
-       unsigned long k_rlim[2];
-       struct rlimit tmp;
-       if (SYSCALL_RLIM_INFINITY != RLIM_INFINITY) {
-               tmp = *rlim;
-               FIX(tmp.rlim_cur);
-               FIX(tmp.rlim_max);
-               rlim = &tmp;
-       }
-       int ret = __syscall(SYS_prlimit64, 0, resource, rlim, 0);
-       if (ret != -ENOSYS) return ret;
-       k_rlim[0] = MIN(rlim->rlim_cur, MIN(-1UL, SYSCALL_RLIM_INFINITY));
-       k_rlim[1] = MIN(rlim->rlim_max, MIN(-1UL, SYSCALL_RLIM_INFINITY));
-       return __syscall(SYS_setrlimit, resource, k_rlim);
-}
-
 struct ctx {
-       const struct rlimit *rlim;
+       unsigned long lim[2];
        int res;
        int err;
 };
 
+#ifdef SYS_setrlimit
 static void do_setrlimit(void *p)
 {
        struct ctx *c = p;
-       if (c->err) return;
-       c->err = -__setrlimit(c->res, c->rlim);
+       if (c->err>0) return;
+       c->err = -__syscall(SYS_setrlimit, c->res, c->lim);
 }
+#endif
 
 int setrlimit(int resource, const struct rlimit *rlim)
 {
-       struct ctx c = { .res = resource, .rlim = rlim };
+       struct rlimit tmp;
+       if (SYSCALL_RLIM_INFINITY != RLIM_INFINITY) {
+               tmp = *rlim;
+               FIX(tmp.rlim_cur);
+               FIX(tmp.rlim_max);
+               rlim = &tmp;
+       }
+       int ret = __syscall(SYS_prlimit64, 0, resource, rlim, 0);
+#ifdef SYS_setrlimit
+       if (ret != -ENOSYS) return __syscall_ret(ret);
+
+       struct ctx c = {
+               .lim[0] = MIN(rlim->rlim_cur, MIN(-1UL, SYSCALL_RLIM_INFINITY)),
+               .lim[1] = MIN(rlim->rlim_max, MIN(-1UL, SYSCALL_RLIM_INFINITY)),
+               .res = resource, .err = -1
+       };
        __synccall(do_setrlimit, &c);
        if (c.err) {
-               errno = c.err;
+               if (c.err>0) errno = c.err;
                return -1;
        }
        return 0;
+#else
+       return __syscall_ret(ret);
+#endif
 }
 
-LFS64(setrlimit);
+weak_alias(setrlimit, setrlimit64);