mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-27 13:30:05 +00:00
[yaml2obj] Add support for sh_link via Link
key.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@184022 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
5aee09da12
commit
fe57e347a5
@ -55,6 +55,7 @@ struct Section {
|
|||||||
ELF_SHF Flags;
|
ELF_SHF Flags;
|
||||||
llvm::yaml::Hex64 Address;
|
llvm::yaml::Hex64 Address;
|
||||||
object::yaml::BinaryRef Content;
|
object::yaml::BinaryRef Content;
|
||||||
|
StringRef Link;
|
||||||
llvm::yaml::Hex64 AddressAlign;
|
llvm::yaml::Hex64 AddressAlign;
|
||||||
};
|
};
|
||||||
struct Object {
|
struct Object {
|
||||||
|
@ -267,6 +267,7 @@ void MappingTraits<ELFYAML::Section>::mapping(IO &IO,
|
|||||||
IO.mapOptional("Flags", Section.Flags, ELFYAML::ELF_SHF(0));
|
IO.mapOptional("Flags", Section.Flags, ELFYAML::ELF_SHF(0));
|
||||||
IO.mapOptional("Address", Section.Address, Hex64(0));
|
IO.mapOptional("Address", Section.Address, Hex64(0));
|
||||||
IO.mapOptional("Content", Section.Content);
|
IO.mapOptional("Content", Section.Content);
|
||||||
|
IO.mapOptional("Link", Section.Link);
|
||||||
IO.mapOptional("AddressAlign", Section.AddressAlign, Hex64(0));
|
IO.mapOptional("AddressAlign", Section.AddressAlign, Hex64(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -10,6 +10,7 @@ Sections:
|
|||||||
Type: SHT_PROGBITS
|
Type: SHT_PROGBITS
|
||||||
Flags: [ SHF_ALLOC, SHF_EXECINSTR ]
|
Flags: [ SHF_ALLOC, SHF_EXECINSTR ]
|
||||||
Address: 0xCAFEBABE
|
Address: 0xCAFEBABE
|
||||||
|
Link: .text # Doesn't make sense for SHT_PROGBITS, but good enough for test.
|
||||||
Content: EBFE
|
Content: EBFE
|
||||||
AddressAlign: 2
|
AddressAlign: 2
|
||||||
|
|
||||||
@ -26,6 +27,8 @@ Sections:
|
|||||||
# CHECK-NEXT: ]
|
# CHECK-NEXT: ]
|
||||||
# CHECK-NEXT: Address: 0xCAFEBABE
|
# CHECK-NEXT: Address: 0xCAFEBABE
|
||||||
# CHECK: Size: 2
|
# CHECK: Size: 2
|
||||||
|
# Check that Link != 0.
|
||||||
|
# CHECK: Link: {{[1-9][0-9]*}}
|
||||||
# CHECK: AddressAlignment: 2
|
# CHECK: AddressAlignment: 2
|
||||||
# CHECK: SectionData (
|
# CHECK: SectionData (
|
||||||
# CHECK-NEXT: 0000: EBFE
|
# CHECK-NEXT: 0000: EBFE
|
||||||
|
@ -76,6 +76,29 @@ public:
|
|||||||
void writeBlobToStream(raw_ostream &Out) { Out << OS.str(); }
|
void writeBlobToStream(raw_ostream &Out) { Out << OS.str(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Used to keep track of section names, so that in the YAML file sections
|
||||||
|
// can be referenced by name instead of by index.
|
||||||
|
class SectionNameToIdxMap {
|
||||||
|
StringMap<int> Map;
|
||||||
|
public:
|
||||||
|
/// \returns true if name is already present in the map.
|
||||||
|
bool addName(StringRef SecName, unsigned i) {
|
||||||
|
StringMapEntry<int> &Entry = Map.GetOrCreateValue(SecName, -1);
|
||||||
|
if (Entry.getValue() != -1)
|
||||||
|
return true;
|
||||||
|
Entry.setValue((int)i);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
/// \returns true if name is not present in the map
|
||||||
|
bool lookupSection(StringRef SecName, unsigned &Idx) const {
|
||||||
|
StringMap<int>::const_iterator I = Map.find(SecName);
|
||||||
|
if (I == Map.end())
|
||||||
|
return true;
|
||||||
|
Idx = I->getValue();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
template <class T>
|
template <class T>
|
||||||
static size_t vectorDataSize(const std::vector<T> &Vec) {
|
static size_t vectorDataSize(const std::vector<T> &Vec) {
|
||||||
return Vec.size() * sizeof(T);
|
return Vec.size() * sizeof(T);
|
||||||
@ -138,6 +161,18 @@ static void writeELF(raw_ostream &OS, const ELFYAML::Object &Doc) {
|
|||||||
// Place section header string table last.
|
// Place section header string table last.
|
||||||
Header.e_shstrndx = Sections.size();
|
Header.e_shstrndx = Sections.size();
|
||||||
|
|
||||||
|
SectionNameToIdxMap SN2I;
|
||||||
|
for (unsigned i = 0, e = Sections.size(); i != e; ++i) {
|
||||||
|
StringRef Name = Sections[i].Name;
|
||||||
|
if (Name.empty())
|
||||||
|
continue;
|
||||||
|
if (SN2I.addName(Name, i)) {
|
||||||
|
errs() << "error: Repeated section name: '" << Name
|
||||||
|
<< "' at YAML section number " << i << ".\n";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
StringTableBuilder StrTab;
|
StringTableBuilder StrTab;
|
||||||
SmallVector<char, 128> Buf;
|
SmallVector<char, 128> Buf;
|
||||||
// XXX: This offset is tightly coupled with the order that we write
|
// XXX: This offset is tightly coupled with the order that we write
|
||||||
@ -159,7 +194,15 @@ static void writeELF(raw_ostream &OS, const ELFYAML::Object &Doc) {
|
|||||||
SHeader.sh_size = Sec.Content.binary_size();
|
SHeader.sh_size = Sec.Content.binary_size();
|
||||||
Sec.Content.writeAsBinary(CBA.getOS());
|
Sec.Content.writeAsBinary(CBA.getOS());
|
||||||
|
|
||||||
SHeader.sh_link = 0;
|
if (!Sec.Link.empty()) {
|
||||||
|
unsigned Index;
|
||||||
|
if (SN2I.lookupSection(Sec.Link, Index)) {
|
||||||
|
errs() << "error: Unknown section referenced: '" << Sec.Link
|
||||||
|
<< "' at YAML section number " << i << ".\n";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
SHeader.sh_link = Index;
|
||||||
|
}
|
||||||
SHeader.sh_info = 0;
|
SHeader.sh_info = 0;
|
||||||
SHeader.sh_addralign = Sec.AddressAlign;
|
SHeader.sh_addralign = Sec.AddressAlign;
|
||||||
SHeader.sh_entsize = 0;
|
SHeader.sh_entsize = 0;
|
||||||
|
Loading…
Reference in New Issue
Block a user