boolSubstitution::runOnFunction(Function &F){ //检查是否设置了有效的混淆次数 if (ObfTimes <= 0) { errs()<<"Substitution application number -sub_loop=x must be x > 0"; returnfalse; }
Function *tmp = &F; //判断是否需要对当前函数进行混淆操作 if (toObfuscate(flag, tmp, "sub")) { substitute(tmp); returntrue; }
boolSubstitution::substitute(Function *f){ Function *tmp = f;
// 执行替换混淆操作的次数 int times = ObfTimes; do { //第一个 for 循环遍历函数中的所有基本块,第二个 for 循环遍历当前基本块中的所有指令 for (Function::iterator bb = tmp->begin(); bb != tmp->end(); ++bb) { for (BasicBlock::iterator inst = bb->begin(); inst != bb->end(); ++inst) { //检查当前指令是否是二元操作符 if (inst->isBinaryOp()) { //根据指令的操作符类型(getOpcode() 返回操作符类型)进入不同的分支 switch (inst->getOpcode()) { case BinaryOperator::Add: //随机选择一个加法替换函数,然后对当前指令进行替换。 (this->*funcAdd[llvm::cryptoutils->get_range(NUMBER_ADD_SUBST)])( cast<BinaryOperator>(inst)); ++Add; break; case BinaryOperator::Sub: //如果是减法操作 (Sub),则从 funcSub 数组中随机选择一个减法替换函数进行替换,并增加 Sub 计数器。 (this->*funcSub[llvm::cryptoutils->get_range(NUMBER_SUB_SUBST)])( cast<BinaryOperator>(inst)); ++Sub; break; ... default: break; } // End switch } // End isBinaryOp } // End for basickblock } // End for Function } while (--times > 0); // for times returnfalse; }
addNeg
实现了对加法指令的替换,具体将 加法操作(a = b + c) 转换为等效的操作 **a = b - (-c)**。
virtualboolrunOnFunction(Function &F){ //检查混淆次数是否有效 if (ObfTimes <= 0) { errs()<<"BogusControlFlow application number -bcf_loop=x must be x > 0"; returnfalse; }
// 检查混淆概率是否有效 if ( !((ObfProbRate > 0) && (ObfProbRate <= 100)) ) { errs()<<"BogusControlFlow application basic blocks percentage -bcf_prob=x must be 0 < x <= 100"; returnfalse; } //判断是否进行混淆 if(toObfuscate(flag,&F,"bcf")) { bogus(F);//混淆函数 doF(*F.getParent());//进一步的处理 returntrue; }
voidbogus(Function &F){ //初始统计和调试信息 ++NumFunction; int NumBasicBlocks = 0; bool firstTime = true; // First time we do the loop in this function bool hasBeenModified = false; //... NumTimesOnFunctions = ObfTimes; int NumObfTimes = ObfTimes;
// 主混淆逻辑 do{//调试输出 CFG(控制流图):在混淆前查看函数的基本块结构 DEBUG_WITH_TYPE("cfg", errs() << "bcf: Function " << F.getName()<<", before the pass:\n"); DEBUG_WITH_TYPE("cfg", F.viewCFG()); // 将当前函数的所有基本块保存到 basicBlocks 列表中,用于后续处理。 std::list<BasicBlock *> basicBlocks; for (Function::iterator i=F.begin();i!=F.end();++i) { basicBlocks.push_back(&*i); } DEBUG_WITH_TYPE("gen", errs() << "bcf: Iterating on the Function's Basic Blocks\n"); //遍历所有基本块 while(!basicBlocks.empty()){ NumBasicBlocks ++; //根据 ObfProbRate 决定是否对当前基本块应用混淆 if((int)llvm::cryptoutils->get_range(100) <= ObfProbRate){ DEBUG_WITH_TYPE("opt", errs() << "bcf: Block "<< NumBasicBlocks <<" selected. \n"); hasBeenModified = true; ++NumModifiedBasicBlocks; NumAddedBasicBlocks += 3; FinalNumBasicBlocks += 3; // Add bogus flow to the given Basic Block (see description) BasicBlock *basicBlock = basicBlocks.front(); //调用 addBogusFlow,对基本块插入假的控制流。 addBogusFlow(basicBlock, F); } else{ DEBUG_WITH_TYPE("opt", errs() << "bcf: Block "<< NumBasicBlocks <<" not selected.\n"); } // remove the block from the list basicBlocks.pop_front(); if(firstTime){ // 如果是第一次处理当前函数,初始化基本块计数器。 ++InitNumBasicBlocks; ++FinalNumBasicBlocks; } } // end of while(!basicBlocks.empty()) ... firstTime = false; }while(--NumObfTimes > 0); }
addBogusFlow对基本块插入假的控制流
1
virtualvoidaddBogusFlow(BasicBlock * basicBlock, Function &F){
DEBUG_WITH_TYPE("gen", errs() << "bcf: The cloned basic block is now correct\n"); DEBUG_WITH_TYPE("gen", errs() << "bcf: Starting to add junk code in the cloned bloc...\n");
// 遍历基本块中的指令 for (BasicBlock::iterator i = alteredBB->begin(), e = alteredBB->end() ; i != e; ++i){ // 如果指令是运算指令,进行垃圾指令添加 if(i->isBinaryOp()){ // binary instructions unsigned opcode = i->getOpcode(); BinaryOperator *op, *op1 = NULL; Twine *var = newTwine("_"); // treat differently float or int // Binary int if(opcode == Instruction::Add || opcode == Instruction::Sub || opcode == Instruction::Mul || opcode == Instruction::UDiv || opcode == Instruction::SDiv || opcode == Instruction::URem || opcode == Instruction::SRem || opcode == Instruction::Shl || opcode == Instruction::LShr || opcode == Instruction::AShr || opcode == Instruction::And || opcode == Instruction::Or || opcode == Instruction::Xor){ for(int random = (int)llvm::cryptoutils->get_range(10); random < 10; ++random){ switch(llvm::cryptoutils->get_range(4)){ // to improve case0: //do nothing break; case1: op = BinaryOperator::CreateNeg(i->getOperand(0),*var,&*i); op1 = BinaryOperator::Create(Instruction::Add,op, i->getOperand(1),"gen",&*i); break; case2: op1 = BinaryOperator::Create(Instruction::Sub, i->getOperand(0), i->getOperand(1),*var,&*i); op = BinaryOperator::Create(Instruction::Mul,op1, i->getOperand(1),"gen",&*i); break; case3: op = BinaryOperator::Create(Instruction::Shl, i->getOperand(0), i->getOperand(1),*var,&*i); break; } } } // Binary float if(opcode == Instruction::FAdd || opcode == Instruction::FSub || opcode == Instruction::FMul || opcode == Instruction::FDiv || opcode == Instruction::FRem){ for(int random = (int)llvm::cryptoutils->get_range(10); random < 10; ++random){ switch(llvm::cryptoutils->get_range(3)){ // can be improved case0: //do nothing break; case1: op = BinaryOperator::CreateFNeg(i->getOperand(0),*var,&*i); op1 = BinaryOperator::Create(Instruction::FAdd,op, i->getOperand(1),"gen",&*i); break; case2: op = BinaryOperator::Create(Instruction::FSub, i->getOperand(0), i->getOperand(1),*var,&*i); op1 = BinaryOperator::Create(Instruction::FMul,op, i->getOperand(1),"gen",&*i); break; } } } if(opcode == Instruction::ICmp){ // Condition (with int) ICmpInst *currentI = (ICmpInst*)(&i); switch(llvm::cryptoutils->get_range(3)){ // must be improved case0: //do nothing break; case1: currentI->swapOperands(); break; case2: // randomly change the predicate switch(llvm::cryptoutils->get_range(10)){ case0: currentI->setPredicate(ICmpInst::ICMP_EQ); break; // equal case1: currentI->setPredicate(ICmpInst::ICMP_NE); break; // not equal case2: currentI->setPredicate(ICmpInst::ICMP_UGT); break; // unsigned greater than case3: currentI->setPredicate(ICmpInst::ICMP_UGE); break; // unsigned greater or equal case4: currentI->setPredicate(ICmpInst::ICMP_ULT); break; // unsigned less than case5: currentI->setPredicate(ICmpInst::ICMP_ULE); break; // unsigned less or equal case6: currentI->setPredicate(ICmpInst::ICMP_SGT); break; // signed greater than case7: currentI->setPredicate(ICmpInst::ICMP_SGE); break; // signed greater or equal case8: currentI->setPredicate(ICmpInst::ICMP_SLT); break; // signed less than case9: currentI->setPredicate(ICmpInst::ICMP_SLE); break; // signed less or equal } break; }
} if(opcode == Instruction::FCmp){ // Conditions (with float) FCmpInst *currentI = (FCmpInst*)(&i); switch(llvm::cryptoutils->get_range(3)){ // must be improved case0: //do nothing break; case1: currentI->swapOperands(); break; case2: // randomly change the predicate switch(llvm::cryptoutils->get_range(10)){ case0: currentI->setPredicate(FCmpInst::FCMP_OEQ); break; // ordered and equal case1: currentI->setPredicate(FCmpInst::FCMP_ONE); break; // ordered and operands are unequal case2: currentI->setPredicate(FCmpInst::FCMP_UGT); break; // unordered or greater than case3: currentI->setPredicate(FCmpInst::FCMP_UGE); break; // unordered, or greater than, or equal case4: currentI->setPredicate(FCmpInst::FCMP_ULT); break; // unordered or less than case5: currentI->setPredicate(FCmpInst::FCMP_ULE); break; // unordered, or less than, or equal case6: currentI->setPredicate(FCmpInst::FCMP_OGT); break; // ordered and greater than case7: currentI->setPredicate(FCmpInst::FCMP_OGE); break; // ordered and greater than or equal case8: currentI->setPredicate(FCmpInst::FCMP_OLT); break; // ordered and less than case9: currentI->setPredicate(FCmpInst::FCMP_OLE); break; // ordered or less than, or equal } break; } } } } return alteredBB; }
// 处理双分支(条件跳转)的基本块 if (i->getTerminator()->getNumSuccessors() == 2) { // Get next cases ConstantInt *numCaseTrue =switchI->findCaseDest(i->getTerminator()->getSuccessor(0)); ConstantInt *numCaseFalse =switchI->findCaseDest(i->getTerminator()->getSuccessor(1));
// Check if next case == default case (switchDefault) if (numCaseTrue == NULL) { numCaseTrue = cast<ConstantInt>( ConstantInt::get(switchI->getCondition()->getType(), llvm::cryptoutils->scramble32( switchI->getNumCases() - 1, scrambling_key))); }