pools.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. /*
  2. * Copyright 2021 ByteDance Inc.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package jitdec
  17. import (
  18. `sync`
  19. `unsafe`
  20. `github.com/bytedance/sonic/internal/caching`
  21. `github.com/bytedance/sonic/internal/native/types`
  22. `github.com/bytedance/sonic/internal/rt`
  23. )
  24. const (
  25. _MinSlice = 2
  26. _MaxStack = 4096 // 4k slots
  27. _MaxStackBytes = _MaxStack * _PtrBytes
  28. _MaxDigitNums = types.MaxDigitNums // used in atof fallback algorithm
  29. )
  30. const (
  31. _PtrBytes = _PTR_SIZE / 8
  32. _FsmOffset = (_MaxStack + 1) * _PtrBytes
  33. _DbufOffset = _FsmOffset + int64(unsafe.Sizeof(types.StateMachine{})) + types.MAX_RECURSE * _PtrBytes
  34. _EpOffset = _DbufOffset + _MaxDigitNums
  35. _StackSize = unsafe.Sizeof(_Stack{})
  36. )
  37. var (
  38. stackPool = sync.Pool{}
  39. valueCache = []unsafe.Pointer(nil)
  40. fieldCache = []*caching.FieldMap(nil)
  41. fieldCacheMux = sync.Mutex{}
  42. programCache = caching.CreateProgramCache()
  43. )
  44. type _Stack struct {
  45. sp uintptr
  46. sb [_MaxStack]unsafe.Pointer
  47. mm types.StateMachine
  48. vp [types.MAX_RECURSE]unsafe.Pointer
  49. dp [_MaxDigitNums]byte
  50. ep unsafe.Pointer
  51. }
  52. type _Decoder func(
  53. s string,
  54. i int,
  55. vp unsafe.Pointer,
  56. sb *_Stack,
  57. fv uint64,
  58. sv string, // DO NOT pass value to this argument, since it is only used for local _VAR_sv
  59. vk unsafe.Pointer, // DO NOT pass value to this argument, since it is only used for local _VAR_vk
  60. ) (int, error)
  61. var _KeepAlive struct {
  62. s string
  63. i int
  64. vp unsafe.Pointer
  65. sb *_Stack
  66. fv uint64
  67. sv string
  68. vk unsafe.Pointer
  69. ret int
  70. err error
  71. frame_decoder [_FP_offs]byte
  72. frame_generic [_VD_offs]byte
  73. }
  74. var (
  75. argPtrs = []bool{true, false, false, true, true, false, true, false, true}
  76. localPtrs = []bool{}
  77. )
  78. var (
  79. argPtrs_generic = []bool{true}
  80. localPtrs_generic = []bool{}
  81. )
  82. func newStack() *_Stack {
  83. if ret := stackPool.Get(); ret == nil {
  84. return new(_Stack)
  85. } else {
  86. return ret.(*_Stack)
  87. }
  88. }
  89. func resetStack(p *_Stack) {
  90. rt.MemclrNoHeapPointers(unsafe.Pointer(p), _StackSize)
  91. }
  92. func freeStack(p *_Stack) {
  93. p.sp = 0
  94. stackPool.Put(p)
  95. }
  96. func freezeValue(v unsafe.Pointer) uintptr {
  97. valueCache = append(valueCache, v)
  98. return uintptr(v)
  99. }
  100. func freezeFields(v *caching.FieldMap) int64 {
  101. fieldCacheMux.Lock()
  102. fieldCache = append(fieldCache, v)
  103. fieldCacheMux.Unlock()
  104. return referenceFields(v)
  105. }
  106. func referenceFields(v *caching.FieldMap) int64 {
  107. return int64(uintptr(unsafe.Pointer(v)))
  108. }
  109. func makeDecoder(vt *rt.GoType, _ ...interface{}) (interface{}, error) {
  110. if pp, err := newCompiler().compile(vt.Pack()); err != nil {
  111. return nil, err
  112. } else {
  113. return newAssembler(pp).Load(), nil
  114. }
  115. }
  116. func findOrCompile(vt *rt.GoType) (_Decoder, error) {
  117. if val := programCache.Get(vt); val != nil {
  118. return val.(_Decoder), nil
  119. } else if ret, err := programCache.Compute(vt, makeDecoder); err == nil {
  120. return ret.(_Decoder), nil
  121. } else {
  122. return nil, err
  123. }
  124. }