mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-10 02:36:06 +00:00
b138caba43
This makes the buffer ownership on error conditions very natural. The buffer is only moved out of the argument if an object is constructed that now owns the buffer. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@211546 91177308-0d34-0410-b5e6-96231b3b80d8
86 lines
2.8 KiB
C++
86 lines
2.8 KiB
C++
//===- ELFObjectFile.cpp - ELF object file implementation -------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// Part of the ELFObjectFile class implementation.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/Object/ELFObjectFile.h"
|
|
#include "llvm/Support/MathExtras.h"
|
|
|
|
namespace llvm {
|
|
using namespace object;
|
|
|
|
ErrorOr<ObjectFile *>
|
|
ObjectFile::createELFObjectFile(std::unique_ptr<MemoryBuffer> &Obj) {
|
|
std::pair<unsigned char, unsigned char> Ident = getElfArchType(Obj.get());
|
|
std::size_t MaxAlignment =
|
|
1ULL << countTrailingZeros(uintptr_t(Obj->getBufferStart()));
|
|
|
|
std::error_code EC;
|
|
std::unique_ptr<ObjectFile> R;
|
|
if (Ident.first == ELF::ELFCLASS32 && Ident.second == ELF::ELFDATA2LSB)
|
|
#if !LLVM_IS_UNALIGNED_ACCESS_FAST
|
|
if (MaxAlignment >= 4)
|
|
R.reset(new ELFObjectFile<ELFType<support::little, 4, false>>(
|
|
Obj.release(), EC));
|
|
else
|
|
#endif
|
|
if (MaxAlignment >= 2)
|
|
R.reset(new ELFObjectFile<ELFType<support::little, 2, false>>(
|
|
Obj.release(), EC));
|
|
else
|
|
return object_error::parse_failed;
|
|
else if (Ident.first == ELF::ELFCLASS32 && Ident.second == ELF::ELFDATA2MSB)
|
|
#if !LLVM_IS_UNALIGNED_ACCESS_FAST
|
|
if (MaxAlignment >= 4)
|
|
R.reset(new ELFObjectFile<ELFType<support::big, 4, false>>(Obj.release(),
|
|
EC));
|
|
else
|
|
#endif
|
|
if (MaxAlignment >= 2)
|
|
R.reset(new ELFObjectFile<ELFType<support::big, 2, false>>(Obj.release(),
|
|
EC));
|
|
else
|
|
return object_error::parse_failed;
|
|
else if (Ident.first == ELF::ELFCLASS64 && Ident.second == ELF::ELFDATA2MSB)
|
|
#if !LLVM_IS_UNALIGNED_ACCESS_FAST
|
|
if (MaxAlignment >= 8)
|
|
R.reset(
|
|
new ELFObjectFile<ELFType<support::big, 8, true>>(Obj.release(), EC));
|
|
else
|
|
#endif
|
|
if (MaxAlignment >= 2)
|
|
R.reset(
|
|
new ELFObjectFile<ELFType<support::big, 2, true>>(Obj.release(), EC));
|
|
else
|
|
return object_error::parse_failed;
|
|
else if (Ident.first == ELF::ELFCLASS64 && Ident.second == ELF::ELFDATA2LSB) {
|
|
#if !LLVM_IS_UNALIGNED_ACCESS_FAST
|
|
if (MaxAlignment >= 8)
|
|
R.reset(new ELFObjectFile<ELFType<support::little, 8, true>>(
|
|
Obj.release(), EC));
|
|
else
|
|
#endif
|
|
if (MaxAlignment >= 2)
|
|
R.reset(new ELFObjectFile<ELFType<support::little, 2, true>>(
|
|
Obj.release(), EC));
|
|
else
|
|
return object_error::parse_failed;
|
|
}
|
|
else
|
|
llvm_unreachable("Buffer is not an ELF object file!");
|
|
|
|
if (EC)
|
|
return EC;
|
|
return R.release();
|
|
}
|
|
|
|
} // end namespace llvm
|