mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-01-31 11:37:35 +00:00
172 lines
5.9 KiB
C
172 lines
5.9 KiB
C
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
│ vi: set et ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi │
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
│ Copyright 2020 Justine Alexandra Roberts Tunney │
|
|
│ │
|
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
│ any purpose with or without fee is hereby granted, provided that the │
|
|
│ above copyright notice and this permission notice appear in all copies. │
|
|
│ │
|
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
#include "libc/calls/calls.h"
|
|
#include "libc/calls/struct/metastat.internal.h"
|
|
#include "libc/calls/syscall-nt.internal.h"
|
|
#include "libc/calls/syscall-sysv.internal.h"
|
|
#include "libc/dce.h"
|
|
#include "libc/errno.h"
|
|
#include "libc/intrin/strace.h"
|
|
#include "libc/limits.h"
|
|
#include "libc/nt/files.h"
|
|
#include "libc/stdio/sysparam.h"
|
|
#include "libc/str/str.h"
|
|
#include "libc/sysv/consts/at.h"
|
|
#include "libc/sysv/consts/o.h"
|
|
#include "libc/sysv/errfuns.h"
|
|
|
|
#define XNU_F_GETPATH 50
|
|
#define XNU_MAXPATHLEN 1024
|
|
|
|
static int sys_getcwd_xnu(char *res, size_t size) {
|
|
int fd, len, rc = -1;
|
|
union metastat st[2];
|
|
char buf[XNU_MAXPATHLEN];
|
|
if ((fd = __sys_openat_nc(AT_FDCWD, ".", O_RDONLY | O_DIRECTORY, 0)) != -1) {
|
|
if (__sys_fstat(fd, &st[0]) != -1) {
|
|
if (st[0].xnu.st_dev && st[0].xnu.st_ino) {
|
|
if (__sys_fcntl(fd, XNU_F_GETPATH, (uintptr_t)buf) != -1) {
|
|
if (__sys_fstatat(AT_FDCWD, buf, &st[1], 0) != -1) {
|
|
if (st[0].xnu.st_dev == st[1].xnu.st_dev &&
|
|
st[0].xnu.st_ino == st[1].xnu.st_ino) {
|
|
if ((len = strlen(buf)) < size) {
|
|
memcpy(res, buf, (rc = len + 1));
|
|
} else {
|
|
erange();
|
|
}
|
|
} else {
|
|
einval();
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
einval();
|
|
}
|
|
}
|
|
sys_close(fd);
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
static int sys_getcwd_metal(char *buf, size_t size) {
|
|
if (size >= 5) {
|
|
strcpy(buf, "/zip");
|
|
return 5;
|
|
} else {
|
|
return erange();
|
|
}
|
|
}
|
|
|
|
static inline int IsAlpha(int c) {
|
|
return ('A' <= c && c <= 'Z') || ('a' <= c && c <= 'z');
|
|
}
|
|
|
|
static dontinline textwindows int sys_getcwd_nt(char *buf, size_t size) {
|
|
|
|
// get current directory from the system
|
|
char16_t p16[PATH_MAX];
|
|
uint32_t n = GetCurrentDirectory(PATH_MAX, p16);
|
|
if (!n)
|
|
return eacces(); // system call failed
|
|
if (n >= PATH_MAX)
|
|
return erange(); // not enough room?!?
|
|
|
|
// convert utf-16 to utf-8
|
|
// we can't modify `buf` until we're certain of success
|
|
char p8[PATH_MAX], *p = p8;
|
|
n = tprecode16to8(p, PATH_MAX, p16).ax;
|
|
if (n >= PATH_MAX)
|
|
return erange(); // utf-8 explosion
|
|
|
|
// turn \\?\c:\... into c:\...
|
|
if (p[0] == '\\' && //
|
|
p[1] == '\\' && //
|
|
p[2] == '?' && //
|
|
p[3] == '\\' && //
|
|
IsAlpha(p[4]) && //
|
|
p[5] == ':' && //
|
|
p[6] == '\\') {
|
|
p += 4;
|
|
n -= 4;
|
|
}
|
|
|
|
// turn c:\... into \c\...
|
|
if (IsAlpha(p[0]) && //
|
|
p[1] == ':' && //
|
|
p[2] == '\\') {
|
|
p[1] = p[0];
|
|
p[0] = '\\';
|
|
}
|
|
|
|
// we now know the final length
|
|
// check if the user supplied a buffer large enough
|
|
if (n >= size) {
|
|
return erange();
|
|
}
|
|
|
|
// copy bytes converting backslashes to slash
|
|
for (int i = 0; i < n; ++i) {
|
|
int c = p[i];
|
|
if (c == '\\') {
|
|
c = '/';
|
|
}
|
|
buf[i] = c;
|
|
}
|
|
|
|
// return number of bytes including nul
|
|
buf[n++] = 0;
|
|
return n;
|
|
}
|
|
|
|
/**
|
|
* Returns current working directory.
|
|
*
|
|
* Cosmo provides this function to address the shortcomings of getcwd().
|
|
* First, this function doesn't link malloc(). Secondly, this offers the
|
|
* Linux and NetBSD's getcwd() API across platforms since it's the best.
|
|
*
|
|
* @param buf receives utf-8 path and isn't modified on error
|
|
* @param size is byte capacity of `buf`
|
|
* @return bytes copied including nul on success, or -1 w/ errno
|
|
* @raise EACCES if the current directory path couldn't be accessed
|
|
* @raise ERANGE if `size` wasn't big enough for path and nul byte
|
|
* @raise EFAULT if `buf` points to invalid memory
|
|
*/
|
|
int __getcwd(char *buf, size_t size) {
|
|
int rc;
|
|
if (IsLinux() || IsNetbsd()) {
|
|
rc = sys_getcwd(buf, size);
|
|
} else if (IsXnu()) {
|
|
rc = sys_getcwd_xnu(buf, size);
|
|
} else if (IsWindows()) {
|
|
rc = sys_getcwd_nt(buf, size);
|
|
} else if (IsFreebsd() || IsOpenbsd()) {
|
|
if (sys_getcwd(buf, size) != -1) {
|
|
rc = strlen(buf) + 1;
|
|
} else if (SupportsFreebsd() && (errno == ENOMEM || errno == EINVAL)) {
|
|
rc = erange();
|
|
} else {
|
|
rc = -1;
|
|
}
|
|
} else {
|
|
rc = sys_getcwd_metal(buf, size);
|
|
}
|
|
STRACE("getcwd([%#hhs], %'zu) → %d% m", rc != -1 ? buf : "n/a", size, rc);
|
|
return rc;
|
|
}
|