Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(gnovm): align Gno constant handling with Go specifications #2828

Open
wants to merge 32 commits into
base: master
Choose a base branch
from

Conversation

omarsy
Copy link
Member

@omarsy omarsy commented Sep 21, 2024

Related Issues:
Closes #2628

This PR aims to align Gno's constant handling with the Go specification regarding constant expressions (see Go Specification on Constants).

  1. Primitive Type Requirement

    • Should Work:
      const t = 1
    • Should Return an Error:
      const t = []string{"1"}
      Error:
      (const (slice[("1" string)] []string)) (value of type []string) is not constant
      
  2. Function Calls Disallowed
    Only built-in functions should be allowed.

    • Should Work:
      const t = len("s")
    • Should Return an Error:
      func v() string {
          return ""
      }
      const t = v()
      Error:
      v<VPBlock(3,0)>() (value of type string) is not constant
      
  3. Constant Operands Requirement
    Constant expressions may contain only constant operands and are evaluated at compile time.

    • Should Work:
      const t = 1
      const v = t
    • Should Raise an Error:
      t := 1
      const v = t
      Error:
      t (variable of type int) is not constant
      
  4. Type Assertion Forbidden

    • This code:
      var i interface{} = 1
      const t, ok = i.(int)
    • Should Raise This Error:
      i.(int) (comma, ok expression of type int) is not constant
      
  5. Index Expression Forbidden

    • This code:
      var i = []string{}
      const t, ok = i[0]
    • Should Return This Error:
      i[0] (variable of type string) is not constant
      
Contributors' checklist...
  • Added new tests, or not needed, or not feasible
  • Provided an example (e.g. screenshot) to aid review or the PR is self-explanatory
  • Updated the official documentation or not needed
  • No breaking changes were made, or a BREAKING CHANGE: xxx message was included in the description
  • Added references to related issues and PRs
  • Provided any useful hints for running manual tests
  • Added new benchmarks to generated graphs, if any. More info here.

@github-actions github-actions bot added the 📦 🤖 gnovm Issues or PRs gnovm related label Sep 21, 2024
Copy link

codecov bot commented Sep 21, 2024

Codecov Report

Attention: Patch coverage is 65.18519% with 47 lines in your changes missing coverage. Please review.

Files with missing lines Patch % Lines
gnovm/pkg/gnolang/type_check.go 64.92% 37 Missing and 10 partials ⚠️

📢 Thoughts on this report? Let us know!

@omarsy omarsy force-pushed the fix/2628 branch 4 times, most recently from c9b02d2 to f49c088 Compare September 25, 2024 11:54
@github-actions github-actions bot added the 🧾 package/realm Tag used for new Realms or Packages. label Sep 25, 2024
@omarsy omarsy force-pushed the fix/2628 branch 2 times, most recently from 355d799 to e9a9be7 Compare September 26, 2024 10:31
@omarsy omarsy marked this pull request as ready for review October 18, 2024 20:52
@omarsy omarsy requested review from jaekwon, moul, piux2, thehowl, mvertes and a team as code owners October 18, 2024 20:52
@omarsy omarsy changed the title feat(gnovm): align Gno constant handling with Go specifications WIP feat(gnovm): align Gno constant handling with Go specifications Oct 18, 2024
@jefft0
Copy link
Contributor

jefft0 commented Oct 19, 2024

@omarsy , can you fix the lint-pr-title CI check?

@jefft0 jefft0 added the review/triage-pending PRs opened by external contributors that are waiting for the 1st review label Oct 19, 2024
@omarsy omarsy changed the title WIP feat(gnovm): align Gno constant handling with Go specifications feat(gnovm): align Gno constant handling with Go specifications Oct 19, 2024
@Kouteki Kouteki added this to the 🚀 Mainnet launch milestone Oct 22, 2024
@omarsy omarsy requested a review from ltzmaxwell November 28, 2024 20:55
@petar-dambovaliev
Copy link
Contributor

Part of your PR related to this but from what i see doesn't address the whole problem. I've fixed it here.

@Gno2D2
Copy link
Collaborator

Gno2D2 commented Nov 30, 2024

🛠 PR Checks Summary

🔴 Maintainers must be able to edit this pull request (more info)

Manual Checks (for Reviewers):
  • IGNORE the bot requirements for this PR (force green CI check)
Read More

🤖 This bot helps streamline PR reviews by verifying automated checks and providing guidance for contributors and reviewers.

✅ Automated Checks (for Contributors):

🔴 Maintainers must be able to edit this pull request (more info)

☑️ Contributor Actions:
  1. Fix any issues flagged by automated checks.
  2. Follow the Contributor Checklist to ensure your PR is ready for review.
    • Add new tests, or document why they are unnecessary.
    • Provide clear examples/screenshots, if necessary.
    • Update documentation, if required.
    • Ensure no breaking changes, or include BREAKING CHANGE notes.
    • Link related issues/PRs, where applicable.
☑️ Reviewer Actions:
  1. Complete manual checks for the PR, including the guidelines and additional checks if applicable.
📚 Resources:
Debug
Automated Checks
Maintainers must be able to edit this pull request (more info)

If

🟢 Condition met
└── 🟢 The pull request was created from a fork (head branch repo: TERITORI/gno)

Then

🔴 Requirement not satisfied
└── 🔴 Maintainer can modify this pull request

Manual Checks
**IGNORE** the bot requirements for this PR (force green CI check)

If

🟢 Condition met
└── 🟢 On every pull request

Can be checked by

  • Any user with comment edit permission

@petar-dambovaliev
Copy link
Contributor

Part of your PR related to this but from what i see doesn't address the whole problem. I've fixed it here.

@omarsy i am going to close my PR. You can integrate that weird Go feature for arrays in your PR. Better in the preprocessor so no functions are called during runtime.

@omarsy
Copy link
Member Author

omarsy commented Dec 3, 2024

Part of your PR related to this but from what i see doesn't address the whole problem. I've fixed it here.

@omarsy i am going to close my PR. You can integrate that weird Go feature for arrays in your PR. Better in the preprocessor so no functions are called during runtime.

I think it is better to handle it in another PR. WDYT ?

switch vx := vx.(type) {
case *NameExpr:
t := evalStaticTypeOf(store, last, vx)
if _, ok := t.(*ArrayType); ok {
Copy link
Contributor

@ltzmaxwell ltzmaxwell Dec 9, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

hmmm, this logic seems incorrect:

if _, ok := t.(*ArrayType); ok {
	break Main
}

Due to this logic, the error in const43.gno was not caught.

I guess your intention is to check len(arr), right? I think that needs additional context to know if the outer expr is a built func call: len or max?

Not so sure, but we need to focus on ensuring that the logic of this function is correct(not limited to the above-mentioned).

assertValidConstantExprRecursively(store, last, expr, nil)
}

func assertValidConstantExprRecursively(store Store, last BlockNode, currExpr, parentExpr Expr) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
func assertValidConstantExprRecursively(store Store, last BlockNode, currExpr, parentExpr Expr) {
func assertValidConstValue(store Store, last BlockNode, currExpr, parentExpr Expr) {

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

should be done here 54878f0

@omarsy omarsy requested a review from ltzmaxwell December 11, 2024 20:36
@thehowl
Copy link
Member

thehowl commented Dec 21, 2024

@ltzmaxwell ping me if you approve and would like me to take a look as well, otherwise go ahead and merge after you've approved

Copy link
Contributor

@ltzmaxwell ltzmaxwell left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

trying to find more counter cases, please check.

case *TypeType:
for _, arg := range currExpr.Args {
assertValidConstValue(store, last, arg, currExpr)
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

please check this:

package main

const a = interface{}(nil)

func main() {
	println("ok")
}

seems also need to check the type, right?

Copy link
Member Author

@omarsy omarsy Dec 26, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

My bad break on my last refactoring when I move the check type

Nice catch ^^

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

so the above clause: case *TypeType is not needed? didn't see any test case targeting this.


tt := pv.GetBlock(store).Source.GetStaticTypeOf(store, currExpr.Sel)
panic(fmt.Sprintf("%s (variable of type %s) is not constant", currExpr.String(), tt))
case *PointerType, *DeclaredType, *StructType, *InterfaceType:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this is a counter case:

package main

type MyStruct struct {
	arr [2]int
}

const a = len(MyStruct{arr: [2]int{1, 2}}.arr)

func main() {
	println("ok")
}

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

:o I miss this case. Thank you ^^
Done here c45f63e

}
case *ConstExpr:
case *BasicLitExpr:
case *CompositeLitExpr:
Copy link
Contributor

@ltzmaxwell ltzmaxwell Dec 26, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can we just check parent and panic here?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since we verify the expression type before calling this function, this case should not occur. Removed for array: 4da7934

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

will it be more straightforward that just check if parentExpr exist, and the type of compositeLitExpr is array type, otherwise panic?


tt := pv.GetBlock(store).Source.GetStaticTypeOf(store, currExpr.Sel)
panic(fmt.Sprintf("%s (variable of type %s) is not constant", currExpr.String(), tt))
case *PointerType, *DeclaredType, *StructType, *InterfaceType, *TypeType, *NativeType:
Copy link
Contributor

@ltzmaxwell ltzmaxwell Dec 31, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

seems *TypeType does not apply. please double check.


func isParentCallExprWithArrayArg(currType Type, parentExpr Expr) bool {
_, okArray := baseOf(currType).(*ArrayType)
_, okCallExpr := parentExpr.(*CallExpr)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

please add a comment.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done here 29b6ab0

assertValidConstValue(store, last, expr, nil)
}

func assertValidConstValue(store Store, last BlockNode, currExpr, parentExpr Expr) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

At the same time, I’m wondering—many of the checks here are for functions like len and cap. So, is this recursive call really necessary? Could most of the cases be handled under callExpr instead? Please consider this.

Apologies for the continuous iterations in reviewing this PR. Thank you for understanding.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you for pointing this out! To clarify, we don't check only if the call expression has len or cap. Our implementation ensures that all expressions are validated for their correctness as constant expressions. For instance, a binary expression with a valid type we will check if it is a valid constant expression.

Here's an example to illustrate this:

package main

func main() {
    v := 5
    const a = 1 + 2 + len([2]int{1, 2}) + v
    println("ok")
}

In this example, the constant expression 1 + 2 + len([2]int{1, 2}) is valid because all sub-expressions are constant and have valid types. However, the addition of v, which is a variable, makes the entire expression invalid as a constant expression, and this is correctly flagged by our checks.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
in focus Core team is prioritizing this work 📦 🌐 tendermint v2 Issues or PRs tm2 related 📦 ⛰️ gno.land Issues or PRs gno.land package related 📦 🤖 gnovm Issues or PRs gnovm related 🧾 package/realm Tag used for new Realms or Packages.
Projects
Status: In Progress
Status: In Review
Development

Successfully merging this pull request may close these issues.

Constant Declaration with Slice Type Compiles Without Error
7 participants