From c32032ac1b86c5db29a94a5e35ac9f50bf5b4457 Mon Sep 17 00:00:00 2001 From: Nikolay Date: Fri, 13 Sep 2024 12:22:25 +0200 Subject: [PATCH] lib/fs: properly call windows APIs (#6998) Previously we manually imported system windows DDLs and made direct syscall. But golang exposes syscall wrappers with sys/windows package. It seems, that direct syscall was broken at 1.23 golang release. It was `GetDiskFreeSpace` syscall in our case. This commit replaces all manual syscalls with wrappers Related issue: https://github.com/VictoriaMetrics/VictoriaMetrics/issues/6973 Related golang issue: https://github.com/golang/go/issues/69029 Signed-off-by: f41gh7 --- docs/changelog/CHANGELOG.md | 3 ++- lib/fs/fs_windows.go | 37 +++++++++---------------------------- 2 files changed, 11 insertions(+), 29 deletions(-) diff --git a/docs/changelog/CHANGELOG.md b/docs/changelog/CHANGELOG.md index 7d4a86f6f3..6e977856b6 100644 --- a/docs/changelog/CHANGELOG.md +++ b/docs/changelog/CHANGELOG.md @@ -31,7 +31,8 @@ See also [LTS releases](https://docs.victoriametrics.com/lts-releases/). * BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent/) fix service discovery of Azure Virtual Machines for response contains `nextLink` in `Host:Port` format. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/6912). * BUGFIX: [vmagent dashboard](https://github.com/VictoriaMetrics/VictoriaMetrics/blob/master/dashboards/vmagent.json): fix legend captions for stream aggregation related panels. Before they were displaying wrong label names. -* BUGFIX: [vmgateway](https://docs.victoriametrics.com/vmgateway/): add missing `datadog`, `newrelic`, `opentelemetry` and `pushgateway` routes to the `JWT` authorization routes. Allows prefixed (`prometheus/graphite`) routes for query requests. +* BUGFIX: [vmgateway](https://docs.victoriametrics.com/vmgateway/): add missing `datadog`, `newrelic`, `opentelemetry` and `pushgateway` routes to the `JWT` authorization routes. Allows prefixed (`promtheus/graphite`) routes for query requests. +* BUGFIX: [Single-node VictoriaMetrics](https://docs.victoriametrics.com/) and `vmstorage` in [VictoriaMetrics cluster](https://docs.victoriametrics.com/cluster-victoriametrics/): Fixes start-up crash on Windows OS. See this [issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/6973) for details. * BUGFIX: [Single-node VictoriaMetrics](https://docs.victoriametrics.com/) and `vmstorage` in [VictoriaMetrics cluster](https://docs.victoriametrics.com/cluster-victoriametrics/): fix metric `vm_object_references{type="indexdb"}`. Previously, it was overcounted. * BUGFIX: [Single-node VictoriaMetrics](https://docs.victoriametrics.com/) and `vmstorage` in [VictoriaMetrics cluster](https://docs.victoriametrics.com/cluster-victoriametrics/): properly ingest stale NaN samples. Previously it could be dropped if series didn't exist at storage node. See this issue [https://github.com/VictoriaMetrics/VictoriaMetrics/issues/5069] for details. diff --git a/lib/fs/fs_windows.go b/lib/fs/fs_windows.go index 1c2c280972..315c822ba8 100644 --- a/lib/fs/fs_windows.go +++ b/lib/fs/fs_windows.go @@ -10,13 +10,6 @@ import ( "golang.org/x/sys/windows" ) -var ( - kernelDLL = windows.MustLoadDLL("kernel32.dll") - procLock = kernelDLL.MustFindProc("LockFileEx") - procEvent = kernelDLL.MustFindProc("CreateEventW") - procDisk = kernelDLL.MustFindProc("GetDiskFreeSpaceExW") -) - // at windows only files could be synced // Sync for directories is not supported. func mustSyncPath(path string) { @@ -61,8 +54,8 @@ func createFlockFile(flockFile string) (*os.File, error) { return nil, fmt.Errorf("cannot create Overlapped handler: %w", err) } // https://docs.microsoft.com/en-us/windows/win32/api/fileapi/nf-fileapi-lockfileex - r1, _, err := procLock.Call(uintptr(handle), uintptr(lockfileExclusiveLock), uintptr(0), uintptr(1), uintptr(0), uintptr(unsafe.Pointer(ol))) - if r1 == 0 { + err = windows.LockFileEx(handle, lockfileExclusiveLock, 0, 0, 0, ol) + if err != nil { return nil, err } return os.NewFile(uintptr(handle), flockFile), nil @@ -118,13 +111,13 @@ func mUnmap(data []byte) error { } func mustGetFreeSpace(path string) uint64 { - var freeBytes int64 - r, _, err := procDisk.Call(uintptr(unsafe.Pointer(windows.StringToUTF16Ptr(path))), - uintptr(unsafe.Pointer(&freeBytes))) - if r == 0 { + var freeBytes uint64 + // https://learn.microsoft.com/en-us/windows/win32/api/fileapi/nf-fileapi-getdiskfreespaceexw + err := windows.GetDiskFreeSpaceEx(windows.StringToUTF16Ptr(path), &freeBytes, nil, nil) + if err != nil { logger.Panicf("FATAL: cannot get free space for %q : %s", path, err) } - return uint64(freeBytes) + return freeBytes } // stub @@ -132,23 +125,11 @@ func fadviseSequentialRead(f *os.File, prefetch bool) error { return nil } -// copied from https://github.com/juju/fslock/blob/master/fslock_windows.go // https://docs.microsoft.com/en-us/windows/win32/api/minwinbase/ns-minwinbase-overlapped func newOverlapped() (*windows.Overlapped, error) { - event, err := createEvent(nil, nil) + event, err := windows.CreateEvent(nil, 1, 1, nil) if err != nil { - return nil, err + return nil, fmt.Errorf("cannot create event: %w", err) } return &windows.Overlapped{HEvent: event}, nil } - -// copied from https://github.com/juju/fslock/blob/master/fslock_windows.go -// https://docs.microsoft.com/en-us/windows/win32/api/synchapi/nf-synchapi-createeventa -func createEvent(sa *windows.SecurityAttributes, name *uint16) (windows.Handle, error) { - r0, _, err := procEvent.Call(uintptr(unsafe.Pointer(sa)), uintptr(1), uintptr(1), uintptr(unsafe.Pointer(name))) - handle := windows.Handle(r0) - if handle == windows.InvalidHandle { - return 0, err - } - return handle, nil -}