|
|
|
@ -110,6 +110,8 @@ var (
|
|
|
|
|
envGarbleListPkgs = os.Getenv("GARBLE_LISTPKGS")
|
|
|
|
|
|
|
|
|
|
seed []byte
|
|
|
|
|
|
|
|
|
|
garbleMapFile = "garble.map"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func saveListedPackages(w io.Writer, flags, patterns []string) error {
|
|
|
|
@ -517,6 +519,8 @@ func transformCompile(args []string) ([]string, error) {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
privateNameMap := make(map[string]string)
|
|
|
|
|
|
|
|
|
|
// TODO: randomize the order and names of the files
|
|
|
|
|
newPaths := make([]string, 0, len(files))
|
|
|
|
|
for i, file := range files {
|
|
|
|
@ -550,7 +554,7 @@ func transformCompile(args []string) ([]string, error) {
|
|
|
|
|
if !envGarbleTiny {
|
|
|
|
|
extraComments, file = transformLineInfo(file)
|
|
|
|
|
}
|
|
|
|
|
file = transformGo(file, info, blacklist)
|
|
|
|
|
file = transformGo(file, info, blacklist, privateNameMap)
|
|
|
|
|
|
|
|
|
|
// Uncomment for some quick debugging. Do not delete.
|
|
|
|
|
// fmt.Fprintf(os.Stderr, "\n-- %s/%s --\n", pkgPath, origName)
|
|
|
|
@ -594,6 +598,15 @@ func transformCompile(args []string) ([]string, error) {
|
|
|
|
|
newPaths = append(newPaths, tempFile.Name())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if len(privateNameMap) > 0 {
|
|
|
|
|
outputDirectory := filepath.Dir(flagValue(flags, "-o"))
|
|
|
|
|
data, _ := json.Marshal(privateNameMap)
|
|
|
|
|
err := ioutil.WriteFile(filepath.Join(outputDirectory, garbleMapFile), data, 0644)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return append(flags, newPaths...), nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -697,7 +710,7 @@ func buildidOf(path string) (string, error) {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func hashWith(salt, value string) string {
|
|
|
|
|
const length = 8
|
|
|
|
|
const length = 4
|
|
|
|
|
|
|
|
|
|
d := sha256.New()
|
|
|
|
|
io.WriteString(d, salt)
|
|
|
|
@ -711,6 +724,17 @@ func hashWith(salt, value string) string {
|
|
|
|
|
return "z" + sum[:length]
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func encodeIntToName(i int) string {
|
|
|
|
|
const privateNameCharset = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_"
|
|
|
|
|
name := "z"
|
|
|
|
|
for i > 0 {
|
|
|
|
|
charIdx := i % len(privateNameCharset)
|
|
|
|
|
i -= charIdx + 1
|
|
|
|
|
name += string(privateNameCharset[charIdx])
|
|
|
|
|
}
|
|
|
|
|
return name
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// buildBlacklist collects all the objects in a package which are known to be
|
|
|
|
|
// used with reflect.TypeOf or reflect.ValueOf. Since we obfuscate one package
|
|
|
|
|
// at a time, we only detect those if the type definition and the reflect usage
|
|
|
|
@ -776,7 +800,7 @@ func buildBlacklist(files []*ast.File, info *types.Info, pkg *types.Package) map
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// transformGo garbles the provided Go syntax node.
|
|
|
|
|
func transformGo(file *ast.File, info *types.Info, blacklist map[types.Object]struct{}) *ast.File {
|
|
|
|
|
func transformGo(file *ast.File, info *types.Info, blacklist map[types.Object]struct{}, nameMap map[string]string) *ast.File {
|
|
|
|
|
// Shuffle top level declarations
|
|
|
|
|
mathrand.Shuffle(len(file.Decls), func(i, j int) {
|
|
|
|
|
decl1 := file.Decls[i]
|
|
|
|
@ -886,8 +910,22 @@ func transformGo(file *ast.File, info *types.Info, blacklist map[types.Object]st
|
|
|
|
|
}
|
|
|
|
|
buildID = id
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if token.IsExported(node.Name) || !envGarbleTiny {
|
|
|
|
|
node.Name = hashWith(buildID, node.Name)
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if name, ok := nameMap[node.Name]; ok {
|
|
|
|
|
node.Name = name
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
name := encodeIntToName(len(nameMap) + 1)
|
|
|
|
|
|
|
|
|
|
// orig := node.Name
|
|
|
|
|
node.Name = hashWith(buildID, node.Name)
|
|
|
|
|
nameMap[node.Name] = name
|
|
|
|
|
node.Name = name
|
|
|
|
|
// log.Printf("%q hashed with %q to %q", orig, buildID, node.Name)
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
@ -947,7 +985,7 @@ func transformLink(args []string) ([]string, error) {
|
|
|
|
|
importCfgPath := flagValue(flags, "-importcfg")
|
|
|
|
|
// there should only ever be one archive/object file passed to the linker,
|
|
|
|
|
// the file for the main package or entrypoint
|
|
|
|
|
garbledImports, err := obfuscateImports(paths[0], importCfgPath)
|
|
|
|
|
garbledImports, nameMap, err := obfuscateImports(paths[0], importCfgPath)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
@ -976,9 +1014,12 @@ func transformLink(args []string) ([]string, error) {
|
|
|
|
|
pkgPath = buildInfo.firstImport
|
|
|
|
|
}
|
|
|
|
|
if id := buildInfo.imports[pkgPath].buildID; id != "" {
|
|
|
|
|
newName, ok := nameMap[pkgPath+"."+name]
|
|
|
|
|
if !ok {
|
|
|
|
|
newName = hashWith(id, name)
|
|
|
|
|
}
|
|
|
|
|
garbledPkg := garbledImports[pkg]
|
|
|
|
|
name = hashWith(id, name)
|
|
|
|
|
flags = append(flags, fmt.Sprintf("-X=%s.%s=%s", garbledPkg, name, str))
|
|
|
|
|
flags = append(flags, fmt.Sprintf("-X=%s.%s=%s", garbledPkg, newName, str))
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|