blob: e8bbe217b8622b8536f52928c94a3fea4dbf4992 [file] [log] [blame]
Nate Begemaneb883af2006-08-23 21:08:52 +00001//===-- MachOWriter.cpp - Target-independent Mach-O Writer code -----------===//
2//
3// The LLVM Compiler Infrastructure
4//
Chris Lattner4ee451d2007-12-29 20:36:04 +00005// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
Nate Begemaneb883af2006-08-23 21:08:52 +00007//
8//===----------------------------------------------------------------------===//
9//
10// This file implements the target-independent Mach-O writer. This file writes
11// out the Mach-O file in the following order:
12//
13// #1 FatHeader (universal-only)
14// #2 FatArch (universal-only, 1 per universal arch)
15// Per arch:
16// #3 Header
17// #4 Load Commands
18// #5 Sections
19// #6 Relocations
20// #7 Symbols
21// #8 Strings
22//
23//===----------------------------------------------------------------------===//
24
Bill Wendling8f84f1f2007-02-08 01:35:27 +000025#include "MachOWriter.h"
Nate Begeman3fe980b2010-01-15 18:51:18 +000026#include "llvm/Function.h"
27#include "llvm/CodeGen/FileWriters.h"
28#include "llvm/CodeGen/MachineFunction.h"
Chris Lattneraf76e592009-08-22 20:48:53 +000029#include "llvm/MC/MCAsmInfo.h"
Nate Begeman3fe980b2010-01-15 18:51:18 +000030#include "llvm/MC/MCContext.h"
31#include "llvm/MC/MCCodeEmitter.h"
32#include "llvm/MC/MCInst.h"
33#include "llvm/MC/MCStreamer.h"
Torok Edwin7d696d82009-07-11 13:10:19 +000034#include "llvm/Support/ErrorHandling.h"
Nate Begeman3fe980b2010-01-15 18:51:18 +000035#include "llvm/Support/FormattedStream.h"
Owen Andersoncb371882008-08-21 00:14:44 +000036#include "llvm/Support/raw_ostream.h"
Chris Lattner45111d12010-01-16 21:57:06 +000037#include "llvm/Target/Mangler.h"
Nate Begeman3fe980b2010-01-15 18:51:18 +000038#include "llvm/Target/TargetData.h"
39#include "llvm/Target/TargetLowering.h"
40#include "llvm/Target/TargetLoweringObjectFile.h"
41using namespace llvm;
Bruno Cardoso Lopesa321dcd2009-06-03 03:43:31 +000042
Nate Begeman3fe980b2010-01-15 18:51:18 +000043namespace llvm {
44MachineFunctionPass *createMachOWriter(formatted_raw_ostream &O,
45 TargetMachine &TM,
46 const MCAsmInfo *T,
47 MCCodeEmitter *MCE) {
48 return new MachOWriter(O, TM, T, MCE);
49}
Bill Wendling8f84f1f2007-02-08 01:35:27 +000050}
51
Nate Begemaneb883af2006-08-23 21:08:52 +000052//===----------------------------------------------------------------------===//
Nate Begemaneb883af2006-08-23 21:08:52 +000053// MachOWriter Implementation
54//===----------------------------------------------------------------------===//
55
Devang Patel19974732007-05-03 01:11:54 +000056char MachOWriter::ID = 0;
Bruno Cardoso Lopesa321dcd2009-06-03 03:43:31 +000057
Nate Begeman3fe980b2010-01-15 18:51:18 +000058MachOWriter::MachOWriter(formatted_raw_ostream &o, TargetMachine &tm,
59 const MCAsmInfo *T, MCCodeEmitter *MCE)
60 : MachineFunctionPass(&ID), O(o), TM(tm), MAI(T), MCCE(MCE),
61 OutContext(*new MCContext()),
62 OutStreamer(*createMachOStreamer(OutContext, O, MCCE)) {
Nate Begemaneb883af2006-08-23 21:08:52 +000063}
64
65MachOWriter::~MachOWriter() {
Nate Begeman3fe980b2010-01-15 18:51:18 +000066 delete &OutStreamer;
67 delete &OutContext;
68 delete MCCE;
Nate Begemaneb883af2006-08-23 21:08:52 +000069}
70
Bruno Cardoso Lopesa321dcd2009-06-03 03:43:31 +000071bool MachOWriter::doInitialization(Module &M) {
Nate Begeman3fe980b2010-01-15 18:51:18 +000072 // Initialize TargetLoweringObjectFile.
73 TM.getTargetLowering()->getObjFileLowering().Initialize(OutContext, TM);
Bruno Cardoso Lopesa321dcd2009-06-03 03:43:31 +000074
Bruno Cardoso Lopesa321dcd2009-06-03 03:43:31 +000075 return false;
76}
77
78/// doFinalization - Now that the module has been completely processed, emit
79/// the Mach-O file to 'O'.
80bool MachOWriter::doFinalization(Module &M) {
Nate Begeman3fe980b2010-01-15 18:51:18 +000081 OutStreamer.Finish();
Bruno Cardoso Lopesa321dcd2009-06-03 03:43:31 +000082 return false;
83}
84
Nate Begeman3fe980b2010-01-15 18:51:18 +000085bool MachOWriter::runOnMachineFunction(MachineFunction &MF) {
86 const Function *F = MF.getFunction();
87 TargetLoweringObjectFile &TLOF = TM.getTargetLowering()->getObjFileLowering();
88 const MCSection *S = TLOF.SectionForGlobal(F, Mang, TM);
89 OutStreamer.SwitchSection(S);
Bruno Cardoso Lopes752e9282009-07-06 06:40:51 +000090
Nate Begeman3fe980b2010-01-15 18:51:18 +000091 for (MachineFunction::const_iterator I = MF.begin(), E = MF.end();
92 I != E; ++I) {
93 // Print a label for the basic block.
94 for (MachineBasicBlock::const_iterator II = I->begin(), IE = I->end();
95 II != IE; ++II) {
96 const MachineInstr *MI = II;
97 MCInst OutMI;
98 OutMI.setOpcode(MI->getOpcode());
Bruno Cardoso Lopes752e9282009-07-06 06:40:51 +000099
Nate Begeman3fe980b2010-01-15 18:51:18 +0000100 for (unsigned i = 0, e = MI->getNumOperands(); i != e; ++i) {
101 const MachineOperand &MO = MI->getOperand(i);
102 MCOperand MCOp;
Bruno Cardoso Lopes752e9282009-07-06 06:40:51 +0000103
Nate Begeman3fe980b2010-01-15 18:51:18 +0000104 switch (MO.getType()) {
105 default:
106 MI->dump();
107 llvm_unreachable("unknown operand type");
108 case MachineOperand::MO_Register:
109 // Ignore all implicit register operands.
110 if (MO.isImplicit()) continue;
111 MCOp = MCOperand::CreateReg(MO.getReg());
112 break;
113 case MachineOperand::MO_Immediate:
114 MCOp = MCOperand::CreateImm(MO.getImm());
115 break;
Nate Begemanbfaaaa62006-12-11 02:20:45 +0000116 }
Nate Begeman3fe980b2010-01-15 18:51:18 +0000117 OutMI.addOperand(MCOp);
Reid Spencera54b7cb2007-01-12 07:05:14 +0000118 }
Nate Begeman3fe980b2010-01-15 18:51:18 +0000119
120 OutStreamer.EmitInstruction(OutMI);
Nate Begemanbfaaaa62006-12-11 02:20:45 +0000121 }
122 }
Nate Begeman3fe980b2010-01-15 18:51:18 +0000123
124 return false;
Nate Begemanbfaaaa62006-12-11 02:20:45 +0000125}