From c3fa2b8b4120c7313f31d04a946f0ae173fa9e7c Mon Sep 17 00:00:00 2001 From: Kamyar Mirzavaziri Date: Fri, 19 Jan 2024 04:14:25 +0330 Subject: [PATCH] windows: fix parsing of non-ASCII entries in token.Environ Fixes golang/go#65055, the unexpected behavior of token.Environ in parsing entries containing runes larger than 2 bytes in size Change-Id: I753d2c605e3a2d7a1d90cd18601d6b918f0d3f7a Reviewed-on: https://go-review.googlesource.com/c/sys/+/556895 Auto-Submit: Bryan Mills LUCI-TryBot-Result: Go LUCI Reviewed-by: Cherry Mui Reviewed-by: Bryan Mills Reviewed-by: Quim Muntal --- windows/env_windows.go | 17 +++++++++------- windows/env_windows_test.go | 40 +++++++++++++++++++++++++++++++++++++ 2 files changed, 50 insertions(+), 7 deletions(-) create mode 100644 windows/env_windows_test.go diff --git a/windows/env_windows.go b/windows/env_windows.go index b8ad19250..d4577a423 100644 --- a/windows/env_windows.go +++ b/windows/env_windows.go @@ -37,14 +37,17 @@ func (token Token) Environ(inheritExisting bool) (env []string, err error) { return nil, err } defer DestroyEnvironmentBlock(block) - blockp := unsafe.Pointer(block) - for { - entry := UTF16PtrToString((*uint16)(blockp)) - if len(entry) == 0 { - break + size := unsafe.Sizeof(*block) + for *block != 0 { + // find NUL terminator + end := unsafe.Pointer(block) + for *(*uint16)(end) != 0 { + end = unsafe.Add(end, size) } - env = append(env, entry) - blockp = unsafe.Add(blockp, 2*(len(entry)+1)) + + entry := unsafe.Slice(block, (uintptr(end)-uintptr(unsafe.Pointer(block)))/size) + env = append(env, UTF16ToString(entry)) + block = (*uint16)(unsafe.Add(end, size)) } return env, nil } diff --git a/windows/env_windows_test.go b/windows/env_windows_test.go new file mode 100644 index 000000000..6824e6ed6 --- /dev/null +++ b/windows/env_windows_test.go @@ -0,0 +1,40 @@ +// Copyright 2024 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package windows_test + +import ( + "fmt" + "slices" + "testing" + + "golang.org/x/sys/windows" +) + +func TestEnvironUTF8(t *testing.T) { + testEnvVariable1Key := "__GO_X_SYS_WINDOWS_ENV_WINDOWS_TEST_VAR_BEAVER" + testEnvVariable1Val := "🦫" + t.Setenv(testEnvVariable1Key, testEnvVariable1Val) + + testEnvVariable2Key := "__GO_X_SYS_WINDOWS_ENV_WINDOWS_TEST_VAR_WHALE" + testEnvVariable2Val := "🐳" + t.Setenv(testEnvVariable2Key, testEnvVariable2Val) + + var userToken windows.Token + + env, err := userToken.Environ(true) + if err != nil { + t.Error(err) + } + + testEnvVariable1 := fmt.Sprintf("%s=%s", testEnvVariable1Key, testEnvVariable1Val) + if !slices.Contains(env, testEnvVariable1) { + t.Fatalf("expected to find %s in env", testEnvVariable1) + } + + testEnvVariable2 := fmt.Sprintf("%s=%s", testEnvVariable2Key, testEnvVariable2Val) + if !slices.Contains(env, testEnvVariable2) { + t.Fatalf("expected to find %s in env", testEnvVariable2) + } +}