Source file
src/runtime/crash_cgo_test.go
1
2
3
4
5
6
7 package runtime_test
8
9 import (
10 "fmt"
11 "internal/goos"
12 "internal/platform"
13 "internal/testenv"
14 "os"
15 "os/exec"
16 "runtime"
17 "strconv"
18 "strings"
19 "testing"
20 "time"
21 )
22
23 func TestCgoCrashHandler(t *testing.T) {
24 t.Parallel()
25 testCrashHandler(t, true)
26 }
27
28 func TestCgoSignalDeadlock(t *testing.T) {
29
30
31
32
33 if testing.Short() && runtime.GOOS == "windows" {
34 t.Skip("Skipping in short mode")
35 }
36 got := runTestProg(t, "testprogcgo", "CgoSignalDeadlock")
37 want := "OK\n"
38 if got != want {
39 t.Fatalf("expected %q, but got:\n%s", want, got)
40 }
41 }
42
43 func TestCgoTraceback(t *testing.T) {
44 t.Parallel()
45 got := runTestProg(t, "testprogcgo", "CgoTraceback")
46 want := "OK\n"
47 if got != want {
48 t.Fatalf("expected %q, but got:\n%s", want, got)
49 }
50 }
51
52 func TestCgoCallbackGC(t *testing.T) {
53 t.Parallel()
54 switch runtime.GOOS {
55 case "plan9", "windows":
56 t.Skipf("no pthreads on %s", runtime.GOOS)
57 }
58 if testing.Short() {
59 switch {
60 case runtime.GOOS == "dragonfly":
61 t.Skip("see golang.org/issue/11990")
62 case runtime.GOOS == "linux" && runtime.GOARCH == "arm":
63 t.Skip("too slow for arm builders")
64 case runtime.GOOS == "linux" && (runtime.GOARCH == "mips64" || runtime.GOARCH == "mips64le"):
65 t.Skip("too slow for mips64x builders")
66 }
67 }
68 got := runTestProg(t, "testprogcgo", "CgoCallbackGC")
69 want := "OK\n"
70 if got != want {
71 t.Fatalf("expected %q, but got:\n%s", want, got)
72 }
73 }
74
75 func TestCgoCallbackPprof(t *testing.T) {
76 t.Parallel()
77 switch runtime.GOOS {
78 case "plan9", "windows":
79 t.Skipf("no pthreads on %s", runtime.GOOS)
80 }
81 if testenv.CPUProfilingBroken() {
82 t.Skip("skipping on platform with broken profiling")
83 }
84
85 got := runTestProg(t, "testprogcgo", "CgoCallbackPprof")
86 if want := "OK\n"; got != want {
87 t.Fatalf("expected %q, but got:\n%s", want, got)
88 }
89 }
90
91 func TestCgoExternalThreadPanic(t *testing.T) {
92 t.Parallel()
93 if runtime.GOOS == "plan9" {
94 t.Skipf("no pthreads on %s", runtime.GOOS)
95 }
96 got := runTestProg(t, "testprogcgo", "CgoExternalThreadPanic")
97 want := "panic: BOOM"
98 if !strings.Contains(got, want) {
99 t.Fatalf("want failure containing %q. output:\n%s\n", want, got)
100 }
101 }
102
103 func TestCgoExternalThreadSIGPROF(t *testing.T) {
104 t.Parallel()
105
106 switch runtime.GOOS {
107 case "plan9", "windows":
108 t.Skipf("no pthreads on %s", runtime.GOOS)
109 }
110
111 got := runTestProg(t, "testprogcgo", "CgoExternalThreadSIGPROF", "GO_START_SIGPROF_THREAD=1")
112 if want := "OK\n"; got != want {
113 t.Fatalf("expected %q, but got:\n%s", want, got)
114 }
115 }
116
117 func TestCgoExternalThreadSignal(t *testing.T) {
118 t.Parallel()
119
120 switch runtime.GOOS {
121 case "plan9", "windows":
122 t.Skipf("no pthreads on %s", runtime.GOOS)
123 }
124
125 got := runTestProg(t, "testprogcgo", "CgoExternalThreadSignal")
126 if want := "OK\n"; got != want {
127 if runtime.GOOS == "ios" && strings.Contains(got, "C signal did not crash as expected") {
128 testenv.SkipFlaky(t, 59913)
129 }
130 t.Fatalf("expected %q, but got:\n%s", want, got)
131 }
132 }
133
134 func TestCgoDLLImports(t *testing.T) {
135
136 if runtime.GOOS != "windows" {
137 t.Skip("skipping windows specific test")
138 }
139 got := runTestProg(t, "testprogcgo", "CgoDLLImportsMain")
140 want := "OK\n"
141 if got != want {
142 t.Fatalf("expected %q, but got %v", want, got)
143 }
144 }
145
146 func TestCgoExecSignalMask(t *testing.T) {
147 t.Parallel()
148
149 switch runtime.GOOS {
150 case "windows", "plan9":
151 t.Skipf("skipping signal mask test on %s", runtime.GOOS)
152 }
153 got := runTestProg(t, "testprogcgo", "CgoExecSignalMask", "GOTRACEBACK=system")
154 want := "OK\n"
155 if got != want {
156 t.Errorf("expected %q, got %v", want, got)
157 }
158 }
159
160 func TestEnsureDropM(t *testing.T) {
161 t.Parallel()
162
163 switch runtime.GOOS {
164 case "windows", "plan9":
165 t.Skipf("skipping dropm test on %s", runtime.GOOS)
166 }
167 got := runTestProg(t, "testprogcgo", "EnsureDropM")
168 want := "OK\n"
169 if got != want {
170 t.Errorf("expected %q, got %v", want, got)
171 }
172 }
173
174
175
176
177 func TestCgoCheckBytes(t *testing.T) {
178 t.Parallel()
179
180 testenv.MustHaveGoBuild(t)
181 exe, err := buildTestProg(t, "testprogcgo")
182 if err != nil {
183 t.Fatal(err)
184 }
185
186
187 const tries = 10
188 var tot1, tot2 time.Duration
189 for i := 0; i < tries; i++ {
190 cmd := testenv.CleanCmdEnv(exec.Command(exe, "CgoCheckBytes"))
191 cmd.Env = append(cmd.Env, "GODEBUG=cgocheck=0", fmt.Sprintf("GO_CGOCHECKBYTES_TRY=%d", i))
192
193 start := time.Now()
194 cmd.Run()
195 d1 := time.Since(start)
196
197 cmd = testenv.CleanCmdEnv(exec.Command(exe, "CgoCheckBytes"))
198 cmd.Env = append(cmd.Env, fmt.Sprintf("GO_CGOCHECKBYTES_TRY=%d", i))
199
200 start = time.Now()
201 cmd.Run()
202 d2 := time.Since(start)
203
204 if d1*20 > d2 {
205
206
207 return
208 }
209
210 tot1 += d1
211 tot2 += d2
212 }
213
214 t.Errorf("cgo check too slow: got %v, expected at most %v", tot2/tries, (tot1/tries)*20)
215 }
216
217 func TestCgoPanicDeadlock(t *testing.T) {
218 t.Parallel()
219
220 got := runTestProg(t, "testprogcgo", "CgoPanicDeadlock")
221 want := "panic: cgo error\n\n"
222 if !strings.HasPrefix(got, want) {
223 t.Fatalf("output does not start with %q:\n%s", want, got)
224 }
225 }
226
227 func TestCgoCCodeSIGPROF(t *testing.T) {
228 t.Parallel()
229 got := runTestProg(t, "testprogcgo", "CgoCCodeSIGPROF")
230 want := "OK\n"
231 if got != want {
232 t.Errorf("expected %q got %v", want, got)
233 }
234 }
235
236 func TestCgoPprofCallback(t *testing.T) {
237 if testing.Short() {
238 t.Skip("skipping in short mode")
239 }
240 switch runtime.GOOS {
241 case "windows", "plan9":
242 t.Skipf("skipping cgo pprof callback test on %s", runtime.GOOS)
243 }
244 got := runTestProg(t, "testprogcgo", "CgoPprofCallback")
245 want := "OK\n"
246 if got != want {
247 t.Errorf("expected %q got %v", want, got)
248 }
249 }
250
251 func TestCgoCrashTraceback(t *testing.T) {
252 t.Parallel()
253 switch platform := runtime.GOOS + "/" + runtime.GOARCH; platform {
254 case "darwin/amd64":
255 case "linux/amd64":
256 case "linux/arm64":
257 case "linux/loong64":
258 case "linux/ppc64le":
259 default:
260 t.Skipf("not yet supported on %s", platform)
261 }
262 got := runTestProg(t, "testprogcgo", "CrashTraceback")
263 for i := 1; i <= 3; i++ {
264 if !strings.Contains(got, fmt.Sprintf("cgo symbolizer:%d", i)) {
265 t.Errorf("missing cgo symbolizer:%d", i)
266 }
267 }
268 }
269
270 func TestCgoCrashTracebackGo(t *testing.T) {
271 t.Parallel()
272 switch platform := runtime.GOOS + "/" + runtime.GOARCH; platform {
273 case "darwin/amd64":
274 case "linux/amd64":
275 case "linux/arm64":
276 case "linux/loong64":
277 case "linux/ppc64le":
278 default:
279 t.Skipf("not yet supported on %s", platform)
280 }
281 got := runTestProg(t, "testprogcgo", "CrashTracebackGo")
282 for i := 1; i <= 3; i++ {
283 want := fmt.Sprintf("main.h%d", i)
284 if !strings.Contains(got, want) {
285 t.Errorf("missing %s", want)
286 }
287 }
288 }
289
290 func TestCgoTracebackContext(t *testing.T) {
291 t.Parallel()
292 got := runTestProg(t, "testprogcgo", "TracebackContext")
293 want := "OK\n"
294 if got != want {
295 t.Errorf("expected %q got %v", want, got)
296 }
297 }
298
299 func TestCgoTracebackContextPreemption(t *testing.T) {
300 t.Parallel()
301 got := runTestProg(t, "testprogcgo", "TracebackContextPreemption")
302 want := "OK\n"
303 if got != want {
304 t.Errorf("expected %q got %v", want, got)
305 }
306 }
307
308 func testCgoPprof(t *testing.T, buildArg, runArg, top, bottom string) {
309 t.Parallel()
310 if runtime.GOOS != "linux" || (runtime.GOARCH != "amd64" && runtime.GOARCH != "ppc64le" && runtime.GOARCH != "arm64" && runtime.GOARCH != "loong64") {
311 t.Skipf("not yet supported on %s/%s", runtime.GOOS, runtime.GOARCH)
312 }
313 testenv.MustHaveGoRun(t)
314
315 exe, err := buildTestProg(t, "testprogcgo", buildArg)
316 if err != nil {
317 t.Fatal(err)
318 }
319
320 cmd := testenv.CleanCmdEnv(exec.Command(exe, runArg))
321 got, err := cmd.CombinedOutput()
322 if err != nil {
323 if testenv.Builder() == "linux-amd64-alpine" {
324
325 t.Skipf("Skipping failing test on Alpine (golang.org/issue/18243). Ignoring error: %v", err)
326 }
327 t.Fatalf("%s\n\n%v", got, err)
328 }
329 fn := strings.TrimSpace(string(got))
330 defer os.Remove(fn)
331
332 for try := 0; try < 2; try++ {
333 cmd := testenv.CleanCmdEnv(exec.Command(testenv.GoToolPath(t), "tool", "pprof", "-tagignore=ignore", "-traces"))
334
335 if try == 0 {
336 cmd.Args = append(cmd.Args, exe, fn)
337 } else {
338 cmd.Args = append(cmd.Args, fn)
339 }
340
341 found := false
342 for i, e := range cmd.Env {
343 if strings.HasPrefix(e, "PPROF_TMPDIR=") {
344 cmd.Env[i] = "PPROF_TMPDIR=" + os.TempDir()
345 found = true
346 break
347 }
348 }
349 if !found {
350 cmd.Env = append(cmd.Env, "PPROF_TMPDIR="+os.TempDir())
351 }
352
353 out, err := cmd.CombinedOutput()
354 t.Logf("%s:\n%s", cmd.Args, out)
355 if err != nil {
356 t.Error(err)
357 continue
358 }
359
360 trace := findTrace(string(out), top)
361 if len(trace) == 0 {
362 t.Errorf("%s traceback missing.", top)
363 continue
364 }
365 if trace[len(trace)-1] != bottom {
366 t.Errorf("invalid traceback origin: got=%v; want=[%s ... %s]", trace, top, bottom)
367 }
368 }
369 }
370
371 func TestCgoPprof(t *testing.T) {
372 testCgoPprof(t, "", "CgoPprof", "cpuHog", "runtime.main")
373 }
374
375 func TestCgoPprofPIE(t *testing.T) {
376 testCgoPprof(t, "-buildmode=pie", "CgoPprof", "cpuHog", "runtime.main")
377 }
378
379 func TestCgoPprofThread(t *testing.T) {
380 testCgoPprof(t, "", "CgoPprofThread", "cpuHogThread", "cpuHogThread2")
381 }
382
383 func TestCgoPprofThreadNoTraceback(t *testing.T) {
384 testCgoPprof(t, "", "CgoPprofThreadNoTraceback", "cpuHogThread", "runtime._ExternalCode")
385 }
386
387 func TestRaceProf(t *testing.T) {
388 if !platform.RaceDetectorSupported(runtime.GOOS, runtime.GOARCH) {
389 t.Skipf("skipping on %s/%s because race detector not supported", runtime.GOOS, runtime.GOARCH)
390 }
391 if runtime.GOOS == "windows" {
392 t.Skipf("skipping: test requires pthread support")
393
394 }
395
396 testenv.MustHaveGoRun(t)
397
398
399
400 if testing.Short() {
401 t.Skip("skipping test in -short mode")
402 }
403
404 exe, err := buildTestProg(t, "testprogcgo", "-race")
405 if err != nil {
406 t.Fatal(err)
407 }
408
409 got, err := testenv.CleanCmdEnv(exec.Command(exe, "CgoRaceprof")).CombinedOutput()
410 if err != nil {
411 t.Fatal(err)
412 }
413 want := "OK\n"
414 if string(got) != want {
415 t.Errorf("expected %q got %s", want, got)
416 }
417 }
418
419 func TestRaceSignal(t *testing.T) {
420 if !platform.RaceDetectorSupported(runtime.GOOS, runtime.GOARCH) {
421 t.Skipf("skipping on %s/%s because race detector not supported", runtime.GOOS, runtime.GOARCH)
422 }
423 if runtime.GOOS == "windows" {
424 t.Skipf("skipping: test requires pthread support")
425
426 }
427 if runtime.GOOS == "darwin" || runtime.GOOS == "ios" {
428 testenv.SkipFlaky(t, 60316)
429 }
430
431 t.Parallel()
432
433 testenv.MustHaveGoRun(t)
434
435
436
437 if testing.Short() {
438 t.Skip("skipping test in -short mode")
439 }
440
441 exe, err := buildTestProg(t, "testprogcgo", "-race")
442 if err != nil {
443 t.Fatal(err)
444 }
445
446 got, err := testenv.CleanCmdEnv(testenv.Command(t, exe, "CgoRaceSignal")).CombinedOutput()
447 if err != nil {
448 t.Logf("%s\n", got)
449 t.Fatal(err)
450 }
451 want := "OK\n"
452 if string(got) != want {
453 t.Errorf("expected %q got %s", want, got)
454 }
455 }
456
457 func TestCgoNumGoroutine(t *testing.T) {
458 switch runtime.GOOS {
459 case "windows", "plan9":
460 t.Skipf("skipping numgoroutine test on %s", runtime.GOOS)
461 }
462 t.Parallel()
463 got := runTestProg(t, "testprogcgo", "NumGoroutine")
464 want := "OK\n"
465 if got != want {
466 t.Errorf("expected %q got %v", want, got)
467 }
468 }
469
470 func TestCatchPanic(t *testing.T) {
471 t.Parallel()
472 switch runtime.GOOS {
473 case "plan9", "windows":
474 t.Skipf("no signals on %s", runtime.GOOS)
475 case "darwin":
476 if runtime.GOARCH == "amd64" {
477 t.Skipf("crash() on darwin/amd64 doesn't raise SIGABRT")
478 }
479 }
480
481 testenv.MustHaveGoRun(t)
482
483 exe, err := buildTestProg(t, "testprogcgo")
484 if err != nil {
485 t.Fatal(err)
486 }
487
488 for _, early := range []bool{true, false} {
489 cmd := testenv.CleanCmdEnv(exec.Command(exe, "CgoCatchPanic"))
490
491 cmd.Env = append(cmd.Env, "GOTRACEBACK=crash")
492 if early {
493
494 cmd.Env = append(cmd.Env, "CGOCATCHPANIC_EARLY_HANDLER=1")
495 }
496 if out, err := cmd.CombinedOutput(); err != nil {
497 t.Errorf("testprogcgo CgoCatchPanic failed: %v\n%s", err, out)
498 }
499 }
500 }
501
502 func TestCgoLockOSThreadExit(t *testing.T) {
503 switch runtime.GOOS {
504 case "plan9", "windows":
505 t.Skipf("no pthreads on %s", runtime.GOOS)
506 }
507 t.Parallel()
508 testLockOSThreadExit(t, "testprogcgo")
509 }
510
511 func TestWindowsStackMemoryCgo(t *testing.T) {
512 if runtime.GOOS != "windows" {
513 t.Skip("skipping windows specific test")
514 }
515 testenv.SkipFlaky(t, 22575)
516 o := runTestProg(t, "testprogcgo", "StackMemory")
517 stackUsage, err := strconv.Atoi(o)
518 if err != nil {
519 t.Fatalf("Failed to read stack usage: %v", err)
520 }
521 if expected, got := 100<<10, stackUsage; got > expected {
522 t.Fatalf("expected < %d bytes of memory per thread, got %d", expected, got)
523 }
524 }
525
526 func TestSigStackSwapping(t *testing.T) {
527 switch runtime.GOOS {
528 case "plan9", "windows":
529 t.Skipf("no sigaltstack on %s", runtime.GOOS)
530 }
531 t.Parallel()
532 got := runTestProg(t, "testprogcgo", "SigStack")
533 want := "OK\n"
534 if got != want {
535 t.Errorf("expected %q got %v", want, got)
536 }
537 }
538
539 func TestCgoTracebackSigpanic(t *testing.T) {
540
541
542 if runtime.GOOS == "windows" {
543
544
545
546 t.Skip("no sigpanic in C on windows")
547 }
548 if runtime.GOOS == "ios" {
549 testenv.SkipFlaky(t, 59912)
550 }
551 t.Parallel()
552 got := runTestProg(t, "testprogcgo", "TracebackSigpanic")
553 t.Log(got)
554
555 want := "main.TracebackSigpanic"
556 if !strings.Contains(got, want) {
557 if runtime.GOOS == "android" && (runtime.GOARCH == "arm" || runtime.GOARCH == "arm64") {
558 testenv.SkipFlaky(t, 58794)
559 }
560 t.Errorf("did not see %q in output", want)
561 }
562
563 nowant := "runtime.sigpanic"
564 if strings.Contains(got, nowant) {
565 t.Errorf("unexpectedly saw %q in output", nowant)
566 }
567
568 nowant = "runtime: "
569 if strings.Contains(got, nowant) {
570 t.Errorf("unexpectedly saw %q in output", nowant)
571 }
572 }
573
574 func TestCgoPanicCallback(t *testing.T) {
575 t.Parallel()
576 got := runTestProg(t, "testprogcgo", "PanicCallback")
577 t.Log(got)
578 want := "panic: runtime error: invalid memory address or nil pointer dereference"
579 if !strings.Contains(got, want) {
580 t.Errorf("did not see %q in output", want)
581 }
582 want = "panic_callback"
583 if !strings.Contains(got, want) {
584 t.Errorf("did not see %q in output", want)
585 }
586 want = "PanicCallback"
587 if !strings.Contains(got, want) {
588 t.Errorf("did not see %q in output", want)
589 }
590
591 nowant := "runtime: "
592 if strings.Contains(got, nowant) {
593 t.Errorf("did not see %q in output", want)
594 }
595 }
596
597
598
599
600
601 func TestBigStackCallbackCgo(t *testing.T) {
602 if runtime.GOOS != "windows" {
603 t.Skip("skipping windows specific test")
604 }
605 t.Parallel()
606 got := runTestProg(t, "testprogcgo", "BigStack")
607 want := "OK\n"
608 if got != want {
609 t.Errorf("expected %q got %v", want, got)
610 }
611 }
612
613 func nextTrace(lines []string) ([]string, []string) {
614 var trace []string
615 for n, line := range lines {
616 if strings.HasPrefix(line, "---") {
617 return trace, lines[n+1:]
618 }
619 fields := strings.Fields(strings.TrimSpace(line))
620 if len(fields) == 0 {
621 continue
622 }
623
624 trace = append(trace, fields[len(fields)-1])
625 }
626 return nil, nil
627 }
628
629 func findTrace(text, top string) []string {
630 lines := strings.Split(text, "\n")
631 _, lines = nextTrace(lines)
632 for len(lines) > 0 {
633 var t []string
634 t, lines = nextTrace(lines)
635 if len(t) == 0 {
636 continue
637 }
638 if t[0] == top {
639 return t
640 }
641 }
642 return nil
643 }
644
645 func TestSegv(t *testing.T) {
646 switch runtime.GOOS {
647 case "plan9", "windows":
648 t.Skipf("no signals on %s", runtime.GOOS)
649 }
650
651 for _, test := range []string{"Segv", "SegvInCgo", "TgkillSegv", "TgkillSegvInCgo"} {
652 test := test
653
654
655 if runtime.GOOS != "linux" && strings.HasPrefix(test, "Tgkill") {
656 continue
657 }
658
659 t.Run(test, func(t *testing.T) {
660 if test == "SegvInCgo" && runtime.GOOS == "ios" {
661 testenv.SkipFlaky(t, 59947)
662 }
663
664 t.Parallel()
665 prog := "testprog"
666 if strings.HasSuffix(test, "InCgo") {
667 prog = "testprogcgo"
668 }
669 got := runTestProg(t, prog, test)
670 t.Log(got)
671 want := "SIGSEGV"
672 if !strings.Contains(got, want) {
673 if runtime.GOOS == "darwin" && runtime.GOARCH == "amd64" && strings.Contains(got, "fatal: morestack on g0") {
674 testenv.SkipFlaky(t, 39457)
675 }
676 t.Errorf("did not see %q in output", want)
677 }
678
679
680 switch runtime.GOOS {
681 case "darwin", "ios", "illumos", "solaris":
682
683 testenv.SkipFlaky(t, 49182)
684 case "linux":
685 if runtime.GOARCH == "386" {
686
687
688 testenv.SkipFlaky(t, 50504)
689 }
690 }
691 if test == "SegvInCgo" && strings.Contains(got, "unknown pc") {
692 testenv.SkipFlaky(t, 50979)
693 }
694
695 for _, nowant := range []string{"fatal error: ", "runtime: "} {
696 if strings.Contains(got, nowant) {
697 if runtime.GOOS == "darwin" && strings.Contains(got, "0xb01dfacedebac1e") {
698
699 t.Skip("skipping due to Darwin handling of malformed addresses")
700 }
701 t.Errorf("unexpectedly saw %q in output", nowant)
702 }
703 }
704 })
705 }
706 }
707
708 func TestAbortInCgo(t *testing.T) {
709 switch runtime.GOOS {
710 case "plan9", "windows":
711
712
713 t.Skipf("no signals on %s", runtime.GOOS)
714 }
715
716 t.Parallel()
717 got := runTestProg(t, "testprogcgo", "Abort")
718 t.Log(got)
719 want := "SIGABRT"
720 if !strings.Contains(got, want) {
721 t.Errorf("did not see %q in output", want)
722 }
723
724 nowant := "runtime: "
725 if strings.Contains(got, nowant) {
726 t.Errorf("did not see %q in output", want)
727 }
728 }
729
730
731
732 func TestEINTR(t *testing.T) {
733 switch runtime.GOOS {
734 case "plan9", "windows":
735 t.Skipf("no EINTR on %s", runtime.GOOS)
736 case "linux":
737 if runtime.GOARCH == "386" {
738
739
740
741
742
743
744
745 t.Skip("skipping on linux-386; C sigaction does not preserve Go restorer")
746 }
747 }
748
749 t.Parallel()
750 output := runTestProg(t, "testprogcgo", "EINTR")
751 want := "OK\n"
752 if output != want {
753 t.Fatalf("want %s, got %s\n", want, output)
754 }
755 }
756
757
758 func TestNeedmDeadlock(t *testing.T) {
759 switch runtime.GOOS {
760 case "plan9", "windows":
761 t.Skipf("no signals on %s", runtime.GOOS)
762 }
763 output := runTestProg(t, "testprogcgo", "NeedmDeadlock")
764 want := "OK\n"
765 if output != want {
766 t.Fatalf("want %s, got %s\n", want, output)
767 }
768 }
769
770 func TestCgoNoCallback(t *testing.T) {
771 got := runTestProg(t, "testprogcgo", "CgoNoCallback")
772 want := "function marked with #cgo nocallback called back into Go"
773 if !strings.Contains(got, want) {
774 t.Fatalf("did not see %q in output:\n%s", want, got)
775 }
776 }
777
778 func TestCgoNoEscape(t *testing.T) {
779 got := runTestProg(t, "testprogcgo", "CgoNoEscape")
780 want := "OK\n"
781 if got != want {
782 t.Fatalf("want %s, got %s\n", want, got)
783 }
784 }
785
786
787 func TestCgoEscapeWithMultiplePointers(t *testing.T) {
788 got := runTestProg(t, "testprogcgo", "CgoEscapeWithMultiplePointers")
789 want := "OK\n"
790 if got != want {
791 t.Fatalf("output is %s; want %s", got, want)
792 }
793 }
794
795 func TestCgoTracebackGoroutineProfile(t *testing.T) {
796 output := runTestProg(t, "testprogcgo", "GoroutineProfile")
797 want := "OK\n"
798 if output != want {
799 t.Fatalf("want %s, got %s\n", want, output)
800 }
801 }
802
803 func TestCgoSigfwd(t *testing.T) {
804 t.Parallel()
805 if !goos.IsUnix {
806 t.Skipf("no signals on %s", runtime.GOOS)
807 }
808
809 got := runTestProg(t, "testprogcgo", "CgoSigfwd", "GO_TEST_CGOSIGFWD=1")
810 if want := "OK\n"; got != want {
811 t.Fatalf("expected %q, but got:\n%s", want, got)
812 }
813 }
814
815 func TestDestructorCallback(t *testing.T) {
816 t.Parallel()
817 got := runTestProg(t, "testprogcgo", "DestructorCallback")
818 if want := "OK\n"; got != want {
819 t.Errorf("expected %q, but got:\n%s", want, got)
820 }
821 }
822
823 func TestDestructorCallbackRace(t *testing.T) {
824
825
826 if testing.Short() {
827 t.Skip("skipping test in -short mode")
828 }
829
830 if !platform.RaceDetectorSupported(runtime.GOOS, runtime.GOARCH) {
831 t.Skipf("skipping on %s/%s because race detector not supported", runtime.GOOS, runtime.GOARCH)
832 }
833
834 t.Parallel()
835
836 exe, err := buildTestProg(t, "testprogcgo", "-race")
837 if err != nil {
838 t.Fatal(err)
839 }
840
841 got, err := testenv.CleanCmdEnv(exec.Command(exe, "DestructorCallback")).CombinedOutput()
842 if err != nil {
843 t.Fatal(err)
844 }
845
846 if want := "OK\n"; string(got) != want {
847 t.Errorf("expected %q, but got:\n%s", want, got)
848 }
849 }
850
851 func TestEnsureBindM(t *testing.T) {
852 t.Parallel()
853 switch runtime.GOOS {
854 case "windows", "plan9":
855 t.Skipf("skipping bindm test on %s", runtime.GOOS)
856 }
857 got := runTestProg(t, "testprogcgo", "EnsureBindM")
858 want := "OK\n"
859 if got != want {
860 t.Errorf("expected %q, got %v", want, got)
861 }
862 }
863
864 func TestStackSwitchCallback(t *testing.T) {
865 t.Parallel()
866 switch runtime.GOOS {
867 case "windows", "plan9", "android", "ios", "openbsd":
868 t.Skipf("skipping test on %s", runtime.GOOS)
869 }
870 got := runTestProg(t, "testprogcgo", "StackSwitchCallback")
871 skip := "SKIP\n"
872 if got == skip {
873 t.Skip("skipping on musl/bionic libc")
874 }
875 want := "OK\n"
876 if got != want {
877 t.Errorf("expected %q, got %v", want, got)
878 }
879 }
880
881 func TestCgoToGoCallGoexit(t *testing.T) {
882 if runtime.GOOS == "plan9" || runtime.GOOS == "windows" {
883 t.Skipf("no pthreads on %s", runtime.GOOS)
884 }
885 output := runTestProg(t, "testprogcgo", "CgoToGoCallGoexit")
886 if !strings.Contains(output, "runtime.Goexit called in a thread that was not created by the Go runtime") {
887 t.Fatalf("output should contain %s, got %s", "runtime.Goexit called in a thread that was not created by the Go runtime", output)
888 }
889 }
890
View as plain text