From 5a6f49cda570b0c9f67decaae124e86075d057d0 Mon Sep 17 00:00:00 2001 From: deelawn Date: Tue, 21 May 2024 09:59:56 +0200 Subject: [PATCH 1/3] blank identifier constants --- gnovm/pkg/gnolang/helpers.go | 2 +- gnovm/pkg/gnolang/machine.go | 4 ++-- gnovm/pkg/gnolang/nodes.go | 12 +++++------ gnovm/pkg/gnolang/op_eval.go | 4 ++-- gnovm/pkg/gnolang/preprocess.go | 35 ++++++++++++++++++--------------- gnovm/pkg/gnolang/values.go | 10 +++++----- 6 files changed, 35 insertions(+), 32 deletions(-) diff --git a/gnovm/pkg/gnolang/helpers.go b/gnovm/pkg/gnolang/helpers.go index b163b6a52a7..564ac0622c2 100644 --- a/gnovm/pkg/gnolang/helpers.go +++ b/gnovm/pkg/gnolang/helpers.go @@ -91,7 +91,7 @@ func Flds(args ...interface{}) FieldTypeExprs { func Recv(n, t interface{}) FieldTypeExpr { if n == "" { - n = "_" + n = blankIdentifier } return FieldTypeExpr{ Name: N(n), diff --git a/gnovm/pkg/gnolang/machine.go b/gnovm/pkg/gnolang/machine.go index 68eb44290e2..40174e1771b 100644 --- a/gnovm/pkg/gnolang/machine.go +++ b/gnovm/pkg/gnolang/machine.go @@ -315,7 +315,7 @@ func checkDuplicates(fset *FileSet) bool { name = d.Name case *ValueDecl: for _, nx := range d.NameExprs { - if nx.Name == "_" { + if nx.Name == blankNameIdentifer { continue } if _, ok := defined[nx.Name]; ok { @@ -327,7 +327,7 @@ func checkDuplicates(fset *FileSet) bool { default: continue } - if name == "_" { + if name == blankNameIdentifer { continue } if _, ok := defined[name]; ok { diff --git a/gnovm/pkg/gnolang/nodes.go b/gnovm/pkg/gnolang/nodes.go index 8f2c5054a8a..0c182749f92 100644 --- a/gnovm/pkg/gnolang/nodes.go +++ b/gnovm/pkg/gnolang/nodes.go @@ -1014,7 +1014,7 @@ type ValueDecl struct { func (x *ValueDecl) GetDeclNames() []Name { ns := make([]Name, 0, len(x.NameExprs)) for _, nx := range x.NameExprs { - if nx.Name == "_" { + if nx.Name == blankNameIdentifer { // ignore } else { ns = append(ns, nx.Name) @@ -1031,7 +1031,7 @@ type TypeDecl struct { } func (x *TypeDecl) GetDeclNames() []Name { - if x.NameExpr.Name == "_" { + if x.NameExpr.Name == blankNameIdentifer { return nil // ignore } else { return []Name{x.NameExpr.Name} @@ -1588,8 +1588,8 @@ func (sb *StaticBlock) GetParentNode(store Store) BlockNode { // Implements BlockNode. // As a side effect, notes externally defined names. func (sb *StaticBlock) GetPathForName(store Store, n Name) ValuePath { - if n == "_" { - return NewValuePathBlock(0, 0, "_") + if n == blankNameIdentifer { + return NewValuePathBlock(0, 0, blankNameIdentifer) } // Check local. gen := 1 @@ -1780,7 +1780,7 @@ func (sb *StaticBlock) Define2(isConst bool, n Name, st Type, tv TypedValue) { if tv.T == nil && tv.V != nil { panic("StaticBlock.Define2() requires .T if .V is set") } - if n == "_" { + if n == blankNameIdentifer { return // ignore } idx, exists := sb.GetLocalIndex(n) @@ -1976,7 +1976,7 @@ func (vp ValuePath) Validate() { panic("uverse value path must have depth 0") } case VPBlock: - // 0 ok ("_" blank) + // 0 ok (blankIdentifier blank) case VPField: if vp.Depth > 1 { panic("field value path must have depth 0 or 1") diff --git a/gnovm/pkg/gnolang/op_eval.go b/gnovm/pkg/gnolang/op_eval.go index bf4ac88aa25..701615fff13 100644 --- a/gnovm/pkg/gnolang/op_eval.go +++ b/gnovm/pkg/gnolang/op_eval.go @@ -47,7 +47,7 @@ func (m *Machine) doOpEval() { m.PopExpr() switch x.Kind { case INT: - x.Value = strings.ReplaceAll(x.Value, "_", "") + x.Value = strings.ReplaceAll(x.Value, blankIdentifier, "") // temporary optimization bi := big.NewInt(0) // TODO optimize. @@ -84,7 +84,7 @@ func (m *Machine) doOpEval() { V: BigintValue{V: bi}, }) case FLOAT: - x.Value = strings.ReplaceAll(x.Value, "_", "") + x.Value = strings.ReplaceAll(x.Value, blankIdentifier, "") if reFloat.MatchString(x.Value) { value := x.Value diff --git a/gnovm/pkg/gnolang/preprocess.go b/gnovm/pkg/gnolang/preprocess.go index ddc72c3a048..02c38025962 100644 --- a/gnovm/pkg/gnolang/preprocess.go +++ b/gnovm/pkg/gnolang/preprocess.go @@ -9,7 +9,10 @@ import ( "github.com/gnolang/gno/tm2/pkg/errors" ) -const blankIdentifer string = "_" +const ( + blankIdentifier string = "_" + blankNameIdentifer Name = "_" +) // In the case of a *FileSet, some declaration steps have to happen // in a restricted parallel way across all the files. @@ -191,7 +194,7 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { var defined bool for _, lx := range n.Lhs { ln := lx.(*NameExpr).Name - if ln == "_" { + if ln == blankNameIdentifer { // ignore. } else { _, ok := last.GetLocalIndex(ln) @@ -235,7 +238,7 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { case *FuncTypeExpr: for i := range n.Params { p := &n.Params[i] - if p.Name == "" || p.Name == "_" { + if p.Name == "" || p.Name == blankNameIdentifer { // create a hidden var with leading dot. // NOTE: document somewhere. pn := fmt.Sprintf(".arg_%d", i) @@ -244,7 +247,7 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { } for i := range n.Results { r := &n.Results[i] - if r.Name == "_" { + if r.Name == blankNameIdentifer { // create a hidden var with leading dot. // NOTE: document somewhere. rn := fmt.Sprintf(".res_%d", i) @@ -683,8 +686,8 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { } // specific and general cases switch n.Name { - case "_": - n.Path = NewValuePathBlock(0, 0, "_") + case blankNameIdentifer: + n.Path = NewValuePathBlock(0, 0, blankNameIdentifer) return n, TRANS_CONTINUE case "iota": pd := lastDecl(ns) @@ -1241,7 +1244,7 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { } // Type assertions on the blank identifier are illegal. - if nx, ok := n.X.(*NameExpr); ok && string(nx.Name) == blankIdentifer { + if nx, ok := n.X.(*NameExpr); ok && string(nx.Name) == blankIdentifier { panic("cannot use _ as value or type") } @@ -1924,8 +1927,8 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { pn := fn.GetParentNode(nil).(*PackageNode) for i := 0; i < numNames; i++ { nx := &n.NameExprs[i] - if nx.Name == "_" { - nx.Path = NewValuePathBlock(0, 0, "_") + if nx.Name == blankNameIdentifer { + nx.Path = NewValuePathBlock(0, 0, blankNameIdentifer) } else { pn.Define2(n.Const, nx.Name, sts[i], tvs[i]) nx.Path = last.GetPathForName(nil, nx.Name) @@ -1934,8 +1937,8 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { } else { for i := 0; i < numNames; i++ { nx := &n.NameExprs[i] - if nx.Name == "_" { - nx.Path = NewValuePathBlock(0, 0, "_") + if nx.Name == blankNameIdentifer { + nx.Path = NewValuePathBlock(0, 0, blankNameIdentifer) } else { last.Define2(n.Const, nx.Name, sts[i], tvs[i]) nx.Path = last.GetPathForName(nil, nx.Name) @@ -3049,7 +3052,7 @@ func predefineNow2(store Store, last BlockNode, d Decl, m map[Name]struct{}) (De // NOTE: unlike the *ValueDecl case, this case doesn't // preprocess d itself (only d.Type). if cd.IsMethod { - if cd.Recv.Name == "" || cd.Recv.Name == "_" { + if cd.Recv.Name == "" || cd.Recv.Name == blankNameIdentifer { // create a hidden var with leading dot. // NOTE: document somewhere. cd.Recv.Name = ".recv" @@ -3168,7 +3171,7 @@ func tryPredefine(store Store, last BlockNode, d Decl) (un Name) { } if d.Name == "" { // use default d.Name = pv.PkgName - } else if d.Name == "_" { // no definition + } else if d.Name == blankNameIdentifer { // no definition return } else if d.Name == "." { // dot import panic("dot imports not allowed in Gno") @@ -3195,8 +3198,8 @@ func tryPredefine(store Store, last BlockNode, d Decl) (un Name) { } for i := 0; i < len(d.NameExprs); i++ { nx := &d.NameExprs[i] - if nx.Name == "_" { - nx.Path.Name = "_" + if nx.Name == blankNameIdentifer { + nx.Path.Name = blankNameIdentifer } else { last2 := skipFile(last) last2.Predefine(d.Const, nx.Name) @@ -3376,7 +3379,7 @@ func constUntypedBigint(source Expr, i64 int64) *ConstExpr { } func fillNameExprPath(last BlockNode, nx *NameExpr, isDefineLHS bool) { - if nx.Name == "_" { + if nx.Name == blankNameIdentifer { // Blank name has no path; caller error. panic("should not happen") } diff --git a/gnovm/pkg/gnolang/values.go b/gnovm/pkg/gnolang/values.go index 948730c4697..fff2396de47 100644 --- a/gnovm/pkg/gnolang/values.go +++ b/gnovm/pkg/gnolang/values.go @@ -163,7 +163,7 @@ func (dbv DataByteValue) SetByte(b byte) { // value after a block var escapes "to the heap". // *(PointerValue.TypedValue) must have already become // initialized, namely T set if a typed-nil. -// Index is -1 for the shared "_" block var, +// Index is -1 for the shared blankIdentifier block var, // and -2 for (gno and native) map items. // // Allocation for PointerValue is not immediate, @@ -179,7 +179,7 @@ type PointerValue struct { } const ( - PointerIndexBlockBlank = -1 // for the "_" identifier in blocks + PointerIndexBlockBlank = -1 // for the blankIdentifier identifier in blocks PointerIndexMap = -2 // Base is Map, use Key. PointerIndexNative = -3 // Base is *NativeValue. ) @@ -2257,7 +2257,7 @@ type Block struct { Source BlockNode Values []TypedValue Parent Value - Blank TypedValue // captures "_" // XXX remove and replace with global instance. + Blank TypedValue // captures blankIdentifier // XXX remove and replace with global instance. bodyStmt bodyStmt // XXX expose for persistence, not needed for MVP. } @@ -2343,7 +2343,7 @@ func (b *Block) GetPointerToInt(store Store, index int) PointerValue { func (b *Block) GetPointerTo(store Store, path ValuePath) PointerValue { if path.IsBlockBlankPath() { if debug { - if path.Name != "_" { + if path.Name != blankNameIdentifer { panic(fmt.Sprintf( "zero value path is reserved for \"_\", but got %s", path.Name)) @@ -2365,7 +2365,7 @@ func (b *Block) GetPointerTo(store Store, path ValuePath) PointerValue { return b.GetPointerToInt(store, int(path.Index)) } -// Result is used has lhs for any assignments to "_". +// Result is used has lhs for any assignments to blankIdentifier. func (b *Block) GetBlankRef() *TypedValue { return &b.Blank } From fcf2b2389e4b78aa23f9e6a7dd8b35c4f10cd888 Mon Sep 17 00:00:00 2001 From: deelawn Date: Tue, 28 May 2024 07:20:46 -0700 Subject: [PATCH 2/3] use untyped constant --- gnovm/pkg/gnolang/machine.go | 4 ++-- gnovm/pkg/gnolang/nodes.go | 10 +++++----- gnovm/pkg/gnolang/preprocess.go | 31 +++++++++++++++---------------- gnovm/pkg/gnolang/values.go | 2 +- 4 files changed, 23 insertions(+), 24 deletions(-) diff --git a/gnovm/pkg/gnolang/machine.go b/gnovm/pkg/gnolang/machine.go index 40174e1771b..2d939246dd9 100644 --- a/gnovm/pkg/gnolang/machine.go +++ b/gnovm/pkg/gnolang/machine.go @@ -315,7 +315,7 @@ func checkDuplicates(fset *FileSet) bool { name = d.Name case *ValueDecl: for _, nx := range d.NameExprs { - if nx.Name == blankNameIdentifer { + if nx.Name == blankIdentifier { continue } if _, ok := defined[nx.Name]; ok { @@ -327,7 +327,7 @@ func checkDuplicates(fset *FileSet) bool { default: continue } - if name == blankNameIdentifer { + if name == blankIdentifier { continue } if _, ok := defined[name]; ok { diff --git a/gnovm/pkg/gnolang/nodes.go b/gnovm/pkg/gnolang/nodes.go index 0c182749f92..6a2eebb9897 100644 --- a/gnovm/pkg/gnolang/nodes.go +++ b/gnovm/pkg/gnolang/nodes.go @@ -1014,7 +1014,7 @@ type ValueDecl struct { func (x *ValueDecl) GetDeclNames() []Name { ns := make([]Name, 0, len(x.NameExprs)) for _, nx := range x.NameExprs { - if nx.Name == blankNameIdentifer { + if nx.Name == blankIdentifier { // ignore } else { ns = append(ns, nx.Name) @@ -1031,7 +1031,7 @@ type TypeDecl struct { } func (x *TypeDecl) GetDeclNames() []Name { - if x.NameExpr.Name == blankNameIdentifer { + if x.NameExpr.Name == blankIdentifier { return nil // ignore } else { return []Name{x.NameExpr.Name} @@ -1588,8 +1588,8 @@ func (sb *StaticBlock) GetParentNode(store Store) BlockNode { // Implements BlockNode. // As a side effect, notes externally defined names. func (sb *StaticBlock) GetPathForName(store Store, n Name) ValuePath { - if n == blankNameIdentifer { - return NewValuePathBlock(0, 0, blankNameIdentifer) + if n == blankIdentifier { + return NewValuePathBlock(0, 0, blankIdentifier) } // Check local. gen := 1 @@ -1780,7 +1780,7 @@ func (sb *StaticBlock) Define2(isConst bool, n Name, st Type, tv TypedValue) { if tv.T == nil && tv.V != nil { panic("StaticBlock.Define2() requires .T if .V is set") } - if n == blankNameIdentifer { + if n == blankIdentifier { return // ignore } idx, exists := sb.GetLocalIndex(n) diff --git a/gnovm/pkg/gnolang/preprocess.go b/gnovm/pkg/gnolang/preprocess.go index 02c38025962..4f5826e10e2 100644 --- a/gnovm/pkg/gnolang/preprocess.go +++ b/gnovm/pkg/gnolang/preprocess.go @@ -10,8 +10,7 @@ import ( ) const ( - blankIdentifier string = "_" - blankNameIdentifer Name = "_" + blankIdentifier = "_" ) // In the case of a *FileSet, some declaration steps have to happen @@ -194,7 +193,7 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { var defined bool for _, lx := range n.Lhs { ln := lx.(*NameExpr).Name - if ln == blankNameIdentifer { + if ln == blankIdentifier { // ignore. } else { _, ok := last.GetLocalIndex(ln) @@ -238,7 +237,7 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { case *FuncTypeExpr: for i := range n.Params { p := &n.Params[i] - if p.Name == "" || p.Name == blankNameIdentifer { + if p.Name == "" || p.Name == blankIdentifier { // create a hidden var with leading dot. // NOTE: document somewhere. pn := fmt.Sprintf(".arg_%d", i) @@ -247,7 +246,7 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { } for i := range n.Results { r := &n.Results[i] - if r.Name == blankNameIdentifer { + if r.Name == blankIdentifier { // create a hidden var with leading dot. // NOTE: document somewhere. rn := fmt.Sprintf(".res_%d", i) @@ -686,8 +685,8 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { } // specific and general cases switch n.Name { - case blankNameIdentifer: - n.Path = NewValuePathBlock(0, 0, blankNameIdentifer) + case blankIdentifier: + n.Path = NewValuePathBlock(0, 0, blankIdentifier) return n, TRANS_CONTINUE case "iota": pd := lastDecl(ns) @@ -1927,8 +1926,8 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { pn := fn.GetParentNode(nil).(*PackageNode) for i := 0; i < numNames; i++ { nx := &n.NameExprs[i] - if nx.Name == blankNameIdentifer { - nx.Path = NewValuePathBlock(0, 0, blankNameIdentifer) + if nx.Name == blankIdentifier { + nx.Path = NewValuePathBlock(0, 0, blankIdentifier) } else { pn.Define2(n.Const, nx.Name, sts[i], tvs[i]) nx.Path = last.GetPathForName(nil, nx.Name) @@ -1937,8 +1936,8 @@ func Preprocess(store Store, ctx BlockNode, n Node) Node { } else { for i := 0; i < numNames; i++ { nx := &n.NameExprs[i] - if nx.Name == blankNameIdentifer { - nx.Path = NewValuePathBlock(0, 0, blankNameIdentifer) + if nx.Name == blankIdentifier { + nx.Path = NewValuePathBlock(0, 0, blankIdentifier) } else { last.Define2(n.Const, nx.Name, sts[i], tvs[i]) nx.Path = last.GetPathForName(nil, nx.Name) @@ -3052,7 +3051,7 @@ func predefineNow2(store Store, last BlockNode, d Decl, m map[Name]struct{}) (De // NOTE: unlike the *ValueDecl case, this case doesn't // preprocess d itself (only d.Type). if cd.IsMethod { - if cd.Recv.Name == "" || cd.Recv.Name == blankNameIdentifer { + if cd.Recv.Name == "" || cd.Recv.Name == blankIdentifier { // create a hidden var with leading dot. // NOTE: document somewhere. cd.Recv.Name = ".recv" @@ -3171,7 +3170,7 @@ func tryPredefine(store Store, last BlockNode, d Decl) (un Name) { } if d.Name == "" { // use default d.Name = pv.PkgName - } else if d.Name == blankNameIdentifer { // no definition + } else if d.Name == blankIdentifier { // no definition return } else if d.Name == "." { // dot import panic("dot imports not allowed in Gno") @@ -3198,8 +3197,8 @@ func tryPredefine(store Store, last BlockNode, d Decl) (un Name) { } for i := 0; i < len(d.NameExprs); i++ { nx := &d.NameExprs[i] - if nx.Name == blankNameIdentifer { - nx.Path.Name = blankNameIdentifer + if nx.Name == blankIdentifier { + nx.Path.Name = blankIdentifier } else { last2 := skipFile(last) last2.Predefine(d.Const, nx.Name) @@ -3379,7 +3378,7 @@ func constUntypedBigint(source Expr, i64 int64) *ConstExpr { } func fillNameExprPath(last BlockNode, nx *NameExpr, isDefineLHS bool) { - if nx.Name == blankNameIdentifer { + if nx.Name == blankIdentifier { // Blank name has no path; caller error. panic("should not happen") } diff --git a/gnovm/pkg/gnolang/values.go b/gnovm/pkg/gnolang/values.go index fff2396de47..f1422f283c3 100644 --- a/gnovm/pkg/gnolang/values.go +++ b/gnovm/pkg/gnolang/values.go @@ -2343,7 +2343,7 @@ func (b *Block) GetPointerToInt(store Store, index int) PointerValue { func (b *Block) GetPointerTo(store Store, path ValuePath) PointerValue { if path.IsBlockBlankPath() { if debug { - if path.Name != blankNameIdentifer { + if path.Name != blankIdentifier { panic(fmt.Sprintf( "zero value path is reserved for \"_\", but got %s", path.Name)) From 62b56db735cb926618142d53dbefc54fcefa016a Mon Sep 17 00:00:00 2001 From: deelawn Date: Tue, 28 May 2024 08:09:31 -0700 Subject: [PATCH 3/3] fixed replaced comments --- gnovm/pkg/gnolang/nodes.go | 2 +- gnovm/pkg/gnolang/values.go | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/gnovm/pkg/gnolang/nodes.go b/gnovm/pkg/gnolang/nodes.go index 6a2eebb9897..9c1e9883f50 100644 --- a/gnovm/pkg/gnolang/nodes.go +++ b/gnovm/pkg/gnolang/nodes.go @@ -1976,7 +1976,7 @@ func (vp ValuePath) Validate() { panic("uverse value path must have depth 0") } case VPBlock: - // 0 ok (blankIdentifier blank) + // 0 ok ("_" blank) case VPField: if vp.Depth > 1 { panic("field value path must have depth 0 or 1") diff --git a/gnovm/pkg/gnolang/values.go b/gnovm/pkg/gnolang/values.go index f1422f283c3..f36d8bae19f 100644 --- a/gnovm/pkg/gnolang/values.go +++ b/gnovm/pkg/gnolang/values.go @@ -163,7 +163,7 @@ func (dbv DataByteValue) SetByte(b byte) { // value after a block var escapes "to the heap". // *(PointerValue.TypedValue) must have already become // initialized, namely T set if a typed-nil. -// Index is -1 for the shared blankIdentifier block var, +// Index is -1 for the shared "_" block var, // and -2 for (gno and native) map items. // // Allocation for PointerValue is not immediate, @@ -179,7 +179,7 @@ type PointerValue struct { } const ( - PointerIndexBlockBlank = -1 // for the blankIdentifier identifier in blocks + PointerIndexBlockBlank = -1 // for the "_" identifier in blocks PointerIndexMap = -2 // Base is Map, use Key. PointerIndexNative = -3 // Base is *NativeValue. ) @@ -2257,7 +2257,7 @@ type Block struct { Source BlockNode Values []TypedValue Parent Value - Blank TypedValue // captures blankIdentifier // XXX remove and replace with global instance. + Blank TypedValue // captures "_" // XXX remove and replace with global instance. bodyStmt bodyStmt // XXX expose for persistence, not needed for MVP. } @@ -2365,7 +2365,7 @@ func (b *Block) GetPointerTo(store Store, path ValuePath) PointerValue { return b.GetPointerToInt(store, int(path.Index)) } -// Result is used has lhs for any assignments to blankIdentifier. +// Result is used has lhs for any assignments to "_". func (b *Block) GetBlankRef() *TypedValue { return &b.Blank }