From 78948ef7af8fd7246d2c9c8536499787d0620a63 Mon Sep 17 00:00:00 2001 From: Evgenii Stratonikov Date: Tue, 15 Sep 2020 09:52:56 +0300 Subject: [PATCH 1/3] compiler: emit error for non-byte subslices They are not supported for now, as VM has only `SUBSTR` opcode for Buffers (`[]byte` in Go). --- pkg/compiler/codegen.go | 4 ++++ pkg/compiler/slice_test.go | 11 +++++++++++ 2 files changed, 15 insertions(+) diff --git a/pkg/compiler/codegen.go b/pkg/compiler/codegen.go index be9e8ec2c..7aa889e0a 100644 --- a/pkg/compiler/codegen.go +++ b/pkg/compiler/codegen.go @@ -517,6 +517,10 @@ func (c *codegen) Visit(node ast.Node) ast.Visitor { return nil case *ast.SliceExpr: + if isCompoundSlice(c.typeOf(n.X.(*ast.Ident)).Underlying()) { + c.prog.Err = errors.New("subslices are supported only for []byte") + return nil + } name := n.X.(*ast.Ident).Name c.emitLoadVar("", name) diff --git a/pkg/compiler/slice_test.go b/pkg/compiler/slice_test.go index ee776407f..7d14650d5 100644 --- a/pkg/compiler/slice_test.go +++ b/pkg/compiler/slice_test.go @@ -316,6 +316,17 @@ func TestSliceOperations(t *testing.T) { runTestCases(t, sliceTestCases) } +func TestSubsliceCompound(t *testing.T) { + src := `package foo + func Main() []int { + a := []int{0, 1, 2, 3} + b := a[1:3] + return b + }` + _, err := compiler.Compile("", strings.NewReader(src)) + require.Error(t, err) +} + func TestJumps(t *testing.T) { src := ` package foo From bcc11cbd741c55c4d7df27fc4305e606caa71dd3 Mon Sep 17 00:00:00 2001 From: Evgenii Stratonikov Date: Tue, 15 Sep 2020 10:05:41 +0300 Subject: [PATCH 2/3] compiler: support removing slice elements Go-way of removing elements from slice is via `append` builtin. There is a separate opcode for removing elements from Arrays, which is cheaper and supported in this commit. --- pkg/compiler/analysis.go | 2 +- pkg/compiler/codegen.go | 6 ++++++ pkg/compiler/slice_test.go | 24 ++++++++++++++++++++++++ pkg/interop/util/util.go | 5 +++++ 4 files changed, 36 insertions(+), 1 deletion(-) diff --git a/pkg/compiler/analysis.go b/pkg/compiler/analysis.go index cd9e9148d..0dcc3d262 100644 --- a/pkg/compiler/analysis.go +++ b/pkg/compiler/analysis.go @@ -17,7 +17,7 @@ var ( goBuiltins = []string{"len", "append", "panic", "make", "copy", "recover", "delete"} // Custom builtin utility functions. customBuiltins = []string{ - "FromAddress", "Equals", + "FromAddress", "Equals", "Remove", "ToBool", "ToByteArray", "ToInteger", } ) diff --git a/pkg/compiler/codegen.go b/pkg/compiler/codegen.go index 7aa889e0a..8ce385f12 100644 --- a/pkg/compiler/codegen.go +++ b/pkg/compiler/codegen.go @@ -1542,6 +1542,12 @@ func (c *codegen) convertBuiltin(expr *ast.CallExpr) { typ = stackitem.BooleanT } c.emitConvert(typ) + case "Remove": + if !isCompoundSlice(c.typeOf(expr.Args[0])) { + c.prog.Err = errors.New("`Remove` supports only non-byte slices") + return + } + emit.Opcode(c.prog.BinWriter, opcode.REMOVE) case "Equals": emit.Opcode(c.prog.BinWriter, opcode.EQUAL) case "FromAddress": diff --git a/pkg/compiler/slice_test.go b/pkg/compiler/slice_test.go index 7d14650d5..e4bc8c6e9 100644 --- a/pkg/compiler/slice_test.go +++ b/pkg/compiler/slice_test.go @@ -327,6 +327,30 @@ func TestSubsliceCompound(t *testing.T) { require.Error(t, err) } +func TestRemove(t *testing.T) { + t.Run("Valid", func(t *testing.T) { + src := `package foo + import "github.com/nspcc-dev/neo-go/pkg/interop/util" + func Main() int { + a := []int{11, 22, 33} + util.Remove(a, 1) + return len(a) + a[0] + a[1] + }` + eval(t, src, big.NewInt(46)) + }) + t.Run("ByteSlice", func(t *testing.T) { + src := `package foo + import "github.com/nspcc-dev/neo-go/pkg/interop/util" + func Main() int { + a := []byte{11, 22, 33} + util.Remove(a, 1) + return len(a) + }` + _, err := compiler.Compile("", strings.NewReader(src)) + require.Error(t, err) + }) +} + func TestJumps(t *testing.T) { src := ` package foo diff --git a/pkg/interop/util/util.go b/pkg/interop/util/util.go index 3e80a77b0..030918f43 100644 --- a/pkg/interop/util/util.go +++ b/pkg/interop/util/util.go @@ -17,3 +17,8 @@ func FromAddress(address string) []byte { func Equals(a, b interface{}) bool { return false } + +// Remove removes element with index i from slice. +// This is done in place and slice must have type other than `[]byte`. +func Remove(slice interface{}, i int) { +} From 5ba22a02f40d8d21755859d4122680f0881d1d16 Mon Sep 17 00:00:00 2001 From: Evgenii Stratonikov Date: Tue, 15 Sep 2020 16:33:34 +0300 Subject: [PATCH 3/3] docs: mention `util` and `convert` packages in compiler --- docs/compiler.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/docs/compiler.md b/docs/compiler.md index 79069be64..ebee25c9d 100644 --- a/docs/compiler.md +++ b/docs/compiler.md @@ -32,6 +32,9 @@ pkg.go.dev](https://pkg.go.dev/github.com/nspcc-dev/neo-go/pkg/interop) for full API documentation. In general it provides the same level of functionality as Neo .net Framework library. +Compiler provides some helpful builtins in `util` and `convert` packages. +Refer to them for detailed documentation. + ## Quick start ### Compiling