| //===--- Bitcode/Writer/BitcodeWriterPass.cpp - Bitcode Writer ------------===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // BitcodeWriterPass implementation. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "ReaderWriter_2_9.h" |
| #include "llvm/Function.h" |
| #include "llvm/Instructions.h" |
| #include "llvm/Module.h" |
| #include "llvm/Pass.h" |
| using namespace llvm; |
| |
| namespace { |
| class WriteBitcodePass : public ModulePass { |
| raw_ostream &OS; // raw_ostream to print on |
| |
| bool expandCaseRange(Function &F); |
| public: |
| static char ID; // Pass identification, replacement for typeid |
| explicit WriteBitcodePass(raw_ostream &o) |
| : ModulePass(ID), OS(o) {} |
| |
| const char *getPassName() const { return "Bitcode Writer"; } |
| |
| bool runOnModule(Module &M) { |
| bool Changed = false; |
| for (Module::iterator F = M.begin(), E = M.end(); F != E; ++F) |
| if (!F->isDeclaration()) |
| Changed |= expandCaseRange(*F); |
| |
| llvm_2_9::WriteBitcodeToFile(&M, OS); |
| return Changed; |
| } |
| }; |
| } |
| |
| char WriteBitcodePass::ID = 0; |
| |
| /// expandCaseRange - Expand case range into explicit case values within the |
| /// range |
| bool WriteBitcodePass::expandCaseRange(Function &F) { |
| bool Changed = false; |
| |
| for (Function::iterator BB = F.begin(), E = F.end(); BB != E; ++BB) { |
| SwitchInst *SI = dyn_cast<SwitchInst>(BB->getTerminator()); |
| if (SI == NULL) { |
| continue; |
| } |
| |
| for (SwitchInst::CaseIt i = SI->case_begin(), e = SI->case_end(); |
| i != e; ++i) { |
| IntegersSubset& CaseRanges = i.getCaseValueEx(); |
| |
| // All case ranges are already in single case values |
| if (CaseRanges.isSingleNumbersOnly()) { |
| continue; |
| } |
| |
| // Create a new case |
| Type *IntTy = SI->getCondition()->getType(); |
| IntegersSubsetToBB CaseBuilder; |
| Changed = true; |
| |
| for (unsigned ri = 0, rn = CaseRanges.getNumItems(); ri != rn; ++ri) { |
| IntegersSubset::Range r = CaseRanges.getItem(ri); |
| bool IsSingleNumber = CaseRanges.isSingleNumber(ri); |
| |
| if (IsSingleNumber) { |
| CaseBuilder.add(r); |
| } else { |
| const APInt &Low = r.getLow(); |
| const APInt &High = r.getHigh(); |
| |
| for (APInt V = Low; V != High; V++) { |
| assert(r.isInRange(V) && "Unexpected out-of-range case value!"); |
| CaseBuilder.add(IntItem::fromType(IntTy, V)); |
| } |
| } |
| |
| IntegersSubset Case = CaseBuilder.getCase(); |
| i.setValueEx(Case); |
| } |
| } |
| } |
| return Changed; |
| } |
| |
| /// createBitcodeWriterPass - Create and return a pass that writes the module |
| /// to the specified ostream. |
| llvm::ModulePass *llvm_2_9::createBitcodeWriterPass(llvm::raw_ostream &Str) { |
| return new WriteBitcodePass(Str); |
| } |