mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-02-01 03:53:33 +00:00
f531acc8f9
- Invent openatemp() API - Invent O_UNLINK open flag - Introduce getenv_secure() API - Remove `git pull` from cosmocc - Fix utimes() when path is NULL - Fix mktemp() to never return NULL - Fix utimensat() UTIME_OMIT on XNU - Improve utimensat() code for RHEL5 - Turn `argv[0]` C:/ to /C/ on Windows - Introduce tmpnam() and tmpnam_r() APIs - Fix more const issues with internal APIs - Permit utimes() on WIN32 in O_RDONLY mode - Fix fdopendir() to check fd is a directory - Fix recent crash regression in landlock make - Fix futimens(AT_FDCWD, NULL) to return EBADF - Use workaround so `make -j` doesn't fork bomb - Rename dontdiscard to __wur (just like glibc) - Fix st_size for WIN32 symlinks containing UTF-8 - Introduce stdio ext APIs needed by GNU coreutils - Fix lstat() on WIN32 for symlinks to directories - Move some constants from normalize.inc to limits.h - Fix segv with memchr() and memcmp() overlapping page - Implement POSIX fflush() behavior for reader streams - Implement AT_SYMLINK_NOFOLLOW for utimensat() on WIN32 - Don't change read-only status of existing files on WIN32 - Correctly handle `0x[^[:xdigit:]]` case in strtol() functions
439 lines
13 KiB
C
439 lines
13 KiB
C
/*-*- 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/limits.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(fdopendir, test) {
|
|
ASSERT_SYS(0, 0, touch("foo", 0644));
|
|
ASSERT_SYS(0, 3, open("foo", O_RDONLY));
|
|
ASSERT_SYS(ENOTDIR, 0, fdopendir(3));
|
|
ASSERT_SYS(0, 0, close(3));
|
|
}
|
|
|
|
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;
|
|
}
|