diff --git a/_fixtures/issue3310.go b/_fixtures/issue3310.go
new file mode 100644
index 0000000000..69a9ca6d09
--- /dev/null
+++ b/_fixtures/issue3310.go
@@ -0,0 +1,19 @@
+package main
+
+import (
+ "fmt"
+ "reflect"
+ _ "runtime"
+)
+
+var i = 2
+var val = reflect.ValueOf(i)
+
+func reflectFunc(value reflect.Value) {
+ fmt.Printf("%s\n", value.Type().Name())
+}
+
+func main() {
+ reflectFunc(val)
+ fmt.Println(&i)
+}
diff --git a/pkg/proc/dwarf_export_test.go b/pkg/proc/dwarf_export_test.go
index 7add16c718..38944a0a78 100644
--- a/pkg/proc/dwarf_export_test.go
+++ b/pkg/proc/dwarf_export_test.go
@@ -2,6 +2,7 @@ package proc
import (
"github.com/go-delve/delve/pkg/dwarf/op"
+ "github.com/go-delve/delve/pkg/proc/evalop"
"golang.org/x/arch/x86/x86asm"
)
@@ -30,3 +31,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.lookupOneFunc(evalop.DebugPinnerFunctionName) != nil
+}
+
+// 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 e46d5cb08e..3056903611 100644
--- a/pkg/proc/eval.go
+++ b/pkg/proc/eval.go
@@ -172,9 +172,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.lookupOneFunc(evalop.DebugPinnerFunctionName) != nil {
+ 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
}
@@ -577,7 +585,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
}
@@ -750,9 +758,13 @@ type evalStack struct {
scope *EvalScope
curthread Thread
lastRetiredFncall *functionCallState
+ debugPinner *Variable
}
func (s *evalStack) push(v *Variable) {
+ if v == nil {
+ panic(fmt.Errorf("internal debugger error, nil pushed onto variables stack"))
+ }
s.stack = append(s.stack, v)
}
@@ -854,6 +866,18 @@ func (stack *evalStack) resume(g *G) {
}
// call injection protocol suspended or concluded, resume normal opcode execution
+ if len(stack.fncalls) == 0 && g.Thread != nil {
+ so := scope.image()
+ if regs, err := g.Thread.Registers(); err == nil {
+ cfa := scope.Regs.CFA
+ frameBase := scope.Regs.FrameBase
+ dwarfRegs := *(scope.BinInfo.Arch.RegistersToDwarfRegisters(so.StaticBase, regs))
+ dwarfRegs.ChangeFunc = g.Thread.SetReg
+ scope.Regs = dwarfRegs
+ scope.Regs.CFA = cfa
+ scope.Regs.FrameBase = frameBase
+ }
+ }
stack.run()
}
@@ -864,7 +888,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
}
@@ -879,25 +903,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
@@ -1087,6 +1121,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-- {
@@ -1109,9 +1148,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, stack.fncallPeek(), regs, curthread)
+ stack.callInjectionContinue = true
+ } else {
+ stack.err = err
+ }
+
case *evalop.CallInjectionStartSpecial:
stack.callInjectionContinue = scope.callInjectionStartSpecial(stack, op, curthread)
@@ -1123,6 +1182,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)
}
@@ -1131,7 +1210,7 @@ func (stack *evalStack) executeOp() {
}
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
}
@@ -1147,9 +1226,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
@@ -1169,6 +1253,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 625a780af6..ef5db8e358 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.debugPinner"
)
type compileCtx struct {
@@ -26,6 +27,8 @@ type compileCtx struct {
ops []Op
allowCalls bool
curCall int
+ flags Flags
+ firstCall bool
}
type evalLookup interface {
@@ -36,14 +39,23 @@ type evalLookup interface {
LookupRegisterName(string) (int, bool)
}
+type Flags uint8
+
+const (
+ CanSet Flags = 1 << iota
+ HasDebugPinner
+)
+
// 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
@@ -53,18 +65,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) {
@@ -77,7 +89,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
@@ -87,7 +99,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
@@ -123,13 +135,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{
@@ -139,11 +155,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) {
@@ -175,6 +220,8 @@ func (ctx *compileCtx) depthCheck(endDepth int) error {
}
}
+ debugPinnerSeen := false
+
for i, op := range ctx.ops {
npop, npush := op.depthCheck()
if depth[i] < npop {
@@ -182,8 +229,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
@@ -547,6 +601,16 @@ func (ctx *compileCtx) compileFunctionCall(node *ast.CallExpr) error {
id := ctx.curCall
ctx.curCall++
+ if ctx.flags&HasDebugPinner != 0 {
+ return ctx.compileFunctionCallNew(node, id)
+ }
+
+ return ctx.compileFunctionCallOld(node, id)
+}
+
+// compileFunctionCallOld compiles a function call when runtime.debugPinner is
+// not available in the target.
+func (ctx *compileCtx) compileFunctionCallOld(node *ast.CallExpr, id int) error {
oldAllowCalls := ctx.allowCalls
oldOps := ctx.ops
ctx.allowCalls = false
@@ -596,6 +660,61 @@ func (ctx *compileCtx) compileFunctionCall(node *ast.CallExpr) error {
return nil
}
+// compileFunctionCallNew compiles a function call when runtime.debugPinner
+// is available in the target.
+func (ctx *compileCtx) compileFunctionCallNew(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/evalop_test.go b/pkg/proc/evalop/evalop_test.go
index a05b5bb895..2d9203c3af 100644
--- a/pkg/proc/evalop/evalop_test.go
+++ b/pkg/proc/evalop/evalop_test.go
@@ -15,7 +15,7 @@ func assertNoError(err error, t testing.TB, s string) {
}
func TestEvalSwitchExhaustiveness(t *testing.T) {
- // Checks that the switch statement in (*EvalScope).evalOne of
+ // Checks that the switch statement in (*EvalScope).executeOp of
// pkg/proc/eval.go exhaustively covers all implementations of the
// evalop.Op interface.
@@ -69,7 +69,7 @@ func TestEvalSwitchExhaustiveness(t *testing.T) {
for op := range ops {
if !ops[op] {
- t.Errorf("evalop.Op %s not used in evalOne", op)
+ t.Errorf("evalop.Op %s not used in executeOp", op)
}
}
}
diff --git a/pkg/proc/evalop/ops.go b/pkg/proc/evalop/ops.go
index a27d2d0f29..8bf4a290d6 100644
--- a/pkg/proc/evalop/ops.go
+++ b/pkg/proc/evalop/ops.go
@@ -153,8 +153,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
}
@@ -166,6 +171,8 @@ const (
JumpIfFalse JumpCond = iota
JumpIfTrue
JumpIfAllocStringChecksFail
+ JumpAlways
+ JumpIfPinningDone
)
// Binary pops two variables from the stack, applies the specified binary
@@ -189,6 +196,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 {
@@ -229,11 +243,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.
@@ -261,3 +294,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 9a29476e53..1204051d4c 100644
--- a/pkg/proc/fncall.go
+++ b/pkg/proc/fncall.go
@@ -42,6 +42,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 +92,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 +134,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 +187,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 +201,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 +304,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.lookupOneFunc(evalop.DebugPinnerFunctionName) != nil,
}
if op.HasFunc {
@@ -365,10 +388,20 @@ 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 {
+ err := cachemem.load()
+ _ = err
+ //TODO: do something with err but what?!
+ }
+}
+
func funcCallFinish(scope *EvalScope, stack *evalStack) {
fncall := stack.fncallPop()
if fncall.err != nil {
@@ -697,7 +730,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 +746,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 +771,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 +907,33 @@ 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)
+ //fmt.Printf("addr %#v %s %s %#v\n", v.Addr, v.Name, v.DwarfType.String(), v.mem)
+ 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
+ })
}
+ sort.Slice(fncall.addrsToPin, func(i, j int) bool { return fncall.addrsToPin[i] < fncall.addrsToPin[j] })
+ fncall.addrsToPin = uniqUint64(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 +960,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 +1011,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 +1062,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 +1083,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
@@ -1180,7 +1253,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,
+ "runtime.debugPinner": true,
+ "runtime.(*Pinner).Unpin": true,
+ "runtime.(*Pinner).Pin": true,
}
// runtimeOptimizedWorkaround modifies the input DIE so that arguments and
@@ -1227,3 +1303,18 @@ func runtimeOptimizedWorkaround(bi *BinaryInfo, image *Image, in *godwarf.Tree)
}
}
}
+
+func uniqUint64(s []uint64) []uint64 {
+ if len(s) == 0 {
+ return s
+ }
+ src, dst := 1, 1
+ for src < len(s) {
+ if s[src] != s[dst-1] {
+ s[dst] = s[src]
+ dst++
+ }
+ src++
+ }
+ return s[:dst]
+}
diff --git a/pkg/proc/mem.go b/pkg/proc/mem.go
index 6138a59ba1..d581d6a5d2 100644
--- a/pkg/proc/mem.go
+++ b/pkg/proc/mem.go
@@ -38,14 +38,25 @@ func (m *memCache) contains(addr uint64, size int) bool {
return addr >= m.cacheAddr && addr <= (m.cacheAddr+uint64(len(m.cache)-size))
}
+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/types.go b/pkg/proc/types.go
index 3a761f68d7..b1e73528d1 100644
--- a/pkg/proc/types.go
+++ b/pkg/proc/types.go
@@ -200,7 +200,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 15eb3b9a85..ff5c7aff84 100644
--- a/pkg/proc/variables.go
+++ b/pkg/proc/variables.go
@@ -86,6 +86,8 @@ const (
VariableCPtr
// VariableCPURegister means this variable is a CPU register.
VariableCPURegister
+
+ variableSaved
)
// Variable represents a variable. It contains the address, name,
@@ -1232,7 +1234,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]
}
@@ -1459,7 +1461,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 655d5b975f..febd9d6314 100644
--- a/pkg/proc/variables_test.go
+++ b/pkg/proc/variables_test.go
@@ -819,7 +819,7 @@ func getEvalExpressionTestCases() []varTest {
{"(unknownthing)(1, 2)", false, "", "", "", errors.New("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")},
+ {"(*afunc)(2)", false, "", "", "", errors.New("*")},
{"unknownthing(2)", false, "", "", "", errors.New("could not evaluate function or type unknownthing: could not find symbol value for unknownthing")},
{"(*unknownthing)(2)", false, "", "", "", errors.New("could not evaluate function or type (*unknownthing): could not find symbol value for unknownthing")},
{"(*strings.Split)(2)", false, "", "", "", errors.New("could not evaluate function or type (*strings.Split): could not find symbol value for strings")},
@@ -903,7 +903,7 @@ 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() {
+ if tc.err.Error() != "*" && tc.err.Error() != err.Error() {
t.Fatalf("Unexpected error. Expected %s got %s", tc.err.Error(), err.Error())
}
}
@@ -1175,9 +1175,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) {
@@ -1189,126 +1190,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(`error evaluating "s1" as argument 1 in function stringsJoin: 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(`error evaluating "s1" as argument 1 in function stringsJoin: 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(`error evaluating "s1" as argument 1 in function strings.Join: could not find symbol value for s1`)},
+ {`strings.Join(s1, comma)`, nil, errors.New(`error evaluating "s1" as argument 1 in function strings.Join: could not find symbol value for s1`), 1},
}
var testcases114 = []testCaseCallFunction{
- {`strings.Join(s1, comma)`, nil, errors.New(`error evaluating "s1" as argument 1 in function strings.Join: could not find symbol value for s1`)},
+ {`strings.Join(s1, comma)`, nil, errors.New(`error evaluating "s1" as argument 1 in function strings.Join: 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) {
@@ -1350,7 +1351,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})
})
}
@@ -1366,6 +1367,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
@@ -1396,7 +1403,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)
@@ -1441,6 +1452,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) {
@@ -1744,3 +1762,15 @@ func TestCapturedVariable(t *testing.T) {
})
})
}
+
+func TestCallFunctionRegisterArg(t *testing.T) {
+ if !goversion.VersionAfterOrEqual(runtime.Version(), 1, 22) {
+ t.Skip("not supported")
+ }
+ withTestProcessArgs("issue3310", t, ".", []string{}, protest.AllNonOptimized, func(p *proc.Target, grp *proc.TargetGroup, fixture protest.Fixture) {
+ setFileBreakpoint(p, t, fixture.Source, 13)
+ assertNoError(grp.Continue(), t, "Continue()")
+ assertNoError(proc.EvalExpressionWithCalls(grp, p.SelectedGoroutine(), "value.Type()", pnormalLoadConfig, true), t, "EvalExpressionWithCalls")
+
+ })
+}