mirror of
https://github.com/json-iterator/go.git
synced 2024-11-24 08:22:14 +02:00
98 lines
2.5 KiB
Go
98 lines
2.5 KiB
Go
package jsoniter
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"reflect"
|
|
"unsafe"
|
|
)
|
|
|
|
func decoderOfArray(cfg *frozenConfig, typ reflect.Type) (ValDecoder, error) {
|
|
decoder, err := decoderOfType(cfg, typ.Elem())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &arrayDecoder{typ, typ.Elem(), decoder}, nil
|
|
}
|
|
|
|
func encoderOfArray(cfg *frozenConfig, typ reflect.Type) (ValEncoder, error) {
|
|
encoder, err := encoderOfType(cfg, typ.Elem())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if typ.Elem().Kind() == reflect.Map {
|
|
encoder = &optionalEncoder{encoder}
|
|
}
|
|
return &arrayEncoder{typ, typ.Elem(), encoder}, nil
|
|
}
|
|
|
|
type arrayEncoder struct {
|
|
arrayType reflect.Type
|
|
elemType reflect.Type
|
|
elemEncoder ValEncoder
|
|
}
|
|
|
|
func (encoder *arrayEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
|
|
stream.WriteArrayStart()
|
|
elemPtr := uintptr(ptr)
|
|
encoder.elemEncoder.Encode(unsafe.Pointer(elemPtr), stream)
|
|
for i := 1; i < encoder.arrayType.Len(); i++ {
|
|
stream.WriteMore()
|
|
elemPtr += encoder.elemType.Size()
|
|
encoder.elemEncoder.Encode(unsafe.Pointer(elemPtr), stream)
|
|
}
|
|
stream.WriteArrayEnd()
|
|
if stream.Error != nil && stream.Error != io.EOF {
|
|
stream.Error = fmt.Errorf("%v: %s", encoder.arrayType, stream.Error.Error())
|
|
}
|
|
}
|
|
|
|
func (encoder *arrayEncoder) EncodeInterface(val interface{}, stream *Stream) {
|
|
// special optimization for interface{}
|
|
e := (*emptyInterface)(unsafe.Pointer(&val))
|
|
if e.word == nil {
|
|
stream.WriteArrayStart()
|
|
stream.WriteNil()
|
|
stream.WriteArrayEnd()
|
|
return
|
|
}
|
|
elemType := encoder.arrayType.Elem()
|
|
if encoder.arrayType.Len() == 1 && (elemType.Kind() == reflect.Ptr || elemType.Kind() == reflect.Map) {
|
|
ptr := uintptr(e.word)
|
|
e.word = unsafe.Pointer(&ptr)
|
|
}
|
|
if reflect.TypeOf(val).Kind() == reflect.Ptr {
|
|
encoder.Encode(unsafe.Pointer(&e.word), stream)
|
|
} else {
|
|
encoder.Encode(e.word, stream)
|
|
}
|
|
}
|
|
|
|
func (encoder *arrayEncoder) IsEmpty(ptr unsafe.Pointer) bool {
|
|
return false
|
|
}
|
|
|
|
type arrayDecoder struct {
|
|
arrayType reflect.Type
|
|
elemType reflect.Type
|
|
elemDecoder ValDecoder
|
|
}
|
|
|
|
func (decoder *arrayDecoder) Decode(ptr unsafe.Pointer, iter *Iterator) {
|
|
decoder.doDecode(ptr, iter)
|
|
if iter.Error != nil && iter.Error != io.EOF {
|
|
iter.Error = fmt.Errorf("%v: %s", decoder.arrayType, iter.Error.Error())
|
|
}
|
|
}
|
|
|
|
func (decoder *arrayDecoder) doDecode(ptr unsafe.Pointer, iter *Iterator) {
|
|
offset := uintptr(0)
|
|
for ; iter.ReadArray(); offset += decoder.elemType.Size() {
|
|
if offset < decoder.arrayType.Size() {
|
|
decoder.elemDecoder.Decode(unsafe.Pointer(uintptr(ptr)+offset), iter)
|
|
} else {
|
|
iter.Skip()
|
|
}
|
|
}
|
|
}
|