github.com/wasilibs/wazerox@v0.0.0-20240124024944-4923be63ab5f/imports/wasi_snapshot_preview1/fs_test.go (about)

     1  package wasi_snapshot_preview1_test
     2  
     3  import (
     4  	"bytes"
     5  	_ "embed"
     6  	"fmt"
     7  	"io"
     8  	"io/fs"
     9  	"math"
    10  	"os"
    11  	"path"
    12  	"runtime"
    13  	"testing"
    14  	gofstest "testing/fstest"
    15  	"time"
    16  
    17  	wazero "github.com/wasilibs/wazerox"
    18  	"github.com/wasilibs/wazerox/api"
    19  	experimentalsys "github.com/wasilibs/wazerox/experimental/sys"
    20  	"github.com/wasilibs/wazerox/internal/fsapi"
    21  	"github.com/wasilibs/wazerox/internal/fstest"
    22  	"github.com/wasilibs/wazerox/internal/platform"
    23  	"github.com/wasilibs/wazerox/internal/sys"
    24  	"github.com/wasilibs/wazerox/internal/sysfs"
    25  	"github.com/wasilibs/wazerox/internal/testing/require"
    26  	"github.com/wasilibs/wazerox/internal/u64"
    27  	"github.com/wasilibs/wazerox/internal/wasip1"
    28  	"github.com/wasilibs/wazerox/internal/wasm"
    29  	sysapi "github.com/wasilibs/wazerox/sys"
    30  )
    31  
    32  func Test_fdAdvise(t *testing.T) {
    33  	mod, r, _ := requireProxyModule(t, wazero.NewModuleConfig().WithFS(fstest.FS))
    34  	defer r.Close(testCtx)
    35  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdAdviseName, uint64(3), 0, 0, uint64(wasip1.FdAdviceNormal))
    36  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdAdviseName, uint64(3), 0, 0, uint64(wasip1.FdAdviceSequential))
    37  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdAdviseName, uint64(3), 0, 0, uint64(wasip1.FdAdviceRandom))
    38  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdAdviseName, uint64(3), 0, 0, uint64(wasip1.FdAdviceWillNeed))
    39  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdAdviseName, uint64(3), 0, 0, uint64(wasip1.FdAdviceDontNeed))
    40  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdAdviseName, uint64(3), 0, 0, uint64(wasip1.FdAdviceNoReuse))
    41  	requireErrnoResult(t, wasip1.ErrnoInval, mod, wasip1.FdAdviseName, uint64(3), 0, 0, uint64(wasip1.FdAdviceNoReuse+1))
    42  	requireErrnoResult(t, wasip1.ErrnoBadf, mod, wasip1.FdAdviseName, uint64(1111111), 0, 0, uint64(wasip1.FdAdviceNoReuse+1))
    43  }
    44  
    45  // Test_fdAllocate only tests it is stubbed for GrainLang per #271
    46  func Test_fdAllocate(t *testing.T) {
    47  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
    48  	const fileName = "file.txt"
    49  
    50  	// Create the target file.
    51  	realPath := joinPath(tmpDir, fileName)
    52  	require.NoError(t, os.WriteFile(realPath, []byte("0123456789"), 0o600))
    53  
    54  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(
    55  		wazero.NewFSConfig().WithDirMount(tmpDir, "/"),
    56  	))
    57  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
    58  	preopen := fsc.RootFS()
    59  	defer r.Close(testCtx)
    60  
    61  	fd, errno := fsc.OpenFile(preopen, fileName, experimentalsys.O_RDWR, 0)
    62  	require.EqualErrno(t, 0, errno)
    63  
    64  	f, ok := fsc.LookupFile(fd)
    65  	require.True(t, ok)
    66  
    67  	requireSizeEqual := func(exp int64) {
    68  		st, errno := f.File.Stat()
    69  		require.EqualErrno(t, 0, errno)
    70  		require.Equal(t, exp, st.Size)
    71  	}
    72  
    73  	t.Run("errors", func(t *testing.T) {
    74  		requireErrnoResult(t, wasip1.ErrnoBadf, mod, wasip1.FdAllocateName, uint64(12345), 0, 0)
    75  		minusOne := int64(-1)
    76  		requireErrnoResult(t, wasip1.ErrnoInval, mod, wasip1.FdAllocateName, uint64(fd), uint64(minusOne), uint64(minusOne))
    77  		requireErrnoResult(t, wasip1.ErrnoInval, mod, wasip1.FdAllocateName, uint64(fd), 0, uint64(minusOne))
    78  		requireErrnoResult(t, wasip1.ErrnoInval, mod, wasip1.FdAllocateName, uint64(fd), uint64(minusOne), 0)
    79  	})
    80  
    81  	t.Run("do not change size", func(t *testing.T) {
    82  		for _, tc := range []struct{ offset, length uint64 }{
    83  			{offset: 0, length: 10},
    84  			{offset: 5, length: 5},
    85  			{offset: 4, length: 0},
    86  			{offset: 10, length: 0},
    87  		} {
    88  			// This shouldn't change the size.
    89  			requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdAllocateName,
    90  				uint64(fd), tc.offset, tc.length)
    91  			requireSizeEqual(10)
    92  		}
    93  	})
    94  
    95  	t.Run("increase", func(t *testing.T) {
    96  		// 10 + 10 > the current size -> increase the size.
    97  		requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdAllocateName,
    98  			uint64(fd), 10, 10)
    99  		requireSizeEqual(20)
   100  
   101  		// But the original content must be kept.
   102  		buf, err := os.ReadFile(realPath)
   103  		require.NoError(t, err)
   104  		require.Equal(t, "0123456789", string(buf[:10]))
   105  	})
   106  
   107  	require.Equal(t, `
   108  ==> wasi_snapshot_preview1.fd_allocate(fd=12345,offset=0,len=0)
   109  <== errno=EBADF
   110  ==> wasi_snapshot_preview1.fd_allocate(fd=4,offset=-1,len=-1)
   111  <== errno=EINVAL
   112  ==> wasi_snapshot_preview1.fd_allocate(fd=4,offset=0,len=-1)
   113  <== errno=EINVAL
   114  ==> wasi_snapshot_preview1.fd_allocate(fd=4,offset=-1,len=0)
   115  <== errno=EINVAL
   116  ==> wasi_snapshot_preview1.fd_allocate(fd=4,offset=0,len=10)
   117  <== errno=ESUCCESS
   118  ==> wasi_snapshot_preview1.fd_allocate(fd=4,offset=5,len=5)
   119  <== errno=ESUCCESS
   120  ==> wasi_snapshot_preview1.fd_allocate(fd=4,offset=4,len=0)
   121  <== errno=ESUCCESS
   122  ==> wasi_snapshot_preview1.fd_allocate(fd=4,offset=10,len=0)
   123  <== errno=ESUCCESS
   124  ==> wasi_snapshot_preview1.fd_allocate(fd=4,offset=10,len=10)
   125  <== errno=ESUCCESS
   126  `, "\n"+log.String())
   127  }
   128  
   129  func Test_fdClose(t *testing.T) {
   130  	// fd_close needs to close an open file descriptor. Open two files so that we can tell which is closed.
   131  	path1, path2 := "dir/-", "dir/a-"
   132  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFS(fstest.FS))
   133  	defer r.Close(testCtx)
   134  
   135  	// open both paths without using WASI
   136  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
   137  	preopen := fsc.RootFS()
   138  
   139  	fdToClose, errno := fsc.OpenFile(preopen, path1, experimentalsys.O_RDONLY, 0)
   140  	require.EqualErrno(t, 0, errno)
   141  
   142  	fdToKeep, errno := fsc.OpenFile(preopen, path2, experimentalsys.O_RDONLY, 0)
   143  	require.EqualErrno(t, 0, errno)
   144  
   145  	// Close
   146  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdCloseName, uint64(fdToClose))
   147  	require.Equal(t, `
   148  ==> wasi_snapshot_preview1.fd_close(fd=4)
   149  <== errno=ESUCCESS
   150  `, "\n"+log.String())
   151  
   152  	// Verify fdToClose is closed and removed from the file descriptor table.
   153  	_, ok := fsc.LookupFile(fdToClose)
   154  	require.False(t, ok)
   155  
   156  	// Verify fdToKeep is not closed
   157  	_, ok = fsc.LookupFile(fdToKeep)
   158  	require.True(t, ok)
   159  
   160  	log.Reset()
   161  	t.Run("ErrnoBadF for an invalid FD", func(t *testing.T) {
   162  		requireErrnoResult(t, wasip1.ErrnoBadf, mod, wasip1.FdCloseName, uint64(42)) // 42 is an arbitrary invalid Fd
   163  		require.Equal(t, `
   164  ==> wasi_snapshot_preview1.fd_close(fd=42)
   165  <== errno=EBADF
   166  `, "\n"+log.String())
   167  	})
   168  	log.Reset()
   169  	t.Run("Can close a pre-open", func(t *testing.T) {
   170  		requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdCloseName, uint64(sys.FdPreopen))
   171  		require.Equal(t, `
   172  ==> wasi_snapshot_preview1.fd_close(fd=3)
   173  <== errno=ESUCCESS
   174  `, "\n"+log.String())
   175  	})
   176  }
   177  
   178  // Test_fdDatasync only tests that the call succeeds; it's hard to test its effectiveness.
   179  func Test_fdDatasync(t *testing.T) {
   180  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
   181  	pathName := "test_path"
   182  	mod, fd, log, r := requireOpenFile(t, tmpDir, pathName, []byte{}, false)
   183  	defer r.Close(testCtx)
   184  
   185  	tests := []struct {
   186  		name          string
   187  		fd            int32
   188  		expectedErrno wasip1.Errno
   189  		expectedLog   string
   190  	}{
   191  		{
   192  			name:          "invalid FD",
   193  			fd:            42, // arbitrary invalid fd
   194  			expectedErrno: wasip1.ErrnoBadf,
   195  			expectedLog: `
   196  ==> wasi_snapshot_preview1.fd_datasync(fd=42)
   197  <== errno=EBADF
   198  `,
   199  		},
   200  		{
   201  			name:          "valid FD",
   202  			fd:            fd,
   203  			expectedErrno: wasip1.ErrnoSuccess,
   204  			expectedLog: `
   205  ==> wasi_snapshot_preview1.fd_datasync(fd=4)
   206  <== errno=ESUCCESS
   207  `,
   208  		},
   209  	}
   210  
   211  	for _, tt := range tests {
   212  		tc := tt
   213  		t.Run(tc.name, func(t *testing.T) {
   214  			defer log.Reset()
   215  
   216  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdDatasyncName, uint64(tc.fd))
   217  			require.Equal(t, tc.expectedLog, "\n"+log.String())
   218  		})
   219  	}
   220  }
   221  
   222  func openPipe(t *testing.T) (*os.File, *os.File) {
   223  	r, w, err := os.Pipe()
   224  	require.NoError(t, err)
   225  	return r, w
   226  }
   227  
   228  func closePipe(r, w *os.File) {
   229  	r.Close()
   230  	w.Close()
   231  }
   232  
   233  func Test_fdFdstatGet(t *testing.T) {
   234  	file, dir := "animals.txt", "sub"
   235  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFS(fstest.FS))
   236  	defer r.Close(testCtx)
   237  	memorySize := mod.Memory().Size()
   238  
   239  	// open both paths without using WASI
   240  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
   241  	preopen := fsc.RootFS()
   242  
   243  	// replace stdin with a fake TTY file.
   244  	// TODO: Make this easier once we have in-memory sys.File
   245  	stdin, _ := fsc.LookupFile(sys.FdStdin)
   246  	stdinFile, errno := (&sysfs.AdaptFS{FS: &gofstest.MapFS{"stdin": &gofstest.MapFile{
   247  		Mode: fs.ModeDevice | fs.ModeCharDevice | 0o600,
   248  	}}}).OpenFile("stdin", 0, 0)
   249  	require.EqualErrno(t, 0, errno)
   250  
   251  	stdin.File = fsapi.Adapt(stdinFile)
   252  
   253  	// Make this file writeable, to ensure flags read-back correctly.
   254  	fileFD, errno := fsc.OpenFile(preopen, file, experimentalsys.O_RDWR, 0)
   255  	require.EqualErrno(t, 0, errno)
   256  
   257  	dirFD, errno := fsc.OpenFile(preopen, dir, experimentalsys.O_RDONLY, 0)
   258  	require.EqualErrno(t, 0, errno)
   259  
   260  	tests := []struct {
   261  		name           string
   262  		fd             int32
   263  		resultFdstat   uint32
   264  		expectedMemory []byte
   265  		expectedErrno  wasip1.Errno
   266  		expectedLog    string
   267  	}{
   268  		{
   269  			name: "stdin is a tty",
   270  			fd:   sys.FdStdin,
   271  			expectedMemory: []byte{
   272  				2, 0, // fs_filetype
   273  				0, 0, 0, 0, 0, 0, // fs_flags
   274  				0xdb, 0x1, 0xe0, 0x8, 0x0, 0x0, 0x0, 0x0, // fs_rights_base
   275  				0, 0, 0, 0, 0, 0, 0, 0, // fs_rights_inheriting
   276  			}, // We shouldn't see RIGHT_FD_SEEK|RIGHT_FD_TELL on a tty file:
   277  			expectedLog: `
   278  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=0)
   279  <== (stat={filetype=CHARACTER_DEVICE,fdflags=,fs_rights_base=FD_DATASYNC|FD_READ|FDSTAT_SET_FLAGS|FD_SYNC|FD_WRITE|FD_ADVISE|FD_ALLOCATE,fs_rights_inheriting=},errno=ESUCCESS)
   280  `,
   281  		},
   282  		{
   283  			name: "stdout",
   284  			fd:   sys.FdStdout,
   285  			expectedMemory: []byte{
   286  				1, 0, // fs_filetype
   287  				0, 0, 0, 0, 0, 0, // fs_flags
   288  				0xff, 0x1, 0xe0, 0x8, 0x0, 0x0, 0x0, 0x0, // fs_rights_base
   289  				0, 0, 0, 0, 0, 0, 0, 0, // fs_rights_inheriting
   290  			},
   291  			expectedLog: `
   292  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=1)
   293  <== (stat={filetype=BLOCK_DEVICE,fdflags=,fs_rights_base=FD_DATASYNC|FD_READ|FD_SEEK|FDSTAT_SET_FLAGS|FD_SYNC|FD_TELL|FD_WRITE|FD_ADVISE|FD_ALLOCATE,fs_rights_inheriting=},errno=ESUCCESS)
   294  `,
   295  		},
   296  		{
   297  			name: "stderr",
   298  			fd:   sys.FdStderr,
   299  			expectedMemory: []byte{
   300  				1, 0, // fs_filetype
   301  				0, 0, 0, 0, 0, 0, // fs_flags
   302  				0xff, 0x1, 0xe0, 0x8, 0x0, 0x0, 0x0, 0x0, // fs_rights_base
   303  				0, 0, 0, 0, 0, 0, 0, 0, // fs_rights_inheriting
   304  			},
   305  			expectedLog: `
   306  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=2)
   307  <== (stat={filetype=BLOCK_DEVICE,fdflags=,fs_rights_base=FD_DATASYNC|FD_READ|FD_SEEK|FDSTAT_SET_FLAGS|FD_SYNC|FD_TELL|FD_WRITE|FD_ADVISE|FD_ALLOCATE,fs_rights_inheriting=},errno=ESUCCESS)
   308  `,
   309  		},
   310  		{
   311  			name: "root",
   312  			fd:   sys.FdPreopen,
   313  			expectedMemory: []byte{
   314  				3, 0, // fs_filetype
   315  				0, 0, 0, 0, 0, 0, // fs_flags
   316  				0x19, 0xfe, 0xbf, 0x7, 0x0, 0x0, 0x0, 0x0, // fs_rights_base
   317  				0xff, 0xff, 0xff, 0xf, 0x0, 0x0, 0x0, 0x0, // fs_rights_inheriting
   318  			},
   319  			expectedLog: `
   320  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=3)
   321  <== (stat={filetype=DIRECTORY,fdflags=,fs_rights_base=FD_DATASYNC|FDSTAT_SET_FLAGS|FD_SYNC|PATH_CREATE_DIRECTORY|PATH_CREATE_FILE|PATH_LINK_SOURCE|PATH_LINK_TARGET|PATH_OPEN|FD_READDIR|PATH_READLINK,fs_rights_inheriting=FD_DATASYNC|FD_READ|FD_SEEK|FDSTAT_SET_FLAGS|FD_SYNC|FD_TELL|FD_WRITE|FD_ADVISE|FD_ALLOCATE|PATH_CREATE_DIRECTORY|PATH_CREATE_FILE|PATH_LINK_SOURCE|PATH_LINK_TARGET|PATH_OPEN|FD_READDIR|PATH_READLINK},errno=ESUCCESS)
   322  `,
   323  		},
   324  		{
   325  			name: "file",
   326  			fd:   fileFD,
   327  			expectedMemory: []byte{
   328  				4, 0, // fs_filetype
   329  				0, 0, 0, 0, 0, 0, // fs_flags
   330  				0xff, 0x1, 0xe0, 0x8, 0x0, 0x0, 0x0, 0x0, // fs_rights_base
   331  				0, 0, 0, 0, 0, 0, 0, 0, // fs_rights_inheriting
   332  			},
   333  			expectedLog: `
   334  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=4)
   335  <== (stat={filetype=REGULAR_FILE,fdflags=,fs_rights_base=FD_DATASYNC|FD_READ|FD_SEEK|FDSTAT_SET_FLAGS|FD_SYNC|FD_TELL|FD_WRITE|FD_ADVISE|FD_ALLOCATE,fs_rights_inheriting=},errno=ESUCCESS)
   336  `,
   337  		},
   338  		{
   339  			name: "dir",
   340  			fd:   dirFD,
   341  			expectedMemory: []byte{
   342  				3, 0, // fs_filetype
   343  				0, 0, 0, 0, 0, 0, // fs_flags
   344  				0x19, 0xfe, 0xbf, 0x7, 0x0, 0x0, 0x0, 0x0, // fs_rights_base
   345  				0xff, 0xff, 0xff, 0xf, 0x0, 0x0, 0x0, 0x0, // fs_rights_inheriting
   346  			},
   347  			expectedLog: `
   348  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=5)
   349  <== (stat={filetype=DIRECTORY,fdflags=,fs_rights_base=FD_DATASYNC|FDSTAT_SET_FLAGS|FD_SYNC|PATH_CREATE_DIRECTORY|PATH_CREATE_FILE|PATH_LINK_SOURCE|PATH_LINK_TARGET|PATH_OPEN|FD_READDIR|PATH_READLINK,fs_rights_inheriting=FD_DATASYNC|FD_READ|FD_SEEK|FDSTAT_SET_FLAGS|FD_SYNC|FD_TELL|FD_WRITE|FD_ADVISE|FD_ALLOCATE|PATH_CREATE_DIRECTORY|PATH_CREATE_FILE|PATH_LINK_SOURCE|PATH_LINK_TARGET|PATH_OPEN|FD_READDIR|PATH_READLINK},errno=ESUCCESS)
   350  `,
   351  		},
   352  		{
   353  			name:          "bad FD",
   354  			fd:            -1,
   355  			expectedErrno: wasip1.ErrnoBadf,
   356  			expectedLog: `
   357  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=-1)
   358  <== (stat=,errno=EBADF)
   359  `,
   360  		},
   361  		{
   362  			name:          "resultFdstat exceeds the maximum valid address by 1",
   363  			fd:            dirFD,
   364  			resultFdstat:  memorySize - 24 + 1,
   365  			expectedErrno: wasip1.ErrnoFault,
   366  			expectedLog: `
   367  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=5)
   368  <== (stat=,errno=EFAULT)
   369  `,
   370  		},
   371  	}
   372  
   373  	for _, tt := range tests {
   374  		tc := tt
   375  
   376  		t.Run(tc.name, func(t *testing.T) {
   377  			defer log.Reset()
   378  
   379  			maskMemory(t, mod, len(tc.expectedMemory))
   380  
   381  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdFdstatGetName, uint64(tc.fd), uint64(tc.resultFdstat))
   382  			require.Equal(t, tc.expectedLog, "\n"+log.String())
   383  
   384  			actual, ok := mod.Memory().Read(0, uint32(len(tc.expectedMemory)))
   385  			require.True(t, ok)
   386  			require.Equal(t, tc.expectedMemory, actual)
   387  		})
   388  	}
   389  }
   390  
   391  func Test_fdFdstatGet_StdioNonblock(t *testing.T) {
   392  	stdinR, stdinW := openPipe(t)
   393  	defer closePipe(stdinR, stdinW)
   394  
   395  	stdoutR, stdoutW := openPipe(t)
   396  	defer closePipe(stdoutR, stdoutW)
   397  
   398  	stderrR, stderrW := openPipe(t)
   399  	defer closePipe(stderrR, stderrW)
   400  
   401  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().
   402  		WithStdin(stdinR).
   403  		WithStdout(stdoutW).
   404  		WithStderr(stderrW))
   405  	defer r.Close(testCtx)
   406  
   407  	stdin, stdout, stderr := uint64(0), uint64(1), uint64(2)
   408  	requireErrnoResult(t, 0, mod, wasip1.FdFdstatSetFlagsName, stdin, uint64(wasip1.FD_NONBLOCK))
   409  	requireErrnoResult(t, 0, mod, wasip1.FdFdstatSetFlagsName, stdout, uint64(wasip1.FD_NONBLOCK))
   410  	requireErrnoResult(t, 0, mod, wasip1.FdFdstatSetFlagsName, stderr, uint64(wasip1.FD_NONBLOCK))
   411  	log.Reset()
   412  
   413  	tests := []struct {
   414  		name           string
   415  		fd             int32
   416  		resultFdstat   uint32
   417  		expectedMemory []byte
   418  		expectedErrno  wasip1.Errno
   419  		expectedLog    string
   420  	}{
   421  		{
   422  			name: "stdin",
   423  			fd:   sys.FdStdin,
   424  			expectedMemory: []byte{
   425  				0, 0, // fs_filetype
   426  				5, 0, 0, 0, 0, 0, // fs_flags
   427  				0xff, 0x1, 0xe0, 0x8, 0x0, 0x0, 0x0, 0x0, // fs_rights_base
   428  				0, 0, 0, 0, 0, 0, 0, 0, // fs_rights_inheriting
   429  			},
   430  			expectedLog: `
   431  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=0)
   432  <== (stat={filetype=UNKNOWN,fdflags=APPEND|NONBLOCK,fs_rights_base=FD_DATASYNC|FD_READ|FD_SEEK|FDSTAT_SET_FLAGS|FD_SYNC|FD_TELL|FD_WRITE|FD_ADVISE|FD_ALLOCATE,fs_rights_inheriting=},errno=ESUCCESS)
   433  `,
   434  		},
   435  		{
   436  			name: "stdout",
   437  			fd:   sys.FdStdout,
   438  			expectedMemory: []byte{
   439  				0, 0, // fs_filetype
   440  				5, 0, 0, 0, 0, 0, // fs_flags
   441  				0xff, 0x1, 0xe0, 0x8, 0x0, 0x0, 0x0, 0x0, // fs_rights_base
   442  				0, 0, 0, 0, 0, 0, 0, 0, // fs_rights_inheriting
   443  			},
   444  			expectedLog: `
   445  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=1)
   446  <== (stat={filetype=UNKNOWN,fdflags=APPEND|NONBLOCK,fs_rights_base=FD_DATASYNC|FD_READ|FD_SEEK|FDSTAT_SET_FLAGS|FD_SYNC|FD_TELL|FD_WRITE|FD_ADVISE|FD_ALLOCATE,fs_rights_inheriting=},errno=ESUCCESS)
   447  `,
   448  		},
   449  		{
   450  			name: "stderr",
   451  			fd:   sys.FdStderr,
   452  			expectedMemory: []byte{
   453  				0, 0, // fs_filetype
   454  				5, 0, 0, 0, 0, 0, // fs_flags
   455  				0xff, 0x1, 0xe0, 0x8, 0x0, 0x0, 0x0, 0x0, // fs_rights_base
   456  				0, 0, 0, 0, 0, 0, 0, 0, // fs_rights_inheriting
   457  			},
   458  			expectedLog: `
   459  ==> wasi_snapshot_preview1.fd_fdstat_get(fd=2)
   460  <== (stat={filetype=UNKNOWN,fdflags=APPEND|NONBLOCK,fs_rights_base=FD_DATASYNC|FD_READ|FD_SEEK|FDSTAT_SET_FLAGS|FD_SYNC|FD_TELL|FD_WRITE|FD_ADVISE|FD_ALLOCATE,fs_rights_inheriting=},errno=ESUCCESS)
   461  `,
   462  		},
   463  	}
   464  
   465  	for _, tt := range tests {
   466  		tc := tt
   467  
   468  		t.Run(tc.name, func(t *testing.T) {
   469  			defer log.Reset()
   470  
   471  			maskMemory(t, mod, len(tc.expectedMemory))
   472  
   473  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdFdstatGetName, uint64(tc.fd), uint64(tc.resultFdstat))
   474  			require.Equal(t, tc.expectedLog, "\n"+log.String())
   475  
   476  			actual, ok := mod.Memory().Read(0, uint32(len(tc.expectedMemory)))
   477  			require.True(t, ok)
   478  			require.Equal(t, tc.expectedMemory, actual)
   479  		})
   480  	}
   481  }
   482  
   483  func Test_fdFdstatSetFlagsWithTrunc(t *testing.T) {
   484  	tmpDir := t.TempDir()
   485  	fileName := "test"
   486  
   487  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().
   488  		WithFSConfig(wazero.NewFSConfig().WithDirMount(tmpDir, "/")))
   489  	defer r.Close(testCtx)
   490  
   491  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
   492  	preopen := fsc.RootFS()
   493  
   494  	fd, errno := fsc.OpenFile(preopen, fileName, experimentalsys.O_RDWR|experimentalsys.O_CREAT|experimentalsys.O_EXCL|experimentalsys.O_TRUNC, 0o600)
   495  	require.EqualErrno(t, 0, errno)
   496  
   497  	// Write the initial text to the file.
   498  	f, ok := fsc.LookupFile(fd)
   499  	require.True(t, ok)
   500  	n, _ := f.File.Write([]byte("abc"))
   501  	require.Equal(t, n, 3)
   502  
   503  	buf, err := os.ReadFile(joinPath(tmpDir, fileName))
   504  	require.NoError(t, err)
   505  	require.Equal(t, "abc", string(buf))
   506  
   507  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdFdstatSetFlagsName, uint64(fd), uint64(0))
   508  	require.Equal(t, `
   509  ==> wasi_snapshot_preview1.fd_fdstat_set_flags(fd=4,flags=)
   510  <== errno=ESUCCESS
   511  `, "\n"+log.String())
   512  	log.Reset()
   513  
   514  	buf, err = os.ReadFile(joinPath(tmpDir, fileName))
   515  	require.NoError(t, err)
   516  	require.Equal(t, "abc", string(buf))
   517  }
   518  
   519  func Test_fdFdstatSetFlags(t *testing.T) {
   520  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
   521  
   522  	stdinR, stdinW := openPipe(t)
   523  	defer closePipe(stdinR, stdinW)
   524  
   525  	stdoutR, stdoutW := openPipe(t)
   526  	defer closePipe(stdoutR, stdoutW)
   527  
   528  	stderrR, stderrW := openPipe(t)
   529  	defer closePipe(stderrR, stderrW)
   530  
   531  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().
   532  		WithStdin(stdinR).
   533  		WithStdout(stdoutW).
   534  		WithStderr(stderrW).
   535  		WithFSConfig(wazero.NewFSConfig().WithDirMount(tmpDir, "/")))
   536  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
   537  	preopen := fsc.RootFS()
   538  	defer r.Close(testCtx)
   539  
   540  	// First, O_CREAT the file with O_APPEND. We use O_EXCL because that
   541  	// triggers an EEXIST error if called a second time with O_CREAT. Our
   542  	// logic should clear O_CREAT preventing this.
   543  	const fileName = "file.txt"
   544  	// Create the target file.
   545  	fd, errno := fsc.OpenFile(preopen, fileName, experimentalsys.O_RDWR|experimentalsys.O_APPEND|experimentalsys.O_CREAT|experimentalsys.O_EXCL, 0o600)
   546  	require.EqualErrno(t, 0, errno)
   547  
   548  	// Write the initial text to the file.
   549  	f, ok := fsc.LookupFile(fd)
   550  	require.True(t, ok)
   551  	_, errno = f.File.Write([]byte("0123456789"))
   552  	require.EqualErrno(t, 0, errno)
   553  
   554  	writeWazero := func() {
   555  		iovs := uint32(1) // arbitrary offset
   556  		initialMemory := []byte{
   557  			'?',         // `iovs` is after this
   558  			18, 0, 0, 0, // = iovs[0].offset
   559  			4, 0, 0, 0, // = iovs[0].length
   560  			23, 0, 0, 0, // = iovs[1].offset
   561  			2, 0, 0, 0, // = iovs[1].length
   562  			'?',                // iovs[0].offset is after this
   563  			'w', 'a', 'z', 'e', // iovs[0].length bytes
   564  			'?',      // iovs[1].offset is after this
   565  			'r', 'o', // iovs[1].length bytes
   566  			'?',
   567  		}
   568  		iovsCount := uint32(2)       // The count of iovs
   569  		resultNwritten := uint32(26) // arbitrary offset
   570  
   571  		ok := mod.Memory().Write(0, initialMemory)
   572  		require.True(t, ok)
   573  
   574  		requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdWriteName, uint64(fd), uint64(iovs), uint64(iovsCount), uint64(resultNwritten))
   575  		require.Equal(t, `
   576  ==> wasi_snapshot_preview1.fd_write(fd=4,iovs=1,iovs_len=2)
   577  <== (nwritten=6,errno=ESUCCESS)
   578  `, "\n"+log.String())
   579  		log.Reset()
   580  	}
   581  
   582  	requireFileContent := func(exp string) {
   583  		buf, err := os.ReadFile(joinPath(tmpDir, fileName))
   584  		require.NoError(t, err)
   585  		require.Equal(t, exp, string(buf))
   586  	}
   587  
   588  	// with O_APPEND flag, the data is appended to buffer.
   589  	writeWazero()
   590  	requireFileContent("0123456789" + "wazero")
   591  
   592  	// Let's remove O_APPEND.
   593  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdFdstatSetFlagsName, uint64(fd), uint64(0))
   594  	require.Equal(t, `
   595  ==> wasi_snapshot_preview1.fd_fdstat_set_flags(fd=4,flags=)
   596  <== errno=ESUCCESS
   597  `, "\n"+log.String()) // FIXME? flags==0 prints 'flags='
   598  	log.Reset()
   599  
   600  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdSeekName, uint64(fd), uint64(0), uint64(0), uint64(1024))
   601  	require.Equal(t, `
   602  ==> wasi_snapshot_preview1.fd_seek(fd=4,offset=0,whence=0)
   603  <== (newoffset=0,errno=ESUCCESS)
   604  `, "\n"+log.String())
   605  	log.Reset()
   606  
   607  	// Without O_APPEND flag, the data is written at the beginning.
   608  	writeWazero()
   609  	requireFileContent("wazero6789" + "wazero")
   610  
   611  	// Restore the O_APPEND flag.
   612  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdFdstatSetFlagsName, uint64(fd), uint64(wasip1.FD_APPEND))
   613  	require.Equal(t, `
   614  ==> wasi_snapshot_preview1.fd_fdstat_set_flags(fd=4,flags=APPEND)
   615  <== errno=ESUCCESS
   616  `, "\n"+log.String()) // FIXME? flags==1 prints 'flags=APPEND'
   617  	log.Reset()
   618  
   619  	// Restoring the O_APPEND flag should not reset fd offset.
   620  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdTellName, uint64(fd), uint64(1024))
   621  	require.Equal(t, `
   622  ==> wasi_snapshot_preview1.fd_tell(fd=4,result.offset=1024)
   623  <== errno=ESUCCESS
   624  `, "\n"+log.String())
   625  	log.Reset()
   626  	offset, _ := mod.Memory().Read(1024, 4)
   627  	require.Equal(t, offset, []byte{6, 0, 0, 0})
   628  
   629  	// with O_APPEND flag, the data is appended to buffer.
   630  	writeWazero()
   631  	requireFileContent("wazero6789" + "wazero" + "wazero")
   632  
   633  	t.Run("nonblock", func(t *testing.T) {
   634  		stdin, stdout, stderr := uint64(0), uint64(1), uint64(2)
   635  		requireErrnoResult(t, 0, mod, wasip1.FdFdstatSetFlagsName, stdin, uint64(wasip1.FD_NONBLOCK))
   636  		requireErrnoResult(t, 0, mod, wasip1.FdFdstatSetFlagsName, stdout, uint64(wasip1.FD_NONBLOCK))
   637  		requireErrnoResult(t, 0, mod, wasip1.FdFdstatSetFlagsName, stderr, uint64(wasip1.FD_NONBLOCK))
   638  	})
   639  
   640  	t.Run("errors", func(t *testing.T) {
   641  		requireErrnoResult(t, wasip1.ErrnoInval, mod, wasip1.FdFdstatSetFlagsName, uint64(fd), uint64(wasip1.FD_DSYNC))
   642  		requireErrnoResult(t, wasip1.ErrnoInval, mod, wasip1.FdFdstatSetFlagsName, uint64(fd), uint64(wasip1.FD_RSYNC))
   643  		requireErrnoResult(t, wasip1.ErrnoInval, mod, wasip1.FdFdstatSetFlagsName, uint64(fd), uint64(wasip1.FD_SYNC))
   644  		requireErrnoResult(t, wasip1.ErrnoBadf, mod, wasip1.FdFdstatSetFlagsName, uint64(12345), uint64(wasip1.FD_APPEND))
   645  		requireErrnoResult(t, wasip1.ErrnoIsdir, mod, wasip1.FdFdstatSetFlagsName, uint64(3) /* preopen */, uint64(wasip1.FD_APPEND))
   646  		requireErrnoResult(t, wasip1.ErrnoIsdir, mod, wasip1.FdFdstatSetFlagsName, uint64(3), uint64(wasip1.FD_NONBLOCK))
   647  	})
   648  }
   649  
   650  // Test_fdFdstatSetRights only tests it is stubbed for GrainLang per #271
   651  func Test_fdFdstatSetRights(t *testing.T) {
   652  	log := requireErrnoNosys(t, wasip1.FdFdstatSetRightsName, 0, 0, 0)
   653  	require.Equal(t, `
   654  ==> wasi_snapshot_preview1.fd_fdstat_set_rights(fd=0,fs_rights_base=,fs_rights_inheriting=)
   655  <== errno=ENOSYS
   656  `, log)
   657  }
   658  
   659  func Test_fdFilestatGet(t *testing.T) {
   660  	file, dir := "animals.txt", "sub"
   661  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFS(fstest.FS))
   662  	defer r.Close(testCtx)
   663  	memorySize := mod.Memory().Size()
   664  
   665  	// open both paths without using WASI
   666  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
   667  	preopen := fsc.RootFS()
   668  
   669  	fileFD, errno := fsc.OpenFile(preopen, file, experimentalsys.O_RDONLY, 0)
   670  	require.EqualErrno(t, 0, errno)
   671  
   672  	dirFD, errno := fsc.OpenFile(preopen, dir, experimentalsys.O_RDONLY, 0)
   673  	require.EqualErrno(t, 0, errno)
   674  
   675  	tests := []struct {
   676  		name           string
   677  		fd             int32
   678  		resultFilestat uint32
   679  		expectedMemory []byte
   680  		expectedErrno  wasip1.Errno
   681  		expectedLog    string
   682  	}{
   683  		{
   684  			name: "stdin",
   685  			fd:   sys.FdStdin,
   686  			expectedMemory: []byte{
   687  				0, 0, 0, 0, 0, 0, 0, 0, // dev
   688  				0, 0, 0, 0, 0, 0, 0, 0, // ino
   689  				// expect block device because stdin isn't a real file
   690  				1, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
   691  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
   692  				0, 0, 0, 0, 0, 0, 0, 0, // size
   693  				0, 0, 0, 0, 0, 0, 0, 0, // atim
   694  				0, 0, 0, 0, 0, 0, 0, 0, // mtim
   695  				0, 0, 0, 0, 0, 0, 0, 0, // ctim
   696  			},
   697  			expectedLog: `
   698  ==> wasi_snapshot_preview1.fd_filestat_get(fd=0)
   699  <== (filestat={filetype=BLOCK_DEVICE,size=0,mtim=0},errno=ESUCCESS)
   700  `,
   701  		},
   702  		{
   703  			name: "stdout",
   704  			fd:   sys.FdStdout,
   705  			expectedMemory: []byte{
   706  				0, 0, 0, 0, 0, 0, 0, 0, // dev
   707  				0, 0, 0, 0, 0, 0, 0, 0, // ino
   708  				// expect block device because stdout isn't a real file
   709  				1, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
   710  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
   711  				0, 0, 0, 0, 0, 0, 0, 0, // size
   712  				0, 0, 0, 0, 0, 0, 0, 0, // atim
   713  				0, 0, 0, 0, 0, 0, 0, 0, // mtim
   714  				0, 0, 0, 0, 0, 0, 0, 0, // ctim
   715  			},
   716  			expectedLog: `
   717  ==> wasi_snapshot_preview1.fd_filestat_get(fd=1)
   718  <== (filestat={filetype=BLOCK_DEVICE,size=0,mtim=0},errno=ESUCCESS)
   719  `,
   720  		},
   721  		{
   722  			name: "stderr",
   723  			fd:   sys.FdStderr,
   724  			expectedMemory: []byte{
   725  				0, 0, 0, 0, 0, 0, 0, 0, // dev
   726  				0, 0, 0, 0, 0, 0, 0, 0, // ino
   727  				// expect block device because stderr isn't a real file
   728  				1, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
   729  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
   730  				0, 0, 0, 0, 0, 0, 0, 0, // size
   731  				0, 0, 0, 0, 0, 0, 0, 0, // atim
   732  				0, 0, 0, 0, 0, 0, 0, 0, // mtim
   733  				0, 0, 0, 0, 0, 0, 0, 0, // ctim
   734  			},
   735  			expectedLog: `
   736  ==> wasi_snapshot_preview1.fd_filestat_get(fd=2)
   737  <== (filestat={filetype=BLOCK_DEVICE,size=0,mtim=0},errno=ESUCCESS)
   738  `,
   739  		},
   740  		{
   741  			name: "root",
   742  			fd:   sys.FdPreopen,
   743  			expectedMemory: []byte{
   744  				0, 0, 0, 0, 0, 0, 0, 0, // dev
   745  				0, 0, 0, 0, 0, 0, 0, 0, // ino
   746  				3, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
   747  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
   748  				0, 0, 0, 0, 0, 0, 0, 0, // size
   749  				0x0, 0x0, 0x7c, 0x78, 0x9d, 0xf2, 0x55, 0x16, // atim
   750  				0x0, 0x0, 0x7c, 0x78, 0x9d, 0xf2, 0x55, 0x16, // mtim
   751  				0x0, 0x0, 0x7c, 0x78, 0x9d, 0xf2, 0x55, 0x16, // ctim
   752  			},
   753  			expectedLog: `
   754  ==> wasi_snapshot_preview1.fd_filestat_get(fd=3)
   755  <== (filestat={filetype=DIRECTORY,size=0,mtim=1609459200000000000},errno=ESUCCESS)
   756  `,
   757  		},
   758  		{
   759  			name: "file",
   760  			fd:   fileFD,
   761  			expectedMemory: []byte{
   762  				0, 0, 0, 0, 0, 0, 0, 0, // dev
   763  				0, 0, 0, 0, 0, 0, 0, 0, // ino
   764  				4, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
   765  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
   766  				30, 0, 0, 0, 0, 0, 0, 0, // size
   767  				0x0, 0x82, 0x13, 0x80, 0x6b, 0x16, 0x24, 0x17, // atim
   768  				0x0, 0x82, 0x13, 0x80, 0x6b, 0x16, 0x24, 0x17, // mtim
   769  				0x0, 0x82, 0x13, 0x80, 0x6b, 0x16, 0x24, 0x17, // ctim
   770  			},
   771  			expectedLog: `
   772  ==> wasi_snapshot_preview1.fd_filestat_get(fd=4)
   773  <== (filestat={filetype=REGULAR_FILE,size=30,mtim=1667482413000000000},errno=ESUCCESS)
   774  `,
   775  		},
   776  		{
   777  			name: "dir",
   778  			fd:   dirFD,
   779  			expectedMemory: []byte{
   780  				0, 0, 0, 0, 0, 0, 0, 0, // dev
   781  				0, 0, 0, 0, 0, 0, 0, 0, // ino
   782  				3, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
   783  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
   784  				0, 0, 0, 0, 0, 0, 0, 0, // size
   785  				0x0, 0x0, 0x1f, 0xa6, 0x70, 0xfc, 0xc5, 0x16, // atim
   786  				0x0, 0x0, 0x1f, 0xa6, 0x70, 0xfc, 0xc5, 0x16, // mtim
   787  				0x0, 0x0, 0x1f, 0xa6, 0x70, 0xfc, 0xc5, 0x16, // ctim
   788  			},
   789  			expectedLog: `
   790  ==> wasi_snapshot_preview1.fd_filestat_get(fd=5)
   791  <== (filestat={filetype=DIRECTORY,size=0,mtim=1640995200000000000},errno=ESUCCESS)
   792  `,
   793  		},
   794  		{
   795  			name:          "bad FD",
   796  			fd:            -1,
   797  			expectedErrno: wasip1.ErrnoBadf,
   798  			expectedLog: `
   799  ==> wasi_snapshot_preview1.fd_filestat_get(fd=-1)
   800  <== (filestat=,errno=EBADF)
   801  `,
   802  		},
   803  		{
   804  			name:           "resultFilestat exceeds the maximum valid address by 1",
   805  			fd:             dirFD,
   806  			resultFilestat: memorySize - 64 + 1,
   807  			expectedErrno:  wasip1.ErrnoFault,
   808  			expectedLog: `
   809  ==> wasi_snapshot_preview1.fd_filestat_get(fd=5)
   810  <== (filestat=,errno=EFAULT)
   811  `,
   812  		},
   813  	}
   814  
   815  	for _, tt := range tests {
   816  		tc := tt
   817  
   818  		t.Run(tc.name, func(t *testing.T) {
   819  			defer log.Reset()
   820  
   821  			maskMemory(t, mod, len(tc.expectedMemory))
   822  
   823  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdFilestatGetName, uint64(tc.fd), uint64(tc.resultFilestat))
   824  			require.Equal(t, tc.expectedLog, "\n"+log.String())
   825  
   826  			actual, ok := mod.Memory().Read(0, uint32(len(tc.expectedMemory)))
   827  			require.True(t, ok)
   828  			require.Equal(t, tc.expectedMemory, actual)
   829  		})
   830  	}
   831  }
   832  
   833  func Test_fdFilestatSetSize(t *testing.T) {
   834  	tmpDir := t.TempDir()
   835  
   836  	tests := []struct {
   837  		name                     string
   838  		size                     uint64
   839  		content, expectedContent []byte
   840  		expectedLog              string
   841  		expectedErrno            wasip1.Errno
   842  	}{
   843  		{
   844  			name:            "badf",
   845  			content:         []byte("badf"),
   846  			expectedContent: []byte("badf"),
   847  			expectedErrno:   wasip1.ErrnoBadf,
   848  			expectedLog: `
   849  ==> wasi_snapshot_preview1.fd_filestat_set_size(fd=5,size=0)
   850  <== errno=EBADF
   851  `,
   852  		},
   853  		{
   854  			name:            "truncate",
   855  			content:         []byte("123456"),
   856  			expectedContent: []byte("12345"),
   857  			size:            5,
   858  			expectedErrno:   wasip1.ErrnoSuccess,
   859  			expectedLog: `
   860  ==> wasi_snapshot_preview1.fd_filestat_set_size(fd=4,size=5)
   861  <== errno=ESUCCESS
   862  `,
   863  		},
   864  		{
   865  			name:            "truncate to zero",
   866  			content:         []byte("123456"),
   867  			expectedContent: []byte(""),
   868  			size:            0,
   869  			expectedErrno:   wasip1.ErrnoSuccess,
   870  			expectedLog: `
   871  ==> wasi_snapshot_preview1.fd_filestat_set_size(fd=4,size=0)
   872  <== errno=ESUCCESS
   873  `,
   874  		},
   875  		{
   876  			name:            "truncate to expand",
   877  			content:         []byte("123456"),
   878  			expectedContent: append([]byte("123456"), make([]byte, 100)...),
   879  			size:            106,
   880  			expectedErrno:   wasip1.ErrnoSuccess,
   881  			expectedLog: `
   882  ==> wasi_snapshot_preview1.fd_filestat_set_size(fd=4,size=106)
   883  <== errno=ESUCCESS
   884  `,
   885  		},
   886  		{
   887  			name:            "large size",
   888  			content:         []byte(""),
   889  			expectedContent: []byte(""),
   890  			size:            math.MaxUint64,
   891  			expectedErrno:   wasip1.ErrnoInval,
   892  			expectedLog: `
   893  ==> wasi_snapshot_preview1.fd_filestat_set_size(fd=4,size=-1)
   894  <== errno=EINVAL
   895  `,
   896  		},
   897  	}
   898  
   899  	for _, tt := range tests {
   900  		tc := tt
   901  		t.Run(tc.name, func(t *testing.T) {
   902  			filepath := path.Base(t.Name())
   903  			mod, fd, log, r := requireOpenFile(t, tmpDir, filepath, tc.content, false)
   904  			defer r.Close(testCtx)
   905  
   906  			if filepath == "badf" {
   907  				fd++
   908  			}
   909  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdFilestatSetSizeName, uint64(fd), uint64(tc.size))
   910  
   911  			actual, err := os.ReadFile(joinPath(tmpDir, filepath))
   912  			require.NoError(t, err)
   913  			require.Equal(t, tc.expectedContent, actual)
   914  
   915  			require.Equal(t, tc.expectedLog, "\n"+log.String())
   916  		})
   917  	}
   918  }
   919  
   920  func Test_fdFilestatSetTimes(t *testing.T) {
   921  	tmpDir := t.TempDir()
   922  
   923  	tests := []struct {
   924  		name          string
   925  		mtime, atime  int64
   926  		flags         uint16
   927  		expectedLog   string
   928  		expectedErrno wasip1.Errno
   929  	}{
   930  		{
   931  			name:          "badf",
   932  			expectedErrno: wasip1.ErrnoBadf,
   933  			expectedLog: `
   934  ==> wasi_snapshot_preview1.fd_filestat_set_times(fd=-1,atim=0,mtim=0,fst_flags=)
   935  <== errno=EBADF
   936  `,
   937  		},
   938  		{
   939  			name:          "a=omit,m=omit",
   940  			mtime:         1234,   // Must be ignored.
   941  			atime:         123451, // Must be ignored.
   942  			expectedErrno: wasip1.ErrnoSuccess,
   943  			expectedLog: `
   944  ==> wasi_snapshot_preview1.fd_filestat_set_times(fd=4,atim=123451,mtim=1234,fst_flags=)
   945  <== errno=ESUCCESS
   946  `,
   947  		},
   948  		{
   949  			name:          "a=now,m=omit",
   950  			expectedErrno: wasip1.ErrnoSuccess,
   951  			mtime:         1234,   // Must be ignored.
   952  			atime:         123451, // Must be ignored.
   953  			flags:         wasip1.FstflagsAtimNow,
   954  			expectedLog: `
   955  ==> wasi_snapshot_preview1.fd_filestat_set_times(fd=4,atim=123451,mtim=1234,fst_flags=ATIM_NOW)
   956  <== errno=ESUCCESS
   957  `,
   958  		},
   959  		{
   960  			name:          "a=omit,m=now",
   961  			expectedErrno: wasip1.ErrnoSuccess,
   962  			mtime:         1234,   // Must be ignored.
   963  			atime:         123451, // Must be ignored.
   964  			flags:         wasip1.FstflagsMtimNow,
   965  			expectedLog: `
   966  ==> wasi_snapshot_preview1.fd_filestat_set_times(fd=4,atim=123451,mtim=1234,fst_flags=MTIM_NOW)
   967  <== errno=ESUCCESS
   968  `,
   969  		},
   970  		{
   971  			name:          "a=now,m=now",
   972  			expectedErrno: wasip1.ErrnoSuccess,
   973  			mtime:         1234,   // Must be ignored.
   974  			atime:         123451, // Must be ignored.
   975  			flags:         wasip1.FstflagsAtimNow | wasip1.FstflagsMtimNow,
   976  			expectedLog: `
   977  ==> wasi_snapshot_preview1.fd_filestat_set_times(fd=4,atim=123451,mtim=1234,fst_flags=ATIM_NOW|MTIM_NOW)
   978  <== errno=ESUCCESS
   979  `,
   980  		},
   981  		{
   982  			name:          "a=now,m=set",
   983  			expectedErrno: wasip1.ErrnoSuccess,
   984  			mtime:         55555500,
   985  			atime:         1234, // Must be ignored.
   986  			flags:         wasip1.FstflagsAtimNow | wasip1.FstflagsMtim,
   987  			expectedLog: `
   988  ==> wasi_snapshot_preview1.fd_filestat_set_times(fd=4,atim=1234,mtim=55555500,fst_flags=ATIM_NOW|MTIM)
   989  <== errno=ESUCCESS
   990  `,
   991  		},
   992  		{
   993  			name:          "a=set,m=now",
   994  			expectedErrno: wasip1.ErrnoSuccess,
   995  			mtime:         1234, // Must be ignored.
   996  			atime:         55555500,
   997  			flags:         wasip1.FstflagsAtim | wasip1.FstflagsMtimNow,
   998  			expectedLog: `
   999  ==> wasi_snapshot_preview1.fd_filestat_set_times(fd=4,atim=55555500,mtim=1234,fst_flags=ATIM|MTIM_NOW)
  1000  <== errno=ESUCCESS
  1001  `,
  1002  		},
  1003  		{
  1004  			name:          "a=set,m=omit",
  1005  			expectedErrno: wasip1.ErrnoSuccess,
  1006  			mtime:         1234, // Must be ignored.
  1007  			atime:         55555500,
  1008  			flags:         wasip1.FstflagsAtim,
  1009  			expectedLog: `
  1010  ==> wasi_snapshot_preview1.fd_filestat_set_times(fd=4,atim=55555500,mtim=1234,fst_flags=ATIM)
  1011  <== errno=ESUCCESS
  1012  `,
  1013  		},
  1014  
  1015  		{
  1016  			name:          "a=omit,m=set",
  1017  			expectedErrno: wasip1.ErrnoSuccess,
  1018  			mtime:         55555500,
  1019  			atime:         1234, // Must be ignored.
  1020  			flags:         wasip1.FstflagsMtim,
  1021  			expectedLog: `
  1022  ==> wasi_snapshot_preview1.fd_filestat_set_times(fd=4,atim=1234,mtim=55555500,fst_flags=MTIM)
  1023  <== errno=ESUCCESS
  1024  `,
  1025  		},
  1026  
  1027  		{
  1028  			name:          "a=set,m=set",
  1029  			expectedErrno: wasip1.ErrnoSuccess,
  1030  			mtime:         55555500,
  1031  			atime:         6666666600,
  1032  			flags:         wasip1.FstflagsAtim | wasip1.FstflagsMtim,
  1033  			expectedLog: `
  1034  ==> wasi_snapshot_preview1.fd_filestat_set_times(fd=4,atim=6666666600,mtim=55555500,fst_flags=ATIM|MTIM)
  1035  <== errno=ESUCCESS
  1036  `,
  1037  		},
  1038  	}
  1039  
  1040  	for _, tt := range tests {
  1041  		tc := tt
  1042  		t.Run(tc.name, func(t *testing.T) {
  1043  			filepath := path.Base(t.Name())
  1044  			mod, fd, log, r := requireOpenFile(t, tmpDir, filepath, []byte("anything"), false)
  1045  			defer r.Close(testCtx)
  1046  
  1047  			sys := mod.(*wasm.ModuleInstance).Sys
  1048  			fsc := sys.FS()
  1049  
  1050  			paramFd := fd
  1051  			if filepath == "badf" {
  1052  				paramFd = -1
  1053  			}
  1054  
  1055  			f, ok := fsc.LookupFile(fd)
  1056  			require.True(t, ok)
  1057  
  1058  			st, errno := f.File.Stat()
  1059  			require.EqualErrno(t, 0, errno)
  1060  			prevAtime, prevMtime := st.Atim, st.Mtim
  1061  
  1062  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdFilestatSetTimesName,
  1063  				uint64(paramFd), uint64(tc.atime), uint64(tc.mtime),
  1064  				uint64(tc.flags),
  1065  			)
  1066  
  1067  			if tc.expectedErrno == wasip1.ErrnoSuccess {
  1068  				f, ok := fsc.LookupFile(fd)
  1069  				require.True(t, ok)
  1070  
  1071  				st, errno = f.File.Stat()
  1072  				require.EqualErrno(t, 0, errno)
  1073  				if tc.flags&wasip1.FstflagsAtim != 0 {
  1074  					require.Equal(t, tc.atime, st.Atim)
  1075  				} else if tc.flags&wasip1.FstflagsAtimNow != 0 {
  1076  					require.True(t, (sys.WalltimeNanos()-st.Atim) < time.Second.Nanoseconds())
  1077  				} else {
  1078  					require.Equal(t, prevAtime, st.Atim)
  1079  				}
  1080  				if tc.flags&wasip1.FstflagsMtim != 0 {
  1081  					require.Equal(t, tc.mtime, st.Mtim)
  1082  				} else if tc.flags&wasip1.FstflagsMtimNow != 0 {
  1083  					require.True(t, (sys.WalltimeNanos()-st.Mtim) < time.Second.Nanoseconds())
  1084  				} else {
  1085  					require.Equal(t, prevMtime, st.Mtim)
  1086  				}
  1087  			}
  1088  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  1089  		})
  1090  	}
  1091  }
  1092  
  1093  func Test_fdPread(t *testing.T) {
  1094  	tmpDir := t.TempDir()
  1095  	mod, fd, log, r := requireOpenFile(t, tmpDir, "test_path", []byte("wazero"), true)
  1096  	defer r.Close(testCtx)
  1097  
  1098  	iovs := uint32(1) // arbitrary offset
  1099  	initialMemory := []byte{
  1100  		'?',         // `iovs` is after this
  1101  		18, 0, 0, 0, // = iovs[0].offset
  1102  		4, 0, 0, 0, // = iovs[0].length
  1103  		23, 0, 0, 0, // = iovs[1].offset
  1104  		2, 0, 0, 0, // = iovs[1].length
  1105  		'?',
  1106  	}
  1107  
  1108  	iovsCount := uint32(2)    // The count of iovs
  1109  	resultNread := uint32(26) // arbitrary offset
  1110  
  1111  	tests := []struct {
  1112  		name           string
  1113  		offset         int64
  1114  		expectedMemory []byte
  1115  		expectedLog    string
  1116  	}{
  1117  		{
  1118  			name:   "offset zero",
  1119  			offset: 0,
  1120  			expectedMemory: append(
  1121  				initialMemory,
  1122  				'w', 'a', 'z', 'e', // iovs[0].length bytes
  1123  				'?',      // iovs[1].offset is after this
  1124  				'r', 'o', // iovs[1].length bytes
  1125  				'?',        // resultNread is after this
  1126  				6, 0, 0, 0, // sum(iovs[...].length) == length of "wazero"
  1127  				'?',
  1128  			),
  1129  			expectedLog: `
  1130  ==> wasi_snapshot_preview1.fd_pread(fd=4,iovs=1,iovs_len=2,offset=0)
  1131  <== (nread=6,errno=ESUCCESS)
  1132  `,
  1133  		},
  1134  		{
  1135  			name:   "offset 2",
  1136  			offset: 2,
  1137  			expectedMemory: append(
  1138  				initialMemory,
  1139  				'z', 'e', 'r', 'o', // iovs[0].length bytes
  1140  				'?', '?', '?', '?', // resultNread is after this
  1141  				4, 0, 0, 0, // sum(iovs[...].length) == length of "zero"
  1142  				'?',
  1143  			),
  1144  			expectedLog: `
  1145  ==> wasi_snapshot_preview1.fd_pread(fd=4,iovs=1,iovs_len=2,offset=2)
  1146  <== (nread=4,errno=ESUCCESS)
  1147  `,
  1148  		},
  1149  	}
  1150  
  1151  	for _, tt := range tests {
  1152  		tc := tt
  1153  		t.Run(tc.name, func(t *testing.T) {
  1154  			defer log.Reset()
  1155  
  1156  			maskMemory(t, mod, len(tc.expectedMemory))
  1157  
  1158  			ok := mod.Memory().Write(0, initialMemory)
  1159  			require.True(t, ok)
  1160  
  1161  			requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdPreadName, uint64(fd), uint64(iovs), uint64(iovsCount), uint64(tc.offset), uint64(resultNread))
  1162  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  1163  
  1164  			actual, ok := mod.Memory().Read(0, uint32(len(tc.expectedMemory)))
  1165  			require.True(t, ok)
  1166  			require.Equal(t, tc.expectedMemory, actual)
  1167  		})
  1168  	}
  1169  }
  1170  
  1171  func Test_fdPread_offset(t *testing.T) {
  1172  	tmpDir := t.TempDir()
  1173  	mod, fd, log, r := requireOpenFile(t, tmpDir, "test_path", []byte("wazero"), true)
  1174  	defer r.Close(testCtx)
  1175  
  1176  	// Do an initial fdPread.
  1177  
  1178  	iovs := uint32(1) // arbitrary offset
  1179  	initialMemory := []byte{
  1180  		'?',         // `iovs` is after this
  1181  		18, 0, 0, 0, // = iovs[0].offset
  1182  		4, 0, 0, 0, // = iovs[0].length
  1183  		23, 0, 0, 0, // = iovs[1].offset
  1184  		2, 0, 0, 0, // = iovs[1].length
  1185  		'?',
  1186  	}
  1187  	iovsCount := uint32(2)    // The count of iovs
  1188  	resultNread := uint32(26) // arbitrary offset
  1189  
  1190  	expectedMemory := append(
  1191  		initialMemory,
  1192  		'z', 'e', 'r', 'o', // iovs[0].length bytes
  1193  		'?', '?', '?', '?', // resultNread is after this
  1194  		4, 0, 0, 0, // sum(iovs[...].length) == length of "zero"
  1195  		'?',
  1196  	)
  1197  
  1198  	maskMemory(t, mod, len(expectedMemory))
  1199  
  1200  	ok := mod.Memory().Write(0, initialMemory)
  1201  	require.True(t, ok)
  1202  
  1203  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdPreadName, uint64(fd), uint64(iovs), uint64(iovsCount), 2, uint64(resultNread))
  1204  	actual, ok := mod.Memory().Read(0, uint32(len(expectedMemory)))
  1205  	require.True(t, ok)
  1206  	require.Equal(t, expectedMemory, actual)
  1207  
  1208  	// Verify that the fdPread didn't affect the fdRead offset.
  1209  
  1210  	expectedMemory = append(
  1211  		initialMemory,
  1212  		'w', 'a', 'z', 'e', // iovs[0].length bytes
  1213  		'?',      // iovs[1].offset is after this
  1214  		'r', 'o', // iovs[1].length bytes
  1215  		'?',        // resultNread is after this
  1216  		6, 0, 0, 0, // sum(iovs[...].length) == length of "wazero"
  1217  		'?',
  1218  	)
  1219  
  1220  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdReadName, uint64(fd), uint64(iovs), uint64(iovsCount), uint64(resultNread))
  1221  	actual, ok = mod.Memory().Read(0, uint32(len(expectedMemory)))
  1222  	require.True(t, ok)
  1223  	require.Equal(t, expectedMemory, actual)
  1224  
  1225  	expectedLog := `
  1226  ==> wasi_snapshot_preview1.fd_pread(fd=4,iovs=1,iovs_len=2,offset=2)
  1227  <== (nread=4,errno=ESUCCESS)
  1228  ==> wasi_snapshot_preview1.fd_read(fd=4,iovs=1,iovs_len=2)
  1229  <== (nread=6,errno=ESUCCESS)
  1230  `
  1231  	require.Equal(t, expectedLog, "\n"+log.String())
  1232  }
  1233  
  1234  func Test_fdPread_Errors(t *testing.T) {
  1235  	tmpDir := t.TempDir()
  1236  	contents := []byte("wazero")
  1237  	mod, fd, log, r := requireOpenFile(t, tmpDir, "test_path", contents, true)
  1238  	defer r.Close(testCtx)
  1239  
  1240  	tests := []struct {
  1241  		name                         string
  1242  		fd                           int32
  1243  		iovs, iovsCount, resultNread uint32
  1244  		offset                       int64
  1245  		memory                       []byte
  1246  		expectedErrno                wasip1.Errno
  1247  		expectedLog                  string
  1248  	}{
  1249  		{
  1250  			name:          "invalid FD",
  1251  			fd:            42,                         // arbitrary invalid fd
  1252  			memory:        []byte{'?', '?', '?', '?'}, // pass result.nread validation
  1253  			expectedErrno: wasip1.ErrnoBadf,
  1254  			expectedLog: `
  1255  ==> wasi_snapshot_preview1.fd_pread(fd=42,iovs=65532,iovs_len=0,offset=0)
  1256  <== (nread=,errno=EBADF)
  1257  `,
  1258  		},
  1259  		{
  1260  			name:          "out-of-memory reading iovs[0].offset",
  1261  			fd:            fd,
  1262  			iovs:          1,
  1263  			memory:        []byte{'?'},
  1264  			expectedErrno: wasip1.ErrnoFault,
  1265  			expectedLog: `
  1266  ==> wasi_snapshot_preview1.fd_pread(fd=4,iovs=65536,iovs_len=0,offset=0)
  1267  <== (nread=,errno=EFAULT)
  1268  `,
  1269  		},
  1270  		{
  1271  			name: "out-of-memory reading iovs[0].length",
  1272  			fd:   fd,
  1273  			iovs: 1, iovsCount: 1,
  1274  			memory: []byte{
  1275  				'?',        // `iovs` is after this
  1276  				9, 0, 0, 0, // = iovs[0].offset
  1277  			},
  1278  			expectedErrno: wasip1.ErrnoFault,
  1279  			expectedLog: `
  1280  ==> wasi_snapshot_preview1.fd_pread(fd=4,iovs=65532,iovs_len=1,offset=0)
  1281  <== (nread=,errno=EFAULT)
  1282  `,
  1283  		},
  1284  		{
  1285  			name: "iovs[0].offset is outside memory",
  1286  			fd:   fd,
  1287  			iovs: 1, iovsCount: 1,
  1288  			memory: []byte{
  1289  				'?',          // `iovs` is after this
  1290  				0, 0, 0x1, 0, // = iovs[0].offset on the second page
  1291  				1, 0, 0, 0, // = iovs[0].length
  1292  			},
  1293  			expectedErrno: wasip1.ErrnoFault,
  1294  			expectedLog: `
  1295  ==> wasi_snapshot_preview1.fd_pread(fd=4,iovs=65528,iovs_len=1,offset=0)
  1296  <== (nread=,errno=EFAULT)
  1297  `,
  1298  		},
  1299  		{
  1300  			name: "length to read exceeds memory by 1",
  1301  			fd:   fd,
  1302  			iovs: 1, iovsCount: 1,
  1303  			memory: []byte{
  1304  				'?',        // `iovs` is after this
  1305  				9, 0, 0, 0, // = iovs[0].offset
  1306  				0, 0, 0x1, 0, // = iovs[0].length on the second page
  1307  				'?',
  1308  			},
  1309  			expectedErrno: wasip1.ErrnoFault,
  1310  			expectedLog: `
  1311  ==> wasi_snapshot_preview1.fd_pread(fd=4,iovs=65527,iovs_len=1,offset=0)
  1312  <== (nread=,errno=EFAULT)
  1313  `,
  1314  		},
  1315  		{
  1316  			name: "resultNread offset is outside memory",
  1317  			fd:   fd,
  1318  			iovs: 1, iovsCount: 1,
  1319  			resultNread: 10, // 1 past memory
  1320  			memory: []byte{
  1321  				'?',        // `iovs` is after this
  1322  				9, 0, 0, 0, // = iovs[0].offset
  1323  				1, 0, 0, 0, // = iovs[0].length
  1324  				'?',
  1325  			},
  1326  			expectedErrno: wasip1.ErrnoFault,
  1327  			expectedLog: `
  1328  ==> wasi_snapshot_preview1.fd_pread(fd=4,iovs=65527,iovs_len=1,offset=0)
  1329  <== (nread=,errno=EFAULT)
  1330  `,
  1331  		},
  1332  		{
  1333  			name: "offset negative",
  1334  			fd:   fd,
  1335  			iovs: 1, iovsCount: 1,
  1336  			resultNread: 10,
  1337  			memory: []byte{
  1338  				'?',        // `iovs` is after this
  1339  				9, 0, 0, 0, // = iovs[0].offset
  1340  				1, 0, 0, 0, // = iovs[0].length
  1341  				'?',
  1342  				'?', '?', '?', '?',
  1343  			},
  1344  			offset:        int64(-1),
  1345  			expectedErrno: wasip1.ErrnoIo,
  1346  			expectedLog: `
  1347  ==> wasi_snapshot_preview1.fd_pread(fd=4,iovs=65523,iovs_len=1,offset=-1)
  1348  <== (nread=,errno=EIO)
  1349  `,
  1350  		},
  1351  	}
  1352  
  1353  	for _, tt := range tests {
  1354  		tc := tt
  1355  		t.Run(tc.name, func(t *testing.T) {
  1356  			defer log.Reset()
  1357  
  1358  			offset := uint32(wasm.MemoryPagesToBytesNum(testMemoryPageSize) - uint64(len(tc.memory)))
  1359  
  1360  			memoryWriteOK := mod.Memory().Write(offset, tc.memory)
  1361  			require.True(t, memoryWriteOK)
  1362  
  1363  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdPreadName, uint64(tc.fd), uint64(tc.iovs+offset), uint64(tc.iovsCount), uint64(tc.offset), uint64(tc.resultNread+offset))
  1364  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  1365  		})
  1366  	}
  1367  }
  1368  
  1369  func Test_fdPrestatGet(t *testing.T) {
  1370  	fsConfig := wazero.NewFSConfig().WithDirMount(t.TempDir(), "/")
  1371  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  1372  	defer r.Close(testCtx)
  1373  
  1374  	resultPrestat := uint32(1) // arbitrary offset
  1375  	expectedMemory := []byte{
  1376  		'?',     // resultPrestat after this
  1377  		0,       // 8-bit tag indicating `prestat_dir`, the only available tag
  1378  		0, 0, 0, // 3-byte padding
  1379  		// the result path length field after this
  1380  		1, 0, 0, 0, // = in little endian encoding
  1381  		'?',
  1382  	}
  1383  
  1384  	maskMemory(t, mod, len(expectedMemory))
  1385  
  1386  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdPrestatGetName, uint64(sys.FdPreopen), uint64(resultPrestat))
  1387  	require.Equal(t, `
  1388  ==> wasi_snapshot_preview1.fd_prestat_get(fd=3)
  1389  <== (prestat={pr_name_len=1},errno=ESUCCESS)
  1390  `, "\n"+log.String())
  1391  
  1392  	actual, ok := mod.Memory().Read(0, uint32(len(expectedMemory)))
  1393  	require.True(t, ok)
  1394  	require.Equal(t, expectedMemory, actual)
  1395  }
  1396  
  1397  func Test_fdPrestatGet_Errors(t *testing.T) {
  1398  	mod, dirFD, log, r := requireOpenFile(t, t.TempDir(), "tmp", nil, true)
  1399  	defer r.Close(testCtx)
  1400  
  1401  	memorySize := mod.Memory().Size()
  1402  	tests := []struct {
  1403  		name          string
  1404  		fd            int32
  1405  		resultPrestat uint32
  1406  		expectedErrno wasip1.Errno
  1407  		expectedLog   string
  1408  	}{
  1409  		{
  1410  			name:          "unopened FD",
  1411  			fd:            42, // arbitrary invalid Fd
  1412  			resultPrestat: 0,  // valid offset
  1413  			expectedErrno: wasip1.ErrnoBadf,
  1414  			expectedLog: `
  1415  ==> wasi_snapshot_preview1.fd_prestat_get(fd=42)
  1416  <== (prestat=,errno=EBADF)
  1417  `,
  1418  		},
  1419  		{
  1420  			name:          "not pre-opened FD",
  1421  			fd:            dirFD,
  1422  			resultPrestat: 0, // valid offset
  1423  			expectedErrno: wasip1.ErrnoBadf,
  1424  			expectedLog: `
  1425  ==> wasi_snapshot_preview1.fd_prestat_get(fd=4)
  1426  <== (prestat=,errno=EBADF)
  1427  `,
  1428  		},
  1429  		{
  1430  			name:          "out-of-memory resultPrestat",
  1431  			fd:            sys.FdPreopen,
  1432  			resultPrestat: memorySize,
  1433  			expectedErrno: wasip1.ErrnoFault,
  1434  			expectedLog: `
  1435  ==> wasi_snapshot_preview1.fd_prestat_get(fd=3)
  1436  <== (prestat=,errno=EFAULT)
  1437  `,
  1438  		},
  1439  	}
  1440  
  1441  	for _, tt := range tests {
  1442  		tc := tt
  1443  
  1444  		t.Run(tc.name, func(t *testing.T) {
  1445  			defer log.Reset()
  1446  
  1447  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdPrestatGetName, uint64(tc.fd), uint64(tc.resultPrestat))
  1448  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  1449  		})
  1450  	}
  1451  }
  1452  
  1453  func Test_fdPrestatDirName(t *testing.T) {
  1454  	fsConfig := wazero.NewFSConfig().WithDirMount(t.TempDir(), "/")
  1455  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  1456  	defer r.Close(testCtx)
  1457  
  1458  	path := uint32(1)    // arbitrary offset
  1459  	pathLen := uint32(0) // shorter than len("/") to prove truncation is ok
  1460  	expectedMemory := []byte{
  1461  		'?', '?', '?', '?',
  1462  	}
  1463  
  1464  	maskMemory(t, mod, len(expectedMemory))
  1465  
  1466  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdPrestatDirNameName, uint64(sys.FdPreopen), uint64(path), uint64(pathLen))
  1467  	require.Equal(t, `
  1468  ==> wasi_snapshot_preview1.fd_prestat_dir_name(fd=3)
  1469  <== (path=,errno=ESUCCESS)
  1470  `, "\n"+log.String())
  1471  
  1472  	actual, ok := mod.Memory().Read(0, uint32(len(expectedMemory)))
  1473  	require.True(t, ok)
  1474  	require.Equal(t, expectedMemory, actual)
  1475  }
  1476  
  1477  func Test_fdPrestatDirName_Errors(t *testing.T) {
  1478  	mod, dirFD, log, r := requireOpenFile(t, t.TempDir(), "tmp", nil, true)
  1479  	defer r.Close(testCtx)
  1480  
  1481  	memorySize := mod.Memory().Size()
  1482  	maskMemory(t, mod, 10)
  1483  
  1484  	validAddress := uint32(0) // Arbitrary valid address as arguments to fd_prestat_dir_name. We chose 0 here.
  1485  	pathLen := uint32(len("/"))
  1486  
  1487  	tests := []struct {
  1488  		name          string
  1489  		fd            int32
  1490  		path          uint32
  1491  		pathLen       uint32
  1492  		expectedErrno wasip1.Errno
  1493  		expectedLog   string
  1494  	}{
  1495  		{
  1496  			name:          "out-of-memory path",
  1497  			fd:            sys.FdPreopen,
  1498  			path:          memorySize,
  1499  			pathLen:       pathLen,
  1500  			expectedErrno: wasip1.ErrnoFault,
  1501  			expectedLog: `
  1502  ==> wasi_snapshot_preview1.fd_prestat_dir_name(fd=3)
  1503  <== (path=,errno=EFAULT)
  1504  `,
  1505  		},
  1506  		{
  1507  			name:          "path exceeds the maximum valid address by 1",
  1508  			fd:            sys.FdPreopen,
  1509  			path:          memorySize - pathLen + 1,
  1510  			pathLen:       pathLen,
  1511  			expectedErrno: wasip1.ErrnoFault,
  1512  			expectedLog: `
  1513  ==> wasi_snapshot_preview1.fd_prestat_dir_name(fd=3)
  1514  <== (path=,errno=EFAULT)
  1515  `,
  1516  		},
  1517  		{
  1518  			name:          "pathLen exceeds the length of the dir name",
  1519  			fd:            sys.FdPreopen,
  1520  			path:          validAddress,
  1521  			pathLen:       pathLen + 1,
  1522  			expectedErrno: wasip1.ErrnoNametoolong,
  1523  			expectedLog: `
  1524  ==> wasi_snapshot_preview1.fd_prestat_dir_name(fd=3)
  1525  <== (path=,errno=ENAMETOOLONG)
  1526  `,
  1527  		},
  1528  		{
  1529  			name:          "unopened FD",
  1530  			fd:            42, // arbitrary invalid fd
  1531  			path:          validAddress,
  1532  			pathLen:       pathLen,
  1533  			expectedErrno: wasip1.ErrnoBadf,
  1534  			expectedLog: `
  1535  ==> wasi_snapshot_preview1.fd_prestat_dir_name(fd=42)
  1536  <== (path=,errno=EBADF)
  1537  `,
  1538  		},
  1539  		{
  1540  			name:          "not pre-opened FD",
  1541  			fd:            dirFD,
  1542  			path:          validAddress,
  1543  			pathLen:       pathLen,
  1544  			expectedErrno: wasip1.ErrnoBadf,
  1545  			expectedLog: `
  1546  ==> wasi_snapshot_preview1.fd_prestat_dir_name(fd=4)
  1547  <== (path=,errno=EBADF)
  1548  `,
  1549  		},
  1550  	}
  1551  
  1552  	for _, tt := range tests {
  1553  		tc := tt
  1554  
  1555  		t.Run(tc.name, func(t *testing.T) {
  1556  			defer log.Reset()
  1557  
  1558  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdPrestatDirNameName, uint64(tc.fd), uint64(tc.path), uint64(tc.pathLen))
  1559  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  1560  		})
  1561  	}
  1562  }
  1563  
  1564  func Test_fdPwrite(t *testing.T) {
  1565  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  1566  	pathName := "test_path"
  1567  	mod, fd, log, r := requireOpenFile(t, tmpDir, pathName, []byte{}, false)
  1568  	defer r.Close(testCtx)
  1569  
  1570  	iovs := uint32(1) // arbitrary offset
  1571  	initialMemory := []byte{
  1572  		'?',         // `iovs` is after this
  1573  		18, 0, 0, 0, // = iovs[0].offset
  1574  		4, 0, 0, 0, // = iovs[0].length
  1575  		23, 0, 0, 0, // = iovs[1].offset
  1576  		2, 0, 0, 0, // = iovs[1].length
  1577  		'?',
  1578  		'w', 'a', 'z', 'e', // iovs[0].length bytes
  1579  		'?',      // iovs[1].offset is after this
  1580  		'r', 'o', // iovs[1].length bytes
  1581  	}
  1582  
  1583  	iovsCount := uint32(2) // The count of iovs
  1584  	resultNwritten := len(initialMemory) + 1
  1585  
  1586  	tests := []struct {
  1587  		name             string
  1588  		offset           int64
  1589  		expectedMemory   []byte
  1590  		expectedContents string
  1591  		expectedLog      string
  1592  	}{
  1593  		{
  1594  			name:   "offset zero",
  1595  			offset: 0,
  1596  			expectedMemory: append(
  1597  				initialMemory,
  1598  				'?',        // resultNwritten is after this
  1599  				6, 0, 0, 0, // sum(iovs[...].length) == length of "wazero"
  1600  				'?',
  1601  			),
  1602  			expectedContents: "wazero",
  1603  			expectedLog: `
  1604  ==> wasi_snapshot_preview1.fd_pwrite(fd=4,iovs=1,iovs_len=2,offset=0)
  1605  <== (nwritten=6,errno=ESUCCESS)
  1606  `,
  1607  		},
  1608  		{
  1609  			name:   "offset 2",
  1610  			offset: 2,
  1611  			expectedMemory: append(
  1612  				initialMemory,
  1613  				'?',        // resultNwritten is after this
  1614  				6, 0, 0, 0, // sum(iovs[...].length) == length of "wazero"
  1615  				'?',
  1616  			),
  1617  			expectedContents: "wawazero", // "wa" from the first test!
  1618  			expectedLog: `
  1619  ==> wasi_snapshot_preview1.fd_pwrite(fd=4,iovs=1,iovs_len=2,offset=2)
  1620  <== (nwritten=6,errno=ESUCCESS)
  1621  `,
  1622  		},
  1623  	}
  1624  
  1625  	for _, tt := range tests {
  1626  		tc := tt
  1627  		t.Run(tc.name, func(t *testing.T) {
  1628  			defer log.Reset()
  1629  
  1630  			maskMemory(t, mod, len(tc.expectedMemory))
  1631  
  1632  			ok := mod.Memory().Write(0, initialMemory)
  1633  			require.True(t, ok)
  1634  
  1635  			requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdPwriteName, uint64(fd), uint64(iovs), uint64(iovsCount), uint64(tc.offset), uint64(resultNwritten))
  1636  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  1637  
  1638  			actual, ok := mod.Memory().Read(0, uint32(len(tc.expectedMemory)))
  1639  			require.True(t, ok)
  1640  			require.Equal(t, tc.expectedMemory, actual)
  1641  
  1642  			// Ensure the contents were really written
  1643  			b, err := os.ReadFile(joinPath(tmpDir, pathName))
  1644  			require.NoError(t, err)
  1645  			require.Equal(t, tc.expectedContents, string(b))
  1646  		})
  1647  	}
  1648  }
  1649  
  1650  func Test_fdPwrite_offset(t *testing.T) {
  1651  	tmpDir := t.TempDir()
  1652  	pathName := "test_path"
  1653  	mod, fd, log, r := requireOpenFile(t, tmpDir, pathName, []byte{}, false)
  1654  	defer r.Close(testCtx)
  1655  
  1656  	// Do an initial fdPwrite.
  1657  
  1658  	iovs := uint32(1) // arbitrary offset
  1659  	pwriteMemory := []byte{
  1660  		'?',         // `iovs` is after this
  1661  		10, 0, 0, 0, // = iovs[0].offset
  1662  		3, 0, 0, 0, // = iovs[0].length
  1663  		'?',
  1664  		'e', 'r', 'o', // iovs[0].length bytes
  1665  		'?', // resultNwritten is after this
  1666  	}
  1667  	iovsCount := uint32(1) // The count of iovs
  1668  	resultNwritten := len(pwriteMemory) + 4
  1669  
  1670  	expectedMemory := append(
  1671  		pwriteMemory,
  1672  		'?', '?', '?', '?', // resultNwritten is after this
  1673  		3, 0, 0, 0, // sum(iovs[...].length) == length of "ero"
  1674  		'?',
  1675  	)
  1676  
  1677  	maskMemory(t, mod, len(expectedMemory))
  1678  
  1679  	ok := mod.Memory().Write(0, pwriteMemory)
  1680  	require.True(t, ok)
  1681  
  1682  	// Write the last half first, to offset 3
  1683  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdPwriteName, uint64(fd), uint64(iovs), uint64(iovsCount), 3, uint64(resultNwritten))
  1684  	actual, ok := mod.Memory().Read(0, uint32(len(expectedMemory)))
  1685  	require.True(t, ok)
  1686  	require.Equal(t, expectedMemory, actual)
  1687  
  1688  	// Verify that the fdPwrite didn't affect the fdWrite offset.
  1689  	writeMemory := []byte{
  1690  		'?',         // `iovs` is after this
  1691  		10, 0, 0, 0, // = iovs[0].offset
  1692  		3, 0, 0, 0, // = iovs[0].length
  1693  		'?',
  1694  		'w', 'a', 'z', // iovs[0].length bytes
  1695  		'?', // resultNwritten is after this
  1696  	}
  1697  	expectedMemory = append(
  1698  		writeMemory,
  1699  		'?', '?', '?', '?', // resultNwritten is after this
  1700  		3, 0, 0, 0, // sum(iovs[...].length) == length of "waz"
  1701  		'?',
  1702  	)
  1703  
  1704  	ok = mod.Memory().Write(0, writeMemory)
  1705  	require.True(t, ok)
  1706  
  1707  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdWriteName, uint64(fd), uint64(iovs), uint64(iovsCount), uint64(resultNwritten))
  1708  	actual, ok = mod.Memory().Read(0, uint32(len(expectedMemory)))
  1709  	require.True(t, ok)
  1710  	require.Equal(t, expectedMemory, actual)
  1711  
  1712  	expectedLog := `
  1713  ==> wasi_snapshot_preview1.fd_pwrite(fd=4,iovs=1,iovs_len=1,offset=3)
  1714  <== (nwritten=3,errno=ESUCCESS)
  1715  ==> wasi_snapshot_preview1.fd_write(fd=4,iovs=1,iovs_len=1)
  1716  <== (nwritten=3,errno=ESUCCESS)
  1717  `
  1718  	require.Equal(t, expectedLog, "\n"+log.String())
  1719  
  1720  	// Ensure the contents were really written
  1721  	b, err := os.ReadFile(joinPath(tmpDir, pathName))
  1722  	require.NoError(t, err)
  1723  	require.Equal(t, "wazero", string(b))
  1724  }
  1725  
  1726  func Test_fdPwrite_Errors(t *testing.T) {
  1727  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  1728  	pathName := "test_path"
  1729  	mod, fd, log, r := requireOpenFile(t, tmpDir, pathName, []byte{}, false)
  1730  	defer r.Close(testCtx)
  1731  
  1732  	tests := []struct {
  1733  		name                            string
  1734  		fd                              int32
  1735  		iovs, iovsCount, resultNwritten uint32
  1736  		offset                          int64
  1737  		memory                          []byte
  1738  		expectedErrno                   wasip1.Errno
  1739  		expectedLog                     string
  1740  	}{
  1741  		{
  1742  			name:          "invalid FD",
  1743  			fd:            42,                         // arbitrary invalid fd
  1744  			memory:        []byte{'?', '?', '?', '?'}, // pass result.nwritten validation
  1745  			expectedErrno: wasip1.ErrnoBadf,
  1746  			expectedLog: `
  1747  ==> wasi_snapshot_preview1.fd_pwrite(fd=42,iovs=65532,iovs_len=0,offset=0)
  1748  <== (nwritten=,errno=EBADF)
  1749  `,
  1750  		},
  1751  		{
  1752  			name:          "out-of-memory writing iovs[0].offset",
  1753  			fd:            fd,
  1754  			iovs:          1,
  1755  			memory:        []byte{'?'},
  1756  			expectedErrno: wasip1.ErrnoFault,
  1757  			expectedLog: `
  1758  ==> wasi_snapshot_preview1.fd_pwrite(fd=4,iovs=65536,iovs_len=0,offset=0)
  1759  <== (nwritten=,errno=EFAULT)
  1760  `,
  1761  		},
  1762  		{
  1763  			name: "out-of-memory writing iovs[0].length",
  1764  			fd:   fd,
  1765  			iovs: 1, iovsCount: 1,
  1766  			memory: []byte{
  1767  				'?',        // `iovs` is after this
  1768  				9, 0, 0, 0, // = iovs[0].offset
  1769  			},
  1770  			expectedErrno: wasip1.ErrnoFault,
  1771  			expectedLog: `
  1772  ==> wasi_snapshot_preview1.fd_pwrite(fd=4,iovs=65532,iovs_len=1,offset=0)
  1773  <== (nwritten=,errno=EFAULT)
  1774  `,
  1775  		},
  1776  		{
  1777  			name: "iovs[0].offset is outside memory",
  1778  			fd:   fd,
  1779  			iovs: 1, iovsCount: 1,
  1780  			memory: []byte{
  1781  				'?',          // `iovs` is after this
  1782  				0, 0, 0x1, 0, // = iovs[0].offset on the second page
  1783  				1, 0, 0, 0, // = iovs[0].length
  1784  			},
  1785  			expectedErrno: wasip1.ErrnoFault,
  1786  			expectedLog: `
  1787  ==> wasi_snapshot_preview1.fd_pwrite(fd=4,iovs=65528,iovs_len=1,offset=0)
  1788  <== (nwritten=,errno=EFAULT)
  1789  `,
  1790  		},
  1791  		{
  1792  			name: "length to write exceeds memory by 1",
  1793  			fd:   fd,
  1794  			iovs: 1, iovsCount: 1,
  1795  			memory: []byte{
  1796  				'?',        // `iovs` is after this
  1797  				9, 0, 0, 0, // = iovs[0].offset
  1798  				0, 0, 0x1, 0, // = iovs[0].length on the second page
  1799  				'?',
  1800  			},
  1801  			expectedErrno: wasip1.ErrnoFault,
  1802  			expectedLog: `
  1803  ==> wasi_snapshot_preview1.fd_pwrite(fd=4,iovs=65527,iovs_len=1,offset=0)
  1804  <== (nwritten=,errno=EFAULT)
  1805  `,
  1806  		},
  1807  		{
  1808  			name: "resultNwritten offset is outside memory",
  1809  			fd:   fd,
  1810  			iovs: 1, iovsCount: 1,
  1811  			resultNwritten: 10, // 1 past memory
  1812  			memory: []byte{
  1813  				'?',        // `iovs` is after this
  1814  				9, 0, 0, 0, // = iovs[0].offset
  1815  				1, 0, 0, 0, // = iovs[0].length
  1816  				'?',
  1817  			},
  1818  			expectedErrno: wasip1.ErrnoFault,
  1819  			expectedLog: `
  1820  ==> wasi_snapshot_preview1.fd_pwrite(fd=4,iovs=65527,iovs_len=1,offset=0)
  1821  <== (nwritten=,errno=EFAULT)
  1822  `,
  1823  		},
  1824  		{
  1825  			name: "offset negative",
  1826  			fd:   fd,
  1827  			iovs: 1, iovsCount: 1,
  1828  			resultNwritten: 10,
  1829  			memory: []byte{
  1830  				'?',        // `iovs` is after this
  1831  				9, 0, 0, 0, // = iovs[0].offset
  1832  				1, 0, 0, 0, // = iovs[0].length
  1833  				'?',
  1834  				'?', '?', '?', '?',
  1835  			},
  1836  			offset:        int64(-1),
  1837  			expectedErrno: wasip1.ErrnoIo,
  1838  			expectedLog: `
  1839  ==> wasi_snapshot_preview1.fd_pwrite(fd=4,iovs=65523,iovs_len=1,offset=-1)
  1840  <== (nwritten=,errno=EIO)
  1841  `,
  1842  		},
  1843  	}
  1844  
  1845  	for _, tt := range tests {
  1846  		tc := tt
  1847  		t.Run(tc.name, func(t *testing.T) {
  1848  			defer log.Reset()
  1849  
  1850  			offset := uint32(wasm.MemoryPagesToBytesNum(testMemoryPageSize) - uint64(len(tc.memory)))
  1851  
  1852  			memoryWriteOK := mod.Memory().Write(offset, tc.memory)
  1853  			require.True(t, memoryWriteOK)
  1854  
  1855  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdPwriteName, uint64(tc.fd), uint64(tc.iovs+offset), uint64(tc.iovsCount), uint64(tc.offset), uint64(tc.resultNwritten+offset))
  1856  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  1857  		})
  1858  	}
  1859  }
  1860  
  1861  func Test_fdRead(t *testing.T) {
  1862  	mod, fd, log, r := requireOpenFile(t, t.TempDir(), "test_path", []byte("wazero"), true)
  1863  	defer r.Close(testCtx)
  1864  
  1865  	iovs := uint32(1) // arbitrary offset
  1866  	initialMemory := []byte{
  1867  		'?',         // `iovs` is after this
  1868  		26, 0, 0, 0, // = iovs[0].offset
  1869  		4, 0, 0, 0, // = iovs[0].length
  1870  		31, 0, 0, 0, // = iovs[1].offset
  1871  		0, 0, 0, 0, // = iovs[1].length == 0 !!
  1872  		31, 0, 0, 0, // = iovs[2].offset
  1873  		2, 0, 0, 0, // = iovs[2].length
  1874  		'?',
  1875  	}
  1876  	iovsCount := uint32(3)    // The count of iovs
  1877  	resultNread := uint32(34) // arbitrary offset
  1878  	expectedMemory := append(
  1879  		initialMemory,
  1880  		'w', 'a', 'z', 'e', // iovs[0].length bytes
  1881  		'?',      // iovs[2].offset is after this
  1882  		'r', 'o', // iovs[2].length bytes
  1883  		'?',        // resultNread is after this
  1884  		6, 0, 0, 0, // sum(iovs[...].length) == length of "wazero"
  1885  		'?',
  1886  	)
  1887  
  1888  	maskMemory(t, mod, len(expectedMemory))
  1889  
  1890  	ok := mod.Memory().Write(0, initialMemory)
  1891  	require.True(t, ok)
  1892  
  1893  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdReadName, uint64(fd), uint64(iovs), uint64(iovsCount), uint64(resultNread))
  1894  	require.Equal(t, `
  1895  ==> wasi_snapshot_preview1.fd_read(fd=4,iovs=1,iovs_len=3)
  1896  <== (nread=6,errno=ESUCCESS)
  1897  `, "\n"+log.String())
  1898  
  1899  	actual, ok := mod.Memory().Read(0, uint32(len(expectedMemory)))
  1900  	require.True(t, ok)
  1901  	require.Equal(t, expectedMemory, actual)
  1902  }
  1903  
  1904  func Test_fdRead_Errors(t *testing.T) {
  1905  	mod, fd, log, r := requireOpenFile(t, t.TempDir(), "test_path", []byte("wazero"), true)
  1906  	defer r.Close(testCtx)
  1907  
  1908  	tests := []struct {
  1909  		name                         string
  1910  		fd                           int32
  1911  		iovs, iovsCount, resultNread uint32
  1912  		memory                       []byte
  1913  		expectedErrno                wasip1.Errno
  1914  		expectedLog                  string
  1915  	}{
  1916  		{
  1917  			name:          "invalid FD",
  1918  			fd:            42,                         // arbitrary invalid fd
  1919  			memory:        []byte{'?', '?', '?', '?'}, // pass result.nread validation
  1920  			expectedErrno: wasip1.ErrnoBadf,
  1921  			expectedLog: `
  1922  ==> wasi_snapshot_preview1.fd_read(fd=42,iovs=65532,iovs_len=65532)
  1923  <== (nread=,errno=EBADF)
  1924  `,
  1925  		},
  1926  		{
  1927  			name:          "out-of-memory reading iovs[0].offset",
  1928  			fd:            fd,
  1929  			iovs:          1,
  1930  			memory:        []byte{'?'},
  1931  			expectedErrno: wasip1.ErrnoFault,
  1932  			expectedLog: `
  1933  ==> wasi_snapshot_preview1.fd_read(fd=4,iovs=65536,iovs_len=65535)
  1934  <== (nread=,errno=EFAULT)
  1935  `,
  1936  		},
  1937  		{
  1938  			name: "out-of-memory reading iovs[0].length",
  1939  			fd:   fd,
  1940  			iovs: 1, iovsCount: 1,
  1941  			memory: []byte{
  1942  				'?',        // `iovs` is after this
  1943  				9, 0, 0, 0, // = iovs[0].offset
  1944  			},
  1945  			expectedErrno: wasip1.ErrnoFault,
  1946  			expectedLog: `
  1947  ==> wasi_snapshot_preview1.fd_read(fd=4,iovs=65532,iovs_len=65532)
  1948  <== (nread=,errno=EFAULT)
  1949  `,
  1950  		},
  1951  		{
  1952  			name: "iovs[0].offset is outside memory",
  1953  			fd:   fd,
  1954  			iovs: 1, iovsCount: 1,
  1955  			memory: []byte{
  1956  				'?',          // `iovs` is after this
  1957  				0, 0, 0x1, 0, // = iovs[0].offset on the second page
  1958  				1, 0, 0, 0, // = iovs[0].length
  1959  			},
  1960  			expectedErrno: wasip1.ErrnoFault,
  1961  			expectedLog: `
  1962  ==> wasi_snapshot_preview1.fd_read(fd=4,iovs=65528,iovs_len=65528)
  1963  <== (nread=,errno=EFAULT)
  1964  `,
  1965  		},
  1966  		{
  1967  			name: "length to read exceeds memory by 1",
  1968  			fd:   fd,
  1969  			iovs: 1, iovsCount: 1,
  1970  			memory: []byte{
  1971  				'?',        // `iovs` is after this
  1972  				9, 0, 0, 0, // = iovs[0].offset
  1973  				0, 0, 0x1, 0, // = iovs[0].length on the second page
  1974  				'?',
  1975  			},
  1976  			expectedErrno: wasip1.ErrnoFault,
  1977  			expectedLog: `
  1978  ==> wasi_snapshot_preview1.fd_read(fd=4,iovs=65527,iovs_len=65527)
  1979  <== (nread=,errno=EFAULT)
  1980  `,
  1981  		},
  1982  		{
  1983  			name: "resultNread offset is outside memory",
  1984  			fd:   fd,
  1985  			iovs: 1, iovsCount: 1,
  1986  			resultNread: 10, // 1 past memory
  1987  			memory: []byte{
  1988  				'?',        // `iovs` is after this
  1989  				9, 0, 0, 0, // = iovs[0].offset
  1990  				1, 0, 0, 0, // = iovs[0].length
  1991  				'?',
  1992  			},
  1993  			expectedErrno: wasip1.ErrnoFault,
  1994  			expectedLog: `
  1995  ==> wasi_snapshot_preview1.fd_read(fd=4,iovs=65527,iovs_len=65527)
  1996  <== (nread=,errno=EFAULT)
  1997  `,
  1998  		},
  1999  	}
  2000  
  2001  	for _, tt := range tests {
  2002  		tc := tt
  2003  		t.Run(tc.name, func(t *testing.T) {
  2004  			defer log.Reset()
  2005  
  2006  			offset := uint32(wasm.MemoryPagesToBytesNum(testMemoryPageSize) - uint64(len(tc.memory)))
  2007  
  2008  			memoryWriteOK := mod.Memory().Write(offset, tc.memory)
  2009  			require.True(t, memoryWriteOK)
  2010  
  2011  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdReadName, uint64(tc.fd), uint64(tc.iovs+offset), uint64(tc.iovsCount+offset), uint64(tc.resultNread+offset))
  2012  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  2013  		})
  2014  	}
  2015  }
  2016  
  2017  var (
  2018  	direntDot = []byte{
  2019  		1, 0, 0, 0, 0, 0, 0, 0, // d_next = 1
  2020  		0, 0, 0, 0, 0, 0, 0, 0, // d_ino = 0
  2021  		1, 0, 0, 0, // d_namlen = 1 character
  2022  		3, 0, 0, 0, // d_type =  directory
  2023  		'.', // name
  2024  	}
  2025  	direntDotDot = []byte{
  2026  		2, 0, 0, 0, 0, 0, 0, 0, // d_next = 2
  2027  		0, 0, 0, 0, 0, 0, 0, 0, // d_ino = 0
  2028  		2, 0, 0, 0, // d_namlen = 2 characters
  2029  		3, 0, 0, 0, // d_type =  directory
  2030  		'.', '.', // name
  2031  	}
  2032  	dirent1 = []byte{
  2033  		3, 0, 0, 0, 0, 0, 0, 0, // d_next = 3
  2034  		0, 0, 0, 0, 0, 0, 0, 0, // d_ino = 0
  2035  		1, 0, 0, 0, // d_namlen = 1 character
  2036  		4, 0, 0, 0, // d_type = regular_file
  2037  		'-', // name
  2038  	}
  2039  	dirent2 = []byte{
  2040  		4, 0, 0, 0, 0, 0, 0, 0, // d_next = 4
  2041  		0, 0, 0, 0, 0, 0, 0, 0, // d_ino = 0
  2042  		2, 0, 0, 0, // d_namlen = 1 character
  2043  		3, 0, 0, 0, // d_type =  directory
  2044  		'a', '-', // name
  2045  	}
  2046  	dirent3 = []byte{
  2047  		5, 0, 0, 0, 0, 0, 0, 0, // d_next = 5
  2048  		0, 0, 0, 0, 0, 0, 0, 0, // d_ino = 0
  2049  		3, 0, 0, 0, // d_namlen = 3 characters
  2050  		4, 0, 0, 0, // d_type = regular_file
  2051  		'a', 'b', '-', // name
  2052  	}
  2053  
  2054  	// TODO: this entry is intended to test reading of a symbolic link entry,
  2055  	// tho it requires modifying fstest.FS to contain this file.
  2056  	// dirent4 = []byte{
  2057  	// 	6, 0, 0, 0, 0, 0, 0, 0, // d_next = 6
  2058  	// 	0, 0, 0, 0, 0, 0, 0, 0, // d_ino = 0
  2059  	// 	2, 0, 0, 0, // d_namlen = 2 characters
  2060  	// 	7, 0, 0, 0, // d_type = symbolic_link
  2061  	// 	'l', 'n', // name
  2062  	// }
  2063  
  2064  	dirents = bytes.Join([][]byte{
  2065  		direntDot,
  2066  		direntDotDot,
  2067  		dirent1,
  2068  		dirent2,
  2069  		dirent3,
  2070  		// dirent4,
  2071  	}, nil)
  2072  )
  2073  
  2074  func Test_fdReaddir(t *testing.T) {
  2075  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFS(fstest.FS))
  2076  	defer r.Close(testCtx)
  2077  
  2078  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  2079  	preopen := fsc.RootFS()
  2080  	fd := sys.FdPreopen + 1
  2081  
  2082  	tests := []struct {
  2083  		name            string
  2084  		initialDir      string
  2085  		dir             func()
  2086  		bufLen          uint32
  2087  		cookie          int64
  2088  		expectedMem     []byte
  2089  		expectedMemSize int
  2090  		expectedBufused uint32
  2091  	}{
  2092  		{
  2093  			name:            "empty dir",
  2094  			initialDir:      "emptydir",
  2095  			bufLen:          wasip1.DirentSize + 1, // size of one entry
  2096  			cookie:          0,
  2097  			expectedBufused: wasip1.DirentSize + 1, // one dot entry
  2098  			expectedMem:     direntDot,
  2099  		},
  2100  		{
  2101  			name:            "full read",
  2102  			initialDir:      "dir",
  2103  			bufLen:          4096,
  2104  			cookie:          0,
  2105  			expectedBufused: 129, // length of all entries
  2106  			expectedMem:     dirents,
  2107  		},
  2108  		{
  2109  			name:            "can't read name",
  2110  			initialDir:      "dir",
  2111  			bufLen:          wasip1.DirentSize, // length is long enough for first, but not the name.
  2112  			cookie:          0,
  2113  			expectedBufused: wasip1.DirentSize,             // == bufLen which is the size of the dirent
  2114  			expectedMem:     direntDot[:wasip1.DirentSize], // header without name
  2115  		},
  2116  		{
  2117  			name:            "read exactly first",
  2118  			initialDir:      "dir",
  2119  			bufLen:          25, // length is long enough for first + the name, but not more.
  2120  			cookie:          0,
  2121  			expectedBufused: 25, // length to read exactly first.
  2122  			expectedMem:     direntDot,
  2123  		},
  2124  		{
  2125  			name:       "read exactly second",
  2126  			initialDir: "dir",
  2127  			dir: func() {
  2128  				f, _ := fsc.LookupFile(fd)
  2129  				rdd, _ := f.DirentCache()
  2130  				_, _ = rdd.Read(0, 1)
  2131  			},
  2132  			bufLen:          27, // length is long enough for exactly second.
  2133  			cookie:          1,  // d_next of first
  2134  			expectedBufused: 27, // length to read exactly second.
  2135  			expectedMem:     direntDotDot,
  2136  		},
  2137  		{
  2138  			name:       "read second and a little more",
  2139  			initialDir: "dir",
  2140  			dir: func() {
  2141  				f, _ := fsc.LookupFile(fd)
  2142  				rdd, _ := f.DirentCache()
  2143  				_, _ = rdd.Read(0, 1)
  2144  			},
  2145  			bufLen:          30, // length is longer than the second entry, but not long enough for a header.
  2146  			cookie:          1,  // d_next of first
  2147  			expectedBufused: 30, // length to read some more, but not enough for a header, so buf was exhausted.
  2148  			expectedMem:     direntDotDot,
  2149  			expectedMemSize: len(direntDotDot), // we do not want to compare the full buffer since we don't know what the leftover 4 bytes will contain.
  2150  		},
  2151  		{
  2152  			name:       "read second and header of third",
  2153  			initialDir: "dir",
  2154  			dir: func() {
  2155  				f, _ := fsc.LookupFile(fd)
  2156  				rdd, _ := f.DirentCache()
  2157  				_, _ = rdd.Read(0, 1)
  2158  			},
  2159  			bufLen:          50, // length is longer than the second entry + enough for the header of third.
  2160  			cookie:          1,  // d_next of first
  2161  			expectedBufused: 50, // length to read exactly second and the header of third.
  2162  			expectedMem:     append(direntDotDot, dirent1[0:24]...),
  2163  		},
  2164  		{
  2165  			name:       "read second and third",
  2166  			initialDir: "dir",
  2167  			dir: func() {
  2168  				f, _ := fsc.LookupFile(fd)
  2169  				rdd, _ := f.DirentCache()
  2170  				_, _ = rdd.Read(0, 1)
  2171  			},
  2172  			bufLen:          53, // length is long enough for second and third.
  2173  			cookie:          1,  // d_next of first
  2174  			expectedBufused: 53, // length to read exactly one second and third.
  2175  			expectedMem:     append(direntDotDot, dirent1...),
  2176  		},
  2177  		{
  2178  			name:       "read exactly third",
  2179  			initialDir: "dir",
  2180  			dir: func() {
  2181  				f, _ := fsc.LookupFile(fd)
  2182  				rdd, _ := f.DirentCache()
  2183  				_, _ = rdd.Read(0, 2)
  2184  			},
  2185  			bufLen:          27, // length is long enough for exactly third.
  2186  			cookie:          2,  // d_next of second.
  2187  			expectedBufused: 27, // length to read exactly third.
  2188  			expectedMem:     dirent1,
  2189  		},
  2190  		{
  2191  			name:       "read third and beyond",
  2192  			initialDir: "dir",
  2193  			dir: func() {
  2194  				f, _ := fsc.LookupFile(fd)
  2195  				rdd, _ := f.DirentCache()
  2196  				_, _ = rdd.Read(0, 2)
  2197  			},
  2198  			bufLen:          300, // length is long enough for third and more
  2199  			cookie:          2,   // d_next of second.
  2200  			expectedBufused: 78,  // length to read the rest
  2201  			expectedMem:     append(dirent1, dirent2...),
  2202  		},
  2203  		{
  2204  			name:       "read exhausted directory",
  2205  			initialDir: "dir",
  2206  			dir: func() {
  2207  				f, _ := fsc.LookupFile(fd)
  2208  				rdd, _ := f.DirentCache()
  2209  				_, _ = rdd.Read(0, 5)
  2210  			},
  2211  			bufLen:          300, // length is long enough for third and more
  2212  			cookie:          5,   // d_next after entries.
  2213  			expectedBufused: 0,   // nothing read
  2214  		},
  2215  	}
  2216  
  2217  	for _, tt := range tests {
  2218  		tc := tt
  2219  		t.Run(tc.name, func(t *testing.T) {
  2220  			defer log.Reset()
  2221  
  2222  			fd, errno := fsc.OpenFile(preopen, tc.initialDir, experimentalsys.O_RDONLY, 0)
  2223  			require.EqualErrno(t, 0, errno)
  2224  			defer fsc.CloseFile(fd) // nolint
  2225  
  2226  			if tc.dir != nil {
  2227  				tc.dir()
  2228  			}
  2229  
  2230  			maskMemory(t, mod, int(tc.bufLen))
  2231  
  2232  			resultBufused := uint32(0) // where to write the amount used out of bufLen
  2233  			buf := uint32(8)           // where to start the dirents
  2234  			requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdReaddirName,
  2235  				uint64(fd), uint64(buf), uint64(tc.bufLen), uint64(tc.cookie), uint64(resultBufused))
  2236  
  2237  			// read back the bufused and compare memory against it
  2238  			bufused, ok := mod.Memory().ReadUint32Le(resultBufused)
  2239  			require.True(t, ok)
  2240  			require.Equal(t, tc.expectedBufused, bufused)
  2241  
  2242  			mem, ok := mod.Memory().Read(buf, bufused)
  2243  			require.True(t, ok)
  2244  
  2245  			if tc.expectedMem != nil {
  2246  				if tc.expectedMemSize == 0 {
  2247  					tc.expectedMemSize = len(tc.expectedMem)
  2248  				}
  2249  				require.Equal(t, tc.expectedMem, mem[:tc.expectedMemSize])
  2250  			}
  2251  		})
  2252  	}
  2253  }
  2254  
  2255  // This is similar to https://github.com/WebAssembly/wasi-testsuite/blob/ac32f57400cdcdd0425d3085c24fc7fc40011d1c/tests/rust/src/bin/fd_readdir.rs#L120
  2256  func Test_fdReaddir_Rewind(t *testing.T) {
  2257  	tmpDir := t.TempDir()
  2258  
  2259  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFS(os.DirFS(tmpDir)))
  2260  	defer r.Close(testCtx)
  2261  
  2262  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  2263  
  2264  	fd, errno := fsc.OpenFile(fsc.RootFS(), ".", experimentalsys.O_RDONLY, 0)
  2265  	require.EqualErrno(t, 0, errno)
  2266  
  2267  	mem := mod.Memory()
  2268  	const resultBufused, buf = 0, 8
  2269  	fdReaddir := func(cookie uint64) uint32 {
  2270  		requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdReaddirName,
  2271  			uint64(fd), buf, 256, cookie, uint64(resultBufused))
  2272  		bufused, ok := mem.ReadUint32Le(resultBufused)
  2273  		require.True(t, ok)
  2274  		return bufused
  2275  	}
  2276  
  2277  	// Read the empty directory, which should only have the dot entries.
  2278  	bufused := fdReaddir(0)
  2279  	dotDirentsLen := (wasip1.DirentSize + 1) + (wasip1.DirentSize + 2)
  2280  	require.Equal(t, dotDirentsLen, bufused)
  2281  
  2282  	// Write a new file to the directory
  2283  	fileName := "file"
  2284  	require.NoError(t, os.WriteFile(path.Join(tmpDir, fileName), nil, 0o0666))
  2285  	fileDirentLen := wasip1.DirentSize + uint32(len(fileName))
  2286  
  2287  	// Read it again, which should see the new file.
  2288  	bufused = fdReaddir(0)
  2289  	require.Equal(t, dotDirentsLen+fileDirentLen, bufused)
  2290  
  2291  	// Read it again, using the file position.
  2292  	bufused = fdReaddir(2)
  2293  	require.Equal(t, fileDirentLen, bufused)
  2294  
  2295  	require.Equal(t, `
  2296  ==> wasi_snapshot_preview1.fd_readdir(fd=4,buf=8,buf_len=256,cookie=0)
  2297  <== (bufused=51,errno=ESUCCESS)
  2298  ==> wasi_snapshot_preview1.fd_readdir(fd=4,buf=8,buf_len=256,cookie=0)
  2299  <== (bufused=79,errno=ESUCCESS)
  2300  ==> wasi_snapshot_preview1.fd_readdir(fd=4,buf=8,buf_len=256,cookie=2)
  2301  <== (bufused=28,errno=ESUCCESS)
  2302  `, "\n"+log.String())
  2303  }
  2304  
  2305  func Test_fdReaddir_Errors(t *testing.T) {
  2306  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFS(fstest.FS))
  2307  	defer r.Close(testCtx)
  2308  	memLen := mod.Memory().Size()
  2309  
  2310  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  2311  	preopen := fsc.RootFS()
  2312  
  2313  	fileFD, errno := fsc.OpenFile(preopen, "animals.txt", experimentalsys.O_RDONLY, 0)
  2314  	require.EqualErrno(t, 0, errno)
  2315  
  2316  	// Directories are stateful, so we open them during the test.
  2317  	dirFD := fileFD + 1
  2318  
  2319  	tests := []struct {
  2320  		name                       string
  2321  		fd                         int32
  2322  		buf, bufLen, resultBufused uint32
  2323  		cookie                     int64
  2324  		expectedErrno              wasip1.Errno
  2325  		expectedLog                string
  2326  	}{
  2327  		{
  2328  			name:          "out-of-memory reading buf",
  2329  			fd:            dirFD,
  2330  			buf:           memLen,
  2331  			bufLen:        1000,
  2332  			expectedErrno: wasip1.ErrnoFault,
  2333  			expectedLog: `
  2334  ==> wasi_snapshot_preview1.fd_readdir(fd=5,buf=65536,buf_len=1000,cookie=0)
  2335  <== (bufused=,errno=EFAULT)
  2336  `,
  2337  		},
  2338  		{
  2339  			name: "invalid FD",
  2340  			fd:   42,                           // arbitrary invalid fd
  2341  			buf:  0, bufLen: wasip1.DirentSize, // enough to read the dirent
  2342  			resultBufused: 1000, // arbitrary
  2343  			expectedErrno: wasip1.ErrnoBadf,
  2344  			expectedLog: `
  2345  ==> wasi_snapshot_preview1.fd_readdir(fd=42,buf=0,buf_len=24,cookie=0)
  2346  <== (bufused=,errno=EBADF)
  2347  `,
  2348  		},
  2349  		{
  2350  			name: "not a dir",
  2351  			fd:   fileFD,
  2352  			buf:  0, bufLen: wasip1.DirentSize, // enough to read the dirent
  2353  			resultBufused: 1000, // arbitrary
  2354  			expectedErrno: wasip1.ErrnoBadf,
  2355  			expectedLog: `
  2356  ==> wasi_snapshot_preview1.fd_readdir(fd=4,buf=0,buf_len=24,cookie=0)
  2357  <== (bufused=,errno=EBADF)
  2358  `,
  2359  		},
  2360  		{
  2361  			name:          "out-of-memory reading bufLen",
  2362  			fd:            dirFD,
  2363  			buf:           memLen - 1,
  2364  			bufLen:        1000,
  2365  			expectedErrno: wasip1.ErrnoFault,
  2366  			expectedLog: `
  2367  ==> wasi_snapshot_preview1.fd_readdir(fd=5,buf=65535,buf_len=1000,cookie=0)
  2368  <== (bufused=,errno=EFAULT)
  2369  `,
  2370  		},
  2371  		{
  2372  			name: "bufLen must be enough to write a struct",
  2373  			fd:   dirFD,
  2374  			buf:  0, bufLen: 1,
  2375  			resultBufused: 1000,
  2376  			expectedErrno: wasip1.ErrnoInval, // Arbitrary error choice.
  2377  			expectedLog: `
  2378  ==> wasi_snapshot_preview1.fd_readdir(fd=5,buf=0,buf_len=1,cookie=0)
  2379  <== (bufused=,errno=EINVAL)
  2380  `,
  2381  		},
  2382  		{
  2383  			name: "cookie invalid when no prior state",
  2384  			fd:   dirFD,
  2385  			buf:  0, bufLen: 1000,
  2386  			cookie:        1,
  2387  			resultBufused: 2000,
  2388  			expectedErrno: wasip1.ErrnoNoent,
  2389  			expectedLog: `
  2390  ==> wasi_snapshot_preview1.fd_readdir(fd=5,buf=0,buf_len=1000,cookie=1)
  2391  <== (bufused=,errno=ENOENT)
  2392  `,
  2393  		},
  2394  		{
  2395  			// cookie should be treated opaquely. When negative, it is a
  2396  			// position not yet read,
  2397  			name: "negative cookie invalid",
  2398  			fd:   dirFD,
  2399  			buf:  0, bufLen: 1000,
  2400  			cookie:        -1,
  2401  			resultBufused: 2000,
  2402  			expectedErrno: wasip1.ErrnoNoent,
  2403  			expectedLog: `
  2404  ==> wasi_snapshot_preview1.fd_readdir(fd=5,buf=0,buf_len=1000,cookie=-1)
  2405  <== (bufused=,errno=ENOENT)
  2406  `,
  2407  		},
  2408  	}
  2409  
  2410  	for _, tt := range tests {
  2411  		tc := tt
  2412  		t.Run(tc.name, func(t *testing.T) {
  2413  			defer log.Reset()
  2414  
  2415  			// Reset the directory so that tests don't taint each other.
  2416  			if tc.fd == dirFD {
  2417  				dirFD, errno = fsc.OpenFile(preopen, "dir", experimentalsys.O_RDONLY, 0)
  2418  				require.EqualErrno(t, 0, errno)
  2419  				defer fsc.CloseFile(dirFD) // nolint
  2420  			}
  2421  
  2422  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdReaddirName,
  2423  				uint64(tc.fd), uint64(tc.buf), uint64(tc.bufLen), uint64(tc.cookie), uint64(tc.resultBufused))
  2424  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  2425  		})
  2426  	}
  2427  }
  2428  
  2429  func Test_fdRenumber(t *testing.T) {
  2430  	const fileFD, dirFD = 4, 5
  2431  
  2432  	tests := []struct {
  2433  		name          string
  2434  		from, to      int32
  2435  		expectedErrno wasip1.Errno
  2436  		expectedLog   string
  2437  	}{
  2438  		{
  2439  			name:          "from=preopen",
  2440  			from:          sys.FdPreopen,
  2441  			to:            dirFD,
  2442  			expectedErrno: wasip1.ErrnoNotsup,
  2443  			expectedLog: `
  2444  ==> wasi_snapshot_preview1.fd_renumber(fd=3,to=5)
  2445  <== errno=ENOTSUP
  2446  `,
  2447  		},
  2448  		{
  2449  			name:          "from=badf",
  2450  			from:          -1,
  2451  			to:            sys.FdPreopen,
  2452  			expectedErrno: wasip1.ErrnoBadf,
  2453  			expectedLog: `
  2454  ==> wasi_snapshot_preview1.fd_renumber(fd=-1,to=3)
  2455  <== errno=EBADF
  2456  `,
  2457  		},
  2458  		{
  2459  			name:          "to=badf",
  2460  			from:          sys.FdPreopen,
  2461  			to:            -1,
  2462  			expectedErrno: wasip1.ErrnoBadf,
  2463  			expectedLog: `
  2464  ==> wasi_snapshot_preview1.fd_renumber(fd=3,to=-1)
  2465  <== errno=EBADF
  2466  `,
  2467  		},
  2468  		{
  2469  			name:          "to=preopen",
  2470  			from:          dirFD,
  2471  			to:            sys.FdPreopen,
  2472  			expectedErrno: wasip1.ErrnoNotsup,
  2473  			expectedLog: `
  2474  ==> wasi_snapshot_preview1.fd_renumber(fd=5,to=3)
  2475  <== errno=ENOTSUP
  2476  `,
  2477  		},
  2478  		{
  2479  			name:          "file to dir",
  2480  			from:          fileFD,
  2481  			to:            dirFD,
  2482  			expectedErrno: wasip1.ErrnoSuccess,
  2483  			expectedLog: `
  2484  ==> wasi_snapshot_preview1.fd_renumber(fd=4,to=5)
  2485  <== errno=ESUCCESS
  2486  `,
  2487  		},
  2488  		{
  2489  			name:          "dir to file",
  2490  			from:          dirFD,
  2491  			to:            fileFD,
  2492  			expectedErrno: wasip1.ErrnoSuccess,
  2493  			expectedLog: `
  2494  ==> wasi_snapshot_preview1.fd_renumber(fd=5,to=4)
  2495  <== errno=ESUCCESS
  2496  `,
  2497  		},
  2498  		{
  2499  			name:          "dir to any",
  2500  			from:          dirFD,
  2501  			to:            12345,
  2502  			expectedErrno: wasip1.ErrnoSuccess,
  2503  			expectedLog: `
  2504  ==> wasi_snapshot_preview1.fd_renumber(fd=5,to=12345)
  2505  <== errno=ESUCCESS
  2506  `,
  2507  		},
  2508  		{
  2509  			name:          "file to any",
  2510  			from:          fileFD,
  2511  			to:            54,
  2512  			expectedErrno: wasip1.ErrnoSuccess,
  2513  			expectedLog: `
  2514  ==> wasi_snapshot_preview1.fd_renumber(fd=4,to=54)
  2515  <== errno=ESUCCESS
  2516  `,
  2517  		},
  2518  	}
  2519  
  2520  	for _, tt := range tests {
  2521  		tc := tt
  2522  		t.Run(tc.name, func(t *testing.T) {
  2523  			mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFS(fstest.FS))
  2524  			defer r.Close(testCtx)
  2525  
  2526  			fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  2527  			preopen := fsc.RootFS()
  2528  
  2529  			// Sanity check of the file descriptor assignment.
  2530  			fileFDAssigned, errno := fsc.OpenFile(preopen, "animals.txt", experimentalsys.O_RDONLY, 0)
  2531  			require.EqualErrno(t, 0, errno)
  2532  			require.Equal(t, int32(fileFD), fileFDAssigned)
  2533  
  2534  			dirFDAssigned, errno := fsc.OpenFile(preopen, "dir", experimentalsys.O_RDONLY, 0)
  2535  			require.EqualErrno(t, 0, errno)
  2536  			require.Equal(t, int32(dirFD), dirFDAssigned)
  2537  
  2538  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdRenumberName, uint64(tc.from), uint64(tc.to))
  2539  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  2540  		})
  2541  	}
  2542  }
  2543  
  2544  func Test_fdSeek(t *testing.T) {
  2545  	mod, fd, log, r := requireOpenFile(t, t.TempDir(), "test_path", []byte("wazero"), true)
  2546  	defer r.Close(testCtx)
  2547  
  2548  	resultNewoffset := uint32(1) // arbitrary offset in api.Memory for the new offset value
  2549  
  2550  	tests := []struct {
  2551  		name           string
  2552  		offset         int64
  2553  		whence         int
  2554  		expectedOffset int64
  2555  		expectedMemory []byte
  2556  		expectedLog    string
  2557  	}{
  2558  		{
  2559  			name:           "SeekStart",
  2560  			offset:         4, // arbitrary offset
  2561  			whence:         io.SeekStart,
  2562  			expectedOffset: 4, // = offset
  2563  			expectedMemory: []byte{
  2564  				'?',                    // resultNewoffset is after this
  2565  				4, 0, 0, 0, 0, 0, 0, 0, // = expectedOffset
  2566  				'?',
  2567  			},
  2568  			expectedLog: `
  2569  ==> wasi_snapshot_preview1.fd_seek(fd=4,offset=4,whence=0)
  2570  <== (newoffset=4,errno=ESUCCESS)
  2571  `,
  2572  		},
  2573  		{
  2574  			name:           "SeekCurrent",
  2575  			offset:         1, // arbitrary offset
  2576  			whence:         io.SeekCurrent,
  2577  			expectedOffset: 2, // = 1 (the initial offset of the test file) + 1 (offset)
  2578  			expectedMemory: []byte{
  2579  				'?',                    // resultNewoffset is after this
  2580  				2, 0, 0, 0, 0, 0, 0, 0, // = expectedOffset
  2581  				'?',
  2582  			},
  2583  			expectedLog: `
  2584  ==> wasi_snapshot_preview1.fd_seek(fd=4,offset=1,whence=1)
  2585  <== (newoffset=2,errno=ESUCCESS)
  2586  `,
  2587  		},
  2588  		{
  2589  			name:           "SeekEnd",
  2590  			offset:         -1, // arbitrary offset, note that offset can be negative
  2591  			whence:         io.SeekEnd,
  2592  			expectedOffset: 5, // = 6 (the size of the test file with content "wazero") + -1 (offset)
  2593  			expectedMemory: []byte{
  2594  				'?',                    // resultNewoffset is after this
  2595  				5, 0, 0, 0, 0, 0, 0, 0, // = expectedOffset
  2596  				'?',
  2597  			},
  2598  			expectedLog: `
  2599  ==> wasi_snapshot_preview1.fd_seek(fd=4,offset=-1,whence=2)
  2600  <== (newoffset=5,errno=ESUCCESS)
  2601  `,
  2602  		},
  2603  	}
  2604  
  2605  	for _, tt := range tests {
  2606  		tc := tt
  2607  		t.Run(tc.name, func(t *testing.T) {
  2608  			defer log.Reset()
  2609  
  2610  			maskMemory(t, mod, len(tc.expectedMemory))
  2611  
  2612  			// Since we initialized this file, we know it is a seeker (because it is a MapFile)
  2613  			fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  2614  			f, ok := fsc.LookupFile(fd)
  2615  			require.True(t, ok)
  2616  
  2617  			// set the initial offset of the file to 1
  2618  			offset, errno := f.File.Seek(1, io.SeekStart)
  2619  			require.EqualErrno(t, 0, errno)
  2620  			require.Equal(t, int64(1), offset)
  2621  
  2622  			requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdSeekName, uint64(fd), uint64(tc.offset), uint64(tc.whence), uint64(resultNewoffset))
  2623  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  2624  
  2625  			actual, ok := mod.Memory().Read(0, uint32(len(tc.expectedMemory)))
  2626  			require.True(t, ok)
  2627  			require.Equal(t, tc.expectedMemory, actual)
  2628  
  2629  			offset, errno = f.File.Seek(0, io.SeekCurrent)
  2630  			require.EqualErrno(t, 0, errno)
  2631  			require.Equal(t, tc.expectedOffset, offset) // test that the offset of file is actually updated.
  2632  		})
  2633  	}
  2634  }
  2635  
  2636  func Test_fdSeek_Errors(t *testing.T) {
  2637  	mod, fileFD, log, r := requireOpenFile(t, t.TempDir(), "test_path", []byte("wazero"), false)
  2638  	defer r.Close(testCtx)
  2639  
  2640  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  2641  	require.Zero(t, fsc.RootFS().Mkdir("dir", 0o0777))
  2642  	dirFD := requireOpenFD(t, mod, "dir")
  2643  
  2644  	memorySize := mod.Memory().Size()
  2645  
  2646  	tests := []struct {
  2647  		name                    string
  2648  		fd                      int32
  2649  		offset                  uint64
  2650  		whence, resultNewoffset uint32
  2651  		expectedErrno           wasip1.Errno
  2652  		expectedLog             string
  2653  	}{
  2654  		{
  2655  			name:          "invalid FD",
  2656  			fd:            42, // arbitrary invalid fd
  2657  			expectedErrno: wasip1.ErrnoBadf,
  2658  			expectedLog: `
  2659  ==> wasi_snapshot_preview1.fd_seek(fd=42,offset=0,whence=0)
  2660  <== (newoffset=,errno=EBADF)
  2661  `,
  2662  		},
  2663  		{
  2664  			name:          "invalid whence",
  2665  			fd:            fileFD,
  2666  			whence:        3, // invalid whence, the largest whence io.SeekEnd(2) + 1
  2667  			expectedErrno: wasip1.ErrnoInval,
  2668  			expectedLog: `
  2669  ==> wasi_snapshot_preview1.fd_seek(fd=4,offset=0,whence=3)
  2670  <== (newoffset=,errno=EINVAL)
  2671  `,
  2672  		},
  2673  		{
  2674  			name:          "dir not file",
  2675  			fd:            dirFD,
  2676  			expectedErrno: wasip1.ErrnoIsdir,
  2677  			expectedLog: `
  2678  ==> wasi_snapshot_preview1.fd_seek(fd=5,offset=0,whence=0)
  2679  <== (newoffset=,errno=EISDIR)
  2680  `,
  2681  		},
  2682  		{
  2683  			name:            "out-of-memory writing resultNewoffset",
  2684  			fd:              fileFD,
  2685  			resultNewoffset: memorySize,
  2686  			expectedErrno:   wasip1.ErrnoFault,
  2687  			expectedLog: `
  2688  ==> wasi_snapshot_preview1.fd_seek(fd=4,offset=0,whence=0)
  2689  <== (newoffset=,errno=EFAULT)
  2690  `,
  2691  		},
  2692  	}
  2693  
  2694  	for _, tt := range tests {
  2695  		tc := tt
  2696  		t.Run(tc.name, func(t *testing.T) {
  2697  			defer log.Reset()
  2698  
  2699  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdSeekName, uint64(tc.fd), tc.offset, uint64(tc.whence), uint64(tc.resultNewoffset))
  2700  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  2701  		})
  2702  	}
  2703  }
  2704  
  2705  // Test_fdSync only tests that the call succeeds; it's hard to test its effectiveness.
  2706  func Test_fdSync(t *testing.T) {
  2707  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  2708  	pathName := "test_path"
  2709  	mod, fd, log, r := requireOpenFile(t, tmpDir, pathName, []byte{}, false)
  2710  	defer r.Close(testCtx)
  2711  
  2712  	tests := []struct {
  2713  		name          string
  2714  		fd            int32
  2715  		expectedErrno wasip1.Errno
  2716  		expectedLog   string
  2717  	}{
  2718  		{
  2719  			name:          "invalid FD",
  2720  			fd:            42, // arbitrary invalid fd
  2721  			expectedErrno: wasip1.ErrnoBadf,
  2722  			expectedLog: `
  2723  ==> wasi_snapshot_preview1.fd_sync(fd=42)
  2724  <== errno=EBADF
  2725  `,
  2726  		},
  2727  		{
  2728  			name:          "valid FD",
  2729  			fd:            fd,
  2730  			expectedErrno: wasip1.ErrnoSuccess,
  2731  			expectedLog: `
  2732  ==> wasi_snapshot_preview1.fd_sync(fd=4)
  2733  <== errno=ESUCCESS
  2734  `,
  2735  		},
  2736  	}
  2737  
  2738  	for _, tt := range tests {
  2739  		tc := tt
  2740  		t.Run(tc.name, func(t *testing.T) {
  2741  			defer log.Reset()
  2742  
  2743  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdSyncName, uint64(tc.fd))
  2744  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  2745  		})
  2746  	}
  2747  }
  2748  
  2749  func Test_fdTell(t *testing.T) {
  2750  	mod, fd, log, r := requireOpenFile(t, t.TempDir(), "test_path", []byte("wazero"), true)
  2751  	defer r.Close(testCtx)
  2752  	defer log.Reset()
  2753  
  2754  	resultNewoffset := uint32(1) // arbitrary offset in api.Memory for the new offset value
  2755  
  2756  	expectedOffset := int64(1) // = offset
  2757  	expectedMemory := []byte{
  2758  		'?',                    // resultNewoffset is after this
  2759  		1, 0, 0, 0, 0, 0, 0, 0, // = expectedOffset
  2760  		'?',
  2761  	}
  2762  	expectedLog := `
  2763  ==> wasi_snapshot_preview1.fd_tell(fd=4,result.offset=1)
  2764  <== errno=ESUCCESS
  2765  `
  2766  
  2767  	maskMemory(t, mod, len(expectedMemory))
  2768  
  2769  	// Since we initialized this file, we know it is a seeker (because it is a MapFile)
  2770  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  2771  	f, ok := fsc.LookupFile(fd)
  2772  	require.True(t, ok)
  2773  
  2774  	// set the initial offset of the file to 1
  2775  	offset, errno := f.File.Seek(1, io.SeekStart)
  2776  	require.EqualErrno(t, 0, errno)
  2777  	require.Equal(t, int64(1), offset)
  2778  
  2779  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdTellName, uint64(fd), uint64(resultNewoffset))
  2780  	require.Equal(t, expectedLog, "\n"+log.String())
  2781  
  2782  	actual, ok := mod.Memory().Read(0, uint32(len(expectedMemory)))
  2783  	require.True(t, ok)
  2784  	require.Equal(t, expectedMemory, actual)
  2785  
  2786  	offset, errno = f.File.Seek(0, io.SeekCurrent)
  2787  	require.EqualErrno(t, 0, errno)
  2788  	require.Equal(t, expectedOffset, offset) // test that the offset of file is actually updated.
  2789  }
  2790  
  2791  func Test_fdTell_Errors(t *testing.T) {
  2792  	mod, fd, log, r := requireOpenFile(t, t.TempDir(), "test_path", []byte("wazero"), true)
  2793  	defer r.Close(testCtx)
  2794  
  2795  	memorySize := mod.Memory().Size()
  2796  
  2797  	tests := []struct {
  2798  		name            string
  2799  		fd              int32
  2800  		resultNewoffset uint32
  2801  		expectedErrno   wasip1.Errno
  2802  		expectedLog     string
  2803  	}{
  2804  		{
  2805  			name:          "invalid FD",
  2806  			fd:            42, // arbitrary invalid fd
  2807  			expectedErrno: wasip1.ErrnoBadf,
  2808  			expectedLog: `
  2809  ==> wasi_snapshot_preview1.fd_tell(fd=42,result.offset=0)
  2810  <== errno=EBADF
  2811  `,
  2812  		},
  2813  		{
  2814  			name:            "out-of-memory writing resultNewoffset",
  2815  			fd:              fd,
  2816  			resultNewoffset: memorySize,
  2817  			expectedErrno:   wasip1.ErrnoFault,
  2818  			expectedLog: `
  2819  ==> wasi_snapshot_preview1.fd_tell(fd=4,result.offset=65536)
  2820  <== errno=EFAULT
  2821  `,
  2822  		},
  2823  	}
  2824  
  2825  	for _, tt := range tests {
  2826  		tc := tt
  2827  		t.Run(tc.name, func(t *testing.T) {
  2828  			defer log.Reset()
  2829  
  2830  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdTellName, uint64(tc.fd), uint64(tc.resultNewoffset))
  2831  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  2832  		})
  2833  	}
  2834  }
  2835  
  2836  func Test_fdWrite(t *testing.T) {
  2837  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  2838  	pathName := "test_path"
  2839  	mod, fd, log, r := requireOpenFile(t, tmpDir, pathName, []byte{}, false)
  2840  	defer r.Close(testCtx)
  2841  
  2842  	iovs := uint32(1) // arbitrary offset
  2843  	initialMemory := []byte{
  2844  		'?',         // `iovs` is after this
  2845  		18, 0, 0, 0, // = iovs[0].offset
  2846  		4, 0, 0, 0, // = iovs[0].length
  2847  		23, 0, 0, 0, // = iovs[1].offset
  2848  		2, 0, 0, 0, // = iovs[1].length
  2849  		'?',                // iovs[0].offset is after this
  2850  		'w', 'a', 'z', 'e', // iovs[0].length bytes
  2851  		'?',      // iovs[1].offset is after this
  2852  		'r', 'o', // iovs[1].length bytes
  2853  		'?',
  2854  	}
  2855  	iovsCount := uint32(2)       // The count of iovs
  2856  	resultNwritten := uint32(26) // arbitrary offset
  2857  	expectedMemory := append(
  2858  		initialMemory,
  2859  		6, 0, 0, 0, // sum(iovs[...].length) == length of "wazero"
  2860  		'?',
  2861  	)
  2862  
  2863  	maskMemory(t, mod, len(expectedMemory))
  2864  	ok := mod.Memory().Write(0, initialMemory)
  2865  	require.True(t, ok)
  2866  
  2867  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdWriteName, uint64(fd), uint64(iovs), uint64(iovsCount), uint64(resultNwritten))
  2868  	require.Equal(t, `
  2869  ==> wasi_snapshot_preview1.fd_write(fd=4,iovs=1,iovs_len=2)
  2870  <== (nwritten=6,errno=ESUCCESS)
  2871  `, "\n"+log.String())
  2872  
  2873  	actual, ok := mod.Memory().Read(0, uint32(len(expectedMemory)))
  2874  	require.True(t, ok)
  2875  	require.Equal(t, expectedMemory, actual)
  2876  
  2877  	// Since we initialized this file, we know we can read it by path
  2878  	buf, err := os.ReadFile(joinPath(tmpDir, pathName))
  2879  	require.NoError(t, err)
  2880  
  2881  	require.Equal(t, []byte("wazero"), buf) // verify the file was actually written
  2882  }
  2883  
  2884  func Test_fdWrite_Errors(t *testing.T) {
  2885  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  2886  	pathName := "test_path"
  2887  	mod, fd, log, r := requireOpenFile(t, tmpDir, pathName, []byte{1, 2, 3, 4}, false)
  2888  	defer r.Close(testCtx)
  2889  
  2890  	// Setup valid test memory
  2891  	iovsCount := uint32(1)
  2892  	memSize := mod.Memory().Size()
  2893  
  2894  	tests := []struct {
  2895  		name                 string
  2896  		fd                   int32
  2897  		iovs, resultNwritten uint32
  2898  		expectedErrno        wasip1.Errno
  2899  		expectedLog          string
  2900  	}{
  2901  		{
  2902  			name:          "invalid FD",
  2903  			fd:            42, // arbitrary invalid fd
  2904  			expectedErrno: wasip1.ErrnoBadf,
  2905  			expectedLog: `
  2906  ==> wasi_snapshot_preview1.fd_write(fd=42,iovs=0,iovs_len=1)
  2907  <== (nwritten=,errno=EBADF)
  2908  `,
  2909  		},
  2910  		{
  2911  			name:          "not writable FD",
  2912  			fd:            sys.FdStdin,
  2913  			expectedErrno: wasip1.ErrnoBadf,
  2914  			expectedLog:   "\n", // stdin is not sampled
  2915  		},
  2916  		{
  2917  			name:          "out-of-memory reading iovs[0].offset",
  2918  			fd:            fd,
  2919  			iovs:          memSize - 2,
  2920  			expectedErrno: wasip1.ErrnoFault,
  2921  			expectedLog: `
  2922  ==> wasi_snapshot_preview1.fd_write(fd=4,iovs=65534,iovs_len=1)
  2923  <== (nwritten=,errno=EFAULT)
  2924  `,
  2925  		},
  2926  		{
  2927  			name:          "out-of-memory reading iovs[0].length",
  2928  			fd:            fd,
  2929  			iovs:          memSize - 4, // iovs[0].offset was 4 bytes and iovs[0].length next, but not enough mod.Memory()!
  2930  			expectedErrno: wasip1.ErrnoFault,
  2931  			expectedLog: `
  2932  ==> wasi_snapshot_preview1.fd_write(fd=4,iovs=65532,iovs_len=1)
  2933  <== (nwritten=,errno=EFAULT)
  2934  `,
  2935  		},
  2936  		{
  2937  			name:          "iovs[0].offset is outside memory",
  2938  			fd:            fd,
  2939  			iovs:          memSize - 5, // iovs[0].offset (where to read "hi") is outside memory.
  2940  			expectedErrno: wasip1.ErrnoFault,
  2941  			expectedLog: `
  2942  ==> wasi_snapshot_preview1.fd_write(fd=4,iovs=65531,iovs_len=1)
  2943  <== (nwritten=,errno=EFAULT)
  2944  `,
  2945  		},
  2946  		{
  2947  			name:          "length to read exceeds memory by 1",
  2948  			fd:            fd,
  2949  			iovs:          memSize - 7, // iovs[0].offset (where to read "hi") is in memory, but truncated.
  2950  			expectedErrno: wasip1.ErrnoFault,
  2951  			expectedLog: `
  2952  ==> wasi_snapshot_preview1.fd_write(fd=4,iovs=65529,iovs_len=1)
  2953  <== (nwritten=,errno=EFAULT)
  2954  `,
  2955  		},
  2956  		{
  2957  			name:           "resultNwritten offset is outside memory",
  2958  			fd:             fd,
  2959  			resultNwritten: memSize, // read was ok, but there wasn't enough memory to write the result.
  2960  			expectedErrno:  wasip1.ErrnoFault,
  2961  			expectedLog: `
  2962  ==> wasi_snapshot_preview1.fd_write(fd=4,iovs=0,iovs_len=1)
  2963  <== (nwritten=,errno=EFAULT)
  2964  `,
  2965  		},
  2966  	}
  2967  
  2968  	for _, tt := range tests {
  2969  		tc := tt
  2970  		t.Run(tc.name, func(t *testing.T) {
  2971  			defer log.Reset()
  2972  
  2973  			mod.Memory().Write(tc.iovs, append(
  2974  				u64.LeBytes(uint64(tc.iovs+8)), // = iovs[0].offset (where the data "hi" begins)
  2975  				// = iovs[0].length (how many bytes are in "hi")
  2976  				2, 0, 0, 0,
  2977  				'h', 'i', // iovs[0].length bytes
  2978  			))
  2979  
  2980  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.FdWriteName, uint64(tc.fd), uint64(tc.iovs), uint64(iovsCount),
  2981  				uint64(tc.resultNwritten))
  2982  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  2983  		})
  2984  	}
  2985  }
  2986  
  2987  func Test_pathCreateDirectory(t *testing.T) {
  2988  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  2989  	fsConfig := wazero.NewFSConfig().WithDirMount(tmpDir, "/")
  2990  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  2991  	defer r.Close(testCtx)
  2992  
  2993  	// set up the initial memory to include the path name starting at an offset.
  2994  	pathName := "wazero"
  2995  	realPath := joinPath(tmpDir, pathName)
  2996  	ok := mod.Memory().Write(0, append([]byte{'?'}, pathName...))
  2997  	require.True(t, ok)
  2998  
  2999  	fd := sys.FdPreopen
  3000  	name := 1
  3001  	nameLen := len(pathName)
  3002  
  3003  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.PathCreateDirectoryName, uint64(fd), uint64(name), uint64(nameLen))
  3004  	require.Equal(t, `
  3005  ==> wasi_snapshot_preview1.path_create_directory(fd=3,path=wazero)
  3006  <== errno=ESUCCESS
  3007  `, "\n"+log.String())
  3008  
  3009  	// ensure the directory was created
  3010  	stat, err := os.Stat(realPath)
  3011  	require.NoError(t, err)
  3012  	require.True(t, stat.IsDir())
  3013  	require.Equal(t, pathName, stat.Name())
  3014  }
  3015  
  3016  func Test_pathCreateDirectory_Errors(t *testing.T) {
  3017  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  3018  	fsConfig := wazero.NewFSConfig().WithDirMount(tmpDir, "/")
  3019  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  3020  	defer r.Close(testCtx)
  3021  
  3022  	file := "file"
  3023  	err := os.WriteFile(joinPath(tmpDir, file), []byte{}, 0o700)
  3024  	require.NoError(t, err)
  3025  	fileFD := requireOpenFD(t, mod, file)
  3026  
  3027  	dir := "dir"
  3028  	err = os.Mkdir(joinPath(tmpDir, dir), 0o700)
  3029  	require.NoError(t, err)
  3030  
  3031  	tests := []struct {
  3032  		name, pathName string
  3033  		fd             int32
  3034  		path, pathLen  uint32
  3035  		expectedErrno  wasip1.Errno
  3036  		expectedLog    string
  3037  	}{
  3038  		{
  3039  			name:          "unopened FD",
  3040  			fd:            42, // arbitrary invalid fd
  3041  			expectedErrno: wasip1.ErrnoBadf,
  3042  			expectedLog: `
  3043  ==> wasi_snapshot_preview1.path_create_directory(fd=42,path=)
  3044  <== errno=EBADF
  3045  `,
  3046  		},
  3047  		{
  3048  			name:          "Fd not a directory",
  3049  			fd:            fileFD,
  3050  			pathName:      file,
  3051  			path:          0,
  3052  			pathLen:       uint32(len(file)),
  3053  			expectedErrno: wasip1.ErrnoNotdir,
  3054  			expectedLog: `
  3055  ==> wasi_snapshot_preview1.path_create_directory(fd=4,path=file)
  3056  <== errno=ENOTDIR
  3057  `,
  3058  		},
  3059  		{
  3060  			name:          "out-of-memory reading path",
  3061  			fd:            sys.FdPreopen,
  3062  			path:          mod.Memory().Size(),
  3063  			pathLen:       1,
  3064  			expectedErrno: wasip1.ErrnoFault,
  3065  			expectedLog: `
  3066  ==> wasi_snapshot_preview1.path_create_directory(fd=3,path=OOM(65536,1))
  3067  <== errno=EFAULT
  3068  `,
  3069  		},
  3070  		{
  3071  			name:          "out-of-memory reading pathLen",
  3072  			fd:            sys.FdPreopen,
  3073  			path:          0,
  3074  			pathLen:       mod.Memory().Size() + 1, // path is in the valid memory range, but pathLen is OOM for path
  3075  			expectedErrno: wasip1.ErrnoFault,
  3076  			expectedLog: `
  3077  ==> wasi_snapshot_preview1.path_create_directory(fd=3,path=OOM(0,65537))
  3078  <== errno=EFAULT
  3079  `,
  3080  		},
  3081  		{
  3082  			name:          "file exists",
  3083  			fd:            sys.FdPreopen,
  3084  			pathName:      file,
  3085  			path:          0,
  3086  			pathLen:       uint32(len(file)),
  3087  			expectedErrno: wasip1.ErrnoExist,
  3088  			expectedLog: `
  3089  ==> wasi_snapshot_preview1.path_create_directory(fd=3,path=file)
  3090  <== errno=EEXIST
  3091  `,
  3092  		},
  3093  		{
  3094  			name:          "dir exists",
  3095  			fd:            sys.FdPreopen,
  3096  			pathName:      dir,
  3097  			path:          0,
  3098  			pathLen:       uint32(len(dir)),
  3099  			expectedErrno: wasip1.ErrnoExist,
  3100  			expectedLog: `
  3101  ==> wasi_snapshot_preview1.path_create_directory(fd=3,path=dir)
  3102  <== errno=EEXIST
  3103  `,
  3104  		},
  3105  	}
  3106  
  3107  	for _, tt := range tests {
  3108  		tc := tt
  3109  		t.Run(tc.name, func(t *testing.T) {
  3110  			defer log.Reset()
  3111  
  3112  			mod.Memory().Write(tc.path, []byte(tc.pathName))
  3113  
  3114  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.PathCreateDirectoryName, uint64(tc.fd), uint64(tc.path), uint64(tc.pathLen))
  3115  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  3116  		})
  3117  	}
  3118  }
  3119  
  3120  func Test_pathFilestatGet(t *testing.T) {
  3121  	file, dir, fileInDir := "animals.txt", "sub", "sub/test.txt"
  3122  
  3123  	initialMemoryFile := append([]byte{'?'}, file...)
  3124  	initialMemoryDir := append([]byte{'?'}, dir...)
  3125  	initialMemoryFileInDir := append([]byte{'?'}, fileInDir...)
  3126  	initialMemoryNotExists := []byte{'?', '?'}
  3127  
  3128  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFS(fstest.FS))
  3129  	defer r.Close(testCtx)
  3130  	memorySize := mod.Memory().Size()
  3131  
  3132  	fileFD := requireOpenFD(t, mod, file)
  3133  
  3134  	tests := []struct {
  3135  		name                    string
  3136  		fd                      int32
  3137  		pathLen, resultFilestat uint32
  3138  		flags                   uint16
  3139  		memory, expectedMemory  []byte
  3140  		expectedErrno           wasip1.Errno
  3141  		expectedLog             string
  3142  	}{
  3143  		{
  3144  			name:           "file under root",
  3145  			fd:             sys.FdPreopen,
  3146  			memory:         initialMemoryFile,
  3147  			pathLen:        uint32(len(file)),
  3148  			resultFilestat: uint32(len(file)) + 1,
  3149  			expectedMemory: append(
  3150  				initialMemoryFile,
  3151  				0, 0, 0, 0, 0, 0, 0, 0, // dev
  3152  				0, 0, 0, 0, 0, 0, 0, 0, // ino
  3153  				4, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
  3154  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
  3155  				30, 0, 0, 0, 0, 0, 0, 0, // size
  3156  				0x0, 0x82, 0x13, 0x80, 0x6b, 0x16, 0x24, 0x17, // atim
  3157  				0x0, 0x82, 0x13, 0x80, 0x6b, 0x16, 0x24, 0x17, // mtim
  3158  				0x0, 0x82, 0x13, 0x80, 0x6b, 0x16, 0x24, 0x17, // ctim
  3159  			),
  3160  			expectedLog: `
  3161  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=,path=animals.txt)
  3162  <== (filestat={filetype=REGULAR_FILE,size=30,mtim=1667482413000000000},errno=ESUCCESS)
  3163  `,
  3164  		},
  3165  		{
  3166  			name:           "file under dir",
  3167  			fd:             sys.FdPreopen, // root
  3168  			memory:         initialMemoryFileInDir,
  3169  			pathLen:        uint32(len(fileInDir)),
  3170  			resultFilestat: uint32(len(fileInDir)) + 1,
  3171  			expectedMemory: append(
  3172  				initialMemoryFileInDir,
  3173  				0, 0, 0, 0, 0, 0, 0, 0, // dev
  3174  				0, 0, 0, 0, 0, 0, 0, 0, // ino
  3175  				4, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
  3176  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
  3177  				14, 0, 0, 0, 0, 0, 0, 0, // size
  3178  				0x0, 0x0, 0xc2, 0xd3, 0x43, 0x6, 0x36, 0x17, // atim
  3179  				0x0, 0x0, 0xc2, 0xd3, 0x43, 0x6, 0x36, 0x17, // mtim
  3180  				0x0, 0x0, 0xc2, 0xd3, 0x43, 0x6, 0x36, 0x17, // ctim
  3181  			),
  3182  			expectedLog: `
  3183  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=,path=sub/test.txt)
  3184  <== (filestat={filetype=REGULAR_FILE,size=14,mtim=1672531200000000000},errno=ESUCCESS)
  3185  `,
  3186  		},
  3187  		{
  3188  			name:           "dir under root",
  3189  			fd:             sys.FdPreopen,
  3190  			memory:         initialMemoryDir,
  3191  			pathLen:        uint32(len(dir)),
  3192  			resultFilestat: uint32(len(dir)) + 1,
  3193  			expectedMemory: append(
  3194  				initialMemoryDir,
  3195  				0, 0, 0, 0, 0, 0, 0, 0, // dev
  3196  				0, 0, 0, 0, 0, 0, 0, 0, // ino
  3197  				3, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
  3198  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
  3199  				0, 0, 0, 0, 0, 0, 0, 0, // size
  3200  				0x0, 0x0, 0x1f, 0xa6, 0x70, 0xfc, 0xc5, 0x16, // atim
  3201  				0x0, 0x0, 0x1f, 0xa6, 0x70, 0xfc, 0xc5, 0x16, // mtim
  3202  				0x0, 0x0, 0x1f, 0xa6, 0x70, 0xfc, 0xc5, 0x16, // ctim
  3203  			),
  3204  			expectedLog: `
  3205  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=,path=sub)
  3206  <== (filestat={filetype=DIRECTORY,size=0,mtim=1640995200000000000},errno=ESUCCESS)
  3207  `,
  3208  		},
  3209  		{
  3210  			name:          "unopened FD",
  3211  			fd:            -1,
  3212  			expectedErrno: wasip1.ErrnoBadf,
  3213  			expectedLog: `
  3214  ==> wasi_snapshot_preview1.path_filestat_get(fd=-1,flags=,path=)
  3215  <== (filestat=,errno=EBADF)
  3216  `,
  3217  		},
  3218  		{
  3219  			name:           "Fd not a directory",
  3220  			fd:             fileFD,
  3221  			memory:         initialMemoryFile,
  3222  			pathLen:        uint32(len(file)),
  3223  			resultFilestat: 2,
  3224  			expectedErrno:  wasip1.ErrnoNotdir,
  3225  			expectedLog: `
  3226  ==> wasi_snapshot_preview1.path_filestat_get(fd=4,flags=,path=animals.txt)
  3227  <== (filestat=,errno=ENOTDIR)
  3228  `,
  3229  		},
  3230  		{
  3231  			name:           "path under root doesn't exist",
  3232  			fd:             sys.FdPreopen,
  3233  			memory:         initialMemoryNotExists,
  3234  			pathLen:        1,
  3235  			resultFilestat: 2,
  3236  			expectedErrno:  wasip1.ErrnoNoent,
  3237  			expectedLog: `
  3238  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=,path=?)
  3239  <== (filestat=,errno=ENOENT)
  3240  `,
  3241  		},
  3242  		{
  3243  			name:          "path is out of memory",
  3244  			fd:            sys.FdPreopen,
  3245  			memory:        initialMemoryFile,
  3246  			pathLen:       memorySize,
  3247  			expectedErrno: wasip1.ErrnoFault,
  3248  			expectedLog: `
  3249  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=,path=OOM(1,65536))
  3250  <== (filestat=,errno=EFAULT)
  3251  `,
  3252  		},
  3253  		{
  3254  			name:           "resultFilestat exceeds the maximum valid address by 1",
  3255  			fd:             sys.FdPreopen,
  3256  			memory:         initialMemoryFile,
  3257  			pathLen:        uint32(len(file)),
  3258  			resultFilestat: memorySize - 64 + 1,
  3259  			expectedErrno:  wasip1.ErrnoFault,
  3260  			expectedLog: `
  3261  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=,path=animals.txt)
  3262  <== (filestat=,errno=EFAULT)
  3263  `,
  3264  		},
  3265  		{
  3266  			name:           "file under root (follow symlinks)",
  3267  			fd:             sys.FdPreopen,
  3268  			flags:          wasip1.LOOKUP_SYMLINK_FOLLOW,
  3269  			memory:         initialMemoryFile,
  3270  			pathLen:        uint32(len(file)),
  3271  			resultFilestat: uint32(len(file)) + 1,
  3272  			expectedMemory: append(
  3273  				initialMemoryFile,
  3274  				0, 0, 0, 0, 0, 0, 0, 0, // dev
  3275  				0, 0, 0, 0, 0, 0, 0, 0, // ino
  3276  				4, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
  3277  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
  3278  				30, 0, 0, 0, 0, 0, 0, 0, // size
  3279  				0x0, 0x82, 0x13, 0x80, 0x6b, 0x16, 0x24, 0x17, // atim
  3280  				0x0, 0x82, 0x13, 0x80, 0x6b, 0x16, 0x24, 0x17, // mtim
  3281  				0x0, 0x82, 0x13, 0x80, 0x6b, 0x16, 0x24, 0x17, // ctim
  3282  			),
  3283  			expectedLog: `
  3284  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=SYMLINK_FOLLOW,path=animals.txt)
  3285  <== (filestat={filetype=REGULAR_FILE,size=30,mtim=1667482413000000000},errno=ESUCCESS)
  3286  `,
  3287  		},
  3288  		{
  3289  			name:           "file under dir (follow symlinks)",
  3290  			fd:             sys.FdPreopen, // root
  3291  			flags:          wasip1.LOOKUP_SYMLINK_FOLLOW,
  3292  			memory:         initialMemoryFileInDir,
  3293  			pathLen:        uint32(len(fileInDir)),
  3294  			resultFilestat: uint32(len(fileInDir)) + 1,
  3295  			expectedMemory: append(
  3296  				initialMemoryFileInDir,
  3297  				0, 0, 0, 0, 0, 0, 0, 0, // dev
  3298  				0, 0, 0, 0, 0, 0, 0, 0, // ino
  3299  				4, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
  3300  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
  3301  				14, 0, 0, 0, 0, 0, 0, 0, // size
  3302  				0x0, 0x0, 0xc2, 0xd3, 0x43, 0x6, 0x36, 0x17, // atim
  3303  				0x0, 0x0, 0xc2, 0xd3, 0x43, 0x6, 0x36, 0x17, // mtim
  3304  				0x0, 0x0, 0xc2, 0xd3, 0x43, 0x6, 0x36, 0x17, // ctim
  3305  			),
  3306  			expectedLog: `
  3307  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=SYMLINK_FOLLOW,path=sub/test.txt)
  3308  <== (filestat={filetype=REGULAR_FILE,size=14,mtim=1672531200000000000},errno=ESUCCESS)
  3309  `,
  3310  		},
  3311  		{
  3312  			name:           "dir under root (follow symlinks)",
  3313  			fd:             sys.FdPreopen,
  3314  			flags:          wasip1.LOOKUP_SYMLINK_FOLLOW,
  3315  			memory:         initialMemoryDir,
  3316  			pathLen:        uint32(len(dir)),
  3317  			resultFilestat: uint32(len(dir)) + 1,
  3318  			expectedMemory: append(
  3319  				initialMemoryDir,
  3320  				0, 0, 0, 0, 0, 0, 0, 0, // dev
  3321  				0, 0, 0, 0, 0, 0, 0, 0, // ino
  3322  				3, 0, 0, 0, 0, 0, 0, 0, // filetype + padding
  3323  				1, 0, 0, 0, 0, 0, 0, 0, // nlink
  3324  				0, 0, 0, 0, 0, 0, 0, 0, // size
  3325  				0x0, 0x0, 0x1f, 0xa6, 0x70, 0xfc, 0xc5, 0x16, // atim
  3326  				0x0, 0x0, 0x1f, 0xa6, 0x70, 0xfc, 0xc5, 0x16, // mtim
  3327  				0x0, 0x0, 0x1f, 0xa6, 0x70, 0xfc, 0xc5, 0x16, // ctim
  3328  			),
  3329  			expectedLog: `
  3330  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=SYMLINK_FOLLOW,path=sub)
  3331  <== (filestat={filetype=DIRECTORY,size=0,mtim=1640995200000000000},errno=ESUCCESS)
  3332  `,
  3333  		},
  3334  		{
  3335  			name:          "unopened FD (follow symlinks)",
  3336  			fd:            -1,
  3337  			flags:         wasip1.LOOKUP_SYMLINK_FOLLOW,
  3338  			expectedErrno: wasip1.ErrnoBadf,
  3339  			expectedLog: `
  3340  ==> wasi_snapshot_preview1.path_filestat_get(fd=-1,flags=SYMLINK_FOLLOW,path=)
  3341  <== (filestat=,errno=EBADF)
  3342  `,
  3343  		},
  3344  		{
  3345  			name:           "Fd not a directory (follow symlinks)",
  3346  			fd:             fileFD,
  3347  			flags:          wasip1.LOOKUP_SYMLINK_FOLLOW,
  3348  			memory:         initialMemoryFile,
  3349  			pathLen:        uint32(len(file)),
  3350  			resultFilestat: 2,
  3351  			expectedErrno:  wasip1.ErrnoNotdir,
  3352  			expectedLog: `
  3353  ==> wasi_snapshot_preview1.path_filestat_get(fd=4,flags=SYMLINK_FOLLOW,path=animals.txt)
  3354  <== (filestat=,errno=ENOTDIR)
  3355  `,
  3356  		},
  3357  		{
  3358  			name:           "path under root doesn't exist (follow symlinks)",
  3359  			fd:             sys.FdPreopen,
  3360  			flags:          wasip1.LOOKUP_SYMLINK_FOLLOW,
  3361  			memory:         initialMemoryNotExists,
  3362  			pathLen:        1,
  3363  			resultFilestat: 2,
  3364  			expectedErrno:  wasip1.ErrnoNoent,
  3365  			expectedLog: `
  3366  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=SYMLINK_FOLLOW,path=?)
  3367  <== (filestat=,errno=ENOENT)
  3368  `,
  3369  		},
  3370  		{
  3371  			name:          "path is out of memory (follow symlinks)",
  3372  			fd:            sys.FdPreopen,
  3373  			flags:         wasip1.LOOKUP_SYMLINK_FOLLOW,
  3374  			memory:        initialMemoryFile,
  3375  			pathLen:       memorySize,
  3376  			expectedErrno: wasip1.ErrnoFault,
  3377  			expectedLog: `
  3378  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=SYMLINK_FOLLOW,path=OOM(1,65536))
  3379  <== (filestat=,errno=EFAULT)
  3380  `,
  3381  		},
  3382  		{
  3383  			name:           "resultFilestat exceeds the maximum valid address by 1 (follow symlinks)",
  3384  			fd:             sys.FdPreopen,
  3385  			flags:          wasip1.LOOKUP_SYMLINK_FOLLOW,
  3386  			memory:         initialMemoryFile,
  3387  			pathLen:        uint32(len(file)),
  3388  			resultFilestat: memorySize - 64 + 1,
  3389  			expectedErrno:  wasip1.ErrnoFault,
  3390  			expectedLog: `
  3391  ==> wasi_snapshot_preview1.path_filestat_get(fd=3,flags=SYMLINK_FOLLOW,path=animals.txt)
  3392  <== (filestat=,errno=EFAULT)
  3393  `,
  3394  		},
  3395  	}
  3396  
  3397  	for _, tt := range tests {
  3398  		tc := tt
  3399  
  3400  		t.Run(tc.name, func(t *testing.T) {
  3401  			defer log.Reset()
  3402  
  3403  			maskMemory(t, mod, len(tc.expectedMemory))
  3404  			mod.Memory().Write(0, tc.memory)
  3405  
  3406  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.PathFilestatGetName, uint64(tc.fd), uint64(tc.flags), uint64(1), uint64(tc.pathLen), uint64(tc.resultFilestat))
  3407  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  3408  
  3409  			actual, ok := mod.Memory().Read(0, uint32(len(tc.expectedMemory)))
  3410  			require.True(t, ok)
  3411  			require.Equal(t, tc.expectedMemory, actual)
  3412  		})
  3413  	}
  3414  }
  3415  
  3416  func Test_pathFilestatSetTimes(t *testing.T) {
  3417  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  3418  
  3419  	file := "file"
  3420  	writeFile(t, tmpDir, file, []byte("012"))
  3421  	link := file + "-link"
  3422  	require.NoError(t, os.Symlink(joinPath(tmpDir, file), joinPath(tmpDir, link)))
  3423  
  3424  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().
  3425  		WithSysWalltime().
  3426  		WithFSConfig(wazero.NewFSConfig().WithDirMount(tmpDir, "")))
  3427  	defer r.Close(testCtx)
  3428  
  3429  	tests := []struct {
  3430  		name          string
  3431  		flags         uint16
  3432  		pathName      string
  3433  		mtime, atime  int64
  3434  		fstFlags      uint16
  3435  		expectedLog   string
  3436  		expectedErrno wasip1.Errno
  3437  	}{
  3438  		{
  3439  			name:  "a=omit,m=omit",
  3440  			flags: wasip1.LOOKUP_SYMLINK_FOLLOW,
  3441  			atime: 123451, // Must be ignored.
  3442  			mtime: 1234,   // Must be ignored.
  3443  			expectedLog: `
  3444  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=SYMLINK_FOLLOW,path=file,atim=123451,mtim=1234,fst_flags=)
  3445  <== errno=ESUCCESS
  3446  `,
  3447  		},
  3448  		{
  3449  			name:     "a=now,m=omit",
  3450  			flags:    wasip1.LOOKUP_SYMLINK_FOLLOW,
  3451  			atime:    123451, // Must be ignored.
  3452  			mtime:    1234,   // Must be ignored.
  3453  			fstFlags: wasip1.FstflagsAtimNow,
  3454  			expectedLog: `
  3455  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=SYMLINK_FOLLOW,path=file,atim=123451,mtim=1234,fst_flags=ATIM_NOW)
  3456  <== errno=ESUCCESS
  3457  `,
  3458  		},
  3459  		{
  3460  			name:     "a=omit,m=now",
  3461  			flags:    wasip1.LOOKUP_SYMLINK_FOLLOW,
  3462  			atime:    123451, // Must be ignored.
  3463  			mtime:    1234,   // Must be ignored.
  3464  			fstFlags: wasip1.FstflagsMtimNow,
  3465  			expectedLog: `
  3466  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=SYMLINK_FOLLOW,path=file,atim=123451,mtim=1234,fst_flags=MTIM_NOW)
  3467  <== errno=ESUCCESS
  3468  `,
  3469  		},
  3470  		{
  3471  			name:     "a=now,m=now",
  3472  			flags:    wasip1.LOOKUP_SYMLINK_FOLLOW,
  3473  			atime:    123451, // Must be ignored.
  3474  			mtime:    1234,   // Must be ignored.
  3475  			fstFlags: wasip1.FstflagsAtimNow | wasip1.FstflagsMtimNow,
  3476  			expectedLog: `
  3477  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=SYMLINK_FOLLOW,path=file,atim=123451,mtim=1234,fst_flags=ATIM_NOW|MTIM_NOW)
  3478  <== errno=ESUCCESS
  3479  `,
  3480  		},
  3481  		{
  3482  			name:     "a=now,m=set",
  3483  			flags:    wasip1.LOOKUP_SYMLINK_FOLLOW,
  3484  			atime:    1234, // Must be ignored.
  3485  			mtime:    55555500,
  3486  			fstFlags: wasip1.FstflagsAtimNow | wasip1.FstflagsMtim,
  3487  			expectedLog: `
  3488  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=SYMLINK_FOLLOW,path=file,atim=1234,mtim=55555500,fst_flags=ATIM_NOW|MTIM)
  3489  <== errno=ESUCCESS
  3490  `,
  3491  		},
  3492  		{
  3493  			name:     "a=set,m=now",
  3494  			flags:    wasip1.LOOKUP_SYMLINK_FOLLOW,
  3495  			atime:    55555500,
  3496  			mtime:    1234, // Must be ignored.
  3497  			fstFlags: wasip1.FstflagsAtim | wasip1.FstflagsMtimNow,
  3498  			expectedLog: `
  3499  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=SYMLINK_FOLLOW,path=file,atim=55555500,mtim=1234,fst_flags=ATIM|MTIM_NOW)
  3500  <== errno=ESUCCESS
  3501  `,
  3502  		},
  3503  		{
  3504  			name:     "a=set,m=omit",
  3505  			flags:    wasip1.LOOKUP_SYMLINK_FOLLOW,
  3506  			atime:    55555500,
  3507  			mtime:    1234, // Must be ignored.
  3508  			fstFlags: wasip1.FstflagsAtim,
  3509  			expectedLog: `
  3510  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=SYMLINK_FOLLOW,path=file,atim=55555500,mtim=1234,fst_flags=ATIM)
  3511  <== errno=ESUCCESS
  3512  `,
  3513  		},
  3514  		{
  3515  			name:     "a=omit,m=set",
  3516  			flags:    wasip1.LOOKUP_SYMLINK_FOLLOW,
  3517  			atime:    1234, // Must be ignored.
  3518  			mtime:    55555500,
  3519  			fstFlags: wasip1.FstflagsMtim,
  3520  			expectedLog: `
  3521  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=SYMLINK_FOLLOW,path=file,atim=1234,mtim=55555500,fst_flags=MTIM)
  3522  <== errno=ESUCCESS
  3523  `,
  3524  		},
  3525  		{
  3526  			name:     "a=set,m=set",
  3527  			flags:    wasip1.LOOKUP_SYMLINK_FOLLOW,
  3528  			atime:    6666666600,
  3529  			mtime:    55555500,
  3530  			fstFlags: wasip1.FstflagsAtim | wasip1.FstflagsMtim,
  3531  			expectedLog: `
  3532  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=SYMLINK_FOLLOW,path=file,atim=6666666600,mtim=55555500,fst_flags=ATIM|MTIM)
  3533  <== errno=ESUCCESS
  3534  `,
  3535  		},
  3536  		{
  3537  			name:     "not found",
  3538  			pathName: "nope",
  3539  			flags:    wasip1.LOOKUP_SYMLINK_FOLLOW,
  3540  			fstFlags: wasip1.FstflagsAtimNow, // Choose one flag to ensure an update occurs
  3541  			expectedLog: `
  3542  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=SYMLINK_FOLLOW,path=nope,atim=0,mtim=0,fst_flags=ATIM_NOW)
  3543  <== errno=ENOENT
  3544  `,
  3545  			expectedErrno: wasip1.ErrnoNoent,
  3546  		},
  3547  		{
  3548  			name:     "no_symlink_follow",
  3549  			pathName: link,
  3550  			flags:    0,
  3551  			atime:    123451, // Must be ignored.
  3552  			mtime:    1234,   // Must be ignored.
  3553  			fstFlags: wasip1.FstflagsMtimNow,
  3554  			expectedLog: `
  3555  ==> wasi_snapshot_preview1.path_filestat_set_times(fd=3,flags=,path=file-link,atim=123451,mtim=1234,fst_flags=MTIM_NOW)
  3556  <== errno=ESUCCESS
  3557  `,
  3558  		},
  3559  	}
  3560  
  3561  	if runtime.GOOS == "windows" && !platform.IsAtLeastGo120 {
  3562  		// Windows 1.19 returns ENOSYS on no_symlink_follow
  3563  		tests = tests[:len(tests)-1]
  3564  	}
  3565  
  3566  	for _, tt := range tests {
  3567  		tc := tt
  3568  
  3569  		t.Run(tc.name, func(t *testing.T) {
  3570  			defer log.Reset()
  3571  
  3572  			pathName := tc.pathName
  3573  			if pathName == "" {
  3574  				pathName = file
  3575  			}
  3576  			mod.Memory().Write(0, []byte(pathName))
  3577  
  3578  			fd := sys.FdPreopen
  3579  			path := 0
  3580  			pathLen := uint32(len(pathName))
  3581  
  3582  			sys := mod.(*wasm.ModuleInstance).Sys
  3583  			fsc := sys.FS()
  3584  
  3585  			var oldSt sysapi.Stat_t
  3586  			var errno experimentalsys.Errno
  3587  			if tc.expectedErrno == wasip1.ErrnoSuccess {
  3588  				oldSt, errno = fsc.RootFS().Stat(pathName)
  3589  				require.EqualErrno(t, 0, errno)
  3590  			}
  3591  
  3592  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.PathFilestatSetTimesName, uint64(fd), uint64(tc.flags),
  3593  				uint64(path), uint64(pathLen), uint64(tc.atime), uint64(tc.mtime), uint64(tc.fstFlags))
  3594  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  3595  
  3596  			if tc.expectedErrno != wasip1.ErrnoSuccess {
  3597  				return
  3598  			}
  3599  
  3600  			newSt, errno := fsc.RootFS().Stat(pathName)
  3601  			require.EqualErrno(t, 0, errno)
  3602  
  3603  			if platform.CompilerSupported() {
  3604  				if tc.fstFlags&wasip1.FstflagsAtim != 0 {
  3605  					require.Equal(t, tc.atime, newSt.Atim)
  3606  				} else if tc.fstFlags&wasip1.FstflagsAtimNow != 0 {
  3607  					now := time.Now().UnixNano()
  3608  					require.True(t, newSt.Atim <= now, "expected atim %d <= now %d", newSt.Atim, now)
  3609  				} else { // omit
  3610  					require.Equal(t, oldSt.Atim, newSt.Atim)
  3611  				}
  3612  			}
  3613  
  3614  			// When compiler isn't supported, we can still check mtim.
  3615  			if tc.fstFlags&wasip1.FstflagsMtim != 0 {
  3616  				require.Equal(t, tc.mtime, newSt.Mtim)
  3617  			} else if tc.fstFlags&wasip1.FstflagsMtimNow != 0 {
  3618  				now := time.Now().UnixNano()
  3619  				require.True(t, newSt.Mtim <= now, "expected mtim %d <= now %d", newSt.Mtim, now)
  3620  			} else { // omit
  3621  				require.Equal(t, oldSt.Mtim, newSt.Mtim)
  3622  			}
  3623  		})
  3624  	}
  3625  }
  3626  
  3627  func Test_pathLink(t *testing.T) {
  3628  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  3629  
  3630  	oldDirName := "my-old-dir"
  3631  	oldDirPath := joinPath(tmpDir, oldDirName)
  3632  	mod, oldFd, log, r := requireOpenFile(t, tmpDir, oldDirName, nil, false)
  3633  	defer r.Close(testCtx)
  3634  
  3635  	newDirName := "my-new-dir/sub"
  3636  	newDirPath := joinPath(tmpDir, newDirName)
  3637  	require.NoError(t, os.MkdirAll(joinPath(tmpDir, newDirName), 0o700))
  3638  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  3639  	newFd, errno := fsc.OpenFile(fsc.RootFS(), newDirName, 0o600, 0)
  3640  	require.EqualErrno(t, 0, errno)
  3641  
  3642  	mem := mod.Memory()
  3643  
  3644  	fileName := "file"
  3645  	err := os.WriteFile(joinPath(oldDirPath, fileName), []byte{1, 2, 3, 4}, 0o700)
  3646  	require.NoError(t, err)
  3647  
  3648  	file := uint32(0xff)
  3649  	ok := mem.Write(file, []byte(fileName))
  3650  	require.True(t, ok)
  3651  
  3652  	notFoundFile := uint32(0xaa)
  3653  	notFoundFileName := "nope"
  3654  	ok = mem.Write(notFoundFile, []byte(notFoundFileName))
  3655  	require.True(t, ok)
  3656  
  3657  	destination := uint32(0xcc)
  3658  	destinationName := "hard-linked"
  3659  	ok = mem.Write(destination, []byte(destinationName))
  3660  	require.True(t, ok)
  3661  
  3662  	destinationRealPath := joinPath(newDirPath, destinationName)
  3663  
  3664  	t.Run("success", func(t *testing.T) {
  3665  		requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.PathLinkName,
  3666  			uint64(oldFd), 0, uint64(file), uint64(len(fileName)),
  3667  			uint64(newFd), uint64(destination), uint64(len(destinationName)))
  3668  		require.Contains(t, log.String(), wasip1.ErrnoName(wasip1.ErrnoSuccess))
  3669  
  3670  		f := openFile(t, destinationRealPath, experimentalsys.O_RDONLY, 0)
  3671  		defer f.Close()
  3672  
  3673  		st, errno := f.Stat()
  3674  		require.EqualErrno(t, 0, errno)
  3675  		require.False(t, st.Mode&os.ModeSymlink == os.ModeSymlink)
  3676  		require.Equal(t, uint64(2), st.Nlink)
  3677  	})
  3678  
  3679  	t.Run("errors", func(t *testing.T) {
  3680  		for _, tc := range []struct {
  3681  			errno wasip1.Errno
  3682  			oldFd int32
  3683  			/* oldFlags, */ oldPath, oldPathLen uint32
  3684  			newFd                               int32
  3685  			newPath, newPathLen                 uint32
  3686  		}{
  3687  			{errno: wasip1.ErrnoBadf, oldFd: 1000},
  3688  			{errno: wasip1.ErrnoBadf, oldFd: oldFd, newFd: 1000},
  3689  			{errno: wasip1.ErrnoNotdir, oldFd: oldFd, newFd: 1},
  3690  			{errno: wasip1.ErrnoNotdir, oldFd: 1, newFd: 1},
  3691  			{errno: wasip1.ErrnoNotdir, oldFd: 1, newFd: newFd},
  3692  			{errno: wasip1.ErrnoFault, oldFd: oldFd, newFd: newFd, oldPathLen: math.MaxUint32},
  3693  			{errno: wasip1.ErrnoFault, oldFd: oldFd, newFd: newFd, newPathLen: math.MaxUint32},
  3694  			{
  3695  				errno: wasip1.ErrnoFault, oldFd: oldFd, newFd: newFd,
  3696  				oldPath: math.MaxUint32, oldPathLen: 100, newPathLen: 100,
  3697  			},
  3698  			{
  3699  				errno: wasip1.ErrnoFault, oldFd: oldFd, newFd: newFd,
  3700  				oldPath: 1, oldPathLen: 100, newPath: math.MaxUint32, newPathLen: 100,
  3701  			},
  3702  		} {
  3703  			name := wasip1.ErrnoName(tc.errno)
  3704  			t.Run(name, func(t *testing.T) {
  3705  				requireErrnoResult(t, tc.errno, mod, wasip1.PathLinkName,
  3706  					uint64(tc.oldFd), 0, uint64(tc.oldPath), uint64(tc.oldPathLen),
  3707  					uint64(tc.newFd), uint64(tc.newPath), uint64(tc.newPathLen))
  3708  				require.Contains(t, log.String(), name)
  3709  			})
  3710  		}
  3711  	})
  3712  }
  3713  
  3714  func Test_pathOpen(t *testing.T) {
  3715  	dir := t.TempDir() // open before loop to ensure no locking problems.
  3716  	writeFS := sysfs.DirFS(dir)
  3717  	readFS := &sysfs.ReadFS{FS: writeFS}
  3718  
  3719  	fileName := "file"
  3720  	fileContents := []byte("012")
  3721  	writeFile(t, dir, fileName, fileContents)
  3722  
  3723  	appendName := "append"
  3724  	appendContents := []byte("345")
  3725  	writeFile(t, dir, appendName, appendContents)
  3726  
  3727  	truncName := "trunc"
  3728  	truncContents := []byte("678")
  3729  	writeFile(t, dir, truncName, truncContents)
  3730  
  3731  	dirName := "dir"
  3732  	mkdir(t, dir, dirName)
  3733  
  3734  	dirFileName := joinPath(dirName, fileName)
  3735  	dirFileContents := []byte("def")
  3736  	writeFile(t, dir, dirFileName, dirFileContents)
  3737  
  3738  	expectedOpenedFd := sys.FdPreopen + 1
  3739  
  3740  	tests := []struct {
  3741  		name          string
  3742  		fs            experimentalsys.FS
  3743  		path          func(t *testing.T) string
  3744  		oflags        uint16
  3745  		fdflags       uint16
  3746  		rights        uint32
  3747  		expected      func(t *testing.T, fsc *sys.FSContext)
  3748  		expectedErrno wasip1.Errno
  3749  		expectedLog   string
  3750  	}{
  3751  		{
  3752  			name: "sysfs.ReadFS",
  3753  			fs:   readFS,
  3754  			path: func(*testing.T) string { return fileName },
  3755  			expected: func(t *testing.T, fsc *sys.FSContext) {
  3756  				requireContents(t, fsc, expectedOpenedFd, fileName, fileContents)
  3757  			},
  3758  			expectedLog: `
  3759  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=file,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  3760  <== (opened_fd=4,errno=ESUCCESS)
  3761  `,
  3762  		},
  3763  		{
  3764  			name: "sysfs.DirFS",
  3765  			fs:   writeFS,
  3766  			path: func(*testing.T) string { return fileName },
  3767  			expected: func(t *testing.T, fsc *sys.FSContext) {
  3768  				requireContents(t, fsc, expectedOpenedFd, fileName, fileContents)
  3769  			},
  3770  			expectedLog: `
  3771  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=file,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  3772  <== (opened_fd=4,errno=ESUCCESS)
  3773  `,
  3774  		},
  3775  		{
  3776  			name:          "sysfs.ReadFS FD_APPEND",
  3777  			fs:            readFS,
  3778  			fdflags:       wasip1.FD_APPEND,
  3779  			path:          func(t *testing.T) (file string) { return appendName },
  3780  			expectedErrno: wasip1.ErrnoNosys,
  3781  			expectedLog: `
  3782  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=append,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=APPEND)
  3783  <== (opened_fd=,errno=ENOSYS)
  3784  `,
  3785  		},
  3786  		{
  3787  			name:    "sysfs.DirFS FD_APPEND",
  3788  			fs:      writeFS,
  3789  			path:    func(t *testing.T) (file string) { return appendName },
  3790  			fdflags: wasip1.FD_APPEND,
  3791  			expected: func(t *testing.T, fsc *sys.FSContext) {
  3792  				contents := writeAndCloseFile(t, fsc, expectedOpenedFd)
  3793  
  3794  				// verify the contents were appended
  3795  				b := readFile(t, dir, appendName)
  3796  				require.Equal(t, append(appendContents, contents...), b)
  3797  			},
  3798  			expectedLog: `
  3799  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=append,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=APPEND)
  3800  <== (opened_fd=4,errno=ESUCCESS)
  3801  `,
  3802  		},
  3803  		{
  3804  			name:          "sysfs.ReadFS O_CREAT",
  3805  			fs:            readFS,
  3806  			oflags:        wasip1.O_CREAT,
  3807  			expectedErrno: wasip1.ErrnoNosys,
  3808  			path:          func(*testing.T) string { return "creat" },
  3809  			expectedLog: `
  3810  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=creat,oflags=CREAT,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  3811  <== (opened_fd=,errno=ENOSYS)
  3812  `,
  3813  		},
  3814  		{
  3815  			name:   "sysfs.DirFS O_CREAT",
  3816  			fs:     writeFS,
  3817  			path:   func(t *testing.T) (file string) { return "creat" },
  3818  			oflags: wasip1.O_CREAT,
  3819  			expected: func(t *testing.T, fsc *sys.FSContext) {
  3820  				// expect to create a new file
  3821  				contents := writeAndCloseFile(t, fsc, expectedOpenedFd)
  3822  
  3823  				// verify the contents were written
  3824  				b := readFile(t, dir, "creat")
  3825  				require.Equal(t, contents, b)
  3826  			},
  3827  			expectedLog: `
  3828  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=creat,oflags=CREAT,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  3829  <== (opened_fd=4,errno=ESUCCESS)
  3830  `,
  3831  		},
  3832  		{
  3833  			name:          "sysfs.ReadFS O_CREAT O_TRUNC",
  3834  			fs:            readFS,
  3835  			oflags:        wasip1.O_CREAT | wasip1.O_TRUNC,
  3836  			expectedErrno: wasip1.ErrnoNosys,
  3837  			path:          func(t *testing.T) (file string) { return joinPath(dirName, "O_CREAT-O_TRUNC") },
  3838  			expectedLog: `
  3839  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=dir/O_CREAT-O_TRUNC,oflags=CREAT|TRUNC,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  3840  <== (opened_fd=,errno=ENOSYS)
  3841  `,
  3842  		},
  3843  		{
  3844  			name:   "sysfs.DirFS O_CREAT O_TRUNC",
  3845  			fs:     writeFS,
  3846  			path:   func(t *testing.T) (file string) { return joinPath(dirName, "O_CREAT-O_TRUNC") },
  3847  			oflags: wasip1.O_CREAT | wasip1.O_TRUNC,
  3848  			expected: func(t *testing.T, fsc *sys.FSContext) {
  3849  				// expect to create a new file
  3850  				contents := writeAndCloseFile(t, fsc, expectedOpenedFd)
  3851  
  3852  				// verify the contents were written
  3853  				b := readFile(t, dir, joinPath(dirName, "O_CREAT-O_TRUNC"))
  3854  				require.Equal(t, contents, b)
  3855  			},
  3856  			expectedLog: `
  3857  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=dir/O_CREAT-O_TRUNC,oflags=CREAT|TRUNC,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  3858  <== (opened_fd=4,errno=ESUCCESS)
  3859  `,
  3860  		},
  3861  		{
  3862  			name:   "sysfs.ReadFS O_DIRECTORY",
  3863  			fs:     readFS,
  3864  			oflags: wasip1.O_DIRECTORY,
  3865  			path:   func(*testing.T) string { return dirName },
  3866  			expected: func(t *testing.T, fsc *sys.FSContext) {
  3867  				f, ok := fsc.LookupFile(expectedOpenedFd)
  3868  				require.True(t, ok)
  3869  				isDir, errno := f.File.IsDir()
  3870  				require.EqualErrno(t, 0, errno)
  3871  				require.True(t, isDir)
  3872  			},
  3873  			expectedLog: `
  3874  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=dir,oflags=DIRECTORY,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  3875  <== (opened_fd=4,errno=ESUCCESS)
  3876  `,
  3877  		},
  3878  		{
  3879  			name:   "sysfs.DirFS O_DIRECTORY",
  3880  			fs:     writeFS,
  3881  			path:   func(*testing.T) string { return dirName },
  3882  			oflags: wasip1.O_DIRECTORY,
  3883  			expected: func(t *testing.T, fsc *sys.FSContext) {
  3884  				f, ok := fsc.LookupFile(expectedOpenedFd)
  3885  				require.True(t, ok)
  3886  				isDir, errno := f.File.IsDir()
  3887  				require.EqualErrno(t, 0, errno)
  3888  				require.True(t, isDir)
  3889  			},
  3890  			expectedLog: `
  3891  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=dir,oflags=DIRECTORY,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  3892  <== (opened_fd=4,errno=ESUCCESS)
  3893  `,
  3894  		},
  3895  		{
  3896  			name:          "sysfs.ReadFS O_TRUNC",
  3897  			fs:            readFS,
  3898  			oflags:        wasip1.O_TRUNC,
  3899  			expectedErrno: wasip1.ErrnoNosys,
  3900  			path:          func(*testing.T) string { return "trunc" },
  3901  			expectedLog: `
  3902  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=trunc,oflags=TRUNC,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  3903  <== (opened_fd=,errno=ENOSYS)
  3904  `,
  3905  		},
  3906  		{
  3907  			name:   "sysfs.DirFS O_TRUNC",
  3908  			fs:     writeFS,
  3909  			path:   func(t *testing.T) (file string) { return "trunc" },
  3910  			oflags: wasip1.O_TRUNC,
  3911  			expected: func(t *testing.T, fsc *sys.FSContext) {
  3912  				contents := writeAndCloseFile(t, fsc, expectedOpenedFd)
  3913  
  3914  				// verify the contents were truncated
  3915  				b := readFile(t, dir, "trunc")
  3916  				require.Equal(t, contents, b)
  3917  			},
  3918  			expectedLog: `
  3919  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=trunc,oflags=TRUNC,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  3920  <== (opened_fd=4,errno=ESUCCESS)
  3921  `,
  3922  		},
  3923  		{
  3924  			name:   "sysfs.DirFS RIGHT_FD_READ|RIGHT_FD_WRITE",
  3925  			fs:     writeFS,
  3926  			path:   func(*testing.T) string { return fileName },
  3927  			oflags: 0,
  3928  			rights: wasip1.RIGHT_FD_READ | wasip1.RIGHT_FD_WRITE,
  3929  			expected: func(t *testing.T, fsc *sys.FSContext) {
  3930  				requireContents(t, fsc, expectedOpenedFd, fileName, fileContents)
  3931  			},
  3932  			expectedLog: `
  3933  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=file,oflags=,fs_rights_base=FD_READ|FD_WRITE,fs_rights_inheriting=,fdflags=)
  3934  <== (opened_fd=4,errno=ESUCCESS)
  3935  `,
  3936  		},
  3937  	}
  3938  
  3939  	for _, tt := range tests {
  3940  		tc := tt
  3941  
  3942  		t.Run(tc.name, func(t *testing.T) {
  3943  			mod, r, log := requireProxyModule(t, wazero.NewModuleConfig())
  3944  			defer r.Close(testCtx)
  3945  
  3946  			mod.(*wasm.ModuleInstance).Sys = sys.DefaultContext(tc.fs)
  3947  
  3948  			pathName := tc.path(t)
  3949  			mod.Memory().Write(0, []byte(pathName))
  3950  
  3951  			path := uint32(0)
  3952  			pathLen := uint32(len(pathName))
  3953  			resultOpenedFd := pathLen
  3954  			fd := sys.FdPreopen
  3955  
  3956  			// TODO: dirflags is a lookupflags and it only has one bit: symlink_follow
  3957  			// https://github.com/WebAssembly/WASI/blob/snapshot-01/phases/snapshot/docs.md#lookupflags
  3958  			dirflags := 0
  3959  
  3960  			// inherited rights aren't used
  3961  			fsRightsInheriting := uint64(0)
  3962  
  3963  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.PathOpenName, uint64(fd), uint64(dirflags), uint64(path),
  3964  				uint64(pathLen), uint64(tc.oflags), uint64(tc.rights), fsRightsInheriting, uint64(tc.fdflags), uint64(resultOpenedFd))
  3965  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  3966  
  3967  			if tc.expectedErrno == wasip1.ErrnoSuccess {
  3968  				openedFd, ok := mod.Memory().ReadUint32Le(pathLen)
  3969  				require.True(t, ok)
  3970  				require.Equal(t, expectedOpenedFd, int32(openedFd))
  3971  
  3972  				tc.expected(t, mod.(*wasm.ModuleInstance).Sys.FS())
  3973  			}
  3974  		})
  3975  	}
  3976  }
  3977  
  3978  func writeAndCloseFile(t *testing.T, fsc *sys.FSContext, fd int32) []byte {
  3979  	contents := []byte("hello")
  3980  	f, ok := fsc.LookupFile(fd)
  3981  	require.True(t, ok)
  3982  	_, errno := f.File.Write([]byte("hello"))
  3983  	require.EqualErrno(t, 0, errno)
  3984  	require.EqualErrno(t, 0, fsc.CloseFile(fd))
  3985  	return contents
  3986  }
  3987  
  3988  func requireOpenFD(t *testing.T, mod api.Module, path string) int32 {
  3989  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  3990  	preopen := fsc.RootFS()
  3991  
  3992  	fd, errno := fsc.OpenFile(preopen, path, experimentalsys.O_RDONLY, 0)
  3993  	require.EqualErrno(t, 0, errno)
  3994  	return fd
  3995  }
  3996  
  3997  func requireContents(t *testing.T, fsc *sys.FSContext, expectedOpenedFd int32, fileName string, fileContents []byte) {
  3998  	// verify the file was actually opened
  3999  	f, ok := fsc.LookupFile(expectedOpenedFd)
  4000  	require.True(t, ok)
  4001  	require.Equal(t, fileName, f.Name)
  4002  
  4003  	// verify the contents are readable
  4004  	buf := readAll(t, f.File)
  4005  	require.Equal(t, fileContents, buf)
  4006  }
  4007  
  4008  func readAll(t *testing.T, f experimentalsys.File) []byte {
  4009  	st, errno := f.Stat()
  4010  	require.EqualErrno(t, 0, errno)
  4011  	buf := make([]byte, st.Size)
  4012  	_, errno = f.Read(buf)
  4013  	require.EqualErrno(t, 0, errno)
  4014  	return buf
  4015  }
  4016  
  4017  func mkdir(t *testing.T, tmpDir, dir string) {
  4018  	err := os.Mkdir(joinPath(tmpDir, dir), 0o700)
  4019  	require.NoError(t, err)
  4020  }
  4021  
  4022  func readFile(t *testing.T, tmpDir, file string) []byte {
  4023  	contents, err := os.ReadFile(joinPath(tmpDir, file))
  4024  	require.NoError(t, err)
  4025  	return contents
  4026  }
  4027  
  4028  func writeFile(t *testing.T, tmpDir, file string, contents []byte) {
  4029  	err := os.WriteFile(joinPath(tmpDir, file), contents, 0o600)
  4030  	require.NoError(t, err)
  4031  }
  4032  
  4033  func Test_pathOpen_Errors(t *testing.T) {
  4034  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  4035  	fsConfig := wazero.NewFSConfig().WithDirMount(tmpDir, "/")
  4036  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  4037  	defer r.Close(testCtx)
  4038  
  4039  	file := "file"
  4040  	err := os.WriteFile(joinPath(tmpDir, file), []byte{}, 0o700)
  4041  	require.NoError(t, err)
  4042  	fileFD := requireOpenFD(t, mod, file)
  4043  
  4044  	dir := "dir"
  4045  	err = os.Mkdir(joinPath(tmpDir, dir), 0o700)
  4046  	require.NoError(t, err)
  4047  
  4048  	nested := "dir/nested"
  4049  	err = os.Mkdir(joinPath(tmpDir, nested), 0o700)
  4050  	require.NoError(t, err)
  4051  
  4052  	nestedFile := "dir/nested/file"
  4053  	err = os.WriteFile(joinPath(tmpDir, nestedFile), []byte{}, 0o700)
  4054  	require.NoError(t, err)
  4055  
  4056  	tests := []struct {
  4057  		name, pathName                        string
  4058  		fd                                    int32
  4059  		path, pathLen, oflags, resultOpenedFd uint32
  4060  		expectedErrno                         wasip1.Errno
  4061  		expectedLog                           string
  4062  	}{
  4063  		{
  4064  			name:          "unopened FD",
  4065  			fd:            42, // arbitrary invalid fd
  4066  			expectedErrno: wasip1.ErrnoBadf,
  4067  			expectedLog: `
  4068  ==> wasi_snapshot_preview1.path_open(fd=42,dirflags=,path=,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4069  <== (opened_fd=,errno=EBADF)
  4070  `,
  4071  		},
  4072  		{
  4073  			name:          "Fd not a directory",
  4074  			fd:            fileFD,
  4075  			pathName:      file,
  4076  			path:          0,
  4077  			pathLen:       uint32(len(file)),
  4078  			expectedErrno: wasip1.ErrnoNotdir,
  4079  			expectedLog: `
  4080  ==> wasi_snapshot_preview1.path_open(fd=4,dirflags=,path=file,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4081  <== (opened_fd=,errno=ENOTDIR)
  4082  `,
  4083  		},
  4084  		{
  4085  			name:          "out-of-memory reading path",
  4086  			fd:            sys.FdPreopen,
  4087  			path:          mod.Memory().Size(),
  4088  			pathLen:       uint32(len(file)),
  4089  			expectedErrno: wasip1.ErrnoFault,
  4090  			expectedLog: `
  4091  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=OOM(65536,4),oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4092  <== (opened_fd=,errno=EFAULT)
  4093  `,
  4094  		},
  4095  		{
  4096  			name:          "out-of-memory reading pathLen",
  4097  			fd:            sys.FdPreopen,
  4098  			path:          0,
  4099  			pathLen:       mod.Memory().Size() + 1, // path is in the valid memory range, but pathLen is OOM for path
  4100  			expectedErrno: wasip1.ErrnoFault,
  4101  			expectedLog: `
  4102  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=OOM(0,65537),oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4103  <== (opened_fd=,errno=EFAULT)
  4104  `,
  4105  		},
  4106  		{
  4107  			name:          "no such file exists",
  4108  			fd:            sys.FdPreopen,
  4109  			pathName:      dir,
  4110  			path:          0,
  4111  			pathLen:       uint32(len(dir)) - 1,
  4112  			expectedErrno: wasip1.ErrnoNoent,
  4113  			expectedLog: `
  4114  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=di,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4115  <== (opened_fd=,errno=ENOENT)
  4116  `,
  4117  		},
  4118  		{
  4119  			name:          "trailing slash on directory",
  4120  			fd:            sys.FdPreopen,
  4121  			pathName:      nested + "/",
  4122  			path:          0,
  4123  			pathLen:       uint32(len(nested)) + 1,
  4124  			expectedErrno: wasip1.ErrnoSuccess,
  4125  			expectedLog: `
  4126  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=dir/nested/,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4127  <== (opened_fd=5,errno=ESUCCESS)
  4128  `,
  4129  		},
  4130  		{
  4131  			name:          "path under preopen",
  4132  			fd:            sys.FdPreopen,
  4133  			pathName:      "../" + file,
  4134  			path:          0,
  4135  			pathLen:       uint32(len(file)) + 3,
  4136  			expectedErrno: wasip1.ErrnoPerm,
  4137  			expectedLog: `
  4138  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=../file,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4139  <== (opened_fd=,errno=EPERM)
  4140  `,
  4141  		},
  4142  		{
  4143  			name:          "rooted path",
  4144  			fd:            sys.FdPreopen,
  4145  			pathName:      "/" + file,
  4146  			path:          0,
  4147  			pathLen:       uint32(len(file)) + 1,
  4148  			expectedErrno: wasip1.ErrnoPerm,
  4149  			expectedLog: `
  4150  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=/file,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4151  <== (opened_fd=,errno=EPERM)
  4152  `,
  4153  		},
  4154  		{
  4155  			name:          "trailing slash on file",
  4156  			fd:            sys.FdPreopen,
  4157  			pathName:      nestedFile + "/",
  4158  			path:          0,
  4159  			pathLen:       uint32(len(nestedFile)) + 1,
  4160  			expectedErrno: wasip1.ErrnoNotdir,
  4161  			expectedLog: `
  4162  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=dir/nested/file/,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4163  <== (opened_fd=,errno=ENOTDIR)
  4164  `,
  4165  		},
  4166  		{
  4167  			name:           "out-of-memory writing resultOpenedFd",
  4168  			fd:             sys.FdPreopen,
  4169  			pathName:       dir,
  4170  			path:           0,
  4171  			pathLen:        uint32(len(dir)),
  4172  			resultOpenedFd: mod.Memory().Size(), // path and pathLen correctly point to the right path, but where to write the opened FD is outside memory.
  4173  			expectedErrno:  wasip1.ErrnoFault,
  4174  			expectedLog: `
  4175  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=dir,oflags=,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4176  <== (opened_fd=,errno=EFAULT)
  4177  `,
  4178  		},
  4179  		{
  4180  			name:          "O_DIRECTORY, but not a directory",
  4181  			oflags:        uint32(wasip1.O_DIRECTORY),
  4182  			fd:            sys.FdPreopen,
  4183  			pathName:      file,
  4184  			path:          0,
  4185  			pathLen:       uint32(len(file)),
  4186  			expectedErrno: wasip1.ErrnoNotdir,
  4187  			expectedLog: `
  4188  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=file,oflags=DIRECTORY,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4189  <== (opened_fd=,errno=ENOTDIR)
  4190  `,
  4191  		},
  4192  		{
  4193  			name:          "oflags=directory and create invalid",
  4194  			oflags:        uint32(wasip1.O_DIRECTORY | wasip1.O_CREAT),
  4195  			fd:            sys.FdPreopen,
  4196  			pathName:      file,
  4197  			path:          0,
  4198  			pathLen:       uint32(len(file)),
  4199  			expectedErrno: wasip1.ErrnoInval,
  4200  			expectedLog: `
  4201  ==> wasi_snapshot_preview1.path_open(fd=3,dirflags=,path=file,oflags=CREAT|DIRECTORY,fs_rights_base=,fs_rights_inheriting=,fdflags=)
  4202  <== (opened_fd=,errno=EINVAL)
  4203  `,
  4204  		},
  4205  	}
  4206  
  4207  	for _, tt := range tests {
  4208  		tc := tt
  4209  		t.Run(tc.name, func(t *testing.T) {
  4210  			defer log.Reset()
  4211  
  4212  			mod.Memory().Write(tc.path, []byte(tc.pathName))
  4213  
  4214  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.PathOpenName, uint64(tc.fd), uint64(0), uint64(tc.path),
  4215  				uint64(tc.pathLen), uint64(tc.oflags), 0, 0, 0, uint64(tc.resultOpenedFd))
  4216  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  4217  		})
  4218  	}
  4219  }
  4220  
  4221  func Test_pathReadlink(t *testing.T) {
  4222  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  4223  
  4224  	dirName := "dir"
  4225  	dirPath := joinPath(tmpDir, dirName)
  4226  	mod, dirFD, log, r := requireOpenFile(t, tmpDir, dirName, nil, false)
  4227  	defer r.Close(testCtx)
  4228  
  4229  	subDirName := "sub-dir"
  4230  	require.NoError(t, os.Mkdir(joinPath(dirPath, subDirName), 0o700))
  4231  
  4232  	mem := mod.Memory()
  4233  
  4234  	originalFileName := "top-original-file"
  4235  	destinationPath := uint32(0x77)
  4236  	destinationPathName := "top-symlinked"
  4237  	ok := mem.Write(destinationPath, []byte(destinationPathName))
  4238  	require.True(t, ok)
  4239  
  4240  	originalSubDirFileName := joinPath(subDirName, "subdir-original-file")
  4241  	destinationSubDirFileName := joinPath(subDirName, "subdir-symlinked")
  4242  	destinationSubDirPathNamePtr := uint32(0xcc)
  4243  	ok = mem.Write(destinationSubDirPathNamePtr, []byte(destinationSubDirFileName))
  4244  	require.True(t, ok)
  4245  
  4246  	// Create original file and symlink to the destination.
  4247  	originalRelativePath := joinPath(dirName, originalFileName)
  4248  	err := os.WriteFile(joinPath(tmpDir, originalRelativePath), []byte{4, 3, 2, 1}, 0o700)
  4249  	require.NoError(t, err)
  4250  	err = os.Symlink(originalRelativePath, joinPath(dirPath, destinationPathName))
  4251  	require.NoError(t, err)
  4252  	originalSubDirRelativePath := joinPath(dirName, originalSubDirFileName)
  4253  	err = os.WriteFile(joinPath(tmpDir, originalSubDirRelativePath), []byte{1, 2, 3, 4}, 0o700)
  4254  	require.NoError(t, err)
  4255  	err = os.Symlink(originalSubDirRelativePath, joinPath(dirPath, destinationSubDirFileName))
  4256  	require.NoError(t, err)
  4257  
  4258  	t.Run("ok", func(t *testing.T) {
  4259  		for _, tc := range []struct {
  4260  			name          string
  4261  			path, pathLen uint32
  4262  			expectedBuf   string
  4263  		}{
  4264  			{
  4265  				name:        "top",
  4266  				path:        destinationPath,
  4267  				pathLen:     uint32(len(destinationPathName)),
  4268  				expectedBuf: originalRelativePath,
  4269  			},
  4270  			{
  4271  				name:        "subdir",
  4272  				path:        destinationSubDirPathNamePtr,
  4273  				pathLen:     uint32(len(destinationSubDirFileName)),
  4274  				expectedBuf: originalSubDirRelativePath,
  4275  			},
  4276  		} {
  4277  			t.Run(tc.name, func(t *testing.T) {
  4278  				const buf, bufLen, resultBufused = 0x100, 0xff, 0x200
  4279  				requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.PathReadlinkName,
  4280  					uint64(dirFD), uint64(tc.path), uint64(tc.pathLen),
  4281  					buf, bufLen, resultBufused)
  4282  				require.Contains(t, log.String(), wasip1.ErrnoName(wasip1.ErrnoSuccess))
  4283  
  4284  				size, ok := mem.ReadUint32Le(resultBufused)
  4285  				require.True(t, ok)
  4286  				actual, ok := mem.Read(buf, size)
  4287  				require.True(t, ok)
  4288  				require.Equal(t, tc.expectedBuf, string(actual))
  4289  			})
  4290  		}
  4291  	})
  4292  
  4293  	t.Run("errors", func(t *testing.T) {
  4294  		for _, tc := range []struct {
  4295  			name                                      string
  4296  			fd                                        int32
  4297  			path, pathLen, buf, bufLen, resultBufused uint32
  4298  			expectedErrno                             wasip1.Errno
  4299  		}{
  4300  			{expectedErrno: wasip1.ErrnoInval},
  4301  			{expectedErrno: wasip1.ErrnoInval, pathLen: 100},
  4302  			{expectedErrno: wasip1.ErrnoInval, bufLen: 100},
  4303  			{
  4304  				name:          "bufLen too short",
  4305  				expectedErrno: wasip1.ErrnoRange,
  4306  				fd:            dirFD,
  4307  				bufLen:        10,
  4308  				path:          destinationPath,
  4309  				pathLen:       uint32(len(destinationPathName)),
  4310  				buf:           0,
  4311  			},
  4312  			{
  4313  				name:          "path past memory",
  4314  				expectedErrno: wasip1.ErrnoFault,
  4315  				bufLen:        100,
  4316  				pathLen:       100,
  4317  				buf:           50,
  4318  				path:          math.MaxUint32,
  4319  			},
  4320  			{expectedErrno: wasip1.ErrnoNotdir, bufLen: 100, pathLen: 100, buf: 50, path: 50, fd: 1},
  4321  			{expectedErrno: wasip1.ErrnoBadf, bufLen: 100, pathLen: 100, buf: 50, path: 50, fd: 1000},
  4322  			{
  4323  				expectedErrno: wasip1.ErrnoNoent,
  4324  				bufLen:        100, buf: 50,
  4325  				path: destinationPath, pathLen: uint32(len(destinationPathName)) - 1,
  4326  				fd: dirFD,
  4327  			},
  4328  		} {
  4329  			name := tc.name
  4330  			if name == "" {
  4331  				name = wasip1.ErrnoName(tc.expectedErrno)
  4332  			}
  4333  			t.Run(name, func(t *testing.T) {
  4334  				requireErrnoResult(t, tc.expectedErrno, mod, wasip1.PathReadlinkName,
  4335  					uint64(tc.fd), uint64(tc.path), uint64(tc.pathLen), uint64(tc.buf),
  4336  					uint64(tc.bufLen), uint64(tc.resultBufused))
  4337  				require.Contains(t, log.String(), wasip1.ErrnoName(tc.expectedErrno))
  4338  			})
  4339  		}
  4340  	})
  4341  }
  4342  
  4343  func Test_pathRemoveDirectory(t *testing.T) {
  4344  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  4345  	fsConfig := wazero.NewFSConfig().WithDirMount(tmpDir, "/")
  4346  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  4347  	defer r.Close(testCtx)
  4348  
  4349  	// set up the initial memory to include the path name starting at an offset.
  4350  	pathName := "wazero"
  4351  	realPath := joinPath(tmpDir, pathName)
  4352  	ok := mod.Memory().Write(0, append([]byte{'?'}, pathName...))
  4353  	require.True(t, ok)
  4354  
  4355  	// create the directory
  4356  	err := os.Mkdir(realPath, 0o700)
  4357  	require.NoError(t, err)
  4358  
  4359  	fd := sys.FdPreopen
  4360  	name := 1
  4361  	nameLen := len(pathName)
  4362  
  4363  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.PathRemoveDirectoryName, uint64(fd), uint64(name), uint64(nameLen))
  4364  	require.Equal(t, `
  4365  ==> wasi_snapshot_preview1.path_remove_directory(fd=3,path=wazero)
  4366  <== errno=ESUCCESS
  4367  `, "\n"+log.String())
  4368  
  4369  	// ensure the directory was removed
  4370  	_, err = os.Stat(realPath)
  4371  	require.Error(t, err)
  4372  }
  4373  
  4374  func Test_pathRemoveDirectory_Errors(t *testing.T) {
  4375  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  4376  	fsConfig := wazero.NewFSConfig().WithDirMount(tmpDir, "/")
  4377  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  4378  	defer r.Close(testCtx)
  4379  
  4380  	file := "file"
  4381  	err := os.WriteFile(joinPath(tmpDir, file), []byte{}, 0o700)
  4382  	require.NoError(t, err)
  4383  	fileFD := requireOpenFD(t, mod, file)
  4384  
  4385  	dirNotEmpty := "notempty"
  4386  	dirNotEmptyPath := joinPath(tmpDir, dirNotEmpty)
  4387  	err = os.Mkdir(dirNotEmptyPath, 0o700)
  4388  	require.NoError(t, err)
  4389  
  4390  	dir := "dir"
  4391  	err = os.Mkdir(joinPath(dirNotEmptyPath, dir), 0o700)
  4392  	require.NoError(t, err)
  4393  
  4394  	tests := []struct {
  4395  		name, pathName string
  4396  		fd             int32
  4397  		path, pathLen  uint32
  4398  		expectedErrno  wasip1.Errno
  4399  		expectedLog    string
  4400  	}{
  4401  		{
  4402  			name:          "unopened FD",
  4403  			fd:            42, // arbitrary invalid fd
  4404  			expectedErrno: wasip1.ErrnoBadf,
  4405  			expectedLog: `
  4406  ==> wasi_snapshot_preview1.path_remove_directory(fd=42,path=)
  4407  <== errno=EBADF
  4408  `,
  4409  		},
  4410  		{
  4411  			name:          "Fd not a directory",
  4412  			fd:            fileFD,
  4413  			pathName:      file,
  4414  			path:          0,
  4415  			pathLen:       uint32(len(file)),
  4416  			expectedErrno: wasip1.ErrnoNotdir,
  4417  			expectedLog: `
  4418  ==> wasi_snapshot_preview1.path_remove_directory(fd=4,path=file)
  4419  <== errno=ENOTDIR
  4420  `,
  4421  		},
  4422  		{
  4423  			name:          "out-of-memory reading path",
  4424  			fd:            sys.FdPreopen,
  4425  			path:          mod.Memory().Size(),
  4426  			pathLen:       1,
  4427  			expectedErrno: wasip1.ErrnoFault,
  4428  			expectedLog: `
  4429  ==> wasi_snapshot_preview1.path_remove_directory(fd=3,path=OOM(65536,1))
  4430  <== errno=EFAULT
  4431  `,
  4432  		},
  4433  		{
  4434  			name:          "out-of-memory reading pathLen",
  4435  			fd:            sys.FdPreopen,
  4436  			path:          0,
  4437  			pathLen:       mod.Memory().Size() + 1, // path is in the valid memory range, but pathLen is OOM for path
  4438  			expectedErrno: wasip1.ErrnoFault,
  4439  			expectedLog: `
  4440  ==> wasi_snapshot_preview1.path_remove_directory(fd=3,path=OOM(0,65537))
  4441  <== errno=EFAULT
  4442  `,
  4443  		},
  4444  		{
  4445  			name:          "no such file exists",
  4446  			fd:            sys.FdPreopen,
  4447  			pathName:      file,
  4448  			path:          0,
  4449  			pathLen:       uint32(len(file) - 1),
  4450  			expectedErrno: wasip1.ErrnoNoent,
  4451  			expectedLog: `
  4452  ==> wasi_snapshot_preview1.path_remove_directory(fd=3,path=fil)
  4453  <== errno=ENOENT
  4454  `,
  4455  		},
  4456  		{
  4457  			name:          "file not dir",
  4458  			fd:            sys.FdPreopen,
  4459  			pathName:      file,
  4460  			path:          0,
  4461  			pathLen:       uint32(len(file)),
  4462  			expectedErrno: wasip1.ErrnoNotdir,
  4463  			expectedLog: fmt.Sprintf(`
  4464  ==> wasi_snapshot_preview1.path_remove_directory(fd=3,path=file)
  4465  <== errno=%s
  4466  `, wasip1.ErrnoName(wasip1.ErrnoNotdir)),
  4467  		},
  4468  		{
  4469  			name:          "dir not empty",
  4470  			fd:            sys.FdPreopen,
  4471  			pathName:      dirNotEmpty,
  4472  			path:          0,
  4473  			pathLen:       uint32(len(dirNotEmpty)),
  4474  			expectedErrno: wasip1.ErrnoNotempty,
  4475  			expectedLog: `
  4476  ==> wasi_snapshot_preview1.path_remove_directory(fd=3,path=notempty)
  4477  <== errno=ENOTEMPTY
  4478  `,
  4479  		},
  4480  	}
  4481  
  4482  	for _, tt := range tests {
  4483  		tc := tt
  4484  		t.Run(tc.name, func(t *testing.T) {
  4485  			defer log.Reset()
  4486  
  4487  			mod.Memory().Write(tc.path, []byte(tc.pathName))
  4488  
  4489  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.PathRemoveDirectoryName, uint64(tc.fd), uint64(tc.path), uint64(tc.pathLen))
  4490  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  4491  		})
  4492  	}
  4493  }
  4494  
  4495  func Test_pathSymlink_errors(t *testing.T) {
  4496  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  4497  
  4498  	dirName := "dir"
  4499  	dirPath := joinPath(tmpDir, dirName)
  4500  	mod, fd, log, r := requireOpenFile(t, tmpDir, dirName, nil, false)
  4501  	defer r.Close(testCtx)
  4502  
  4503  	mem := mod.Memory()
  4504  
  4505  	fileName := "file"
  4506  	err := os.WriteFile(joinPath(dirPath, fileName), []byte{1, 2, 3, 4}, 0o700)
  4507  	require.NoError(t, err)
  4508  
  4509  	file := uint32(0xff)
  4510  	ok := mem.Write(file, []byte(fileName))
  4511  	require.True(t, ok)
  4512  
  4513  	notFoundFile := uint32(0xaa)
  4514  	notFoundFileName := "nope"
  4515  	ok = mem.Write(notFoundFile, []byte(notFoundFileName))
  4516  	require.True(t, ok)
  4517  
  4518  	link := uint32(0xcc)
  4519  	linkName := fileName + "-link"
  4520  	ok = mem.Write(link, []byte(linkName))
  4521  	require.True(t, ok)
  4522  
  4523  	t.Run("success", func(t *testing.T) {
  4524  		requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.PathSymlinkName,
  4525  			uint64(file), uint64(len(fileName)), uint64(fd), uint64(link), uint64(len(linkName)))
  4526  		require.Contains(t, log.String(), wasip1.ErrnoName(wasip1.ErrnoSuccess))
  4527  		st, err := os.Lstat(joinPath(dirPath, linkName))
  4528  		require.NoError(t, err)
  4529  		require.Equal(t, st.Mode()&os.ModeSymlink, os.ModeSymlink)
  4530  	})
  4531  
  4532  	t.Run("errors", func(t *testing.T) {
  4533  		for _, tc := range []struct {
  4534  			errno               wasip1.Errno
  4535  			oldPath, oldPathLen uint32
  4536  			fd                  int32
  4537  			newPath, newPathLen uint32
  4538  		}{
  4539  			{errno: wasip1.ErrnoBadf, fd: 1000},
  4540  			{errno: wasip1.ErrnoNotdir, fd: 2},
  4541  			// Length zero buffer is not valid.
  4542  			{errno: wasip1.ErrnoInval, fd: fd},
  4543  			{errno: wasip1.ErrnoInval, oldPathLen: 100, fd: fd},
  4544  			{errno: wasip1.ErrnoInval, newPathLen: 100, fd: fd},
  4545  			// Invalid pointer to the names.
  4546  			{errno: wasip1.ErrnoFault, oldPath: math.MaxUint32, oldPathLen: 100, newPathLen: 100, fd: fd},
  4547  			{errno: wasip1.ErrnoFault, newPath: math.MaxUint32, oldPathLen: 100, newPathLen: 100, fd: fd},
  4548  			{errno: wasip1.ErrnoFault, oldPath: math.MaxUint32, newPath: math.MaxUint32, oldPathLen: 100, newPathLen: 100, fd: fd},
  4549  			// Non-existing path as source.
  4550  			{
  4551  				errno: wasip1.ErrnoInval, oldPath: notFoundFile, oldPathLen: uint32(len(notFoundFileName)),
  4552  				newPath: 0, newPathLen: 5, fd: fd,
  4553  			},
  4554  			// Linking to existing file.
  4555  			{
  4556  				errno: wasip1.ErrnoExist, oldPath: file, oldPathLen: uint32(len(fileName)),
  4557  				newPath: file, newPathLen: uint32(len(fileName)), fd: fd,
  4558  			},
  4559  		} {
  4560  			name := wasip1.ErrnoName(tc.errno)
  4561  			t.Run(name, func(t *testing.T) {
  4562  				requireErrnoResult(t, tc.errno, mod, wasip1.PathSymlinkName,
  4563  					uint64(tc.oldPath), uint64(tc.oldPathLen), uint64(tc.fd), uint64(tc.newPath), uint64(tc.newPathLen))
  4564  				require.Contains(t, log.String(), name)
  4565  			})
  4566  		}
  4567  	})
  4568  }
  4569  
  4570  func Test_pathRename(t *testing.T) {
  4571  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  4572  	fsConfig := wazero.NewFSConfig().WithDirMount(tmpDir, "/")
  4573  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  4574  	defer r.Close(testCtx)
  4575  
  4576  	// set up the initial memory to include the old path name starting at an offset.
  4577  	oldfd := sys.FdPreopen
  4578  	oldPathName := "wazero"
  4579  	realOldPath := joinPath(tmpDir, oldPathName)
  4580  	oldPath := uint32(0)
  4581  	oldPathLen := len(oldPathName)
  4582  	ok := mod.Memory().Write(oldPath, []byte(oldPathName))
  4583  	require.True(t, ok)
  4584  
  4585  	// create the file
  4586  	err := os.WriteFile(realOldPath, []byte{}, 0o600)
  4587  	require.NoError(t, err)
  4588  
  4589  	newfd := sys.FdPreopen
  4590  	newPathName := "wahzero"
  4591  	realNewPath := joinPath(tmpDir, newPathName)
  4592  	newPath := uint32(16)
  4593  	newPathLen := len(newPathName)
  4594  	ok = mod.Memory().Write(newPath, []byte(newPathName))
  4595  	require.True(t, ok)
  4596  
  4597  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.PathRenameName,
  4598  		uint64(oldfd), uint64(oldPath), uint64(oldPathLen),
  4599  		uint64(newfd), uint64(newPath), uint64(newPathLen))
  4600  	require.Equal(t, `
  4601  ==> wasi_snapshot_preview1.path_rename(fd=3,old_path=wazero,new_fd=3,new_path=wahzero)
  4602  <== errno=ESUCCESS
  4603  `, "\n"+log.String())
  4604  
  4605  	// ensure the file was renamed
  4606  	_, err = os.Stat(realOldPath)
  4607  	require.Error(t, err)
  4608  	_, err = os.Stat(realNewPath)
  4609  	require.NoError(t, err)
  4610  }
  4611  
  4612  func Test_pathRename_Errors(t *testing.T) {
  4613  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  4614  	fsConfig := wazero.NewFSConfig().WithDirMount(tmpDir, "/")
  4615  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  4616  	defer r.Close(testCtx)
  4617  
  4618  	file := "file"
  4619  	err := os.WriteFile(joinPath(tmpDir, file), []byte{}, 0o700)
  4620  	require.NoError(t, err)
  4621  
  4622  	// We have to test FD validation with a path not under test. Otherwise,
  4623  	// Windows may fail for the wrong reason, like:
  4624  	//	The process cannot access the file because it is being used by another process.
  4625  	file1 := "file1"
  4626  	err = os.WriteFile(joinPath(tmpDir, file1), []byte{}, 0o700)
  4627  	require.NoError(t, err)
  4628  	fileFD := requireOpenFD(t, mod, file1)
  4629  
  4630  	dirNotEmpty := "notempty"
  4631  	err = os.Mkdir(joinPath(tmpDir, dirNotEmpty), 0o700)
  4632  	require.NoError(t, err)
  4633  
  4634  	dir := joinPath(dirNotEmpty, "dir")
  4635  	err = os.Mkdir(joinPath(tmpDir, dir), 0o700)
  4636  	require.NoError(t, err)
  4637  
  4638  	tests := []struct {
  4639  		name, oldPathName, newPathName string
  4640  		oldFd                          int32
  4641  		oldPath, oldPathLen            uint32
  4642  		newFd                          int32
  4643  		newPath, newPathLen            uint32
  4644  		expectedErrno                  wasip1.Errno
  4645  		expectedLog                    string
  4646  	}{
  4647  		{
  4648  			name:          "unopened old FD",
  4649  			oldFd:         42, // arbitrary invalid fd
  4650  			newFd:         sys.FdPreopen,
  4651  			expectedErrno: wasip1.ErrnoBadf,
  4652  			expectedLog: `
  4653  ==> wasi_snapshot_preview1.path_rename(fd=42,old_path=,new_fd=3,new_path=)
  4654  <== errno=EBADF
  4655  `,
  4656  		},
  4657  		{
  4658  			name:          "old FD not a directory",
  4659  			oldFd:         fileFD,
  4660  			newFd:         sys.FdPreopen,
  4661  			expectedErrno: wasip1.ErrnoNotdir,
  4662  			expectedLog: `
  4663  ==> wasi_snapshot_preview1.path_rename(fd=4,old_path=,new_fd=3,new_path=)
  4664  <== errno=ENOTDIR
  4665  `,
  4666  		},
  4667  		{
  4668  			name:          "unopened new FD",
  4669  			oldFd:         sys.FdPreopen,
  4670  			newFd:         42, // arbitrary invalid fd
  4671  			expectedErrno: wasip1.ErrnoBadf,
  4672  			expectedLog: `
  4673  ==> wasi_snapshot_preview1.path_rename(fd=3,old_path=,new_fd=42,new_path=)
  4674  <== errno=EBADF
  4675  `,
  4676  		},
  4677  		{
  4678  			name:          "new FD not a directory",
  4679  			oldFd:         sys.FdPreopen,
  4680  			newFd:         fileFD,
  4681  			expectedErrno: wasip1.ErrnoNotdir,
  4682  			expectedLog: `
  4683  ==> wasi_snapshot_preview1.path_rename(fd=3,old_path=,new_fd=4,new_path=)
  4684  <== errno=ENOTDIR
  4685  `,
  4686  		},
  4687  		{
  4688  			name:          "out-of-memory reading old path",
  4689  			oldFd:         sys.FdPreopen,
  4690  			newFd:         sys.FdPreopen,
  4691  			oldPath:       mod.Memory().Size(),
  4692  			oldPathLen:    1,
  4693  			expectedErrno: wasip1.ErrnoFault,
  4694  			expectedLog: `
  4695  ==> wasi_snapshot_preview1.path_rename(fd=3,old_path=OOM(65536,1),new_fd=3,new_path=)
  4696  <== errno=EFAULT
  4697  `,
  4698  		},
  4699  		{
  4700  			name:          "out-of-memory reading new path",
  4701  			oldFd:         sys.FdPreopen,
  4702  			newFd:         sys.FdPreopen,
  4703  			oldPath:       0,
  4704  			oldPathName:   "a",
  4705  			oldPathLen:    1,
  4706  			newPath:       mod.Memory().Size(),
  4707  			newPathLen:    1,
  4708  			expectedErrno: wasip1.ErrnoFault,
  4709  			expectedLog: `
  4710  ==> wasi_snapshot_preview1.path_rename(fd=3,old_path=a,new_fd=3,new_path=OOM(65536,1))
  4711  <== errno=EFAULT
  4712  `,
  4713  		},
  4714  		{
  4715  			name:          "out-of-memory reading old pathLen",
  4716  			oldFd:         sys.FdPreopen,
  4717  			newFd:         sys.FdPreopen,
  4718  			oldPath:       0,
  4719  			oldPathLen:    mod.Memory().Size() + 1, // path is in the valid memory range, but pathLen is OOM for path
  4720  			expectedErrno: wasip1.ErrnoFault,
  4721  			expectedLog: `
  4722  ==> wasi_snapshot_preview1.path_rename(fd=3,old_path=OOM(0,65537),new_fd=3,new_path=)
  4723  <== errno=EFAULT
  4724  `,
  4725  		},
  4726  		{
  4727  			name:          "out-of-memory reading new pathLen",
  4728  			oldFd:         sys.FdPreopen,
  4729  			newFd:         sys.FdPreopen,
  4730  			oldPathName:   file,
  4731  			oldPathLen:    uint32(len(file)),
  4732  			newPath:       0,
  4733  			newPathLen:    mod.Memory().Size() + 1, // path is in the valid memory range, but pathLen is OOM for path
  4734  			expectedErrno: wasip1.ErrnoFault,
  4735  			expectedLog: `
  4736  ==> wasi_snapshot_preview1.path_rename(fd=3,old_path=file,new_fd=3,new_path=OOM(0,65537))
  4737  <== errno=EFAULT
  4738  `,
  4739  		},
  4740  		{
  4741  			name:          "no such file exists",
  4742  			oldFd:         sys.FdPreopen,
  4743  			newFd:         sys.FdPreopen,
  4744  			oldPathName:   file,
  4745  			oldPathLen:    uint32(len(file)) - 1,
  4746  			newPath:       16,
  4747  			newPathName:   file,
  4748  			newPathLen:    uint32(len(file)),
  4749  			expectedErrno: wasip1.ErrnoNoent,
  4750  			expectedLog: `
  4751  ==> wasi_snapshot_preview1.path_rename(fd=3,old_path=fil,new_fd=3,new_path=file)
  4752  <== errno=ENOENT
  4753  `,
  4754  		},
  4755  		{
  4756  			name:          "dir not file",
  4757  			oldFd:         sys.FdPreopen,
  4758  			newFd:         sys.FdPreopen,
  4759  			oldPathName:   file,
  4760  			oldPathLen:    uint32(len(file)),
  4761  			newPath:       16,
  4762  			newPathName:   dir,
  4763  			newPathLen:    uint32(len(dir)),
  4764  			expectedErrno: wasip1.ErrnoIsdir,
  4765  			expectedLog: `
  4766  ==> wasi_snapshot_preview1.path_rename(fd=3,old_path=file,new_fd=3,new_path=notempty/dir)
  4767  <== errno=EISDIR
  4768  `,
  4769  		},
  4770  	}
  4771  
  4772  	for _, tt := range tests {
  4773  		tc := tt
  4774  		t.Run(tc.name, func(t *testing.T) {
  4775  			defer log.Reset()
  4776  
  4777  			mod.Memory().Write(tc.oldPath, []byte(tc.oldPathName))
  4778  			mod.Memory().Write(tc.newPath, []byte(tc.newPathName))
  4779  
  4780  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.PathRenameName,
  4781  				uint64(tc.oldFd), uint64(tc.oldPath), uint64(tc.oldPathLen),
  4782  				uint64(tc.newFd), uint64(tc.newPath), uint64(tc.newPathLen))
  4783  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  4784  		})
  4785  	}
  4786  }
  4787  
  4788  func Test_pathUnlinkFile(t *testing.T) {
  4789  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  4790  	fsConfig := wazero.NewFSConfig().WithDirMount(tmpDir, "/")
  4791  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  4792  	defer r.Close(testCtx)
  4793  
  4794  	// set up the initial memory to include the path name starting at an offset.
  4795  	pathName := "wazero"
  4796  	realPath := joinPath(tmpDir, pathName)
  4797  	ok := mod.Memory().Write(0, append([]byte{'?'}, pathName...))
  4798  	require.True(t, ok)
  4799  
  4800  	// create the file
  4801  	err := os.WriteFile(realPath, []byte{}, 0o600)
  4802  	require.NoError(t, err)
  4803  
  4804  	fd := sys.FdPreopen
  4805  	name := 1
  4806  	nameLen := len(pathName)
  4807  
  4808  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.PathUnlinkFileName, uint64(fd), uint64(name), uint64(nameLen))
  4809  	require.Equal(t, `
  4810  ==> wasi_snapshot_preview1.path_unlink_file(fd=3,path=wazero)
  4811  <== errno=ESUCCESS
  4812  `, "\n"+log.String())
  4813  
  4814  	// ensure the file was removed
  4815  	_, err = os.Stat(realPath)
  4816  	require.Error(t, err)
  4817  }
  4818  
  4819  func Test_pathUnlinkFile_Errors(t *testing.T) {
  4820  	tmpDir := t.TempDir() // open before loop to ensure no locking problems.
  4821  	fsConfig := wazero.NewFSConfig().WithDirMount(tmpDir, "/")
  4822  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  4823  	defer r.Close(testCtx)
  4824  
  4825  	file := "file"
  4826  	err := os.WriteFile(joinPath(tmpDir, file), []byte{}, 0o700)
  4827  	require.NoError(t, err)
  4828  	fileFD := requireOpenFD(t, mod, file)
  4829  
  4830  	dir := "dir"
  4831  	err = os.Mkdir(joinPath(tmpDir, dir), 0o700)
  4832  	require.NoError(t, err)
  4833  
  4834  	tests := []struct {
  4835  		name, pathName string
  4836  		fd             int32
  4837  		path, pathLen  uint32
  4838  		expectedErrno  wasip1.Errno
  4839  		expectedLog    string
  4840  	}{
  4841  		{
  4842  			name:          "unopened FD",
  4843  			fd:            42, // arbitrary invalid fd
  4844  			expectedErrno: wasip1.ErrnoBadf,
  4845  			expectedLog: `
  4846  ==> wasi_snapshot_preview1.path_unlink_file(fd=42,path=)
  4847  <== errno=EBADF
  4848  `,
  4849  		},
  4850  		{
  4851  			name:          "Fd not a directory",
  4852  			fd:            fileFD,
  4853  			expectedErrno: wasip1.ErrnoNotdir,
  4854  			expectedLog: `
  4855  ==> wasi_snapshot_preview1.path_unlink_file(fd=4,path=)
  4856  <== errno=ENOTDIR
  4857  `,
  4858  		},
  4859  		{
  4860  			name:          "out-of-memory reading path",
  4861  			fd:            sys.FdPreopen,
  4862  			path:          mod.Memory().Size(),
  4863  			pathLen:       1,
  4864  			expectedErrno: wasip1.ErrnoFault,
  4865  			expectedLog: `
  4866  ==> wasi_snapshot_preview1.path_unlink_file(fd=3,path=OOM(65536,1))
  4867  <== errno=EFAULT
  4868  `,
  4869  		},
  4870  		{
  4871  			name:          "out-of-memory reading pathLen",
  4872  			fd:            sys.FdPreopen,
  4873  			path:          0,
  4874  			pathLen:       mod.Memory().Size() + 1, // path is in the valid memory range, but pathLen is OOM for path
  4875  			expectedErrno: wasip1.ErrnoFault,
  4876  			expectedLog: `
  4877  ==> wasi_snapshot_preview1.path_unlink_file(fd=3,path=OOM(0,65537))
  4878  <== errno=EFAULT
  4879  `,
  4880  		},
  4881  		{
  4882  			name:          "no such file exists",
  4883  			fd:            sys.FdPreopen,
  4884  			pathName:      file,
  4885  			path:          0,
  4886  			pathLen:       uint32(len(file) - 1),
  4887  			expectedErrno: wasip1.ErrnoNoent,
  4888  			expectedLog: `
  4889  ==> wasi_snapshot_preview1.path_unlink_file(fd=3,path=fil)
  4890  <== errno=ENOENT
  4891  `,
  4892  		},
  4893  		{
  4894  			name:          "dir not file",
  4895  			fd:            sys.FdPreopen,
  4896  			pathName:      dir,
  4897  			path:          0,
  4898  			pathLen:       uint32(len(dir)),
  4899  			expectedErrno: wasip1.ErrnoIsdir,
  4900  			expectedLog: `
  4901  ==> wasi_snapshot_preview1.path_unlink_file(fd=3,path=dir)
  4902  <== errno=EISDIR
  4903  `,
  4904  		},
  4905  	}
  4906  
  4907  	for _, tt := range tests {
  4908  		tc := tt
  4909  		t.Run(tc.name, func(t *testing.T) {
  4910  			defer log.Reset()
  4911  
  4912  			mod.Memory().Write(tc.path, []byte(tc.pathName))
  4913  
  4914  			requireErrnoResult(t, tc.expectedErrno, mod, wasip1.PathUnlinkFileName, uint64(tc.fd), uint64(tc.path), uint64(tc.pathLen))
  4915  			require.Equal(t, tc.expectedLog, "\n"+log.String())
  4916  		})
  4917  	}
  4918  }
  4919  
  4920  func requireOpenFile(t *testing.T, tmpDir string, pathName string, data []byte, readOnly bool) (api.Module, int32, *bytes.Buffer, api.Closer) {
  4921  	oflags := experimentalsys.O_RDWR
  4922  	if readOnly {
  4923  		oflags = experimentalsys.O_RDONLY
  4924  	}
  4925  
  4926  	realPath := joinPath(tmpDir, pathName)
  4927  	if data == nil {
  4928  		oflags = experimentalsys.O_RDONLY
  4929  		require.NoError(t, os.Mkdir(realPath, 0o700))
  4930  	} else {
  4931  		require.NoError(t, os.WriteFile(realPath, data, 0o600))
  4932  	}
  4933  
  4934  	fsConfig := wazero.NewFSConfig()
  4935  
  4936  	if readOnly {
  4937  		fsConfig = fsConfig.WithReadOnlyDirMount(tmpDir, "/")
  4938  	} else {
  4939  		fsConfig = fsConfig.WithDirMount(tmpDir, "/")
  4940  	}
  4941  
  4942  	mod, r, log := requireProxyModule(t, wazero.NewModuleConfig().WithFSConfig(fsConfig))
  4943  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  4944  	preopen := fsc.RootFS()
  4945  
  4946  	fd, errno := fsc.OpenFile(preopen, pathName, oflags, 0)
  4947  	require.EqualErrno(t, 0, errno)
  4948  
  4949  	return mod, fd, log, r
  4950  }
  4951  
  4952  // Test_fdReaddir_dotEntryHasARealInode because wasi-testsuite requires it.
  4953  func Test_fdReaddir_dotEntryHasARealInode(t *testing.T) {
  4954  	if runtime.GOOS == "windows" && !platform.IsAtLeastGo120 {
  4955  		t.Skip("windows before go 1.20 has trouble reading the inode information on directories.")
  4956  	}
  4957  
  4958  	root := t.TempDir()
  4959  	mod, r, _ := requireProxyModule(t, wazero.NewModuleConfig().
  4960  		WithFSConfig(wazero.NewFSConfig().WithDirMount(root, "/")),
  4961  	)
  4962  	defer r.Close(testCtx)
  4963  
  4964  	mem := mod.Memory()
  4965  
  4966  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  4967  	preopen := fsc.RootFS()
  4968  
  4969  	readDirTarget := "dir"
  4970  	mem.Write(0, []byte(readDirTarget))
  4971  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.PathCreateDirectoryName,
  4972  		uint64(sys.FdPreopen), uint64(0), uint64(len(readDirTarget)))
  4973  
  4974  	// Open the directory, before writing files!
  4975  	fd, errno := fsc.OpenFile(preopen, readDirTarget, experimentalsys.O_RDONLY, 0)
  4976  	require.EqualErrno(t, 0, errno)
  4977  
  4978  	// get the real inode of the current directory
  4979  	st, errno := preopen.Stat(readDirTarget)
  4980  	require.EqualErrno(t, 0, errno)
  4981  	dirents := []byte{1, 0, 0, 0, 0, 0, 0, 0}         // d_next = 1
  4982  	dirents = append(dirents, u64.LeBytes(st.Ino)...) // d_ino
  4983  	dirents = append(dirents, 1, 0, 0, 0)             // d_namlen = 1 character
  4984  	dirents = append(dirents, 3, 0, 0, 0)             // d_type = directory
  4985  	dirents = append(dirents, '.')                    // name
  4986  
  4987  	require.EqualErrno(t, 0, errno)
  4988  	dirents = append(dirents, 2, 0, 0, 0, 0, 0, 0, 0) // d_next = 2
  4989  	// See /RATIONALE.md for why we don't attempt to get an inode for ".."
  4990  	dirents = append(dirents, 0, 0, 0, 0, 0, 0, 0, 0) // d_ino
  4991  	dirents = append(dirents, 2, 0, 0, 0)             // d_namlen = 2 characters
  4992  	dirents = append(dirents, 3, 0, 0, 0)             // d_type = directory
  4993  	dirents = append(dirents, '.', '.')               // name
  4994  
  4995  	// Try to list them!
  4996  	resultBufused := uint32(0) // where to write the amount used out of bufLen
  4997  	buf := uint32(8)           // where to start the dirents
  4998  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdReaddirName,
  4999  		uint64(fd), uint64(buf), uint64(0x2000), 0, uint64(resultBufused))
  5000  
  5001  	used, _ := mem.ReadUint32Le(resultBufused)
  5002  
  5003  	results, _ := mem.Read(buf, used)
  5004  	require.Equal(t, dirents, results)
  5005  }
  5006  
  5007  // Test_fdReaddir_opened_file_written ensures that writing files to the already-opened directory
  5008  // is visible. This is significant on Windows.
  5009  // https://github.com/ziglang/zig/blob/2ccff5115454bab4898bae3de88f5619310bc5c1/lib/std/fs/test.zig#L156-L184
  5010  func Test_fdReaddir_opened_file_written(t *testing.T) {
  5011  	if runtime.GOOS == "windows" && !platform.IsAtLeastGo120 {
  5012  		t.Skip("windows before go 1.20 has trouble reading the inode information on directories.")
  5013  	}
  5014  
  5015  	tmpDir := t.TempDir()
  5016  	mod, r, _ := requireProxyModule(t, wazero.NewModuleConfig().
  5017  		WithFSConfig(wazero.NewFSConfig().WithDirMount(tmpDir, "/")),
  5018  	)
  5019  	defer r.Close(testCtx)
  5020  
  5021  	mem := mod.Memory()
  5022  
  5023  	fsc := mod.(*wasm.ModuleInstance).Sys.FS()
  5024  	preopen := fsc.RootFS()
  5025  
  5026  	dirName := "dir"
  5027  	dirPath := joinPath(tmpDir, dirName)
  5028  	mem.Write(0, []byte(dirName))
  5029  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.PathCreateDirectoryName,
  5030  		uint64(sys.FdPreopen), uint64(0), uint64(len(dirName)))
  5031  
  5032  	// Open the directory, before writing files!
  5033  	dirFD, errno := fsc.OpenFile(preopen, dirName, experimentalsys.O_RDONLY, 0)
  5034  	require.EqualErrno(t, 0, errno)
  5035  
  5036  	// Then write a file to the directory.
  5037  	f := openFile(t, joinPath(dirPath, "file"), experimentalsys.O_CREAT, 0)
  5038  	defer f.Close()
  5039  
  5040  	// get the real inode of the current directory
  5041  	st, errno := preopen.Stat(dirName)
  5042  	require.EqualErrno(t, 0, errno)
  5043  	dirents := []byte{1, 0, 0, 0, 0, 0, 0, 0}         // d_next = 1
  5044  	dirents = append(dirents, u64.LeBytes(st.Ino)...) // d_ino
  5045  	dirents = append(dirents, 1, 0, 0, 0)             // d_namlen = 1 character
  5046  	dirents = append(dirents, 3, 0, 0, 0)             // d_type = directory
  5047  	dirents = append(dirents, '.')                    // name
  5048  
  5049  	// get the real inode of the parent directory
  5050  	st, errno = preopen.Stat(".")
  5051  	require.EqualErrno(t, 0, errno)
  5052  	dirents = append(dirents, 2, 0, 0, 0, 0, 0, 0, 0) // d_next = 2
  5053  	// See /RATIONALE.md for why we don't attempt to get an inode for ".."
  5054  	dirents = append(dirents, 0, 0, 0, 0, 0, 0, 0, 0) // d_ino
  5055  	dirents = append(dirents, 2, 0, 0, 0)             // d_namlen = 2 characters
  5056  	dirents = append(dirents, 3, 0, 0, 0)             // d_type = directory
  5057  	dirents = append(dirents, '.', '.')               // name
  5058  
  5059  	// get the real inode of the file
  5060  	st, errno = f.Stat()
  5061  	require.EqualErrno(t, 0, errno)
  5062  	dirents = append(dirents, 3, 0, 0, 0, 0, 0, 0, 0) // d_next = 3
  5063  	dirents = append(dirents, u64.LeBytes(st.Ino)...) // d_ino
  5064  	dirents = append(dirents, 4, 0, 0, 0)             // d_namlen = 4 characters
  5065  	dirents = append(dirents, 4, 0, 0, 0)             // d_type = regular_file
  5066  	dirents = append(dirents, 'f', 'i', 'l', 'e')     // name
  5067  
  5068  	// Try to list them!
  5069  	resultBufused := uint32(0) // where to write the amount used out of bufLen
  5070  	buf := uint32(8)           // where to start the dirents
  5071  	requireErrnoResult(t, wasip1.ErrnoSuccess, mod, wasip1.FdReaddirName,
  5072  		uint64(dirFD), uint64(buf), uint64(0x2000), 0, uint64(resultBufused))
  5073  
  5074  	used, _ := mem.ReadUint32Le(resultBufused)
  5075  
  5076  	results, _ := mem.Read(buf, used)
  5077  	require.Equal(t, dirents, results)
  5078  }
  5079  
  5080  // joinPath avoids us having to rename fields just to avoid conflict with the
  5081  // path package.
  5082  func joinPath(dirName, baseName string) string {
  5083  	return path.Join(dirName, baseName)
  5084  }
  5085  
  5086  func openFile(t *testing.T, path string, flag experimentalsys.Oflag, perm fs.FileMode) experimentalsys.File {
  5087  	f, errno := sysfs.OpenOSFile(path, flag, perm)
  5088  	require.EqualErrno(t, 0, errno)
  5089  	return f
  5090  }