mirror of
				https://github.com/go-delve/delve.git
				synced 2025-11-01 03:42:59 +08:00 
			
		
		
		
	 281f3920dd
			
		
	
	281f3920dd
	
	
	
		
			
			TestStacktraceGoroutine failed intermittently on freebsd/amd64/go1.20. This happens because of two windows, in the scheduler (right after parking a goroutine and just before resuming a parked goroutine) where a thread is associated with a goroutine but running in the system stack and there is no systemstack_switch frame to connect the two stacks.
		
			
				
	
	
		
			248 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			248 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package proc
 | |
| 
 | |
| import (
 | |
| 	"encoding/binary"
 | |
| 	"fmt"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/go-delve/delve/pkg/dwarf/frame"
 | |
| 	"github.com/go-delve/delve/pkg/dwarf/op"
 | |
| 	"github.com/go-delve/delve/pkg/dwarf/regnum"
 | |
| )
 | |
| 
 | |
| var i386BreakInstruction = []byte{0xCC}
 | |
| 
 | |
| // I386Arch returns an initialized I386Arch
 | |
| // struct.
 | |
| func I386Arch(goos string) *Arch {
 | |
| 	return &Arch{
 | |
| 		Name:                             "386",
 | |
| 		ptrSize:                          4,
 | |
| 		maxInstructionLength:             15,
 | |
| 		breakpointInstruction:            i386BreakInstruction,
 | |
| 		altBreakpointInstruction:         []byte{0xcd, 0x03},
 | |
| 		breakInstrMovesPC:                true,
 | |
| 		derefTLS:                         false,
 | |
| 		prologues:                        prologuesI386,
 | |
| 		fixFrameUnwindContext:            i386FixFrameUnwindContext,
 | |
| 		switchStack:                      i386SwitchStack,
 | |
| 		regSize:                          i386RegSize,
 | |
| 		RegistersToDwarfRegisters:        i386RegistersToDwarfRegisters,
 | |
| 		addrAndStackRegsToDwarfRegisters: i386AddrAndStackRegsToDwarfRegisters,
 | |
| 		DwarfRegisterToString:            i386DwarfRegisterToString,
 | |
| 		inhibitStepInto:                  i386InhibitStepInto,
 | |
| 		asmDecode:                        i386AsmDecode,
 | |
| 		PCRegNum:                         regnum.I386_Eip,
 | |
| 		SPRegNum:                         regnum.I386_Esp,
 | |
| 		asmRegisters:                     i386AsmRegisters,
 | |
| 		RegisterNameToDwarf:              nameToDwarfFunc(regnum.I386NameToDwarf),
 | |
| 		RegnumToString:                   regnum.I386ToName,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func i386FixFrameUnwindContext(fctxt *frame.FrameContext, pc uint64, bi *BinaryInfo) *frame.FrameContext {
 | |
| 	i := bi.Arch
 | |
| 	if i.sigreturnfn == nil {
 | |
| 		i.sigreturnfn = bi.lookupOneFunc("runtime.sigreturn")
 | |
| 	}
 | |
| 
 | |
| 	if fctxt == nil || (i.sigreturnfn != nil && pc >= i.sigreturnfn.Entry && pc < i.sigreturnfn.End) {
 | |
| 		// When there's no frame descriptor entry use BP (the frame pointer) instead
 | |
| 		// - return register is [bp + i.PtrSize()] (i.e. [cfa-i.PtrSize()])
 | |
| 		// - cfa is bp + i.PtrSize()*2
 | |
| 		// - bp is [bp] (i.e. [cfa-i.PtrSize()*2])
 | |
| 		// - sp is cfa
 | |
| 
 | |
| 		// When the signal handler runs it will move the execution to the signal
 | |
| 		// handling stack (installed using the sigaltstack system call).
 | |
| 		// This isn't i proper stack switch: the pointer to g in TLS will still
 | |
| 		// refer to whatever g was executing on that thread before the signal was
 | |
| 		// received.
 | |
| 		// Since go did not execute i stack switch the previous value of sp, pc
 | |
| 		// and bp is not saved inside g.sched, as it normally would.
 | |
| 		// The only way to recover is to either read sp/pc from the signal context
 | |
| 		// parameter (the ucontext_t* parameter) or to unconditionally follow the
 | |
| 		// frame pointer when we get to runtime.sigreturn (which is what we do
 | |
| 		// here).
 | |
| 
 | |
| 		return &frame.FrameContext{
 | |
| 			RetAddrReg: regnum.I386_Eip,
 | |
| 			Regs: map[uint64]frame.DWRule{
 | |
| 				regnum.I386_Eip: {
 | |
| 					Rule:   frame.RuleOffset,
 | |
| 					Offset: int64(-i.PtrSize()),
 | |
| 				},
 | |
| 				regnum.I386_Ebp: {
 | |
| 					Rule:   frame.RuleOffset,
 | |
| 					Offset: int64(-2 * i.PtrSize()),
 | |
| 				},
 | |
| 				regnum.I386_Esp: {
 | |
| 					Rule:   frame.RuleValOffset,
 | |
| 					Offset: 0,
 | |
| 				},
 | |
| 			},
 | |
| 			CFA: frame.DWRule{
 | |
| 				Rule:   frame.RuleCFA,
 | |
| 				Reg:    regnum.I386_Ebp,
 | |
| 				Offset: int64(2 * i.PtrSize()),
 | |
| 			},
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if i.crosscall2fn == nil {
 | |
| 		i.crosscall2fn = bi.lookupOneFunc("crosscall2")
 | |
| 	}
 | |
| 
 | |
| 	// TODO(chainhelen), need to check whether there is a bad frame descriptor like amd64.
 | |
| 	// crosscall2 is defined in $GOROOT/src/runtime/cgo/asm_386.s.
 | |
| 	if i.crosscall2fn != nil && pc >= i.crosscall2fn.Entry && pc < i.crosscall2fn.End {
 | |
| 		rule := fctxt.CFA
 | |
| 		fctxt.CFA = rule
 | |
| 	}
 | |
| 
 | |
| 	// We assume that EBP is the frame pointer and we want to keep it updated,
 | |
| 	// so that we can use it to unwind the stack even when we encounter frames
 | |
| 	// without descriptor entries.
 | |
| 	// If there isn't i rule already we emit one.
 | |
| 	if fctxt.Regs[regnum.I386_Ebp].Rule == frame.RuleUndefined {
 | |
| 		fctxt.Regs[regnum.I386_Ebp] = frame.DWRule{
 | |
| 			Rule:   frame.RuleFramePointer,
 | |
| 			Reg:    regnum.I386_Ebp,
 | |
| 			Offset: 0,
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return fctxt
 | |
| }
 | |
| 
 | |
| // SwitchStack will use the current frame to determine if it's time to
 | |
| func i386SwitchStack(it *stackIterator, _ *op.DwarfRegisters) bool {
 | |
| 	if it.frame.Current.Fn == nil {
 | |
| 		if it.systemstack && it.g != nil && it.top {
 | |
| 			it.switchToGoroutineStack()
 | |
| 			return true
 | |
| 		}
 | |
| 		return false
 | |
| 	}
 | |
| 	switch it.frame.Current.Fn.Name {
 | |
| 	case "runtime.asmcgocall", "runtime.cgocallback_gofunc": // TODO(chainhelen), need to support cgo stacktraces.
 | |
| 		return false
 | |
| 	case "runtime.goexit", "runtime.rt0_go":
 | |
| 		// Look for "top of stack" functions.
 | |
| 		it.atend = true
 | |
| 		return true
 | |
| 
 | |
| 	case "runtime.mcall":
 | |
| 		if it.systemstack && it.g != nil {
 | |
| 			it.switchToGoroutineStack()
 | |
| 			return true
 | |
| 		}
 | |
| 		it.atend = true
 | |
| 		return true
 | |
| 
 | |
| 	case "runtime.mstart":
 | |
| 		// Calls to runtime.systemstack will switch to the systemstack then:
 | |
| 		// 1. alter the goroutine stack so that it looks like systemstack_switch
 | |
| 		//    was called
 | |
| 		// 2. alter the system stack so that it looks like the bottom-most frame
 | |
| 		//    belongs to runtime.mstart
 | |
| 		// If we find a runtime.mstart frame on the system stack of a goroutine
 | |
| 		// parked on runtime.systemstack_switch we assume runtime.systemstack was
 | |
| 		// called and continue tracing from the parked position.
 | |
| 
 | |
| 		if it.top || !it.systemstack || it.g == nil {
 | |
| 			return false
 | |
| 		}
 | |
| 		if fn := it.bi.PCToFunc(it.g.PC); fn == nil || fn.Name != "runtime.systemstack_switch" {
 | |
| 			return false
 | |
| 		}
 | |
| 
 | |
| 		it.switchToGoroutineStack()
 | |
| 		return true
 | |
| 
 | |
| 	case "runtime.newstack", "runtime.systemstack":
 | |
| 		if it.systemstack && it.g != nil {
 | |
| 			it.switchToGoroutineStack()
 | |
| 			return true
 | |
| 		}
 | |
| 
 | |
| 		return false
 | |
| 
 | |
| 	default:
 | |
| 		return false
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // RegSize returns the size (in bytes) of register regnum.
 | |
| // The mapping between hardware registers and DWARF registers is specified
 | |
| // in the System V ABI Intel386 Architecture Processor Supplement page 25,
 | |
| // table 2.14
 | |
| // https://www.uclibc.org/docs/psABI-i386.pdf
 | |
| func i386RegSize(regnum uint64) int {
 | |
| 	// XMM registers
 | |
| 	if regnum >= 21 && regnum <= 36 {
 | |
| 		return 16
 | |
| 	}
 | |
| 	// x87 registers
 | |
| 	if regnum >= 11 && regnum <= 18 {
 | |
| 		return 10
 | |
| 	}
 | |
| 	return 4
 | |
| }
 | |
| 
 | |
| func i386RegistersToDwarfRegisters(staticBase uint64, regs Registers) *op.DwarfRegisters {
 | |
| 	dregs := initDwarfRegistersFromSlice(regnum.I386MaxRegNum(), regs, regnum.I386NameToDwarf)
 | |
| 	dr := op.NewDwarfRegisters(staticBase, dregs, binary.LittleEndian, regnum.I386_Eip, regnum.I386_Esp, regnum.I386_Ebp, 0)
 | |
| 	dr.SetLoadMoreCallback(loadMoreDwarfRegistersFromSliceFunc(dr, regs, regnum.I386NameToDwarf))
 | |
| 
 | |
| 	return dr
 | |
| }
 | |
| 
 | |
| func i386AddrAndStackRegsToDwarfRegisters(staticBase, pc, sp, bp, lr uint64) op.DwarfRegisters {
 | |
| 	dregs := make([]*op.DwarfRegister, regnum.I386_Eip+1)
 | |
| 	dregs[regnum.I386_Eip] = op.DwarfRegisterFromUint64(pc)
 | |
| 	dregs[regnum.I386_Esp] = op.DwarfRegisterFromUint64(sp)
 | |
| 	dregs[regnum.I386_Ebp] = op.DwarfRegisterFromUint64(bp)
 | |
| 
 | |
| 	return *op.NewDwarfRegisters(staticBase, dregs, binary.LittleEndian, regnum.I386_Eip, regnum.I386_Esp, regnum.I386_Ebp, 0)
 | |
| }
 | |
| 
 | |
| func i386DwarfRegisterToString(j int, reg *op.DwarfRegister) (name string, floatingPoint bool, repr string) {
 | |
| 	name = regnum.I386ToName(uint64(j))
 | |
| 
 | |
| 	if reg == nil {
 | |
| 		return name, false, ""
 | |
| 	}
 | |
| 
 | |
| 	switch n := strings.ToLower(name); n {
 | |
| 	case "eflags":
 | |
| 		return name, false, eflagsDescription.Describe(reg.Uint64Val, 32)
 | |
| 
 | |
| 	case "tw", "fop":
 | |
| 		return name, true, fmt.Sprintf("%#04x", reg.Uint64Val)
 | |
| 
 | |
| 	default:
 | |
| 		if reg.Bytes != nil && strings.HasPrefix(n, "xmm") {
 | |
| 			return name, true, formatSSEReg(name, reg.Bytes)
 | |
| 		} else if reg.Bytes != nil && strings.HasPrefix(n, "st(") {
 | |
| 			return name, true, formatX87Reg(reg.Bytes)
 | |
| 		} else if reg.Bytes == nil || (reg.Bytes != nil && len(reg.Bytes) <= 8) {
 | |
| 			return name, false, fmt.Sprintf("%#016x", reg.Uint64Val)
 | |
| 		} else {
 | |
| 			return name, false, fmt.Sprintf("%#x", reg.Bytes)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // i386InhibitStepInto returns whether StepBreakpoint can be set at pc.
 | |
| // When cgo or pie on 386 linux, compiler will insert more instructions (ex: call __x86.get_pc_thunk.).
 | |
| // StepBreakpoint shouldn't be set on __x86.get_pc_thunk and skip it.
 | |
| // See comments on stacksplit in $GOROOT/src/cmd/internal/obj/x86/obj6.go for generated instructions details.
 | |
| func i386InhibitStepInto(bi *BinaryInfo, pc uint64) bool {
 | |
| 	if bi.SymNames != nil && bi.SymNames[pc] != nil &&
 | |
| 		strings.HasPrefix(bi.SymNames[pc].Name, "__x86.get_pc_thunk.") {
 | |
| 		return true
 | |
| 	}
 | |
| 	return false
 | |
| }
 |