2012-06-03 19:27:17 +10:00
|
|
|
// Copyright 2012 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 syscall_test
|
|
|
|
|
|
|
|
|
|
import (
|
2021-04-29 15:38:56 +10:00
|
|
|
"fmt"
|
|
|
|
|
"internal/testenv"
|
2012-06-03 19:27:17 +10:00
|
|
|
"os"
|
2021-04-29 15:38:56 +10:00
|
|
|
"os/exec"
|
2012-06-03 19:27:17 +10:00
|
|
|
"path/filepath"
|
2021-04-29 15:38:56 +10:00
|
|
|
"strings"
|
2012-06-03 19:27:17 +10:00
|
|
|
"syscall"
|
|
|
|
|
"testing"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func TestWin32finddata(t *testing.T) {
|
2021-04-07 22:32:56 +08:00
|
|
|
dir := t.TempDir()
|
2012-06-03 19:27:17 +10:00
|
|
|
|
|
|
|
|
path := filepath.Join(dir, "long_name.and_extension")
|
|
|
|
|
f, err := os.Create(path)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("failed to create %v: %v", path, err)
|
|
|
|
|
}
|
|
|
|
|
f.Close()
|
|
|
|
|
|
|
|
|
|
type X struct {
|
2012-06-08 14:04:44 -04:00
|
|
|
fd syscall.Win32finddata
|
2012-06-03 19:27:17 +10:00
|
|
|
got byte
|
|
|
|
|
pad [10]byte // to protect ourselves
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
var want byte = 2 // it is unlikely to have this character in the filename
|
|
|
|
|
x := X{got: want}
|
|
|
|
|
|
syscall: return EINVAL when string arguments have NUL characters
Since NUL usually terminates strings in underlying syscalls, allowing
it when converting string arguments is a security risk, especially
when dealing with filenames. For example, a program might reason that
filename like "/root/..\x00/" is a subdirectory or "/root/" and allow
access to it, while underlying syscall will treat "\x00" as an end of
that string and the actual filename will be "/root/..", which might
be unexpected. Returning EINVAL when string arguments have NUL in
them makes sure this attack vector is unusable.
R=golang-dev, r, bradfitz, fullung, rsc, minux.ma
CC=golang-dev
https://golang.org/cl/6458050
2012-08-05 17:24:32 -04:00
|
|
|
pathp, _ := syscall.UTF16PtrFromString(path)
|
|
|
|
|
h, err := syscall.FindFirstFile(pathp, &(x.fd))
|
2012-06-03 19:27:17 +10:00
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("FindFirstFile failed: %v", err)
|
|
|
|
|
}
|
|
|
|
|
err = syscall.FindClose(h)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("FindClose failed: %v", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if x.got != want {
|
|
|
|
|
t.Fatalf("memory corruption: want=%d got=%d", want, x.got)
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-08-29 21:44:46 +08:00
|
|
|
|
|
|
|
|
func abort(funcname string, err error) {
|
|
|
|
|
panic(funcname + " failed: " + err.Error())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func ExampleLoadLibrary() {
|
|
|
|
|
h, err := syscall.LoadLibrary("kernel32.dll")
|
|
|
|
|
if err != nil {
|
|
|
|
|
abort("LoadLibrary", err)
|
|
|
|
|
}
|
|
|
|
|
defer syscall.FreeLibrary(h)
|
|
|
|
|
proc, err := syscall.GetProcAddress(h, "GetVersion")
|
|
|
|
|
if err != nil {
|
|
|
|
|
abort("GetProcAddress", err)
|
|
|
|
|
}
|
|
|
|
|
r, _, _ := syscall.Syscall(uintptr(proc), 0, 0, 0, 0)
|
|
|
|
|
major := byte(r)
|
|
|
|
|
minor := uint8(r >> 8)
|
|
|
|
|
build := uint16(r >> 16)
|
|
|
|
|
print("windows version ", major, ".", minor, " (Build ", build, ")\n")
|
|
|
|
|
}
|
2018-06-10 01:01:27 +07:00
|
|
|
|
|
|
|
|
func TestTOKEN_ALL_ACCESS(t *testing.T) {
|
|
|
|
|
if syscall.TOKEN_ALL_ACCESS != 0xF01FF {
|
|
|
|
|
t.Errorf("TOKEN_ALL_ACCESS = %x, want 0xF01FF", syscall.TOKEN_ALL_ACCESS)
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-04-29 15:38:56 +10:00
|
|
|
|
|
|
|
|
func TestStdioAreInheritable(t *testing.T) {
|
|
|
|
|
testenv.MustHaveGoBuild(t)
|
|
|
|
|
testenv.MustHaveExecPath(t, "gcc")
|
|
|
|
|
|
|
|
|
|
tmpdir := t.TempDir()
|
|
|
|
|
|
|
|
|
|
// build go dll
|
|
|
|
|
const dlltext = `
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import "C"
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
//export HelloWorld
|
|
|
|
|
func HelloWorld() {
|
|
|
|
|
fmt.Println("Hello World")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func main() {}
|
|
|
|
|
`
|
|
|
|
|
dllsrc := filepath.Join(tmpdir, "helloworld.go")
|
|
|
|
|
err := os.WriteFile(dllsrc, []byte(dlltext), 0644)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
dll := filepath.Join(tmpdir, "helloworld.dll")
|
|
|
|
|
cmd := exec.Command(testenv.GoToolPath(t), "build", "-o", dll, "-buildmode", "c-shared", dllsrc)
|
|
|
|
|
out, err := testenv.CleanCmdEnv(cmd).CombinedOutput()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("failed to build go library: %s\n%s", err, out)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// run powershell script
|
|
|
|
|
psscript := fmt.Sprintf(`
|
|
|
|
|
hostname;
|
|
|
|
|
$signature = " [DllImport("%q")] public static extern void HelloWorld(); ";
|
|
|
|
|
Add-Type -MemberDefinition $signature -Name World -Namespace Hello;
|
|
|
|
|
[Hello.World]::HelloWorld();
|
|
|
|
|
hostname;
|
|
|
|
|
`, dll)
|
|
|
|
|
psscript = strings.ReplaceAll(psscript, "\n", "")
|
|
|
|
|
out, err = exec.Command("powershell", "-Command", psscript).CombinedOutput()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("Powershell command failed: %v: %v", err, string(out))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
hostname, err := os.Hostname()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
have := strings.ReplaceAll(string(out), "\n", "")
|
|
|
|
|
have = strings.ReplaceAll(have, "\r", "")
|
|
|
|
|
want := fmt.Sprintf("%sHello World%s", hostname, hostname)
|
|
|
|
|
if have != want {
|
|
|
|
|
t.Fatalf("Powershell command output is wrong: got %q, want %q", have, want)
|
|
|
|
|
}
|
|
|
|
|
}
|