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

Side by Side Diff: src/IceTargetLoweringX8632.cpp

Issue 586943003: Subzero: Change the way bitcast stack slot lowering is handled. (Closed) Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master
Patch Set: Created 6 years, 3 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 unified diff | Download patch
« src/IceOperand.cpp ('K') | « src/IceRegAlloc.cpp ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 //===- subzero/src/IceTargetLoweringX8632.cpp - x86-32 lowering -----------===// 1 //===- subzero/src/IceTargetLoweringX8632.cpp - x86-32 lowering -----------===//
2 // 2 //
3 // The Subzero Code Generator 3 // The Subzero Code Generator
4 // 4 //
5 // This file is distributed under the University of Illinois Open Source 5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details. 6 // License. See LICENSE.TXT for details.
7 // 7 //
8 //===----------------------------------------------------------------------===// 8 //===----------------------------------------------------------------------===//
9 // 9 //
10 // This file implements the TargetLoweringX8632 class, which 10 // This file implements the TargetLoweringX8632 class, which
(...skipping 683 matching lines...) Expand 10 before | Expand all | Expand 10 after
694 getRegisterSet(RegSet_CalleeSave, RegSet_None); 694 getRegisterSet(RegSet_CalleeSave, RegSet_None);
695 695
696 size_t GlobalsSize = 0; 696 size_t GlobalsSize = 0;
697 std::vector<size_t> LocalsSize(Func->getNumNodes()); 697 std::vector<size_t> LocalsSize(Func->getNumNodes());
698 698
699 // Prepass. Compute RegsUsed, PreservedRegsSizeBytes, and 699 // Prepass. Compute RegsUsed, PreservedRegsSizeBytes, and
700 // SpillAreaSizeBytes. 700 // SpillAreaSizeBytes.
701 RegsUsed = llvm::SmallBitVector(CalleeSaves.size()); 701 RegsUsed = llvm::SmallBitVector(CalleeSaves.size());
702 const VarList &Variables = Func->getVariables(); 702 const VarList &Variables = Func->getVariables();
703 const VarList &Args = Func->getArgs(); 703 const VarList &Args = Func->getArgs();
704 VarList SpilledVariables, SortedSpilledVariables, 704 VarList SpilledVariables, SortedSpilledVariables, VariablesLinkedToSpillSlots;
705 VariablesLinkedToSpillSplots;
706 705
707 // If there is a separate locals area, this specifies the alignment 706 // If there is a separate locals area, this specifies the alignment
708 // for it. 707 // for it.
709 uint32_t LocalsSlotsAlignmentBytes = 0; 708 uint32_t LocalsSlotsAlignmentBytes = 0;
710 // The entire spill locations area gets aligned to largest natural 709 // The entire spill locations area gets aligned to largest natural
711 // alignment of the variables that have a spill slot. 710 // alignment of the variables that have a spill slot.
712 uint32_t SpillAreaAlignmentBytes = 0; 711 uint32_t SpillAreaAlignmentBytes = 0;
713 for (VarList::const_iterator I = Variables.begin(), E = Variables.end(); 712 for (VarList::const_iterator I = Variables.begin(), E = Variables.end();
714 I != E; ++I) { 713 I != E; ++I) {
715 Variable *Var = *I; 714 Variable *Var = *I;
716 if (Var->hasReg()) { 715 if (Var->hasReg()) {
717 RegsUsed[Var->getRegNum()] = true; 716 RegsUsed[Var->getRegNum()] = true;
718 continue; 717 continue;
719 } 718 }
720 // An argument either does not need a stack slot (if passed in a 719 // An argument either does not need a stack slot (if passed in a
721 // register) or already has one (if passed on the stack). 720 // register) or already has one (if passed on the stack).
722 if (Var->getIsArg()) 721 if (Var->getIsArg())
723 continue; 722 continue;
724 // An unreferenced variable doesn't need a stack slot. 723 // An unreferenced variable doesn't need a stack slot.
725 if (ComputedLiveRanges && Var->getLiveRange().isEmpty()) 724 if (ComputedLiveRanges && Var->getLiveRange().isEmpty())
726 continue; 725 continue;
727 // A spill slot linked to a variable with a stack slot should reuse 726 // A spill slot linked to a variable with a stack slot should reuse
728 // that stack slot. 727 // that stack slot.
729 if (Var->getWeight() == RegWeight::Zero && Var->getRegisterOverlap()) { 728 if (SpillVariable *SpillVar = llvm::dyn_cast<SpillVariable>(Var)) {
730 if (Variable *Linked = Var->getPreferredRegister()) { 729 assert(Var->getWeight() == RegWeight::Zero);
731 if (!Linked->hasReg()) { 730 if (!SpillVar->getLinkedTo()->hasReg()) {
732 VariablesLinkedToSpillSplots.push_back(Var); 731 VariablesLinkedToSpillSlots.push_back(Var);
733 continue; 732 continue;
734 }
735 } 733 }
736 } 734 }
737 SpilledVariables.push_back(Var); 735 SpilledVariables.push_back(Var);
738 } 736 }
739 737
740 SortedSpilledVariables.reserve(SpilledVariables.size()); 738 SortedSpilledVariables.reserve(SpilledVariables.size());
741 sortByAlignment(SortedSpilledVariables, SpilledVariables); 739 sortByAlignment(SortedSpilledVariables, SpilledVariables);
742 for (VarList::const_iterator I = SortedSpilledVariables.begin(), 740 for (VarList::const_iterator I = SortedSpilledVariables.begin(),
743 E = SortedSpilledVariables.end(); 741 E = SortedSpilledVariables.end();
744 I != E; ++I) { 742 I != E; ++I) {
(...skipping 127 matching lines...) Expand 10 before | Expand all | Expand 10 after
872 if (IsEbpBasedFrame) 870 if (IsEbpBasedFrame)
873 Var->setStackOffset(-NextStackOffset); 871 Var->setStackOffset(-NextStackOffset);
874 else 872 else
875 Var->setStackOffset(SpillAreaSizeBytes - NextStackOffset); 873 Var->setStackOffset(SpillAreaSizeBytes - NextStackOffset);
876 } 874 }
877 this->FrameSizeLocals = NextStackOffset - SpillAreaPaddingBytes; 875 this->FrameSizeLocals = NextStackOffset - SpillAreaPaddingBytes;
878 this->HasComputedFrame = true; 876 this->HasComputedFrame = true;
879 877
880 // Assign stack offsets to variables that have been linked to spilled 878 // Assign stack offsets to variables that have been linked to spilled
881 // variables. 879 // variables.
882 for (VarList::const_iterator I = VariablesLinkedToSpillSplots.begin(), 880 for (VarList::const_iterator I = VariablesLinkedToSpillSlots.begin(),
883 E = VariablesLinkedToSpillSplots.end(); 881 E = VariablesLinkedToSpillSlots.end();
884 I != E; ++I) { 882 I != E; ++I) {
885 Variable *Var = *I; 883 Variable *Var = *I;
886 Variable *Linked = Var->getPreferredRegister(); 884 Variable *Linked = (llvm::cast<SpillVariable>(Var))->getLinkedTo();
887 Var->setStackOffset(Linked->getStackOffset()); 885 Var->setStackOffset(Linked->getStackOffset());
888 } 886 }
889 887
890 if (Func->getContext()->isVerbose(IceV_Frame)) { 888 if (Func->getContext()->isVerbose(IceV_Frame)) {
891 Ostream &Str = Func->getContext()->getStrDump(); 889 Ostream &Str = Func->getContext()->getStrDump();
892 890
893 Str << "Stack layout:\n"; 891 Str << "Stack layout:\n";
894 uint32_t EspAdjustmentPaddingSize = 892 uint32_t EspAdjustmentPaddingSize =
895 SpillAreaSizeBytes - LocalsSpillAreaSize - 893 SpillAreaSizeBytes - LocalsSpillAreaSize -
896 GlobalsAndSubsequentPaddingSize - SpillAreaPaddingBytes; 894 GlobalsAndSubsequentPaddingSize - SpillAreaPaddingBytes;
(...skipping 1375 matching lines...) Expand 10 before | Expand all | Expand 10 after
2272 (void)DestType; 2270 (void)DestType;
2273 assert((DestType == IceType_i32 && SrcType == IceType_f32) || 2271 assert((DestType == IceType_i32 && SrcType == IceType_f32) ||
2274 (DestType == IceType_f32 && SrcType == IceType_i32)); 2272 (DestType == IceType_f32 && SrcType == IceType_i32));
2275 // a.i32 = bitcast b.f32 ==> 2273 // a.i32 = bitcast b.f32 ==>
2276 // t.f32 = b.f32 2274 // t.f32 = b.f32
2277 // s.f32 = spill t.f32 2275 // s.f32 = spill t.f32
2278 // a.i32 = s.f32 2276 // a.i32 = s.f32
2279 Variable *T = NULL; 2277 Variable *T = NULL;
2280 // TODO: Should be able to force a spill setup by calling legalize() with 2278 // TODO: Should be able to force a spill setup by calling legalize() with
2281 // Legal_Mem and not Legal_Reg or Legal_Imm. 2279 // Legal_Mem and not Legal_Reg or Legal_Imm.
2282 Variable *Spill = Func->makeVariable(SrcType, Context.getNode()); 2280 SpillVariable *SpillVar =
2281 Func->makeVariable<SpillVariable>(SrcType, Context.getNode());
2282 SpillVar->setLinkedTo(Dest);
2283 Variable *Spill = SpillVar;
2283 Spill->setWeight(RegWeight::Zero); 2284 Spill->setWeight(RegWeight::Zero);
2284 Spill->setPreferredRegister(Dest, true);
2285 _mov(T, Src0RM); 2285 _mov(T, Src0RM);
2286 _mov(Spill, T); 2286 _mov(Spill, T);
2287 _mov(Dest, Spill); 2287 _mov(Dest, Spill);
2288 } break; 2288 } break;
2289 case IceType_i64: { 2289 case IceType_i64: {
2290 Operand *Src0RM = legalize(Src0, Legal_Reg | Legal_Mem); 2290 Operand *Src0RM = legalize(Src0, Legal_Reg | Legal_Mem);
2291 assert(Src0RM->getType() == IceType_f64); 2291 assert(Src0RM->getType() == IceType_f64);
2292 // a.i64 = bitcast b.f64 ==> 2292 // a.i64 = bitcast b.f64 ==>
2293 // s.f64 = spill b.f64 2293 // s.f64 = spill b.f64
2294 // t_lo.i32 = lo(s.f64) 2294 // t_lo.i32 = lo(s.f64)
2295 // a_lo.i32 = t_lo.i32 2295 // a_lo.i32 = t_lo.i32
2296 // t_hi.i32 = hi(s.f64) 2296 // t_hi.i32 = hi(s.f64)
2297 // a_hi.i32 = t_hi.i32 2297 // a_hi.i32 = t_hi.i32
2298 Variable *Spill = Func->makeVariable(IceType_f64, Context.getNode()); 2298 SpillVariable *SpillVar =
2299 Func->makeVariable<SpillVariable>(IceType_f64, Context.getNode());
2300 SpillVar->setLinkedTo(llvm::dyn_cast<Variable>(Src0RM));
2301 Variable *Spill = SpillVar;
2299 Spill->setWeight(RegWeight::Zero); 2302 Spill->setWeight(RegWeight::Zero);
2300 Spill->setPreferredRegister(llvm::dyn_cast<Variable>(Src0RM), true);
2301 _movq(Spill, Src0RM); 2303 _movq(Spill, Src0RM);
2302 2304
2303 Variable *DestLo = llvm::cast<Variable>(loOperand(Dest)); 2305 Variable *DestLo = llvm::cast<Variable>(loOperand(Dest));
2304 Variable *DestHi = llvm::cast<Variable>(hiOperand(Dest)); 2306 Variable *DestHi = llvm::cast<Variable>(hiOperand(Dest));
2305 Variable *T_Lo = makeReg(IceType_i32); 2307 Variable *T_Lo = makeReg(IceType_i32);
2306 Variable *T_Hi = makeReg(IceType_i32); 2308 Variable *T_Hi = makeReg(IceType_i32);
2307 VariableSplit *SpillLo = 2309 VariableSplit *SpillLo =
2308 VariableSplit::create(Func, Spill, VariableSplit::Low); 2310 VariableSplit::create(Func, Spill, VariableSplit::Low);
2309 VariableSplit *SpillHi = 2311 VariableSplit *SpillHi =
2310 VariableSplit::create(Func, Spill, VariableSplit::High); 2312 VariableSplit::create(Func, Spill, VariableSplit::High);
2311 2313
2312 _mov(T_Lo, SpillLo); 2314 _mov(T_Lo, SpillLo);
2313 _mov(DestLo, T_Lo); 2315 _mov(DestLo, T_Lo);
2314 _mov(T_Hi, SpillHi); 2316 _mov(T_Hi, SpillHi);
2315 _mov(DestHi, T_Hi); 2317 _mov(DestHi, T_Hi);
2316 } break; 2318 } break;
2317 case IceType_f64: { 2319 case IceType_f64: {
2318 Src0 = legalize(Src0); 2320 Src0 = legalize(Src0);
2319 assert(Src0->getType() == IceType_i64); 2321 assert(Src0->getType() == IceType_i64);
2320 // a.f64 = bitcast b.i64 ==> 2322 // a.f64 = bitcast b.i64 ==>
2321 // t_lo.i32 = b_lo.i32 2323 // t_lo.i32 = b_lo.i32
2322 // FakeDef(s.f64) 2324 // FakeDef(s.f64)
2323 // lo(s.f64) = t_lo.i32 2325 // lo(s.f64) = t_lo.i32
2324 // t_hi.i32 = b_hi.i32 2326 // t_hi.i32 = b_hi.i32
2325 // hi(s.f64) = t_hi.i32 2327 // hi(s.f64) = t_hi.i32
2326 // a.f64 = s.f64 2328 // a.f64 = s.f64
2327 Variable *Spill = Func->makeVariable(IceType_f64, Context.getNode()); 2329 SpillVariable *SpillVar =
2330 Func->makeVariable<SpillVariable>(IceType_f64, Context.getNode());
2331 SpillVar->setLinkedTo(Dest);
2332 Variable *Spill = SpillVar;
2328 Spill->setWeight(RegWeight::Zero); 2333 Spill->setWeight(RegWeight::Zero);
2329 Spill->setPreferredRegister(Dest, true);
2330 2334
2331 Variable *T_Lo = NULL, *T_Hi = NULL; 2335 Variable *T_Lo = NULL, *T_Hi = NULL;
2332 VariableSplit *SpillLo = 2336 VariableSplit *SpillLo =
2333 VariableSplit::create(Func, Spill, VariableSplit::Low); 2337 VariableSplit::create(Func, Spill, VariableSplit::Low);
2334 VariableSplit *SpillHi = 2338 VariableSplit *SpillHi =
2335 VariableSplit::create(Func, Spill, VariableSplit::High); 2339 VariableSplit::create(Func, Spill, VariableSplit::High);
2336 _mov(T_Lo, loOperand(Src0)); 2340 _mov(T_Lo, loOperand(Src0));
2337 // Technically, the Spill is defined after the _store happens, but 2341 // Technically, the Spill is defined after the _store happens, but
2338 // SpillLo is considered a "use" of Spill so define Spill before it 2342 // SpillLo is considered a "use" of Spill so define Spill before it
2339 // is used. 2343 // is used.
(...skipping 1374 matching lines...) Expand 10 before | Expand all | Expand 10 after
3714 Base = Var; 3718 Base = Var;
3715 Offset += MoreOffset; 3719 Offset += MoreOffset;
3716 Reason = BaseInst; 3720 Reason = BaseInst;
3717 return true; 3721 return true;
3718 } 3722 }
3719 return false; 3723 return false;
3720 } 3724 }
3721 3725
3722 void computeAddressOpt(Cfg *Func, const Inst *Instr, Variable *&Base, 3726 void computeAddressOpt(Cfg *Func, const Inst *Instr, Variable *&Base,
3723 Variable *&Index, uint16_t &Shift, int32_t &Offset) { 3727 Variable *&Index, uint16_t &Shift, int32_t &Offset) {
3724 Func->setCurrentNode(NULL); 3728 Func->resetCurrentNode();
3725 if (Func->getContext()->isVerbose(IceV_AddrOpt)) { 3729 if (Func->getContext()->isVerbose(IceV_AddrOpt)) {
3726 Ostream &Str = Func->getContext()->getStrDump(); 3730 Ostream &Str = Func->getContext()->getStrDump();
3727 Str << "\nStarting computeAddressOpt for instruction:\n "; 3731 Str << "\nStarting computeAddressOpt for instruction:\n ";
3728 Instr->dumpDecorated(Func); 3732 Instr->dumpDecorated(Func);
3729 } 3733 }
3730 (void)Offset; // TODO: pattern-match for non-zero offsets. 3734 (void)Offset; // TODO: pattern-match for non-zero offsets.
3731 if (Base == NULL) 3735 if (Base == NULL)
3732 return; 3736 return;
3733 // If the Base has more than one use or is live across multiple 3737 // If the Base has more than one use or is live across multiple
3734 // blocks, then don't go further. Alternatively (?), never consider 3738 // blocks, then don't go further. Alternatively (?), never consider
(...skipping 814 matching lines...) Expand 10 before | Expand all | Expand 10 after
4549 Str << "\t.align\t" << Align << "\n"; 4553 Str << "\t.align\t" << Align << "\n";
4550 Str << MangledName << ":\n"; 4554 Str << MangledName << ":\n";
4551 for (SizeT i = 0; i < Size; ++i) { 4555 for (SizeT i = 0; i < Size; ++i) {
4552 Str << "\t.byte\t" << (((unsigned)Data[i]) & 0xff) << "\n"; 4556 Str << "\t.byte\t" << (((unsigned)Data[i]) & 0xff) << "\n";
4553 } 4557 }
4554 Str << "\t.size\t" << MangledName << ", " << Size << "\n"; 4558 Str << "\t.size\t" << MangledName << ", " << Size << "\n";
4555 } 4559 }
4556 } 4560 }
4557 4561
4558 } // end of namespace Ice 4562 } // end of namespace Ice
OLDNEW
« src/IceOperand.cpp ('K') | « src/IceRegAlloc.cpp ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698