/* * Copyright (c) 2017 FUJITSU LIMITED. All rights reserved. * Author: Xiao Yang <yangx.jy@cn.fujitsu.com> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. */ /* * Test for EFAULT when rlim points outside the accessible address space. */ #define _GNU_SOURCE #include <errno.h> #include <sys/resource.h> #include <sys/time.h> #include <sys/wait.h> #include <stdlib.h> #include "tst_test.h" static void verify_setrlimit(void) { int status; pid_t pid; pid = SAFE_FORK(); if (!pid) { TEST(setrlimit(RLIMIT_NOFILE, (void *) -1)); if (TEST_RETURN != -1) { tst_res(TFAIL, "setrlimit() succeeded unexpectedly"); exit(0); } /* Usually, setrlimit() should return EFAULT */ if (TEST_ERRNO == EFAULT) { tst_res(TPASS | TTERRNO, "setrlimit() failed as expected"); } else { tst_res(TFAIL | TTERRNO, "setrlimit() should fail with EFAULT, got"); } exit(0); } SAFE_WAITPID(pid, &status, 0); /* If glibc has to convert between 32bit and 64bit struct rlimit * in some cases, it is possible to get SegFault. */ if (WIFSIGNALED(status) && WTERMSIG(status) == SIGSEGV) { tst_res(TPASS, "setrlimit() caused SIGSEGV"); return; } if (WIFEXITED(status) && WEXITSTATUS(status) == 0) return; tst_res(TBROK, "child %s", tst_strstatus(status)); } static struct tst_test test = { .test_all = verify_setrlimit, .forks_child = 1, };