mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
cmd/cgo: discard trailing zero-sized fields in a non-empty C struct
In order to fix issue #9401 the compiler was changed to add a padding byte to any non-empty Go struct that ends in a zero-sized field. That causes the Go version of such a C struct to have a different size than the C struct, which can considerable confusion. Change cgo so that it discards any such zero-sized fields, so that the Go and C structs are the same size. This is a change from previous releases, in that it used to be possible to refer to a zero-sized trailing field (by taking its address), and with this change it no longer is. That is unfortunate, but something has to change. It seems better to visibly break programs that do this rather than to silently break programs that rely on the struct sizes being the same. Update #9401. Fixes #11925. Change-Id: I3fba3f02f11265b3c41d68616f79dedb05b81225 Reviewed-on: https://go-review.googlesource.com/12864 Reviewed-by: Russ Cox <rsc@golang.org>
This commit is contained in:
parent
c9d2c7f0d2
commit
7904946eeb
4 changed files with 65 additions and 6 deletions
|
|
@ -64,5 +64,6 @@ func TestReturnAfterGrowFromGo(t *testing.T) { testReturnAfterGrowFromGo(t) }
|
||||||
func Test9026(t *testing.T) { test9026(t) }
|
func Test9026(t *testing.T) { test9026(t) }
|
||||||
func Test9557(t *testing.T) { test9557(t) }
|
func Test9557(t *testing.T) { test9557(t) }
|
||||||
func Test10303(t *testing.T) { test10303(t, 10) }
|
func Test10303(t *testing.T) { test10303(t, 10) }
|
||||||
|
func Test11925(t *testing.T) { test11925(t) }
|
||||||
|
|
||||||
func BenchmarkCgoCall(b *testing.B) { benchCgoCall(b) }
|
func BenchmarkCgoCall(b *testing.B) { benchCgoCall(b) }
|
||||||
|
|
|
||||||
37
misc/cgo/test/issue11925.go
Normal file
37
misc/cgo/test/issue11925.go
Normal file
|
|
@ -0,0 +1,37 @@
|
||||||
|
// 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 11925. Structs with zero-length trailing fields are now
|
||||||
|
// padded by the Go compiler.
|
||||||
|
|
||||||
|
package cgotest
|
||||||
|
|
||||||
|
/*
|
||||||
|
struct a11925 {
|
||||||
|
int i;
|
||||||
|
char a[0];
|
||||||
|
char b[0];
|
||||||
|
};
|
||||||
|
|
||||||
|
struct b11925 {
|
||||||
|
int i;
|
||||||
|
char a[0];
|
||||||
|
char b[];
|
||||||
|
};
|
||||||
|
*/
|
||||||
|
import "C"
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
func test11925(t *testing.T) {
|
||||||
|
if C.sizeof_struct_a11925 != unsafe.Sizeof(C.struct_a11925{}) {
|
||||||
|
t.Errorf("size of a changed: C %d, Go %d", C.sizeof_struct_a11925, unsafe.Sizeof(C.struct_a11925{}))
|
||||||
|
}
|
||||||
|
if C.sizeof_struct_b11925 != unsafe.Sizeof(C.struct_b11925{}) {
|
||||||
|
t.Errorf("size of b changed: C %d, Go %d", C.sizeof_struct_b11925, unsafe.Sizeof(C.struct_b11925{}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -20,6 +20,7 @@ struct issue8428two {
|
||||||
void *p;
|
void *p;
|
||||||
char b;
|
char b;
|
||||||
char rest[0];
|
char rest[0];
|
||||||
|
char pad;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct issue8428three {
|
struct issue8428three {
|
||||||
|
|
@ -34,8 +35,10 @@ import "C"
|
||||||
import "unsafe"
|
import "unsafe"
|
||||||
|
|
||||||
var _ = C.struct_issue8428one{
|
var _ = C.struct_issue8428one{
|
||||||
b: C.char(0),
|
b: C.char(0),
|
||||||
rest: [0]C.char{},
|
// The trailing rest field is not available in cgo.
|
||||||
|
// See issue 11925.
|
||||||
|
// rest: [0]C.char{},
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ = C.struct_issue8428two{
|
var _ = C.struct_issue8428two{
|
||||||
|
|
|
||||||
|
|
@ -1544,11 +1544,13 @@ func (c *typeConv) intExpr(n int64) ast.Expr {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add padding of given size to fld.
|
// Add padding of given size to fld.
|
||||||
func (c *typeConv) pad(fld []*ast.Field, size int64) []*ast.Field {
|
func (c *typeConv) pad(fld []*ast.Field, sizes []int64, size int64) ([]*ast.Field, []int64) {
|
||||||
n := len(fld)
|
n := len(fld)
|
||||||
fld = fld[0 : n+1]
|
fld = fld[0 : n+1]
|
||||||
fld[n] = &ast.Field{Names: []*ast.Ident{c.Ident("_")}, Type: c.Opaque(size)}
|
fld[n] = &ast.Field{Names: []*ast.Ident{c.Ident("_")}, Type: c.Opaque(size)}
|
||||||
return fld
|
sizes = sizes[0 : n+1]
|
||||||
|
sizes[n] = size
|
||||||
|
return fld, sizes
|
||||||
}
|
}
|
||||||
|
|
||||||
// Struct conversion: return Go and (gc) C syntax for type.
|
// Struct conversion: return Go and (gc) C syntax for type.
|
||||||
|
|
@ -1559,6 +1561,7 @@ func (c *typeConv) Struct(dt *dwarf.StructType, pos token.Pos) (expr *ast.Struct
|
||||||
var buf bytes.Buffer
|
var buf bytes.Buffer
|
||||||
buf.WriteString("struct {")
|
buf.WriteString("struct {")
|
||||||
fld := make([]*ast.Field, 0, 2*len(dt.Field)+1) // enough for padding around every field
|
fld := make([]*ast.Field, 0, 2*len(dt.Field)+1) // enough for padding around every field
|
||||||
|
sizes := make([]int64, 0, 2*len(dt.Field)+1)
|
||||||
off := int64(0)
|
off := int64(0)
|
||||||
|
|
||||||
// Rename struct fields that happen to be named Go keywords into
|
// Rename struct fields that happen to be named Go keywords into
|
||||||
|
|
@ -1594,7 +1597,7 @@ func (c *typeConv) Struct(dt *dwarf.StructType, pos token.Pos) (expr *ast.Struct
|
||||||
anon := 0
|
anon := 0
|
||||||
for _, f := range dt.Field {
|
for _, f := range dt.Field {
|
||||||
if f.ByteOffset > off {
|
if f.ByteOffset > off {
|
||||||
fld = c.pad(fld, f.ByteOffset-off)
|
fld, sizes = c.pad(fld, sizes, f.ByteOffset-off)
|
||||||
off = f.ByteOffset
|
off = f.ByteOffset
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -1652,6 +1655,8 @@ func (c *typeConv) Struct(dt *dwarf.StructType, pos token.Pos) (expr *ast.Struct
|
||||||
ident[name] = name
|
ident[name] = name
|
||||||
}
|
}
|
||||||
fld[n] = &ast.Field{Names: []*ast.Ident{c.Ident(ident[name])}, Type: tgo}
|
fld[n] = &ast.Field{Names: []*ast.Ident{c.Ident(ident[name])}, Type: tgo}
|
||||||
|
sizes = sizes[0 : n+1]
|
||||||
|
sizes[n] = size
|
||||||
off += size
|
off += size
|
||||||
buf.WriteString(t.C.String())
|
buf.WriteString(t.C.String())
|
||||||
buf.WriteString(" ")
|
buf.WriteString(" ")
|
||||||
|
|
@ -1662,9 +1667,22 @@ func (c *typeConv) Struct(dt *dwarf.StructType, pos token.Pos) (expr *ast.Struct
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if off < dt.ByteSize {
|
if off < dt.ByteSize {
|
||||||
fld = c.pad(fld, dt.ByteSize-off)
|
fld, sizes = c.pad(fld, sizes, dt.ByteSize-off)
|
||||||
off = dt.ByteSize
|
off = dt.ByteSize
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If the last field in a non-zero-sized struct is zero-sized
|
||||||
|
// the compiler is going to pad it by one (see issue 9401).
|
||||||
|
// We can't permit that, because then the size of the Go
|
||||||
|
// struct will not be the same as the size of the C struct.
|
||||||
|
// Our only option in such a case is to remove the field,
|
||||||
|
// which means that it can not be referenced from Go.
|
||||||
|
for off > 0 && sizes[len(sizes)-1] == 0 {
|
||||||
|
n := len(sizes)
|
||||||
|
fld = fld[0 : n-1]
|
||||||
|
sizes = sizes[0 : n-1]
|
||||||
|
}
|
||||||
|
|
||||||
if off != dt.ByteSize {
|
if off != dt.ByteSize {
|
||||||
fatalf("%s: struct size calculation error off=%d bytesize=%d", lineno(pos), off, dt.ByteSize)
|
fatalf("%s: struct size calculation error off=%d bytesize=%d", lineno(pos), off, dt.ByteSize)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue