|
|
|
@ -7,12 +7,54 @@ import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"go/ast"
|
|
|
|
|
"go/token"
|
|
|
|
|
"math"
|
|
|
|
|
mathrand "math/rand"
|
|
|
|
|
ah "mvdan.cc/garble/internal/asthelper"
|
|
|
|
|
"slices"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// extKeyRarity probability of using an external key.
|
|
|
|
|
// Larger value, greater probability of using an external key.
|
|
|
|
|
// Must be between 0 and 1
|
|
|
|
|
type extKeyRarity float32
|
|
|
|
|
|
|
|
|
|
const (
|
|
|
|
|
rareRarity extKeyRarity = 0.4
|
|
|
|
|
normalRarity extKeyRarity = 0.6
|
|
|
|
|
commonRarity extKeyRarity = 0.8
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func (r extKeyRarity) Try(rand *mathrand.Rand) bool {
|
|
|
|
|
return rand.Float32() < float32(r)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// extKey contains all information about the external key
|
|
|
|
|
type extKey struct {
|
|
|
|
|
name, typ string
|
|
|
|
|
value uint64
|
|
|
|
|
bits int
|
|
|
|
|
refs int
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (k *extKey) Type() *ast.Ident {
|
|
|
|
|
return ast.NewIdent(k.typ)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (k *extKey) Name() *ast.Ident {
|
|
|
|
|
return ast.NewIdent(k.name)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (k *extKey) AddRef() {
|
|
|
|
|
k.refs++
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (k *extKey) IsUsed() bool {
|
|
|
|
|
return k.refs > 0
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// obfuscator takes a byte slice and converts it to a ast.BlockStmt
|
|
|
|
|
type obfuscator interface {
|
|
|
|
|
obfuscate(obfRand *mathrand.Rand, data []byte) *ast.BlockStmt
|
|
|
|
|
obfuscate(obfRand *mathrand.Rand, data []byte, extKeys []*extKey) *ast.BlockStmt
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var (
|
|
|
|
@ -79,9 +121,147 @@ func operatorToReversedBinaryExpr(t token.Token, x, y ast.Expr) *ast.BinaryExpr
|
|
|
|
|
return expr
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const (
|
|
|
|
|
// minExtKeyCount is minimum number of external keys for one lambda call
|
|
|
|
|
minExtKeyCount = 2
|
|
|
|
|
// maxExtKeyCount is maximum number of external keys for one lambda call
|
|
|
|
|
maxExtKeyCount = 6
|
|
|
|
|
|
|
|
|
|
// minByteSliceExtKeyOps minimum number of operations with external keys for one byte slice
|
|
|
|
|
minByteSliceExtKeyOps = 2
|
|
|
|
|
// maxByteSliceExtKeyOps maximum number of operations with external keys for one byte slice
|
|
|
|
|
maxByteSliceExtKeyOps = 12
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// extKeyRanges contains a list of different ranges of random numbers for external keys
|
|
|
|
|
// Different types and bitnesses will increase the chance of changing patterns
|
|
|
|
|
var extKeyRanges = []struct {
|
|
|
|
|
typ string
|
|
|
|
|
max uint64
|
|
|
|
|
bits int
|
|
|
|
|
}{
|
|
|
|
|
{"uint8", math.MaxUint8, 8},
|
|
|
|
|
{"uint16", math.MaxUint16, 16},
|
|
|
|
|
{"uint32", math.MaxUint32, 32},
|
|
|
|
|
{"uint64", math.MaxUint64, 64},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func randExtKey(obfRand *mathrand.Rand, idx int) *extKey {
|
|
|
|
|
r := extKeyRanges[obfRand.Intn(len(extKeyRanges))]
|
|
|
|
|
return &extKey{
|
|
|
|
|
name: fmt.Sprintf("__garble_ext_key_%d", idx),
|
|
|
|
|
typ: r.typ,
|
|
|
|
|
value: obfRand.Uint64() & r.max,
|
|
|
|
|
bits: r.bits,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func randExtKeys(obfRand *mathrand.Rand) []*extKey {
|
|
|
|
|
count := minExtKeyCount + obfRand.Intn(maxExtKeyCount-minExtKeyCount)
|
|
|
|
|
|
|
|
|
|
keys := make([]*extKey, count)
|
|
|
|
|
for i := 0; i < count; i++ {
|
|
|
|
|
keys[i] = randExtKey(obfRand, i)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return keys
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func extKeysToParams(obfRand *obfRand, keys []*extKey) (params *ast.FieldList, args []ast.Expr) {
|
|
|
|
|
params = &ast.FieldList{}
|
|
|
|
|
for _, key := range keys {
|
|
|
|
|
name := key.Name()
|
|
|
|
|
if !key.IsUsed() {
|
|
|
|
|
name.Name = "_"
|
|
|
|
|
}
|
|
|
|
|
params.List = append(params.List, &ast.Field{
|
|
|
|
|
Names: []*ast.Ident{name},
|
|
|
|
|
Type: key.Type(),
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
if rareRarity.Try(obfRand.Rand) {
|
|
|
|
|
args = append(args, obfRand.scrambleUsingGlobalKey(key))
|
|
|
|
|
} else {
|
|
|
|
|
args = append(args, &ast.BasicLit{
|
|
|
|
|
Kind: token.INT,
|
|
|
|
|
Value: fmt.Sprint(key.value),
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// dataToByteSliceWithExtKeys scramble and turns a byte slice into an AST expression like:
|
|
|
|
|
//
|
|
|
|
|
// func() []byte {
|
|
|
|
|
// data := []byte("<data>")
|
|
|
|
|
// data[<index>] = data[<index>] <random operator> byte(<external key> >> <random shift>) // repeated random times
|
|
|
|
|
// return data
|
|
|
|
|
// }()
|
|
|
|
|
func dataToByteSliceWithExtKeys(obfRand *mathrand.Rand, data []byte, extKeys []*extKey) ast.Expr {
|
|
|
|
|
extKeyOpCount := minByteSliceExtKeyOps + obfRand.Intn(maxByteSliceExtKeyOps-minByteSliceExtKeyOps)
|
|
|
|
|
|
|
|
|
|
var stmts []ast.Stmt
|
|
|
|
|
for i := 0; i < extKeyOpCount; i++ {
|
|
|
|
|
key := extKeys[obfRand.Intn(len(extKeys))]
|
|
|
|
|
key.AddRef()
|
|
|
|
|
|
|
|
|
|
idx, op, b := obfRand.Intn(len(data)), randOperator(obfRand), obfRand.Intn(key.bits/8)
|
|
|
|
|
data[idx] = evalOperator(op, data[idx], byte(key.value>>(b*8)))
|
|
|
|
|
stmts = append(stmts, &ast.AssignStmt{
|
|
|
|
|
Lhs: []ast.Expr{ah.IndexExpr("data", ah.IntLit(idx))},
|
|
|
|
|
Tok: token.ASSIGN,
|
|
|
|
|
Rhs: []ast.Expr{
|
|
|
|
|
operatorToReversedBinaryExpr(op,
|
|
|
|
|
ah.IndexExpr("data", ah.IntLit(idx)),
|
|
|
|
|
ah.CallExprByName("byte", &ast.BinaryExpr{
|
|
|
|
|
X: key.Name(),
|
|
|
|
|
Op: token.SHR,
|
|
|
|
|
Y: ah.IntLit(b * 8),
|
|
|
|
|
}),
|
|
|
|
|
),
|
|
|
|
|
},
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// External keys can be applied several times to the same array element,
|
|
|
|
|
// and it is important to invert the order of execution to correctly restore the original value
|
|
|
|
|
slices.Reverse(stmts)
|
|
|
|
|
|
|
|
|
|
stmts = append([]ast.Stmt{ah.AssignDefineStmt(ast.NewIdent("data"), ah.DataToByteSlice(data))}, append(stmts, ah.ReturnStmt(ast.NewIdent("data")))...)
|
|
|
|
|
return ah.LambdaCall(&ast.ArrayType{Elt: ast.NewIdent("byte")}, ah.BlockStmt(stmts...))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// dataToByteSliceWithExtKeys scramble and turns a byte into an AST expression like:
|
|
|
|
|
//
|
|
|
|
|
// byte(<obfuscated value>) <random operator> byte(<external key> >> <random shift>)
|
|
|
|
|
func byteLitWithExtKey(obfRand *mathrand.Rand, val byte, extKeys []*extKey, rarity extKeyRarity) ast.Expr {
|
|
|
|
|
if !rarity.Try(obfRand) {
|
|
|
|
|
return ah.IntLit(int(val))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
key := extKeys[obfRand.Intn(len(extKeys))]
|
|
|
|
|
key.AddRef()
|
|
|
|
|
|
|
|
|
|
op, b := randOperator(obfRand), obfRand.Intn(key.bits/8)
|
|
|
|
|
newVal := evalOperator(op, val, byte(key.value>>(b*8)))
|
|
|
|
|
return operatorToReversedBinaryExpr(op,
|
|
|
|
|
ah.CallExprByName("byte", ah.IntLit(int(newVal))),
|
|
|
|
|
ah.CallExprByName("byte", &ast.BinaryExpr{
|
|
|
|
|
X: key.Name(),
|
|
|
|
|
Op: token.SHR,
|
|
|
|
|
Y: ah.IntLit(b * 8),
|
|
|
|
|
}),
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type obfRand struct {
|
|
|
|
|
*mathrand.Rand
|
|
|
|
|
testObfuscator obfuscator
|
|
|
|
|
|
|
|
|
|
globalKeysVarName string
|
|
|
|
|
globalKeys []uint64
|
|
|
|
|
globalKeysUsed bool
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (r *obfRand) nextObfuscator() obfuscator {
|
|
|
|
@ -98,7 +278,26 @@ func (r *obfRand) nextLinearTimeObfuscator() obfuscator {
|
|
|
|
|
return Obfuscators[r.Intn(len(LinearTimeObfuscators))]
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (r *obfRand) scrambleUsingGlobalKey(k *extKey) ast.Expr {
|
|
|
|
|
r.globalKeysUsed = true
|
|
|
|
|
|
|
|
|
|
extKeyIdx := r.Intn(len(r.globalKeys))
|
|
|
|
|
return &ast.BinaryExpr{
|
|
|
|
|
X: ah.CallExprByName(k.typ, ah.IndexExpr(r.globalKeysVarName, ah.IntLit(extKeyIdx))),
|
|
|
|
|
Op: token.XOR,
|
|
|
|
|
Y: &ast.BasicLit{
|
|
|
|
|
Kind: token.INT,
|
|
|
|
|
// To avoid an overflow error at compile time, truncate global key to external key bitness
|
|
|
|
|
Value: fmt.Sprint(k.value ^ (r.globalKeys[extKeyIdx] & ((1 << k.bits) - 1))),
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func newObfRand(rand *mathrand.Rand, file *ast.File) *obfRand {
|
|
|
|
|
testObf := testPkgToObfuscatorMap[file.Name.Name]
|
|
|
|
|
return &obfRand{rand, testObf}
|
|
|
|
|
globalKeys := make([]uint64, minExtKeyCount+rand.Intn(maxExtKeyCount-minExtKeyCount))
|
|
|
|
|
for i := 0; i < len(globalKeys); i++ {
|
|
|
|
|
globalKeys[i] = rand.Uint64()
|
|
|
|
|
}
|
|
|
|
|
return &obfRand{rand, testObf, fmt.Sprintf("__garble_global_keys_%d", rand.Uint64()), globalKeys, false}
|
|
|
|
|
}
|
|
|
|
|