/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│ │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│ ╞══════════════════════════════════════════════════════════════════════════════╡ │ Copyright 2021 Justine Alexandra Roberts Tunney │ │ │ │ Permission to use, copy, modify, and/or distribute this software for │ │ any purpose with or without fee is hereby granted, provided that the │ │ above copyright notice and this permission notice appear in all copies. │ │ │ │ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │ │ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │ │ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │ │ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │ │ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │ │ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │ │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │ │ PERFORMANCE OF THIS SOFTWARE. │ ╚─────────────────────────────────────────────────────────────────────────────*/ #include "libc/calls/calls.h" #include "libc/dce.h" #include "libc/errno.h" #include "libc/runtime/runtime.h" #include "libc/sysv/consts/at.h" #include "libc/testlib/testlib.h" #include "libc/x/x.h" void SetUpOnce(void) { testlib_enable_tmp_setup_teardown(); ASSERT_SYS(0, 0, pledge("stdio rpath wpath cpath fattr", 0)); } TEST(rename, enoent) { EXPECT_SYS(ENOENT, -1, rename("foo", "")); EXPECT_SYS(ENOENT, -1, rename("", "foo")); EXPECT_SYS(ENOENT, -1, rename("foo", "o/bar")); EXPECT_SYS(ENOENT, -1, rename("o/bar", "foo")); } TEST(renameat, enotdir) { EXPECT_SYS(0, 0, close(creat("yo", 0644))); EXPECT_SYS(ENOTDIR, -1, rename("yo/there", "hrcue")); // this test makes platforms crazy // EXPECT_SYS(ENOTDIR, -1, rename("zoo", "yo/there")); } TEST(rename, eisdir) { // new is a directory but old is not a directory ASSERT_SYS(0, 0, mkdir("foo", 0755)); ASSERT_SYS(0, 0, touch("foo/bar", 0644)); ASSERT_SYS(0, 0, touch("lol", 0644)); ASSERT_SYS(EISDIR, -1, rename("lol", "foo")); } TEST(rename, enotdir) { // old is a directory but new is not a directory ASSERT_SYS(0, 0, mkdir("lol", 0755)); ASSERT_SYS(0, 0, touch("foo", 0644)); ASSERT_SYS(ENOTDIR, -1, rename("lol", "foo")); } TEST(rename, eisdir_again) { // old is a directory but new is not a directory ASSERT_SYS(0, 0, touch("foo", 0644)); ASSERT_SYS(0, 0, touch("bar", 0644)); ASSERT_SYS(ENOTDIR, -1, rename("foo/", "bar")); ASSERT_SYS(ENOTDIR, -1, rename("foo", "bar/")); } TEST(rename, moveDirectoryOverDirectory_replacesOldDirectory) { ASSERT_SYS(0, 0, mkdir("foo//", 0755)); ASSERT_SYS(0, 0, mkdir("lol", 0755)); ASSERT_SYS(0, 0, rename("lol", "foo")); ASSERT_TRUE(fileexists("foo")); ASSERT_FALSE(fileexists("lol")); } TEST(rename, enotempty) { // POSIX specifies EEXIST or ENOTEMPTY when new path is non-empty dir. // Old version of Linux (e.g. RHEL7) return EEXIST. // Everything else returns ENOTEMPTY. int rc; ASSERT_SYS(0, 0, mkdir("foo", 0755)); ASSERT_SYS(0, 0, touch("foo/bar", 0644)); ASSERT_SYS(0, 0, mkdir("lol", 0755)); ASSERT_EQ(-1, (rc = rename("lol", "foo"))); ASSERT_TRUE(errno == ENOTEMPTY || errno == EEXIST); errno = 0; } TEST(rename, moveIntoNonWritableDirectory_raisesEacces) { // old versions of linux allow this // new versions of linux report exdev?! if (IsLinux()) return; // netbsd and openbsd allow this if (IsNetbsd() || IsOpenbsd()) return; // windows doesn't really have permissions if (IsWindows()) return; // posix specifies this behavior ASSERT_SYS(0, 0, mkdir("foo", 0111)); ASSERT_SYS(0, 0, touch("lol", 0644)); ASSERT_SYS(EACCES, -1, rename("lol", "foo/bar")); } TEST(renameat, testNull_returnsEfault) { ASSERT_SYS(0, 0, close(creat("hello", 0644))); ASSERT_SYS(EFAULT, -1, renameat(AT_FDCWD, 0, AT_FDCWD, 0)); ASSERT_SYS(EFAULT, -1, renameat(AT_FDCWD, 0, AT_FDCWD, "hello")); ASSERT_SYS(EFAULT, -1, renameat(AT_FDCWD, "hello", AT_FDCWD, 0)); } TEST(renameat, test) { ASSERT_SYS(0, 0, close(creat("first", 0644))); ASSERT_TRUE(fileexists("first")); ASSERT_TRUE(!fileexists("second")); ASSERT_SYS(0, 0, renameat(AT_FDCWD, "first", AT_FDCWD, "second")); ASSERT_TRUE(!fileexists("first")); ASSERT_TRUE(fileexists("second")); }