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

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 issues in patch set 13. 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
« no previous file with comments | « src/IceConverter.h ('k') | src/IceInst.h » ('j') | 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/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 // model.
61 SubzeroPointerType = Ice::IceType_i32;
62 }
63 60
64 // Caller is expected to delete the returned Ice::Cfg object. 61 // Caller is expected to delete the returned Ice::Cfg object.
65 Ice::Cfg *convertFunction(const Function *F) { 62 Ice::Cfg *convertFunction(const Function *F) {
66 VarMap.clear(); 63 VarMap.clear();
67 NodeMap.clear(); 64 NodeMap.clear();
68 Func = new Ice::Cfg(Ctx); 65 Func = new Ice::Cfg(Ctx);
69 Func->setFunctionName(F->getName()); 66 Func->setFunctionName(F->getName());
70 Func->setReturnType(convertType(F->getReturnType())); 67 Func->setReturnType(convertToIceType(F->getReturnType()));
71 Func->setInternal(F->hasInternalLinkage()); 68 Func->setInternal(F->hasInternalLinkage());
72 69
73 // The initial definition/use of each arg is the entry node. 70 // The initial definition/use of each arg is the entry node.
74 CurrentNode = mapBasicBlockToNode(&F->getEntryBlock()); 71 CurrentNode = mapBasicBlockToNode(&F->getEntryBlock());
75 for (Function::const_arg_iterator ArgI = F->arg_begin(), 72 for (Function::const_arg_iterator ArgI = F->arg_begin(),
76 ArgE = F->arg_end(); 73 ArgE = F->arg_end();
77 ArgI != ArgE; ++ArgI) { 74 ArgI != ArgE; ++ArgI) {
78 Func->addArg(mapValueToIceVar(ArgI)); 75 Func->addArg(mapValueToIceVar(ArgI));
79 } 76 }
80 77
(...skipping 14 matching lines...) Expand all
95 Func->computePredecessors(); 92 Func->computePredecessors();
96 93
97 return Func; 94 return Func;
98 } 95 }
99 96
100 // convertConstant() does not use Func or require it to be a valid 97 // 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 98 // Ice::Cfg pointer. As such, it's suitable for e.g. constructing
102 // global initializers. 99 // global initializers.
103 Ice::Constant *convertConstant(const Constant *Const) { 100 Ice::Constant *convertConstant(const Constant *Const) {
104 if (const GlobalValue *GV = dyn_cast<GlobalValue>(Const)) { 101 if (const GlobalValue *GV = dyn_cast<GlobalValue>(Const)) {
105 return Ctx->getConstantSym(convertType(GV->getType()), 0, GV->getName()); 102 return Ctx->getConstantSym(convertToIceType(GV->getType()), 0,
103 GV->getName());
106 } else if (const ConstantInt *CI = dyn_cast<ConstantInt>(Const)) { 104 } else if (const ConstantInt *CI = dyn_cast<ConstantInt>(Const)) {
107 return Ctx->getConstantInt(convertIntegerType(CI->getType()), 105 return Ctx->getConstantInt(convertToIceType(CI->getType()),
108 CI->getZExtValue()); 106 CI->getZExtValue());
109 } else if (const ConstantFP *CFP = dyn_cast<ConstantFP>(Const)) { 107 } else if (const ConstantFP *CFP = dyn_cast<ConstantFP>(Const)) {
110 Ice::Type Type = convertType(CFP->getType()); 108 Ice::Type Type = convertToIceType(CFP->getType());
111 if (Type == Ice::IceType_f32) 109 if (Type == Ice::IceType_f32)
112 return Ctx->getConstantFloat(CFP->getValueAPF().convertToFloat()); 110 return Ctx->getConstantFloat(CFP->getValueAPF().convertToFloat());
113 else if (Type == Ice::IceType_f64) 111 else if (Type == Ice::IceType_f64)
114 return Ctx->getConstantDouble(CFP->getValueAPF().convertToDouble()); 112 return Ctx->getConstantDouble(CFP->getValueAPF().convertToDouble());
115 llvm_unreachable("Unexpected floating point type"); 113 llvm_unreachable("Unexpected floating point type");
116 return NULL; 114 return NULL;
117 } else if (const UndefValue *CU = dyn_cast<UndefValue>(Const)) { 115 } else if (const UndefValue *CU = dyn_cast<UndefValue>(Const)) {
118 return Ctx->getConstantUndef(convertType(CU->getType())); 116 return Ctx->getConstantUndef(convertToIceType(CU->getType()));
119 } else { 117 } else {
120 llvm_unreachable("Unhandled constant type"); 118 llvm_unreachable("Unhandled constant type");
121 return NULL; 119 return NULL;
122 } 120 }
123 } 121 }
124 122
125 private: 123 private:
126 // LLVM values (instructions, etc.) are mapped directly to ICE variables. 124 // LLVM values (instructions, etc.) are mapped directly to ICE variables.
127 // mapValueToIceVar has a version that forces an ICE type on the variable, 125 // mapValueToIceVar has a version that forces an ICE type on the variable,
128 // and a version that just uses convertType on V. 126 // and a version that just uses convertToIceType on V.
129 Ice::Variable *mapValueToIceVar(const Value *V, Ice::Type IceTy) { 127 Ice::Variable *mapValueToIceVar(const Value *V, Ice::Type IceTy) {
130 if (IceTy == Ice::IceType_void) 128 if (IceTy == Ice::IceType_void)
131 return NULL; 129 return NULL;
132 if (VarMap.find(V) == VarMap.end()) { 130 if (VarMap.find(V) == VarMap.end()) {
133 assert(CurrentNode); 131 assert(CurrentNode);
134 VarMap[V] = Func->makeVariable(IceTy, CurrentNode, V->getName()); 132 VarMap[V] = Func->makeVariable(IceTy, CurrentNode, V->getName());
135 } 133 }
136 return VarMap[V]; 134 return VarMap[V];
137 } 135 }
138 136
139 Ice::Variable *mapValueToIceVar(const Value *V) { 137 Ice::Variable *mapValueToIceVar(const Value *V) {
140 return mapValueToIceVar(V, convertType(V->getType())); 138 return mapValueToIceVar(V, convertToIceType(V->getType()));
141 } 139 }
142 140
143 Ice::CfgNode *mapBasicBlockToNode(const BasicBlock *BB) { 141 Ice::CfgNode *mapBasicBlockToNode(const BasicBlock *BB) {
144 if (NodeMap.find(BB) == NodeMap.end()) { 142 if (NodeMap.find(BB) == NodeMap.end()) {
145 NodeMap[BB] = Func->makeNode(BB->getName()); 143 NodeMap[BB] = Func->makeNode(BB->getName());
146 } 144 }
147 return NodeMap[BB]; 145 return NodeMap[BB];
148 } 146 }
149 147
150 Ice::Type convertIntegerType(const IntegerType *IntTy) const { 148 Ice::Type convertToIceType(Type *LLVMTy) const {
151 switch (IntTy->getBitWidth()) { 149 Ice::Type IceTy = TypeConverter.convertToIceType(LLVMTy);
152 case 1: 150 if (IceTy == Ice::IceType_NUM)
153 return Ice::IceType_i1; 151 llvm::report_fatal_error(std::string("Invalid PNaCl type ") +
154 case 8: 152 LLVMObjectAsString(LLVMTy));
155 return Ice::IceType_i8; 153 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 } 154 }
230 155
231 // Given an LLVM instruction and an operand number, produce the 156 // Given an LLVM instruction and an operand number, produce the
232 // Ice::Operand this refers to. If there's no such operand, return 157 // Ice::Operand this refers to. If there's no such operand, return
233 // NULL. 158 // NULL.
234 Ice::Operand *convertOperand(const Instruction *Inst, unsigned OpNum) { 159 Ice::Operand *convertOperand(const Instruction *Inst, unsigned OpNum) {
235 if (OpNum >= Inst->getNumOperands()) { 160 if (OpNum >= Inst->getNumOperands()) {
236 return NULL; 161 return NULL;
237 } 162 }
238 const Value *Op = Inst->getOperand(OpNum); 163 const Value *Op = Inst->getOperand(OpNum);
(...skipping 158 matching lines...) Expand 10 before | Expand all | Expand 10 after
397 Ice::CfgNode *NodeElse = mapBasicBlockToNode(BBElse); 322 Ice::CfgNode *NodeElse = mapBasicBlockToNode(BBElse);
398 return Ice::InstBr::create(Func, Src, NodeThen, NodeElse); 323 return Ice::InstBr::create(Func, Src, NodeThen, NodeElse);
399 } else { 324 } else {
400 BasicBlock *BBSucc = Inst->getSuccessor(0); 325 BasicBlock *BBSucc = Inst->getSuccessor(0);
401 return Ice::InstBr::create(Func, mapBasicBlockToNode(BBSucc)); 326 return Ice::InstBr::create(Func, mapBasicBlockToNode(BBSucc));
402 } 327 }
403 } 328 }
404 329
405 Ice::Inst *convertIntToPtrInstruction(const IntToPtrInst *Inst) { 330 Ice::Inst *convertIntToPtrInstruction(const IntToPtrInst *Inst) {
406 Ice::Operand *Src = convertOperand(Inst, 0); 331 Ice::Operand *Src = convertOperand(Inst, 0);
407 Ice::Variable *Dest = mapValueToIceVar(Inst, SubzeroPointerType); 332 Ice::Variable *Dest =
333 mapValueToIceVar(Inst, TypeConverter.getIcePointerType());
408 return Ice::InstAssign::create(Func, Dest, Src); 334 return Ice::InstAssign::create(Func, Dest, Src);
409 } 335 }
410 336
411 Ice::Inst *convertPtrToIntInstruction(const PtrToIntInst *Inst) { 337 Ice::Inst *convertPtrToIntInstruction(const PtrToIntInst *Inst) {
412 Ice::Operand *Src = convertOperand(Inst, 0); 338 Ice::Operand *Src = convertOperand(Inst, 0);
413 Ice::Variable *Dest = mapValueToIceVar(Inst); 339 Ice::Variable *Dest = mapValueToIceVar(Inst);
414 return Ice::InstAssign::create(Func, Dest, Src); 340 return Ice::InstAssign::create(Func, Dest, Src);
415 } 341 }
416 342
417 Ice::Inst *convertRetInstruction(const ReturnInst *Inst) { 343 Ice::Inst *convertRetInstruction(const ReturnInst *Inst) {
(...skipping 197 matching lines...) Expand 10 before | Expand all | Expand 10 after
615 if (Info) { 541 if (Info) {
616 validateIntrinsicCall(NewInst, Info); 542 validateIntrinsicCall(NewInst, Info);
617 } 543 }
618 return NewInst; 544 return NewInst;
619 } 545 }
620 546
621 Ice::Inst *convertAllocaInstruction(const AllocaInst *Inst) { 547 Ice::Inst *convertAllocaInstruction(const AllocaInst *Inst) {
622 // PNaCl bitcode only contains allocas of byte-granular objects. 548 // PNaCl bitcode only contains allocas of byte-granular objects.
623 Ice::Operand *ByteCount = convertValue(Inst->getArraySize()); 549 Ice::Operand *ByteCount = convertValue(Inst->getArraySize());
624 uint32_t Align = Inst->getAlignment(); 550 uint32_t Align = Inst->getAlignment();
625 Ice::Variable *Dest = mapValueToIceVar(Inst, SubzeroPointerType); 551 Ice::Variable *Dest =
552 mapValueToIceVar(Inst, TypeConverter.getIcePointerType());
626 553
627 return Ice::InstAlloca::create(Func, ByteCount, Align, Dest); 554 return Ice::InstAlloca::create(Func, ByteCount, Align, Dest);
628 } 555 }
629 556
630 Ice::Inst *convertUnreachableInstruction(const UnreachableInst * /*Inst*/) { 557 Ice::Inst *convertUnreachableInstruction(const UnreachableInst * /*Inst*/) {
631 return Ice::InstUnreachable::create(Func); 558 return Ice::InstUnreachable::create(Func);
632 } 559 }
633 560
634 Ice::CfgNode *convertBasicBlock(const BasicBlock *BB) { 561 Ice::CfgNode *convertBasicBlock(const BasicBlock *BB) {
635 Ice::CfgNode *Node = mapBasicBlockToNode(BB); 562 Ice::CfgNode *Node = mapBasicBlockToNode(BB);
(...skipping 28 matching lines...) Expand all
664 report_fatal_error("Mismatched argument type."); 591 report_fatal_error("Mismatched argument type.");
665 } 592 }
666 } 593 }
667 } 594 }
668 595
669 private: 596 private:
670 // Data 597 // Data
671 Ice::GlobalContext *Ctx; 598 Ice::GlobalContext *Ctx;
672 Ice::Cfg *Func; 599 Ice::Cfg *Func;
673 Ice::CfgNode *CurrentNode; 600 Ice::CfgNode *CurrentNode;
674 Ice::Type SubzeroPointerType;
675 std::map<const Value *, Ice::Variable *> VarMap; 601 std::map<const Value *, Ice::Variable *> VarMap;
676 std::map<const BasicBlock *, Ice::CfgNode *> NodeMap; 602 std::map<const BasicBlock *, Ice::CfgNode *> NodeMap;
603 Ice::TypeConverter TypeConverter;
677 }; 604 };
678 605
679 } // end of anonymous namespace 606 } // end of anonymous namespace
680 607
681 namespace Ice { 608 namespace Ice {
682 609
683 void Converter::convertToIce(Module *Mod) { 610 void Converter::convertToIce() {
684 if (!Ctx->getFlags().DisableGlobals) 611 if (!Ctx->getFlags().DisableGlobals)
685 convertGlobals(Mod); 612 convertGlobals();
686 convertFunctions(Mod); 613 convertFunctions();
687 } 614 }
688 615
689 void Converter::convertGlobals(Module *Mod) { 616 void Converter::convertGlobals() {
690 OwningPtr<TargetGlobalInitLowering> GlobalLowering( 617 OwningPtr<TargetGlobalInitLowering> GlobalLowering(
691 TargetGlobalInitLowering::createLowering(Ctx->getTargetArch(), Ctx)); 618 TargetGlobalInitLowering::createLowering(Ctx->getTargetArch(), Ctx));
692 for (Module::const_global_iterator I = Mod->global_begin(), 619 for (Module::const_global_iterator I = Mod->global_begin(),
693 E = Mod->global_end(); 620 E = Mod->global_end();
694 I != E; ++I) { 621 I != E; ++I) {
695 if (!I->hasInitializer()) 622 if (!I->hasInitializer())
696 continue; 623 continue;
697 const llvm::Constant *Initializer = I->getInitializer(); 624 const llvm::Constant *Initializer = I->getInitializer();
698 IceString Name = I->getName(); 625 IceString Name = I->getName();
699 unsigned Align = I->getAlignment(); 626 unsigned Align = I->getAlignment();
(...skipping 22 matching lines...) Expand all
722 llvm_unreachable("Unhandled global initializer"); 649 llvm_unreachable("Unhandled global initializer");
723 } 650 }
724 651
725 GlobalLowering->lower(Name, Align, IsInternal, IsConst, IsZeroInitializer, 652 GlobalLowering->lower(Name, Align, IsInternal, IsConst, IsZeroInitializer,
726 NumElements, Data, 653 NumElements, Data,
727 Ctx->getFlags().DisableTranslation); 654 Ctx->getFlags().DisableTranslation);
728 } 655 }
729 GlobalLowering.reset(); 656 GlobalLowering.reset();
730 } 657 }
731 658
732 void Converter::convertFunctions(Module *Mod) { 659 void Converter::convertFunctions() {
733 for (Module::const_iterator I = Mod->begin(), E = Mod->end(); I != E; ++I) { 660 for (Module::const_iterator I = Mod->begin(), E = Mod->end(); I != E; ++I) {
734 if (I->empty()) 661 if (I->empty())
735 continue; 662 continue;
736 LLVM2ICEConverter FunctionConverter(Ctx); 663 LLVM2ICEConverter FunctionConverter(Ctx, Mod->getContext());
737 664
738 Timer TConvert; 665 Timer TConvert;
739 Cfg *Fcn = FunctionConverter.convertFunction(I); 666 Cfg *Fcn = FunctionConverter.convertFunction(I);
740 if (Ctx->getFlags().SubzeroTimingEnabled) { 667 if (Ctx->getFlags().SubzeroTimingEnabled) {
741 std::cerr << "[Subzero timing] Convert function " 668 std::cerr << "[Subzero timing] Convert function "
742 << Fcn->getFunctionName() << ": " << TConvert.getElapsedSec() 669 << Fcn->getFunctionName() << ": " << TConvert.getElapsedSec()
743 << " sec\n"; 670 << " sec\n";
744 } 671 }
745 translateFcn(Fcn); 672 translateFcn(Fcn);
746 } 673 }
747 674
748 emitConstants(); 675 emitConstants();
749 } 676 }
750 677
751 } // end of namespace Ice 678 } // end of namespace Ice
OLDNEW
« no previous file with comments | « src/IceConverter.h ('k') | src/IceInst.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698