diff --git a/pkg/proc/bininfo.go b/pkg/proc/bininfo.go index 3f999946cf..d6eff4ea90 100644 --- a/pkg/proc/bininfo.go +++ b/pkg/proc/bininfo.go @@ -34,6 +34,7 @@ import ( "github.com/go-delve/delve/pkg/internal/gosym" "github.com/go-delve/delve/pkg/logflags" "github.com/go-delve/delve/pkg/proc/debuginfod" + "github.com/go-delve/delve/pkg/proc/evalop" "github.com/hashicorp/golang-lru/simplelru" ) @@ -110,7 +111,8 @@ type BinaryInfo struct { // Go 1.17 register ABI is enabled. regabi bool - logger logflags.Logger + debugPinnerFn *Function + logger logflags.Logger } var ( @@ -2574,13 +2576,23 @@ func (bi *BinaryInfo) LookupFunc() map[string][]*Function { } func (bi *BinaryInfo) lookupOneFunc(name string) *Function { + if name == evalop.DebugPinnerFunctionName && bi.debugPinnerFn != nil { + return bi.debugPinnerFn + } fns := bi.LookupFunc()[name] if fns == nil { return nil } + if name == evalop.DebugPinnerFunctionName { + bi.debugPinnerFn = fns[0] + } return fns[0] } +func (bi *BinaryInfo) hasDebugPinner() bool { + return bi.lookupOneFunc(evalop.DebugPinnerFunctionName) != nil +} + // loadDebugInfoMapsCompileUnit loads entry from a single compile unit. func (bi *BinaryInfo) loadDebugInfoMapsCompileUnit(ctxt *loadDebugInfoMapsContext, image *Image, reader *reader.Reader, cu *compileUnit) { hasAttrGoPkgName := goversion.ProducerAfterOrEqual(cu.producer, 1, 13) diff --git a/pkg/proc/dwarf_export_test.go b/pkg/proc/dwarf_export_test.go index 7add16c718..3431f93baa 100644 --- a/pkg/proc/dwarf_export_test.go +++ b/pkg/proc/dwarf_export_test.go @@ -30,3 +30,14 @@ func NewCompositeMemory(p *Target, pieces []op.Piece, base uint64) (*compositeMe func IsJNZ(inst archInst) bool { return inst.(*x86Inst).Op == x86asm.JNE } + +// HasDebugPinner returns true if the target has runtime.debugPinner. +func (bi *BinaryInfo) HasDebugPinner() bool { + return bi.hasDebugPinner() +} + +// DebugPinCount returns the number of addresses pinned during the last +// function call injection. +func DebugPinCount() int { + return debugPinCount +} diff --git a/pkg/proc/eval.go b/pkg/proc/eval.go index 08133d6547..da4679fee9 100644 --- a/pkg/proc/eval.go +++ b/pkg/proc/eval.go @@ -186,9 +186,17 @@ func GoroutineScope(t *Target, thread Thread) (*EvalScope, error) { return FrameToScope(t, thread.ProcessMemory(), g, threadID, locations...), nil } +func (scope *EvalScope) evalopFlags() evalop.Flags { + flags := evalop.Flags(0) + if scope.BinInfo.hasDebugPinner() { + flags |= evalop.HasDebugPinner + } + return flags +} + // EvalExpression returns the value of the given expression. func (scope *EvalScope) EvalExpression(expr string, cfg LoadConfig) (*Variable, error) { - ops, err := evalop.Compile(scopeToEvalLookup{scope}, expr, false) + ops, err := evalop.Compile(scopeToEvalLookup{scope}, expr, scope.evalopFlags()) if err != nil { return nil, err } @@ -642,7 +650,7 @@ func (scope *EvalScope) setValue(dstv, srcv *Variable, srcExpr string) error { // SetVariable sets the value of the named variable func (scope *EvalScope) SetVariable(name, value string) error { - ops, err := evalop.CompileSet(scopeToEvalLookup{scope}, name, value) + ops, err := evalop.CompileSet(scopeToEvalLookup{scope}, name, value, scope.evalopFlags()) if err != nil { return err } @@ -828,9 +836,13 @@ type evalStack struct { scope *EvalScope curthread Thread lastRetiredFncall *functionCallState + debugPinner *Variable } func (s *evalStack) push(v *Variable) { + if v == nil { + panic(errors.New("internal debugger error, nil pushed onto variables stack")) + } s.stack = append(s.stack, v) } @@ -944,7 +956,7 @@ func (stack *evalStack) run() { stack.executeOp() // If the instruction we just executed requests the call injection // protocol by setting callInjectionContinue we switch to it. - if stack.callInjectionContinue { + if stack.callInjectionContinue && stack.err == nil { scope.callCtx.injectionThread = nil return } @@ -959,25 +971,35 @@ func (stack *evalStack) run() { // injections before returning. if len(stack.fncalls) > 0 { + fncallLog("undoing calls (%v)", stack.err) fncall := stack.fncallPeek() if fncall == stack.lastRetiredFncall { stack.err = fmt.Errorf("internal debugger error: could not undo injected call during error recovery, original error: %v", stack.err) return } if fncall.undoInjection != nil { - // setTargetExecuted is set if evalop.CallInjectionSetTarget has been - // executed but evalop.CallInjectionComplete hasn't, we must undo the callOP - // call in evalop.CallInjectionSetTarget before continuing. - switch scope.BinInfo.Arch.Name { - case "amd64": - regs, _ := curthread.Registers() - setSP(curthread, regs.SP()+uint64(scope.BinInfo.Arch.PtrSize())) - setPC(curthread, fncall.undoInjection.oldpc) - case "arm64", "ppc64le": - setLR(curthread, fncall.undoInjection.oldlr) - setPC(curthread, fncall.undoInjection.oldpc) - default: - panic("not implemented") + if fncall.undoInjection.doComplete2 { + // doComplete2 is set if CallInjectionComplete{DoPinning: true} has been + // executed but CallInjectionComplete2 hasn't. + regs, err := curthread.Registers() + if err == nil { + callInjectionComplete2(scope, scope.BinInfo, fncall, regs, curthread) + } + } else { + // undoInjection is set if evalop.CallInjectionSetTarget has been + // executed but evalop.CallInjectionComplete hasn't, we must undo the callOP + // call in evalop.CallInjectionSetTarget before continuing. + switch scope.BinInfo.Arch.Name { + case "amd64": + regs, _ := curthread.Registers() + setSP(curthread, regs.SP()+uint64(scope.BinInfo.Arch.PtrSize())) + setPC(curthread, fncall.undoInjection.oldpc) + case "arm64", "ppc64le": + setLR(curthread, fncall.undoInjection.oldlr) + setPC(curthread, fncall.undoInjection.oldpc) + default: + panic("not implemented") + } } } stack.lastRetiredFncall = fncall @@ -1137,6 +1159,11 @@ func (stack *evalStack) executeOp() { case *evalop.Pop: stack.pop() + case *evalop.Roll: + rolled := stack.stack[len(stack.stack)-op.N-1] + copy(stack.stack[len(stack.stack)-op.N-1:], stack.stack[len(stack.stack)-op.N:]) + stack.stack[len(stack.stack)-1] = rolled + case *evalop.BuiltinCall: vars := make([]*Variable, len(op.Args)) for i := len(op.Args) - 1; i >= 0; i-- { @@ -1159,9 +1186,29 @@ func (stack *evalStack) executeOp() { stack.err = funcCallCopyOneArg(scope, fncall, actualArg, &fncall.formalArgs[op.ArgNum], curthread) case *evalop.CallInjectionComplete: - stack.fncallPeek().undoInjection = nil + fncall := stack.fncallPeek() + fncall.doPinning = op.DoPinning + if op.DoPinning { + fncall.undoInjection.doComplete2 = true + } else { + fncall.undoInjection = nil + } stack.callInjectionContinue = true + case *evalop.CallInjectionComplete2: + fncall := stack.fncallPeek() + if len(fncall.addrsToPin) != 0 { + stack.err = fmt.Errorf("internal debugger error: CallInjectionComplete2 called when there still are addresses to pin") + } + fncall.undoInjection = nil + regs, err := curthread.Registers() + if err == nil { + callInjectionComplete2(scope, scope.BinInfo, fncall, regs, curthread) + stack.callInjectionContinue = true + } else { + stack.err = err + } + case *evalop.CallInjectionStartSpecial: stack.callInjectionContinue = scope.callInjectionStartSpecial(stack, op, curthread) @@ -1173,6 +1220,26 @@ func (stack *evalStack) executeOp() { rhv := stack.pop() stack.err = scope.setValue(lhv, rhv, exprToString(op.Rhe)) + case *evalop.PushPinAddress: + debugPinCount++ + fncall := stack.fncallPeek() + addrToPin := fncall.addrsToPin[len(fncall.addrsToPin)-1] + fncall.addrsToPin = fncall.addrsToPin[:len(fncall.addrsToPin)-1] + typ, err := scope.BinInfo.findType("unsafe.Pointer") + if ptyp, ok := typ.(*godwarf.PtrType); err == nil && ok { + v := newVariable("", 0, typ, scope.BinInfo, scope.Mem) + v.Children = []Variable{*(newVariable("", uint64(addrToPin), ptyp.Type, scope.BinInfo, scope.Mem))} + stack.push(v) + } else { + stack.err = fmt.Errorf("can not pin address: %v", err) + } + + case *evalop.SetDebugPinner: + stack.debugPinner = stack.pop() + + case *evalop.PushDebugPinner: + stack.push(stack.debugPinner) + default: stack.err = fmt.Errorf("internal debugger error: unknown eval opcode: %#v", op) } @@ -1273,7 +1340,7 @@ func (stack *evalStack) pushIdent(scope *EvalScope, name string) (found bool) { } func (scope *EvalScope) evalAST(t ast.Expr) (*Variable, error) { - ops, err := evalop.CompileAST(scopeToEvalLookup{scope}, t) + ops, err := evalop.CompileAST(scopeToEvalLookup{scope}, t, scope.evalopFlags()) if err != nil { return nil, err } @@ -1289,9 +1356,14 @@ func exprToString(t ast.Expr) string { } func (scope *EvalScope) evalJump(op *evalop.Jump, stack *evalStack) { - x := stack.peek() - if op.Pop { - stack.pop() + var x *Variable + + switch op.When { + case evalop.JumpIfTrue, evalop.JumpIfFalse, evalop.JumpIfAllocStringChecksFail: + x = stack.peek() + if op.Pop { + stack.pop() + } } var v bool @@ -1308,6 +1380,17 @@ func (scope *EvalScope) evalJump(op *evalop.Jump, stack *evalStack) { return } return + case evalop.JumpAlways: + stack.opidx = op.Target - 1 + return + case evalop.JumpIfPinningDone: + fncall := stack.fncallPeek() + if len(fncall.addrsToPin) == 0 { + stack.opidx = op.Target - 1 + } + return + default: + panic("internal error, bad jump condition") } if x.Kind != reflect.Bool { diff --git a/pkg/proc/evalop/evalcompile.go b/pkg/proc/evalop/evalcompile.go index 660c787942..14cb53049d 100644 --- a/pkg/proc/evalop/evalcompile.go +++ b/pkg/proc/evalop/evalcompile.go @@ -18,7 +18,8 @@ import ( ) var ( - ErrFuncCallNotAllowed = errors.New("function calls not allowed without using 'call'") + ErrFuncCallNotAllowed = errors.New("function calls not allowed without using 'call'") + DebugPinnerFunctionName = "runtime.debugPinnerV1" ) type compileCtx struct { @@ -26,6 +27,8 @@ type compileCtx struct { ops []Op allowCalls bool curCall int + flags Flags + firstCall bool } type evalLookup interface { @@ -33,14 +36,24 @@ type evalLookup interface { HasBuiltin(string) bool } +// Flags describes flags used to control Compile and CompileAST +type Flags uint8 + +const ( + CanSet Flags = 1 << iota // Assignment is allowed + HasDebugPinner // runtime.debugPinner is available +) + // CompileAST compiles the expression t into a list of instructions. -func CompileAST(lookup evalLookup, t ast.Expr) ([]Op, error) { - ctx := &compileCtx{evalLookup: lookup, allowCalls: true} +func CompileAST(lookup evalLookup, t ast.Expr, flags Flags) ([]Op, error) { + ctx := &compileCtx{evalLookup: lookup, allowCalls: true, flags: flags, firstCall: true} err := ctx.compileAST(t) if err != nil { return nil, err } + ctx.compileDebugUnpin() + err = ctx.depthCheck(1) if err != nil { return ctx.ops, err @@ -50,18 +63,18 @@ func CompileAST(lookup evalLookup, t ast.Expr) ([]Op, error) { // Compile compiles the expression expr into a list of instructions. // If canSet is true expressions like "x = y" are also accepted. -func Compile(lookup evalLookup, expr string, canSet bool) ([]Op, error) { +func Compile(lookup evalLookup, expr string, flags Flags) ([]Op, error) { t, err := parser.ParseExpr(expr) if err != nil { - if canSet { + if flags&CanSet != 0 { eqOff, isAs := isAssignment(err) if isAs { - return CompileSet(lookup, expr[:eqOff], expr[eqOff+1:]) + return CompileSet(lookup, expr[:eqOff], expr[eqOff+1:], flags) } } return nil, err } - return CompileAST(lookup, t) + return CompileAST(lookup, t, flags) } func isAssignment(err error) (int, bool) { @@ -74,7 +87,7 @@ func isAssignment(err error) (int, bool) { // CompileSet compiles the expression setting lhexpr to rhexpr into a list of // instructions. -func CompileSet(lookup evalLookup, lhexpr, rhexpr string) ([]Op, error) { +func CompileSet(lookup evalLookup, lhexpr, rhexpr string, flags Flags) ([]Op, error) { lhe, err := parser.ParseExpr(lhexpr) if err != nil { return nil, err @@ -84,7 +97,7 @@ func CompileSet(lookup evalLookup, lhexpr, rhexpr string) ([]Op, error) { return nil, err } - ctx := &compileCtx{evalLookup: lookup, allowCalls: true} + ctx := &compileCtx{evalLookup: lookup, allowCalls: true, flags: flags, firstCall: true} err = ctx.compileAST(rhe) if err != nil { return nil, err @@ -120,13 +133,17 @@ func (ctx *compileCtx) compileAllocLiteralString() { &PushLen{}, &PushNil{}, &PushConst{constant.MakeBool(false)}, - }) + }, true) ctx.pushOp(&ConvertAllocToString{}) jmp.Target = len(ctx.ops) } -func (ctx *compileCtx) compileSpecialCall(fnname string, argAst []ast.Expr, args []Op) { +func (ctx *compileCtx) compileSpecialCall(fnname string, argAst []ast.Expr, args []Op, doPinning bool) { + if doPinning { + ctx.compileGetDebugPinner() + } + id := ctx.curCall ctx.curCall++ ctx.pushOp(&CallInjectionStartSpecial{ @@ -136,11 +153,40 @@ func (ctx *compileCtx) compileSpecialCall(fnname string, argAst []ast.Expr, args ctx.pushOp(&CallInjectionSetTarget{id: id}) for i := range args { - ctx.pushOp(args[i]) + if args[i] != nil { + ctx.pushOp(args[i]) + } ctx.pushOp(&CallInjectionCopyArg{id: id, ArgNum: i}) } - ctx.pushOp(&CallInjectionComplete{id: id}) + doPinning = doPinning && (ctx.flags&HasDebugPinner != 0) + + ctx.pushOp(&CallInjectionComplete{id: id, DoPinning: doPinning}) + + if doPinning { + ctx.compilePinningLoop(id) + } +} + +func (ctx *compileCtx) compileGetDebugPinner() { + if ctx.firstCall && ctx.flags&HasDebugPinner != 0 { + ctx.compileSpecialCall(DebugPinnerFunctionName, []ast.Expr{}, []Op{}, false) + ctx.pushOp(&SetDebugPinner{}) + ctx.firstCall = false + } +} + +func (ctx *compileCtx) compileDebugUnpin() { + if !ctx.firstCall && ctx.flags&HasDebugPinner != 0 { + ctx.compileSpecialCall("runtime.(*Pinner).Unpin", []ast.Expr{ + &ast.Ident{Name: "debugPinner"}, + }, []Op{ + &PushDebugPinner{}, + }, false) + ctx.pushOp(&Pop{}) + ctx.pushOp(&PushNil{}) + ctx.pushOp(&SetDebugPinner{}) + } } func (ctx *compileCtx) pushOp(op Op) { @@ -172,6 +218,8 @@ func (ctx *compileCtx) depthCheck(endDepth int) error { } } + debugPinnerSeen := false + for i, op := range ctx.ops { npop, npush := op.depthCheck() if depth[i] < npop { @@ -179,8 +227,15 @@ func (ctx *compileCtx) depthCheck(endDepth int) error { } d := depth[i] - npop + npush checkAndSet(i+1, d) - if jmp, _ := op.(*Jump); jmp != nil { - checkAndSet(jmp.Target, d) + switch op := op.(type) { + case *Jump: + checkAndSet(op.Target, d) + case *CallInjectionStartSpecial: + debugPinnerSeen = true + case *CallInjectionComplete: + if op.DoPinning && !debugPinnerSeen { + err = fmt.Errorf("internal debugger error: pinning call injection seen before call to %s at instrution %d", DebugPinnerFunctionName, i) + } } if err != nil { return err @@ -521,6 +576,16 @@ func (ctx *compileCtx) compileFunctionCall(node *ast.CallExpr) error { id := ctx.curCall ctx.curCall++ + if ctx.flags&HasDebugPinner != 0 { + return ctx.compileFunctionCallWithPinning(node, id) + } + + return ctx.compileFunctionCallNoPinning(node, id) +} + +// compileFunctionCallNoPinning compiles a function call when runtime.debugPinner is +// not available in the target. +func (ctx *compileCtx) compileFunctionCallNoPinning(node *ast.CallExpr, id int) error { oldAllowCalls := ctx.allowCalls oldOps := ctx.ops ctx.allowCalls = false @@ -570,6 +635,61 @@ func (ctx *compileCtx) compileFunctionCall(node *ast.CallExpr) error { return nil } +// compileFunctionCallWithPinning compiles a function call when runtime.debugPinner +// is available in the target. +func (ctx *compileCtx) compileFunctionCallWithPinning(node *ast.CallExpr, id int) error { + ctx.compileGetDebugPinner() + + err := ctx.compileAST(node.Fun) + if err != nil { + return err + } + + for i, arg := range node.Args { + err := ctx.compileAST(arg) + if isStringLiteral(arg) { + ctx.compileAllocLiteralString() + } + if err != nil { + return fmt.Errorf("error evaluating %q as argument %d in function %s: %v", exprToString(arg), i+1, exprToString(node.Fun), err) + } + } + + ctx.pushOp(&Roll{len(node.Args)}) + ctx.pushOp(&CallInjectionStart{HasFunc: true, id: id, Node: node}) + ctx.pushOp(&Pop{}) + ctx.pushOp(&CallInjectionSetTarget{id: id}) + + for i := len(node.Args) - 1; i >= 0; i-- { + arg := node.Args[i] + ctx.pushOp(&CallInjectionCopyArg{id: id, ArgNum: i, ArgExpr: arg}) + } + + ctx.pushOp(&CallInjectionComplete{id: id, DoPinning: true}) + + ctx.compilePinningLoop(id) + + return nil +} + +func (ctx *compileCtx) compilePinningLoop(id int) { + loopStart := len(ctx.ops) + jmp := &Jump{When: JumpIfPinningDone} + ctx.pushOp(jmp) + ctx.pushOp(&PushPinAddress{}) + ctx.compileSpecialCall("runtime.(*Pinner).Pin", []ast.Expr{ + &ast.Ident{Name: "debugPinner"}, + &ast.Ident{Name: "pinAddress"}, + }, []Op{ + &PushDebugPinner{}, + nil, + }, false) + ctx.pushOp(&Pop{}) + ctx.pushOp(&Jump{When: JumpAlways, Target: loopStart}) + jmp.Target = len(ctx.ops) + ctx.pushOp(&CallInjectionComplete2{id: id}) +} + func Listing(depth []int, ops []Op) string { if depth == nil { depth = make([]int, len(ops)+1) diff --git a/pkg/proc/evalop/ops.go b/pkg/proc/evalop/ops.go index 43ae96c563..d64c099545 100644 --- a/pkg/proc/evalop/ops.go +++ b/pkg/proc/evalop/ops.go @@ -164,8 +164,13 @@ type Jump struct { } func (jmpif *Jump) depthCheck() (npop, npush int) { - if jmpif.Pop { - return 1, 0 + switch jmpif.When { + case JumpIfTrue, JumpIfFalse, JumpIfAllocStringChecksFail: + if jmpif.Pop { + return 1, 0 + } else { + return 1, 1 + } } return 0, 0 } @@ -177,6 +182,8 @@ const ( JumpIfFalse JumpCond = iota JumpIfTrue JumpIfAllocStringChecksFail + JumpAlways + JumpIfPinningDone ) // Binary pops two variables from the stack, applies the specified binary @@ -200,6 +207,13 @@ type Pop struct { func (*Pop) depthCheck() (npop, npush int) { return 1, 0 } +// Roll removes the n-th element of the stack and pushes it back in at the top +type Roll struct { + N int +} + +func (*Roll) depthCheck() (npop, npush int) { return 1, 1 } + // BuiltinCall pops len(Args) argument from the stack, calls the specified // builtin on them and pushes the result back on the stack. type BuiltinCall struct { @@ -240,11 +254,30 @@ type CallInjectionCopyArg struct { func (*CallInjectionCopyArg) depthCheck() (npop, npush int) { return 1, 0 } // CallInjectionComplete resumes target execution so that the injected call can run. +// If DoPinning is true it stops after the call is completed without undoing +// the call injection frames so that address pinning for the return value +// can be performed, see CallInjectionComplete2. type CallInjectionComplete struct { + id int + DoPinning bool +} + +func (op *CallInjectionComplete) depthCheck() (npop, npush int) { + if op.DoPinning { + return 0, 0 + } else { + return 0, 1 + } +} + +// CallInjectionComplete2 if DoPinning was passed to CallInjectionComplete +// this will finish the call injection protocol and push the evaluation +// result on the stack. +type CallInjectionComplete2 struct { id int } -func (*CallInjectionComplete) depthCheck() (npop, npush int) { return 0, 1 } +func (*CallInjectionComplete2) depthCheck() (npop, npush int) { return 0, 1 } // CallInjectionStartSpecial starts call injection for a function with a // name and arguments known at compile time. @@ -272,3 +305,22 @@ type SetValue struct { } func (*SetValue) depthCheck() (npop, npush int) { return 2, 0 } + +// SetDebugPinner pops one variable from the stack and uses it as the saved debug pinner. +type SetDebugPinner struct { +} + +func (*SetDebugPinner) depthCheck() (npop, npush int) { return 1, 0 } + +// PushDebugPinner pushes the debug pinner on the stack. +type PushDebugPinner struct { +} + +func (*PushDebugPinner) depthCheck() (npop, npush int) { return 0, 1 } + +// PushPinAddress pushes an address to pin on the stack (as an +// unsafe.Pointer) and removes it from the list of addresses to pin. +type PushPinAddress struct { +} + +func (*PushPinAddress) depthCheck() (npop, npush int) { return 0, 1 } diff --git a/pkg/proc/fncall.go b/pkg/proc/fncall.go index f045c3ab1c..02d4803f01 100644 --- a/pkg/proc/fncall.go +++ b/pkg/proc/fncall.go @@ -8,6 +8,7 @@ import ( "go/ast" "go/constant" "reflect" + "slices" "sort" "strconv" "strings" @@ -42,6 +43,9 @@ import ( // - evalop.CallInjectionSetTarget // - evalCallInjectionCopyArg // - evalCallInjectionComplete +// +// When the target has runtime.debugPinner then evalCallInjectionPinPointer +// must be also called in a loop until it returns false. const ( debugCallFunctionNamePrefix1 = "debugCall" @@ -89,12 +93,20 @@ type functionCallState struct { // it contains information on how to undo a function call injection without running it undoInjection *undoInjection + // hasDebugPinner is true if the target has runtime.debugPinner + hasDebugPinner bool + // doPinning is true if this call injection should pin the results + doPinning bool + // addrsToPin addresses from return variables that should be pinned + addrsToPin []uint64 + protocolReg uint64 debugCallName string } type undoInjection struct { oldpc, oldlr uint64 + doComplete2 bool } type callContext struct { @@ -123,10 +135,14 @@ type callInjection struct { endCallInjection func() } +//lint:ignore U1000 this variable is only used by tests +var debugPinCount int + // EvalExpressionWithCalls is like EvalExpression but allows function calls in 'expr'. // Because this can only be done in the current goroutine, unlike // EvalExpression, EvalExpressionWithCalls is not a method of EvalScope. func EvalExpressionWithCalls(grp *TargetGroup, g *G, expr string, retLoadCfg LoadConfig, checkEscape bool) error { + debugPinCount = 0 t := grp.Selected bi := t.BinInfo() if !t.SupportsFunctionCalls() { @@ -172,7 +188,7 @@ func EvalExpressionWithCalls(grp *TargetGroup, g *G, expr string, retLoadCfg Loa return err } - ops, err := evalop.Compile(scopeToEvalLookup{scope}, expr, true) + ops, err := evalop.Compile(scopeToEvalLookup{scope}, expr, scope.evalopFlags()|evalop.CanSet) if err != nil { return err } @@ -186,7 +202,7 @@ func EvalExpressionWithCalls(grp *TargetGroup, g *G, expr string, retLoadCfg Loa } stack.eval(scope, ops) - if stack.callInjectionContinue { + if stack.callInjectionContinue && stack.err == nil { return grp.Continue() } @@ -289,11 +305,19 @@ func (scope *EvalScope) evalCallInjectionStart(op *evalop.CallInjectionStart, st return } + for _, v := range stack.stack { + if v.Flags&(VariableFakeAddress|VariableCPURegister|variableSaved) != 0 || v.Unreadable != nil || v.DwarfType == nil || v.RealType == nil || v.Addr == 0 { + continue + } + saveVariable(v) + } + fncall := functionCallState{ - expr: op.Node, - savedRegs: regs, - protocolReg: protocolReg, - debugCallName: dbgcallfn.Name, + expr: op.Node, + savedRegs: regs, + protocolReg: protocolReg, + debugCallName: dbgcallfn.Name, + hasDebugPinner: scope.BinInfo.hasDebugPinner(), } if op.HasFunc { @@ -365,10 +389,18 @@ func (scope *EvalScope) evalCallInjectionStart(op *evalop.CallInjectionStart, st p.fncallForG[scope.g.ID].startThreadID = thread.ThreadID() stack.fncallPush(&fncall) - stack.push(newConstant(constant.MakeBool(fncall.fn == nil || fncall.receiver != nil || fncall.closureAddr != 0), scope.Mem)) + stack.push(newConstant(constant.MakeBool(!fncall.hasDebugPinner && (fncall.fn == nil || fncall.receiver != nil || fncall.closureAddr != 0)), scope.Mem)) stack.callInjectionContinue = true } +func saveVariable(v *Variable) { + v.mem = cacheMemory(v.mem, v.Addr, int(v.RealType.Size())) + v.Flags |= variableSaved + if cachemem, ok := v.mem.(*memCache); ok { + v.Unreadable = cachemem.load() + } +} + func funcCallFinish(scope *EvalScope, stack *evalStack) { fncall := stack.fncallPop() if fncall.err != nil { @@ -697,7 +729,7 @@ func allPointers(v *Variable, name string, f func(addr uint64, name string) erro return fmt.Errorf("escape check for %s failed, variable unreadable: %v", name, v.Unreadable) } switch v.Kind { - case reflect.Ptr: + case reflect.Ptr, reflect.UnsafePointer: var w *Variable if len(v.Children) == 1 { // this branch is here to support pointers constructed with typecasts from ints or the '&' operator @@ -713,6 +745,12 @@ func allPointers(v *Variable, name string, f func(addr uint64, name string) erro sv.RealType = resolveTypedef(&(v.RealType.(*godwarf.MapType).TypedefType)) sv = sv.maybeDereference() return f(sv.Addr, name) + case reflect.Interface: + sv := v.clone() + sv.RealType = resolveTypedef(&(v.RealType.(*godwarf.InterfaceType).TypedefType)) + sv = sv.maybeDereference() + sv.Kind = reflect.Struct + return allPointers(sv, name, f) case reflect.Struct: t := v.RealType.(*godwarf.StructType) for _, field := range t.Field { @@ -732,6 +770,10 @@ func allPointers(v *Variable, name string, f func(addr uint64, name string) erro if err := f(v.funcvalAddr(), name); err != nil { return err } + case reflect.Complex64, reflect.Complex128, reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64, reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr, reflect.Bool, reflect.Float32, reflect.Float64: + // nothing to do + default: + panic(fmt.Errorf("not implemented: %s", v.Kind)) } return nil @@ -864,29 +906,32 @@ func funcCallStep(callScope *EvalScope, stack *evalStack, thread Thread) bool { return (v.Flags & VariableReturnArgument) != 0 }) - loadValues(fncall.retvars, callScope.callCtx.retLoadCfg) + if !fncall.doPinning { + loadValues(fncall.retvars, callScope.callCtx.retLoadCfg) + } for _, v := range fncall.retvars { v.Flags |= VariableFakeAddress } - // Store the stack span of the currently running goroutine (which in Go >= - // 1.15 might be different from the original injection goroutine) so that - // later on we can use it to perform the escapeCheck - if threadg, _ := GetG(thread); threadg != nil { - callScope.callCtx.stacks = append(callScope.callCtx.stacks, threadg.stack) - } - if bi.Arch.Name == "arm64" || bi.Arch.Name == "ppc64le" { - oldlr, err := readUintRaw(thread.ProcessMemory(), regs.SP(), int64(bi.Arch.PtrSize())) - if err != nil { - fncall.err = fmt.Errorf("could not restore LR: %v", err) - break - } - if err = setLR(thread, oldlr); err != nil { - fncall.err = fmt.Errorf("could not restore LR: %v", err) - break + if fncall.doPinning { + stack.callInjectionContinue = false + for _, v := range fncall.retvars { + saveVariable(v) + allPointers(v, "", func(addr uint64, _ string) error { + if addr != 0 && pointerEscapes(addr, callScope.g.stack, callScope.callCtx.stacks) { + fncall.addrsToPin = append(fncall.addrsToPin, addr) + } + return nil + }) } + slices.Sort(fncall.addrsToPin) + fncall.addrsToPin = slices.Compact(fncall.addrsToPin) + + return false // will continue with evalop.CallInjectionComplete2 } + callInjectionComplete2(callScope, bi, fncall, regs, thread) + case debugCallRegReadPanic: // 2 // read panic value from stack stack.callInjectionContinue = true @@ -913,9 +958,29 @@ func funcCallStep(callScope *EvalScope, stack *evalStack, thread Thread) bool { return false } +func callInjectionComplete2(callScope *EvalScope, bi *BinaryInfo, fncall *functionCallState, regs Registers, thread Thread) { + // Store the stack span of the currently running goroutine (which in Go >= + // 1.15 might be different from the original injection goroutine) so that + // later on we can use it to perform the escapeCheck + if threadg, _ := GetG(thread); threadg != nil { + callScope.callCtx.stacks = append(callScope.callCtx.stacks, threadg.stack) + } + if bi.Arch.Name == "arm64" || bi.Arch.Name == "ppc64le" { + oldlr, err := readUintRaw(thread.ProcessMemory(), regs.SP(), int64(bi.Arch.PtrSize())) + if err != nil { + fncall.err = fmt.Errorf("could not restore LR: %v", err) + return + } + if err = setLR(thread, oldlr); err != nil { + fncall.err = fmt.Errorf("could not restore LR: %v", err) + return + } + } +} + func (scope *EvalScope) evalCallInjectionSetTarget(op *evalop.CallInjectionSetTarget, stack *evalStack, thread Thread) { fncall := stack.fncallPeek() - if fncall.fn == nil || fncall.receiver != nil || fncall.closureAddr != 0 { + if !fncall.hasDebugPinner && (fncall.fn == nil || fncall.receiver != nil || fncall.closureAddr != 0) { funcCallEvalFuncExpr(scope, stack, fncall) } stack.pop() // target function, consumed by funcCallEvalFuncExpr either above or in evalop.CallInjectionStart @@ -944,7 +1009,7 @@ func (scope *EvalScope) evalCallInjectionSetTarget(op *evalop.CallInjectionSetTa if fncall.receiver != nil { err := funcCallCopyOneArg(scope, fncall, fncall.receiver, &fncall.formalArgs[0], thread) if err != nil { - stack.err = err + stack.err = fmt.Errorf("could not set call receiver: %v", err) return } fncall.formalArgs = fncall.formalArgs[1:] @@ -995,6 +1060,9 @@ func fakeFunctionEntryScope(scope *EvalScope, fn *Function, cfa int64, sp uint64 func (scope *EvalScope) callInjectionStartSpecial(stack *evalStack, op *evalop.CallInjectionStartSpecial, curthread Thread) bool { fnv, err := scope.findGlobalInternal(op.FnName) if fnv == nil { + if err == nil { + err = fmt.Errorf("function %s not found", op.FnName) + } stack.err = err return false } @@ -1013,6 +1081,9 @@ func (scope *EvalScope) callInjectionStartSpecial(stack *evalStack, op *evalop.C func (scope *EvalScope) convertAllocToString(stack *evalStack) { mallocv := stack.pop() v := stack.pop() + + mallocv.loadValue(loadFullValue) + if mallocv.Unreadable != nil { stack.err = mallocv.Unreadable return @@ -1058,7 +1129,7 @@ func isCallInjectionStop(t *Target, thread Thread, loc *Location) bool { // callInjectionProtocol is the function called from Continue to progress // the injection protocol for all threads. // Returns true if a call injection terminated -func callInjectionProtocol(t *Target, threads []Thread) (done bool, err error) { +func callInjectionProtocol(t *Target, trapthread Thread, threads []Thread) (done bool, err error) { if len(t.fncallForG) == 0 { // we aren't injecting any calls, no need to check the threads. return false, nil @@ -1072,6 +1143,9 @@ func callInjectionProtocol(t *Target, threads []Thread) (done bool, err error) { if err != nil { continue } + if (thread.ThreadID() != trapthread.ThreadID()) && !thread.SoftExc() { + continue + } if !isCallInjectionStop(t, thread, loc) { continue } @@ -1180,7 +1254,10 @@ func debugCallProtocolReg(archName string, version int) (uint64, bool) { // runtimeWhitelist is a list of functions in the runtime that we can call // (through call injection) even if they are optimized. var runtimeWhitelist = map[string]bool{ - "runtime.mallocgc": true, + "runtime.mallocgc": true, + evalop.DebugPinnerFunctionName: true, + "runtime.(*Pinner).Unpin": true, + "runtime.(*Pinner).Pin": true, } // runtimeOptimizedWorkaround modifies the input DIE so that arguments and diff --git a/pkg/proc/mem.go b/pkg/proc/mem.go index 5f2ad383a7..530ca5c0f9 100644 --- a/pkg/proc/mem.go +++ b/pkg/proc/mem.go @@ -43,14 +43,25 @@ func (m *memCache) contains(addr uint64, size int) bool { return addr >= m.cacheAddr && end <= m.cacheAddr+uint64(len(m.cache)) } +func (m *memCache) load() error { + if m.loaded { + return nil + } + _, err := m.mem.ReadMemory(m.cache, m.cacheAddr) + if err != nil { + return err + } + m.loaded = true + return nil +} + func (m *memCache) ReadMemory(data []byte, addr uint64) (n int, err error) { if m.contains(addr, len(data)) { if !m.loaded { - _, err := m.mem.ReadMemory(m.cache, m.cacheAddr) + err := m.load() if err != nil { return 0, err } - m.loaded = true } copy(data, m.cache[addr-m.cacheAddr:]) return len(data), nil diff --git a/pkg/proc/target_exec.go b/pkg/proc/target_exec.go index cbc9389b1e..b6ee2e2fe5 100644 --- a/pkg/proc/target_exec.go +++ b/pkg/proc/target_exec.go @@ -159,7 +159,7 @@ func (grp *TargetGroup) Continue() error { log.Debugf("\t%d PC=%#x", th.ThreadID(), regs.PC()) } } - callInjectionDoneThis, callErrThis := callInjectionProtocol(dbp, threads) + callInjectionDoneThis, callErrThis := callInjectionProtocol(dbp, trapthread, threads) callInjectionDone = callInjectionDone || callInjectionDoneThis if callInjectionDoneThis { dbp.StopReason = StopCallReturned diff --git a/pkg/proc/types.go b/pkg/proc/types.go index bcfc610773..eaa2b7719f 100644 --- a/pkg/proc/types.go +++ b/pkg/proc/types.go @@ -206,7 +206,10 @@ func dwarfToRuntimeType(bi *BinaryInfo, mem MemoryReadWriter, typ godwarf.Type) if kindv == nil || kindv.Unreadable != nil || kindv.Kind != reflect.Uint { kindv = _type.loadFieldNamed("Kind_") } - if kindv == nil || kindv.Unreadable != nil || kindv.Kind != reflect.Uint { + if kindv == nil { + return 0, 0, false, fmt.Errorf("unreadable interace type (no kind field)") + } + if kindv.Unreadable != nil || kindv.Kind != reflect.Uint { return 0, 0, false, fmt.Errorf("unreadable interface type: %v", kindv.Unreadable) } typeKind, _ = constant.Uint64Val(kindv.Value) diff --git a/pkg/proc/variables.go b/pkg/proc/variables.go index 95c1222d31..1c3c19fd6a 100644 --- a/pkg/proc/variables.go +++ b/pkg/proc/variables.go @@ -89,6 +89,8 @@ const ( // variableTrustLen means that when this variable is loaded its length // should be trusted and used instead of MaxArrayValues variableTrustLen + + variableSaved ) // Variable represents a variable. It contains the address, name, @@ -1235,7 +1237,7 @@ func (v *Variable) maybeDereference() *Variable { switch t := v.RealType.(type) { case *godwarf.PtrType: - if v.Addr == 0 && len(v.Children) == 1 && v.loaded { + if (v.Addr == 0 || v.Flags&VariableFakeAddress != 0) && len(v.Children) == 1 && v.loaded { // fake pointer variable constructed by casting an integer to a pointer type return &v.Children[0] } @@ -1469,7 +1471,7 @@ func convertToEface(srcv, dstv *Variable) error { } typeAddr, typeKind, runtimeTypeFound, err := dwarfToRuntimeType(srcv.bi, srcv.mem, srcv.RealType) if err != nil { - return err + return fmt.Errorf("can not convert value of type %s to %s: %v", srcv.DwarfType.String(), dstv.DwarfType.String(), err) } if !runtimeTypeFound || typeKind&kindDirectIface == 0 { return &typeConvErr{srcv.DwarfType, dstv.RealType} diff --git a/pkg/proc/variables_test.go b/pkg/proc/variables_test.go index 0fe452dafc..2e2ad1c2ca 100644 --- a/pkg/proc/variables_test.go +++ b/pkg/proc/variables_test.go @@ -912,14 +912,14 @@ func getEvalExpressionTestCases() []varTest { {"bytearray[0] * bytearray[0]", false, "144", "144", "uint8", nil}, // function call / typecast errors - {"unknownthing(1, 2)", false, "", "", "", errors.New("could not find symbol value for unknownthing")}, - {"(unknownthing)(1, 2)", false, "", "", "", errors.New("could not find symbol value for unknownthing")}, + {"unknownthing(1, 2)", false, "", "", "", altErrors("function calls not allowed without using 'call'", "could not find symbol value for unknownthing")}, + {"(unknownthing)(1, 2)", false, "", "", "", altErrors("function calls not allowed without using 'call'", "could not find symbol value for unknownthing")}, {"afunc(2)", false, "", "", "", errors.New("function calls not allowed without using 'call'")}, {"(afunc)(2)", false, "", "", "", errors.New("function calls not allowed without using 'call'")}, - {"(*afunc)(2)", false, "", "", "", errors.New("expression \"afunc\" (func()) can not be dereferenced")}, - {"unknownthing(2)", false, "", "", "", errors.New("could not find symbol value for unknownthing")}, - {"(*unknownthing)(2)", false, "", "", "", errors.New("could not find symbol value for unknownthing")}, - {"(*strings.Split)(2)", false, "", "", "", errors.New("could not find symbol value for strings")}, + {"(*afunc)(2)", false, "", "", "", errors.New("*")}, + {"unknownthing(2)", false, "", "", "", altErrors("function calls not allowed without using 'call'", "could not find symbol value for unknownthing")}, + {"(*unknownthing)(2)", false, "", "", "", altErrors("function calls not allowed without using 'call'", "could not find symbol value for unknownthing")}, + {"(*strings.Split)(2)", false, "", "", "", altErrors("function calls not allowed without using 'call'", "could not find symbol value for strings")}, // pretty printing special types {"tim1", false, `time.Time(1977-05-25T18:00:00Z)…`, `time.Time(1977-05-25T18:00:00Z)…`, "time.Time", nil}, @@ -977,6 +977,18 @@ func getEvalExpressionTestCases() []varTest { return testcases } +func altErrors(errs ...string) *altError { + return &altError{errs} +} + +type altError struct { + errs []string +} + +func (err *altError) Error() string { + return "[multiple alternatives]" +} + func TestEvalExpression(t *testing.T) { testcases := getEvalExpressionTestCases() protest.AllowRecording(t) @@ -1000,8 +1012,22 @@ func TestEvalExpression(t *testing.T) { if err == nil { t.Fatalf("Expected error %s, got no error (%s)", tc.err.Error(), tc.name) } - if tc.err.Error() != err.Error() { - t.Fatalf("Unexpected error. Expected %s got %s", tc.err.Error(), err.Error()) + switch e := tc.err.(type) { + case *altError: + ok := false + for _, tgtErr := range e.errs { + if tgtErr == err.Error() { + ok = true + break + } + } + if !ok { + t.Fatalf("Unexpected error. Expected %s got %s", tc.err.Error(), err.Error()) + } + default: + if tc.err.Error() != "*" && tc.err.Error() != err.Error() { + t.Fatalf("Unexpected error. Expected %s got %s", tc.err.Error(), err.Error()) + } } } }) @@ -1272,9 +1298,10 @@ func TestIssue1075(t *testing.T) { } type testCaseCallFunction struct { - expr string // call expression to evaluate - outs []string // list of return parameters in this format: :: - err error // if not nil should return an error + expr string // call expression to evaluate + outs []string // list of return parameters in this format: :: + err error // if not nil should return an error + pinCount int // where debugPinner is supported this is the number of pins created during the function call injection } func TestCallFunction(t *testing.T) { @@ -1286,126 +1313,126 @@ func TestCallFunction(t *testing.T) { var testcases = []testCaseCallFunction{ // Basic function call injection tests - {"call1(one, two)", []string{":int:3"}, nil}, - {"call1(one+two, 4)", []string{":int:7"}, nil}, - {"callpanic()", []string{`~panic:interface {}:interface {}(string) "callpanic panicked"`}, nil}, - {`stringsJoin(nil, "")`, []string{`:string:""`}, nil}, - {`stringsJoin(stringslice, comma)`, []string{`:string:"one,two,three"`}, nil}, - {`stringsJoin(stringslice, "~~")`, []string{`:string:"one~~two~~three"`}, nil}, - {`stringsJoin(s1, comma)`, nil, errors.New(`could not find symbol value for s1`)}, - {`stringsJoin(intslice, comma)`, nil, errors.New("can not convert value of type []int to []string")}, - {`noreturncall(2)`, nil, nil}, + {"call1(one, two)", []string{":int:3"}, nil, 0}, + {"call1(one+two, 4)", []string{":int:7"}, nil, 0}, + {"callpanic()", []string{`~panic:interface {}:interface {}(string) "callpanic panicked"`}, nil, 0}, + {`stringsJoin(nil, "")`, []string{`:string:""`}, nil, 0}, + {`stringsJoin(stringslice, comma)`, []string{`:string:"one,two,three"`}, nil, 1}, + {`stringsJoin(stringslice, "~~")`, []string{`:string:"one~~two~~three"`}, nil, 2}, + {`stringsJoin(s1, comma)`, nil, errors.New(`could not find symbol value for s1`), 1}, + {`stringsJoin(intslice, comma)`, nil, errors.New("can not convert value of type []int to []string"), 1}, + {`noreturncall(2)`, nil, nil, 0}, // Expression tests - {`square(2) + 1`, []string{":int:5"}, nil}, - {`intcallpanic(1) + 1`, []string{":int:2"}, nil}, - {`intcallpanic(0) + 1`, []string{`~panic:interface {}:interface {}(string) "panic requested"`}, nil}, - {`onetwothree(5)[1] + 2`, []string{":int:9"}, nil}, + {`square(2) + 1`, []string{":int:5"}, nil, 0}, + {`intcallpanic(1) + 1`, []string{":int:2"}, nil, 0}, + {`intcallpanic(0) + 1`, []string{`~panic:interface {}:interface {}(string) "panic requested"`}, nil, 0}, + {`onetwothree(5)[1] + 2`, []string{":int:9"}, nil, 1}, // Call types tests (methods, function pointers, etc.) // The following set of calls was constructed using https://docs.google.com/document/d/1bMwCey-gmqZVTpRax-ESeVuZGmjwbocYs1iHplK-cjo/pub as a reference - {`a.VRcvr(1)`, []string{`:string:"1 + 3 = 4"`}, nil}, // direct call of a method with value receiver / on a value + {`a.VRcvr(1)`, []string{`:string:"1 + 3 = 4"`}, nil, 1}, // direct call of a method with value receiver / on a value - {`a.PRcvr(2)`, []string{`:string:"2 - 3 = -1"`}, nil}, // direct call of a method with pointer receiver / on a value - {`pa.VRcvr(3)`, []string{`:string:"3 + 6 = 9"`}, nil}, // direct call of a method with value receiver / on a pointer - {`pa.PRcvr(4)`, []string{`:string:"4 - 6 = -2"`}, nil}, // direct call of a method with pointer receiver / on a pointer + {`a.PRcvr(2)`, []string{`:string:"2 - 3 = -1"`}, nil, 1}, // direct call of a method with pointer receiver / on a value + {`pa.VRcvr(3)`, []string{`:string:"3 + 6 = 9"`}, nil, 1}, // direct call of a method with value receiver / on a pointer + {`pa.PRcvr(4)`, []string{`:string:"4 - 6 = -2"`}, nil, 1}, // direct call of a method with pointer receiver / on a pointer - {`vable_pa.VRcvr(6)`, []string{`:string:"6 + 6 = 12"`}, nil}, // indirect call of method on interface / containing value with value method - {`pable_pa.PRcvr(7)`, []string{`:string:"7 - 6 = 1"`}, nil}, // indirect call of method on interface / containing pointer with value method - {`vable_a.VRcvr(5)`, []string{`:string:"5 + 3 = 8"`}, nil}, // indirect call of method on interface / containing pointer with pointer method + {`vable_pa.VRcvr(6)`, []string{`:string:"6 + 6 = 12"`}, nil, 1}, // indirect call of method on interface / containing value with value method + {`pable_pa.PRcvr(7)`, []string{`:string:"7 - 6 = 1"`}, nil, 1}, // indirect call of method on interface / containing pointer with value method + {`vable_a.VRcvr(5)`, []string{`:string:"5 + 3 = 8"`}, nil, 1}, // indirect call of method on interface / containing pointer with pointer method - {`pa.nonexistent()`, nil, errors.New("pa has no member nonexistent")}, - {`a.nonexistent()`, nil, errors.New("a has no member nonexistent")}, - {`vable_pa.nonexistent()`, nil, errors.New("vable_pa has no member nonexistent")}, - {`vable_a.nonexistent()`, nil, errors.New("vable_a has no member nonexistent")}, - {`pable_pa.nonexistent()`, nil, errors.New("pable_pa has no member nonexistent")}, + {`pa.nonexistent()`, nil, errors.New("pa has no member nonexistent"), 0}, + {`a.nonexistent()`, nil, errors.New("a has no member nonexistent"), 0}, + {`vable_pa.nonexistent()`, nil, errors.New("vable_pa has no member nonexistent"), 0}, + {`vable_a.nonexistent()`, nil, errors.New("vable_a has no member nonexistent"), 0}, + {`pable_pa.nonexistent()`, nil, errors.New("pable_pa has no member nonexistent"), 0}, - {`fn2glob(10, 20)`, []string{":int:30"}, nil}, // indirect call of func value / set to top-level func - {`fn2clos(11)`, []string{`:string:"1 + 6 + 11 = 18"`}, nil}, // indirect call of func value / set to func literal - {`fn2clos(12)`, []string{`:string:"2 + 6 + 12 = 20"`}, nil}, - {`fn2valmeth(13)`, []string{`:string:"13 + 6 = 19"`}, nil}, // indirect call of func value / set to value method - {`fn2ptrmeth(14)`, []string{`:string:"14 - 6 = 8"`}, nil}, // indirect call of func value / set to pointer method + {`fn2glob(10, 20)`, []string{":int:30"}, nil, 0}, // indirect call of func value / set to top-level func + {`fn2clos(11)`, []string{`:string:"1 + 6 + 11 = 18"`}, nil, 1}, // indirect call of func value / set to func literal + {`fn2clos(12)`, []string{`:string:"2 + 6 + 12 = 20"`}, nil, 1}, + {`fn2valmeth(13)`, []string{`:string:"13 + 6 = 19"`}, nil, 1}, // indirect call of func value / set to value method + {`fn2ptrmeth(14)`, []string{`:string:"14 - 6 = 8"`}, nil, 1}, // indirect call of func value / set to pointer method - {"fn2nil()", nil, errors.New("nil pointer dereference")}, + {"fn2nil()", nil, errors.New("nil pointer dereference"), 0}, - {"ga.PRcvr(2)", []string{`:string:"2 - 0 = 2"`}, nil}, + {"ga.PRcvr(2)", []string{`:string:"2 - 0 = 2"`}, nil, 1}, - {"x.CallMe()", nil, nil}, - {"x2.CallMe(5)", []string{":int:25"}, nil}, + {"x.CallMe()", nil, nil, 0}, + {"x2.CallMe(5)", []string{":int:25"}, nil, 0}, - {"\"delve\".CallMe()", nil, errors.New("\"delve\" (type string) is not a struct")}, + {"\"delve\".CallMe()", nil, errors.New("\"delve\" (type string) is not a struct"), 0}, // Nested function calls tests - {`onetwothree(intcallpanic(2))`, []string{`:[]int:[]int len: 3, cap: 3, [3,4,5]`}, nil}, - {`onetwothree(intcallpanic(0))`, []string{`~panic:interface {}:interface {}(string) "panic requested"`}, nil}, - {`onetwothree(intcallpanic(2)+1)`, []string{`:[]int:[]int len: 3, cap: 3, [4,5,6]`}, nil}, - {`onetwothree(intcallpanic("not a number"))`, nil, errors.New("can not convert \"not a number\" constant to int")}, + {`onetwothree(intcallpanic(2))`, []string{`:[]int:[]int len: 3, cap: 3, [3,4,5]`}, nil, 1}, + {`onetwothree(intcallpanic(0))`, []string{`~panic:interface {}:interface {}(string) "panic requested"`}, nil, 0}, + {`onetwothree(intcallpanic(2)+1)`, []string{`:[]int:[]int len: 3, cap: 3, [4,5,6]`}, nil, 1}, + {`onetwothree(intcallpanic("not a number"))`, nil, errors.New("can not convert \"not a number\" constant to int"), 1}, // Variable setting tests - {`pa2 = getAStructPtr(8); pa2`, []string{`pa2:*main.astruct:*main.astruct {X: 8}`}, nil}, + {`pa2 = getAStructPtr(8); pa2`, []string{`pa2:*main.astruct:*main.astruct {X: 8}`}, nil, 1}, // Escape tests - {"escapeArg(&a2)", nil, errors.New("cannot use &a2 as argument pa2 in function main.escapeArg: stack object passed to escaping pointer: pa2")}, + {"escapeArg(&a2)", nil, errors.New("cannot use &a2 as argument pa2 in function main.escapeArg: stack object passed to escaping pointer: pa2"), 0}, // Issue 1577 - {"1+2", []string{`::3`}, nil}, - {`"de"+"mo"`, []string{`::"demo"`}, nil}, + {"1+2", []string{`::3`}, nil, 0}, + {`"de"+"mo"`, []string{`::"demo"`}, nil, 0}, // Issue 3176 - {`ref.String()[0]`, []string{`:byte:98`}, nil}, - {`ref.String()[20]`, nil, errors.New("index out of bounds")}, + {`ref.String()[0]`, []string{`:byte:98`}, nil, 1}, + {`ref.String()[20]`, nil, errors.New("index out of bounds"), 1}, } var testcases112 = []testCaseCallFunction{ // string allocation requires trusted argument order, which we don't have in Go 1.11 - {`stringsJoin(stringslice, ",")`, []string{`:string:"one,two,three"`}, nil}, - {`str = "a new string"; str`, []string{`str:string:"a new string"`}, nil}, + {`stringsJoin(stringslice, ",")`, []string{`:string:"one,two,three"`}, nil, 2}, + {`str = "a new string"; str`, []string{`str:string:"a new string"`}, nil, 1}, // support calling optimized functions - {`strings.Join(nil, "")`, []string{`:string:""`}, nil}, - {`strings.Join(stringslice, comma)`, []string{`:string:"one,two,three"`}, nil}, - {`strings.Join(intslice, comma)`, nil, errors.New("can not convert value of type []int to []string")}, - {`strings.Join(stringslice, ",")`, []string{`:string:"one,two,three"`}, nil}, - {`strings.LastIndexByte(stringslice[1], 'w')`, []string{":int:1"}, nil}, - {`strings.LastIndexByte(stringslice[1], 'o')`, []string{":int:2"}, nil}, - {`d.Base.Method()`, []string{`:int:4`}, nil}, - {`d.Method()`, []string{`:int:4`}, nil}, + {`strings.Join(nil, "")`, []string{`:string:""`}, nil, 0}, + {`strings.Join(stringslice, comma)`, []string{`:string:"one,two,three"`}, nil, 1}, + {`strings.Join(intslice, comma)`, nil, errors.New("can not convert value of type []int to []string"), 1}, + {`strings.Join(stringslice, ",")`, []string{`:string:"one,two,three"`}, nil, 2}, + {`strings.LastIndexByte(stringslice[1], 'w')`, []string{":int:1"}, nil, 0}, + {`strings.LastIndexByte(stringslice[1], 'o')`, []string{":int:2"}, nil, 0}, + {`d.Base.Method()`, []string{`:int:4`}, nil, 0}, + {`d.Method()`, []string{`:int:4`}, nil, 0}, } var testcases113 = []testCaseCallFunction{ - {`curriedAdd(2)(3)`, []string{`:int:5`}, nil}, + {`curriedAdd(2)(3)`, []string{`:int:5`}, nil, 1}, // Method calls on a value returned by a function - {`getAStruct(3).VRcvr(1)`, []string{`:string:"1 + 3 = 4"`}, nil}, // direct call of a method with value receiver / on a value + {`getAStruct(3).VRcvr(1)`, []string{`:string:"1 + 3 = 4"`}, nil, 1}, // direct call of a method with value receiver / on a value - {`getAStruct(3).PRcvr(2)`, nil, errors.New("cannot use getAStruct(3).PRcvr as argument pa in function main.(*astruct).PRcvr: stack object passed to escaping pointer: pa")}, // direct call of a method with pointer receiver / on a value - {`getAStructPtr(6).VRcvr(3)`, []string{`:string:"3 + 6 = 9"`}, nil}, // direct call of a method with value receiver / on a pointer - {`getAStructPtr(6).PRcvr(4)`, []string{`:string:"4 - 6 = -2"`}, nil}, // direct call of a method with pointer receiver / on a pointer + {`getAStruct(3).PRcvr(2)`, nil, errors.New("could not set call receiver: cannot use getAStruct(3).PRcvr as argument pa in function main.(*astruct).PRcvr: stack object passed to escaping pointer: pa"), 0}, // direct call of a method with pointer receiver / on a value + {`getAStructPtr(6).VRcvr(3)`, []string{`:string:"3 + 6 = 9"`}, nil, 2}, // direct call of a method with value receiver / on a pointer + {`getAStructPtr(6).PRcvr(4)`, []string{`:string:"4 - 6 = -2"`}, nil, 2}, // direct call of a method with pointer receiver / on a pointer - {`getVRcvrableFromAStruct(3).VRcvr(6)`, []string{`:string:"6 + 3 = 9"`}, nil}, // indirect call of method on interface / containing value with value method - {`getPRcvrableFromAStructPtr(6).PRcvr(7)`, []string{`:string:"7 - 6 = 1"`}, nil}, // indirect call of method on interface / containing pointer with value method - {`getVRcvrableFromAStructPtr(6).VRcvr(5)`, []string{`:string:"5 + 6 = 11"`}, nil}, // indirect call of method on interface / containing pointer with pointer method + {`getVRcvrableFromAStruct(3).VRcvr(6)`, []string{`:string:"6 + 3 = 9"`}, nil, 3}, // indirect call of method on interface / containing value with value method + {`getPRcvrableFromAStructPtr(6).PRcvr(7)`, []string{`:string:"7 - 6 = 1"`}, nil, 3}, // indirect call of method on interface / containing pointer with value method + {`getVRcvrableFromAStructPtr(6).VRcvr(5)`, []string{`:string:"5 + 6 = 11"`}, nil, 3}, // indirect call of method on interface / containing pointer with pointer method } var testcasesBefore114After112 = []testCaseCallFunction{ - {`strings.Join(s1, comma)`, nil, errors.New(`could not find symbol value for s1`)}, + {`strings.Join(s1, comma)`, nil, errors.New(`could not find symbol value for s1`), 1}, } var testcases114 = []testCaseCallFunction{ - {`strings.Join(s1, comma)`, nil, errors.New(`could not find symbol value for s1`)}, + {`strings.Join(s1, comma)`, nil, errors.New(`could not find symbol value for s1`), 1}, } var testcases117 = []testCaseCallFunction{ - {`regabistacktest("one", "two", "three", "four", "five", 4)`, []string{`:string:"onetwo"`, `:string:"twothree"`, `:string:"threefour"`, `:string:"fourfive"`, `:string:"fiveone"`, ":uint8:8"}, nil}, - {`regabistacktest2(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)`, []string{":int:3", ":int:5", ":int:7", ":int:9", ":int:11", ":int:13", ":int:15", ":int:17", ":int:19", ":int:11"}, nil}, - {`issue2698.String()`, []string{`:string:"1 2 3 4"`}, nil}, - {`issue3364.String()`, []string{`:string:"1 2"`}, nil}, - {`regabistacktest3(rast3, 5)`, []string{`:[10]string:[10]string ["onetwo","twothree","threefour","fourfive","fivesix","sixseven","sevenheight","heightnine","nineten","tenone"]`, ":uint8:15"}, nil}, - {`floatsum(1, 2)`, []string{":float64:3"}, nil}, + {`regabistacktest("one", "two", "three", "four", "five", 4)`, []string{`:string:"onetwo"`, `:string:"twothree"`, `:string:"threefour"`, `:string:"fourfive"`, `:string:"fiveone"`, ":uint8:8"}, nil, 10}, + {`regabistacktest2(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)`, []string{":int:3", ":int:5", ":int:7", ":int:9", ":int:11", ":int:13", ":int:15", ":int:17", ":int:19", ":int:11"}, nil, 0}, + {`issue2698.String()`, []string{`:string:"1 2 3 4"`}, nil, 1}, + {`issue3364.String()`, []string{`:string:"1 2"`}, nil, 1}, + {`regabistacktest3(rast3, 5)`, []string{`:[10]string:[10]string ["onetwo","twothree","threefour","fourfive","fivesix","sixseven","sevenheight","heightnine","nineten","tenone"]`, ":uint8:15"}, nil, 10}, + {`floatsum(1, 2)`, []string{":float64:3"}, nil, 0}, } withTestProcessArgs("fncall", t, ".", nil, protest.AllNonOptimized, func(p *proc.Target, grp *proc.TargetGroup, fixture protest.Fixture) { @@ -1447,7 +1474,7 @@ func TestCallFunction(t *testing.T) { } // LEAVE THIS AS THE LAST ITEM, IT BREAKS THE TARGET PROCESS!!! - testCallFunction(t, grp, p, testCaseCallFunction{"-unsafe escapeArg(&a2)", nil, nil}) + testCallFunction(t, grp, p, testCaseCallFunction{"-unsafe escapeArg(&a2)", nil, nil, 0}) }) } @@ -1463,6 +1490,12 @@ func testCallFunctionSetBreakpoint(t *testing.T, p *proc.Target, grp *proc.Targe } func testCallFunction(t *testing.T, grp *proc.TargetGroup, p *proc.Target, tc testCaseCallFunction) { + t.Run(tc.expr, func(t *testing.T) { + testCallFunctionIntl(t, grp, p, tc) + }) +} + +func testCallFunctionIntl(t *testing.T, grp *proc.TargetGroup, p *proc.Target, tc testCaseCallFunction) { const unsafePrefix = "-unsafe " var callExpr, varExpr string @@ -1493,7 +1526,11 @@ func testCallFunction(t *testing.T, grp *proc.TargetGroup, p *proc.Target, tc te } if err != nil { - t.Fatalf("call %q: error %q", tc.expr, err.Error()) + if strings.HasPrefix(err.Error(), "internal debugger error") { + t.Fatalf("call %q: error %s", tc.expr, err.Error()) + } else { + t.Fatalf("call %q: error %q", tc.expr, err.Error()) + } } retvalsVar := p.CurrentThread().Common().ReturnValues(pnormalLoadConfig) @@ -1538,6 +1575,13 @@ func testCallFunction(t *testing.T, grp *proc.TargetGroup, p *proc.Target, tc te t.Fatalf("call %q, output parameter %d: expected value %q, got %q", tc.expr, i, tgtValue, cvs) } } + + if p.BinInfo().HasDebugPinner() { + t.Logf("\t(pins = %d)", proc.DebugPinCount()) + if proc.DebugPinCount() != tc.pinCount { + t.Fatalf("call %q, expected pin count %d, got %d", tc.expr, tc.pinCount, proc.DebugPinCount()) + } + } } func TestIssue1531(t *testing.T) {