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

Side by Side Diff: src/IceConverter.cpp

Issue 395193005: Start processing function blocks in Subzero. (Closed) Base URL: https://chromium.googlesource.com/native_client/pnacl-subzero.git@master
Patch Set: Fix nits in patch set 4. Created 6 years, 5 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
OLDNEW
1 //===- subzero/src/IceConverter.cpp - Converts LLVM to Ice ---------------===// 1 //===- subzero/src/IceConverter.cpp - Converts LLVM to Ice ---------------===//
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 LLVM to ICE converter. 10 // This file implements the LLVM to ICE converter.
11 // 11 //
12 //===----------------------------------------------------------------------===// 12 //===----------------------------------------------------------------------===//
13 13
14 #include "IceConverter.h" 14 #include "IceConverter.h"
15 15
16 #include "IceCfg.h" 16 #include "IceCfg.h"
17 #include "IceCfgNode.h" 17 #include "IceCfgNode.h"
18 #include "IceClFlags.h" 18 #include "IceClFlags.h"
19 #include "IceDefs.h" 19 #include "IceDefs.h"
20 #include "IceGlobalContext.h" 20 #include "IceGlobalContext.h"
21 #include "IceInst.h" 21 #include "IceInst.h"
22 #include "IceOperand.h" 22 #include "IceOperand.h"
23 #include "IceTargetLowering.h" 23 #include "IceTargetLowering.h"
24 #include "IceTypes.h" 24 #include "IceTypes.h"
25 #include "IceTypeConverter.h"
25 26
26 #include "llvm/IR/Constant.h" 27 #include "llvm/IR/Constant.h"
27 #include "llvm/IR/Constants.h" 28 #include "llvm/IR/Constants.h"
28 #include "llvm/IR/DataLayout.h" 29 #include "llvm/IR/DataLayout.h"
29 #include "llvm/IR/Instruction.h" 30 #include "llvm/IR/Instruction.h"
30 #include "llvm/IR/Instructions.h" 31 #include "llvm/IR/Instructions.h"
31 #include "llvm/IR/LLVMContext.h" 32 #include "llvm/IR/LLVMContext.h"
32 #include "llvm/IR/Module.h" 33 #include "llvm/IR/Module.h"
33 34
34 #include <iostream> 35 #include <iostream>
(...skipping 12 matching lines...) Expand all
47 48
48 // Converter from LLVM to ICE. The entry point is the convertFunction method. 49 // Converter from LLVM to ICE. The entry point is the convertFunction method.
49 // 50 //
50 // Note: this currently assumes that the given IR was verified to be valid PNaCl 51 // Note: this currently assumes that the given IR was verified to be valid PNaCl
51 // bitcode: 52 // bitcode:
52 // https://developers.google.com/native-client/dev/reference/pnacl-bitcode-abi 53 // https://developers.google.com/native-client/dev/reference/pnacl-bitcode-abi
53 // If not, all kinds of assertions may fire. 54 // If not, all kinds of assertions may fire.
54 // 55 //
55 class LLVM2ICEConverter { 56 class LLVM2ICEConverter {
56 public: 57 public:
57 LLVM2ICEConverter(Ice::GlobalContext *Ctx) 58 LLVM2ICEConverter(Ice::GlobalContext *Ctx, LLVMContext& LLVMContext)
58 : Ctx(Ctx), Func(NULL), CurrentNode(NULL) { 59 : Ctx(Ctx), Func(NULL), CurrentNode(NULL), TypeConverter(LLVMContext) {
59 // All PNaCl pointer widths are 32 bits because of the sandbox 60 // All PNaCl pointer widths are 32 bits because of the sandbox
60 // model. 61 // model.
jvoung (off chromium) 2014/07/22 18:40:18 leftover comment
Karl 2014/07/23 20:22:20 Done.
61 SubzeroPointerType = Ice::IceType_i32;
62 } 62 }
63 63
64 // Caller is expected to delete the returned Ice::Cfg object. 64 // Caller is expected to delete the returned Ice::Cfg object.
65 Ice::Cfg *convertFunction(const Function *F) { 65 Ice::Cfg *convertFunction(const Function *F) {
66 VarMap.clear(); 66 VarMap.clear();
67 NodeMap.clear(); 67 NodeMap.clear();
68 Func = new Ice::Cfg(Ctx); 68 Func = new Ice::Cfg(Ctx);
69 Func->setFunctionName(F->getName()); 69 Func->setFunctionName(F->getName());
70 Func->setReturnType(convertType(F->getReturnType())); 70 Func->setReturnType(convertToIceType(F->getReturnType()));
71 Func->setInternal(F->hasInternalLinkage()); 71 Func->setInternal(F->hasInternalLinkage());
72 72
73 // The initial definition/use of each arg is the entry node. 73 // The initial definition/use of each arg is the entry node.
74 CurrentNode = mapBasicBlockToNode(&F->getEntryBlock()); 74 CurrentNode = mapBasicBlockToNode(&F->getEntryBlock());
75 for (Function::const_arg_iterator ArgI = F->arg_begin(), 75 for (Function::const_arg_iterator ArgI = F->arg_begin(),
76 ArgE = F->arg_end(); 76 ArgE = F->arg_end();
77 ArgI != ArgE; ++ArgI) { 77 ArgI != ArgE; ++ArgI) {
78 Func->addArg(mapValueToIceVar(ArgI)); 78 Func->addArg(mapValueToIceVar(ArgI));
79 } 79 }
80 80
(...skipping 14 matching lines...) Expand all
95 Func->computePredecessors(); 95 Func->computePredecessors();
96 96
97 return Func; 97 return Func;
98 } 98 }
99 99
100 // convertConstant() does not use Func or require it to be a valid 100 // convertConstant() does not use Func or require it to be a valid
101 // Ice::Cfg pointer. As such, it's suitable for e.g. constructing 101 // Ice::Cfg pointer. As such, it's suitable for e.g. constructing
102 // global initializers. 102 // global initializers.
103 Ice::Constant *convertConstant(const Constant *Const) { 103 Ice::Constant *convertConstant(const Constant *Const) {
104 if (const GlobalValue *GV = dyn_cast<GlobalValue>(Const)) { 104 if (const GlobalValue *GV = dyn_cast<GlobalValue>(Const)) {
105 return Ctx->getConstantSym(convertType(GV->getType()), 0, GV->getName()); 105 return Ctx->getConstantSym(convertToIceType(GV->getType()),
106 0, GV->getName());
106 } else if (const ConstantInt *CI = dyn_cast<ConstantInt>(Const)) { 107 } else if (const ConstantInt *CI = dyn_cast<ConstantInt>(Const)) {
107 return Ctx->getConstantInt(convertIntegerType(CI->getType()), 108 return Ctx->getConstantInt(convertToIceType(CI->getType()),
108 CI->getZExtValue()); 109 CI->getZExtValue());
109 } else if (const ConstantFP *CFP = dyn_cast<ConstantFP>(Const)) { 110 } else if (const ConstantFP *CFP = dyn_cast<ConstantFP>(Const)) {
110 Ice::Type Type = convertType(CFP->getType()); 111 Ice::Type Type = convertToIceType(CFP->getType());
111 if (Type == Ice::IceType_f32) 112 if (Type == Ice::IceType_f32)
112 return Ctx->getConstantFloat(CFP->getValueAPF().convertToFloat()); 113 return Ctx->getConstantFloat(CFP->getValueAPF().convertToFloat());
113 else if (Type == Ice::IceType_f64) 114 else if (Type == Ice::IceType_f64)
114 return Ctx->getConstantDouble(CFP->getValueAPF().convertToDouble()); 115 return Ctx->getConstantDouble(CFP->getValueAPF().convertToDouble());
115 llvm_unreachable("Unexpected floating point type"); 116 llvm_unreachable("Unexpected floating point type");
116 return NULL; 117 return NULL;
117 } else if (const UndefValue *CU = dyn_cast<UndefValue>(Const)) { 118 } else if (const UndefValue *CU = dyn_cast<UndefValue>(Const)) {
118 return Ctx->getConstantUndef(convertType(CU->getType())); 119 return Ctx->getConstantUndef(convertToIceType(CU->getType()));
119 } else { 120 } else {
120 llvm_unreachable("Unhandled constant type"); 121 llvm_unreachable("Unhandled constant type");
121 return NULL; 122 return NULL;
122 } 123 }
123 } 124 }
124 125
125 private: 126 private:
126 // LLVM values (instructions, etc.) are mapped directly to ICE variables. 127 // LLVM values (instructions, etc.) are mapped directly to ICE variables.
127 // mapValueToIceVar has a version that forces an ICE type on the variable, 128 // mapValueToIceVar has a version that forces an ICE type on the variable,
128 // and a version that just uses convertType on V. 129 // and a version that just uses convertToIceType on V.
129 Ice::Variable *mapValueToIceVar(const Value *V, Ice::Type IceTy) { 130 Ice::Variable *mapValueToIceVar(const Value *V, Ice::Type IceTy) {
130 if (IceTy == Ice::IceType_void) 131 if (IceTy == Ice::IceType_void)
131 return NULL; 132 return NULL;
132 if (VarMap.find(V) == VarMap.end()) { 133 if (VarMap.find(V) == VarMap.end()) {
133 assert(CurrentNode); 134 assert(CurrentNode);
134 VarMap[V] = Func->makeVariable(IceTy, CurrentNode, V->getName()); 135 VarMap[V] = Func->makeVariable(IceTy, CurrentNode, V->getName());
135 } 136 }
136 return VarMap[V]; 137 return VarMap[V];
137 } 138 }
138 139
139 Ice::Variable *mapValueToIceVar(const Value *V) { 140 Ice::Variable *mapValueToIceVar(const Value *V) {
140 return mapValueToIceVar(V, convertType(V->getType())); 141 return mapValueToIceVar(V, convertToIceType(V->getType()));
141 } 142 }
142 143
143 Ice::CfgNode *mapBasicBlockToNode(const BasicBlock *BB) { 144 Ice::CfgNode *mapBasicBlockToNode(const BasicBlock *BB) {
144 if (NodeMap.find(BB) == NodeMap.end()) { 145 if (NodeMap.find(BB) == NodeMap.end()) {
145 NodeMap[BB] = Func->makeNode(BB->getName()); 146 NodeMap[BB] = Func->makeNode(BB->getName());
146 } 147 }
147 return NodeMap[BB]; 148 return NodeMap[BB];
148 } 149 }
149 150
150 Ice::Type convertIntegerType(const IntegerType *IntTy) const { 151 Ice::Type convertToIceType(Type *LLVMTy) const {
151 switch (IntTy->getBitWidth()) { 152 Ice::Type IceTy = TypeConverter.convertToIceType(LLVMTy);
152 case 1: 153 if (IceTy == Ice::IceType_NUM)
153 return Ice::IceType_i1; 154 llvm::report_fatal_error(std::string("Invalid PNaCl type ") +
154 case 8: 155 LLVMObjectAsString(LLVMTy));
155 return Ice::IceType_i8; 156 return IceTy;
156 case 16:
157 return Ice::IceType_i16;
158 case 32:
159 return Ice::IceType_i32;
160 case 64:
161 return Ice::IceType_i64;
162 default:
163 report_fatal_error(std::string("Invalid PNaCl int type: ") +
164 LLVMObjectAsString(IntTy));
165 return Ice::IceType_void;
166 }
167 }
168
169 Ice::Type convertVectorType(const VectorType *VecTy) const {
170 unsigned NumElements = VecTy->getNumElements();
171 const Type *ElementType = VecTy->getElementType();
172
173 if (ElementType->isFloatTy()) {
174 if (NumElements == 4)
175 return Ice::IceType_v4f32;
176 } else if (ElementType->isIntegerTy()) {
177 switch (cast<IntegerType>(ElementType)->getBitWidth()) {
178 case 1:
179 if (NumElements == 4)
180 return Ice::IceType_v4i1;
181 if (NumElements == 8)
182 return Ice::IceType_v8i1;
183 if (NumElements == 16)
184 return Ice::IceType_v16i1;
185 break;
186 case 8:
187 if (NumElements == 16)
188 return Ice::IceType_v16i8;
189 break;
190 case 16:
191 if (NumElements == 8)
192 return Ice::IceType_v8i16;
193 break;
194 case 32:
195 if (NumElements == 4)
196 return Ice::IceType_v4i32;
197 break;
198 }
199 }
200
201 report_fatal_error(std::string("Unhandled vector type: ") +
202 LLVMObjectAsString(VecTy));
203 return Ice::IceType_void;
204 }
205
206 Ice::Type convertType(const Type *Ty) const {
207 switch (Ty->getTypeID()) {
208 case Type::VoidTyID:
209 return Ice::IceType_void;
210 case Type::IntegerTyID:
211 return convertIntegerType(cast<IntegerType>(Ty));
212 case Type::FloatTyID:
213 return Ice::IceType_f32;
214 case Type::DoubleTyID:
215 return Ice::IceType_f64;
216 case Type::PointerTyID:
217 return SubzeroPointerType;
218 case Type::FunctionTyID:
219 return SubzeroPointerType;
220 case Type::VectorTyID:
221 return convertVectorType(cast<VectorType>(Ty));
222 default:
223 report_fatal_error(std::string("Invalid PNaCl type: ") +
224 LLVMObjectAsString(Ty));
225 }
226
227 llvm_unreachable("convertType");
228 return Ice::IceType_void;
229 } 157 }
230 158
231 // Given an LLVM instruction and an operand number, produce the 159 // Given an LLVM instruction and an operand number, produce the
232 // Ice::Operand this refers to. If there's no such operand, return 160 // Ice::Operand this refers to. If there's no such operand, return
233 // NULL. 161 // NULL.
234 Ice::Operand *convertOperand(const Instruction *Inst, unsigned OpNum) { 162 Ice::Operand *convertOperand(const Instruction *Inst, unsigned OpNum) {
235 if (OpNum >= Inst->getNumOperands()) { 163 if (OpNum >= Inst->getNumOperands()) {
236 return NULL; 164 return NULL;
237 } 165 }
238 const Value *Op = Inst->getOperand(OpNum); 166 const Value *Op = Inst->getOperand(OpNum);
(...skipping 158 matching lines...) Expand 10 before | Expand all | Expand 10 after
397 Ice::CfgNode *NodeElse = mapBasicBlockToNode(BBElse); 325 Ice::CfgNode *NodeElse = mapBasicBlockToNode(BBElse);
398 return Ice::InstBr::create(Func, Src, NodeThen, NodeElse); 326 return Ice::InstBr::create(Func, Src, NodeThen, NodeElse);
399 } else { 327 } else {
400 BasicBlock *BBSucc = Inst->getSuccessor(0); 328 BasicBlock *BBSucc = Inst->getSuccessor(0);
401 return Ice::InstBr::create(Func, mapBasicBlockToNode(BBSucc)); 329 return Ice::InstBr::create(Func, mapBasicBlockToNode(BBSucc));
402 } 330 }
403 } 331 }
404 332
405 Ice::Inst *convertIntToPtrInstruction(const IntToPtrInst *Inst) { 333 Ice::Inst *convertIntToPtrInstruction(const IntToPtrInst *Inst) {
406 Ice::Operand *Src = convertOperand(Inst, 0); 334 Ice::Operand *Src = convertOperand(Inst, 0);
407 Ice::Variable *Dest = mapValueToIceVar(Inst, SubzeroPointerType); 335 Ice::Variable *Dest =
336 mapValueToIceVar(Inst, TypeConverter.getIcePointerType());
408 return Ice::InstAssign::create(Func, Dest, Src); 337 return Ice::InstAssign::create(Func, Dest, Src);
409 } 338 }
410 339
411 Ice::Inst *convertPtrToIntInstruction(const PtrToIntInst *Inst) { 340 Ice::Inst *convertPtrToIntInstruction(const PtrToIntInst *Inst) {
412 Ice::Operand *Src = convertOperand(Inst, 0); 341 Ice::Operand *Src = convertOperand(Inst, 0);
413 Ice::Variable *Dest = mapValueToIceVar(Inst); 342 Ice::Variable *Dest = mapValueToIceVar(Inst);
414 return Ice::InstAssign::create(Func, Dest, Src); 343 return Ice::InstAssign::create(Func, Dest, Src);
415 } 344 }
416 345
417 Ice::Inst *convertRetInstruction(const ReturnInst *Inst) { 346 Ice::Inst *convertRetInstruction(const ReturnInst *Inst) {
(...skipping 197 matching lines...) Expand 10 before | Expand all | Expand 10 after
615 if (Info) { 544 if (Info) {
616 validateIntrinsicCall(NewInst, Info); 545 validateIntrinsicCall(NewInst, Info);
617 } 546 }
618 return NewInst; 547 return NewInst;
619 } 548 }
620 549
621 Ice::Inst *convertAllocaInstruction(const AllocaInst *Inst) { 550 Ice::Inst *convertAllocaInstruction(const AllocaInst *Inst) {
622 // PNaCl bitcode only contains allocas of byte-granular objects. 551 // PNaCl bitcode only contains allocas of byte-granular objects.
623 Ice::Operand *ByteCount = convertValue(Inst->getArraySize()); 552 Ice::Operand *ByteCount = convertValue(Inst->getArraySize());
624 uint32_t Align = Inst->getAlignment(); 553 uint32_t Align = Inst->getAlignment();
625 Ice::Variable *Dest = mapValueToIceVar(Inst, SubzeroPointerType); 554 Ice::Variable *Dest =
555 mapValueToIceVar(Inst, TypeConverter.getIcePointerType());
626 556
627 return Ice::InstAlloca::create(Func, ByteCount, Align, Dest); 557 return Ice::InstAlloca::create(Func, ByteCount, Align, Dest);
628 } 558 }
629 559
630 Ice::Inst *convertUnreachableInstruction(const UnreachableInst * /*Inst*/) { 560 Ice::Inst *convertUnreachableInstruction(const UnreachableInst * /*Inst*/) {
631 return Ice::InstUnreachable::create(Func); 561 return Ice::InstUnreachable::create(Func);
632 } 562 }
633 563
634 Ice::CfgNode *convertBasicBlock(const BasicBlock *BB) { 564 Ice::CfgNode *convertBasicBlock(const BasicBlock *BB) {
635 Ice::CfgNode *Node = mapBasicBlockToNode(BB); 565 Ice::CfgNode *Node = mapBasicBlockToNode(BB);
(...skipping 28 matching lines...) Expand all
664 report_fatal_error("Mismatched argument type."); 594 report_fatal_error("Mismatched argument type.");
665 } 595 }
666 } 596 }
667 } 597 }
668 598
669 private: 599 private:
670 // Data 600 // Data
671 Ice::GlobalContext *Ctx; 601 Ice::GlobalContext *Ctx;
672 Ice::Cfg *Func; 602 Ice::Cfg *Func;
673 Ice::CfgNode *CurrentNode; 603 Ice::CfgNode *CurrentNode;
674 Ice::Type SubzeroPointerType;
675 std::map<const Value *, Ice::Variable *> VarMap; 604 std::map<const Value *, Ice::Variable *> VarMap;
676 std::map<const BasicBlock *, Ice::CfgNode *> NodeMap; 605 std::map<const BasicBlock *, Ice::CfgNode *> NodeMap;
606 Ice::TypeConverter TypeConverter;
677 }; 607 };
678 608
679 } // end of anonymous namespace. 609 } // end of anonymous namespace.
680 610
681 namespace Ice { 611 namespace Ice {
682 612
683 void Converter::convertToIce(Module *Mod) { 613 void Converter::convertToIce() {
684 convertGlobals(Mod); 614 convertGlobals();
685 convertFunctions(Mod); 615 convertFunctions();
686 } 616 }
687 617
688 void Converter::convertGlobals(Module *Mod) { 618 void Converter::convertGlobals() {
689 OwningPtr<TargetGlobalInitLowering> GlobalLowering( 619 OwningPtr<TargetGlobalInitLowering> GlobalLowering(
690 TargetGlobalInitLowering::createLowering(Ctx->getTargetArch(), Ctx)); 620 TargetGlobalInitLowering::createLowering(Ctx->getTargetArch(), Ctx));
691 for (Module::const_global_iterator I = Mod->global_begin(), 621 for (Module::const_global_iterator I = Mod->global_begin(),
692 E = Mod->global_end(); 622 E = Mod->global_end();
693 I != E; ++I) { 623 I != E; ++I) {
694 if (!I->hasInitializer()) 624 if (!I->hasInitializer())
695 continue; 625 continue;
696 const llvm::Constant *Initializer = I->getInitializer(); 626 const llvm::Constant *Initializer = I->getInitializer();
697 IceString Name = I->getName(); 627 IceString Name = I->getName();
698 unsigned Align = I->getAlignment(); 628 unsigned Align = I->getAlignment();
(...skipping 21 matching lines...) Expand all
720 } else { 650 } else {
721 llvm_unreachable("Unhandled global initializer"); 651 llvm_unreachable("Unhandled global initializer");
722 } 652 }
723 653
724 GlobalLowering->lower(Name, Align, IsInternal, IsConst, IsZeroInitializer, 654 GlobalLowering->lower(Name, Align, IsInternal, IsConst, IsZeroInitializer,
725 NumElements, Data, Flags.DisableTranslation); 655 NumElements, Data, Flags.DisableTranslation);
726 } 656 }
727 GlobalLowering.reset(); 657 GlobalLowering.reset();
728 } 658 }
729 659
730 void Converter::convertFunctions(Module *Mod) { 660 void Converter::convertFunctions() {
731 for (Module::const_iterator I = Mod->begin(), E = Mod->end(); I != E; ++I) { 661 for (Module::const_iterator I = Mod->begin(), E = Mod->end(); I != E; ++I) {
732 if (I->empty()) 662 if (I->empty())
733 continue; 663 continue;
734 LLVM2ICEConverter FunctionConverter(Ctx); 664 LLVM2ICEConverter FunctionConverter(Ctx, Mod->getContext());
735 665
736 Timer TConvert; 666 Timer TConvert;
737 Cfg *Fcn = FunctionConverter.convertFunction(I); 667 Cfg *Fcn = FunctionConverter.convertFunction(I);
738 if (Flags.SubzeroTimingEnabled) { 668 if (Flags.SubzeroTimingEnabled) {
739 std::cerr << "[Subzero timing] Convert function " 669 std::cerr << "[Subzero timing] Convert function "
740 << Fcn->getFunctionName() << ": " << TConvert.getElapsedSec() 670 << Fcn->getFunctionName() << ": " << TConvert.getElapsedSec()
741 << " sec\n"; 671 << " sec\n";
742 } 672 }
743 translateFcn(Fcn); 673 translateFcn(Fcn);
744 } 674 }
745 675
746 emitConstants(); 676 emitConstants();
747 } 677 }
748 678
749 } // end of Ice namespace. 679 } // end of Ice namespace.
OLDNEW
« no previous file with comments | « src/IceConverter.h ('k') | src/IceInst.h » ('j') | src/IceTargetLoweringX8632.cpp » ('J')

Powered by Google App Engine
This is Rietveld 408576698