mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-05-22 21:32:31 +00:00
Make ZipOS and Qemu work better
This change improves the dirstream library in a lot of respects, especially for /zip/... files. Also turn off MAP_STACK on Aarch64 because Qemu seems to implement it differently than Linux and it's probably responsible for a lot of mysterious crashes.
This commit is contained in:
parent
4658ae539f
commit
110559ce6a
48 changed files with 748 additions and 500 deletions
|
@ -35,7 +35,6 @@ void SetUpOnce(void) {
|
|||
}
|
||||
|
||||
TEST(access, efault) {
|
||||
ASSERT_SYS(EFAULT, -1, access(0, F_OK));
|
||||
if (IsWindows() || !IsAsan()) return; // not possible
|
||||
ASSERT_SYS(EFAULT, -1, access((void *)77, F_OK));
|
||||
}
|
||||
|
|
|
@ -71,6 +71,15 @@ TEST(read_pipe, canBeInterruptedByAlarm) {
|
|||
close(fds[0]);
|
||||
}
|
||||
|
||||
TEST(read_directory, eisdir) {
|
||||
// TODO(jart): what
|
||||
if (IsWindows() || IsFreebsd()) return;
|
||||
ASSERT_SYS(0, 0, mkdir("boop", 0755));
|
||||
ASSERT_SYS(0, 3, open("boop", O_RDONLY | O_DIRECTORY));
|
||||
ASSERT_SYS(EISDIR, -1, read(3, 0, 0));
|
||||
ASSERT_SYS(0, 0, close(3));
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
BENCH(read, bench) {
|
||||
|
|
|
@ -21,6 +21,7 @@
|
|||
#include "libc/calls/calls.h"
|
||||
#include "libc/calls/internal.h"
|
||||
#include "libc/calls/struct/metastat.internal.h"
|
||||
#include "libc/calls/struct/stat.h"
|
||||
#include "libc/dce.h"
|
||||
#include "libc/errno.h"
|
||||
#include "libc/mem/gc.internal.h"
|
||||
|
@ -47,13 +48,15 @@ TEST(stat_010, testEmptyFile_sizeIsZero) {
|
|||
}
|
||||
|
||||
TEST(stat, enoent) {
|
||||
ASSERT_SYS(ENOENT, -1, stat("hi", 0));
|
||||
ASSERT_SYS(ENOENT, -1, stat("o/doesnotexist", 0));
|
||||
struct stat st;
|
||||
ASSERT_SYS(ENOENT, -1, stat("hi", &st));
|
||||
ASSERT_SYS(ENOENT, -1, stat("o/doesnotexist", &st));
|
||||
}
|
||||
|
||||
TEST(stat, enotdir) {
|
||||
struct stat st;
|
||||
ASSERT_SYS(0, 0, close(creat("yo", 0644)));
|
||||
ASSERT_SYS(ENOTDIR, -1, stat("yo/there", 0));
|
||||
ASSERT_SYS(ENOTDIR, -1, stat("yo/there", &st));
|
||||
}
|
||||
|
||||
TEST(stat, zipos) {
|
||||
|
|
|
@ -17,9 +17,12 @@
|
|||
│ PERFORMANCE OF THIS SOFTWARE. │
|
||||
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||
#include "libc/calls/calls.h"
|
||||
#include "libc/errno.h"
|
||||
#include "libc/limits.h"
|
||||
#include "libc/mem/gc.h"
|
||||
#include "libc/mem/mem.h"
|
||||
#include "libc/runtime/runtime.h"
|
||||
#include "libc/runtime/zipos.internal.h"
|
||||
#include "libc/str/str.h"
|
||||
#include "libc/sysv/consts/o.h"
|
||||
#include "libc/testlib/hyperion.h"
|
||||
|
@ -53,3 +56,39 @@ TEST(zipos, test) {
|
|||
for (i = 0; i < n; ++i) EXPECT_SYS(0, 0, _join(t + i));
|
||||
__print_maps();
|
||||
}
|
||||
|
||||
TEST(zipos, normpath) {
|
||||
{
|
||||
char s[] = "";
|
||||
__zipos_normpath(s);
|
||||
ASSERT_STREQ("", s);
|
||||
}
|
||||
{
|
||||
char s[] = "usr/";
|
||||
__zipos_normpath(s);
|
||||
ASSERT_STREQ("usr", s);
|
||||
}
|
||||
{
|
||||
char s[] = "usr/./";
|
||||
__zipos_normpath(s);
|
||||
ASSERT_STREQ("usr", s);
|
||||
}
|
||||
}
|
||||
|
||||
#if 0
|
||||
TEST(zipos_O_DIRECTORY, blocksOpeningOfNormalFiles) {
|
||||
ASSERT_SYS(ENOTDIR, -1,
|
||||
open("/zip/libc/testlib/hyperion.txt", O_RDONLY | O_DIRECTORY));
|
||||
}
|
||||
#endif
|
||||
|
||||
TEST(zipos, readPastEof) {
|
||||
char buf[512];
|
||||
ASSERT_SYS(0, 3, open("/zip/libc/testlib/hyperion.txt", O_RDONLY));
|
||||
EXPECT_SYS(EINVAL, -1, pread(3, buf, 512, UINT64_MAX));
|
||||
EXPECT_SYS(0, 0, pread(3, buf, 512, INT64_MAX));
|
||||
EXPECT_SYS(EINVAL, -1, lseek(3, UINT64_MAX, SEEK_SET));
|
||||
EXPECT_SYS(0, INT64_MAX, lseek(3, INT64_MAX, SEEK_SET));
|
||||
EXPECT_SYS(0, 0, read(3, buf, 512));
|
||||
EXPECT_SYS(0, 0, close(3));
|
||||
}
|
||||
|
|
|
@ -18,13 +18,18 @@
|
|||
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||
#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/intrin/kprintf.h"
|
||||
#include "libc/mem/gc.h"
|
||||
#include "libc/mem/gc.internal.h"
|
||||
#include "libc/runtime/runtime.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"
|
||||
|
@ -61,6 +66,10 @@ TEST(opendir, enotdir) {
|
|||
TEST(opendir, zipTest_fake) {
|
||||
ASSERT_NE(NULL, (dir = opendir("/zip")));
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ(".", ent->d_name);
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ("..", ent->d_name);
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ("echo.com", ent->d_name);
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ("usr", ent->d_name);
|
||||
|
@ -70,6 +79,10 @@ TEST(opendir, zipTest_fake) {
|
|||
EXPECT_EQ(0, closedir(dir));
|
||||
ASSERT_NE(NULL, (dir = opendir("/zip/")));
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ(".", ent->d_name);
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ("..", ent->d_name);
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ("echo.com", ent->d_name);
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ("usr", ent->d_name);
|
||||
|
@ -79,11 +92,19 @@ TEST(opendir, zipTest_fake) {
|
|||
EXPECT_EQ(0, closedir(dir));
|
||||
ASSERT_NE(NULL, (dir = opendir("/zip/usr")));
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ(".", ent->d_name);
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ("..", ent->d_name);
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ("share", ent->d_name);
|
||||
EXPECT_EQ(NULL, (ent = readdir(dir)));
|
||||
EXPECT_EQ(0, closedir(dir));
|
||||
ASSERT_NE(NULL, (dir = opendir("/zip/usr/")));
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ(".", ent->d_name);
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ("..", ent->d_name);
|
||||
EXPECT_NE(NULL, (ent = readdir(dir)));
|
||||
EXPECT_STREQ("share", ent->d_name);
|
||||
EXPECT_EQ(NULL, (ent = readdir(dir)));
|
||||
EXPECT_EQ(0, closedir(dir));
|
||||
|
@ -91,6 +112,28 @@ TEST(opendir, zipTest_fake) {
|
|||
EXPECT_EQ(NULL, (dir = opendir("/zip/us/")));
|
||||
}
|
||||
|
||||
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;
|
||||
|
@ -115,9 +158,9 @@ 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"));
|
||||
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));
|
||||
|
@ -143,12 +186,11 @@ TEST(dirstream, test) {
|
|||
TEST(dirstream, zipTest) {
|
||||
bool foundNewYork = false;
|
||||
const char *path = "/zip/usr/share/zoneinfo/";
|
||||
ASSERT_NE(0, _gc(xiso8601ts(NULL)));
|
||||
ASSERT_NE(NULL, (dir = opendir(path)));
|
||||
while ((ent = readdir(dir))) {
|
||||
foundNewYork |= !strcmp(ent->d_name, "New_York");
|
||||
}
|
||||
closedir(dir);
|
||||
EXPECT_SYS(0, 0, closedir(dir));
|
||||
EXPECT_TRUE(foundNewYork);
|
||||
}
|
||||
|
||||
|
@ -156,9 +198,9 @@ 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"));
|
||||
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));
|
||||
|
@ -183,3 +225,28 @@ TEST(dirstream, zipTest_notDir) {
|
|||
ASSERT_EQ(NULL, opendir("/zip/usr/share/zoneinfo/New_York"));
|
||||
ASSERT_EQ(ENOTDIR, errno);
|
||||
}
|
||||
|
||||
TEST(dirstream, seek) {
|
||||
if (IsNetbsd()) return; // omg
|
||||
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_SYS(0, 0, closedir(dir));
|
||||
}
|
||||
|
|
69
test/libc/stdio/zipdir_test.c
Normal file
69
test/libc/stdio/zipdir_test.c
Normal file
|
@ -0,0 +1,69 @@
|
|||
/*-*- 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 2023 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/struct/dirent.h"
|
||||
#include "libc/mem/gc.internal.h"
|
||||
#include "libc/runtime/runtime.h"
|
||||
#include "libc/str/str.h"
|
||||
#include "libc/sysv/consts/dt.h"
|
||||
#include "libc/testlib/testlib.h"
|
||||
|
||||
__static_yoink("zipos");
|
||||
__static_yoink("libc/testlib/hyperion.txt");
|
||||
__static_yoink("libc/testlib/moby.txt");
|
||||
__static_yoink("usr/share/zoneinfo/New_York");
|
||||
|
||||
DIR *dir;
|
||||
struct dirent *ent;
|
||||
|
||||
TEST(zipdir, test) {
|
||||
const char *path = "/zip/libc/testlib///";
|
||||
ASSERT_NE(NULL, (dir = opendir(path)));
|
||||
ASSERT_EQ(0, telldir(dir));
|
||||
ASSERT_NE(NULL, (ent = readdir(dir)));
|
||||
ASSERT_EQ(0, strcmp(ent->d_name, "."));
|
||||
ASSERT_EQ(DT_DIR, ent->d_type);
|
||||
ASSERT_NE(NULL, (ent = readdir(dir)));
|
||||
ASSERT_EQ(0, strcmp(ent->d_name, ".."));
|
||||
ASSERT_EQ(DT_DIR, ent->d_type);
|
||||
ASSERT_NE(NULL, (ent = readdir(dir)));
|
||||
ASSERT_EQ(0, strcmp(ent->d_name, "hyperion.txt"));
|
||||
ASSERT_EQ(DT_REG, ent->d_type);
|
||||
long pos = telldir(dir);
|
||||
ASSERT_NE(NULL, (ent = readdir(dir)));
|
||||
ASSERT_EQ(0, strcmp(ent->d_name, "moby.txt"));
|
||||
ASSERT_EQ(DT_REG, ent->d_type);
|
||||
ASSERT_EQ(NULL, (ent = readdir(dir)));
|
||||
seekdir(dir, pos);
|
||||
ASSERT_NE(NULL, (ent = readdir(dir)));
|
||||
ASSERT_EQ(0, strcmp(ent->d_name, "moby.txt"));
|
||||
ASSERT_EQ(DT_REG, ent->d_type);
|
||||
ASSERT_EQ(NULL, (ent = readdir(dir)));
|
||||
ASSERT_SYS(0, 0, closedir(dir));
|
||||
}
|
||||
|
||||
TEST(dirstream, hasDirectoryEntry) {
|
||||
bool gotsome = false;
|
||||
const char *path = "/zip/usr/share/zoneinfo";
|
||||
ASSERT_NE(NULL, (dir = opendir(path)));
|
||||
while ((ent = readdir(dir))) {
|
||||
gotsome = true;
|
||||
}
|
||||
ASSERT_SYS(0, 0, closedir(dir));
|
||||
EXPECT_TRUE(gotsome);
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue