Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(7)

Unified Diff: src/IceTargetLoweringX8632.cpp

Issue 1185703004: Add constant blinding/pooling option for X8632 code translation (Closed) Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master
Patch Set: reformat Created 5 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
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

Powered by Google App Engine
This is Rietveld 408576698