| // Copyright 2015 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. |
| |
| // Issue 10303. Pointers passed to C were not marked as escaping (bug in cgo). |
| |
| package cgotest |
| |
| import "runtime" |
| |
| /* |
| typedef int *intptr; |
| |
| void setintstar(int *x) { |
| *x = 1; |
| } |
| |
| void setintptr(intptr x) { |
| *x = 1; |
| } |
| |
| void setvoidptr(void *x) { |
| *(int*)x = 1; |
| } |
| |
| typedef struct Struct Struct; |
| struct Struct { |
| int *P; |
| }; |
| |
| void setstruct(Struct s) { |
| *s.P = 1; |
| } |
| |
| */ |
| import "C" |
| |
| import ( |
| "testing" |
| "unsafe" |
| ) |
| |
| func test10303(t *testing.T, n int) { |
| if runtime.Compiler == "gccgo" { |
| t.Skip("gccgo permits C pointers on the stack") |
| } |
| |
| // Run at a few different stack depths just to avoid an unlucky pass |
| // due to variables ending up on different pages. |
| if n > 0 { |
| test10303(t, n-1) |
| } |
| if t.Failed() { |
| return |
| } |
| var x, y, z, v, si C.int |
| var s C.Struct |
| C.setintstar(&x) |
| C.setintptr(&y) |
| C.setvoidptr(unsafe.Pointer(&v)) |
| s.P = &si |
| C.setstruct(s) |
| |
| if uintptr(unsafe.Pointer(&x))&^0xfff == uintptr(unsafe.Pointer(&z))&^0xfff { |
| t.Error("C int* argument on stack") |
| } |
| if uintptr(unsafe.Pointer(&y))&^0xfff == uintptr(unsafe.Pointer(&z))&^0xfff { |
| t.Error("C intptr argument on stack") |
| } |
| if uintptr(unsafe.Pointer(&v))&^0xfff == uintptr(unsafe.Pointer(&z))&^0xfff { |
| t.Error("C void* argument on stack") |
| } |
| if uintptr(unsafe.Pointer(&si))&^0xfff == uintptr(unsafe.Pointer(&z))&^0xfff { |
| t.Error("C struct field pointer on stack") |
| } |
| } |