/*-*- 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 2020 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/calls/struct/dirent.h" #include "libc/calls/struct/stat.h" #include "libc/dce.h" #include "libc/errno.h" #include "libc/fmt/conv.h" #include "libc/fmt/fmt.h" #include "libc/fmt/itoa.h" #include "libc/intrin/kprintf.h" #include "libc/mem/critbit0.h" #include "libc/mem/gc.h" #include "libc/mem/gc.internal.h" #include "libc/mem/mem.h" #include "libc/runtime/runtime.h" #include "libc/stdio/append.h" #include "libc/stdio/ftw.h" #include "libc/stdio/rand.h" #include "libc/str/str.h" #include "libc/sysv/consts/dt.h" #include "libc/sysv/consts/o.h" #include "libc/sysv/consts/s.h" #include "libc/testlib/testlib.h" #include "libc/x/xasprintf.h" #include "libc/x/xiso8601.h" __static_yoink("zipos"); __static_yoink("usr/share/zoneinfo/"); __static_yoink("usr/share/zoneinfo/New_York"); __static_yoink("libc/testlib/hyperion.txt"); __static_yoink("libc/testlib/moby.txt"); char testlib_enable_tmp_setup_teardown; DIR *dir; struct dirent *ent; void SetUp(void) { dir = 0; ent = 0; } TEST(opendir, efault) { ASSERT_SYS(EFAULT, NULL, opendir(0)); if (!IsAsan()) return; // not possible ASSERT_SYS(EFAULT, NULL, opendir((void *)77)); } TEST(opendir, enoent) { ASSERT_SYS(ENOENT, NULL, opendir("")); ASSERT_SYS(ENOENT, NULL, opendir("o/foo")); } TEST(opendir, enotdir) { ASSERT_SYS(0, 0, close(creat("yo", 0644))); ASSERT_SYS(ENOTDIR, NULL, opendir("yo/there")); } TEST(opendir, openSyntheticDirEntry) { struct stat st; ASSERT_SYS(0, 3, open("/zip", O_RDONLY | O_DIRECTORY)); ASSERT_SYS(0, 0, fstat(3, &st)); ASSERT_TRUE(S_ISDIR(st.st_mode)); EXPECT_SYS(EISDIR, -1, read(3, 0, 0)); ASSERT_NE(NULL, (dir = fdopendir(3))); EXPECT_NE(NULL, (ent = readdir(dir))); EXPECT_EQ(0, closedir(dir)); } TEST(opendir, openRealDirEntry) { struct stat st; ASSERT_SYS(0, 3, open("/zip/usr/share/zoneinfo", O_RDONLY | O_DIRECTORY)); ASSERT_SYS(0, 0, fstat(3, &st)); ASSERT_TRUE(S_ISDIR(st.st_mode)); EXPECT_SYS(EISDIR, -1, read(3, 0, 0)); ASSERT_NE(NULL, (dir = fdopendir(3))); EXPECT_NE(NULL, (ent = readdir(dir))); EXPECT_EQ(0, closedir(dir)); } TEST(dirstream, testDots) { int hasdot = 0; int hasdotdot = 0; ASSERT_SYS(0, 0, close(creat("foo", 0644))); ASSERT_NE(NULL, (dir = opendir("."))); while ((ent = readdir(dir))) { if (!strcmp(ent->d_name, ".")) { ++hasdot; EXPECT_EQ(DT_DIR, ent->d_type); } if (!strcmp(ent->d_name, "..")) { ++hasdotdot; EXPECT_EQ(DT_DIR, ent->d_type); } } EXPECT_EQ(1, hasdot); EXPECT_EQ(1, hasdotdot); EXPECT_SYS(0, 0, closedir(dir)); } TEST(dirstream, test) { bool hasfoo = false; bool hasbar = false; char *dpath, *file1, *file2; dpath = gc(xasprintf("%s.%d", "dirstream", rand())); file1 = gc(xasprintf("%s/%s", dpath, "foo")); file2 = gc(xasprintf("%s/%s", dpath, "bar")); EXPECT_NE(-1, mkdir(dpath, 0755)); EXPECT_NE(-1, touch(file1, 0644)); EXPECT_NE(-1, touch(file2, 0644)); EXPECT_TRUE(NULL != (dir = opendir(dpath))); while ((ent = readdir(dir))) { if (!strcmp(ent->d_name, "foo")) { EXPECT_EQ(DT_REG, ent->d_type); hasfoo = true; } if (!strcmp(ent->d_name, "bar")) { EXPECT_EQ(DT_REG, ent->d_type); hasbar = true; } } EXPECT_TRUE(hasfoo); EXPECT_TRUE(hasbar); EXPECT_NE(-1, closedir(dir)); EXPECT_NE(-1, unlink(file2)); EXPECT_NE(-1, unlink(file1)); EXPECT_NE(-1, rmdir(dpath)); } TEST(dirstream, zipTest) { bool foundNewYork = false; const char *path = "/zip/usr/share/zoneinfo/"; ASSERT_NE(NULL, (dir = opendir(path))); while ((ent = readdir(dir))) { foundNewYork |= !strcmp(ent->d_name, "New_York"); } EXPECT_SYS(0, 0, closedir(dir)); EXPECT_TRUE(foundNewYork); } TEST(rewinddir, test) { bool hasfoo = false; bool hasbar = false; char *dpath, *file1, *file2; dpath = gc(xasprintf("%s.%d", "dirstream", rand())); file1 = gc(xasprintf("%s/%s", dpath, "foo")); file2 = gc(xasprintf("%s/%s", dpath, "bar")); EXPECT_NE(-1, mkdir(dpath, 0755)); EXPECT_NE(-1, touch(file1, 0644)); EXPECT_NE(-1, touch(file2, 0644)); EXPECT_TRUE(NULL != (dir = opendir(dpath))); readdir(dir); readdir(dir); readdir(dir); rewinddir(dir); while ((ent = readdir(dir))) { if (strcmp(ent->d_name, "foo")) hasfoo = true; if (strcmp(ent->d_name, "bar")) hasbar = true; } EXPECT_TRUE(hasfoo); EXPECT_TRUE(hasbar); EXPECT_NE(-1, closedir(dir)); EXPECT_NE(-1, unlink(file2)); EXPECT_NE(-1, unlink(file1)); EXPECT_NE(-1, rmdir(dpath)); } TEST(dirstream, zipTest_notDir) { ASSERT_EQ(NULL, opendir("/zip/usr/share/zoneinfo/New_York")); ASSERT_EQ(ENOTDIR, errno); } TEST(dirstream, ino) { ASSERT_SYS(0, 0, mkdir("boop", 0755)); EXPECT_SYS(0, 0, touch("boop/a", 0644)); EXPECT_SYS(0, 0, touch("boop/b", 0644)); EXPECT_SYS(0, 0, touch("boop/c", 0644)); ASSERT_NE(NULL, (dir = opendir("boop"))); ASSERT_NE(NULL, (ent = readdir(dir))); // #1 ASSERT_NE(NULL, (ent = readdir(dir))); // #2 long pos = telldir(dir); ASSERT_NE(NULL, (ent = readdir(dir))); // #3 char name[32]; strlcpy(name, ent->d_name, sizeof(name)); ASSERT_NE(NULL, (ent = readdir(dir))); // #4 ASSERT_NE(NULL, (ent = readdir(dir))); // #5 ASSERT_EQ(NULL, (ent = readdir(dir))); // eod seekdir(dir, pos); ASSERT_NE(NULL, (ent = readdir(dir))); // #2 ASSERT_STREQ(name, ent->d_name); ASSERT_NE(NULL, (ent = readdir(dir))); // #3 ASSERT_NE(NULL, (ent = readdir(dir))); // #4 ASSERT_EQ(NULL, (ent = readdir(dir))); // eod ASSERT_EQ(NULL, (ent = readdir(dir))); // eod ASSERT_SYS(0, 0, closedir(dir)); } TEST(dirstream, seek) { ASSERT_SYS(0, 0, mkdir("boop", 0755)); EXPECT_SYS(0, 0, touch("boop/a", 0644)); EXPECT_SYS(0, 0, touch("boop/b", 0644)); EXPECT_SYS(0, 0, touch("boop/c", 0644)); ASSERT_NE(NULL, (dir = opendir("boop"))); ASSERT_NE(NULL, (ent = readdir(dir))); // #1 ASSERT_NE(NULL, (ent = readdir(dir))); // #2 long pos = telldir(dir); ASSERT_NE(NULL, (ent = readdir(dir))); // #3 char name[32]; strlcpy(name, ent->d_name, sizeof(name)); ASSERT_NE(NULL, (ent = readdir(dir))); // #4 ASSERT_NE(NULL, (ent = readdir(dir))); // #5 ASSERT_EQ(NULL, (ent = readdir(dir))); // eod seekdir(dir, pos); ASSERT_NE(NULL, (ent = readdir(dir))); // #2 ASSERT_STREQ(name, ent->d_name); ASSERT_NE(NULL, (ent = readdir(dir))); // #3 ASSERT_NE(NULL, (ent = readdir(dir))); // #4 ASSERT_EQ(NULL, (ent = readdir(dir))); // eod ASSERT_EQ(NULL, (ent = readdir(dir))); // eod ASSERT_SYS(0, 0, closedir(dir)); } TEST(dirstream, seeky) { char name[256]; char path[512]; struct stat golden; int i, j, n = 1000; int goodindex = 500; struct critbit0 tree = {0}; ASSERT_SYS(0, 0, mkdir("boop", 0755)); ASSERT_EQ(1, critbit0_insert(&tree, ".")); ASSERT_EQ(1, critbit0_insert(&tree, "..")); for (i = 0; i < n; ++i) { for (j = 0; j < 255; ++j) { name[j] = '0' + rand() % 10; } // TODO(jart): why does Windows croak with 255 name[100] = 0; strcpy(path, "boop/"); strcat(path, name); path[255] = 0; *FormatInt32(path + 5, i) = '-'; ASSERT_EQ(1, critbit0_insert(&tree, path + 5)); ASSERT_SYS(0, 0, touch(path, 0644)); if (i == goodindex) { ASSERT_SYS(0, 0, stat(path, &golden)); } } // do a full pass { ASSERT_NE(NULL, (dir = opendir("boop"))); long tell = -1; long prev = telldir(dir); while ((ent = readdir(dir))) { if (atoi(ent->d_name) == goodindex) { ASSERT_EQ(golden.st_ino, ent->d_ino); tell = prev; } prev = telldir(dir); ASSERT_EQ(1, critbit0_delete(&tree, ent->d_name)); } ASSERT_EQ(NULL, tree.root); // all entries were found ASSERT_NE(-1, tell); seekdir(dir, tell); ASSERT_NE(NULL, (ent = readdir(dir))); ASSERT_EQ(goodindex, atoi(ent->d_name)); ASSERT_EQ(golden.st_ino, ent->d_ino); ASSERT_SYS(0, 0, closedir(dir)); } // do a partial pass, and seek midway { ASSERT_NE(NULL, (dir = opendir("boop"))); int abort = 700; long tell = -1; bool foundit = false; long prev = telldir(dir); while ((ent = readdir(dir))) { if (atoi(ent->d_name) == goodindex) { ASSERT_EQ(golden.st_ino, ent->d_ino); tell = prev; foundit = true; } prev = telldir(dir); if (--abort <= 0 && foundit) { break; } } ASSERT_NE(-1, tell); seekdir(dir, tell); ASSERT_NE(NULL, (ent = readdir(dir))); ASSERT_EQ(goodindex, atoi(ent->d_name)); ASSERT_EQ(golden.st_ino, ent->d_ino); ASSERT_SYS(0, 0, closedir(dir)); } } TEST(dirstream, dots) { bool got_dot = false; bool got_dot_dot = false; ASSERT_NE(NULL, (dir = opendir("."))); while ((ent = readdir(dir))) { got_dot |= !strcmp(ent->d_name, "."); got_dot_dot |= !strcmp(ent->d_name, ".."); } ASSERT_SYS(0, 0, closedir(dir)); ASSERT_TRUE(got_dot_dot); ASSERT_TRUE(got_dot); } TEST(dirstream, inoFile_isConsistentWithStat) { struct stat st; bool gotsome = false; EXPECT_SYS(0, 0, touch("foo", 0644)); EXPECT_SYS(0, 0, stat("foo", &st)); ASSERT_NE(NULL, (dir = opendir("."))); while ((ent = readdir(dir))) { if (!strcmp(ent->d_name, "foo")) { ASSERT_EQ(st.st_ino, ent->d_ino); gotsome = true; } } ASSERT_SYS(0, 0, closedir(dir)); ASSERT_TRUE(gotsome); } TEST(dirstream_zipos, inoFile_isConsistentWithStat) { struct stat st; const char *dirpath; char filename[PATH_MAX]; dirpath = "/zip/usr/share/zoneinfo"; ASSERT_NE(NULL, (dir = opendir(dirpath))); while ((ent = readdir(dir))) { snprintf(filename, sizeof(filename), "%s/%s", dirpath, ent->d_name); EXPECT_SYS(0, 0, stat(filename, &st)); ASSERT_EQ(st.st_ino, ent->d_ino); } ASSERT_SYS(0, 0, closedir(dir)); } static const char *DescribeDt(int dt) { static char buf[12]; switch (dt) { case DT_UNKNOWN: return "DT_UNKNOWN"; case DT_FIFO: return "DT_FIFO"; case DT_CHR: return "DT_CHR"; case DT_DIR: return "DT_DIR"; case DT_BLK: return "DT_BLK"; case DT_REG: return "DT_REG"; case DT_LNK: return "DT_LNK"; case DT_SOCK: return "DT_SOCK"; default: FormatInt32(buf, dt); return buf; } } static const char *DescribeFtw(int dt) { static char buf[12]; switch (dt) { case FTW_F: return "FTW_F"; case FTW_D: return "FTW_D"; case FTW_DNR: return "FTW_DNR"; case FTW_NS: return "FTW_NS"; case FTW_SL: return "FTW_SL"; case FTW_DP: return "FTW_DP"; case FTW_SLN: return "FTW_SLN"; default: FormatInt32(buf, dt); return buf; } } char *b; static int walk(const char *fpath, // const struct stat *st, // int typeflag, // struct FTW *ftwbuf) { // appendf(&b, "%-6s %s\n", DescribeFtw(typeflag), fpath); return 0; } TEST(dirstream, walk) { ASSERT_SYS(0, 0, nftw("/zip", walk, 128, FTW_PHYS | FTW_DEPTH)); ASSERT_STREQ("FTW_F /zip/echo.com\n" "FTW_F /zip/libc/testlib/hyperion.txt\n" "FTW_F /zip/libc/testlib/moby.txt\n" "FTW_DP /zip/libc/testlib\n" "FTW_DP /zip/libc\n" "FTW_F /zip/usr/share/zoneinfo/New_York\n" "FTW_DP /zip/usr/share/zoneinfo\n" "FTW_DP /zip/usr/share\n" "FTW_DP /zip/usr\n" "FTW_F /zip/.cosmo\n" "FTW_DP /zip\n", b); free(b); b = 0; }