148 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			148 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package jsoniter
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"io"
 | 
						|
	"reflect"
 | 
						|
	"unsafe"
 | 
						|
)
 | 
						|
 | 
						|
func decoderOfSlice(cfg *frozenConfig, typ reflect.Type) (ValDecoder, error) {
 | 
						|
	decoder, err := decoderOfType(cfg, typ.Elem())
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return &sliceDecoder{typ, typ.Elem(), decoder}, nil
 | 
						|
}
 | 
						|
 | 
						|
func encoderOfSlice(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 &sliceEncoder{typ, typ.Elem(), encoder}, nil
 | 
						|
}
 | 
						|
 | 
						|
type sliceEncoder struct {
 | 
						|
	sliceType   reflect.Type
 | 
						|
	elemType    reflect.Type
 | 
						|
	elemEncoder ValEncoder
 | 
						|
}
 | 
						|
 | 
						|
func (encoder *sliceEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
 | 
						|
	slice := (*sliceHeader)(ptr)
 | 
						|
	if slice.Data == nil {
 | 
						|
		stream.WriteNil()
 | 
						|
		return
 | 
						|
	}
 | 
						|
	if slice.Len == 0 {
 | 
						|
		stream.WriteEmptyArray()
 | 
						|
		return
 | 
						|
	}
 | 
						|
	stream.WriteArrayStart()
 | 
						|
	elemPtr := unsafe.Pointer(slice.Data)
 | 
						|
	encoder.elemEncoder.Encode(unsafe.Pointer(elemPtr), stream)
 | 
						|
	for i := 1; i < slice.Len; i++ {
 | 
						|
		stream.WriteMore()
 | 
						|
		elemPtr = unsafe.Pointer(uintptr(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.sliceType, stream.Error.Error())
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (encoder *sliceEncoder) EncodeInterface(val interface{}, stream *Stream) {
 | 
						|
	WriteToStream(val, stream, encoder)
 | 
						|
}
 | 
						|
 | 
						|
func (encoder *sliceEncoder) IsEmpty(ptr unsafe.Pointer) bool {
 | 
						|
	slice := (*sliceHeader)(ptr)
 | 
						|
	return slice.Len == 0
 | 
						|
}
 | 
						|
 | 
						|
type sliceDecoder struct {
 | 
						|
	sliceType   reflect.Type
 | 
						|
	elemType    reflect.Type
 | 
						|
	elemDecoder ValDecoder
 | 
						|
}
 | 
						|
 | 
						|
// sliceHeader is a safe version of SliceHeader used within this package.
 | 
						|
type sliceHeader struct {
 | 
						|
	Data unsafe.Pointer
 | 
						|
	Len  int
 | 
						|
	Cap  int
 | 
						|
}
 | 
						|
 | 
						|
func (decoder *sliceDecoder) 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.sliceType, iter.Error.Error())
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (decoder *sliceDecoder) doDecode(ptr unsafe.Pointer, iter *Iterator) {
 | 
						|
	slice := (*sliceHeader)(ptr)
 | 
						|
	if iter.ReadNil() {
 | 
						|
		slice.Len = 0
 | 
						|
		slice.Cap = 0
 | 
						|
		slice.Data = nil
 | 
						|
		return
 | 
						|
	}
 | 
						|
	reuseSlice(slice, decoder.sliceType, 4)
 | 
						|
	slice.Len = 0
 | 
						|
	offset := uintptr(0)
 | 
						|
	iter.ReadArrayCB(func(iter *Iterator) bool {
 | 
						|
		growOne(slice, decoder.sliceType, decoder.elemType)
 | 
						|
		decoder.elemDecoder.Decode(unsafe.Pointer(uintptr(slice.Data)+offset), iter)
 | 
						|
		offset += decoder.elemType.Size()
 | 
						|
		return true
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
// grow grows the slice s so that it can hold extra more values, allocating
 | 
						|
// more capacity if needed. It also returns the old and new slice lengths.
 | 
						|
func growOne(slice *sliceHeader, sliceType reflect.Type, elementType reflect.Type) {
 | 
						|
	newLen := slice.Len + 1
 | 
						|
	if newLen <= slice.Cap {
 | 
						|
		slice.Len = newLen
 | 
						|
		return
 | 
						|
	}
 | 
						|
	newCap := slice.Cap
 | 
						|
	if newCap == 0 {
 | 
						|
		newCap = 1
 | 
						|
	} else {
 | 
						|
		for newCap < newLen {
 | 
						|
			if slice.Len < 1024 {
 | 
						|
				newCap += newCap
 | 
						|
			} else {
 | 
						|
				newCap += newCap / 4
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	newVal := reflect.MakeSlice(sliceType, newLen, newCap)
 | 
						|
	dst := unsafe.Pointer(newVal.Pointer())
 | 
						|
	// copy old array into new array
 | 
						|
	originalBytesCount := slice.Len * int(elementType.Size())
 | 
						|
	srcSliceHeader := (unsafe.Pointer)(&sliceHeader{slice.Data, originalBytesCount, originalBytesCount})
 | 
						|
	dstSliceHeader := (unsafe.Pointer)(&sliceHeader{dst, originalBytesCount, originalBytesCount})
 | 
						|
	copy(*(*[]byte)(dstSliceHeader), *(*[]byte)(srcSliceHeader))
 | 
						|
	slice.Data = dst
 | 
						|
	slice.Len = newLen
 | 
						|
	slice.Cap = newCap
 | 
						|
}
 | 
						|
 | 
						|
func reuseSlice(slice *sliceHeader, sliceType reflect.Type, expectedCap int) {
 | 
						|
	if expectedCap <= slice.Cap {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	newVal := reflect.MakeSlice(sliceType, 0, expectedCap)
 | 
						|
	dst := unsafe.Pointer(newVal.Pointer())
 | 
						|
	slice.Data = dst
 | 
						|
	slice.Cap = expectedCap
 | 
						|
}
 |