You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
175 lines
3.9 KiB
Go
175 lines
3.9 KiB
Go
// Copyright (c) 2020, The Garble Authors.
|
|
// See LICENSE for licensing information.
|
|
|
|
package asthelper
|
|
|
|
import (
|
|
"fmt"
|
|
"go/ast"
|
|
"go/constant"
|
|
"go/token"
|
|
"strconv"
|
|
)
|
|
|
|
// StringLit returns an ast.BasicLit of kind STRING
|
|
func StringLit(value string) *ast.BasicLit {
|
|
return &ast.BasicLit{
|
|
Kind: token.STRING,
|
|
Value: fmt.Sprintf("%q", value),
|
|
}
|
|
}
|
|
|
|
// IntLit returns an ast.BasicLit of kind INT
|
|
func IntLit(value int) *ast.BasicLit {
|
|
return &ast.BasicLit{
|
|
Kind: token.INT,
|
|
Value: strconv.Itoa(value),
|
|
}
|
|
}
|
|
|
|
// IndexExpr "name[index]"
|
|
func IndexExpr(name string, index ast.Expr) *ast.IndexExpr {
|
|
return &ast.IndexExpr{
|
|
X: ast.NewIdent(name),
|
|
Index: index,
|
|
}
|
|
}
|
|
|
|
// CallExpr "fun(arg)"
|
|
func CallExpr(fun ast.Expr, args ...ast.Expr) *ast.CallExpr {
|
|
return &ast.CallExpr{
|
|
Fun: fun,
|
|
Args: args,
|
|
}
|
|
}
|
|
|
|
// LambdaCall "func() resultType {block}()"
|
|
func LambdaCall(resultType ast.Expr, block *ast.BlockStmt) *ast.CallExpr {
|
|
funcLit := &ast.FuncLit{
|
|
Type: &ast.FuncType{
|
|
Params: &ast.FieldList{},
|
|
Results: &ast.FieldList{
|
|
List: []*ast.Field{
|
|
{Type: resultType},
|
|
},
|
|
},
|
|
},
|
|
Body: block,
|
|
}
|
|
return CallExpr(funcLit)
|
|
}
|
|
|
|
// LambdaCallParams "func(params) resultType {block}(args)"
|
|
func LambdaCallParams(params *ast.FieldList, resultType ast.Expr, block *ast.BlockStmt, args []ast.Expr) *ast.CallExpr {
|
|
funcLit := &ast.FuncLit{
|
|
Type: &ast.FuncType{
|
|
Params: params,
|
|
Results: &ast.FieldList{
|
|
List: []*ast.Field{
|
|
{Type: resultType},
|
|
},
|
|
},
|
|
},
|
|
Body: block,
|
|
}
|
|
return CallExpr(funcLit, args...)
|
|
}
|
|
|
|
// ReturnStmt "return result"
|
|
func ReturnStmt(results ...ast.Expr) *ast.ReturnStmt {
|
|
return &ast.ReturnStmt{
|
|
Results: results,
|
|
}
|
|
}
|
|
|
|
// BlockStmt a block of multiple statements e.g. a function body
|
|
func BlockStmt(stmts ...ast.Stmt) *ast.BlockStmt {
|
|
return &ast.BlockStmt{List: stmts}
|
|
}
|
|
|
|
// ExprStmt convert an ast.Expr to an ast.Stmt
|
|
func ExprStmt(expr ast.Expr) *ast.ExprStmt {
|
|
return &ast.ExprStmt{X: expr}
|
|
}
|
|
|
|
// DataToByteSlice turns a byte slice like []byte{1, 2, 3} into an AST
|
|
// expression
|
|
func DataToByteSlice(data []byte) *ast.CallExpr {
|
|
return &ast.CallExpr{
|
|
Fun: &ast.ArrayType{
|
|
Elt: &ast.Ident{Name: "byte"},
|
|
},
|
|
Args: []ast.Expr{StringLit(string(data))},
|
|
}
|
|
}
|
|
|
|
// DataToArray turns a byte slice like []byte{1, 2, 3} into an AST
|
|
// expression
|
|
func DataToArray(data []byte) *ast.CompositeLit {
|
|
elts := make([]ast.Expr, len(data))
|
|
for i, b := range data {
|
|
elts[i] = IntLit(int(b))
|
|
}
|
|
|
|
return &ast.CompositeLit{
|
|
Type: &ast.ArrayType{
|
|
Len: IntLit(len(data)),
|
|
Elt: ast.NewIdent("byte"),
|
|
},
|
|
Elts: elts,
|
|
}
|
|
}
|
|
|
|
// SelectExpr "x.sel"
|
|
func SelectExpr(x ast.Expr, sel *ast.Ident) *ast.SelectorExpr {
|
|
return &ast.SelectorExpr{
|
|
X: x,
|
|
Sel: sel,
|
|
}
|
|
}
|
|
|
|
// AssignDefineStmt "Lhs := Rhs"
|
|
func AssignDefineStmt(Lhs ast.Expr, Rhs ast.Expr) *ast.AssignStmt {
|
|
return &ast.AssignStmt{
|
|
Lhs: []ast.Expr{Lhs},
|
|
Tok: token.DEFINE,
|
|
Rhs: []ast.Expr{Rhs},
|
|
}
|
|
}
|
|
|
|
// CallExprByName "fun(args...)"
|
|
func CallExprByName(fun string, args ...ast.Expr) *ast.CallExpr {
|
|
return CallExpr(ast.NewIdent(fun), args...)
|
|
}
|
|
|
|
// AssignStmt "Lhs = Rhs"
|
|
func AssignStmt(Lhs ast.Expr, Rhs ast.Expr) *ast.AssignStmt {
|
|
return &ast.AssignStmt{
|
|
Lhs: []ast.Expr{Lhs},
|
|
Tok: token.ASSIGN,
|
|
Rhs: []ast.Expr{Rhs},
|
|
}
|
|
}
|
|
|
|
// IndexExprByExpr "xExpr[indexExpr]"
|
|
func IndexExprByExpr(xExpr, indexExpr ast.Expr) *ast.IndexExpr {
|
|
return &ast.IndexExpr{X: xExpr, Index: indexExpr}
|
|
}
|
|
|
|
func ConstToAst(val constant.Value) ast.Expr {
|
|
switch val.Kind() {
|
|
case constant.Bool:
|
|
return ast.NewIdent(val.ExactString())
|
|
case constant.String:
|
|
return &ast.BasicLit{Kind: token.STRING, Value: val.ExactString()}
|
|
case constant.Int:
|
|
return &ast.BasicLit{Kind: token.INT, Value: val.ExactString()}
|
|
case constant.Float:
|
|
return &ast.BasicLit{Kind: token.FLOAT, Value: val.String()}
|
|
case constant.Complex:
|
|
return CallExprByName("complex", ConstToAst(constant.Real(val)), ConstToAst(constant.Imag(val)))
|
|
default:
|
|
panic("unreachable")
|
|
}
|
|
}
|