don't remove "//go:" compile directives

For example, this broke cgo, since it uses go:linkname.

Updates #12.
pull/22/head
Daniel Martí 5 years ago
parent cf3f54aa88
commit 5aaa086e5d

@ -48,9 +48,8 @@ which is equivalent to the longer:
func main() { os.Exit(main1()) } func main() { os.Exit(main1()) }
var ( var (
deferred []func() error deferred []func() error
fset = token.NewFileSet() fset = token.NewFileSet()
emptyFset = token.NewFileSet()
b64 = base64.NewEncoding("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_z") b64 = base64.NewEncoding("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_z")
printerConfig = printer.Config{Mode: printer.RawFormat} printerConfig = printer.Config{Mode: printer.RawFormat}
@ -257,7 +256,7 @@ func transformCompile(args []string) ([]string, error) {
// log.Printf("%#v", ids) // log.Printf("%#v", ids)
var files []*ast.File var files []*ast.File
for _, path := range paths { for _, path := range paths {
file, err := parser.ParseFile(fset, path, nil, 0) file, err := parser.ParseFile(fset, path, nil, parser.ParseComments)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -295,8 +294,8 @@ func transformCompile(args []string) ([]string, error) {
return nil, err return nil, err
} }
defer f.Close() defer f.Close()
// printerConfig.Fprint(os.Stderr, emptyFset, file) // printerConfig.Fprint(os.Stderr, fset, file)
if err := printerConfig.Fprint(f, emptyFset, file); err != nil { if err := printerConfig.Fprint(f, fset, file); err != nil {
return nil, err return nil, err
} }
if err := f.Close(); err != nil { if err := f.Close(); err != nil {
@ -385,7 +384,22 @@ func hashWith(salt, value string) string {
} }
// transformGo garbles the provided Go syntax node. // transformGo garbles the provided Go syntax node.
func transformGo(node ast.Node, info *types.Info) ast.Node { func transformGo(file *ast.File, info *types.Info) ast.Node {
// Remove all comments, minus the "//go:" compiler directives.
// The final binary should still not contain comment text, but removing
// it helps ensure that (and makes position info less predictable).
origComments := file.Comments
file.Comments = nil
for _, commentGroup := range origComments {
for _, comment := range commentGroup.List {
if strings.HasPrefix(comment.Text, "//go:") {
file.Comments = append(file.Comments, &ast.CommentGroup{
List: []*ast.Comment{comment},
})
}
}
}
pre := func(cursor *astutil.Cursor) bool { pre := func(cursor *astutil.Cursor) bool {
switch node := cursor.Node().(type) { switch node := cursor.Node().(type) {
case *ast.Ident: case *ast.Ident:
@ -411,7 +425,7 @@ func transformGo(node ast.Node, info *types.Info) ast.Node {
return true // might implement an interface return true // might implement an interface
} }
if implementedOutsideGo(x) { if implementedOutsideGo(x) {
return true // implemented elsewhere, like assembly return true // give up in this case
} }
switch node.Name { switch node.Name {
case "main", "init", "TestMain": case "main", "init", "TestMain":
@ -459,7 +473,7 @@ func transformGo(node ast.Node, info *types.Info) ast.Node {
} }
return true return true
} }
return astutil.Apply(node, pre, nil) return astutil.Apply(file, pre, nil)
} }
func isStandardLibrary(path string) bool { func isStandardLibrary(path string) bool {
@ -473,7 +487,7 @@ func isStandardLibrary(path string) bool {
} }
// implementedOutsideGo returns whether a *types.Func does not have a body, for // implementedOutsideGo returns whether a *types.Func does not have a body, for
// example when it's implemented in assembly. // example when it's implemented in assembly, or when one uses go:linkname.
// //
// Note that this function can only return true if the obj parameter was // Note that this function can only return true if the obj parameter was
// type-checked from source - that is, if it's the top-level package we're // type-checked from source - that is, if it's the top-level package we're

@ -12,6 +12,10 @@ rm main$exe
garble build garble build
bincmp main$exe main_old$exe bincmp main$exe main_old$exe
go build
exec ./main
cmp stdout main.stdout
-- go.mod -- -- go.mod --
module foo.com/main module foo.com/main
-- main.go -- -- main.go --
@ -19,18 +23,22 @@ package main
import ( import (
"fmt" "fmt"
_ "unsafe"
"foo.com/main/imported" "foo.com/main/imported"
"rsc.io/quote" "rsc.io/quote"
) )
//go:linkname linkedPrintln fmt.Println
func linkedPrintln(a ...interface{}) (n int, err error)
func main() { func main() {
fmt.Println(imported.ImportedVar) fmt.Println(imported.ImportedVar)
fmt.Println(imported.ImportedConst) fmt.Println(imported.ImportedConst)
imported.ImportedFunc('x') imported.ImportedFunc('x')
fmt.Println(imported.ImportedType(3)) fmt.Println(imported.ImportedType(3))
fmt.Println(nil) linkedPrintln(nil)
fmt.Println(quote.Go()) fmt.Println(quote.Go())
} }
-- main.stdout -- -- main.stdout --

@ -2,6 +2,8 @@ garble build main.go
exec ./main exec ./main
cmp stderr main.stderr cmp stderr main.stderr
! binsubstr main$exe 'valuable information'
-- main.go -- -- main.go --
package main package main
@ -10,6 +12,7 @@ import (
"go/ast" "go/ast"
) )
// This comment contains valuable information. Ensure it's not in the final binary.
var V interface{} var V interface{}
type T struct { type T struct {

Loading…
Cancel
Save