compiler: save both VM and smartcontract types
VM types are used in debugger, while smartcontract ones are used in manifest. We can't save only one of them, because conversion in either side is lossy: 1. VM has `Array` and `Struct` but smartcontract only has `Array`. 2. Smartcontract has `Hash160` etc, which are all `ByteString` or `Buffer` in VM. And to spice things a bit more, return type in debugger can be `Void`, which corresponds to no real stackitem type (as it must exist).
This commit is contained in:
parent
3d0ed6eac3
commit
cbf26f315c
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 {
|
||||||
|
@ -307,8 +313,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