2023-07-19 18:33:50 -04:00
|
|
|
//go:build windows
|
2017-04-17 18:08:24 -04:00
|
|
|
// +build windows
|
|
|
|
|
|
|
|
package winio
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
2017-05-09 19:01:55 -04:00
|
|
|
"runtime"
|
2017-04-17 18:08:24 -04:00
|
|
|
"unsafe"
|
|
|
|
|
2022-03-03 07:20:02 -05:00
|
|
|
"golang.org/x/sys/windows"
|
2017-04-17 18:08:24 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
// FileBasicInfo contains file access time and file attributes information.
|
|
|
|
type FileBasicInfo struct {
|
2022-03-03 07:20:02 -05:00
|
|
|
CreationTime, LastAccessTime, LastWriteTime, ChangeTime windows.Filetime
|
2018-09-05 14:54:38 -04:00
|
|
|
FileAttributes uint32
|
2023-07-19 18:33:50 -04:00
|
|
|
_ uint32 // padding
|
2017-04-17 18:08:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetFileBasicInfo retrieves times and attributes for a file.
|
|
|
|
func GetFileBasicInfo(f *os.File) (*FileBasicInfo, error) {
|
|
|
|
bi := &FileBasicInfo{}
|
2023-07-19 18:33:50 -04:00
|
|
|
if err := windows.GetFileInformationByHandleEx(
|
|
|
|
windows.Handle(f.Fd()),
|
|
|
|
windows.FileBasicInfo,
|
|
|
|
(*byte)(unsafe.Pointer(bi)),
|
|
|
|
uint32(unsafe.Sizeof(*bi)),
|
|
|
|
); err != nil {
|
2017-04-17 18:08:24 -04:00
|
|
|
return nil, &os.PathError{Op: "GetFileInformationByHandleEx", Path: f.Name(), Err: err}
|
|
|
|
}
|
2017-05-09 19:01:55 -04:00
|
|
|
runtime.KeepAlive(f)
|
2017-04-17 18:08:24 -04:00
|
|
|
return bi, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetFileBasicInfo sets times and attributes for a file.
|
|
|
|
func SetFileBasicInfo(f *os.File, bi *FileBasicInfo) error {
|
2023-07-19 18:33:50 -04:00
|
|
|
if err := windows.SetFileInformationByHandle(
|
|
|
|
windows.Handle(f.Fd()),
|
|
|
|
windows.FileBasicInfo,
|
|
|
|
(*byte)(unsafe.Pointer(bi)),
|
|
|
|
uint32(unsafe.Sizeof(*bi)),
|
|
|
|
); err != nil {
|
2017-04-17 18:08:24 -04:00
|
|
|
return &os.PathError{Op: "SetFileInformationByHandle", Path: f.Name(), Err: err}
|
|
|
|
}
|
2017-05-09 19:01:55 -04:00
|
|
|
runtime.KeepAlive(f)
|
2017-04-17 18:08:24 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-03-03 07:20:02 -05:00
|
|
|
// FileStandardInfo contains extended information for the file.
|
|
|
|
// FILE_STANDARD_INFO in WinBase.h
|
|
|
|
// https://docs.microsoft.com/en-us/windows/win32/api/winbase/ns-winbase-file_standard_info
|
|
|
|
type FileStandardInfo struct {
|
|
|
|
AllocationSize, EndOfFile int64
|
|
|
|
NumberOfLinks uint32
|
|
|
|
DeletePending, Directory bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetFileStandardInfo retrieves ended information for the file.
|
|
|
|
func GetFileStandardInfo(f *os.File) (*FileStandardInfo, error) {
|
|
|
|
si := &FileStandardInfo{}
|
2023-07-19 18:33:50 -04:00
|
|
|
if err := windows.GetFileInformationByHandleEx(windows.Handle(f.Fd()),
|
|
|
|
windows.FileStandardInfo,
|
|
|
|
(*byte)(unsafe.Pointer(si)),
|
|
|
|
uint32(unsafe.Sizeof(*si))); err != nil {
|
2022-03-03 07:20:02 -05:00
|
|
|
return nil, &os.PathError{Op: "GetFileInformationByHandleEx", Path: f.Name(), Err: err}
|
|
|
|
}
|
|
|
|
runtime.KeepAlive(f)
|
|
|
|
return si, nil
|
|
|
|
}
|
|
|
|
|
2017-04-17 18:08:24 -04:00
|
|
|
// FileIDInfo contains the volume serial number and file ID for a file. This pair should be
|
|
|
|
// unique on a system.
|
|
|
|
type FileIDInfo struct {
|
|
|
|
VolumeSerialNumber uint64
|
|
|
|
FileID [16]byte
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetFileID retrieves the unique (volume, file ID) pair for a file.
|
|
|
|
func GetFileID(f *os.File) (*FileIDInfo, error) {
|
|
|
|
fileID := &FileIDInfo{}
|
2023-07-19 18:33:50 -04:00
|
|
|
if err := windows.GetFileInformationByHandleEx(
|
|
|
|
windows.Handle(f.Fd()),
|
|
|
|
windows.FileIdInfo,
|
|
|
|
(*byte)(unsafe.Pointer(fileID)),
|
|
|
|
uint32(unsafe.Sizeof(*fileID)),
|
|
|
|
); err != nil {
|
2017-04-17 18:08:24 -04:00
|
|
|
return nil, &os.PathError{Op: "GetFileInformationByHandleEx", Path: f.Name(), Err: err}
|
|
|
|
}
|
2017-05-09 19:01:55 -04:00
|
|
|
runtime.KeepAlive(f)
|
2017-04-17 18:08:24 -04:00
|
|
|
return fileID, nil
|
|
|
|
}
|