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.
144 lines
3.8 KiB
Go
144 lines
3.8 KiB
Go
package main
|
|
|
|
import (
|
|
"go/ast"
|
|
"go/token"
|
|
|
|
ah "mvdan.cc/garble/internal/asthelper"
|
|
)
|
|
|
|
// addRuntimeAPI exposes additional functions in the runtime
|
|
// package that may be helpful when hiding information
|
|
// during execution is required.
|
|
func addRuntimeAPI(filename string, file *ast.File) {
|
|
switchPanicPrints := func(node ast.Node) bool {
|
|
call, ok := node.(*ast.CallExpr)
|
|
if !ok {
|
|
return true
|
|
}
|
|
id, ok := call.Fun.(*ast.Ident)
|
|
if !ok {
|
|
return true
|
|
}
|
|
|
|
switch id.Name {
|
|
case "print":
|
|
id.Name = "panicprint"
|
|
return false
|
|
case "println":
|
|
id.Name = "panicprint"
|
|
call.Args = append(call.Args, &ast.BasicLit{Kind: token.STRING, Value: `"\n"`})
|
|
return false
|
|
default:
|
|
return true
|
|
}
|
|
}
|
|
|
|
switch filename {
|
|
case "debug.go":
|
|
// Add hideFatalErrors function and internal fatalErrorsHidden variable
|
|
file.Decls = append(file.Decls, hideFatalErrorsDecls...)
|
|
case "error.go":
|
|
// Add a function panicprint, that does nothing if panics are
|
|
// hidden, otherwise forwards arguments to printany to print them
|
|
// as normal. Although we add an if statement to printany to do
|
|
// nothing if panics are hidden, printany only takes one argument,
|
|
// and both print and printany are used to print panic messages.
|
|
// panicprint's entire purpose is to act as a replacement to print
|
|
// that respects hideFatalErrors, and print is variadic, so print
|
|
// must be replaced by a variadic function, hence panicprint.
|
|
//
|
|
// We will also add two statements to printany:
|
|
// 1. An if statement that returns early if panics are hidden
|
|
// 2. An additional case statement that handles printing runtime.hex
|
|
// values. Without this case statement, the default case will print
|
|
// the runtime.hex values in a way not consistent with normal panic
|
|
// outputs
|
|
for _, decl := range file.Decls {
|
|
decl, ok := decl.(*ast.FuncDecl)
|
|
if !ok || decl.Name.Name != "printany" {
|
|
continue
|
|
}
|
|
for _, stmt := range decl.Body.List {
|
|
if stmt, ok := stmt.(*ast.TypeSwitchStmt); ok {
|
|
stmt.Body.List = append(stmt.Body.List, printanyHexCase)
|
|
break
|
|
}
|
|
}
|
|
decl.Body.List = append([]ast.Stmt{fatalErrorsHiddenCheckStmt}, decl.Body.List...)
|
|
break
|
|
}
|
|
|
|
file.Decls = append(file.Decls, panicprintDecl)
|
|
default:
|
|
// Change all calls to print, which we don't control, to
|
|
// panicprint, which we do control and does the same thing.
|
|
ast.Inspect(file, switchPanicPrints)
|
|
}
|
|
}
|
|
|
|
var fatalErrorsHiddenCheckStmt = &ast.IfStmt{
|
|
Cond: ah.Ident("fatalErrorsHidden"),
|
|
Body: ah.BlockStmt(ah.ReturnStmt()),
|
|
}
|
|
|
|
var hideFatalErrorsDecls = []ast.Decl{
|
|
&ast.GenDecl{
|
|
Tok: token.VAR,
|
|
Specs: []ast.Spec{&ast.ValueSpec{
|
|
Names: []*ast.Ident{ah.Ident("fatalErrorsHidden")},
|
|
Type: ah.Ident("bool"),
|
|
}},
|
|
},
|
|
&ast.FuncDecl{
|
|
Name: ah.Ident("hideFatalErrors"),
|
|
Type: &ast.FuncType{Params: &ast.FieldList{
|
|
List: []*ast.Field{{
|
|
Names: []*ast.Ident{ah.Ident("hide")},
|
|
Type: ah.Ident("bool"),
|
|
}},
|
|
}},
|
|
Body: ah.BlockStmt(
|
|
&ast.AssignStmt{
|
|
Lhs: []ast.Expr{ah.Ident("fatalErrorsHidden")},
|
|
Tok: token.ASSIGN,
|
|
Rhs: []ast.Expr{ah.Ident("hide")},
|
|
},
|
|
),
|
|
},
|
|
}
|
|
|
|
var printanyHexCase = &ast.CaseClause{
|
|
List: []ast.Expr{ah.Ident("hex")},
|
|
Body: []ast.Stmt{
|
|
ah.ExprStmt(ah.CallExpr(ah.Ident("print"), ah.Ident("v"))),
|
|
},
|
|
}
|
|
|
|
var panicprintDecl = &ast.FuncDecl{
|
|
Name: ah.Ident("panicprint"),
|
|
Type: &ast.FuncType{Params: &ast.FieldList{
|
|
List: []*ast.Field{{
|
|
Names: []*ast.Ident{{Name: "args"}},
|
|
Type: &ast.Ellipsis{Elt: &ast.InterfaceType{
|
|
Methods: &ast.FieldList{},
|
|
}},
|
|
}},
|
|
}},
|
|
Body: ah.BlockStmt(
|
|
&ast.IfStmt{
|
|
Cond: ah.Ident("fatalErrorsHidden"),
|
|
Body: ah.BlockStmt(ah.ReturnStmt()),
|
|
},
|
|
&ast.RangeStmt{
|
|
Key: ah.Ident("_"),
|
|
Value: ah.Ident("arg"),
|
|
Tok: token.DEFINE,
|
|
X: ah.Ident("args"),
|
|
Body: ah.BlockStmt(
|
|
ah.ExprStmt(ah.CallExpr(ah.Ident("printany"), ah.Ident("arg"))),
|
|
),
|
|
},
|
|
),
|
|
}
|