// Copyright 2019 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 runtime_test import ( "bufio" "bytes" "fmt" "internal/testenv" "os/exec" "path/filepath" "runtime" "strings" "syscall" "testing" ) func TestVectoredHandlerExceptionInNonGoThread(t *testing.T) { if *flagQuick { t.Skip("-quick") } if strings.HasPrefix(testenv.Builder(), "windows-amd64-2012") { testenv.SkipFlaky(t, 49681) } testenv.MustHaveGoBuild(t) testenv.MustHaveCGO(t) testenv.MustHaveExecPath(t, "gcc") testprog.Lock() defer testprog.Unlock() dir := t.TempDir() // build c program dll := filepath.Join(dir, "veh.dll") cmd := exec.Command("gcc", "-shared", "-o", dll, "testdata/testwinlibthrow/veh.c") out, err := testenv.CleanCmdEnv(cmd).CombinedOutput() if err != nil { t.Fatalf("failed to build c exe: %s\n%s", err, out) } // build go exe exe := filepath.Join(dir, "test.exe") cmd = exec.Command(testenv.GoToolPath(t), "build", "-o", exe, "testdata/testwinlibthrow/main.go") out, err = testenv.CleanCmdEnv(cmd).CombinedOutput() if err != nil { t.Fatalf("failed to build go library: %s\n%s", err, out) } // run test program in same thread cmd = exec.Command(exe) out, err = testenv.CleanCmdEnv(cmd).CombinedOutput() if err == nil { t.Fatal("error expected") } if _, ok := err.(*exec.ExitError); ok && len(out) > 0 { if !bytes.Contains(out, []byte("Exception 0x2a")) { t.Fatalf("unexpected failure while running executable: %s\n%s", err, out) } } else { t.Fatalf("unexpected error while running executable: %s\n%s", err, out) } // run test program in a new thread cmd = exec.Command(exe, "thread") out, err = testenv.CleanCmdEnv(cmd).CombinedOutput() if err == nil { t.Fatal("error expected") } if err, ok := err.(*exec.ExitError); ok { if err.ExitCode() != 42 { t.Fatalf("unexpected failure while running executable: %s\n%s", err, out) } } else { t.Fatalf("unexpected error while running executable: %s\n%s", err, out) } } func TestVectoredHandlerDontCrashOnLibrary(t *testing.T) { if *flagQuick { t.Skip("-quick") } if runtime.GOARCH == "arm" { //TODO: remove this skip and update testwinlib/main.c // once windows/arm supports c-shared buildmode. // See go.dev/issues/43800. t.Skip("this test can't run on windows/arm") } testenv.MustHaveGoBuild(t) testenv.MustHaveCGO(t) testenv.MustHaveExecPath(t, "gcc") testprog.Lock() defer testprog.Unlock() dir := t.TempDir() // build go dll dll := filepath.Join(dir, "testwinlib.dll") cmd := exec.Command(testenv.GoToolPath(t), "build", "-o", dll, "-buildmode", "c-shared", "testdata/testwinlib/main.go") out, err := testenv.CleanCmdEnv(cmd).CombinedOutput() if err != nil { t.Fatalf("failed to build go library: %s\n%s", err, out) } // build c program exe := filepath.Join(dir, "test.exe") cmd = exec.Command("gcc", "-L"+dir, "-I"+dir, "-ltestwinlib", "-o", exe, "testdata/testwinlib/main.c") out, err = testenv.CleanCmdEnv(cmd).CombinedOutput() if err != nil { t.Fatalf("failed to build c exe: %s\n%s", err, out) } // run test program cmd = exec.Command(exe) out, err = testenv.CleanCmdEnv(cmd).CombinedOutput() if err != nil { t.Fatalf("failure while running executable: %s\n%s", err, out) } var expectedOutput string if runtime.GOARCH == "arm64" || runtime.GOARCH == "arm" { // TODO: remove when windows/arm64 and windows/arm support SEH stack unwinding. expectedOutput = "exceptionCount: 1\ncontinueCount: 1\nunhandledCount: 0\n" } else { expectedOutput = "exceptionCount: 1\ncontinueCount: 1\nunhandledCount: 1\n" } // cleaning output cleanedOut := strings.ReplaceAll(string(out), "\r\n", "\n") if cleanedOut != expectedOutput { t.Errorf("expected output %q, got %q", expectedOutput, cleanedOut) } } func sendCtrlBreak(pid int) error { kernel32, err := syscall.LoadDLL("kernel32.dll") if err != nil { return fmt.Errorf("LoadDLL: %v\n", err) } generateEvent, err := kernel32.FindProc("GenerateConsoleCtrlEvent") if err != nil { return fmt.Errorf("FindProc: %v\n", err) } result, _, err := generateEvent.Call(syscall.CTRL_BREAK_EVENT, uintptr(pid)) if result == 0 { return fmt.Errorf("GenerateConsoleCtrlEvent: %v\n", err) } return nil } // TestCtrlHandler tests that Go can gracefully handle closing the console window. // See https://golang.org/issues/41884. func TestCtrlHandler(t *testing.T) { testenv.MustHaveGoBuild(t) t.Parallel() // build go program exe := filepath.Join(t.TempDir(), "test.exe") cmd := exec.Command(testenv.GoToolPath(t), "build", "-o", exe, "testdata/testwinsignal/main.go") out, err := testenv.CleanCmdEnv(cmd).CombinedOutput() if err != nil { t.Fatalf("failed to build go exe: %v\n%s", err, out) } // run test program cmd = exec.Command(exe) var stdout strings.Builder var stderr strings.Builder cmd.Stdout = &stdout cmd.Stderr = &stderr inPipe, err := cmd.StdinPipe() if err != nil { t.Fatalf("Failed to create stdin pipe: %v", err) } // keep inPipe alive until the end of the test defer inPipe.Close() // in a new command window const _CREATE_NEW_CONSOLE = 0x00000010 cmd.SysProcAttr = &syscall.SysProcAttr{ CreationFlags: _CREATE_NEW_CONSOLE, HideWindow: true, } if err := cmd.Start(); err != nil { t.Fatalf("Start failed: %v", err) } defer func() { cmd.Process.Kill() cmd.Wait() }() // check child exited gracefully, did not timeout if err := cmd.Wait(); err != nil { t.Fatalf("Program exited with error: %v\n%s", err, &stderr) } // check child received, handled SIGTERM if expected, got := syscall.SIGTERM.String(), strings.TrimSpace(stdout.String()); expected != got { t.Fatalf("Expected '%s' got: %s", expected, got) } } // TestLibraryCtrlHandler tests that Go DLL allows calling program to handle console control events. // See https://golang.org/issues/35965. func TestLibraryCtrlHandler(t *testing.T) { if *flagQuick { t.Skip("-quick") } if runtime.GOARCH != "amd64" { t.Skip("this test can only run on windows/amd64") } testenv.MustHaveGoBuild(t) testenv.MustHaveCGO(t) testenv.MustHaveExecPath(t, "gcc") testprog.Lock() defer testprog.Unlock() dir := t.TempDir() // build go dll dll := filepath.Join(dir, "dummy.dll") cmd := exec.Command(testenv.GoToolPath(t), "build", "-o", dll, "-buildmode", "c-shared", "testdata/testwinlibsignal/dummy.go") out, err := testenv.CleanCmdEnv(cmd).CombinedOutput() if err != nil { t.Fatalf("failed to build go library: %s\n%s", err, out) } // build c program exe := filepath.Join(dir, "test.exe") cmd = exec.Command("gcc", "-o", exe, "testdata/testwinlibsignal/main.c") out, err = testenv.CleanCmdEnv(cmd).CombinedOutput() if err != nil { t.Fatalf("failed to build c exe: %s\n%s", err, out) } // run test program cmd = exec.Command(exe) var stderr bytes.Buffer cmd.Stderr = &stderr outPipe, err := cmd.StdoutPipe() if err != nil { t.Fatalf("Failed to create stdout pipe: %v", err) } outReader := bufio.NewReader(outPipe) cmd.SysProcAttr = &syscall.SysProcAttr{ CreationFlags: syscall.CREATE_NEW_PROCESS_GROUP, } if err := cmd.Start(); err != nil { t.Fatalf("Start failed: %v", err) } errCh := make(chan error, 1) go func() { if line, err := outReader.ReadString('\n'); err != nil { errCh <- fmt.Errorf("could not read stdout: %v", err) } else if strings.TrimSpace(line) != "ready" { errCh <- fmt.Errorf("unexpected message: %v", line) } else { errCh <- sendCtrlBreak(cmd.Process.Pid) } }() if err := <-errCh; err != nil { t.Fatal(err) } if err := cmd.Wait(); err != nil { t.Fatalf("Program exited with error: %v\n%s", err, &stderr) } } func TestIssue59213(t *testing.T) { if runtime.GOOS != "windows" { t.Skip("skipping windows only test") } if *flagQuick { t.Skip("-quick") } testenv.MustHaveGoBuild(t) testenv.MustHaveCGO(t) goEnv := func(arg string) string { cmd := testenv.Command(t, testenv.GoToolPath(t), "env", arg) cmd.Stderr = new(bytes.Buffer) line, err := cmd.Output() if err != nil { t.Fatalf("%v: %v\n%s", cmd, err, cmd.Stderr) } out := string(bytes.TrimSpace(line)) t.Logf("%v: %q", cmd, out) return out } cc := goEnv("CC") cgoCflags := goEnv("CGO_CFLAGS") t.Parallel() tmpdir := t.TempDir() dllfile := filepath.Join(tmpdir, "test.dll") exefile := filepath.Join(tmpdir, "gotest.exe") // build go dll cmd := testenv.Command(t, testenv.GoToolPath(t), "build", "-o", dllfile, "-buildmode", "c-shared", "testdata/testwintls/main.go") out, err := testenv.CleanCmdEnv(cmd).CombinedOutput() if err != nil { t.Fatalf("failed to build go library: %s\n%s", err, out) } // build c program cmd = testenv.Command(t, cc, "-o", exefile, "testdata/testwintls/main.c") testenv.CleanCmdEnv(cmd) cmd.Env = append(cmd.Env, "CGO_CFLAGS="+cgoCflags) out, err = cmd.CombinedOutput() if err != nil { t.Fatalf("failed to build c exe: %s\n%s", err, out) } // run test program cmd = testenv.Command(t, exefile, dllfile, "GoFunc") out, err = testenv.CleanCmdEnv(cmd).CombinedOutput() if err != nil { t.Fatalf("failed: %s\n%s", err, out) } }