mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-22 17:30:15 +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
				
			
		
										
											Binary file not shown.
										
									
								
							|  | @ -29,9 +29,9 @@ | ||||||
| #include "libc/intrin/strace.internal.h" | #include "libc/intrin/strace.internal.h" | ||||||
| #include "libc/intrin/weaken.h" | #include "libc/intrin/weaken.h" | ||||||
| #include "libc/log/libfatal.internal.h" | #include "libc/log/libfatal.internal.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/sysv/consts/o.h" | #include "libc/sysv/consts/o.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/runtime/zipos.internal.h" |  | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Replaces current process with program. |  * Replaces current process with program. | ||||||
|  | @ -74,7 +74,7 @@ int execve(const char *prog, char *const argv[], char *const envp[]) { | ||||||
|     if (!rc) { |     if (!rc) { | ||||||
|       if (_weaken(__zipos_parseuri) && |       if (_weaken(__zipos_parseuri) && | ||||||
|           (_weaken(__zipos_parseuri)(prog, &uri) != -1)) { |           (_weaken(__zipos_parseuri)(prog, &uri) != -1)) { | ||||||
|         rc = _weaken(__zipos_open)(&uri, O_RDONLY | O_CLOEXEC, 0); |         rc = _weaken(__zipos_open)(&uri, O_RDONLY | O_CLOEXEC); | ||||||
|         if (rc != -1) { |         if (rc != -1) { | ||||||
|           const int zipFD = rc; |           const int zipFD = rc; | ||||||
|           strace_enabled(-1); |           strace_enabled(-1); | ||||||
|  |  | ||||||
|  | @ -26,9 +26,9 @@ | ||||||
| #include "libc/intrin/describeflags.internal.h" | #include "libc/intrin/describeflags.internal.h" | ||||||
| #include "libc/intrin/strace.internal.h" | #include "libc/intrin/strace.internal.h" | ||||||
| #include "libc/intrin/weaken.h" | #include "libc/intrin/weaken.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/sysv/consts/at.h" | #include "libc/sysv/consts/at.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/runtime/zipos.internal.h" |  | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Checks if effective user can access path in particular ways. |  * Checks if effective user can access path in particular ways. | ||||||
|  | @ -51,7 +51,7 @@ | ||||||
| int faccessat(int dirfd, const char *path, int amode, int flags) { | int faccessat(int dirfd, const char *path, int amode, int flags) { | ||||||
|   int e, rc; |   int e, rc; | ||||||
|   struct ZiposUri zipname; |   struct ZiposUri zipname; | ||||||
|   if (!path || (IsAsan() && !__asan_is_valid_str(path))) { |   if (IsAsan() && !__asan_is_valid_str(path)) { | ||||||
|     rc = efault(); |     rc = efault(); | ||||||
|   } else if (__isfdkind(dirfd, kFdZip)) { |   } else if (__isfdkind(dirfd, kFdZip)) { | ||||||
|     rc = enotsup(); |     rc = enotsup(); | ||||||
|  |  | ||||||
|  | @ -24,8 +24,8 @@ | ||||||
| #include "libc/intrin/describeflags.internal.h" | #include "libc/intrin/describeflags.internal.h" | ||||||
| #include "libc/intrin/strace.internal.h" | #include "libc/intrin/strace.internal.h" | ||||||
| #include "libc/intrin/weaken.h" | #include "libc/intrin/weaken.h" | ||||||
| #include "libc/sysv/errfuns.h" |  | ||||||
| #include "libc/runtime/zipos.internal.h" | #include "libc/runtime/zipos.internal.h" | ||||||
|  | #include "libc/sysv/errfuns.h" | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Returns information about file, via open()'d descriptor. |  * Returns information about file, via open()'d descriptor. | ||||||
|  | @ -38,7 +38,9 @@ | ||||||
|  */ |  */ | ||||||
| int fstat(int fd, struct stat *st) { | int fstat(int fd, struct stat *st) { | ||||||
|   int rc; |   int rc; | ||||||
|   if (__isfdkind(fd, kFdZip)) { |   if (IsAsan() && !__asan_is_valid(st, sizeof(*st))) { | ||||||
|  |     rc = efault(); | ||||||
|  |   } else if (__isfdkind(fd, kFdZip)) { | ||||||
|     rc = _weaken(__zipos_fstat)( |     rc = _weaken(__zipos_fstat)( | ||||||
|         (struct ZiposHandle *)(intptr_t)g_fds.p[fd].handle, st); |         (struct ZiposHandle *)(intptr_t)g_fds.p[fd].handle, st); | ||||||
|   } else if (!IsWindows() && !IsMetal()) { |   } else if (!IsWindows() && !IsMetal()) { | ||||||
|  |  | ||||||
|  | @ -29,10 +29,10 @@ | ||||||
| #include "libc/intrin/weaken.h" | #include "libc/intrin/weaken.h" | ||||||
| #include "libc/log/log.h" | #include "libc/log/log.h" | ||||||
| #include "libc/mem/alloca.h" | #include "libc/mem/alloca.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/str/str.h" | #include "libc/str/str.h" | ||||||
| #include "libc/sysv/consts/at.h" | #include "libc/sysv/consts/at.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/runtime/zipos.internal.h" |  | ||||||
| 
 | 
 | ||||||
| static inline const char *__strace_fstatat_flags(char buf[12], int flags) { | static inline const char *__strace_fstatat_flags(char buf[12], int flags) { | ||||||
|   if (flags == AT_SYMLINK_NOFOLLOW) return "AT_SYMLINK_NOFOLLOW"; |   if (flags == AT_SYMLINK_NOFOLLOW) return "AT_SYMLINK_NOFOLLOW"; | ||||||
|  | @ -56,7 +56,9 @@ int fstatat(int dirfd, const char *path, struct stat *st, int flags) { | ||||||
|   /* execve() depends on this */ |   /* execve() depends on this */ | ||||||
|   int rc; |   int rc; | ||||||
|   struct ZiposUri zipname; |   struct ZiposUri zipname; | ||||||
|   if (__isfdkind(dirfd, kFdZip)) { |   if (IsAsan() && !__asan_is_valid(st, sizeof(*st))) { | ||||||
|  |     rc = efault(); | ||||||
|  |   } else if (__isfdkind(dirfd, kFdZip)) { | ||||||
|     STRACE("zipos dirfd not supported yet"); |     STRACE("zipos dirfd not supported yet"); | ||||||
|     rc = einval(); |     rc = einval(); | ||||||
|   } else if (_weaken(__zipos_stat) && |   } else if (_weaken(__zipos_stat) && | ||||||
|  |  | ||||||
|  | @ -25,8 +25,8 @@ | ||||||
| #include "libc/intrin/strace.internal.h" | #include "libc/intrin/strace.internal.h" | ||||||
| #include "libc/intrin/weaken.h" | #include "libc/intrin/weaken.h" | ||||||
| #include "libc/log/backtrace.internal.h" | #include "libc/log/backtrace.internal.h" | ||||||
| #include "libc/sysv/errfuns.h" |  | ||||||
| #include "libc/runtime/zipos.internal.h" | #include "libc/runtime/zipos.internal.h" | ||||||
|  | #include "libc/sysv/errfuns.h" | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Changes current position of file descriptor, e.g. |  * Changes current position of file descriptor, e.g. | ||||||
|  |  | ||||||
|  | @ -29,11 +29,11 @@ | ||||||
| #include "libc/intrin/strace.internal.h" | #include "libc/intrin/strace.internal.h" | ||||||
| #include "libc/intrin/weaken.h" | #include "libc/intrin/weaken.h" | ||||||
| #include "libc/log/log.h" | #include "libc/log/log.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/str/str.h" | #include "libc/str/str.h" | ||||||
| #include "libc/sysv/consts/at.h" | #include "libc/sysv/consts/at.h" | ||||||
| #include "libc/sysv/consts/o.h" | #include "libc/sysv/consts/o.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/runtime/zipos.internal.h" |  | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Opens file. |  * Opens file. | ||||||
|  | @ -166,7 +166,7 @@ int openat(int dirfd, const char *file, int flags, ...) { | ||||||
|       if (_weaken(__zipos_open) && |       if (_weaken(__zipos_open) && | ||||||
|           _weaken(__zipos_parseuri)(file, &zipname) != -1) { |           _weaken(__zipos_parseuri)(file, &zipname) != -1) { | ||||||
|         if (!__vforked && dirfd == AT_FDCWD) { |         if (!__vforked && dirfd == AT_FDCWD) { | ||||||
|           rc = _weaken(__zipos_open)(&zipname, flags, mode); |           rc = _weaken(__zipos_open)(&zipname, flags); | ||||||
|         } else { |         } else { | ||||||
|           rc = enotsup(); /* TODO */ |           rc = enotsup(); /* TODO */ | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -742,14 +742,6 @@ void abort(void) wontreturn; | ||||||
| #pragma GCC diagnostic error "-Wtrampolines"
 | #pragma GCC diagnostic error "-Wtrampolines"
 | ||||||
| #if __GNUC__ >= 6
 | #if __GNUC__ >= 6
 | ||||||
| #pragma GCC diagnostic error "-Wnonnull-compare"
 | #pragma GCC diagnostic error "-Wnonnull-compare"
 | ||||||
| #if defined(_COSMO_SOURCE) && !defined(MODE_DBG) && \
 |  | ||||||
|     !defined(STACK_FRAME_UNLIMITED) |  | ||||||
| #pragma GCC diagnostic error "-Wframe-larger-than=131072"
 |  | ||||||
| #if __GNUC__ >= 9
 |  | ||||||
| // #pragma GCC diagnostic error "-Walloca-larger-than=1024"
 |  | ||||||
| // #pragma GCC diagnostic error "-Wvla-larger-than=1024"
 |  | ||||||
| #endif /* GCC 9+ */
 |  | ||||||
| #endif /* STACK_FRAME_UNLIMITED */
 |  | ||||||
| #elif __GNUC__ >= 9
 | #elif __GNUC__ >= 9
 | ||||||
| #pragma GCC diagnostic error /* e.g. fabs not abs */ "-Wabsolute-value"
 | #pragma GCC diagnostic error /* e.g. fabs not abs */ "-Wabsolute-value"
 | ||||||
| #endif /* GCC 6+ */
 | #endif /* GCC 6+ */
 | ||||||
|  |  | ||||||
|  | @ -74,7 +74,7 @@ | ||||||
| #define __BIGGEST_ALIGNMENT__ 16
 | #define __BIGGEST_ALIGNMENT__ 16
 | ||||||
| #endif
 | #endif
 | ||||||
| 
 | 
 | ||||||
| #define APE_STACKSIZE 4194304 /* default 4mb stack */
 | #define APE_STACKSIZE 8388608 /* default 8mb stack */
 | ||||||
| #define APE_PAGESIZE  0x10000 /* i386+ */
 | #define APE_PAGESIZE  0x10000 /* i386+ */
 | ||||||
| #ifdef _COSMO_SOURCE
 | #ifdef _COSMO_SOURCE
 | ||||||
| #define FRAMESIZE 0x10000
 | #define FRAMESIZE 0x10000
 | ||||||
|  |  | ||||||
|  | @ -25,7 +25,7 @@ | ||||||
|  * @param delta is added to enabled state |  * @param delta is added to enabled state | ||||||
|  * @return enabled state before `delta` was applied |  * @return enabled state before `delta` was applied | ||||||
|  */ |  */ | ||||||
| dontinstrument int ftrace_enabled(int delta) { | dontasan dontubsan dontinstrument int ftrace_enabled(int delta) { | ||||||
|   int res; |   int res; | ||||||
|   struct CosmoTib *tib; |   struct CosmoTib *tib; | ||||||
|   if (__tls_enabled) { |   if (__tls_enabled) { | ||||||
|  |  | ||||||
|  | @ -25,7 +25,7 @@ | ||||||
|  * @param delta is added to enabled state |  * @param delta is added to enabled state | ||||||
|  * @return enabled state before `delta` was applied |  * @return enabled state before `delta` was applied | ||||||
|  */ |  */ | ||||||
| dontinstrument int strace_enabled(int delta) { | dontasan dontubsan dontinstrument int strace_enabled(int delta) { | ||||||
|   int res; |   int res; | ||||||
|   struct CosmoTib *tib; |   struct CosmoTib *tib; | ||||||
|   if (__tls_enabled) { |   if (__tls_enabled) { | ||||||
|  |  | ||||||
|  | @ -164,7 +164,7 @@ wontreturn textstartup void cosmo(long *sp, struct Syslib *m1) { | ||||||
|   __switch_stacks(argc, argv, envp, auxv, cosmo2, |   __switch_stacks(argc, argv, envp, auxv, cosmo2, | ||||||
|                   (char *)mmap(ape_stack_vaddr, (uintptr_t)ape_stack_memsz, |                   (char *)mmap(ape_stack_vaddr, (uintptr_t)ape_stack_memsz, | ||||||
|                                MAP_FIXED | PROT_READ | PROT_WRITE, |                                MAP_FIXED | PROT_READ | PROT_WRITE, | ||||||
|                                MAP_ANONYMOUS | MAP_STACK, -1, 0) + |                                MAP_PRIVATE | MAP_ANONYMOUS, -1, 0) + | ||||||
|                       (uintptr_t)ape_stack_memsz); |                       (uintptr_t)ape_stack_memsz); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -54,7 +54,6 @@ | ||||||
| #include "libc/sysv/consts/ss.h" | #include "libc/sysv/consts/ss.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/thread/thread.h" | #include "libc/thread/thread.h" | ||||||
| #include "libc/zipos/zipos.internal.h" |  | ||||||
| 
 | 
 | ||||||
| #define MAP_ANONYMOUS_linux   0x00000020 | #define MAP_ANONYMOUS_linux   0x00000020 | ||||||
| #define MAP_ANONYMOUS_openbsd 0x00001000 | #define MAP_ANONYMOUS_openbsd 0x00001000 | ||||||
|  |  | ||||||
|  | @ -18,10 +18,10 @@ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/calls/calls.h" | #include "libc/calls/calls.h" | ||||||
| #include "libc/calls/struct/stat.h" | #include "libc/calls/struct/stat.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/sysv/consts/ok.h" | #include "libc/sysv/consts/ok.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/zip.internal.h" | #include "libc/zip.internal.h" | ||||||
| #include "libc/runtime/zipos.internal.h" |  | ||||||
| 
 | 
 | ||||||
| // TODO: this should check parent directory components
 | // TODO: this should check parent directory components
 | ||||||
| 
 | 
 | ||||||
|  | @ -31,37 +31,30 @@ | ||||||
|  * @param uri is obtained via __zipos_parseuri() |  * @param uri is obtained via __zipos_parseuri() | ||||||
|  * @asyncsignalsafe |  * @asyncsignalsafe | ||||||
|  */ |  */ | ||||||
| int __zipos_access(const struct ZiposUri *name, int amode) { | int __zipos_access(struct ZiposUri *name, int amode) { | ||||||
|   ssize_t cf; | 
 | ||||||
|   int rc, mode; |  | ||||||
|   struct Zipos *z; |   struct Zipos *z; | ||||||
|   if ((z = __zipos_get()) && (cf = __zipos_find(z, name)) != -1) { |   if (!(z = __zipos_get())) { | ||||||
|     mode = GetZipCfileMode(z->map + cf); |     return enoexec(); | ||||||
|     if (amode == F_OK) { |  | ||||||
|       rc = 0; |  | ||||||
|     } else if (amode == R_OK) { |  | ||||||
|       if (mode & 0444) { |  | ||||||
|         rc = 0; |  | ||||||
|       } else { |  | ||||||
|         rc = eacces(); |  | ||||||
|       } |  | ||||||
|     } else if (amode == W_OK) { |  | ||||||
|       if (mode & 0222) { |  | ||||||
|         rc = 0; |  | ||||||
|       } else { |  | ||||||
|         rc = eacces(); |  | ||||||
|       } |  | ||||||
|     } else if (amode == X_OK) { |  | ||||||
|       if (mode & 0111) { |  | ||||||
|         rc = 0; |  | ||||||
|       } else { |  | ||||||
|         rc = eacces(); |  | ||||||
|       } |  | ||||||
|     } else { |  | ||||||
|       rc = einval(); |  | ||||||
|     } |  | ||||||
|   } else { |  | ||||||
|     rc = enoent(); |  | ||||||
|   } |   } | ||||||
|   return rc; | 
 | ||||||
|  |   ssize_t cf; | ||||||
|  |   if ((cf = __zipos_find(z, name)) == -1) { | ||||||
|  |     return enoent(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   int mode = GetZipCfileMode(z->map + cf); | ||||||
|  |   if (amode == F_OK) { | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  |   if (amode & ~(R_OK | W_OK | X_OK)) { | ||||||
|  |     return einval(); | ||||||
|  |   } | ||||||
|  |   if (((amode & X_OK) && !(mode & 0111)) || | ||||||
|  |       ((amode & W_OK) && !(mode & 0222)) || | ||||||
|  |       ((amode & R_OK) && !(mode & 0444))) { | ||||||
|  |     return eacces(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -16,7 +16,6 @@ | ||||||
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ | │ PERFORMANCE OF THIS SOFTWARE.                                                │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/calls/calls.h" |  | ||||||
| #include "libc/calls/internal.h" | #include "libc/calls/internal.h" | ||||||
| #include "libc/calls/state.internal.h" | #include "libc/calls/state.internal.h" | ||||||
| #include "libc/calls/syscall-sysv.internal.h" | #include "libc/calls/syscall-sysv.internal.h" | ||||||
|  | @ -38,10 +37,10 @@ int __zipos_close(int fd) { | ||||||
|   if (!IsWindows()) { |   if (!IsWindows()) { | ||||||
|     rc = sys_close(fd); |     rc = sys_close(fd); | ||||||
|   } else { |   } else { | ||||||
|     rc = 0; /* no system file descriptor needed on nt */ |     rc = 0;  // no system file descriptor needed on nt
 | ||||||
|   } |   } | ||||||
|   if (!__vforked) { |   if (!__vforked) { | ||||||
|     __zipos_free(__zipos_get(), h); |     __zipos_free(h); | ||||||
|   } |   } | ||||||
|   return rc; |   return rc; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -17,34 +17,28 @@ | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ | │ PERFORMANCE OF THIS SOFTWARE.                                                │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/calls/internal.h" | #include "libc/calls/internal.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/sysv/consts/f.h" | #include "libc/sysv/consts/f.h" | ||||||
| #include "libc/sysv/consts/fd.h" | #include "libc/sysv/consts/fd.h" | ||||||
| #include "libc/sysv/consts/o.h" | #include "libc/sysv/consts/o.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/zip.internal.h" |  | ||||||
| #include "libc/runtime/zipos.internal.h" |  | ||||||
| 
 |  | ||||||
| #define ZIPOS  __zipos_get() |  | ||||||
| #define HANDLE ((struct ZiposHandle *)(intptr_t)g_fds.p[fd].handle) |  | ||||||
| 
 | 
 | ||||||
| int __zipos_fcntl(int fd, int cmd, uintptr_t arg) { | int __zipos_fcntl(int fd, int cmd, uintptr_t arg) { | ||||||
|   int rc; |  | ||||||
|   if (cmd == F_GETFD) { |   if (cmd == F_GETFD) { | ||||||
|     if (g_fds.p[fd].flags & O_CLOEXEC) { |     if (g_fds.p[fd].flags & O_CLOEXEC) { | ||||||
|       rc = FD_CLOEXEC; |       return FD_CLOEXEC; | ||||||
|     } else { |     } else { | ||||||
|       rc = 0; |       return 0; | ||||||
|     } |     } | ||||||
|   } else if (cmd == F_SETFD) { |   } else if (cmd == F_SETFD) { | ||||||
|     if (arg & FD_CLOEXEC) { |     if (arg & FD_CLOEXEC) { | ||||||
|       g_fds.p[fd].flags |= O_CLOEXEC; |       g_fds.p[fd].flags |= O_CLOEXEC; | ||||||
|       rc = FD_CLOEXEC; |       return FD_CLOEXEC; | ||||||
|     } else { |     } else { | ||||||
|       g_fds.p[fd].flags &= ~O_CLOEXEC; |       g_fds.p[fd].flags &= ~O_CLOEXEC; | ||||||
|       rc = 0; |       return 0; | ||||||
|     } |     } | ||||||
|   } else { |   } else { | ||||||
|     rc = einval(); |     return einval(); | ||||||
|   } |   } | ||||||
|   return rc; |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -16,39 +16,30 @@ | ||||||
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ | │ PERFORMANCE OF THIS SOFTWARE.                                                │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "ape/relocations.h" |  | ||||||
| #include "libc/assert.h" |  | ||||||
| #include "libc/runtime/runtime.h" |  | ||||||
| #include "libc/str/str.h" |  | ||||||
| #include "libc/zip.internal.h" |  | ||||||
| #include "libc/runtime/zipos.internal.h" | #include "libc/runtime/zipos.internal.h" | ||||||
|  | #include "libc/zip.internal.h" | ||||||
| 
 | 
 | ||||||
| // TODO(jart): improve time complexity here
 | ssize_t __zipos_find(struct Zipos *zipos, struct ZiposUri *name) { | ||||||
| 
 |  | ||||||
| ssize_t __zipos_find(struct Zipos *zipos, const struct ZiposUri *name) { |  | ||||||
|   const char *zname; |  | ||||||
|   size_t i, n, c, znamesize; |  | ||||||
|   if (!name->len) { |   if (!name->len) { | ||||||
|     return 0; |     return ZIPOS_SYNTHETIC_DIRECTORY; | ||||||
|   } |   } | ||||||
|   c = GetZipCdirOffset(zipos->cdir); |   bool found_subfile = false; | ||||||
|   n = GetZipCdirRecords(zipos->cdir); |   size_t c = GetZipCdirOffset(zipos->cdir); | ||||||
|   for (i = 0; i < n; ++i, c += ZIP_CFILE_HDRSIZE(zipos->map + c)) { |   size_t n = GetZipCdirRecords(zipos->cdir); | ||||||
|     npassert(ZIP_CFILE_MAGIC(zipos->map + c) == kZipCfileHdrMagic); |   for (size_t i = 0; i < n; ++i, c += ZIP_CFILE_HDRSIZE(zipos->map + c)) { | ||||||
|     zname = ZIP_CFILE_NAME(zipos->map + c); |     const char *zname = ZIP_CFILE_NAME(zipos->map + c); | ||||||
|     znamesize = ZIP_CFILE_NAMESIZE(zipos->map + c); |     size_t zsize = ZIP_CFILE_NAMESIZE(zipos->map + c); | ||||||
|     if ((name->len == znamesize && !memcmp(name->path, zname, name->len)) || |     if ((name->len == zsize || | ||||||
|         (name->len + 1 == znamesize && !memcmp(name->path, zname, name->len) && |          (name->len + 1 == zsize && zname[name->len] == '/')) && | ||||||
|          zname[name->len] == '/')) { |         !memcmp(name->path, zname, name->len)) { | ||||||
|       return c; |       return c; | ||||||
|     } else if ((name->len < znamesize && |     } else if (name->len + 1 < zsize && zname[name->len] == '/' && | ||||||
|                 !memcmp(name->path, zname, name->len) && |                !memcmp(name->path, zname, name->len)) { | ||||||
|                 zname[name->len - 1] == '/') || |       found_subfile = true; | ||||||
|                (name->len + 1 < znamesize && |  | ||||||
|                 !memcmp(name->path, zname, name->len) && |  | ||||||
|                 zname[name->len] == '/')) { |  | ||||||
|       return 0; |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |   if (found_subfile) { | ||||||
|  |     return ZIPOS_SYNTHETIC_DIRECTORY; | ||||||
|  |   } | ||||||
|   return -1; |   return -1; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -16,9 +16,6 @@ | ||||||
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ | │ PERFORMANCE OF THIS SOFTWARE.                                                │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/assert.h" |  | ||||||
| #include "libc/sysv/errfuns.h" |  | ||||||
| #include "libc/zip.internal.h" |  | ||||||
| #include "libc/runtime/zipos.internal.h" | #include "libc/runtime/zipos.internal.h" | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  | @ -27,12 +24,6 @@ | ||||||
|  * @param uri is obtained via __zipos_parseuri() |  * @param uri is obtained via __zipos_parseuri() | ||||||
|  * @asyncsignalsafe |  * @asyncsignalsafe | ||||||
|  */ |  */ | ||||||
| int __zipos_fstat(const struct ZiposHandle *h, struct stat *st) { | int __zipos_fstat(struct ZiposHandle *h, struct stat *st) { | ||||||
|   int rc; |   return __zipos_stat_impl(h->zipos, h->cfile, st); | ||||||
|   if (st) { |  | ||||||
|     rc = __zipos_stat_impl(__zipos_get(), h->cfile, st); |  | ||||||
|   } else { |  | ||||||
|     rc = efault(); |  | ||||||
|   } |  | ||||||
|   return rc; |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -20,16 +20,17 @@ | ||||||
| #include "libc/calls/metalfile.internal.h" | #include "libc/calls/metalfile.internal.h" | ||||||
| #include "libc/fmt/conv.h" | #include "libc/fmt/conv.h" | ||||||
| #include "libc/intrin/cmpxchg.h" | #include "libc/intrin/cmpxchg.h" | ||||||
|  | #include "libc/intrin/kprintf.h" | ||||||
| #include "libc/intrin/promises.internal.h" | #include "libc/intrin/promises.internal.h" | ||||||
| #include "libc/intrin/strace.internal.h" | #include "libc/intrin/strace.internal.h" | ||||||
| #include "libc/macros.internal.h" | #include "libc/macros.internal.h" | ||||||
| #include "libc/runtime/runtime.h" | #include "libc/runtime/runtime.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/sysv/consts/map.h" | #include "libc/sysv/consts/map.h" | ||||||
| #include "libc/sysv/consts/o.h" | #include "libc/sysv/consts/o.h" | ||||||
| #include "libc/sysv/consts/prot.h" | #include "libc/sysv/consts/prot.h" | ||||||
| #include "libc/thread/thread.h" | #include "libc/thread/thread.h" | ||||||
| #include "libc/zip.internal.h" | #include "libc/zip.internal.h" | ||||||
| #include "libc/runtime/zipos.internal.h" |  | ||||||
| 
 | 
 | ||||||
| #ifdef __x86_64__ | #ifdef __x86_64__ | ||||||
| __static_yoink(APE_COM_NAME); | __static_yoink(APE_COM_NAME); | ||||||
|  | @ -81,7 +82,9 @@ struct Zipos *__zipos_get(void) { | ||||||
|     } |     } | ||||||
|     if (fd != -1 || PLEDGED(RPATH)) { |     if (fd != -1 || PLEDGED(RPATH)) { | ||||||
|       if (fd == -1) { |       if (fd == -1) { | ||||||
|         progpath = GetProgramExecutableName(); |         if (!progpath) { | ||||||
|  |           progpath = GetProgramExecutableName(); | ||||||
|  |         } | ||||||
|         fd = open(progpath, O_RDONLY); |         fd = open(progpath, O_RDONLY); | ||||||
|       } |       } | ||||||
|       if (fd != -1) { |       if (fd != -1) { | ||||||
|  |  | ||||||
|  | @ -17,10 +17,39 @@ | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ | │ PERFORMANCE OF THIS SOFTWARE.                                                │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/calls/calls.h" | #include "libc/calls/calls.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
|  | #include "libc/stdckdint.h" | ||||||
|  | #include "libc/sysv/consts/s.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/thread/thread.h" | #include "libc/thread/thread.h" | ||||||
| #include "libc/zip.internal.h" | #include "libc/zip.internal.h" | ||||||
| #include "libc/runtime/zipos.internal.h" | 
 | ||||||
|  | static int64_t __zipos_lseek_impl(struct ZiposHandle *h, int64_t offset, | ||||||
|  |                                   unsigned whence) { | ||||||
|  |   int64_t pos; | ||||||
|  |   if (h->cfile == ZIPOS_SYNTHETIC_DIRECTORY || | ||||||
|  |       S_ISDIR(GetZipCfileMode(h->zipos->map + h->cfile))) { | ||||||
|  |     return eisdir(); | ||||||
|  |   } | ||||||
|  |   switch (whence) { | ||||||
|  |     case SEEK_SET: | ||||||
|  |       return offset; | ||||||
|  |     case SEEK_CUR: | ||||||
|  |       if (!ckd_add(&pos, h->pos, offset)) { | ||||||
|  |         return pos; | ||||||
|  |       } else { | ||||||
|  |         return eoverflow(); | ||||||
|  |       } | ||||||
|  |     case SEEK_END: | ||||||
|  |       if (!ckd_sub(&pos, h->size, offset)) { | ||||||
|  |         return pos; | ||||||
|  |       } else { | ||||||
|  |         return eoverflow(); | ||||||
|  |       } | ||||||
|  |     default: | ||||||
|  |       return einval(); | ||||||
|  |   } | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Changes current position of zip file handle. |  * Changes current position of zip file handle. | ||||||
|  | @ -31,27 +60,12 @@ | ||||||
|  * @asyncsignalsafe |  * @asyncsignalsafe | ||||||
|  */ |  */ | ||||||
| int64_t __zipos_lseek(struct ZiposHandle *h, int64_t offset, unsigned whence) { | int64_t __zipos_lseek(struct ZiposHandle *h, int64_t offset, unsigned whence) { | ||||||
|   int64_t rc; |   int64_t pos; | ||||||
|  |   if (offset < 0) return einval(); | ||||||
|   pthread_mutex_lock(&h->lock); |   pthread_mutex_lock(&h->lock); | ||||||
|   switch (whence) { |   if ((pos = __zipos_lseek_impl(h, offset, whence)) != -1) { | ||||||
|     case SEEK_SET: |     h->pos = pos; | ||||||
|       rc = offset; |  | ||||||
|       break; |  | ||||||
|     case SEEK_CUR: |  | ||||||
|       rc = h->pos + offset; |  | ||||||
|       break; |  | ||||||
|     case SEEK_END: |  | ||||||
|       rc = h->size - offset; |  | ||||||
|       break; |  | ||||||
|     default: |  | ||||||
|       rc = -1; |  | ||||||
|       break; |  | ||||||
|   } |  | ||||||
|   if (rc >= 0) { |  | ||||||
|     h->pos = rc; |  | ||||||
|   } else { |  | ||||||
|     rc = einval(); |  | ||||||
|   } |   } | ||||||
|   pthread_mutex_unlock(&h->lock); |   pthread_mutex_unlock(&h->lock); | ||||||
|   return rc; |   return pos; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -20,14 +20,15 @@ | ||||||
| #include "libc/calls/struct/iovec.h" | #include "libc/calls/struct/iovec.h" | ||||||
| #include "libc/dce.h" | #include "libc/dce.h" | ||||||
| #include "libc/errno.h" | #include "libc/errno.h" | ||||||
| #include "libc/intrin/likely.h" |  | ||||||
| #include "libc/intrin/strace.internal.h" | #include "libc/intrin/strace.internal.h" | ||||||
| #include "libc/runtime/internal.h" | #include "libc/runtime/internal.h" | ||||||
| #include "libc/runtime/runtime.h" | #include "libc/runtime/runtime.h" | ||||||
| #include "libc/runtime/zipos.internal.h" | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/sysv/consts/map.h" | #include "libc/sysv/consts/map.h" | ||||||
| #include "libc/sysv/consts/prot.h" | #include "libc/sysv/consts/prot.h" | ||||||
|  | #include "libc/sysv/consts/s.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
|  | #include "libc/zip.internal.h" | ||||||
| 
 | 
 | ||||||
| #define IP(X)  (intptr_t)(X) | #define IP(X)  (intptr_t)(X) | ||||||
| #define VIP(X) (void *)IP(X) | #define VIP(X) (void *)IP(X) | ||||||
|  | @ -50,25 +51,36 @@ | ||||||
|  */ |  */ | ||||||
| dontasan void *__zipos_mmap(void *addr, size_t size, int prot, int flags, | dontasan void *__zipos_mmap(void *addr, size_t size, int prot, int flags, | ||||||
|                             struct ZiposHandle *h, int64_t off) { |                             struct ZiposHandle *h, int64_t off) { | ||||||
|   if (!(flags & MAP_PRIVATE) || | 
 | ||||||
|       (flags & ~(MAP_PRIVATE | MAP_FILE | MAP_FIXED | MAP_FIXED_NOREPLACE)) || |   if (off < 0) { | ||||||
|       (!!(flags & MAP_FIXED) ^ !!(flags & MAP_FIXED_NOREPLACE))) { |     STRACE("negative zipos mmap offset"); | ||||||
|     STRACE( |  | ||||||
|         "zipos mappings currently only support MAP_PRIVATE with select flags"); |  | ||||||
|     return VIP(einval()); |     return VIP(einval()); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if (VERY_UNLIKELY(off < 0)) { |   if (h->cfile == ZIPOS_SYNTHETIC_DIRECTORY || | ||||||
|     STRACE("neg off"); |       S_ISDIR(GetZipCfileMode(h->zipos->map + h->cfile))) { | ||||||
|  |     return VIP(eisdir()); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   if (flags & (MAP_SHARED | MAP_ANONYMOUS)) { | ||||||
|  |     STRACE("ZipOS bad flags"); | ||||||
|     return VIP(einval()); |     return VIP(einval()); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   if (prot & ~(PROT_READ | PROT_WRITE | PROT_EXEC)) { | ||||||
|  |     STRACE("ZipOS bad protection"); | ||||||
|  |     return VIP(einval()); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   flags &= MAP_FIXED | MAP_FIXED_NOREPLACE; | ||||||
|  |   flags |= MAP_PRIVATE | MAP_ANONYMOUS; | ||||||
|  | 
 | ||||||
|   const int tempProt = !IsXnu() ? prot | PROT_WRITE : PROT_WRITE; |   const int tempProt = !IsXnu() ? prot | PROT_WRITE : PROT_WRITE; | ||||||
|   void *outAddr = __mmap_unlocked(addr, size, tempProt, |   void *outAddr = __mmap_unlocked(addr, size, tempProt, flags, -1, 0); | ||||||
|                                   (flags & (~MAP_FILE)) | MAP_ANONYMOUS, -1, 0); |  | ||||||
|   if (outAddr == MAP_FAILED) { |   if (outAddr == MAP_FAILED) { | ||||||
|     return MAP_FAILED; |     return MAP_FAILED; | ||||||
|   } |   } | ||||||
|  | 
 | ||||||
|   do { |   do { | ||||||
|     if (__zipos_read(h, &(struct iovec){outAddr, size}, 1, off) == -1) { |     if (__zipos_read(h, &(struct iovec){outAddr, size}, 1, off) == -1) { | ||||||
|       strace_enabled(-1); |       strace_enabled(-1); | ||||||
|  | @ -82,6 +94,7 @@ dontasan void *__zipos_mmap(void *addr, size_t size, int prot, int flags, | ||||||
|     } |     } | ||||||
|     return outAddr; |     return outAddr; | ||||||
|   } while (0); |   } while (0); | ||||||
|  | 
 | ||||||
|   const int e = errno; |   const int e = errno; | ||||||
|   __munmap_unlocked(outAddr, size); |   __munmap_unlocked(outAddr, size); | ||||||
|   errno = e; |   errno = e; | ||||||
|  |  | ||||||
|  | @ -1,7 +1,7 @@ | ||||||
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | /*-*- 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│ | │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||||
| │ Copyright 2022 Justine Alexandra Roberts Tunney                              │ | │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||||
| │                                                                              │ | │                                                                              │ | ||||||
| │ Permission to use, copy, modify, and/or distribute this software for         │ | │ Permission to use, copy, modify, and/or distribute this software for         │ | ||||||
| │ any purpose with or without fee is hereby granted, provided that the         │ | │ any purpose with or without fee is hereby granted, provided that the         │ | ||||||
|  | @ -16,22 +16,65 @@ | ||||||
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ | │ PERFORMANCE OF THIS SOFTWARE.                                                │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/dce.h" |  | ||||||
| #include "libc/intrin/asan.internal.h" |  | ||||||
| #include "libc/intrin/asancodes.h" |  | ||||||
| #include "libc/intrin/cmpxchg.h" |  | ||||||
| #include "libc/str/str.h" |  | ||||||
| #include "libc/thread/thread.h" |  | ||||||
| #include "libc/runtime/zipos.internal.h" | #include "libc/runtime/zipos.internal.h" | ||||||
| 
 | 
 | ||||||
| void __zipos_free(struct Zipos *z, struct ZiposHandle *h) { | static size_t __zipos_trimpath(char *s, int *isabs) { | ||||||
|   if (IsAsan()) { |   char *p = s, *q = s; | ||||||
|     __asan_poison((char *)h + sizeof(struct ZiposHandle), |   for (; *q; ++q) { | ||||||
|                   h->mapsize - sizeof(struct ZiposHandle), kAsanHeapFree); |     if (*q == '/') { | ||||||
|  |       while (q[1] == '/') ++q; | ||||||
|  |       if (q[1] == '.' && (q[2] == '/' || q[2] == '\0')) { | ||||||
|  |         ++q; | ||||||
|  |       } else { | ||||||
|  |         *p++ = '/'; | ||||||
|  |       } | ||||||
|  |     } else { | ||||||
|  |       *p++ = *q; | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|   pthread_mutex_destroy(&h->lock); |   if (s < p && p[-1] == '.' && p[-2] == '.' && (p - 2 == s || p[-3] == '/')) { | ||||||
|   __zipos_lock(); |     *p++ = '/'; | ||||||
|   do h->next = z->freelist; |   } | ||||||
|   while (!_cmpxchg(&z->freelist, h->next, h)); |   *p = '\0'; | ||||||
|   __zipos_unlock(); |   if (isabs) { | ||||||
|  |     *isabs = *s == '/'; | ||||||
|  |   } | ||||||
|  |   return p - s; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | size_t __zipos_normpath(char *s) { | ||||||
|  |   int isabs; | ||||||
|  |   char *p = s, *q = s; | ||||||
|  |   __zipos_trimpath(s, &isabs); | ||||||
|  |   if (!*s) return 0; | ||||||
|  |   for (; *q != '\0'; ++q) { | ||||||
|  |     if (q[0] == '/' && q[1] == '.' && q[2] == '.' && | ||||||
|  |         (q[3] == '/' || q[3] == '\0')) { | ||||||
|  |       char *ep = p; | ||||||
|  |       while (s < ep && *--ep != '/') donothing; | ||||||
|  |       if (ep != p && | ||||||
|  |           (p[-1] != '.' || p[-2] != '.' || (s < p - 3 && p[-3] != '/'))) { | ||||||
|  |         p = ep; | ||||||
|  |         q += 2; | ||||||
|  |         continue; | ||||||
|  |       } else if (ep == s && isabs) { | ||||||
|  |         q += 2; | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (q[0] != '/' || p != s || isabs) { | ||||||
|  |       *p++ = *q; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   if (p == s) { | ||||||
|  |     *p++ = isabs ? '/' : '.'; | ||||||
|  |   } | ||||||
|  |   if (p == s + 1 && s[0] == '.') { | ||||||
|  |     *p++ = '/'; | ||||||
|  |   } | ||||||
|  |   while (p - s > 1 && p[-1] == '/') { | ||||||
|  |     --p; | ||||||
|  |   } | ||||||
|  |   *p = '\0'; | ||||||
|  |   return p - s; | ||||||
| } | } | ||||||
|  | @ -23,6 +23,7 @@ | ||||||
| #include "libc/calls/state.internal.h" | #include "libc/calls/state.internal.h" | ||||||
| #include "libc/calls/struct/sigset.h" | #include "libc/calls/struct/sigset.h" | ||||||
| #include "libc/calls/syscall-sysv.internal.h" | #include "libc/calls/syscall-sysv.internal.h" | ||||||
|  | #include "libc/calls/syscall_support-sysv.internal.h" | ||||||
| #include "libc/dce.h" | #include "libc/dce.h" | ||||||
| #include "libc/errno.h" | #include "libc/errno.h" | ||||||
| #include "libc/intrin/asan.internal.h" | #include "libc/intrin/asan.internal.h" | ||||||
|  | @ -30,19 +31,21 @@ | ||||||
| #include "libc/intrin/cmpxchg.h" | #include "libc/intrin/cmpxchg.h" | ||||||
| #include "libc/intrin/directmap.internal.h" | #include "libc/intrin/directmap.internal.h" | ||||||
| #include "libc/intrin/extend.internal.h" | #include "libc/intrin/extend.internal.h" | ||||||
|  | #include "libc/intrin/strace.internal.h" | ||||||
| #include "libc/intrin/weaken.h" | #include "libc/intrin/weaken.h" | ||||||
| #include "libc/runtime/internal.h" | #include "libc/runtime/internal.h" | ||||||
| #include "libc/runtime/memtrack.internal.h" | #include "libc/runtime/memtrack.internal.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/sysv/consts/f.h" | #include "libc/sysv/consts/f.h" | ||||||
| #include "libc/sysv/consts/fd.h" | #include "libc/sysv/consts/fd.h" | ||||||
| #include "libc/sysv/consts/map.h" | #include "libc/sysv/consts/map.h" | ||||||
| #include "libc/sysv/consts/o.h" | #include "libc/sysv/consts/o.h" | ||||||
| #include "libc/sysv/consts/prot.h" | #include "libc/sysv/consts/prot.h" | ||||||
|  | #include "libc/sysv/consts/s.h" | ||||||
| #include "libc/sysv/consts/sig.h" | #include "libc/sysv/consts/sig.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/thread/thread.h" | #include "libc/thread/thread.h" | ||||||
| #include "libc/zip.internal.h" | #include "libc/zip.internal.h" | ||||||
| #include "libc/runtime/zipos.internal.h" |  | ||||||
| 
 | 
 | ||||||
| static char *mapend; | static char *mapend; | ||||||
| static size_t maptotal; | static size_t maptotal; | ||||||
|  | @ -60,6 +63,18 @@ static void *__zipos_mmap_space(size_t mapsize) { | ||||||
|   return start + offset; |   return start + offset; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void __zipos_free(struct ZiposHandle *h) { | ||||||
|  |   if (IsAsan()) { | ||||||
|  |     __asan_poison((char *)h + sizeof(struct ZiposHandle), | ||||||
|  |                   h->mapsize - sizeof(struct ZiposHandle), kAsanHeapFree); | ||||||
|  |   } | ||||||
|  |   pthread_mutex_destroy(&h->lock); | ||||||
|  |   __zipos_lock(); | ||||||
|  |   do h->next = h->zipos->freelist; | ||||||
|  |   while (!_cmpxchg(&h->zipos->freelist, h->next, h)); | ||||||
|  |   __zipos_unlock(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static struct ZiposHandle *__zipos_alloc(struct Zipos *zipos, size_t size) { | static struct ZiposHandle *__zipos_alloc(struct Zipos *zipos, size_t size) { | ||||||
|   size_t mapsize; |   size_t mapsize; | ||||||
|   struct ZiposHandle *h, **ph; |   struct ZiposHandle *h, **ph; | ||||||
|  | @ -88,6 +103,7 @@ StartOver: | ||||||
|   } |   } | ||||||
|   if (h) { |   if (h) { | ||||||
|     h->size = size; |     h->size = size; | ||||||
|  |     h->zipos = zipos; | ||||||
|     h->mapsize = mapsize; |     h->mapsize = mapsize; | ||||||
|     pthread_mutex_init(&h->lock, 0); |     pthread_mutex_init(&h->lock, 0); | ||||||
|   } |   } | ||||||
|  | @ -95,65 +111,63 @@ StartOver: | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int __zipos_mkfd(int minfd) { | static int __zipos_mkfd(int minfd) { | ||||||
|   int e, fd; |   int fd, e = errno; | ||||||
|   static bool demodernize; |   if ((fd = __sys_fcntl(2, F_DUPFD_CLOEXEC, minfd)) != -1) { | ||||||
|   if (!demodernize) { |     return fd; | ||||||
|     e = errno; |   } else if (errno == EINVAL) { | ||||||
|     if ((fd = __sys_fcntl(2, F_DUPFD_CLOEXEC, minfd)) != -1) { |     errno = e; | ||||||
|       return fd; |     return __fixupnewfd(__sys_fcntl(2, F_DUPFD, minfd), O_CLOEXEC); | ||||||
|     } else if (errno == EINVAL) { |   } else { | ||||||
|       demodernize = true; |     return fd; | ||||||
|       errno = e; |  | ||||||
|     } else { |  | ||||||
|       return fd; |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|   if ((fd = __sys_fcntl(2, F_DUPFD, minfd)) != -1) { |  | ||||||
|     __sys_fcntl(fd, F_SETFD, FD_CLOEXEC); |  | ||||||
|   } |  | ||||||
|   return fd; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int __zipos_setfd(int fd, struct ZiposHandle *h, unsigned flags, | static int __zipos_setfd(int fd, struct ZiposHandle *h, unsigned flags) { | ||||||
|                          int mode) { |  | ||||||
|   int want = fd; |   int want = fd; | ||||||
|   atomic_compare_exchange_strong_explicit( |   atomic_compare_exchange_strong_explicit( | ||||||
|       &g_fds.f, &want, fd + 1, memory_order_release, memory_order_relaxed); |       &g_fds.f, &want, fd + 1, memory_order_release, memory_order_relaxed); | ||||||
|   g_fds.p[fd].kind = kFdZip; |   g_fds.p[fd].kind = kFdZip; | ||||||
|   g_fds.p[fd].handle = (intptr_t)h; |   g_fds.p[fd].handle = (intptr_t)h; | ||||||
|   g_fds.p[fd].flags = flags | O_CLOEXEC; |   g_fds.p[fd].flags = flags | O_CLOEXEC; | ||||||
|   g_fds.p[fd].mode = mode; |  | ||||||
|   g_fds.p[fd].extra = 0; |   g_fds.p[fd].extra = 0; | ||||||
|   __fds_unlock(); |   __fds_unlock(); | ||||||
|   return fd; |   return fd; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int __zipos_load(struct Zipos *zipos, size_t cf, unsigned flags, | static int __zipos_load(struct Zipos *zipos, size_t cf, int flags, | ||||||
|                         int mode) { |                         struct ZiposUri *name) { | ||||||
|   size_t lf; |   size_t lf; | ||||||
|   size_t size; |   size_t size; | ||||||
|   int rc, fd, minfd; |   int rc, fd, minfd; | ||||||
|   struct ZiposHandle *h; |   struct ZiposHandle *h; | ||||||
|   lf = GetZipCfileOffset(zipos->map + cf); |   if (cf == ZIPOS_SYNTHETIC_DIRECTORY) { | ||||||
|   npassert((ZIP_LFILE_MAGIC(zipos->map + lf) == kZipLfileHdrMagic)); |     size = name->len; | ||||||
|   size = GetZipLfileUncompressedSize(zipos->map + lf); |     if (!(h = __zipos_alloc(zipos, size + 1))) return -1; | ||||||
|   switch (ZIP_LFILE_COMPRESSIONMETHOD(zipos->map + lf)) { |     if (size) memcpy(h->data, name->path, size); | ||||||
|     case kZipCompressionNone: |     h->data[size] = 0; | ||||||
|       if (!(h = __zipos_alloc(zipos, 0))) return -1; |     h->mem = h->data; | ||||||
|       h->mem = ZIP_LFILE_CONTENT(zipos->map + lf); |   } else { | ||||||
|       break; |     lf = GetZipCfileOffset(zipos->map + cf); | ||||||
|     case kZipCompressionDeflate: |     npassert((ZIP_LFILE_MAGIC(zipos->map + lf) == kZipLfileHdrMagic)); | ||||||
|       if (!(h = __zipos_alloc(zipos, size))) return -1; |     size = GetZipLfileUncompressedSize(zipos->map + lf); | ||||||
|       if (!__inflate(h->data, size, ZIP_LFILE_CONTENT(zipos->map + lf), |     switch (ZIP_LFILE_COMPRESSIONMETHOD(zipos->map + lf)) { | ||||||
|                      GetZipLfileCompressedSize(zipos->map + lf))) { |       case kZipCompressionNone: | ||||||
|         h->mem = h->data; |         if (!(h = __zipos_alloc(zipos, 0))) return -1; | ||||||
|       } else { |         h->mem = ZIP_LFILE_CONTENT(zipos->map + lf); | ||||||
|         h->mem = 0; |         break; | ||||||
|         eio(); |       case kZipCompressionDeflate: | ||||||
|       } |         if (!(h = __zipos_alloc(zipos, size))) return -1; | ||||||
|       break; |         if (!__inflate(h->data, size, ZIP_LFILE_CONTENT(zipos->map + lf), | ||||||
|     default: |                        GetZipLfileCompressedSize(zipos->map + lf))) { | ||||||
|       return eio(); |           h->mem = h->data; | ||||||
|  |         } else { | ||||||
|  |           h->mem = 0; | ||||||
|  |           eio(); | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |       default: | ||||||
|  |         return eio(); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|   h->pos = 0; |   h->pos = 0; | ||||||
|   h->cfile = cf; |   h->cfile = cf; | ||||||
|  | @ -164,7 +178,7 @@ static int __zipos_load(struct Zipos *zipos, size_t cf, unsigned flags, | ||||||
|   TryAgain: |   TryAgain: | ||||||
|     if (IsWindows() || IsMetal()) { |     if (IsWindows() || IsMetal()) { | ||||||
|       if ((fd = __reservefd_unlocked(-1)) != -1) { |       if ((fd = __reservefd_unlocked(-1)) != -1) { | ||||||
|         return __zipos_setfd(fd, h, flags, mode); |         return __zipos_setfd(fd, h, flags); | ||||||
|       } |       } | ||||||
|     } else if ((fd = __zipos_mkfd(minfd)) != -1) { |     } else if ((fd = __zipos_mkfd(minfd)) != -1) { | ||||||
|       if (__ensurefds_unlocked(fd) != -1) { |       if (__ensurefds_unlocked(fd) != -1) { | ||||||
|  | @ -173,16 +187,45 @@ static int __zipos_load(struct Zipos *zipos, size_t cf, unsigned flags, | ||||||
|           minfd = fd + 1; |           minfd = fd + 1; | ||||||
|           goto TryAgain; |           goto TryAgain; | ||||||
|         } |         } | ||||||
|         return __zipos_setfd(fd, h, flags, mode); |         return __zipos_setfd(fd, h, flags); | ||||||
|       } |       } | ||||||
|       sys_close(fd); |       sys_close(fd); | ||||||
|     } |     } | ||||||
|     __fds_unlock(); |     __fds_unlock(); | ||||||
|   } |   } | ||||||
|   __zipos_free(zipos, h); |   __zipos_free(h); | ||||||
|   return -1; |   return -1; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static int __zipos_open_impl(struct ZiposUri *name, int flags) { | ||||||
|  |   struct Zipos *zipos; | ||||||
|  |   if (!(zipos = __zipos_get())) { | ||||||
|  |     return enoexec(); | ||||||
|  |   } | ||||||
|  |   ssize_t cf; | ||||||
|  |   if ((cf = __zipos_find(zipos, name)) == -1) { | ||||||
|  |     return enoent(); | ||||||
|  |   } | ||||||
|  |   if ((flags & O_ACCMODE) != O_RDONLY || (flags & O_TRUNC)) { | ||||||
|  |     return erofs(); | ||||||
|  |   } | ||||||
|  |   if (flags & O_EXCL) { | ||||||
|  |     return eexist(); | ||||||
|  |   } | ||||||
|  |   if (cf != ZIPOS_SYNTHETIC_DIRECTORY) { | ||||||
|  |     int mode = GetZipCfileMode(zipos->map + cf); | ||||||
|  | #if 0 | ||||||
|  |     if ((flags & O_DIRECTORY) && !S_ISDIR(mode)) { | ||||||
|  |       return enotdir(); | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  |     if (!(mode & 0444)) { | ||||||
|  |       return eacces(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return __zipos_load(zipos, cf, flags, name); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Loads compressed file from αcτµαlly pδrταblε εxεcµταblε object store. |  * Loads compressed file from αcτµαlly pδrταblε εxεcµταblε object store. | ||||||
|  * |  * | ||||||
|  | @ -190,33 +233,15 @@ static int __zipos_load(struct Zipos *zipos, size_t cf, unsigned flags, | ||||||
|  * @asyncsignalsafe |  * @asyncsignalsafe | ||||||
|  * @threadsafe |  * @threadsafe | ||||||
|  */ |  */ | ||||||
| int __zipos_open(const struct ZiposUri *name, unsigned flags, int mode) { | int __zipos_open(struct ZiposUri *name, int flags) { | ||||||
|   int rc; |   int rc; | ||||||
|   ssize_t cf; |  | ||||||
|   struct Zipos *zipos; |  | ||||||
|   if (_weaken(pthread_testcancel_np) && |   if (_weaken(pthread_testcancel_np) && | ||||||
|       (rc = _weaken(pthread_testcancel_np)())) { |       (rc = _weaken(pthread_testcancel_np)())) { | ||||||
|     errno = rc; |     errno = rc; | ||||||
|     return -1; |     return -1; | ||||||
|   } |   } | ||||||
|   BLOCK_SIGNALS; |   BLOCK_SIGNALS; | ||||||
|   if ((flags & O_ACCMODE) == O_RDONLY) { |   rc = __zipos_open_impl(name, flags); | ||||||
|     if ((zipos = __zipos_get())) { |  | ||||||
|       if ((cf = __zipos_find(zipos, name)) != -1) { |  | ||||||
|         rc = __zipos_load(zipos, cf, flags, mode); |  | ||||||
|         assert(rc != 0); |  | ||||||
|       } else { |  | ||||||
|         rc = enoent(); |  | ||||||
|         assert(rc != 0); |  | ||||||
|       } |  | ||||||
|     } else { |  | ||||||
|       rc = enoexec(); |  | ||||||
|       assert(rc != 0); |  | ||||||
|     } |  | ||||||
|   } else { |  | ||||||
|     rc = einval(); |  | ||||||
|     assert(rc != 0); |  | ||||||
|   } |  | ||||||
|   ALLOW_SIGNALS; |   ALLOW_SIGNALS; | ||||||
|   return rc; |   return rc; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -16,19 +16,21 @@ | ||||||
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ | │ PERFORMANCE OF THIS SOFTWARE.                                                │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/str/str.h" |  | ||||||
| #include "libc/runtime/zipos.internal.h" | #include "libc/runtime/zipos.internal.h" | ||||||
|  | #include "libc/str/str.h" | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Extracts information about ZIP URI if it is one. |  * Extracts information about ZIP URI if it is one. | ||||||
|  */ |  */ | ||||||
| ssize_t __zipos_parseuri(const char *uri, struct ZiposUri *out) { | ssize_t __zipos_parseuri(const char *uri, struct ZiposUri *out) { | ||||||
|   size_t len; |   size_t len; | ||||||
|   if ((uri[0] == '/' && uri[1] == 'z' && uri[2] == 'i' && uri[3] == 'p' && |   if ((uri[0] == '/' &&  //
 | ||||||
|  |        uri[1] == 'z' &&  //
 | ||||||
|  |        uri[2] == 'i' &&  //
 | ||||||
|  |        uri[3] == 'p' &&  //
 | ||||||
|        (!uri[4] || uri[4] == '/')) && |        (!uri[4] || uri[4] == '/')) && | ||||||
|       (len = strlen(uri)) < PATH_MAX) { |       strlcpy(out->path, uri + 4 + !!uri[4], ZIPOS_PATH_MAX) < ZIPOS_PATH_MAX) { | ||||||
|     out->path = uri + 4 + !!uri[4]; |     return (out->len = __zipos_normpath(out->path)); | ||||||
|     return (out->len = len - 4 - !!uri[4]); |  | ||||||
|   } else { |   } else { | ||||||
|     return -1; |     return -1; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -18,11 +18,10 @@ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/assert.h" | #include "libc/assert.h" | ||||||
| #include "libc/calls/struct/iovec.h" | #include "libc/calls/struct/iovec.h" | ||||||
| #include "libc/intrin/safemacros.internal.h" |  | ||||||
| #include "libc/str/str.h" |  | ||||||
| #include "libc/thread/thread.h" |  | ||||||
| #include "libc/zip.internal.h" |  | ||||||
| #include "libc/runtime/zipos.internal.h" | #include "libc/runtime/zipos.internal.h" | ||||||
|  | #include "libc/sysv/consts/s.h" | ||||||
|  | #include "libc/sysv/errfuns.h" | ||||||
|  | #include "libc/zip.internal.h" | ||||||
| 
 | 
 | ||||||
| static size_t GetIovSize(const struct iovec *iov, size_t iovlen) { | static size_t GetIovSize(const struct iovec *iov, size_t iovlen) { | ||||||
|   size_t i, r; |   size_t i, r; | ||||||
|  | @ -30,6 +29,29 @@ static size_t GetIovSize(const struct iovec *iov, size_t iovlen) { | ||||||
|   return r; |   return r; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static ssize_t __zipos_read_impl(struct ZiposHandle *h, const struct iovec *iov, | ||||||
|  |                                  size_t iovlen, ssize_t opt_offset) { | ||||||
|  |   int i; | ||||||
|  |   int64_t b, x, y; | ||||||
|  |   if (h->cfile == ZIPOS_SYNTHETIC_DIRECTORY || | ||||||
|  |       S_ISDIR(GetZipCfileMode(h->zipos->map + h->cfile))) { | ||||||
|  |     return eisdir(); | ||||||
|  |   } | ||||||
|  |   if (opt_offset == -1) { | ||||||
|  |     x = y = h->pos; | ||||||
|  |   } else { | ||||||
|  |     x = y = opt_offset; | ||||||
|  |   } | ||||||
|  |   for (i = 0; i < iovlen && y < h->size; ++i, y += b) { | ||||||
|  |     b = MIN(iov[i].iov_len, h->size - y); | ||||||
|  |     if (b) memcpy(iov[i].iov_base, h->mem + y, b); | ||||||
|  |   } | ||||||
|  |   if (opt_offset == -1) { | ||||||
|  |     h->pos = y; | ||||||
|  |   } | ||||||
|  |   return y - x; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Reads data from zip store object. |  * Reads data from zip store object. | ||||||
|  * |  * | ||||||
|  | @ -39,14 +61,10 @@ static size_t GetIovSize(const struct iovec *iov, size_t iovlen) { | ||||||
|  */ |  */ | ||||||
| ssize_t __zipos_read(struct ZiposHandle *h, const struct iovec *iov, | ssize_t __zipos_read(struct ZiposHandle *h, const struct iovec *iov, | ||||||
|                      size_t iovlen, ssize_t opt_offset) { |                      size_t iovlen, ssize_t opt_offset) { | ||||||
|   size_t i, b, x, y; |   ssize_t rc; | ||||||
|  |   unassert(opt_offset >= 0 || opt_offset == -1); | ||||||
|   pthread_mutex_lock(&h->lock); |   pthread_mutex_lock(&h->lock); | ||||||
|   x = y = opt_offset != -1 ? opt_offset : h->pos; |   rc = __zipos_read_impl(h, iov, iovlen, opt_offset); | ||||||
|   for (i = 0; i < iovlen && y < h->size; ++i, y += b) { |  | ||||||
|     b = min(iov[i].iov_len, h->size - y); |  | ||||||
|     if (b) memcpy(iov[i].iov_base, h->mem + y, b); |  | ||||||
|   } |  | ||||||
|   if (opt_offset == -1) h->pos = y; |  | ||||||
|   pthread_mutex_unlock(&h->lock); |   pthread_mutex_unlock(&h->lock); | ||||||
|   return y - x; |   return rc; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -18,17 +18,19 @@ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/calls/struct/stat.h" | #include "libc/calls/struct/stat.h" | ||||||
| #include "libc/intrin/safemacros.internal.h" | #include "libc/intrin/safemacros.internal.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/str/str.h" | #include "libc/str/str.h" | ||||||
| #include "libc/sysv/consts/s.h" | #include "libc/sysv/consts/s.h" | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/zip.internal.h" | #include "libc/zip.internal.h" | ||||||
| #include "libc/runtime/zipos.internal.h" |  | ||||||
| 
 | 
 | ||||||
| int __zipos_stat_impl(struct Zipos *zipos, size_t cf, struct stat *st) { | int __zipos_stat_impl(struct Zipos *zipos, size_t cf, struct stat *st) { | ||||||
|   size_t lf; |   size_t lf; | ||||||
|   if (zipos && st) { |   if (zipos && st) { | ||||||
|     bzero(st, sizeof(*st)); |     bzero(st, sizeof(*st)); | ||||||
|     if (cf) { |     if (cf == ZIPOS_SYNTHETIC_DIRECTORY) { | ||||||
|  |       st->st_mode = S_IFDIR | 0555; | ||||||
|  |     } else { | ||||||
|       lf = GetZipCfileOffset(zipos->map + cf); |       lf = GetZipCfileOffset(zipos->map + cf); | ||||||
|       st->st_mode = GetZipCfileMode(zipos->map + cf); |       st->st_mode = GetZipCfileMode(zipos->map + cf); | ||||||
|       st->st_size = GetZipLfileUncompressedSize(zipos->map + lf); |       st->st_size = GetZipLfileUncompressedSize(zipos->map + lf); | ||||||
|  | @ -37,8 +39,6 @@ int __zipos_stat_impl(struct Zipos *zipos, size_t cf, struct stat *st) { | ||||||
|       GetZipCfileTimestamps(zipos->map + cf, &st->st_mtim, &st->st_atim, |       GetZipCfileTimestamps(zipos->map + cf, &st->st_mtim, &st->st_atim, | ||||||
|                             &st->st_ctim, 0); |                             &st->st_ctim, 0); | ||||||
|       st->st_birthtim = st->st_ctim; |       st->st_birthtim = st->st_ctim; | ||||||
|     } else { |  | ||||||
|       st->st_mode = 0444 | S_IFDIR | 0111; |  | ||||||
|     } |     } | ||||||
|     return 0; |     return 0; | ||||||
|   } else { |   } else { | ||||||
|  |  | ||||||
|  | @ -16,9 +16,8 @@ | ||||||
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │ | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ | │ PERFORMANCE OF THIS SOFTWARE.                                                │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/stdio/stdio.h" |  | ||||||
| #include "libc/sysv/errfuns.h" |  | ||||||
| #include "libc/runtime/zipos.internal.h" | #include "libc/runtime/zipos.internal.h" | ||||||
|  | #include "libc/sysv/errfuns.h" | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * Reads file metadata from αcτµαlly pδrταblε εxεcµταblε object store. |  * Reads file metadata from αcτµαlly pδrταblε εxεcµταblε object store. | ||||||
|  | @ -26,22 +25,10 @@ | ||||||
|  * @param uri is obtained via __zipos_parseuri() |  * @param uri is obtained via __zipos_parseuri() | ||||||
|  * @asyncsignalsafe |  * @asyncsignalsafe | ||||||
|  */ |  */ | ||||||
| int __zipos_stat(const struct ZiposUri *name, struct stat *st) { | int __zipos_stat(struct ZiposUri *name, struct stat *st) { | ||||||
|   int rc; |  | ||||||
|   ssize_t cf; |   ssize_t cf; | ||||||
|   struct Zipos *zipos; |   struct Zipos *zipos; | ||||||
|   if (st) { |   if (!(zipos = __zipos_get())) return enoexec(); | ||||||
|     if ((zipos = __zipos_get())) { |   if ((cf = __zipos_find(zipos, name)) == -1) return enoent(); | ||||||
|       if ((cf = __zipos_find(zipos, name)) != -1) { |   return __zipos_stat_impl(zipos, cf, st); | ||||||
|         rc = __zipos_stat_impl(zipos, cf, st); |  | ||||||
|       } else { |  | ||||||
|         rc = enoent(); |  | ||||||
|       } |  | ||||||
|     } else { |  | ||||||
|       rc = enoexec(); |  | ||||||
|     } |  | ||||||
|   } else { |  | ||||||
|     rc = efault(); |  | ||||||
|   } |  | ||||||
|   return rc; |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -6,23 +6,29 @@ | ||||||
| #if !(__ASSEMBLER__ + __LINKER__ + 0) | #if !(__ASSEMBLER__ + __LINKER__ + 0) | ||||||
| COSMOPOLITAN_C_START_ | COSMOPOLITAN_C_START_ | ||||||
| 
 | 
 | ||||||
|  | #define ZIPOS_PATH_MAX 1024 | ||||||
|  | 
 | ||||||
|  | #define ZIPOS_SYNTHETIC_DIRECTORY 0 | ||||||
|  | 
 | ||||||
| struct stat; | struct stat; | ||||||
| struct iovec; | struct iovec; | ||||||
|  | struct Zipos; | ||||||
| 
 | 
 | ||||||
| struct ZiposUri { | struct ZiposUri { | ||||||
|   const char *path; |   uint32_t len; | ||||||
|   size_t len; |   char path[ZIPOS_PATH_MAX]; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct ZiposHandle { | struct ZiposHandle { | ||||||
|   struct ZiposHandle *next; |   struct ZiposHandle *next; | ||||||
|   pthread_mutex_t lock; |   pthread_mutex_t lock; | ||||||
|   size_t size;    /* byte length of `mem` */ |   struct Zipos *zipos; | ||||||
|   size_t mapsize; /* total size of this struct */ |   size_t size; | ||||||
|   size_t pos;     /* read/write byte offset state */ |   size_t mapsize; | ||||||
|   uint32_t cfile; /* central directory entry rva */ |   size_t pos; | ||||||
|   uint8_t *mem;   /* points to inflated data or uncompressed image */ |   size_t cfile; | ||||||
|   uint8_t data[]; /* uncompressed file memory */ |   uint8_t *mem; | ||||||
|  |   uint8_t data[]; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct Zipos { | struct Zipos { | ||||||
|  | @ -31,17 +37,18 @@ struct Zipos { | ||||||
|   struct ZiposHandle *freelist; |   struct ZiposHandle *freelist; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | int __zipos_close(int); | ||||||
| void __zipos_lock(void); | void __zipos_lock(void); | ||||||
| void __zipos_unlock(void); | void __zipos_unlock(void); | ||||||
| int __zipos_close(int); | size_t __zipos_normpath(char *); | ||||||
| struct Zipos *__zipos_get(void) pureconst; | struct Zipos *__zipos_get(void) pureconst; | ||||||
| void __zipos_free(struct Zipos *, struct ZiposHandle *); | void __zipos_free(struct ZiposHandle *); | ||||||
| ssize_t __zipos_parseuri(const char *, struct ZiposUri *); | ssize_t __zipos_parseuri(const char *, struct ZiposUri *); | ||||||
| ssize_t __zipos_find(struct Zipos *, const struct ZiposUri *); | ssize_t __zipos_find(struct Zipos *, struct ZiposUri *); | ||||||
| int __zipos_open(const struct ZiposUri *, unsigned, int); | int __zipos_open(struct ZiposUri *, int); | ||||||
| int __zipos_access(const struct ZiposUri *, int); | int __zipos_access(struct ZiposUri *, int); | ||||||
| int __zipos_stat(const struct ZiposUri *, struct stat *); | int __zipos_stat(struct ZiposUri *, struct stat *); | ||||||
| int __zipos_fstat(const struct ZiposHandle *, struct stat *); | int __zipos_fstat(struct ZiposHandle *, struct stat *); | ||||||
| int __zipos_stat_impl(struct Zipos *, size_t, struct stat *); | int __zipos_stat_impl(struct Zipos *, size_t, struct stat *); | ||||||
| ssize_t __zipos_read(struct ZiposHandle *, const struct iovec *, size_t, | ssize_t __zipos_read(struct ZiposHandle *, const struct iovec *, size_t, | ||||||
|                      ssize_t); |                      ssize_t); | ||||||
|  |  | ||||||
|  | @ -22,8 +22,9 @@ | ||||||
| #include "libc/calls/struct/dirent.h" | #include "libc/calls/struct/dirent.h" | ||||||
| #include "libc/calls/struct/stat.h" | #include "libc/calls/struct/stat.h" | ||||||
| #include "libc/calls/syscall_support-nt.internal.h" | #include "libc/calls/syscall_support-nt.internal.h" | ||||||
|  | #include "libc/dce.h" | ||||||
| #include "libc/errno.h" | #include "libc/errno.h" | ||||||
| #include "libc/intrin/asan.internal.h" | #include "libc/intrin/kprintf.h" | ||||||
| #include "libc/intrin/nopl.internal.h" | #include "libc/intrin/nopl.internal.h" | ||||||
| #include "libc/intrin/strace.internal.h" | #include "libc/intrin/strace.internal.h" | ||||||
| #include "libc/intrin/weaken.h" | #include "libc/intrin/weaken.h" | ||||||
|  | @ -32,6 +33,7 @@ | ||||||
| #include "libc/nt/enum/filetype.h" | #include "libc/nt/enum/filetype.h" | ||||||
| #include "libc/nt/files.h" | #include "libc/nt/files.h" | ||||||
| #include "libc/nt/struct/win32finddata.h" | #include "libc/nt/struct/win32finddata.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/str/str.h" | #include "libc/str/str.h" | ||||||
| #include "libc/sysv/consts/dt.h" | #include "libc/sysv/consts/dt.h" | ||||||
| #include "libc/sysv/consts/o.h" | #include "libc/sysv/consts/o.h" | ||||||
|  | @ -39,7 +41,6 @@ | ||||||
| #include "libc/sysv/errfuns.h" | #include "libc/sysv/errfuns.h" | ||||||
| #include "libc/thread/thread.h" | #include "libc/thread/thread.h" | ||||||
| #include "libc/zip.internal.h" | #include "libc/zip.internal.h" | ||||||
| #include "libc/runtime/zipos.internal.h" |  | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  * @fileoverview Directory Streams for Linux+Mac+Windows+FreeBSD+OpenBSD. |  * @fileoverview Directory Streams for Linux+Mac+Windows+FreeBSD+OpenBSD. | ||||||
|  | @ -57,15 +58,17 @@ int sys_getdents(unsigned, void *, unsigned, long *); | ||||||
|  * Directory stream object. |  * Directory stream object. | ||||||
|  */ |  */ | ||||||
| struct dirstream { | struct dirstream { | ||||||
|  |   int fd; | ||||||
|   bool iszip; |   bool iszip; | ||||||
|   int64_t fd; |   int64_t hand; | ||||||
|   int64_t tell; |   int64_t tell; | ||||||
|  |   char16_t *name; | ||||||
|   pthread_mutex_t lock; |   pthread_mutex_t lock; | ||||||
|   struct { |   struct { | ||||||
|     uint64_t offset; |     uint64_t offset; | ||||||
|     uint64_t records; |     uint64_t records; | ||||||
|     uint8_t *prefix; |  | ||||||
|     size_t prefixlen; |     size_t prefixlen; | ||||||
|  |     uint8_t prefix[ZIPOS_PATH_MAX]; | ||||||
|   } zip; |   } zip; | ||||||
|   struct dirent ent; |   struct dirent ent; | ||||||
|   union { |   union { | ||||||
|  | @ -76,7 +79,6 @@ struct dirstream { | ||||||
|     }; |     }; | ||||||
|     struct { |     struct { | ||||||
|       bool isdone; |       bool isdone; | ||||||
|       char16_t *name; |  | ||||||
|       struct NtWin32FindData windata; |       struct NtWin32FindData windata; | ||||||
|     }; |     }; | ||||||
|   }; |   }; | ||||||
|  | @ -148,7 +150,7 @@ static textwindows DIR *opendir_nt_impl(char16_t *name, size_t len) { | ||||||
|     } |     } | ||||||
|     name[len] = u'\0'; |     name[len] = u'\0'; | ||||||
|     if ((res = calloc(1, sizeof(DIR)))) { |     if ((res = calloc(1, sizeof(DIR)))) { | ||||||
|       if ((res->fd = FindFirstFile(name, &res->windata)) != -1) { |       if ((res->hand = FindFirstFile(name, &res->windata)) != -1) { | ||||||
|         return res; |         return res; | ||||||
|       } |       } | ||||||
|       __fix_enotdir(-1, name); |       __fix_enotdir(-1, name); | ||||||
|  | @ -160,35 +162,17 @@ static textwindows DIR *opendir_nt_impl(char16_t *name, size_t len) { | ||||||
|   return NULL; |   return NULL; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static textwindows dontinline DIR *opendir_nt(const char *path) { |  | ||||||
|   int len; |  | ||||||
|   DIR *res; |  | ||||||
|   char16_t *name; |  | ||||||
|   if (*path) { |  | ||||||
|     if ((name = malloc(PATH_MAX * 2))) { |  | ||||||
|       if ((len = __mkntpath(path, name)) != -1 && |  | ||||||
|           (res = opendir_nt_impl(name, len))) { |  | ||||||
|         res->name = name; |  | ||||||
|         return res; |  | ||||||
|       } |  | ||||||
|       free(name); |  | ||||||
|     } |  | ||||||
|   } else { |  | ||||||
|     enoent(); |  | ||||||
|   } |  | ||||||
|   return NULL; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static textwindows dontinline DIR *fdopendir_nt(int fd) { | static textwindows dontinline DIR *fdopendir_nt(int fd) { | ||||||
|   DIR *res; |   DIR *res; | ||||||
|   char16_t *name; |   char16_t *name; | ||||||
|   if (__isfdkind(fd, kFdFile)) { |   if (__isfdkind(fd, kFdFile)) { | ||||||
|     if ((name = malloc(PATH_MAX * 2))) { |     if ((name = calloc(1, PATH_MAX * 2))) { | ||||||
|       if ((res = opendir_nt_impl( |       if ((res = opendir_nt_impl( | ||||||
|                name, GetFinalPathNameByHandle( |                name, GetFinalPathNameByHandle( | ||||||
|                          g_fds.p[fd].handle, name, PATH_MAX, |                          g_fds.p[fd].handle, name, PATH_MAX, | ||||||
|                          kNtFileNameNormalized | kNtVolumeNameDos)))) { |                          kNtFileNameNormalized | kNtVolumeNameDos)))) { | ||||||
|         res->name = name; |         res->name = name; | ||||||
|  |         res->fd = -1; | ||||||
|         close(fd); |         close(fd); | ||||||
|         return res; |         return res; | ||||||
|       } |       } | ||||||
|  | @ -235,86 +219,13 @@ static textwindows dontinline struct dirent *readdir_nt(DIR *dir) { | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     dir->ent.d_type = GetNtDirentType(&dir->windata); |     dir->ent.d_type = GetNtDirentType(&dir->windata); | ||||||
|     dir->isdone = !FindNextFile(dir->fd, &dir->windata); |     dir->isdone = !FindNextFile(dir->hand, &dir->windata); | ||||||
|     return &dir->ent; |     return &dir->ent; | ||||||
|   } else { |   } else { | ||||||
|     return NULL; |     return NULL; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 |  | ||||||
|  * Opens directory, e.g. |  | ||||||
|  * |  | ||||||
|  *     DIR *d; |  | ||||||
|  *     struct dirent *e; |  | ||||||
|  *     CHECK((d = opendir(path))); |  | ||||||
|  *     while ((e = readdir(d))) { |  | ||||||
|  *       printf("%s/%s\n", path, e->d_name); |  | ||||||
|  *     } |  | ||||||
|  *     LOGIFNEG1(closedir(d)); |  | ||||||
|  * |  | ||||||
|  * @returns newly allocated DIR object, or NULL w/ errno |  | ||||||
|  * @errors ENOENT, ENOTDIR, EACCES, EMFILE, ENFILE, ENOMEM |  | ||||||
|  * @raise ECANCELED if thread was cancelled in masked mode |  | ||||||
|  * @raise EINTR if we needed to block and a signal was delivered instead |  | ||||||
|  * @cancellationpoint |  | ||||||
|  * @see glob() |  | ||||||
|  */ |  | ||||||
| DIR *opendir(const char *name) { |  | ||||||
|   DIR *res; |  | ||||||
|   int fd, rc; |  | ||||||
|   struct stat st; |  | ||||||
|   struct Zipos *zip; |  | ||||||
|   struct ZiposUri zipname; |  | ||||||
|   if (_weaken(pthread_testcancel_np) && |  | ||||||
|       (rc = _weaken(pthread_testcancel_np)())) { |  | ||||||
|     errno = rc; |  | ||||||
|     return 0; |  | ||||||
|   } |  | ||||||
|   if (!name || (IsAsan() && !__asan_is_valid_str(name))) { |  | ||||||
|     efault(); |  | ||||||
|     res = 0; |  | ||||||
|   } else if (_weaken(__zipos_get) && |  | ||||||
|              _weaken(__zipos_parseuri)(name, &zipname) != -1) { |  | ||||||
|     if (_weaken(__zipos_stat)(&zipname, &st) != -1) { |  | ||||||
|       if (S_ISDIR(st.st_mode)) { |  | ||||||
|         zip = _weaken(__zipos_get)(); |  | ||||||
|         if ((res = calloc(1, sizeof(DIR)))) { |  | ||||||
|           res->iszip = true; |  | ||||||
|           res->fd = -1; |  | ||||||
|           res->zip.offset = GetZipCdirOffset(zip->cdir); |  | ||||||
|           res->zip.records = GetZipCdirRecords(zip->cdir); |  | ||||||
|           res->zip.prefix = malloc(zipname.len + 2); |  | ||||||
|           if (zipname.len) { |  | ||||||
|             memcpy(res->zip.prefix, zipname.path, zipname.len); |  | ||||||
|           } |  | ||||||
|           if (zipname.len && res->zip.prefix[zipname.len - 1] != '/') { |  | ||||||
|             res->zip.prefix[zipname.len++] = '/'; |  | ||||||
|           } |  | ||||||
|           res->zip.prefix[zipname.len] = '\0'; |  | ||||||
|           res->zip.prefixlen = zipname.len; |  | ||||||
|         } |  | ||||||
|       } else { |  | ||||||
|         enotdir(); |  | ||||||
|         res = 0; |  | ||||||
|       } |  | ||||||
|     } else { |  | ||||||
|       res = 0; |  | ||||||
|     } |  | ||||||
|   } else if (!IsWindows()) { |  | ||||||
|     res = 0; |  | ||||||
|     if ((fd = open(name, O_RDONLY | O_NOCTTY | O_DIRECTORY | O_CLOEXEC)) != |  | ||||||
|         -1) { |  | ||||||
|       if (!(res = fdopendir(fd))) { |  | ||||||
|         close(fd); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } else { |  | ||||||
|     res = opendir_nt(name); |  | ||||||
|   } |  | ||||||
|   return res; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /**
 | /**
 | ||||||
|  * Creates directory object for file descriptor. |  * Creates directory object for file descriptor. | ||||||
|  * |  * | ||||||
|  | @ -324,19 +235,98 @@ DIR *opendir(const char *name) { | ||||||
|  * @errors ENOMEM and fd is closed |  * @errors ENOMEM and fd is closed | ||||||
|  */ |  */ | ||||||
| DIR *fdopendir(int fd) { | DIR *fdopendir(int fd) { | ||||||
|  | 
 | ||||||
|  |   // allocate directory iterator object
 | ||||||
|   DIR *dir; |   DIR *dir; | ||||||
|   if (!IsWindows()) { |   if (!(dir = calloc(1, sizeof(*dir)))) { | ||||||
|     if (!(dir = calloc(1, sizeof(*dir)))) return NULL; |     return NULL; | ||||||
|     dir->fd = fd; |  | ||||||
|   } else { |  | ||||||
|     dir = fdopendir_nt(fd); |  | ||||||
|   } |   } | ||||||
|  | 
 | ||||||
|  |   // on unix, file descriptor isn't required to be tracked
 | ||||||
|  |   dir->fd = fd; | ||||||
|  |   if (!__isfdkind(fd, kFdZip)) { | ||||||
|  |     if (IsWindows()) { | ||||||
|  |       free(dir); | ||||||
|  |       return fdopendir_nt(fd); | ||||||
|  |     } | ||||||
|  |     return dir; | ||||||
|  |   } | ||||||
|  |   dir->iszip = true; | ||||||
|  | 
 | ||||||
|  |   // ensure open /zip/... file is a directory
 | ||||||
|  |   struct ZiposHandle *h = (struct ZiposHandle *)(intptr_t)g_fds.p[fd].handle; | ||||||
|  |   if (h->cfile != ZIPOS_SYNTHETIC_DIRECTORY && | ||||||
|  |       !S_ISDIR(GetZipCfileMode(h->zipos->map + h->cfile))) { | ||||||
|  |     free(dir); | ||||||
|  |     enotdir(); | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // get path of this file descriptor and ensure trailing slash
 | ||||||
|  |   size_t len; | ||||||
|  |   char *name; | ||||||
|  |   if (h->cfile != ZIPOS_SYNTHETIC_DIRECTORY) { | ||||||
|  |     len = ZIP_CFILE_NAMESIZE(h->zipos->map + h->cfile); | ||||||
|  |     name = ZIP_CFILE_NAME(h->zipos->map + h->cfile); | ||||||
|  |   } else { | ||||||
|  |     len = h->size; | ||||||
|  |     name = (char *)h->data; | ||||||
|  |   } | ||||||
|  |   if (len + 2 > ZIPOS_PATH_MAX) { | ||||||
|  |     free(dir); | ||||||
|  |     enametoolong(); | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  |   if (len) memcpy(dir->zip.prefix, name, len); | ||||||
|  |   if (len && dir->zip.prefix[len - 1] != '/') { | ||||||
|  |     dir->zip.prefix[len++] = '/'; | ||||||
|  |   } | ||||||
|  |   dir->zip.prefix[len] = 0; | ||||||
|  |   dir->zip.prefixlen = len; | ||||||
|  | 
 | ||||||
|  |   // setup state values for directory iterator
 | ||||||
|  |   dir->zip.offset = GetZipCdirOffset(h->zipos->cdir); | ||||||
|  |   dir->zip.records = GetZipCdirRecords(h->zipos->cdir); | ||||||
|  | 
 | ||||||
|   return dir; |   return dir; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * Opens directory, e.g. | ||||||
|  |  * | ||||||
|  |  *     DIR *d; | ||||||
|  |  *     struct dirent *e; | ||||||
|  |  *     d = opendir(path); | ||||||
|  |  *     while ((e = readdir(d))) { | ||||||
|  |  *       printf("%s/%s\n", path, e->d_name); | ||||||
|  |  *     } | ||||||
|  |  *     closedir(d); | ||||||
|  |  * | ||||||
|  |  * @returns newly allocated DIR object, or NULL w/ errno | ||||||
|  |  * @errors ENOENT, ENOTDIR, EACCES, EMFILE, ENFILE, ENOMEM | ||||||
|  |  * @raise ECANCELED if thread was cancelled in masked mode | ||||||
|  |  * @raise EINTR if we needed to block and a signal was delivered instead | ||||||
|  |  * @cancellationpoint | ||||||
|  |  * @see glob() | ||||||
|  |  */ | ||||||
|  | DIR *opendir(const char *name) { | ||||||
|  |   int rc; | ||||||
|  |   if (_weaken(pthread_testcancel_np) && | ||||||
|  |       (rc = _weaken(pthread_testcancel_np)())) { | ||||||
|  |     errno = rc; | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  |   int fd; | ||||||
|  |   if ((fd = open(name, O_RDONLY | O_DIRECTORY | O_NOCTTY | O_CLOEXEC)) == -1) { | ||||||
|  |     return 0; | ||||||
|  |   } | ||||||
|  |   DIR *res = fdopendir(fd); | ||||||
|  |   if (!res) close(fd); | ||||||
|  |   return res; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static struct dirent *readdir_impl(DIR *dir) { | static struct dirent *readdir_impl(DIR *dir) { | ||||||
|   size_t n; |   size_t n; | ||||||
|   long basep; |  | ||||||
|   int rc, mode; |   int rc, mode; | ||||||
|   uint8_t *s, *p; |   uint8_t *s, *p; | ||||||
|   struct Zipos *zip; |   struct Zipos *zip; | ||||||
|  | @ -348,53 +338,68 @@ static struct dirent *readdir_impl(DIR *dir) { | ||||||
|   if (dir->iszip) { |   if (dir->iszip) { | ||||||
|     ent = 0; |     ent = 0; | ||||||
|     zip = _weaken(__zipos_get)(); |     zip = _weaken(__zipos_get)(); | ||||||
|     while (!ent && dir->tell < dir->zip.records) { |     while (!ent && dir->tell < dir->zip.records + 2) { | ||||||
|       npassert(ZIP_CFILE_MAGIC(zip->map + dir->zip.offset) == |       if (!dir->tell) { | ||||||
|                kZipCfileHdrMagic); |         ent = (struct dirent *)dir->buf; | ||||||
|       s = ZIP_CFILE_NAME(zip->map + dir->zip.offset); |         ent->d_ino++; | ||||||
|       n = ZIP_CFILE_NAMESIZE(zip->map + dir->zip.offset); |         ent->d_off = dir->tell; | ||||||
|       if (dir->zip.prefixlen < n && |         ent->d_reclen = 1; | ||||||
|           !memcmp(dir->zip.prefix, s, dir->zip.prefixlen)) { |         ent->d_type = DT_DIR; | ||||||
|         s += dir->zip.prefixlen; |         strcpy(ent->d_name, "."); | ||||||
|         n -= dir->zip.prefixlen; |       } else if (dir->tell == 1) { | ||||||
|         p = memchr(s, '/', n); |         ent = (struct dirent *)dir->buf; | ||||||
|         if (!p || p + 1 - s == n) { |         ent->d_ino++; | ||||||
|           if (p + 1 - s == n) --n; |         ent->d_off = dir->tell; | ||||||
|           mode = GetZipCfileMode(zip->map + dir->zip.offset); |         ent->d_reclen = 2; | ||||||
|           ent = (struct dirent *)dir->buf; |         ent->d_type = DT_DIR; | ||||||
|           ent->d_ino++; |         strcpy(ent->d_name, ".."); | ||||||
|           ent->d_off = dir->zip.offset; |       } else { | ||||||
|           ent->d_reclen = MIN(n, 255); |         s = ZIP_CFILE_NAME(zip->map + dir->zip.offset); | ||||||
|           ent->d_type = S_ISDIR(mode) ? DT_DIR : DT_REG; |         n = ZIP_CFILE_NAMESIZE(zip->map + dir->zip.offset); | ||||||
|           if (ent->d_reclen) { |         if (dir->zip.prefixlen < n && | ||||||
|             memcpy(ent->d_name, s, ent->d_reclen); |             !memcmp(dir->zip.prefix, s, dir->zip.prefixlen)) { | ||||||
|           } |           s += dir->zip.prefixlen; | ||||||
|           ent->d_name[ent->d_reclen] = 0; |           n -= dir->zip.prefixlen; | ||||||
|         } else { |           p = memchr(s, '/', n); | ||||||
|           lastent = (struct dirent *)dir->buf; |           if (!p || p + 1 - s == n) { | ||||||
|           n = p - s; |             if (p + 1 - s == n) --n; | ||||||
|           n = MIN(n, 255); |             mode = GetZipCfileMode(zip->map + dir->zip.offset); | ||||||
|           if (!lastent->d_ino || (n != lastent->d_reclen) || |             ent = (struct dirent *)dir->buf; | ||||||
|               memcmp(lastent->d_name, s, n)) { |  | ||||||
|             ent = lastent; |  | ||||||
|             ent->d_ino++; |             ent->d_ino++; | ||||||
|             ent->d_off = -1; |             ent->d_off = dir->tell; | ||||||
|             ent->d_reclen = n; |             ent->d_reclen = MIN(n, 255); | ||||||
|             ent->d_type = DT_DIR; |             ent->d_type = S_ISDIR(mode) ? DT_DIR : DT_REG; | ||||||
|             if (ent->d_reclen) { |             if (ent->d_reclen) { | ||||||
|               memcpy(ent->d_name, s, ent->d_reclen); |               memcpy(ent->d_name, s, ent->d_reclen); | ||||||
|             } |             } | ||||||
|             ent->d_name[ent->d_reclen] = 0; |             ent->d_name[ent->d_reclen] = 0; | ||||||
|  |           } else { | ||||||
|  |             lastent = (struct dirent *)dir->buf; | ||||||
|  |             n = p - s; | ||||||
|  |             n = MIN(n, 255); | ||||||
|  |             if (!lastent->d_ino || (n != lastent->d_reclen) || | ||||||
|  |                 memcmp(lastent->d_name, s, n)) { | ||||||
|  |               ent = lastent; | ||||||
|  |               mode = GetZipCfileMode(zip->map + dir->zip.offset); | ||||||
|  |               ent->d_ino++; | ||||||
|  |               ent->d_off = dir->tell; | ||||||
|  |               ent->d_reclen = n; | ||||||
|  |               ent->d_type = S_ISDIR(mode) ? DT_DIR : DT_REG; | ||||||
|  |               if (ent->d_reclen) { | ||||||
|  |                 memcpy(ent->d_name, s, ent->d_reclen); | ||||||
|  |               } | ||||||
|  |               ent->d_name[ent->d_reclen] = 0; | ||||||
|  |             } | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
|  |         dir->zip.offset += ZIP_CFILE_HDRSIZE(zip->map + dir->zip.offset); | ||||||
|       } |       } | ||||||
|       dir->zip.offset += ZIP_CFILE_HDRSIZE(zip->map + dir->zip.offset); |  | ||||||
|       dir->tell++; |       dir->tell++; | ||||||
|     } |     } | ||||||
|     return ent; |     return ent; | ||||||
|   } else if (!IsWindows()) { |   } else if (!IsWindows()) { | ||||||
|     if (dir->buf_pos >= dir->buf_end) { |     if (dir->buf_pos >= dir->buf_end) { | ||||||
|       basep = dir->tell; /* TODO(jart): what does xnu do */ |       long basep = dir->tell; | ||||||
|       rc = sys_getdents(dir->fd, dir->buf, sizeof(dir->buf) - 256, &basep); |       rc = sys_getdents(dir->fd, dir->buf, sizeof(dir->buf) - 256, &basep); | ||||||
|       STRACE("sys_getdents(%d) → %d% m", dir->fd, rc); |       STRACE("sys_getdents(%d) → %d% m", dir->fd, rc); | ||||||
|       if (!rc || rc == -1) return NULL; |       if (!rc || rc == -1) return NULL; | ||||||
|  | @ -408,6 +413,7 @@ static struct dirent *readdir_impl(DIR *dir) { | ||||||
|     } else if (IsOpenbsd()) { |     } else if (IsOpenbsd()) { | ||||||
|       obsd = (struct dirent_openbsd *)((char *)dir->buf + dir->buf_pos); |       obsd = (struct dirent_openbsd *)((char *)dir->buf + dir->buf_pos); | ||||||
|       dir->buf_pos += obsd->d_reclen; |       dir->buf_pos += obsd->d_reclen; | ||||||
|  |       dir->tell = obsd->d_off; | ||||||
|       ent = &dir->ent; |       ent = &dir->ent; | ||||||
|       ent->d_ino = obsd->d_fileno; |       ent->d_ino = obsd->d_fileno; | ||||||
|       ent->d_off = obsd->d_off; |       ent->d_off = obsd->d_off; | ||||||
|  | @ -419,7 +425,7 @@ static struct dirent *readdir_impl(DIR *dir) { | ||||||
|       dir->buf_pos += nbsd->d_reclen; |       dir->buf_pos += nbsd->d_reclen; | ||||||
|       ent = &dir->ent; |       ent = &dir->ent; | ||||||
|       ent->d_ino = nbsd->d_fileno; |       ent->d_ino = nbsd->d_fileno; | ||||||
|       ent->d_off = dir->tell++; |       ent->d_off = (dir->tell += nbsd->d_reclen); | ||||||
|       ent->d_reclen = nbsd->d_reclen; |       ent->d_reclen = nbsd->d_reclen; | ||||||
|       ent->d_type = nbsd->d_type; |       ent->d_type = nbsd->d_type; | ||||||
|       memcpy(ent->d_name, nbsd->d_name, MAX(256, nbsd->d_namlen + 1)); |       memcpy(ent->d_name, nbsd->d_name, MAX(256, nbsd->d_namlen + 1)); | ||||||
|  | @ -428,7 +434,7 @@ static struct dirent *readdir_impl(DIR *dir) { | ||||||
|       dir->buf_pos += bsd->d_reclen; |       dir->buf_pos += bsd->d_reclen; | ||||||
|       ent = &dir->ent; |       ent = &dir->ent; | ||||||
|       ent->d_ino = bsd->d_fileno; |       ent->d_ino = bsd->d_fileno; | ||||||
|       ent->d_off = IsXnu() ? (dir->tell = basep) : dir->tell++; |       ent->d_off = dir->tell++; | ||||||
|       ent->d_reclen = bsd->d_reclen; |       ent->d_reclen = bsd->d_reclen; | ||||||
|       ent->d_type = bsd->d_type; |       ent->d_type = bsd->d_type; | ||||||
|       memcpy(ent->d_name, bsd->d_name, bsd->d_namlen + 1); |       memcpy(ent->d_name, bsd->d_name, bsd->d_namlen + 1); | ||||||
|  | @ -450,9 +456,14 @@ static struct dirent *readdir_impl(DIR *dir) { | ||||||
|  */ |  */ | ||||||
| struct dirent *readdir(DIR *dir) { | struct dirent *readdir(DIR *dir) { | ||||||
|   struct dirent *e; |   struct dirent *e; | ||||||
|   _lockdir(dir); |   if (dir) { | ||||||
|   e = readdir_impl(dir); |     _lockdir(dir); | ||||||
|   _unlockdir(dir); |     e = readdir_impl(dir); | ||||||
|  |     _unlockdir(dir); | ||||||
|  |   } else { | ||||||
|  |     efault(); | ||||||
|  |     e = 0; | ||||||
|  |   } | ||||||
|   return e; |   return e; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -496,20 +507,18 @@ errno_t readdir_r(DIR *dir, struct dirent *output, struct dirent **result) { | ||||||
|  * @return 0 on success or -1 w/ errno |  * @return 0 on success or -1 w/ errno | ||||||
|  */ |  */ | ||||||
| int closedir(DIR *dir) { | int closedir(DIR *dir) { | ||||||
|   int rc; |   int rc = 0; | ||||||
|   if (dir) { |   if (dir) { | ||||||
|     if (dir->iszip) { |     if (dir->fd != -1) { | ||||||
|       free(dir->zip.prefix); |       rc |= close(dir->fd); | ||||||
|       rc = 0; |     } | ||||||
|     } else if (!IsWindows()) { |     free(dir->name); | ||||||
|       rc = close(dir->fd); |     if (IsWindows() && !dir->iszip) { | ||||||
|     } else { |       if (!FindClose(dir->hand)) { | ||||||
|       free(dir->name); |         rc = __winerr(); | ||||||
|       rc = FindClose(dir->fd) ? 0 : __winerr(); |       } | ||||||
|     } |     } | ||||||
|     free(dir); |     free(dir); | ||||||
|   } else { |  | ||||||
|     rc = 0; |  | ||||||
|   } |   } | ||||||
|   return rc; |   return rc; | ||||||
| } | } | ||||||
|  | @ -533,13 +542,7 @@ long telldir(DIR *dir) { | ||||||
| int dirfd(DIR *dir) { | int dirfd(DIR *dir) { | ||||||
|   int rc; |   int rc; | ||||||
|   _lockdir(dir); |   _lockdir(dir); | ||||||
|   if (dir->iszip) { |   rc = dir->fd; | ||||||
|     rc = eopnotsupp(); |  | ||||||
|   } else if (IsWindows()) { |  | ||||||
|     rc = eopnotsupp(); |  | ||||||
|   } else { |  | ||||||
|     rc = dir->fd; |  | ||||||
|   } |  | ||||||
|   _unlockdir(dir); |   _unlockdir(dir); | ||||||
|   return rc; |   return rc; | ||||||
| } | } | ||||||
|  | @ -559,8 +562,8 @@ void rewinddir(DIR *dir) { | ||||||
|       dir->tell = 0; |       dir->tell = 0; | ||||||
|     } |     } | ||||||
|   } else { |   } else { | ||||||
|     FindClose(dir->fd); |     FindClose(dir->hand); | ||||||
|     if ((dir->fd = FindFirstFile(dir->name, &dir->windata)) != -1) { |     if ((dir->hand = FindFirstFile(dir->name, &dir->windata)) != -1) { | ||||||
|       dir->isdone = false; |       dir->isdone = false; | ||||||
|       dir->tell = 0; |       dir->tell = 0; | ||||||
|     } else { |     } else { | ||||||
|  | @ -582,11 +585,27 @@ void seekdir(DIR *dir, long off) { | ||||||
|   if (dir->iszip) { |   if (dir->iszip) { | ||||||
|     dir->zip.offset = GetZipCdirOffset(_weaken(__zipos_get)()->cdir); |     dir->zip.offset = GetZipCdirOffset(_weaken(__zipos_get)()->cdir); | ||||||
|     for (i = 0; i < off && i < dir->zip.records; ++i) { |     for (i = 0; i < off && i < dir->zip.records; ++i) { | ||||||
|       dir->zip.offset += ZIP_CFILE_HDRSIZE(zip->map + dir->zip.offset); |       if (i >= 2) { | ||||||
|  |         dir->zip.offset += ZIP_CFILE_HDRSIZE(zip->map + dir->zip.offset); | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|   } else { |   } else if (!IsWindows()) { | ||||||
|     i = lseek(dir->fd, off, SEEK_SET); |     i = lseek(dir->fd, off, SEEK_SET); | ||||||
|     dir->buf_pos = dir->buf_end = 0; |     dir->buf_pos = dir->buf_end = 0; | ||||||
|  |   } else { | ||||||
|  |     i = 0; | ||||||
|  |     dir->isdone = false; | ||||||
|  |     FindClose(dir->hand); | ||||||
|  |     if ((dir->hand = FindFirstFile(dir->name, &dir->windata)) != -1) { | ||||||
|  |       for (; i < off; ++i) { | ||||||
|  |         if (!FindNextFile(dir->hand, &dir->windata)) { | ||||||
|  |           dir->isdone = true; | ||||||
|  |           break; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } else { | ||||||
|  |       dir->isdone = true; | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|   dir->tell = i; |   dir->tell = i; | ||||||
|   _unlockdir(dir); |   _unlockdir(dir); | ||||||
|  |  | ||||||
|  | @ -957,18 +957,6 @@ syscon	pf	PF_VSOCK				40			40			0			0			0			0			0			0 | ||||||
| syscon	pf	PF_WANPIPE				25			25			0			0			0			0			0			0 | syscon	pf	PF_WANPIPE				25			25			0			0			0			0			0			0 | ||||||
| syscon	pf	PF_X25					9			9			0			0			0			0			0			0 | syscon	pf	PF_X25					9			9			0			0			0			0			0			0 | ||||||
| 
 | 
 | ||||||
| #	getdents() constants |  | ||||||
| # |  | ||||||
| #	group	name					GNU/Systemd		GNU/Systemd (Aarch64)	XNU's Not UNIX!		MacOS (Arm64)		FreeBSD			OpenBSD			NetBSD			The New Technology	Commentary |  | ||||||
| syscon	dt	DT_UNKNOWN				0			0			0			0			0			0			0			0			# consensus |  | ||||||
| syscon	dt	DT_FIFO					1			1			1			1			1			1			1			1			# unix consensus & faked nt |  | ||||||
| syscon	dt	DT_CHR					2			2			2			2			2			2			2			2			# unix consensus & faked nt |  | ||||||
| syscon	dt	DT_DIR					4			4			4			4			4			4			4			4			# unix consensus & faked nt |  | ||||||
| syscon	dt	DT_BLK					6			6			6			6			6			6			6			6			# unix consensus & faked nt |  | ||||||
| syscon	dt	DT_REG					8			8			8			8			8			8			8			8			# unix consensus & faked nt |  | ||||||
| syscon	dt	DT_LNK					10			10			10			10			10			10			10			10			# unix consensus & faked nt |  | ||||||
| syscon	dt	DT_SOCK					12			12			12			12			12			12			12			12			# unix consensus & faked nt |  | ||||||
| 
 |  | ||||||
| #	msync() flags | #	msync() flags | ||||||
| # | # | ||||||
| #	group	name					GNU/Systemd		GNU/Systemd (Aarch64)	XNU's Not UNIX!		MacOS (Arm64)		FreeBSD			OpenBSD			NetBSD			The New Technology	Commentary | #	group	name					GNU/Systemd		GNU/Systemd (Aarch64)	XNU's Not UNIX!		MacOS (Arm64)		FreeBSD			OpenBSD			NetBSD			The New Technology	Commentary | ||||||
|  |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| #include "libc/sysv/consts/syscon.internal.h" |  | ||||||
| .syscon dt,DT_BLK,6,6,6,6,6,6,6,6 |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| #include "libc/sysv/consts/syscon.internal.h" |  | ||||||
| .syscon dt,DT_CHR,2,2,2,2,2,2,2,2 |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| #include "libc/sysv/consts/syscon.internal.h" |  | ||||||
| .syscon dt,DT_DIR,4,4,4,4,4,4,4,4 |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| #include "libc/sysv/consts/syscon.internal.h" |  | ||||||
| .syscon dt,DT_FIFO,1,1,1,1,1,1,1,1 |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| #include "libc/sysv/consts/syscon.internal.h" |  | ||||||
| .syscon dt,DT_LNK,10,10,10,10,10,10,10,10 |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| #include "libc/sysv/consts/syscon.internal.h" |  | ||||||
| .syscon dt,DT_REG,8,8,8,8,8,8,8,8 |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| #include "libc/sysv/consts/syscon.internal.h" |  | ||||||
| .syscon dt,DT_SOCK,12,12,12,12,12,12,12,12 |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| #include "libc/sysv/consts/syscon.internal.h" |  | ||||||
| .syscon dt,DT_UNKNOWN,0,0,0,0,0,0,0,0 |  | ||||||
|  | @ -1,19 +1,5 @@ | ||||||
| #ifndef COSMOPOLITAN_LIBC_SYSV_CONSTS_DT_H_ | #ifndef COSMOPOLITAN_LIBC_SYSV_CONSTS_DT_H_ | ||||||
| #define COSMOPOLITAN_LIBC_SYSV_CONSTS_DT_H_ | #define COSMOPOLITAN_LIBC_SYSV_CONSTS_DT_H_ | ||||||
| #if !(__ASSEMBLER__ + __LINKER__ + 0) |  | ||||||
| COSMOPOLITAN_C_START_ |  | ||||||
| 
 |  | ||||||
| extern const uint8_t DT_UNKNOWN; |  | ||||||
| extern const uint8_t DT_FIFO; |  | ||||||
| extern const uint8_t DT_CHR; |  | ||||||
| extern const uint8_t DT_DIR; |  | ||||||
| extern const uint8_t DT_BLK; |  | ||||||
| extern const uint8_t DT_REG; |  | ||||||
| extern const uint8_t DT_LNK; |  | ||||||
| extern const uint8_t DT_SOCK; |  | ||||||
| 
 |  | ||||||
| COSMOPOLITAN_C_END_ |  | ||||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ |  | ||||||
| 
 | 
 | ||||||
| #define DT_UNKNOWN 0 | #define DT_UNKNOWN 0 | ||||||
| #define DT_FIFO    1 | #define DT_FIFO    1 | ||||||
|  |  | ||||||
|  | @ -35,7 +35,6 @@ void SetUpOnce(void) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| TEST(access, efault) { | TEST(access, efault) { | ||||||
|   ASSERT_SYS(EFAULT, -1, access(0, F_OK)); |  | ||||||
|   if (IsWindows() || !IsAsan()) return;  // not possible
 |   if (IsWindows() || !IsAsan()) return;  // not possible
 | ||||||
|   ASSERT_SYS(EFAULT, -1, access((void *)77, F_OK)); |   ASSERT_SYS(EFAULT, -1, access((void *)77, F_OK)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -71,6 +71,15 @@ TEST(read_pipe, canBeInterruptedByAlarm) { | ||||||
|   close(fds[0]); |   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) { | BENCH(read, bench) { | ||||||
|  |  | ||||||
|  | @ -21,6 +21,7 @@ | ||||||
| #include "libc/calls/calls.h" | #include "libc/calls/calls.h" | ||||||
| #include "libc/calls/internal.h" | #include "libc/calls/internal.h" | ||||||
| #include "libc/calls/struct/metastat.internal.h" | #include "libc/calls/struct/metastat.internal.h" | ||||||
|  | #include "libc/calls/struct/stat.h" | ||||||
| #include "libc/dce.h" | #include "libc/dce.h" | ||||||
| #include "libc/errno.h" | #include "libc/errno.h" | ||||||
| #include "libc/mem/gc.internal.h" | #include "libc/mem/gc.internal.h" | ||||||
|  | @ -47,13 +48,15 @@ TEST(stat_010, testEmptyFile_sizeIsZero) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| TEST(stat, enoent) { | TEST(stat, enoent) { | ||||||
|   ASSERT_SYS(ENOENT, -1, stat("hi", 0)); |   struct stat st; | ||||||
|   ASSERT_SYS(ENOENT, -1, stat("o/doesnotexist", 0)); |   ASSERT_SYS(ENOENT, -1, stat("hi", &st)); | ||||||
|  |   ASSERT_SYS(ENOENT, -1, stat("o/doesnotexist", &st)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| TEST(stat, enotdir) { | TEST(stat, enotdir) { | ||||||
|  |   struct stat st; | ||||||
|   ASSERT_SYS(0, 0, close(creat("yo", 0644))); |   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) { | TEST(stat, zipos) { | ||||||
|  |  | ||||||
|  | @ -17,9 +17,12 @@ | ||||||
| │ PERFORMANCE OF THIS SOFTWARE.                                                │ | │ PERFORMANCE OF THIS SOFTWARE.                                                │ | ||||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||||
| #include "libc/calls/calls.h" | #include "libc/calls/calls.h" | ||||||
|  | #include "libc/errno.h" | ||||||
|  | #include "libc/limits.h" | ||||||
| #include "libc/mem/gc.h" | #include "libc/mem/gc.h" | ||||||
| #include "libc/mem/mem.h" | #include "libc/mem/mem.h" | ||||||
| #include "libc/runtime/runtime.h" | #include "libc/runtime/runtime.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/str/str.h" | #include "libc/str/str.h" | ||||||
| #include "libc/sysv/consts/o.h" | #include "libc/sysv/consts/o.h" | ||||||
| #include "libc/testlib/hyperion.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)); |   for (i = 0; i < n; ++i) EXPECT_SYS(0, 0, _join(t + i)); | ||||||
|   __print_maps(); |   __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/calls.h" | ||||||
| #include "libc/calls/struct/dirent.h" | #include "libc/calls/struct/dirent.h" | ||||||
|  | #include "libc/calls/struct/stat.h" | ||||||
| #include "libc/dce.h" | #include "libc/dce.h" | ||||||
| #include "libc/errno.h" | #include "libc/errno.h" | ||||||
|  | #include "libc/intrin/kprintf.h" | ||||||
| #include "libc/mem/gc.h" | #include "libc/mem/gc.h" | ||||||
|  | #include "libc/mem/gc.internal.h" | ||||||
| #include "libc/runtime/runtime.h" | #include "libc/runtime/runtime.h" | ||||||
| #include "libc/stdio/rand.h" | #include "libc/stdio/rand.h" | ||||||
| #include "libc/str/str.h" | #include "libc/str/str.h" | ||||||
| #include "libc/sysv/consts/dt.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/testlib/testlib.h" | ||||||
| #include "libc/x/xasprintf.h" | #include "libc/x/xasprintf.h" | ||||||
| #include "libc/x/xiso8601.h" | #include "libc/x/xiso8601.h" | ||||||
|  | @ -61,6 +66,10 @@ TEST(opendir, enotdir) { | ||||||
| TEST(opendir, zipTest_fake) { | TEST(opendir, zipTest_fake) { | ||||||
|   ASSERT_NE(NULL, (dir = opendir("/zip"))); |   ASSERT_NE(NULL, (dir = opendir("/zip"))); | ||||||
|   EXPECT_NE(NULL, (ent = readdir(dir))); |   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_STREQ("echo.com", ent->d_name); | ||||||
|   EXPECT_NE(NULL, (ent = readdir(dir))); |   EXPECT_NE(NULL, (ent = readdir(dir))); | ||||||
|   EXPECT_STREQ("usr", ent->d_name); |   EXPECT_STREQ("usr", ent->d_name); | ||||||
|  | @ -70,6 +79,10 @@ TEST(opendir, zipTest_fake) { | ||||||
|   EXPECT_EQ(0, closedir(dir)); |   EXPECT_EQ(0, closedir(dir)); | ||||||
|   ASSERT_NE(NULL, (dir = opendir("/zip/"))); |   ASSERT_NE(NULL, (dir = opendir("/zip/"))); | ||||||
|   EXPECT_NE(NULL, (ent = readdir(dir))); |   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_STREQ("echo.com", ent->d_name); | ||||||
|   EXPECT_NE(NULL, (ent = readdir(dir))); |   EXPECT_NE(NULL, (ent = readdir(dir))); | ||||||
|   EXPECT_STREQ("usr", ent->d_name); |   EXPECT_STREQ("usr", ent->d_name); | ||||||
|  | @ -79,11 +92,19 @@ TEST(opendir, zipTest_fake) { | ||||||
|   EXPECT_EQ(0, closedir(dir)); |   EXPECT_EQ(0, closedir(dir)); | ||||||
|   ASSERT_NE(NULL, (dir = opendir("/zip/usr"))); |   ASSERT_NE(NULL, (dir = opendir("/zip/usr"))); | ||||||
|   EXPECT_NE(NULL, (ent = readdir(dir))); |   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_STREQ("share", ent->d_name); | ||||||
|   EXPECT_EQ(NULL, (ent = readdir(dir))); |   EXPECT_EQ(NULL, (ent = readdir(dir))); | ||||||
|   EXPECT_EQ(0, closedir(dir)); |   EXPECT_EQ(0, closedir(dir)); | ||||||
|   ASSERT_NE(NULL, (dir = opendir("/zip/usr/"))); |   ASSERT_NE(NULL, (dir = opendir("/zip/usr/"))); | ||||||
|   EXPECT_NE(NULL, (ent = readdir(dir))); |   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_STREQ("share", ent->d_name); | ||||||
|   EXPECT_EQ(NULL, (ent = readdir(dir))); |   EXPECT_EQ(NULL, (ent = readdir(dir))); | ||||||
|   EXPECT_EQ(0, closedir(dir)); |   EXPECT_EQ(0, closedir(dir)); | ||||||
|  | @ -91,6 +112,28 @@ TEST(opendir, zipTest_fake) { | ||||||
|   EXPECT_EQ(NULL, (dir = opendir("/zip/us/"))); |   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) { | TEST(dirstream, testDots) { | ||||||
|   int hasdot = 0; |   int hasdot = 0; | ||||||
|   int hasdotdot = 0; |   int hasdotdot = 0; | ||||||
|  | @ -115,9 +158,9 @@ TEST(dirstream, test) { | ||||||
|   bool hasfoo = false; |   bool hasfoo = false; | ||||||
|   bool hasbar = false; |   bool hasbar = false; | ||||||
|   char *dpath, *file1, *file2; |   char *dpath, *file1, *file2; | ||||||
|   dpath = _gc(xasprintf("%s.%d", "dirstream", rand())); |   dpath = gc(xasprintf("%s.%d", "dirstream", rand())); | ||||||
|   file1 = _gc(xasprintf("%s/%s", dpath, "foo")); |   file1 = gc(xasprintf("%s/%s", dpath, "foo")); | ||||||
|   file2 = _gc(xasprintf("%s/%s", dpath, "bar")); |   file2 = gc(xasprintf("%s/%s", dpath, "bar")); | ||||||
|   EXPECT_NE(-1, mkdir(dpath, 0755)); |   EXPECT_NE(-1, mkdir(dpath, 0755)); | ||||||
|   EXPECT_NE(-1, touch(file1, 0644)); |   EXPECT_NE(-1, touch(file1, 0644)); | ||||||
|   EXPECT_NE(-1, touch(file2, 0644)); |   EXPECT_NE(-1, touch(file2, 0644)); | ||||||
|  | @ -143,12 +186,11 @@ TEST(dirstream, test) { | ||||||
| TEST(dirstream, zipTest) { | TEST(dirstream, zipTest) { | ||||||
|   bool foundNewYork = false; |   bool foundNewYork = false; | ||||||
|   const char *path = "/zip/usr/share/zoneinfo/"; |   const char *path = "/zip/usr/share/zoneinfo/"; | ||||||
|   ASSERT_NE(0, _gc(xiso8601ts(NULL))); |  | ||||||
|   ASSERT_NE(NULL, (dir = opendir(path))); |   ASSERT_NE(NULL, (dir = opendir(path))); | ||||||
|   while ((ent = readdir(dir))) { |   while ((ent = readdir(dir))) { | ||||||
|     foundNewYork |= !strcmp(ent->d_name, "New_York"); |     foundNewYork |= !strcmp(ent->d_name, "New_York"); | ||||||
|   } |   } | ||||||
|   closedir(dir); |   EXPECT_SYS(0, 0, closedir(dir)); | ||||||
|   EXPECT_TRUE(foundNewYork); |   EXPECT_TRUE(foundNewYork); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -156,9 +198,9 @@ TEST(rewinddir, test) { | ||||||
|   bool hasfoo = false; |   bool hasfoo = false; | ||||||
|   bool hasbar = false; |   bool hasbar = false; | ||||||
|   char *dpath, *file1, *file2; |   char *dpath, *file1, *file2; | ||||||
|   dpath = _gc(xasprintf("%s.%d", "dirstream", rand())); |   dpath = gc(xasprintf("%s.%d", "dirstream", rand())); | ||||||
|   file1 = _gc(xasprintf("%s/%s", dpath, "foo")); |   file1 = gc(xasprintf("%s/%s", dpath, "foo")); | ||||||
|   file2 = _gc(xasprintf("%s/%s", dpath, "bar")); |   file2 = gc(xasprintf("%s/%s", dpath, "bar")); | ||||||
|   EXPECT_NE(-1, mkdir(dpath, 0755)); |   EXPECT_NE(-1, mkdir(dpath, 0755)); | ||||||
|   EXPECT_NE(-1, touch(file1, 0644)); |   EXPECT_NE(-1, touch(file1, 0644)); | ||||||
|   EXPECT_NE(-1, touch(file2, 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(NULL, opendir("/zip/usr/share/zoneinfo/New_York")); | ||||||
|   ASSERT_EQ(ENOTDIR, errno); |   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); | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								third_party/unzip/process.c
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								third_party/unzip/process.c
									
										
									
									
										vendored
									
									
								
							|  | @ -1330,7 +1330,7 @@ static int find_ecrec64(__G__ searchlen)         /* return PK-class error */ | ||||||
|     if (memcmp((char *)byterec, end_central64_sig, 4) ) { |     if (memcmp((char *)byterec, end_central64_sig, 4) ) { | ||||||
|       /* Zip64 EOCD Record not found */ |       /* Zip64 EOCD Record not found */ | ||||||
|       /* Since we already have seen the Zip64 EOCD Locator, it's
 |       /* Since we already have seen the Zip64 EOCD Locator, it's
 | ||||||
|          possible we got here because there are bytes prepended |          possible we got h-ere because there are bytes prepended | ||||||
|          to the archive, like the sfx prefix. */ |          to the archive, like the sfx prefix. */ | ||||||
| 
 | 
 | ||||||
|       /* Make a guess as to where the Zip64 EOCD Record might be */ |       /* Make a guess as to where the Zip64 EOCD Record might be */ | ||||||
|  |  | ||||||
|  | @ -22,8 +22,11 @@ | ||||||
| #include "libc/limits.h" | #include "libc/limits.h" | ||||||
| #include "libc/log/check.h" | #include "libc/log/check.h" | ||||||
| #include "libc/mem/gc.h" | #include "libc/mem/gc.h" | ||||||
|  | #include "libc/mem/gc.internal.h" | ||||||
|  | #include "libc/mem/mem.h" | ||||||
| #include "libc/nexgen32e/crc32.h" | #include "libc/nexgen32e/crc32.h" | ||||||
| #include "libc/nt/enum/fileflagandattributes.h" | #include "libc/nt/enum/fileflagandattributes.h" | ||||||
|  | #include "libc/runtime/zipos.internal.h" | ||||||
| #include "libc/stdio/rand.h" | #include "libc/stdio/rand.h" | ||||||
| #include "libc/str/str.h" | #include "libc/str/str.h" | ||||||
| #include "libc/sysv/consts/s.h" | #include "libc/sysv/consts/s.h" | ||||||
|  | @ -130,7 +133,7 @@ static void EmitZipCdirHdr(unsigned char *p, const void *name, size_t namesize, | ||||||
| /**
 | /**
 | ||||||
|  * Embeds zip file in elf object. |  * Embeds zip file in elf object. | ||||||
|  */ |  */ | ||||||
| void elfwriter_zip(struct ElfWriter *elf, const char *symbol, const char *name, | void elfwriter_zip(struct ElfWriter *elf, const char *symbol, const char *cname, | ||||||
|                    size_t namesize, const void *data, size_t size, |                    size_t namesize, const void *data, size_t size, | ||||||
|                    uint32_t mode, struct timespec mtim, struct timespec atim, |                    uint32_t mode, struct timespec mtim, struct timespec atim, | ||||||
|                    struct timespec ctim, bool nocompress) { |                    struct timespec ctim, bool nocompress) { | ||||||
|  | @ -144,6 +147,13 @@ void elfwriter_zip(struct ElfWriter *elf, const char *symbol, const char *name, | ||||||
| 
 | 
 | ||||||
|   CHECK_NE(0, mtim.tv_sec); |   CHECK_NE(0, mtim.tv_sec); | ||||||
| 
 | 
 | ||||||
|  |   char *name = gc(strndup(cname, namesize)); | ||||||
|  |   namesize = __zipos_normpath(name); | ||||||
|  |   if (S_ISDIR(mode) && namesize && name[namesize - 1] != '/') { | ||||||
|  |     name[namesize++] = '/'; | ||||||
|  |     name[namesize] = 0; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   gflags = 0; |   gflags = 0; | ||||||
|   iattrs = 0; |   iattrs = 0; | ||||||
|   compsize = size; |   compsize = size; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue