 Chromium Code Reviews
 Chromium Code Reviews Issue 1185703004:
  Add constant blinding/pooling option for X8632 code translation  (Closed) 
  Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master
    
  
    Issue 1185703004:
  Add constant blinding/pooling option for X8632 code translation  (Closed) 
  Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master| Index: src/IceTargetLoweringX8632.cpp | 
| diff --git a/src/IceTargetLoweringX8632.cpp b/src/IceTargetLoweringX8632.cpp | 
| index 506b315079909210d3b1f7c9dc1fbee21e0e53ea..36f1ed386b126fdc6983d83683da7ea47ae1226a 100644 | 
| --- a/src/IceTargetLoweringX8632.cpp | 
| +++ b/src/IceTargetLoweringX8632.cpp | 
| @@ -438,6 +438,16 @@ TargetX8632::TargetX8632(Cfg *Func) | 
| TypeToRegisterSet[IceType_v8i16] = VectorRegisters; | 
| TypeToRegisterSet[IceType_v4i32] = VectorRegisters; | 
| TypeToRegisterSet[IceType_v4f32] = VectorRegisters; | 
| + | 
| + // qining: initialize the assistant pause flag | 
| 
Jim Stichnoth
2015/06/12 23:48:18
"qining:"?  (here and below)
What does "assistant
 
qining
2015/06/17 04:28:53
Done. Removed
 | 
| + // for constant blinding and pooling. | 
| + // When this is set to true, all constant blinding | 
| + // and pooling will be disabled. This should be set to | 
| + // true for doLoadOpt() and advancedPhiLowering(). | 
| + // Note, when a physical regs are available for the Dest | 
| 
Jim Stichnoth
2015/06/12 23:48:18
I'm not 100% sure what this part of the comment me
 | 
| + // of Phi lowering assignment, we should set this flag to | 
| + // false to enable blinding/pooling for that instruction. | 
| + RandomizationPoolingPaused = false; | 
| 
Jim Stichnoth
2015/06/12 23:48:17
Do this simple initialization in the initializer l
 
qining
2015/06/13 00:41:26
Agree, but should we still leave the comments (may
 
qining
2015/06/17 04:28:54
Done.
 | 
| } | 
| void TargetX8632::translateO2() { | 
| @@ -482,7 +492,12 @@ void TargetX8632::translateO2() { | 
| return; | 
| Func->dump("After x86 address mode opt"); | 
| + // qining: disable constant blinding or pooling for load optimization | 
| + bool RPLatchForDoLoadOpt = RandomizationPoolingPaused; | 
| 
Jim Stichnoth
2015/06/12 23:48:17
You are using this pattern enough times (5?) that
 
qining
2015/06/17 04:28:54
Done.
 | 
| + RandomizationPoolingPaused = true; | 
| doLoadOpt(); | 
| + // qining: Resume constant blinding and pooling | 
| + RandomizationPoolingPaused = RPLatchForDoLoadOpt; | 
| Func->genCode(); | 
| if (Func->hasError()) | 
| return; | 
| @@ -509,7 +524,14 @@ void TargetX8632::translateO2() { | 
| Func->dump("After linear scan regalloc"); | 
| if (Ctx->getFlags().getPhiEdgeSplit()) { | 
| + // : In general we need to pause constant blinding or pooling | 
| 
Jim Stichnoth
2015/06/12 23:48:18
":" ?
 
qining
2015/06/17 04:28:53
Done. Should I remove this comment?
 | 
| + // advanced phi lowering, unless the lowering assignment has a | 
| 
Jim Stichnoth
2015/06/12 23:48:17
during advanced phi lowering
 
qining
2015/06/17 04:28:54
Done.
 | 
| + // physical register for the Dest Variable | 
| + bool RPLatchAdvancedPhiLowering = RandomizationPoolingPaused; | 
| + RandomizationPoolingPaused = true; | 
| Func->advancedPhiLowering(); | 
| + // qining: recover the pause flag; | 
| + RandomizationPoolingPaused = RPLatchAdvancedPhiLowering; | 
| Func->dump("After advanced Phi lowering"); | 
| } | 
| @@ -762,8 +784,9 @@ void TargetX8632::emitVariable(const Variable *Var) const { | 
| Str << "%" << getRegName(Var->getRegNum(), Var->getType()); | 
| return; | 
| } | 
| - if (Var->getWeight().isInf()) | 
| + if (Var->getWeight().isInf()) { | 
| llvm_unreachable("Infinite-weight Variable has no register assigned"); | 
| + } | 
| int32_t Offset = Var->getStackOffset(); | 
| if (!hasFramePointer()) | 
| Offset += getStackAdjustment(); | 
| @@ -776,8 +799,9 @@ void TargetX8632::emitVariable(const Variable *Var) const { | 
| X8632::Address TargetX8632::stackVarToAsmOperand(const Variable *Var) const { | 
| if (Var->hasReg()) | 
| llvm_unreachable("Stack Variable has a register assigned"); | 
| - if (Var->getWeight().isInf()) | 
| + if (Var->getWeight().isInf()) { | 
| llvm_unreachable("Infinite-weight Variable has no register assigned"); | 
| + } | 
| int32_t Offset = Var->getStackOffset(); | 
| if (!hasFramePointer()) | 
| Offset += getStackAdjustment(); | 
| @@ -1168,12 +1192,19 @@ Operand *TargetX8632::loOperand(Operand *Operand) { | 
| return Var->getLo(); | 
| } | 
| if (ConstantInteger64 *Const = llvm::dyn_cast<ConstantInteger64>(Operand)) { | 
| - return Ctx->getConstantInt32(static_cast<uint32_t>(Const->getValue())); | 
| + ConstantInteger32 *ConstInt = llvm::dyn_cast<ConstantInteger32>( | 
| 
Jim Stichnoth
2015/06/12 23:48:18
Use cast<> instead of dyn_cast<>.
 
qining
2015/06/17 04:28:54
Done
 | 
| + Ctx->getConstantInt32(static_cast<uint32_t>(Const->getValue()))); | 
| 
Jim Stichnoth
2015/06/12 23:48:18
I think you should use int32_t instead of uint32_t
 
qining
2015/06/17 04:28:54
Done
 | 
| + // check if we need to blind/pool the constant | 
| + return randomizeOrPoolImmediate(ConstInt); | 
| } | 
| if (OperandX8632Mem *Mem = llvm::dyn_cast<OperandX8632Mem>(Operand)) { | 
| - return OperandX8632Mem::create(Func, IceType_i32, Mem->getBase(), | 
| - Mem->getOffset(), Mem->getIndex(), | 
| - Mem->getShift(), Mem->getSegmentRegister()); | 
| + OperandX8632Mem *MemOperand = OperandX8632Mem::create( | 
| + Func, IceType_i32, Mem->getBase(), Mem->getOffset(), Mem->getIndex(), | 
| + Mem->getShift(), Mem->getSegmentRegister()); | 
| + // Test if we should randomize or pool the offset, if so randomize it or | 
| + // pool it then create mem operand with the blinded/pooled constant. | 
| + // Otherwise, return the mem operand as ordinary mem operand. | 
| + return randomizeOrPoolImmediate(MemOperand); | 
| } | 
| llvm_unreachable("Unsupported operand type"); | 
| return nullptr; | 
| @@ -1189,8 +1220,10 @@ Operand *TargetX8632::hiOperand(Operand *Operand) { | 
| return Var->getHi(); | 
| } | 
| if (ConstantInteger64 *Const = llvm::dyn_cast<ConstantInteger64>(Operand)) { | 
| - return Ctx->getConstantInt32( | 
| - static_cast<uint32_t>(Const->getValue() >> 32)); | 
| + ConstantInteger32 *ConstInt = llvm::dyn_cast<ConstantInteger32>( | 
| 
Jim Stichnoth
2015/06/12 23:48:18
cast<>
 
qining
2015/06/17 04:28:54
Done
 | 
| + Ctx->getConstantInt32(static_cast<uint32_t>(Const->getValue() >> 32))); | 
| 
Jim Stichnoth
2015/06/12 23:48:17
int32_t
 
qining
2015/06/17 04:28:54
Done
 | 
| + // check if we need to blind/pool the constant | 
| + return randomizeOrPoolImmediate(ConstInt); | 
| } | 
| if (OperandX8632Mem *Mem = llvm::dyn_cast<OperandX8632Mem>(Operand)) { | 
| Constant *Offset = Mem->getOffset(); | 
| @@ -1206,9 +1239,13 @@ Operand *TargetX8632::hiOperand(Operand *Operand) { | 
| Ctx->getConstantSym(4 + SymOffset->getOffset(), SymOffset->getName(), | 
| SymOffset->getSuppressMangling()); | 
| } | 
| - return OperandX8632Mem::create(Func, IceType_i32, Mem->getBase(), Offset, | 
| - Mem->getIndex(), Mem->getShift(), | 
| - Mem->getSegmentRegister()); | 
| + OperandX8632Mem *MemOperand = OperandX8632Mem::create( | 
| + Func, IceType_i32, Mem->getBase(), Offset, Mem->getIndex(), | 
| + Mem->getShift(), Mem->getSegmentRegister()); | 
| + // Test if the Offset is an eligible i32 constants for randomization and | 
| + // pooling. Blind/pool it if it is. Otherwise return as oridinary mem | 
| + // operand. | 
| + return randomizeOrPoolImmediate(MemOperand); | 
| } | 
| llvm_unreachable("Unsupported operand type"); | 
| return nullptr; | 
| @@ -1839,18 +1876,30 @@ void TargetX8632::lowerAssign(const InstAssign *Inst) { | 
| _mov(DestHi, T_Hi); | 
| } else { | 
| Operand *RI; | 
| - if (Dest->hasReg()) | 
| + if (Dest->hasReg()) { | 
| // If Dest already has a physical register, then legalize the | 
| // Src operand into a Variable with the same register | 
| // assignment. This is mostly a workaround for advanced phi | 
| // lowering's ad-hoc register allocation which assumes no | 
| // register allocation is needed when at least one of the | 
| // operands is non-memory. | 
| + | 
| + // qining: if we have a physical register for the dest variable, | 
| + // we can enable our constant blinding or pooling again. Note | 
| + // this is only for advancedPhiLowering(), the flag flip should | 
| + // leave no other side effect. | 
| + bool RPLatchInLowerAssign = RandomizationPoolingPaused; | 
| + RandomizationPoolingPaused = false; | 
| + | 
| RI = legalize(Src0, Legal_Reg, Dest->getRegNum()); | 
| - else | 
| + | 
| + // qining: Resume constant blinding and pooling | 
| + RandomizationPoolingPaused = RPLatchInLowerAssign; | 
| + } else { | 
| // If Dest could be a stack operand, then RI must be a physical | 
| // register or a scalar integer immediate. | 
| RI = legalize(Src0, Legal_Reg | Legal_Imm); | 
| + } | 
| if (isVectorType(Dest->getType())) | 
| _movp(Dest, RI); | 
| else | 
| @@ -3119,10 +3168,11 @@ void TargetX8632::lowerIntrinsicCall(const InstIntrinsicCall *Instr) { | 
| Func->setError("Unexpected memory ordering for AtomicRMW"); | 
| return; | 
| } | 
| - lowerAtomicRMW(Instr->getDest(), | 
| - static_cast<uint32_t>(llvm::cast<ConstantInteger32>( | 
| - Instr->getArg(0))->getValue()), | 
| - Instr->getArg(1), Instr->getArg(2)); | 
| + lowerAtomicRMW( | 
| + Instr->getDest(), | 
| + static_cast<uint32_t>( | 
| + llvm::cast<ConstantInteger32>(Instr->getArg(0))->getValue()), | 
| + Instr->getArg(1), Instr->getArg(2)); | 
| return; | 
| case Intrinsics::AtomicStore: { | 
| if (!Intrinsics::isMemoryOrderValid( | 
| @@ -4325,6 +4375,10 @@ void TargetX8632::lowerUnreachable(const InstUnreachable * /*Inst*/) { _ud2(); } | 
| // turned into zeroes, since loOperand() and hiOperand() don't expect | 
| // Undef input. | 
| void TargetX8632::prelowerPhis() { | 
| + // Pause constant blinding or pooling, blinding or pooling will be done later | 
| + // during phi lowering assignments | 
| + RandomizationPoolingPaused = true; | 
| + | 
| CfgNode *Node = Context.getNode(); | 
| for (Inst &I : Node->getPhis()) { | 
| auto Phi = llvm::dyn_cast<InstPhi>(&I); | 
| @@ -4349,6 +4403,9 @@ void TargetX8632::prelowerPhis() { | 
| Phi->setDeleted(); | 
| } | 
| } | 
| + | 
| + // Recover the constant blinding/pooling state | 
| + RandomizationPoolingPaused = false; | 
| } | 
| namespace { | 
| @@ -4619,6 +4676,9 @@ Operand *TargetX8632::legalize(Operand *From, LegalMask Allowed, | 
| Mem->getShift(), Mem->getSegmentRegister()); | 
| } | 
| + // qining: For all Memory Operands, we do randomization/pooling from here | 
| + From = randomizeOrPoolImmediate(llvm::dyn_cast<OperandX8632Mem>(From)); | 
| 
Jim Stichnoth
2015/06/12 23:48:17
You've already done a dyn_cast<>(From) in the encl
 
qining
2015/06/17 04:28:53
Done
 | 
| + | 
| if (!(Allowed & Legal_Mem)) { | 
| From = copyToReg(From, RegNum); | 
| } | 
| @@ -4643,6 +4703,16 @@ Operand *TargetX8632::legalize(Operand *From, LegalMask Allowed, | 
| } | 
| // There should be no constants of vector type (other than undef). | 
| assert(!isVectorType(Ty)); | 
| + | 
| + // If the operand is an 32 bit constant integer, we should check | 
| + // whether we need to randomize it or pool it. | 
| + if (From && llvm::isa<ConstantInteger32>(From)) { | 
| 
Jim Stichnoth
2015/06/12 23:48:18
"From" must be non-null, so no need to test that.
 
qining
2015/06/17 04:28:54
Done
 | 
| + Operand *NewFrom = | 
| + randomizeOrPoolImmediate(llvm::cast<Constant>(From), RegNum); | 
| 
Jim Stichnoth
2015/06/12 23:48:17
You can avoid the cast<> and the earlier isa<> che
 
qining
2015/06/17 04:28:54
Done
 | 
| + if (NewFrom != From) | 
| + return NewFrom; | 
| + } | 
| + | 
| // Convert a scalar floating point constant into an explicit | 
| // memory operand. | 
| if (isScalarFloatingType(Ty)) { | 
| @@ -4706,25 +4776,40 @@ Operand *TargetX8632::legalizeSrc0ForCmp(Operand *Src0, Operand *Src1) { | 
| return legalize(Src0, IsSrc1ImmOrReg ? (Legal_Reg | Legal_Mem) : Legal_Reg); | 
| } | 
| -OperandX8632Mem *TargetX8632::formMemoryOperand(Operand *Operand, Type Ty, | 
| +OperandX8632Mem *TargetX8632::formMemoryOperand(Operand *operand, Type Ty, | 
| 
Jim Stichnoth
2015/06/12 23:48:17
Variable names should be capitalized, so maybe Opn
 
qining
2015/06/17 04:28:54
Done, replaced with Opnd.
 | 
| bool DoLegalize) { | 
| - OperandX8632Mem *Mem = llvm::dyn_cast<OperandX8632Mem>(Operand); | 
| + OperandX8632Mem *Mem = llvm::dyn_cast<OperandX8632Mem>(operand); | 
| // It may be the case that address mode optimization already creates | 
| // an OperandX8632Mem, so in that case it wouldn't need another level | 
| // of transformation. | 
| if (!Mem) { | 
| - Variable *Base = llvm::dyn_cast<Variable>(Operand); | 
| - Constant *Offset = llvm::dyn_cast<Constant>(Operand); | 
| + Variable *Base = llvm::dyn_cast<Variable>(operand); | 
| + Constant *Offset = llvm::dyn_cast<Constant>(operand); | 
| assert(Base || Offset); | 
| if (Offset) { | 
| - // Make sure Offset is not undef. | 
| + // qining: during memory operand building, we do not | 
| + // blind or pool the constant offset, we will work on | 
| + // the whole memory operand later as one entity later, | 
| + // this save one instruction. By turning blinding and | 
| + // pooling off, we guarantee legalize(Offset) will return | 
| + // a constant* | 
| + bool RPLatchForMemOperandBuilding = RandomizationPoolingPaused; | 
| + RandomizationPoolingPaused = true; | 
| + | 
| Offset = llvm::cast<Constant>(legalize(Offset)); | 
| + | 
| assert(llvm::isa<ConstantInteger32>(Offset) || | 
| llvm::isa<ConstantRelocatable>(Offset)); | 
| + | 
| + // qining: Resume constant blinding and pooling | 
| + RandomizationPoolingPaused = RPLatchForMemOperandBuilding; | 
| } | 
| Mem = OperandX8632Mem::create(Func, Ty, Base, Offset); | 
| } | 
| - return llvm::cast<OperandX8632Mem>(DoLegalize ? legalize(Mem) : Mem); | 
| + // qining: do legalization, which contains randomization/pooling | 
| + // or do randomization/pooling. | 
| + return llvm::cast<OperandX8632Mem>( | 
| + DoLegalize ? legalize(Mem) : randomizeOrPoolImmediate(Mem)); | 
| } | 
| Variable *TargetX8632::makeReg(Type Type, int32_t RegNum) { | 
| @@ -4965,6 +5050,19 @@ const char *PoolTypeConverter<double>::TypeName = "double"; | 
| const char *PoolTypeConverter<double>::AsmTag = ".quad"; | 
| const char *PoolTypeConverter<double>::PrintfString = "0x%llx"; | 
| +// Add converter for int type constant pooling | 
| +template <> struct PoolTypeConverter<int> { | 
| + typedef uint32_t PrimitiveIntType; | 
| + typedef ConstantInteger32 IceType; | 
| + static const Type Ty = IceType_i32; | 
| + static const char *TypeName; | 
| + static const char *AsmTag; | 
| + static const char *PrintfString; | 
| +}; | 
| +const char *PoolTypeConverter<int>::TypeName = "i32"; | 
| +const char *PoolTypeConverter<int>::AsmTag = ".long"; | 
| +const char *PoolTypeConverter<int>::PrintfString = "0x%x"; | 
| + | 
| template <typename T> | 
| void TargetDataX8632::emitConstantPool(GlobalContext *Ctx) { | 
| if (!ALLOW_DUMP) | 
| @@ -4979,6 +5077,12 @@ void TargetDataX8632::emitConstantPool(GlobalContext *Ctx) { | 
| Str << "\t.align\t" << Align << "\n"; | 
| for (Constant *C : Pool) { | 
| typename T::IceType *Const = llvm::cast<typename T::IceType>(C); | 
| + // When constant pooling turned on, only emit labels for eligible constants | 
| + // If C is i32 constant and not large enough, we should ignore it here. | 
| + if (llvm::isa<ConstantInteger32>(C) && | 
| + llvm::dyn_cast_or_null<ConstantInteger32>(C) | 
| 
Jim Stichnoth
2015/06/12 23:48:18
Same comments here as in IceELFObjectWriter.cpp
 
qining
2015/06/17 04:28:54
Done. shouldBePooled will be checked before poolin
 | 
| + ->shouldBeRandomizedOrPooled(Ctx) == false) | 
| + continue; | 
| typename T::IceType::PrimType Value = Const->getValue(); | 
| // Use memcpy() to copy bits from Value into RawValue in a way | 
| // that avoids breaking strict-aliasing rules. | 
| @@ -5004,12 +5108,22 @@ void TargetDataX8632::lowerConstants() const { | 
| switch (Ctx->getFlags().getOutFileType()) { | 
| case FT_Elf: { | 
| ELFObjectWriter *Writer = Ctx->getObjectWriter(); | 
| + | 
| + // If immediates pooling turned on, pool the integer constants | 
| + if (Ctx->getFlags().shouldPoolImmediates()) | 
| + Writer->writeConstantPool<ConstantInteger32>(IceType_i32); | 
| + | 
| Writer->writeConstantPool<ConstantFloat>(IceType_f32); | 
| Writer->writeConstantPool<ConstantDouble>(IceType_f64); | 
| } break; | 
| case FT_Asm: | 
| case FT_Iasm: { | 
| OstreamLocker L(Ctx); | 
| + | 
| + // If immediates pooling turned on, pool the integer constants | 
| + if (Ctx->getFlags().shouldPoolImmediates()) | 
| + emitConstantPool<PoolTypeConverter<int>>(Ctx); | 
| + | 
| emitConstantPool<PoolTypeConverter<float>>(Ctx); | 
| emitConstantPool<PoolTypeConverter<double>>(Ctx); | 
| } break; | 
| @@ -5019,4 +5133,186 @@ void TargetDataX8632::lowerConstants() const { | 
| TargetHeaderX8632::TargetHeaderX8632(GlobalContext *Ctx) | 
| : TargetHeaderLowering(Ctx) {} | 
| +uint32_t TargetX8632::getRandomizationCookie() { | 
| + static uint64_t cookie = 0; | 
| 
Jim Stichnoth
2015/06/12 23:48:18
Don't use static non-const variables, as they are
 
qining
2015/06/17 04:28:54
Done. Moved getRandomizationCookie() and the cooki
 | 
| + static bool cookie_initialized = false; | 
| + if (!cookie_initialized) { | 
| + RandomNumberGenerator &RNG = Ctx->getRNG(); | 
| + cookie = | 
| + (uint32_t)RNG.next((uint64_t)std::numeric_limits<uint32_t>::max() + 1); | 
| + cookie_initialized = true; | 
| + } | 
| + return cookie; | 
| +} | 
| + | 
| +// Blind/pool an immediate value | 
| +Operand *TargetX8632::randomizeOrPoolImmediate(Constant *immediate, | 
| 
Jim Stichnoth
2015/06/12 23:48:18
Capitalize "immediate" arg name.
 
qining
2015/06/17 04:28:53
Done
 | 
| + int32_t RegNum) { | 
| + assert(llvm::isa<ConstantInteger32>(immediate) || | 
| + llvm::isa<ConstantRelocatable>(immediate)); | 
| + if (Ctx->getFlags().shouldNotRandomizeOrPoolImmediates() == true || | 
| + RandomizationPoolingPaused == true) { | 
| + // immediates randomization/pool turned off | 
| + return immediate; | 
| + } | 
| + if (llvm::isa<ConstantInteger32>(immediate) && | 
| + llvm::dyn_cast<ConstantInteger32>(immediate) | 
| + ->shouldBeRandomizedOrPooled(Ctx)) { | 
| 
Jim Stichnoth
2015/06/12 23:48:18
Call immediate->shouldBeRandomizedOrPooled(Ctx) di
 
qining
2015/06/17 04:28:54
Done. Changed to:
if(Constant *C = dyn_cast_or_nul
 | 
| + Ctx->statsUpdateRPImms(); | 
| + if (Ctx->getFlags().shouldRandomizeImmediates() == true) { | 
| + // blind the constant | 
| + // FROM: | 
| + // imm | 
| + // TO: | 
| + // insert: mov imm+cookie, Reg | 
| + // insert: lea -cookie[Reg], Reg | 
| + // => Reg | 
| + // if we have already assigned a phy register, we must come from | 
| + // andvancedPhiLowering()=>lowerAssign(). In this case we should reuse | 
| 
Jim Stichnoth
2015/06/12 23:48:18
advancedPhiLowering
 
qining
2015/06/17 04:28:54
Done.
 | 
| + // the assigned register as this assignment is that start of its use-def | 
| + // chain. So we add RegNum argument here. | 
| + Variable *Reg = makeReg(immediate->getType(), RegNum); | 
| + ConstantInteger32 *integer = llvm::cast<ConstantInteger32>(immediate); | 
| 
Jim Stichnoth
2015/06/12 23:48:18
Capitalize variable names
 
qining
2015/06/17 04:28:53
Done.
 | 
| + uint32_t value = integer->getValue(); | 
| + uint64_t cookie = getRandomizationCookie(); | 
| 
Jim Stichnoth
2015/06/12 23:48:18
uint32_t
 
qining
2015/06/17 04:28:54
Done.
 | 
| + _mov(Reg, Ctx->getConstantInt(IceType_i32, cookie + value)); | 
| + Constant *offset = Ctx->getConstantInt(IceType_i32, 0 - cookie); | 
| + _lea(Reg, OperandX8632Mem::create(Func, offset->getType(), Reg, offset, | 
| + NULL, 0)); | 
| + // make sure liveness analysis won't kill this variable, otherwise a | 
| + // liveness | 
| + // assertion will be triggered. | 
| + _set_dest_nonkillable(); | 
| + return Reg; | 
| + } else { | 
| 
Jim Stichnoth
2015/06/12 23:48:18
Don't use this pattern:
  if (abc) {
    def;
 
qining
2015/06/17 04:28:54
Done. Changed to:
if (abc) {
   return xyz;
}
if
 | 
| + // pool the constant | 
| + // FROM: | 
| + // imm | 
| + // TO: | 
| + // insert: mov $label, Reg | 
| + // => Reg | 
| + assert(Ctx->getFlags().shouldPoolImmediates() == true); | 
| + // if we have already assigned a phy register, we must come from | 
| + // andvancedPhiLowering()=>lowerAssign(). In this case we should reuse | 
| 
Jim Stichnoth
2015/06/12 23:48:17
advancedPhiLowering
 
qining
2015/06/17 04:28:53
Done.
 | 
| + // the assigned register as this assignment is that start of its use-def | 
| + // chain. So we add RegNum argument here. | 
| + Variable *Reg = makeReg(immediate->getType(), RegNum); | 
| + IceString label; | 
| + llvm::raw_string_ostream label_stream(label); | 
| + immediate->emitPoolLabel(label_stream); | 
| + Constant *symbol = Ctx->getConstantSym(0, label_stream.str(), true); | 
| 
Jim Stichnoth
2015/06/12 23:48:17
Try to avoid using non-obvious constants as argume
 
qining
2015/06/17 04:28:54
Done.
 | 
| + OperandX8632Mem *MemOperand = | 
| + OperandX8632Mem::create(Func, immediate->getType(), NULL, symbol); | 
| + _mov(Reg, MemOperand); | 
| + return Reg; | 
| + } | 
| + } else { | 
| + // the constant immediate is not eligible for blinding/pooling | 
| + return immediate; | 
| + } | 
| +} | 
| + | 
| +OperandX8632Mem * | 
| +TargetX8632::randomizeOrPoolImmediate(OperandX8632Mem *MemOperand, | 
| + int32_t RegNum) { | 
| + assert(llvm::isa<OperandX8632Mem>(MemOperand)); | 
| 
Jim Stichnoth
2015/06/12 23:48:18
This assert isn't very useful and should always su
 
qining
2015/06/17 04:28:53
Done. Changed to assert(MemOperand).
 | 
| + if (Ctx->getFlags().shouldNotRandomizeOrPoolImmediates() == true || | 
| + RandomizationPoolingPaused == true) { | 
| + // immediates randomization/pooling is turned off | 
| + return MemOperand; | 
| + } | 
| + | 
| + if (MemOperand->getOffset() && | 
| 
Jim Stichnoth
2015/06/12 23:48:18
Something like this might be more compact:
if (au
 
qining
2015/06/17 04:28:53
Done.
 | 
| + llvm::isa<ConstantInteger32>(MemOperand->getOffset()) && | 
| + llvm::dyn_cast<ConstantInteger32>(MemOperand->getOffset()) | 
| + ->shouldBeRandomizedOrPooled(Ctx)) { | 
| + // The offset of this mem operand should be blinded or pooled | 
| + Ctx->statsUpdateRPImms(); | 
| + if (Ctx->getFlags().shouldRandomizeImmediates() == true) { | 
| + // blind the constant offset | 
| + // FROM: | 
| + // offset[base, index, shift] | 
| + // TO: | 
| + // insert: lea offset+cookie[base], RegTemp | 
| + // => -cookie[RegTemp, index, shift] | 
| + uint32_t value = | 
| + llvm::dyn_cast<ConstantInteger32>(MemOperand->getOffset()) | 
| + ->getValue(); | 
| + uint64_t cookie = getRandomizationCookie(); | 
| + Constant *mask1 = Ctx->getConstantInt(IceType_i32, cookie + value); | 
| + Constant *mask2 = Ctx->getConstantInt(IceType_i32, 0 - cookie); | 
| + // qining: We need to make sure the MemOperand->getBase() has a physical | 
| + // register, if it is a variable! | 
| + if (MemOperand->getBase() != NULL) | 
| 
Jim Stichnoth
2015/06/12 23:48:18
nullptr, or just leave out the != part
 
qining
2015/06/17 04:28:54
Done.
 | 
| + MemOperand->getBase()->setWeightInfinite(); | 
| 
Jim Stichnoth
2015/06/12 23:48:17
This will get you into trouble.  The base variable
 | 
| + OperandX8632Mem *TempMemOperand = OperandX8632Mem::create( | 
| + Func, MemOperand->getType(), MemOperand->getBase(), mask1); | 
| + // if we have already assigned a phy register, we must come from | 
| + // andvancedPhiLowering()=>lowerAssign(). In this case we should reuse | 
| + // the assigned register as this assignment is that start of its use-def | 
| + // chain. So we add RegNum argument here. | 
| + | 
| + Variable *RegTemp = makeReg(MemOperand->getOffset()->getType(), RegNum); | 
| + _lea(RegTemp, TempMemOperand); | 
| + // as srcreg doesn't include dstreg, we don't need to add | 
| + // _set_dest_nonkillable() | 
| + // qining: but if we use the same Dest Reg, that is, with RegNum | 
| + // assigned, we should add this _set_dest_nonkillable() | 
| + if (RegNum != Variable::NoRegister) | 
| + _set_dest_nonkillable(); | 
| + | 
| + OperandX8632Mem *NewMemOperand = OperandX8632Mem::create( | 
| + Func, MemOperand->getType(), RegTemp, mask2, MemOperand->getIndex(), | 
| + MemOperand->getShift(), MemOperand->getSegmentRegister()); | 
| + | 
| + return NewMemOperand; | 
| + | 
| + } else { | 
| + // pool the constant offset | 
| + // FROM: | 
| + // offset[base, index, shift] | 
| + // TO: | 
| + // insert: mov $label, RegTemp | 
| + // insert: lea [base, RegTemp], RegTemp | 
| + // =>[RegTemp, index, shift] | 
| + assert(Ctx->getFlags().shouldPoolImmediates() == true); | 
| + // qining: Mem operand should never exist as source operands in phi | 
| + // lowering | 
| + // assignments, so there is no need to reuse any registers here. | 
| + // However, for phi lowering, we should not ask for new physical | 
| + // registers in general. | 
| + // However, if we do meet MemOperand during phi lowering, we should not | 
| + // blind or pool the immediates for now | 
| + if (RegNum != Variable::NoRegister) | 
| + return MemOperand; | 
| + Variable *RegTemp = makeReg(MemOperand->getOffset()->getType()); | 
| + IceString label; | 
| + llvm::raw_string_ostream label_stream(label); | 
| + MemOperand->getOffset()->emitPoolLabel(label_stream); | 
| + Constant *symbol = Ctx->getConstantSym(0, label_stream.str(), true); | 
| + OperandX8632Mem *SymbolOperand = OperandX8632Mem::create( | 
| + Func, MemOperand->getOffset()->getType(), NULL, symbol); | 
| + _mov(RegTemp, SymbolOperand); | 
| + // qining: We need to make sure the MemOperand->getBase() has a physical | 
| + // register! If we do not have base register here, we won't need an | 
| + // extra lea instruction anymore. | 
| + if (MemOperand->getBase() != NULL) { | 
| + MemOperand->getBase()->setWeightInfinite(); | 
| + OperandX8632Mem *CalculateOperand = OperandX8632Mem::create( | 
| + Func, MemOperand->getType(), MemOperand->getBase(), NULL, RegTemp, | 
| + 0, MemOperand->getSegmentRegister()); | 
| + _lea(RegTemp, CalculateOperand); | 
| + _set_dest_nonkillable(); | 
| + } | 
| + OperandX8632Mem *NewMemOperand = OperandX8632Mem::create( | 
| + Func, MemOperand->getType(), RegTemp, NULL, MemOperand->getIndex(), | 
| + MemOperand->getShift(), MemOperand->getSegmentRegister()); | 
| + return NewMemOperand; | 
| + } | 
| + } else { | 
| + // the offset is not eligible for blinding or pooling, return the original | 
| + // mem operand | 
| + return MemOperand; | 
| + } | 
| +} | 
| } // end of namespace Ice |