Add split obfuscator

pull/81/head
Pagran 5 years ago committed by pagran
parent 65461aabce
commit c0eaf5ee43

@ -19,6 +19,7 @@ var (
obfuscators = []obfuscator{
xor{},
swap{},
split{},
}
envGarbleSeed = os.Getenv("GARBLE_SEED")
)

@ -0,0 +1,198 @@
package literals
import (
"fmt"
"go/ast"
"go/token"
mathrand "math/rand"
ah "mvdan.cc/garble/internal/asthelper"
)
const maxChunkSize = 4
const minCaseCount = 3
type split struct{}
// check that the obfuscator interface is implemented
var _ obfuscator = split{}
func splitIntoRandomChunks(data []byte) [][]byte {
if len(data) == 1 {
return [][]byte{data}
}
var chunks [][]byte
for {
if len(data) == 0 {
break
}
chunkSize := 1 + mathrand.Intn(maxChunkSize)
if chunkSize > len(data) {
chunkSize = len(data)
}
chunks = append(chunks, data[:chunkSize])
data = data[chunkSize:]
}
return chunks
}
func splitIntoOneByteChunks(data []byte) [][]byte {
var chunks [][]byte
for _, d := range data {
chunks = append(chunks, []byte{d})
}
return chunks
}
func shuffleStmts(stmts []ast.Stmt) []ast.Stmt {
mathrand.Shuffle(len(stmts), func(i, j int) {
stmts[i], stmts[j] = stmts[j], stmts[i]
})
return stmts
}
func (x split) obfuscate(data []byte) *ast.BlockStmt {
var chunks [][]byte
if len(data)/maxChunkSize < minCaseCount { // Short arrays should be divided into single-byte fragments
chunks = splitIntoOneByteChunks(data)
} else {
chunks = splitIntoRandomChunks(data)
}
// Generate indexes for cases chunk count + 1 decrypt case + 1 exit case
indexes := mathrand.Perm(len(chunks) + 2)
decryptKeyInitial := mathrand.Int()
decryptKey := decryptKeyInitial
// Calculate decrypt key based on indexes and position. Ignore exit index
for i, index := range indexes[:len(indexes)-1] {
decryptKey ^= index * i
}
decryptIndex := indexes[len(indexes)-2]
exitIndex := indexes[len(indexes)-1]
for chunkIdx := range chunks {
chunk := chunks[chunkIdx]
for i := range chunk { // Encrypt all data with the decryptKey key
chunk[i] ^= byte(decryptKey)
}
}
switchCases := []ast.Stmt{
&ast.CaseClause{
List: []ast.Expr{ah.IntLit(decryptIndex)},
Body: shuffleStmts([]ast.Stmt{
&ast.AssignStmt{
Lhs: []ast.Expr{ah.Ident("i")},
Tok: token.ASSIGN,
Rhs: []ast.Expr{ah.IntLit(exitIndex)},
},
&ast.RangeStmt{
Key: ah.Ident("y"),
Tok: token.DEFINE,
X: ah.Ident("data"),
Body: ah.BlockStmt(&ast.AssignStmt{
Lhs: []ast.Expr{ah.IndexExpr("data", ah.Ident("y"))},
Tok: token.XOR_ASSIGN,
Rhs: []ast.Expr{ah.CallExpr(ah.Ident("byte"), ah.Ident("decryptKey"))},
}),
},
}),
},
}
for i := range chunks {
index := indexes[i]
nextIndex := indexes[i+1]
chunk := chunks[i]
var literal *ast.BasicLit
if len(chunk) != 1 {
literal = &ast.BasicLit{
Kind: token.STRING,
// TODO: Is it correct to generate append(arr, "str"...) expressions like this?
Value: fmt.Sprintf("%q...", chunk),
}
} else {
literal = ah.IntLit(int(chunk[0]))
}
switchCases = append(switchCases, &ast.CaseClause{
List: []ast.Expr{ah.IntLit(index)},
Body: shuffleStmts([]ast.Stmt{
&ast.AssignStmt{
Lhs: []ast.Expr{ah.Ident("i")},
Tok: token.ASSIGN,
Rhs: []ast.Expr{ah.IntLit(nextIndex)},
},
&ast.AssignStmt{
Lhs: []ast.Expr{ah.Ident("data")},
Tok: token.ASSIGN,
Rhs: []ast.Expr{
&ast.CallExpr{
Fun: ah.Ident("append"),
Args: []ast.Expr{
ah.Ident("data"),
literal,
},
},
},
},
}),
})
}
return ah.BlockStmt(&ast.DeclStmt{
Decl: &ast.GenDecl{
Tok: token.VAR,
Specs: []ast.Spec{
&ast.ValueSpec{
Names: []*ast.Ident{ah.Ident("data")},
Type: &ast.ArrayType{Elt: ah.Ident("byte")},
},
},
},
},
&ast.AssignStmt{
Lhs: []ast.Expr{ah.Ident("i")},
Tok: token.DEFINE,
Rhs: []ast.Expr{ah.IntLit(indexes[0])},
},
&ast.AssignStmt{
Lhs: []ast.Expr{ah.Ident("decryptKey")},
Tok: token.DEFINE,
Rhs: []ast.Expr{ah.IntLit(decryptKeyInitial)},
},
&ast.ForStmt{
Init: &ast.AssignStmt{
Lhs: []ast.Expr{ah.Ident("counter")},
Tok: token.DEFINE,
Rhs: []ast.Expr{ah.IntLit(0)},
},
Cond: &ast.BinaryExpr{
X: ah.Ident("i"),
Op: token.NEQ,
Y: ah.IntLit(indexes[len(indexes)-1]),
},
Post: &ast.IncDecStmt{
X: ah.Ident("counter"),
Tok: token.INC,
},
Body: ah.BlockStmt(
&ast.AssignStmt{
Lhs: []ast.Expr{ah.Ident("decryptKey")},
Tok: token.XOR_ASSIGN,
Rhs: []ast.Expr{
&ast.BinaryExpr{
X: ah.Ident("i"),
Op: token.MUL,
Y: ah.Ident("counter"),
},
},
},
&ast.SwitchStmt{
Tag: ah.Ident("i"),
Body: ah.BlockStmt(shuffleStmts(switchCases)...),
}),
})
}

@ -42,6 +42,9 @@ grep '^\s+\w+\[\w+\] = \w+\[\w+\] \^ \w+$' .obf-src/main/z0.go
# Swap obfuscator. Detect [...]byte|uint16|uint32|uint64{...}
grep '^\s+\w+ := \[\.{3}\](byte|uint16|uint32|uint64)\{[0-9\s,]+\}$' .obf-src/main/z0.go
# Split obfuscator. Detect decryptKey ^= i *- counter
grep '^\s+\w+ \^= \w+ \* \w+$' .obf-src/main/z0.go
-- go.mod --
module test/main

Loading…
Cancel
Save