Skip to content

Commit a94d2d0

Browse files
authored
Avoid varargs conventions when calling open (WebAssembly#126)
* Add an entrypoint for calling open that bypasses the varargs. * Add an entrypoint for calling openat that bypasses the varargs.
1 parent 7fcc4f2 commit a94d2d0

File tree

8 files changed

+38
-10
lines changed

8 files changed

+38
-10
lines changed

expected/wasm32-wasi/defined-symbols.txt

+2
Original file line numberDiff line numberDiff line change
@@ -251,6 +251,8 @@ __uselocale
251251
__utc
252252
__wasilibc_fd_renumber
253253
__wasilibc_find_relpath
254+
__wasilibc_open_nomode
255+
__wasilibc_openat_nomode
254256
__wasilibc_populate_environ
255257
__wasilibc_register_preopened_fd
256258
__wasilibc_rmdirat

libc-bottom-half/cloudlibc/src/libc/dirent/opendirat.c

+5
Original file line numberDiff line numberDiff line change
@@ -2,14 +2,19 @@
22
//
33
// SPDX-License-Identifier: BSD-2-Clause
44

5+
#include <wasi/libc.h>
56
#include <dirent.h>
67
#include <fcntl.h>
78
#include <stddef.h>
89
#include <unistd.h>
910

1011
DIR *opendirat(int dir, const char *dirname) {
1112
// Open directory.
13+
#ifdef __wasilibc_unmodified_upstream // avoid making a varargs call
1214
int fd = openat(dir, dirname, O_RDONLY | O_NONBLOCK | O_DIRECTORY);
15+
#else
16+
int fd = __wasilibc_openat_nomode(dir, dirname, O_RDONLY | O_NONBLOCK | O_DIRECTORY);
17+
#endif
1318
if (fd == -1)
1419
return NULL;
1520

libc-bottom-half/cloudlibc/src/libc/dirent/scandirat.c

+5
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
// SPDX-License-Identifier: BSD-2-Clause
44

55
#include <wasi/core.h>
6+
#include <wasi/libc.h>
67
#include <dirent.h>
78
#include <errno.h>
89
#include <fcntl.h>
@@ -24,7 +25,11 @@ int scandirat(int dirfd, const char *dir, struct dirent ***namelist,
2425
sel = sel_true;
2526

2627
// Open the directory.
28+
#ifdef __wasilibc_unmodified_upstream // avoid making a varargs call
2729
int fd = openat(dirfd, dir, O_RDONLY | O_NONBLOCK | O_DIRECTORY);
30+
#else
31+
int fd = __wasilibc_openat_nomode(dirfd, dir, O_RDONLY | O_NONBLOCK | O_DIRECTORY);
32+
#endif
2833
if (fd == -1)
2934
return -1;
3035

libc-bottom-half/cloudlibc/src/libc/fcntl/openat.c

+8
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66

77
#include <assert.h>
88
#include <wasi/core.h>
9+
#include <wasi/libc.h>
910
#include <errno.h>
1011
#include <fcntl.h>
1112
#include <string.h>
@@ -22,6 +23,13 @@ static_assert(O_EXCL >> 12 == __WASI_O_EXCL, "Value mismatch");
2223
static_assert(O_TRUNC >> 12 == __WASI_O_TRUNC, "Value mismatch");
2324

2425
int openat(int fd, const char *path, int oflag, ...) {
26+
#ifdef __wasilibc_unmodified_upstream // fstat
27+
#else
28+
return __wasilibc_openat_nomode(fd, path, oflag);
29+
}
30+
31+
int __wasilibc_openat_nomode(int fd, const char *path, int oflag) {
32+
#endif
2533
// Compute rights corresponding with the access modes provided.
2634
// Attempt to obtain all rights, except the ones that contradict the
2735
// access mode provided to openat().

libc-bottom-half/headers/public/wasi/libc.h

+2
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,8 @@ int __wasilibc_register_preopened_fd(int fd, const char *path);
1111
int __wasilibc_fd_renumber(int fd, int newfd);
1212
int __wasilibc_unlinkat(int fd, const char *path);
1313
int __wasilibc_rmdirat(int fd, const char *path);
14+
int __wasilibc_open_nomode(const char *path, int oflag);
15+
int __wasilibc_openat_nomode(int fd, const char *path, int oflag);
1416
off_t __wasilibc_tell(int fd);
1517

1618
#ifdef __cplusplus

libc-bottom-half/libpreopen/libpreopen.c

+9-3
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,14 @@
6565

6666
int
6767
open(const char *path, int flags, ...)
68+
{
69+
// WASI libc's openat ignores the mode argument, so call a special
70+
// entrypoint which avoids the varargs calling convention.
71+
return __wasilibc_open_nomode(path, flags);
72+
}
73+
74+
int
75+
__wasilibc_open_nomode(const char *path, int flags)
6876
{
6977
const char *relative_path;
7078
int dirfd = __wasilibc_find_relpath(path, __WASI_RIGHT_PATH_OPEN, 0,
@@ -77,9 +85,7 @@ open(const char *path, int flags, ...)
7785
return -1;
7886
}
7987

80-
// WASI libc's openat ignores the mode argument, so don't bother passing
81-
// the actual mode value through.
82-
return openat(dirfd, relative_path, flags);
88+
return __wasilibc_openat_nomode(dirfd, relative_path, flags);
8389
}
8490

8591
int

libc-bottom-half/sources/truncate.c

+4-6
Original file line numberDiff line numberDiff line change
@@ -1,23 +1,21 @@
11
#include <unistd.h>
22
#include <fcntl.h>
33
#include <errno.h>
4+
#include <wasi/libc.h>
45

56
int truncate(const char *path, off_t length)
67
{
7-
int fd = open(path, O_WRONLY | O_CLOEXEC | O_NOCTTY);
8+
int fd = __wasilibc_open_nomode(path, O_WRONLY | O_CLOEXEC | O_NOCTTY);
89
if (fd < 0)
910
return -1;
1011

1112
int result = ftruncate(fd, length);
12-
if (result < 0) {
13+
if (result != 0) {
1314
int save_errno = errno;
1415
(void)close(fd);
1516
errno = save_errno;
1617
return -1;
1718
}
1819

19-
if (close(fd) < 0)
20-
return -1;
21-
22-
return 0;
20+
return close(fd);
2321
}

libc-top-half/musl/src/stdio/fopen.c

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
#ifdef __wasilibc_unmodified_upstream // WASI has no syscall
22
#else
33
#include <unistd.h>
4+
#include <wasi/libc.h>
45
#endif
56
#include "stdio_impl.h"
67
#include <fcntl.h>
@@ -25,7 +26,8 @@ FILE *fopen(const char *restrict filename, const char *restrict mode)
2526
#ifdef __wasilibc_unmodified_upstream // WASI has no sys_open
2627
fd = sys_open(filename, flags, 0666);
2728
#else
28-
fd = open(filename, flags, 0666);
29+
// WASI libc ignores the mode parameter anyway, so skip the varargs.
30+
fd = __wasilibc_open_nomode(filename, flags);
2931
#endif
3032
if (fd < 0) return 0;
3133
#ifdef __wasilibc_unmodified_upstream // WASI has no syscall

0 commit comments

Comments
 (0)