Files
wazero/internal/platform/wrap_windows.go
Crypt Keeper e5dc733df7 Adds Path to platform.File and refactors tests (#1431)
Signed-off-by: Adrian Cole <adrian@tetrate.io>
Signed-off-by: Edoardo Vacchi <evacchi@users.noreply.github.com>
Co-authored-by: Edoardo Vacchi <evacchi@users.noreply.github.com>
2023-05-04 07:05:40 +08:00

140 lines
3.4 KiB
Go

package platform
import (
"errors"
"io/fs"
"syscall"
)
// windowsWrappedFile deals with errno portability issues in Windows. This code
// is likely to change as we complete WASI and GOOS=js.
//
// If we don't map to syscall.Errno, wasm will crash in odd way attempting the
// same. This approach is an alternative to making our own fs.File public type.
// We aren't doing that yet, as mapping problems are generally contained to
// Windows. Hence, file is intentionally not exported.
//
// Note: Don't test for this type as it is wrapped when using sysfs.NewReadFS.
type windowsWrappedFile struct {
WriteFile
path string
flag int
perm fs.FileMode
dirInitialized bool
fileType *fs.FileMode
// closed is true when closed was called. This ensures proper syscall.EBADF
// TODO: extract a base wrapper type to cover all cases.
closed bool
}
// Path implements PathFile
func (w *windowsWrappedFile) Path() string {
return w.path
}
// Readdir implements readdirFile.
func (w *windowsWrappedFile) Readdir(n int) (fis []fs.FileInfo, err error) {
if err = w.requireFile("Readdir", false, true); err != nil {
return
} else if err = w.maybeInitDir(); err != nil {
return
}
return w.WriteFile.Readdir(n)
}
// ReadDir implements fs.ReadDirFile.
func (w *windowsWrappedFile) ReadDir(n int) (dirents []fs.DirEntry, err error) {
if err = w.requireFile("ReadDir", false, true); err != nil {
return
} else if err = w.maybeInitDir(); err != nil {
return
}
return w.WriteFile.ReadDir(n)
}
// Write implements io.Writer
func (w *windowsWrappedFile) Write(p []byte) (n int, err error) {
if err = w.requireFile("Write", false, false); err != nil {
return
}
n, err = w.WriteFile.Write(p)
// ERROR_ACCESS_DENIED is often returned instead of EBADF
// when a file is used after close.
if errors.Is(err, ERROR_ACCESS_DENIED) {
err = syscall.EBADF
}
return
}
// Close implements io.Closer
func (w *windowsWrappedFile) Close() (err error) {
if w.closed {
return
}
if err = w.WriteFile.Close(); err != nil {
w.closed = true
}
return
}
func (w *windowsWrappedFile) maybeInitDir() error {
if w.dirInitialized {
return nil
}
// On Windows, once the directory is opened, changes to the directory are
// not visible on ReadDir on that already-opened file handle.
//
// To provide consistent behavior with other platforms, we re-open it.
if err := w.WriteFile.Close(); err != nil {
return err
}
newW, errno := openFile(w.path, w.flag, w.perm)
if errno != 0 {
return &fs.PathError{Op: "OpenFile", Path: w.path, Err: errno}
}
w.WriteFile = newW
w.dirInitialized = true
return nil
}
// requireFile is used to making syscalls which will fail.
func (w *windowsWrappedFile) requireFile(op string, readOnly, isDir bool) error {
var ft fs.FileMode
var err error
if w.closed {
err = syscall.EBADF
} else if readOnly && w.flag&syscall.O_RDONLY == 0 {
err = syscall.EBADF
} else if ft, err = w.getFileType(); err == nil && isDir != ft.IsDir() {
if isDir {
err = syscall.ENOTDIR
} else {
err = syscall.EBADF
}
} else if err == nil {
return nil
}
return &fs.PathError{Op: op, Path: w.path, Err: err}
}
// getFileType caches the file type as this cannot change on an open file.
func (w *windowsWrappedFile) getFileType() (fs.FileMode, error) {
if w.fileType == nil {
st, errno := statFile(w.WriteFile)
if errno != 0 {
return 0, nil
}
ft := st.Mode & fs.ModeType
w.fileType = &ft
return ft, nil
}
return *w.fileType, nil
}