125 lines
3.0 KiB
Go
125 lines
3.0 KiB
Go
|
/*
|
||
|
* Copyright 2021 ByteDance Inc.
|
||
|
*
|
||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
* you may not use this file except in compliance with the License.
|
||
|
* You may obtain a copy of the License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* See the License for the specific language governing permissions and
|
||
|
* limitations under the License.
|
||
|
*/
|
||
|
|
||
|
package rt
|
||
|
|
||
|
import (
|
||
|
`os`
|
||
|
`sync/atomic`
|
||
|
`unsafe`
|
||
|
|
||
|
`golang.org/x/arch/x86/x86asm`
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
_MaxInstr = 15
|
||
|
)
|
||
|
|
||
|
func isvar(arg x86asm.Arg) bool {
|
||
|
v, ok := arg.(x86asm.Mem)
|
||
|
return ok && v.Base == x86asm.RIP
|
||
|
}
|
||
|
|
||
|
func iszero(arg x86asm.Arg) bool {
|
||
|
v, ok := arg.(x86asm.Imm)
|
||
|
return ok && v == 0
|
||
|
}
|
||
|
|
||
|
func GcwbAddr() uintptr {
|
||
|
var err error
|
||
|
var off uintptr
|
||
|
var ins x86asm.Inst
|
||
|
|
||
|
/* get the function address */
|
||
|
pc := uintptr(0)
|
||
|
fp := FuncAddr(atomic.StorePointer)
|
||
|
|
||
|
/* search within the first 16 instructions */
|
||
|
for i := 0; i < 16; i++ {
|
||
|
mem := unsafe.Pointer(uintptr(fp) + pc)
|
||
|
buf := BytesFrom(mem, _MaxInstr, _MaxInstr)
|
||
|
|
||
|
/* disassemble the instruction */
|
||
|
if ins, err = x86asm.Decode(buf, 64); err != nil {
|
||
|
panic("gcwbaddr: " + err.Error())
|
||
|
}
|
||
|
|
||
|
/* check for a byte comparison with zero */
|
||
|
if ins.Op == x86asm.CMP && ins.MemBytes == 1 && isvar(ins.Args[0]) && iszero(ins.Args[1]) {
|
||
|
off = pc + uintptr(ins.Len) + uintptr(ins.Args[0].(x86asm.Mem).Disp)
|
||
|
break
|
||
|
}
|
||
|
|
||
|
/* move to next instruction */
|
||
|
nb := ins.Len
|
||
|
pc += uintptr(nb)
|
||
|
}
|
||
|
|
||
|
/* check for address */
|
||
|
if off == 0 {
|
||
|
panic("gcwbaddr: could not locate the variable `writeBarrier`")
|
||
|
} else {
|
||
|
return uintptr(fp) + off
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// StopProfiling is used to stop traceback introduced by SIGPROF while native code is running.
|
||
|
// WARN: this option is only a workaround for traceback issue (https://github.com/bytedance/sonic/issues/310),
|
||
|
// and will be dropped when the issue is fixed.
|
||
|
var StopProfiling = os.Getenv("SONIC_STOP_PROFILING") != ""
|
||
|
|
||
|
// WARN: must be aligned with runtime.Prof
|
||
|
// type Prof struct {
|
||
|
// signalLock uint32
|
||
|
// hz int32
|
||
|
// }
|
||
|
|
||
|
var (
|
||
|
// // go:linkname runtimeProf runtime.prof
|
||
|
// runtimeProf Prof
|
||
|
|
||
|
// count of native-C calls
|
||
|
yieldCount uint32
|
||
|
|
||
|
// previous value of runtimeProf.hz
|
||
|
oldHz int32
|
||
|
)
|
||
|
|
||
|
//go:nosplit
|
||
|
func MoreStack(size uintptr)
|
||
|
|
||
|
func StopProf()
|
||
|
|
||
|
// func StopProf() {
|
||
|
// atomic.AddUint32(&yieldCount, 1)
|
||
|
// if runtimeProf.hz != 0 {
|
||
|
// oldHz = runtimeProf.hz
|
||
|
// runtimeProf.hz = 0
|
||
|
// }
|
||
|
// }
|
||
|
|
||
|
func StartProf()
|
||
|
|
||
|
// func StartProf() {
|
||
|
// atomic.AddUint32(&yieldCount, ^uint32(0))
|
||
|
// if yieldCount == 0 && runtimeProf.hz == 0 {
|
||
|
// if oldHz == 0 {
|
||
|
// oldHz = 100
|
||
|
// }
|
||
|
// runtimeProf.hz = oldHz
|
||
|
// }
|
||
|
// }
|