Skip to content

Commit c3fa2b8

Browse files
kmirzavazirigopherbot
authored andcommitted
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 <[email protected]> LUCI-TryBot-Result: Go LUCI <[email protected]> Reviewed-by: Cherry Mui <[email protected]> Reviewed-by: Bryan Mills <[email protected]> Reviewed-by: Quim Muntal <[email protected]>
1 parent f69d32a commit c3fa2b8

File tree

2 files changed

+50
-7
lines changed

2 files changed

+50
-7
lines changed

windows/env_windows.go

+10-7
Original file line numberDiff line numberDiff line change
@@ -37,14 +37,17 @@ func (token Token) Environ(inheritExisting bool) (env []string, err error) {
3737
return nil, err
3838
}
3939
defer DestroyEnvironmentBlock(block)
40-
blockp := unsafe.Pointer(block)
41-
for {
42-
entry := UTF16PtrToString((*uint16)(blockp))
43-
if len(entry) == 0 {
44-
break
40+
size := unsafe.Sizeof(*block)
41+
for *block != 0 {
42+
// find NUL terminator
43+
end := unsafe.Pointer(block)
44+
for *(*uint16)(end) != 0 {
45+
end = unsafe.Add(end, size)
4546
}
46-
env = append(env, entry)
47-
blockp = unsafe.Add(blockp, 2*(len(entry)+1))
47+
48+
entry := unsafe.Slice(block, (uintptr(end)-uintptr(unsafe.Pointer(block)))/size)
49+
env = append(env, UTF16ToString(entry))
50+
block = (*uint16)(unsafe.Add(end, size))
4851
}
4952
return env, nil
5053
}

windows/env_windows_test.go

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
// Copyright 2024 The Go Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
package windows_test
6+
7+
import (
8+
"fmt"
9+
"slices"
10+
"testing"
11+
12+
"golang.org/x/sys/windows"
13+
)
14+
15+
func TestEnvironUTF8(t *testing.T) {
16+
testEnvVariable1Key := "__GO_X_SYS_WINDOWS_ENV_WINDOWS_TEST_VAR_BEAVER"
17+
testEnvVariable1Val := "🦫"
18+
t.Setenv(testEnvVariable1Key, testEnvVariable1Val)
19+
20+
testEnvVariable2Key := "__GO_X_SYS_WINDOWS_ENV_WINDOWS_TEST_VAR_WHALE"
21+
testEnvVariable2Val := "🐳"
22+
t.Setenv(testEnvVariable2Key, testEnvVariable2Val)
23+
24+
var userToken windows.Token
25+
26+
env, err := userToken.Environ(true)
27+
if err != nil {
28+
t.Error(err)
29+
}
30+
31+
testEnvVariable1 := fmt.Sprintf("%s=%s", testEnvVariable1Key, testEnvVariable1Val)
32+
if !slices.Contains(env, testEnvVariable1) {
33+
t.Fatalf("expected to find %s in env", testEnvVariable1)
34+
}
35+
36+
testEnvVariable2 := fmt.Sprintf("%s=%s", testEnvVariable2Key, testEnvVariable2Val)
37+
if !slices.Contains(env, testEnvVariable2) {
38+
t.Fatalf("expected to find %s in env", testEnvVariable2)
39+
}
40+
}

0 commit comments

Comments
 (0)