Skip to content

Commit

Permalink
feat(reflect): supports nocopy option (#68)
Browse files Browse the repository at this point in the history
  • Loading branch information
xiaost authored Nov 28, 2024
1 parent 1c153bd commit 7402993
Show file tree
Hide file tree
Showing 4 changed files with 115 additions and 9 deletions.
56 changes: 49 additions & 7 deletions internal/reflect/decoder.go
Original file line number Diff line number Diff line change
Expand Up @@ -125,7 +125,13 @@ func (d *tDecoder) Decode(b []byte, base unsafe.Pointer, sd *structDesc, maxdept
if t.FixedSize > 0 {
i += decodeFixedSizeTypes(t.T, b[i:], p)
} else {
n, err := d.decodeType(t, b[i:], p, maxdepth-1)
var n int
var err error
if f.NoCopy {
n, err = decodeStringNoCopy(t, b[i:], p)
} else {
n, err = d.decodeType(t, b[i:], p, maxdepth-1)
}
if err != nil {
return i, fmt.Errorf("decode field %d of struct %s err: %w", fid, sd.rt.String(), err)
}
Expand Down Expand Up @@ -168,6 +174,39 @@ func decodeFixedSizeTypes(t ttype, b []byte, p unsafe.Pointer) int {
}
}

func decodeStringNoCopy(t *tType, b []byte, p unsafe.Pointer) (i int, err error) {
l := int(binary.BigEndian.Uint32(b))
if l < 0 {
err = errNegativeSize
return
}
i += 4
if l == 0 {
if t.Tag == defs.T_binary {
*(*[]byte)(p) = []byte{}
} else {
*(*string)(p) = ""
}
return
}

// assert len, panic if []byte shorter than expected.
_ = b[i+l-1]

if t.Tag == defs.T_binary {
h := (*sliceHeader)(p)
h.Data = uintptr(unsafe.Pointer(&b[i]))
h.Len = l
h.Cap = l
} else { // convert to str
h := (*stringHeader)(p)
h.Data = uintptr(unsafe.Pointer(&b[i]))
h.Len = l
}
i += l
return
}

func (d *tDecoder) decodeType(t *tType, b []byte, p unsafe.Pointer, maxdepth int) (int, error) {
if maxdepth == 0 {
return 0, errDepthLimitExceeded
Expand All @@ -184,12 +223,16 @@ func (d *tDecoder) decodeType(t *tType, b []byte, p unsafe.Pointer, maxdepth int
i := 4
if l == 0 {
if t.Tag == defs.T_binary {
*(*sliceHeader)(p) = zeroSliceHeader
*(*[]byte)(p) = []byte{}
} else {
*(*stringHeader)(p) = zeroStrHeader
*(*string)(p) = ""
}
return i, nil
}

// assert len, panic if []byte shorter than expected.
_ = b[i+l-1]

x := d.Malloc(l, 1, 0)
if t.Tag == defs.T_binary {
h := (*sliceHeader)(p)
Expand Down Expand Up @@ -304,15 +347,14 @@ func (d *tDecoder) decodeType(t *tType, b []byte, p unsafe.Pointer, maxdepth int

// decode list
h := (*sliceHeader)(p) // update the slice field
h.Data = 0
h.Len = l
h.Cap = l
if l <= 0 {
*(*sliceHeader)(p) = zeroSliceHeader
h.Zero()
return i, nil
}
x := d.Malloc(l*et.Size, et.Align, et.MallocAbiType) // malloc for slice. make([]Type, l, l)
h.Data = uintptr(x)
h.Len = l
h.Cap = l

// pre-allocate space for elements if they're pointers
// like
Expand Down
46 changes: 46 additions & 0 deletions internal/reflect/decoder_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
package reflect

import (
"bytes"
"math"
"math/rand"
"testing"
Expand Down Expand Up @@ -358,3 +359,48 @@ func TestDecodeUnknownFields(t *testing.T) {
assert.Equal(t, sz, len(testb))
assert.Equal(t, testb, p._unknownFields)
}

func TestDecodeNoCopy(t *testing.T) {
type Msg struct {
A string `frugal:"1,default,string,nocopy"`
B string `frugal:"2,default,string"`
C []byte `frugal:"3,default,binary,nocopy"`
}

strA := "strA"
strB := "strB"
strC := "strC"
sz := 3*(fieldHeaderLen+strHeaderLen) + len(strA) + len(strB) + len(strC)
b := make([]byte, 0, sz)
b = appendStringField(b, 1, strA)
b = appendStringField(b, 2, strB)
b = appendStringField(b, 3, strC)
b = append(b, 0) // tSTOP

p := &Msg{}
_, err := Decode(b, p)
require.NoError(t, err)

assert.Equal(t, strA, p.A)
assert.Equal(t, strB, p.B)
assert.Equal(t, strC, string(p.C))

// update original buffer
// coz it's nocopy, the fields of p will be changed implicitly as well.
// update strA -> xtrA, strB -> xtrB, strC -> xtrC
for _, s := range []string{strA, strB, strC} {
i := bytes.Index(b, []byte(s))
b[i] = 'x'
}
assert.Equal(t, "xtrA", p.A)
assert.Equal(t, "strB", p.B) // p.B has no `nocopy` option
assert.Equal(t, "xtrC", string(p.C))

type Msg2 struct {
A int32 `frugal:"4,default,i32,nocopy"`
}
p2 := &Msg2{}
_, err = Decode(b, p2)
require.Error(t, err)
_ = p2
}
8 changes: 6 additions & 2 deletions internal/reflect/desc.go
Original file line number Diff line number Diff line change
Expand Up @@ -221,10 +221,10 @@ type tField struct {
Offset uintptr
Type *tType

Opts defs.Options
Spec defs.Requiredness
Default unsafe.Pointer

NoCopy bool
CanSkipEncodeIfNil bool
CanSkipIfDefault bool
}
Expand Down Expand Up @@ -263,11 +263,15 @@ func (f *tField) fromDefsField(x defs.Field) {
f.ID = x.ID
f.Offset = uintptr(x.F)
f.Type = newTType(x.Type)
f.Opts = x.Opts
f.Spec = x.Spec

t := f.Type

f.NoCopy = (x.Opts & defs.NoCopy) != 0
if f.NoCopy && f.Type.WT != tSTRING {
// never goes here, defs will check the tag
panic("[BUG] nocopy on non-STRING type")
}
// for map or slice, t.IsPointer() is false,
// but we can consider the types as pointer as per lang spec
// for defs.T_binary, actually it's []byte, like tLIST
Expand Down
14 changes: 14 additions & 0 deletions internal/reflect/hack.go
Original file line number Diff line number Diff line change
Expand Up @@ -239,5 +239,19 @@ func (h *sliceHeader) UnsafePointer() unsafe.Pointer {
return *(*unsafe.Pointer)(unsafe.Pointer(h))
}

var (
emptyslice = make([]byte, 0)

// for slice, Data should points to zerobase var in `runtime`
// so that it can represent as []type{} instead of []type(nil)
zerobase = ((*sliceHeader)(unsafe.Pointer(&emptyslice))).Data
)

func (h *sliceHeader) Zero() {
h.Len = 0
h.Cap = 0
h.Data = zerobase
}

//go:linkname mallocgc runtime.mallocgc
func mallocgc(size uintptr, typ unsafe.Pointer, needzero bool) unsafe.Pointer

0 comments on commit 7402993

Please sign in to comment.