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

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: Code ready for review. 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)
jvoung (off chromium) 2014/07/17 23:57:55 Before we get too many instances of "Llvm", I'd li
Karl 2014/07/18 20:27:42 Consensus is to use LLVM. Changing.
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.
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 154 matching lines...) Expand 10 before | Expand all | Expand 10 after
393 Ice::CfgNode *NodeElse = mapBasicBlockToNode(BBElse); 321 Ice::CfgNode *NodeElse = mapBasicBlockToNode(BBElse);
394 return Ice::InstBr::create(Func, Src, NodeThen, NodeElse); 322 return Ice::InstBr::create(Func, Src, NodeThen, NodeElse);
395 } else { 323 } else {
396 BasicBlock *BBSucc = Inst->getSuccessor(0); 324 BasicBlock *BBSucc = Inst->getSuccessor(0);
397 return Ice::InstBr::create(Func, mapBasicBlockToNode(BBSucc)); 325 return Ice::InstBr::create(Func, mapBasicBlockToNode(BBSucc));
398 } 326 }
399 } 327 }
400 328
401 Ice::Inst *convertIntToPtrInstruction(const IntToPtrInst *Inst) { 329 Ice::Inst *convertIntToPtrInstruction(const IntToPtrInst *Inst) {
402 Ice::Operand *Src = convertOperand(Inst, 0); 330 Ice::Operand *Src = convertOperand(Inst, 0);
403 Ice::Variable *Dest = mapValueToIceVar(Inst, SubzeroPointerType); 331 Ice::Variable *Dest =
332 mapValueToIceVar(Inst, TypeConverter.getIcePointerType());
404 return Ice::InstAssign::create(Func, Dest, Src); 333 return Ice::InstAssign::create(Func, Dest, Src);
405 } 334 }
406 335
407 Ice::Inst *convertPtrToIntInstruction(const PtrToIntInst *Inst) { 336 Ice::Inst *convertPtrToIntInstruction(const PtrToIntInst *Inst) {
408 Ice::Operand *Src = convertOperand(Inst, 0); 337 Ice::Operand *Src = convertOperand(Inst, 0);
409 Ice::Variable *Dest = mapValueToIceVar(Inst); 338 Ice::Variable *Dest = mapValueToIceVar(Inst);
410 return Ice::InstAssign::create(Func, Dest, Src); 339 return Ice::InstAssign::create(Func, Dest, Src);
411 } 340 }
412 341
413 Ice::Inst *convertRetInstruction(const ReturnInst *Inst) { 342 Ice::Inst *convertRetInstruction(const ReturnInst *Inst) {
(...skipping 181 matching lines...) Expand 10 before | Expand all | Expand 10 after
595 if (Info) { 524 if (Info) {
596 validateIntrinsicCall(NewInst, Info); 525 validateIntrinsicCall(NewInst, Info);
597 } 526 }
598 return NewInst; 527 return NewInst;
599 } 528 }
600 529
601 Ice::Inst *convertAllocaInstruction(const AllocaInst *Inst) { 530 Ice::Inst *convertAllocaInstruction(const AllocaInst *Inst) {
602 // PNaCl bitcode only contains allocas of byte-granular objects. 531 // PNaCl bitcode only contains allocas of byte-granular objects.
603 Ice::Operand *ByteCount = convertValue(Inst->getArraySize()); 532 Ice::Operand *ByteCount = convertValue(Inst->getArraySize());
604 uint32_t Align = Inst->getAlignment(); 533 uint32_t Align = Inst->getAlignment();
605 Ice::Variable *Dest = mapValueToIceVar(Inst, SubzeroPointerType); 534 Ice::Variable *Dest =
535 mapValueToIceVar(Inst, TypeConverter.getIcePointerType());
606 536
607 return Ice::InstAlloca::create(Func, ByteCount, Align, Dest); 537 return Ice::InstAlloca::create(Func, ByteCount, Align, Dest);
608 } 538 }
609 539
610 Ice::Inst *convertUnreachableInstruction(const UnreachableInst * /*Inst*/) { 540 Ice::Inst *convertUnreachableInstruction(const UnreachableInst * /*Inst*/) {
611 return Ice::InstUnreachable::create(Func); 541 return Ice::InstUnreachable::create(Func);
612 } 542 }
613 543
614 Ice::CfgNode *convertBasicBlock(const BasicBlock *BB) { 544 Ice::CfgNode *convertBasicBlock(const BasicBlock *BB) {
615 Ice::CfgNode *Node = mapBasicBlockToNode(BB); 545 Ice::CfgNode *Node = mapBasicBlockToNode(BB);
(...skipping 28 matching lines...) Expand all
644 report_fatal_error("Mismatched argument type."); 574 report_fatal_error("Mismatched argument type.");
645 } 575 }
646 } 576 }
647 } 577 }
648 578
649 private: 579 private:
650 // Data 580 // Data
651 Ice::GlobalContext *Ctx; 581 Ice::GlobalContext *Ctx;
652 Ice::Cfg *Func; 582 Ice::Cfg *Func;
653 Ice::CfgNode *CurrentNode; 583 Ice::CfgNode *CurrentNode;
654 Ice::Type SubzeroPointerType;
655 std::map<const Value *, Ice::Variable *> VarMap; 584 std::map<const Value *, Ice::Variable *> VarMap;
656 std::map<const BasicBlock *, Ice::CfgNode *> NodeMap; 585 std::map<const BasicBlock *, Ice::CfgNode *> NodeMap;
586 Ice::TypeConverter TypeConverter;
657 }; 587 };
658 588
659 } // end of anonymous namespace. 589 } // end of anonymous namespace.
660 590
661 namespace Ice { 591 namespace Ice {
662 592
663 void Converter::convertToIce(Module *Mod) { 593 void Converter::convertToIce() {
664 convertGlobals(Mod); 594 convertGlobals();
665 convertFunctions(Mod); 595 convertFunctions();
666 } 596 }
667 597
668 void Converter::convertGlobals(Module *Mod) { 598 void Converter::convertGlobals() {
669 OwningPtr<TargetGlobalInitLowering> GlobalLowering( 599 OwningPtr<TargetGlobalInitLowering> GlobalLowering(
670 TargetGlobalInitLowering::createLowering(Ctx->getTargetArch(), Ctx)); 600 TargetGlobalInitLowering::createLowering(Ctx->getTargetArch(), Ctx));
671 for (Module::const_global_iterator I = Mod->global_begin(), 601 for (Module::const_global_iterator I = Mod->global_begin(),
672 E = Mod->global_end(); 602 E = Mod->global_end();
673 I != E; ++I) { 603 I != E; ++I) {
674 if (!I->hasInitializer()) 604 if (!I->hasInitializer())
675 continue; 605 continue;
676 const llvm::Constant *Initializer = I->getInitializer(); 606 const llvm::Constant *Initializer = I->getInitializer();
677 IceString Name = I->getName(); 607 IceString Name = I->getName();
678 unsigned Align = I->getAlignment(); 608 unsigned Align = I->getAlignment();
(...skipping 21 matching lines...) Expand all
700 } else { 630 } else {
701 llvm_unreachable("Unhandled global initializer"); 631 llvm_unreachable("Unhandled global initializer");
702 } 632 }
703 633
704 GlobalLowering->lower(Name, Align, IsInternal, IsConst, IsZeroInitializer, 634 GlobalLowering->lower(Name, Align, IsInternal, IsConst, IsZeroInitializer,
705 NumElements, Data, Flags.DisableTranslation); 635 NumElements, Data, Flags.DisableTranslation);
706 } 636 }
707 GlobalLowering.reset(); 637 GlobalLowering.reset();
708 } 638 }
709 639
710 void Converter::convertFunctions(Module *Mod) { 640 void Converter::convertFunctions() {
711 for (Module::const_iterator I = Mod->begin(), E = Mod->end(); I != E; ++I) { 641 for (Module::const_iterator I = Mod->begin(), E = Mod->end(); I != E; ++I) {
712 if (I->empty()) 642 if (I->empty())
713 continue; 643 continue;
714 LLVM2ICEConverter FunctionConverter(Ctx); 644 LLVM2ICEConverter FunctionConverter(Ctx, Mod->getContext());
715 645
716 Timer TConvert; 646 Timer TConvert;
717 Cfg *Fcn = FunctionConverter.convertFunction(I); 647 Cfg *Fcn = FunctionConverter.convertFunction(I);
718 if (Flags.SubzeroTimingEnabled) { 648 if (Flags.SubzeroTimingEnabled) {
719 std::cerr << "[Subzero timing] Convert function " 649 std::cerr << "[Subzero timing] Convert function "
720 << Fcn->getFunctionName() << ": " << TConvert.getElapsedSec() 650 << Fcn->getFunctionName() << ": " << TConvert.getElapsedSec()
721 << " sec\n"; 651 << " sec\n";
722 } 652 }
723 translateFcn(Fcn); 653 translateFcn(Fcn);
724 } 654 }
725 655
726 emitConstants(); 656 emitConstants();
727 } 657 }
728 658
729 } // end of Ice namespace. 659 } // end of Ice namespace.
OLDNEW
« no previous file with comments | « src/IceConverter.h ('k') | src/IceInst.h » ('j') | src/IceTranslator.h » ('J')

Powered by Google App Engine
This is Rietveld 408576698