142 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
		
			Vendored
		
	
	
	
			
		
		
	
	
			142 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
		
			Vendored
		
	
	
	
package encoder
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"sync"
 | 
						|
	"unsafe"
 | 
						|
 | 
						|
	"github.com/goccy/go-json/internal/runtime"
 | 
						|
)
 | 
						|
 | 
						|
type compileContext struct {
 | 
						|
	typ                      *runtime.Type
 | 
						|
	opcodeIndex              uint32
 | 
						|
	ptrIndex                 int
 | 
						|
	indent                   uint32
 | 
						|
	escapeKey                bool
 | 
						|
	structTypeToCompiledCode map[uintptr]*CompiledCode
 | 
						|
 | 
						|
	parent *compileContext
 | 
						|
}
 | 
						|
 | 
						|
func (c *compileContext) context() *compileContext {
 | 
						|
	return &compileContext{
 | 
						|
		typ:                      c.typ,
 | 
						|
		opcodeIndex:              c.opcodeIndex,
 | 
						|
		ptrIndex:                 c.ptrIndex,
 | 
						|
		indent:                   c.indent,
 | 
						|
		escapeKey:                c.escapeKey,
 | 
						|
		structTypeToCompiledCode: c.structTypeToCompiledCode,
 | 
						|
		parent:                   c,
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (c *compileContext) withType(typ *runtime.Type) *compileContext {
 | 
						|
	ctx := c.context()
 | 
						|
	ctx.typ = typ
 | 
						|
	return ctx
 | 
						|
}
 | 
						|
 | 
						|
func (c *compileContext) incIndent() *compileContext {
 | 
						|
	ctx := c.context()
 | 
						|
	ctx.indent++
 | 
						|
	return ctx
 | 
						|
}
 | 
						|
 | 
						|
func (c *compileContext) decIndent() *compileContext {
 | 
						|
	ctx := c.context()
 | 
						|
	ctx.indent--
 | 
						|
	return ctx
 | 
						|
}
 | 
						|
 | 
						|
func (c *compileContext) incIndex() {
 | 
						|
	c.incOpcodeIndex()
 | 
						|
	c.incPtrIndex()
 | 
						|
}
 | 
						|
 | 
						|
func (c *compileContext) decIndex() {
 | 
						|
	c.decOpcodeIndex()
 | 
						|
	c.decPtrIndex()
 | 
						|
}
 | 
						|
 | 
						|
func (c *compileContext) incOpcodeIndex() {
 | 
						|
	c.opcodeIndex++
 | 
						|
	if c.parent != nil {
 | 
						|
		c.parent.incOpcodeIndex()
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (c *compileContext) decOpcodeIndex() {
 | 
						|
	c.opcodeIndex--
 | 
						|
	if c.parent != nil {
 | 
						|
		c.parent.decOpcodeIndex()
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (c *compileContext) incPtrIndex() {
 | 
						|
	c.ptrIndex++
 | 
						|
	if c.parent != nil {
 | 
						|
		c.parent.incPtrIndex()
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (c *compileContext) decPtrIndex() {
 | 
						|
	c.ptrIndex--
 | 
						|
	if c.parent != nil {
 | 
						|
		c.parent.decPtrIndex()
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
const (
 | 
						|
	bufSize = 1024
 | 
						|
)
 | 
						|
 | 
						|
var (
 | 
						|
	runtimeContextPool = sync.Pool{
 | 
						|
		New: func() interface{} {
 | 
						|
			return &RuntimeContext{
 | 
						|
				Buf:      make([]byte, 0, bufSize),
 | 
						|
				Ptrs:     make([]uintptr, 128),
 | 
						|
				KeepRefs: make([]unsafe.Pointer, 0, 8),
 | 
						|
				Option:   &Option{},
 | 
						|
			}
 | 
						|
		},
 | 
						|
	}
 | 
						|
)
 | 
						|
 | 
						|
type RuntimeContext struct {
 | 
						|
	Context    context.Context
 | 
						|
	Buf        []byte
 | 
						|
	MarshalBuf []byte
 | 
						|
	Ptrs       []uintptr
 | 
						|
	KeepRefs   []unsafe.Pointer
 | 
						|
	SeenPtr    []uintptr
 | 
						|
	BaseIndent uint32
 | 
						|
	Prefix     []byte
 | 
						|
	IndentStr  []byte
 | 
						|
	Option     *Option
 | 
						|
}
 | 
						|
 | 
						|
func (c *RuntimeContext) Init(p uintptr, codelen int) {
 | 
						|
	if len(c.Ptrs) < codelen {
 | 
						|
		c.Ptrs = make([]uintptr, codelen)
 | 
						|
	}
 | 
						|
	c.Ptrs[0] = p
 | 
						|
	c.KeepRefs = c.KeepRefs[:0]
 | 
						|
	c.SeenPtr = c.SeenPtr[:0]
 | 
						|
	c.BaseIndent = 0
 | 
						|
}
 | 
						|
 | 
						|
func (c *RuntimeContext) Ptr() uintptr {
 | 
						|
	header := (*runtime.SliceHeader)(unsafe.Pointer(&c.Ptrs))
 | 
						|
	return uintptr(header.Data)
 | 
						|
}
 | 
						|
 | 
						|
func TakeRuntimeContext() *RuntimeContext {
 | 
						|
	return runtimeContextPool.Get().(*RuntimeContext)
 | 
						|
}
 | 
						|
 | 
						|
func ReleaseRuntimeContext(ctx *RuntimeContext) {
 | 
						|
	runtimeContextPool.Put(ctx)
 | 
						|
}
 |