github.com/bytedance/sonic@v1.11.7-0.20240517092252-d2edb31b167b/internal/decoder/asm_stubs_amd64_go117.go (about) 1 // +build go1.17,!go1.21 2 3 // Copyright 2023 CloudWeGo Authors 4 // 5 // Licensed under the Apache License, Version 2.0 (the "License"); 6 // you may not use this file except in compliance with the License. 7 // You may obtain a copy of the License at 8 // 9 // http://www.apache.org/licenses/LICENSE-2.0 10 // 11 // Unless required by applicable law or agreed to in writing, software 12 // distributed under the License is distributed on an "AS IS" BASIS, 13 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14 // See the License for the specific language governing permissions and 15 // limitations under the License. 16 17 package decoder 18 19 import ( 20 `strconv` 21 `unsafe` 22 23 `github.com/bytedance/sonic/internal/jit` 24 `github.com/twitchyliquid64/golang-asm/obj` 25 `github.com/twitchyliquid64/golang-asm/obj/x86` 26 ) 27 28 //go:linkname _runtime_writeBarrier runtime.writeBarrier 29 var _runtime_writeBarrier uintptr 30 31 //go:linkname gcWriteBarrierAX runtime.gcWriteBarrier 32 func gcWriteBarrierAX() 33 34 var ( 35 _V_writeBarrier = jit.Imm(int64(uintptr(unsafe.Pointer(&_runtime_writeBarrier)))) 36 37 _F_gcWriteBarrierAX = jit.Func(gcWriteBarrierAX) 38 ) 39 40 func (self *_Assembler) WritePtrAX(i int, rec obj.Addr, saveDI bool) { 41 self.Emit("MOVQ", _V_writeBarrier, _R9) 42 self.Emit("CMPL", jit.Ptr(_R9, 0), jit.Imm(0)) 43 self.Sjmp("JE", "_no_writeBarrier" + strconv.Itoa(i) + "_{n}") 44 if saveDI { 45 self.save(_DI) 46 } 47 self.Emit("LEAQ", rec, _DI) 48 self.call(_F_gcWriteBarrierAX) 49 if saveDI { 50 self.load(_DI) 51 } 52 self.Sjmp("JMP", "_end_writeBarrier" + strconv.Itoa(i) + "_{n}") 53 self.Link("_no_writeBarrier" + strconv.Itoa(i) + "_{n}") 54 self.Emit("MOVQ", _AX, rec) 55 self.Link("_end_writeBarrier" + strconv.Itoa(i) + "_{n}") 56 } 57 58 func (self *_Assembler) WriteRecNotAX(i int, ptr obj.Addr, rec obj.Addr, saveDI bool, saveAX bool) { 59 if rec.Reg == x86.REG_AX || rec.Index == x86.REG_AX { 60 panic("rec contains AX!") 61 } 62 self.Emit("MOVQ", _V_writeBarrier, _R9) 63 self.Emit("CMPL", jit.Ptr(_R9, 0), jit.Imm(0)) 64 self.Sjmp("JE", "_no_writeBarrier" + strconv.Itoa(i) + "_{n}") 65 if saveAX { 66 self.Emit("XCHGQ", ptr, _AX) 67 } else { 68 self.Emit("MOVQ", ptr, _AX) 69 } 70 if saveDI { 71 self.save(_DI) 72 } 73 self.Emit("LEAQ", rec, _DI) 74 self.call(_F_gcWriteBarrierAX) 75 if saveDI { 76 self.load(_DI) 77 } 78 if saveAX { 79 self.Emit("XCHGQ", ptr, _AX) 80 } 81 self.Sjmp("JMP", "_end_writeBarrier" + strconv.Itoa(i) + "_{n}") 82 self.Link("_no_writeBarrier" + strconv.Itoa(i) + "_{n}") 83 self.Emit("MOVQ", ptr, rec) 84 self.Link("_end_writeBarrier" + strconv.Itoa(i) + "_{n}") 85 } 86 87 88 func (self *_ValueDecoder) WritePtrAX(i int, rec obj.Addr, saveDI bool) { 89 self.Emit("MOVQ", _V_writeBarrier, _R9) 90 self.Emit("CMPL", jit.Ptr(_R9, 0), jit.Imm(0)) 91 self.Sjmp("JE", "_no_writeBarrier" + strconv.Itoa(i) + "_{n}") 92 if saveDI { 93 self.save(_DI) 94 } 95 self.Emit("LEAQ", rec, _DI) 96 self.call(_F_gcWriteBarrierAX) 97 if saveDI { 98 self.load(_DI) 99 } 100 self.Sjmp("JMP", "_end_writeBarrier" + strconv.Itoa(i) + "_{n}") 101 self.Link("_no_writeBarrier" + strconv.Itoa(i) + "_{n}") 102 self.Emit("MOVQ", _AX, rec) 103 self.Link("_end_writeBarrier" + strconv.Itoa(i) + "_{n}") 104 } 105 106 func (self *_ValueDecoder) WriteRecNotAX(i int, ptr obj.Addr, rec obj.Addr, saveDI bool) { 107 if rec.Reg == x86.REG_AX || rec.Index == x86.REG_AX { 108 panic("rec contains AX!") 109 } 110 self.Emit("MOVQ", _V_writeBarrier, _AX) 111 self.Emit("CMPL", jit.Ptr(_AX, 0), jit.Imm(0)) 112 self.Sjmp("JE", "_no_writeBarrier" + strconv.Itoa(i) + "_{n}") 113 self.Emit("MOVQ", ptr, _AX) 114 if saveDI { 115 self.save(_DI) 116 } 117 self.Emit("LEAQ", rec, _DI) 118 self.call(_F_gcWriteBarrierAX) 119 if saveDI { 120 self.load(_DI) 121 } 122 self.Sjmp("JMP", "_end_writeBarrier" + strconv.Itoa(i) + "_{n}") 123 self.Link("_no_writeBarrier" + strconv.Itoa(i) + "_{n}") 124 self.Emit("MOVQ", ptr, rec) 125 self.Link("_end_writeBarrier" + strconv.Itoa(i) + "_{n}") 126 }