/* * Copyright (c) International Business Machines Corp., 2002 * Copyright (c) 2012 Cyril Hrubis <chrubis@suse.cz> * * 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. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ /* * This test verifies that flock cannot unlock a file locked * by another task * * Test Steps: * * Fork a child processes The parent flocks a file with LOCK_EX Child waits * for that to happen, then checks to make sure it is locked. Child then * tries to unlock the file. If the unlock succeeds, the child attempts to * lock the file with LOCK_EX. The test passes if the child is able to lock * the file. */ #include <stdio.h> #include <errno.h> #include <fcntl.h> #include <sys/types.h> #include <sys/stat.h> #include <sys/file.h> #include <sys/wait.h> #include "test.h" #define FILE_NAME "flock03" static void setup(void); static void cleanup(void); static void childfunc(int); #ifdef UCLINUX static int fd_uc; static void childfunc_uc(void) { childfunc(fd_uc); } #endif char *TCID = "flock03"; int TST_TOTAL = 3; int main(int argc, char **argv) { int lc; pid_t pid; int status; int fd; tst_parse_opts(argc, argv, NULL, NULL); #ifdef UCLINUX maybe_run_child(&childfunc_uc, "ds", &fd_uc, FILE_NAME); #endif setup(); for (lc = 0; TEST_LOOPING(lc); lc++) { tst_count = 0; fd = open(FILE_NAME, O_RDWR); if (fd == -1) tst_brkm(TFAIL | TERRNO, cleanup, "parent failed to open the file"); pid = FORK_OR_VFORK(); if (pid == -1) tst_brkm(TFAIL | TERRNO, cleanup, "fork() failed"); if (pid == 0) { #ifdef UCLINUX if (self_exec(argv[0], "ds", fd, FILE_NAME) < 0) tst_brkm(TFAIL | TERRNO, cleanup, "self_exec failed"); #else childfunc(fd); #endif } TEST(flock(fd, LOCK_EX | LOCK_NB)); if (TEST_RETURN != 0) tst_resm(TFAIL | TTERRNO, "Parent: Initial attempt to flock() failed"); else tst_resm(TPASS, "Parent: Initial attempt to flock() passed"); TST_SAFE_CHECKPOINT_WAKE(cleanup, 0); if ((waitpid(pid, &status, 0)) < 0) { tst_resm(TFAIL, "wait() failed"); continue; } if ((WIFEXITED(status)) && (WEXITSTATUS(status) == 0)) tst_resm(TPASS, "flock03 Passed"); else tst_resm(TFAIL, "flock03 Failed"); close(fd); } cleanup(); tst_exit(); } static void childfunc(int fd) { int fd2; TST_SAFE_CHECKPOINT_WAIT(NULL, 0); fd2 = open(FILE_NAME, O_RDWR); if (fd2 == -1) { fprintf(stderr, "CHILD: failed to open the file: %s\n", strerror(errno)); exit(1); } if (flock(fd2, LOCK_EX | LOCK_NB) != -1) { fprintf(stderr, "CHILD: The file was not already locked\n"); exit(1); } TEST(flock(fd, LOCK_UN)); /* XXX: LOCK_UN does not return an error if there was nothing to * unlock. */ if (TEST_RETURN == -1) { fprintf(stderr, "CHILD: Unable to unlock file locked by " "parent: %s\n", strerror(TEST_ERRNO)); exit(1); } else { fprintf(stderr, "CHILD: File locked by parent unlocked\n"); } TEST(flock(fd2, LOCK_EX | LOCK_NB)); if (TEST_RETURN == -1) { fprintf(stderr, "CHILD: Unable to lock file after " "unlocking: %s\n", strerror(TEST_ERRNO)); exit(1); } else { fprintf(stderr, "CHILD: Locking after unlock passed\n"); } close(fd); close(fd2); exit(0); } static void setup(void) { int fd; tst_sig(FORK, DEF_HANDLER, cleanup); TEST_PAUSE; tst_tmpdir(); TST_CHECKPOINT_INIT(tst_rmdir); fd = open(FILE_NAME, O_CREAT | O_TRUNC | O_RDWR, 0666); if (fd < 0) { tst_resm(TBROK, "creating a new file failed"); cleanup(); } close(fd); } static void cleanup(void) { tst_rmdir(); }