Merge pull request #1599 from nspcc-dev/compiler/debuginfo
compiler: save both VM and smartcontract types
This commit is contained in:
commit
d828096cbf
5 changed files with 140 additions and 67 deletions
30
examples/events/events.go
Normal file
30
examples/events/events.go
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
package events
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
||||||
|
)
|
||||||
|
|
||||||
|
// NotifySomeBytes emits notification with ByteArray.
|
||||||
|
func NotifySomeBytes(arg []byte) {
|
||||||
|
runtime.Notify("SomeBytes", arg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifySomeInteger emits notification with Integer.
|
||||||
|
func NotifySomeInteger(arg int) {
|
||||||
|
runtime.Notify("SomeInteger", arg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifySomeString emits notification with String.
|
||||||
|
func NotifySomeString(arg string) {
|
||||||
|
runtime.Notify("SomeString", arg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifySomeMap emits notification with Map.
|
||||||
|
func NotifySomeMap(arg map[string]int) {
|
||||||
|
runtime.Notify("SomeMap", arg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifySomeArray emits notification with Array.
|
||||||
|
func NotifySomeArray(arg []int) {
|
||||||
|
runtime.Notify("SomeArray", arg)
|
||||||
|
}
|
23
examples/events/events.yml
Normal file
23
examples/events/events.yml
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
name: "Event types example"
|
||||||
|
supportedstandards: []
|
||||||
|
events:
|
||||||
|
- name: SomeBytes
|
||||||
|
parameters:
|
||||||
|
- name: bytes
|
||||||
|
type: ByteArray
|
||||||
|
- name: SomeInteger
|
||||||
|
parameters:
|
||||||
|
- name: int
|
||||||
|
type: Integer
|
||||||
|
- name: SomeString
|
||||||
|
parameters:
|
||||||
|
- name: str
|
||||||
|
type: String
|
||||||
|
- name: SomeMap
|
||||||
|
parameters:
|
||||||
|
- name: m
|
||||||
|
type: Map
|
||||||
|
- name: SomeArray
|
||||||
|
parameters:
|
||||||
|
- name: a
|
||||||
|
type: Array
|
|
@ -877,7 +877,8 @@ func (c *codegen) Visit(node ast.Node) ast.Visitor {
|
||||||
tv := c.typeAndValueOf(n.Args[0])
|
tv := c.typeAndValueOf(n.Args[0])
|
||||||
params := make([]string, 0, len(n.Args[1:]))
|
params := make([]string, 0, len(n.Args[1:]))
|
||||||
for _, p := range n.Args[1:] {
|
for _, p := range n.Args[1:] {
|
||||||
params = append(params, c.scTypeFromExpr(p))
|
st, _ := c.scAndVMTypeFromExpr(p)
|
||||||
|
params = append(params, st.String())
|
||||||
}
|
}
|
||||||
// Sometimes event name is stored in a var.
|
// Sometimes event name is stored in a var.
|
||||||
// Skip in this case.
|
// Skip in this case.
|
||||||
|
|
|
@ -13,6 +13,7 @@ import (
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||||||
)
|
)
|
||||||
|
|
||||||
// DebugInfo represents smart-contract debug information.
|
// DebugInfo represents smart-contract debug information.
|
||||||
|
@ -42,8 +43,10 @@ type MethodDebugInfo struct {
|
||||||
// Parameters is a list of method's parameters.
|
// Parameters is a list of method's parameters.
|
||||||
Parameters []DebugParam `json:"params"`
|
Parameters []DebugParam `json:"params"`
|
||||||
// ReturnType is method's return type.
|
// ReturnType is method's return type.
|
||||||
ReturnType string `json:"return"`
|
ReturnType string `json:"return"`
|
||||||
Variables []string `json:"variables"`
|
// ReturnTypeSC is return type to use in manifest.
|
||||||
|
ReturnTypeSC smartcontract.ParamType `json:"-"`
|
||||||
|
Variables []string `json:"variables"`
|
||||||
// SeqPoints is a map between source lines and byte-code instruction offsets.
|
// SeqPoints is a map between source lines and byte-code instruction offsets.
|
||||||
SeqPoints []DebugSeqPoint `json:"sequence-points"`
|
SeqPoints []DebugSeqPoint `json:"sequence-points"`
|
||||||
}
|
}
|
||||||
|
@ -86,8 +89,9 @@ type DebugRange struct {
|
||||||
|
|
||||||
// DebugParam represents variables's name and type.
|
// DebugParam represents variables's name and type.
|
||||||
type DebugParam struct {
|
type DebugParam struct {
|
||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
Type string `json:"type"`
|
Type string `json:"type"`
|
||||||
|
TypeSC smartcontract.ParamType `json:"-"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *codegen) saveSequencePoint(n ast.Node) {
|
func (c *codegen) saveSequencePoint(n ast.Node) {
|
||||||
|
@ -128,8 +132,9 @@ func (c *codegen) emitDebugInfo(contract []byte) *DebugInfo {
|
||||||
Start: 0,
|
Start: 0,
|
||||||
End: uint16(c.initEndOffset),
|
End: uint16(c.initEndOffset),
|
||||||
},
|
},
|
||||||
ReturnType: "Void",
|
ReturnType: "Void",
|
||||||
SeqPoints: c.sequencePoints["init"],
|
ReturnTypeSC: smartcontract.VoidType,
|
||||||
|
SeqPoints: c.sequencePoints["init"],
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if c.deployEndOffset >= 0 {
|
if c.deployEndOffset >= 0 {
|
||||||
|
@ -146,11 +151,13 @@ func (c *codegen) emitDebugInfo(contract []byte) *DebugInfo {
|
||||||
End: uint16(c.deployEndOffset),
|
End: uint16(c.deployEndOffset),
|
||||||
},
|
},
|
||||||
Parameters: []DebugParam{{
|
Parameters: []DebugParam{{
|
||||||
Name: "isUpdate",
|
Name: "isUpdate",
|
||||||
Type: "Boolean",
|
Type: "Boolean",
|
||||||
|
TypeSC: smartcontract.BoolType,
|
||||||
}},
|
}},
|
||||||
ReturnType: "Void",
|
ReturnType: "Void",
|
||||||
SeqPoints: c.sequencePoints[manifest.MethodDeploy],
|
ReturnTypeSC: smartcontract.VoidType,
|
||||||
|
SeqPoints: c.sequencePoints[manifest.MethodDeploy],
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
for name, scope := range c.funcs {
|
for name, scope := range c.funcs {
|
||||||
|
@ -169,8 +176,8 @@ func (c *codegen) registerDebugVariable(name string, expr ast.Expr) {
|
||||||
// do not save globals for now
|
// do not save globals for now
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
typ := c.scTypeFromExpr(expr)
|
_, vt := c.scAndVMTypeFromExpr(expr)
|
||||||
c.scope.variables = append(c.scope.variables, name+","+typ)
|
c.scope.variables = append(c.scope.variables, name+","+vt.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *codegen) methodInfoFromScope(name string, scope *funcScope) *MethodDebugInfo {
|
func (c *codegen) methodInfoFromScope(name string, scope *funcScope) *MethodDebugInfo {
|
||||||
|
@ -178,48 +185,53 @@ func (c *codegen) methodInfoFromScope(name string, scope *funcScope) *MethodDebu
|
||||||
params := make([]DebugParam, 0, ps.NumFields())
|
params := make([]DebugParam, 0, ps.NumFields())
|
||||||
for i := range ps.List {
|
for i := range ps.List {
|
||||||
for j := range ps.List[i].Names {
|
for j := range ps.List[i].Names {
|
||||||
|
st, vt := c.scAndVMTypeFromExpr(ps.List[i].Type)
|
||||||
params = append(params, DebugParam{
|
params = append(params, DebugParam{
|
||||||
Name: ps.List[i].Names[j].Name,
|
Name: ps.List[i].Names[j].Name,
|
||||||
Type: c.scTypeFromExpr(ps.List[i].Type),
|
Type: vt.String(),
|
||||||
|
TypeSC: st,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ss := strings.Split(name, ".")
|
ss := strings.Split(name, ".")
|
||||||
name = ss[len(ss)-1]
|
name = ss[len(ss)-1]
|
||||||
r, n := utf8.DecodeRuneInString(name)
|
r, n := utf8.DecodeRuneInString(name)
|
||||||
|
st, vt := c.scAndVMReturnTypeFromScope(scope)
|
||||||
return &MethodDebugInfo{
|
return &MethodDebugInfo{
|
||||||
ID: name,
|
ID: name,
|
||||||
Name: DebugMethodName{
|
Name: DebugMethodName{
|
||||||
Name: string(unicode.ToLower(r)) + name[n:],
|
Name: string(unicode.ToLower(r)) + name[n:],
|
||||||
Namespace: scope.pkg.Name(),
|
Namespace: scope.pkg.Name(),
|
||||||
},
|
},
|
||||||
IsExported: scope.decl.Name.IsExported(),
|
IsExported: scope.decl.Name.IsExported(),
|
||||||
IsFunction: scope.decl.Recv == nil,
|
IsFunction: scope.decl.Recv == nil,
|
||||||
Range: scope.rng,
|
Range: scope.rng,
|
||||||
Parameters: params,
|
Parameters: params,
|
||||||
ReturnType: c.scReturnTypeFromScope(scope),
|
ReturnType: vt,
|
||||||
SeqPoints: c.sequencePoints[name],
|
ReturnTypeSC: st,
|
||||||
Variables: scope.variables,
|
SeqPoints: c.sequencePoints[name],
|
||||||
|
Variables: scope.variables,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *codegen) scReturnTypeFromScope(scope *funcScope) string {
|
func (c *codegen) scAndVMReturnTypeFromScope(scope *funcScope) (smartcontract.ParamType, string) {
|
||||||
results := scope.decl.Type.Results
|
results := scope.decl.Type.Results
|
||||||
switch results.NumFields() {
|
switch results.NumFields() {
|
||||||
case 0:
|
case 0:
|
||||||
return "Void"
|
return smartcontract.VoidType, "Void"
|
||||||
case 1:
|
case 1:
|
||||||
return c.scTypeFromExpr(results.List[0].Type)
|
st, vt := c.scAndVMTypeFromExpr(results.List[0].Type)
|
||||||
|
return st, vt.String()
|
||||||
default:
|
default:
|
||||||
// multiple return values are not supported in debugger
|
// multiple return values are not supported in debugger
|
||||||
return "Any"
|
return smartcontract.AnyType, "Any"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *codegen) scTypeFromExpr(typ ast.Expr) string {
|
func (c *codegen) scAndVMTypeFromExpr(typ ast.Expr) (smartcontract.ParamType, stackitem.Type) {
|
||||||
t := c.typeOf(typ)
|
t := c.typeOf(typ)
|
||||||
if c.typeOf(typ) == nil {
|
if c.typeOf(typ) == nil {
|
||||||
return "Any"
|
return smartcontract.AnyType, stackitem.AnyT
|
||||||
}
|
}
|
||||||
if named, ok := t.(*types.Named); ok {
|
if named, ok := t.(*types.Named); ok {
|
||||||
if isInteropPath(named.String()) {
|
if isInteropPath(named.String()) {
|
||||||
|
@ -227,13 +239,22 @@ func (c *codegen) scTypeFromExpr(typ ast.Expr) string {
|
||||||
pkg := named.Obj().Pkg().Name()
|
pkg := named.Obj().Pkg().Name()
|
||||||
switch pkg {
|
switch pkg {
|
||||||
case "blockchain", "contract":
|
case "blockchain", "contract":
|
||||||
return "Array" // Block, Transaction, Contract
|
return smartcontract.ArrayType, stackitem.ArrayT // Block, Transaction, Contract
|
||||||
case "interop":
|
case "interop":
|
||||||
if name != "Interface" {
|
if name != "Interface" {
|
||||||
return name
|
switch name {
|
||||||
|
case "Hash160":
|
||||||
|
return smartcontract.Hash160Type, stackitem.ByteArrayT
|
||||||
|
case "Hash256":
|
||||||
|
return smartcontract.Hash256Type, stackitem.ByteArrayT
|
||||||
|
case "PublicKey":
|
||||||
|
return smartcontract.PublicKeyType, stackitem.ByteArrayT
|
||||||
|
case "Signature":
|
||||||
|
return smartcontract.SignatureType, stackitem.ByteArrayT
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return "InteropInterface"
|
return smartcontract.InteropInterfaceType, stackitem.InteropT
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
switch t := t.Underlying().(type) {
|
switch t := t.Underlying().(type) {
|
||||||
|
@ -241,25 +262,25 @@ func (c *codegen) scTypeFromExpr(typ ast.Expr) string {
|
||||||
info := t.Info()
|
info := t.Info()
|
||||||
switch {
|
switch {
|
||||||
case info&types.IsInteger != 0:
|
case info&types.IsInteger != 0:
|
||||||
return "Integer"
|
return smartcontract.IntegerType, stackitem.IntegerT
|
||||||
case info&types.IsBoolean != 0:
|
case info&types.IsBoolean != 0:
|
||||||
return "Boolean"
|
return smartcontract.BoolType, stackitem.BooleanT
|
||||||
case info&types.IsString != 0:
|
case info&types.IsString != 0:
|
||||||
return "String"
|
return smartcontract.StringType, stackitem.ByteArrayT
|
||||||
default:
|
default:
|
||||||
return "Any"
|
return smartcontract.AnyType, stackitem.AnyT
|
||||||
}
|
}
|
||||||
case *types.Map:
|
case *types.Map:
|
||||||
return "Map"
|
return smartcontract.MapType, stackitem.MapT
|
||||||
case *types.Struct:
|
case *types.Struct:
|
||||||
return "Struct"
|
return smartcontract.ArrayType, stackitem.StructT
|
||||||
case *types.Slice:
|
case *types.Slice:
|
||||||
if isByte(t.Elem()) {
|
if isByte(t.Elem()) {
|
||||||
return "ByteString"
|
return smartcontract.ByteArrayType, stackitem.ByteArrayT
|
||||||
}
|
}
|
||||||
return "Array"
|
return smartcontract.ArrayType, stackitem.ArrayT
|
||||||
default:
|
default:
|
||||||
return "Any"
|
return smartcontract.AnyType, stackitem.AnyT
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -310,13 +331,9 @@ func (d *DebugParam) UnmarshalJSON(data []byte) error {
|
||||||
|
|
||||||
// ToManifestParameter converts DebugParam to manifest.Parameter
|
// ToManifestParameter converts DebugParam to manifest.Parameter
|
||||||
func (d *DebugParam) ToManifestParameter() (manifest.Parameter, error) {
|
func (d *DebugParam) ToManifestParameter() (manifest.Parameter, error) {
|
||||||
pType, err := smartcontract.ParseParamType(d.Type)
|
|
||||||
if err != nil {
|
|
||||||
return manifest.Parameter{}, err
|
|
||||||
}
|
|
||||||
return manifest.Parameter{
|
return manifest.Parameter{
|
||||||
Name: d.Name,
|
Name: d.Name,
|
||||||
Type: pType,
|
Type: d.TypeSC,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -333,14 +350,10 @@ func (m *MethodDebugInfo) ToManifestMethod() (manifest.Method, error) {
|
||||||
return result, err
|
return result, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
returnType, err := smartcontract.ParseParamType(m.ReturnType)
|
|
||||||
if err != nil {
|
|
||||||
return result, err
|
|
||||||
}
|
|
||||||
result.Name = m.Name.Name
|
result.Name = m.Name.Name
|
||||||
result.Offset = int(m.Range.Start)
|
result.Offset = int(m.Range.Start)
|
||||||
result.Parameters = parameters
|
result.Parameters = parameters
|
||||||
result.ReturnType = returnType
|
result.ReturnType = m.ReturnTypeSC
|
||||||
return result, nil
|
return result, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -71,8 +71,8 @@ func _deploy(isUpdate bool) {}
|
||||||
t.Run("return types", func(t *testing.T) {
|
t.Run("return types", func(t *testing.T) {
|
||||||
returnTypes := map[string]string{
|
returnTypes := map[string]string{
|
||||||
"MethodInt": "Integer",
|
"MethodInt": "Integer",
|
||||||
"MethodConcat": "String",
|
"MethodConcat": "ByteString",
|
||||||
"MethodString": "String", "MethodByteArray": "ByteString",
|
"MethodString": "ByteString", "MethodByteArray": "ByteString",
|
||||||
"MethodArray": "Array", "MethodStruct": "Struct",
|
"MethodArray": "Array", "MethodStruct": "Struct",
|
||||||
"Main": "Boolean",
|
"Main": "Boolean",
|
||||||
"unexportedMethod": "Integer",
|
"unexportedMethod": "Integer",
|
||||||
|
@ -89,7 +89,7 @@ func _deploy(isUpdate bool) {}
|
||||||
|
|
||||||
t.Run("variables", func(t *testing.T) {
|
t.Run("variables", func(t *testing.T) {
|
||||||
vars := map[string][]string{
|
vars := map[string][]string{
|
||||||
"Main": {"s,String", "res,Integer"},
|
"Main": {"s,ByteString", "res,Integer"},
|
||||||
}
|
}
|
||||||
for i := range d.Methods {
|
for i := range d.Methods {
|
||||||
v, ok := vars[d.Methods[i].ID]
|
v, ok := vars[d.Methods[i].ID]
|
||||||
|
@ -102,30 +102,36 @@ func _deploy(isUpdate bool) {}
|
||||||
t.Run("param types", func(t *testing.T) {
|
t.Run("param types", func(t *testing.T) {
|
||||||
paramTypes := map[string][]DebugParam{
|
paramTypes := map[string][]DebugParam{
|
||||||
"_deploy": {{
|
"_deploy": {{
|
||||||
Name: "isUpdate",
|
Name: "isUpdate",
|
||||||
Type: "Boolean",
|
Type: "Boolean",
|
||||||
|
TypeSC: smartcontract.BoolType,
|
||||||
}},
|
}},
|
||||||
"MethodInt": {{
|
"MethodInt": {{
|
||||||
Name: "a",
|
Name: "a",
|
||||||
Type: "String",
|
Type: "ByteString",
|
||||||
|
TypeSC: smartcontract.StringType,
|
||||||
}},
|
}},
|
||||||
"MethodConcat": {
|
"MethodConcat": {
|
||||||
{
|
{
|
||||||
Name: "a",
|
Name: "a",
|
||||||
Type: "String",
|
Type: "ByteString",
|
||||||
|
TypeSC: smartcontract.StringType,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "b",
|
Name: "b",
|
||||||
Type: "String",
|
Type: "ByteString",
|
||||||
|
TypeSC: smartcontract.StringType,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "c",
|
Name: "c",
|
||||||
Type: "String",
|
Type: "ByteString",
|
||||||
|
TypeSC: smartcontract.StringType,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
"Main": {{
|
"Main": {{
|
||||||
Name: "op",
|
Name: "op",
|
||||||
Type: "String",
|
Type: "ByteString",
|
||||||
|
TypeSC: smartcontract.StringType,
|
||||||
}},
|
}},
|
||||||
}
|
}
|
||||||
for i := range d.Methods {
|
for i := range d.Methods {
|
||||||
|
@ -303,8 +309,8 @@ func TestDebugInfo_MarshalJSON(t *testing.T) {
|
||||||
},
|
},
|
||||||
Range: DebugRange{Start: 10, End: 20},
|
Range: DebugRange{Start: 10, End: 20},
|
||||||
Parameters: []DebugParam{
|
Parameters: []DebugParam{
|
||||||
{"param1", "Integer"},
|
{Name: "param1", Type: "Integer"},
|
||||||
{"ok", "Boolean"},
|
{Name: "ok", Type: "Boolean"},
|
||||||
},
|
},
|
||||||
ReturnType: "ByteString",
|
ReturnType: "ByteString",
|
||||||
Variables: []string{},
|
Variables: []string{},
|
||||||
|
|
Loading…
Reference in a new issue