blob: ed64a32eeff2aecfd950212490aefa493275ab91 [file] [log] [blame]
Rafael Espindola908d2ed2011-12-24 02:14:02 +00001//===-- X86WinCOFFObjectWriter.cpp - X86 Win COFF Writer ------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "MCTargetDesc/X86FixupKinds.h"
11#include "MCTargetDesc/X86MCTargetDesc.h"
Nico Rieck1da45292013-04-10 23:28:17 +000012#include "llvm/MC/MCExpr.h"
13#include "llvm/MC/MCValue.h"
Rafael Espindola908d2ed2011-12-24 02:14:02 +000014#include "llvm/MC/MCWinCOFFObjectWriter.h"
15#include "llvm/Support/COFF.h"
16#include "llvm/Support/ErrorHandling.h"
17
18using namespace llvm;
19
20namespace llvm {
21 class MCObjectWriter;
22}
23
24namespace {
25 class X86WinCOFFObjectWriter : public MCWinCOFFObjectTargetWriter {
26 const bool Is64Bit;
27
28 public:
29 X86WinCOFFObjectWriter(bool Is64Bit_);
30 ~X86WinCOFFObjectWriter();
31
Nico Rieck1da45292013-04-10 23:28:17 +000032 virtual unsigned getRelocType(const MCValue &Target,
33 const MCFixup &Fixup,
34 bool IsCrossSection) const LLVM_OVERRIDE;
Rafael Espindola908d2ed2011-12-24 02:14:02 +000035 };
36}
37
38X86WinCOFFObjectWriter::X86WinCOFFObjectWriter(bool Is64Bit_)
39 : MCWinCOFFObjectTargetWriter(Is64Bit_ ? COFF::IMAGE_FILE_MACHINE_AMD64 :
40 COFF::IMAGE_FILE_MACHINE_I386),
41 Is64Bit(Is64Bit_) {}
42
43X86WinCOFFObjectWriter::~X86WinCOFFObjectWriter() {}
44
Nico Rieck1da45292013-04-10 23:28:17 +000045unsigned X86WinCOFFObjectWriter::getRelocType(const MCValue &Target,
46 const MCFixup &Fixup,
47 bool IsCrossSection) const {
48 unsigned FixupKind = IsCrossSection ? FK_PCRel_4 : Fixup.getKind();
49
50 MCSymbolRefExpr::VariantKind Modifier = Target.isAbsolute() ?
51 MCSymbolRefExpr::VK_None : Target.getSymA()->getKind();
52
Rafael Espindola908d2ed2011-12-24 02:14:02 +000053 switch (FixupKind) {
54 case FK_PCRel_4:
55 case X86::reloc_riprel_4byte:
56 case X86::reloc_riprel_4byte_movq_load:
57 return Is64Bit ? COFF::IMAGE_REL_AMD64_REL32 : COFF::IMAGE_REL_I386_REL32;
Rafael Espindola908d2ed2011-12-24 02:14:02 +000058 case FK_Data_4:
59 case X86::reloc_signed_4byte:
Nico Rieck1da45292013-04-10 23:28:17 +000060 if (Modifier == MCSymbolRefExpr::VK_COFF_IMGREL32)
61 return Is64Bit ? COFF::IMAGE_REL_AMD64_ADDR32NB :
62 COFF::IMAGE_REL_I386_DIR32NB;
Rafael Espindola908d2ed2011-12-24 02:14:02 +000063 return Is64Bit ? COFF::IMAGE_REL_AMD64_ADDR32 : COFF::IMAGE_REL_I386_DIR32;
Rafael Espindola908d2ed2011-12-24 02:14:02 +000064 case FK_Data_8:
65 if (Is64Bit)
66 return COFF::IMAGE_REL_AMD64_ADDR64;
David Blaikie46a9f012012-01-20 21:51:11 +000067 llvm_unreachable("unsupported relocation type");
Rafael Espindolaa56ab0ed2011-12-24 14:47:52 +000068 case FK_SecRel_4:
Anton Korobeynikovc6b40172012-02-11 17:26:53 +000069 return Is64Bit ? COFF::IMAGE_REL_AMD64_SECREL : COFF::IMAGE_REL_I386_SECREL;
Rafael Espindola908d2ed2011-12-24 02:14:02 +000070 default:
71 llvm_unreachable("unsupported relocation type");
72 }
73}
74
75MCObjectWriter *llvm::createX86WinCOFFObjectWriter(raw_ostream &OS,
76 bool Is64Bit) {
77 MCWinCOFFObjectTargetWriter *MOTW = new X86WinCOFFObjectWriter(Is64Bit);
78 return createWinCOFFObjectWriter(MOTW, OS);
79}