@ -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 }
}