Commit 123ff2eb authored by Russ Cox's avatar Russ Cox

bug177: anonymous struct fields in reflect

(reported by iant)

R=r
DELTA=50  (32 added, 12 deleted, 6 changed)
OCL=32263
CL=32385
parent 27b0c351
...@@ -627,7 +627,7 @@ ok: ...@@ -627,7 +627,7 @@ ok:
ot = duint32(s, ot, n); ot = duint32(s, ot, n);
for(t1=t->type; t1!=T; t1=t1->down) { for(t1=t->type; t1!=T; t1=t1->down) {
// ../../pkg/runtime/type.go:/structField // ../../pkg/runtime/type.go:/structField
if(t1->sym) { if(t1->sym && !t1->embedded) {
ot = dgostringptr(s, ot, t1->sym->name); ot = dgostringptr(s, ot, t1->sym->name);
if(exportname(t1->sym->name)) if(exportname(t1->sym->name))
ot = dgostringptr(s, ot, nil); ot = dgostringptr(s, ot, nil);
......
...@@ -666,7 +666,7 @@ func compileDec(wireId typeId, rt reflect.Type) (engine *decEngine, err os.Error ...@@ -666,7 +666,7 @@ func compileDec(wireId typeId, rt reflect.Type) (engine *decEngine, err os.Error
localField, present := srt.FieldByName(wireField.name); localField, present := srt.FieldByName(wireField.name);
ovfl := overflow(wireField.name); ovfl := overflow(wireField.name);
// TODO(r): anonymous names // TODO(r): anonymous names
if !present || localField.Anonymous { if !present {
op, err := decIgnoreOpFor(wireField.id); op, err := decIgnoreOpFor(wireField.id);
if err != nil { if err != nil {
return nil, err return nil, err
......
...@@ -477,7 +477,11 @@ func (t *StructType) Field(i int) (f StructField) { ...@@ -477,7 +477,11 @@ func (t *StructType) Field(i int) (f StructField) {
if p.name != nil { if p.name != nil {
f.Name = *p.name; f.Name = *p.name;
} else { } else {
f.Name = f.Type.Name(); t := f.Type;
if pt, ok := t.(*PtrType); ok {
t = pt.Elem();
}
f.Name = t.Name();
f.Anonymous = true; f.Anonymous = true;
} }
if p.pkgPath != nil { if p.pkgPath != nil {
...@@ -487,29 +491,21 @@ func (t *StructType) Field(i int) (f StructField) { ...@@ -487,29 +491,21 @@ func (t *StructType) Field(i int) (f StructField) {
f.Tag = *p.tag; f.Tag = *p.tag;
} }
f.Offset = p.offset; f.Offset = p.offset;
f.Index = i;
return; return;
} }
// FieldByName returns the field with the provided name and a boolean to indicate // FieldByName returns the field with the provided name and a boolean to indicate
// that the field was found.. // that the field was found.
func (t *StructType) FieldByName(name string) (f StructField, present bool) { func (t *StructType) FieldByName(name string) (f StructField, present bool) {
for i, p := range t.fields { for i, p := range t.fields {
if p.name == nil || *p.name != name { ff := t.Field(i);
continue; if ff.Name == name {
} f = ff;
f.Name = *p.name;
f.Type = toType(*p.typ);
if p.pkgPath != nil {
f.PkgPath = *p.pkgPath;
}
if p.tag != nil {
f.Tag = *p.tag;
}
f.Offset = p.offset;
f.Index = i;
present = true; present = true;
break; break;
} }
}
return; return;
} }
......
// $G $D/$F.go && $L $F.$A && ./$A.out
// Copyright 2009 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 main
import "fmt"
import "reflect"
type S1 struct { i int }
type S2 struct { S1 }
func main() {
typ := reflect.Typeof(S2{}).(*reflect.StructType);
f := typ.Field(0);
if f.Name != "S1" || f.Anonymous != true {
println("BUG: ", f.Name, f.Anonymous);
return;
}
f, ok := typ.FieldByName("S1");
if !ok {
println("BUG: missing S1");
return;
}
if !f.Anonymous {
println("BUG: S1 is not anonymous");
return;
}
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment