361 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			361 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===-- XCOFFYAML.cpp - XCOFF YAMLIO implementation -------------*- C++ -*-===//
 | |
| //
 | |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | |
| // See https://llvm.org/LICENSE.txt for license information.
 | |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| // This file defines classes for handling the YAML representation of XCOFF.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "llvm/ObjectYAML/XCOFFYAML.h"
 | |
| #include "llvm/BinaryFormat/XCOFF.h"
 | |
| #include <string.h>
 | |
| 
 | |
| namespace llvm {
 | |
| namespace XCOFFYAML {
 | |
| 
 | |
| Object::Object() { memset(&Header, 0, sizeof(Header)); }
 | |
| 
 | |
| AuxSymbolEnt::~AuxSymbolEnt() = default;
 | |
| 
 | |
| } // namespace XCOFFYAML
 | |
| 
 | |
| namespace yaml {
 | |
| 
 | |
| void ScalarBitSetTraits<XCOFF::SectionTypeFlags>::bitset(
 | |
|     IO &IO, XCOFF::SectionTypeFlags &Value) {
 | |
| #define ECase(X) IO.bitSetCase(Value, #X, XCOFF::X)
 | |
|   ECase(STYP_PAD);
 | |
|   ECase(STYP_DWARF);
 | |
|   ECase(STYP_TEXT);
 | |
|   ECase(STYP_DATA);
 | |
|   ECase(STYP_BSS);
 | |
|   ECase(STYP_EXCEPT);
 | |
|   ECase(STYP_INFO);
 | |
|   ECase(STYP_TDATA);
 | |
|   ECase(STYP_TBSS);
 | |
|   ECase(STYP_LOADER);
 | |
|   ECase(STYP_DEBUG);
 | |
|   ECase(STYP_TYPCHK);
 | |
|   ECase(STYP_OVRFLO);
 | |
| #undef ECase
 | |
| }
 | |
| 
 | |
| void ScalarEnumerationTraits<XCOFF::StorageClass>::enumeration(
 | |
|     IO &IO, XCOFF::StorageClass &Value) {
 | |
| #define ECase(X) IO.enumCase(Value, #X, XCOFF::X)
 | |
|   ECase(C_NULL);
 | |
|   ECase(C_AUTO);
 | |
|   ECase(C_EXT);
 | |
|   ECase(C_STAT);
 | |
|   ECase(C_REG);
 | |
|   ECase(C_EXTDEF);
 | |
|   ECase(C_LABEL);
 | |
|   ECase(C_ULABEL);
 | |
|   ECase(C_MOS);
 | |
|   ECase(C_ARG);
 | |
|   ECase(C_STRTAG);
 | |
|   ECase(C_MOU);
 | |
|   ECase(C_UNTAG);
 | |
|   ECase(C_TPDEF);
 | |
|   ECase(C_USTATIC);
 | |
|   ECase(C_ENTAG);
 | |
|   ECase(C_MOE);
 | |
|   ECase(C_REGPARM);
 | |
|   ECase(C_FIELD);
 | |
|   ECase(C_BLOCK);
 | |
|   ECase(C_FCN);
 | |
|   ECase(C_EOS);
 | |
|   ECase(C_FILE);
 | |
|   ECase(C_LINE);
 | |
|   ECase(C_ALIAS);
 | |
|   ECase(C_HIDDEN);
 | |
|   ECase(C_HIDEXT);
 | |
|   ECase(C_BINCL);
 | |
|   ECase(C_EINCL);
 | |
|   ECase(C_INFO);
 | |
|   ECase(C_WEAKEXT);
 | |
|   ECase(C_DWARF);
 | |
|   ECase(C_GSYM);
 | |
|   ECase(C_LSYM);
 | |
|   ECase(C_PSYM);
 | |
|   ECase(C_RSYM);
 | |
|   ECase(C_RPSYM);
 | |
|   ECase(C_STSYM);
 | |
|   ECase(C_TCSYM);
 | |
|   ECase(C_BCOMM);
 | |
|   ECase(C_ECOML);
 | |
|   ECase(C_ECOMM);
 | |
|   ECase(C_DECL);
 | |
|   ECase(C_ENTRY);
 | |
|   ECase(C_FUN);
 | |
|   ECase(C_BSTAT);
 | |
|   ECase(C_ESTAT);
 | |
|   ECase(C_GTLS);
 | |
|   ECase(C_STTLS);
 | |
|   ECase(C_EFCN);
 | |
| #undef ECase
 | |
| }
 | |
| 
 | |
| void ScalarEnumerationTraits<XCOFF::StorageMappingClass>::enumeration(
 | |
|     IO &IO, XCOFF::StorageMappingClass &Value) {
 | |
| #define ECase(X) IO.enumCase(Value, #X, XCOFF::X)
 | |
|   ECase(XMC_PR);
 | |
|   ECase(XMC_RO);
 | |
|   ECase(XMC_DB);
 | |
|   ECase(XMC_GL);
 | |
|   ECase(XMC_XO);
 | |
|   ECase(XMC_SV);
 | |
|   ECase(XMC_SV64);
 | |
|   ECase(XMC_SV3264);
 | |
|   ECase(XMC_TI);
 | |
|   ECase(XMC_TB);
 | |
|   ECase(XMC_RW);
 | |
|   ECase(XMC_TC0);
 | |
|   ECase(XMC_TC);
 | |
|   ECase(XMC_TD);
 | |
|   ECase(XMC_DS);
 | |
|   ECase(XMC_UA);
 | |
|   ECase(XMC_BS);
 | |
|   ECase(XMC_UC);
 | |
|   ECase(XMC_TL);
 | |
|   ECase(XMC_UL);
 | |
|   ECase(XMC_TE);
 | |
| #undef ECase
 | |
| }
 | |
| 
 | |
| void ScalarEnumerationTraits<XCOFFYAML::AuxSymbolType>::enumeration(
 | |
|     IO &IO, XCOFFYAML::AuxSymbolType &Type) {
 | |
| #define ECase(X) IO.enumCase(Type, #X, XCOFFYAML::X)
 | |
|   ECase(AUX_EXCEPT);
 | |
|   ECase(AUX_FCN);
 | |
|   ECase(AUX_SYM);
 | |
|   ECase(AUX_FILE);
 | |
|   ECase(AUX_CSECT);
 | |
|   ECase(AUX_SECT);
 | |
|   ECase(AUX_STAT);
 | |
| #undef ECase
 | |
| }
 | |
| 
 | |
| void ScalarEnumerationTraits<XCOFF::CFileStringType>::enumeration(
 | |
|     IO &IO, XCOFF::CFileStringType &Type) {
 | |
| #define ECase(X) IO.enumCase(Type, #X, XCOFF::X)
 | |
|   ECase(XFT_FN);
 | |
|   ECase(XFT_CT);
 | |
|   ECase(XFT_CV);
 | |
|   ECase(XFT_CD);
 | |
| #undef ECase
 | |
| }
 | |
| 
 | |
| struct NSectionFlags {
 | |
|   NSectionFlags(IO &) : Flags(XCOFF::SectionTypeFlags(0)) {}
 | |
|   NSectionFlags(IO &, uint32_t C) : Flags(XCOFF::SectionTypeFlags(C)) {}
 | |
| 
 | |
|   uint32_t denormalize(IO &) { return Flags; }
 | |
| 
 | |
|   XCOFF::SectionTypeFlags Flags;
 | |
| };
 | |
| 
 | |
| void MappingTraits<XCOFFYAML::FileHeader>::mapping(
 | |
|     IO &IO, XCOFFYAML::FileHeader &FileHdr) {
 | |
|   IO.mapOptional("MagicNumber", FileHdr.Magic);
 | |
|   IO.mapOptional("NumberOfSections", FileHdr.NumberOfSections);
 | |
|   IO.mapOptional("CreationTime", FileHdr.TimeStamp);
 | |
|   IO.mapOptional("OffsetToSymbolTable", FileHdr.SymbolTableOffset);
 | |
|   IO.mapOptional("EntriesInSymbolTable", FileHdr.NumberOfSymTableEntries);
 | |
|   IO.mapOptional("AuxiliaryHeaderSize", FileHdr.AuxHeaderSize);
 | |
|   IO.mapOptional("Flags", FileHdr.Flags);
 | |
| }
 | |
| 
 | |
| void MappingTraits<XCOFFYAML::AuxiliaryHeader>::mapping(
 | |
|     IO &IO, XCOFFYAML::AuxiliaryHeader &AuxHdr) {
 | |
|   IO.mapOptional("Magic", AuxHdr.Magic);
 | |
|   IO.mapOptional("Version", AuxHdr.Version);
 | |
|   IO.mapOptional("TextStartAddr", AuxHdr.TextStartAddr);
 | |
|   IO.mapOptional("DataStartAddr", AuxHdr.DataStartAddr);
 | |
|   IO.mapOptional("TOCAnchorAddr", AuxHdr.TOCAnchorAddr);
 | |
|   IO.mapOptional("TextSectionSize", AuxHdr.TextSize);
 | |
|   IO.mapOptional("DataSectionSize", AuxHdr.InitDataSize);
 | |
|   IO.mapOptional("BssSectionSize", AuxHdr.BssDataSize);
 | |
|   IO.mapOptional("SecNumOfEntryPoint", AuxHdr.SecNumOfEntryPoint);
 | |
|   IO.mapOptional("SecNumOfText", AuxHdr.SecNumOfText);
 | |
|   IO.mapOptional("SecNumOfData", AuxHdr.SecNumOfData);
 | |
|   IO.mapOptional("SecNumOfTOC", AuxHdr.SecNumOfTOC);
 | |
|   IO.mapOptional("SecNumOfLoader", AuxHdr.SecNumOfLoader);
 | |
|   IO.mapOptional("SecNumOfBSS", AuxHdr.SecNumOfBSS);
 | |
|   IO.mapOptional("MaxAlignOfText", AuxHdr.MaxAlignOfText);
 | |
|   IO.mapOptional("MaxAlignOfData", AuxHdr.MaxAlignOfData);
 | |
|   IO.mapOptional("ModuleType", AuxHdr.CpuFlag);
 | |
|   IO.mapOptional("TextPageSize", AuxHdr.TextPageSize);
 | |
|   IO.mapOptional("DataPageSize", AuxHdr.DataPageSize);
 | |
|   IO.mapOptional("StackPageSize", AuxHdr.StackPageSize);
 | |
|   IO.mapOptional("FlagAndTDataAlignment", AuxHdr.FlagAndTDataAlignment);
 | |
|   IO.mapOptional("EntryPointAddr", AuxHdr.EntryPointAddr);
 | |
|   IO.mapOptional("MaxStackSize", AuxHdr.MaxStackSize);
 | |
|   IO.mapOptional("MaxDataSize", AuxHdr.MaxDataSize);
 | |
|   IO.mapOptional("SecNumOfTData", AuxHdr.SecNumOfTData);
 | |
|   IO.mapOptional("SecNumOfTBSS", AuxHdr.SecNumOfTBSS);
 | |
|   IO.mapOptional("Flag", AuxHdr.Flag);
 | |
| }
 | |
| 
 | |
| void MappingTraits<XCOFFYAML::Relocation>::mapping(IO &IO,
 | |
|                                                    XCOFFYAML::Relocation &R) {
 | |
|   IO.mapOptional("Address", R.VirtualAddress);
 | |
|   IO.mapOptional("Symbol", R.SymbolIndex);
 | |
|   IO.mapOptional("Info", R.Info);
 | |
|   IO.mapOptional("Type", R.Type);
 | |
| }
 | |
| 
 | |
| void MappingTraits<XCOFFYAML::Section>::mapping(IO &IO,
 | |
|                                                 XCOFFYAML::Section &Sec) {
 | |
|   MappingNormalization<NSectionFlags, uint32_t> NC(IO, Sec.Flags);
 | |
|   IO.mapOptional("Name", Sec.SectionName);
 | |
|   IO.mapOptional("Address", Sec.Address);
 | |
|   IO.mapOptional("Size", Sec.Size);
 | |
|   IO.mapOptional("FileOffsetToData", Sec.FileOffsetToData);
 | |
|   IO.mapOptional("FileOffsetToRelocations", Sec.FileOffsetToRelocations);
 | |
|   IO.mapOptional("FileOffsetToLineNumbers", Sec.FileOffsetToLineNumbers);
 | |
|   IO.mapOptional("NumberOfRelocations", Sec.NumberOfRelocations);
 | |
|   IO.mapOptional("NumberOfLineNumbers", Sec.NumberOfLineNumbers);
 | |
|   IO.mapOptional("Flags", NC->Flags);
 | |
|   IO.mapOptional("SectionData", Sec.SectionData);
 | |
|   IO.mapOptional("Relocations", Sec.Relocations);
 | |
| }
 | |
| 
 | |
| static void auxSymMapping(IO &IO, XCOFFYAML::CsectAuxEnt &AuxSym, bool Is64) {
 | |
|   IO.mapOptional("ParameterHashIndex", AuxSym.ParameterHashIndex);
 | |
|   IO.mapOptional("TypeChkSectNum", AuxSym.TypeChkSectNum);
 | |
|   IO.mapOptional("SymbolAlignmentAndType", AuxSym.SymbolAlignmentAndType);
 | |
|   IO.mapOptional("StorageMappingClass", AuxSym.StorageMappingClass);
 | |
|   if (Is64) {
 | |
|     IO.mapOptional("SectionOrLengthLo", AuxSym.SectionOrLengthLo);
 | |
|     IO.mapOptional("SectionOrLengthHi", AuxSym.SectionOrLengthHi);
 | |
|   } else {
 | |
|     IO.mapOptional("SectionOrLength", AuxSym.SectionOrLength);
 | |
|     IO.mapOptional("StabInfoIndex", AuxSym.StabInfoIndex);
 | |
|     IO.mapOptional("StabSectNum", AuxSym.StabSectNum);
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void auxSymMapping(IO &IO, XCOFFYAML::FileAuxEnt &AuxSym) {
 | |
|   IO.mapOptional("FileNameOrString", AuxSym.FileNameOrString);
 | |
|   IO.mapOptional("FileStringType", AuxSym.FileStringType);
 | |
| }
 | |
| 
 | |
| static void auxSymMapping(IO &IO, XCOFFYAML::BlockAuxEnt &AuxSym, bool Is64) {
 | |
|   if (Is64) {
 | |
|     IO.mapOptional("LineNum", AuxSym.LineNum);
 | |
|   } else {
 | |
|     IO.mapOptional("LineNumHi", AuxSym.LineNumHi);
 | |
|     IO.mapOptional("LineNumLo", AuxSym.LineNumLo);
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void auxSymMapping(IO &IO, XCOFFYAML::FunctionAuxEnt &AuxSym,
 | |
|                           bool Is64) {
 | |
|   if (!Is64)
 | |
|     IO.mapOptional("OffsetToExceptionTbl", AuxSym.OffsetToExceptionTbl);
 | |
|   IO.mapOptional("SizeOfFunction", AuxSym.SizeOfFunction);
 | |
|   IO.mapOptional("SymIdxOfNextBeyond", AuxSym.SymIdxOfNextBeyond);
 | |
|   IO.mapOptional("PtrToLineNum", AuxSym.PtrToLineNum);
 | |
| }
 | |
| 
 | |
| static void auxSymMapping(IO &IO, XCOFFYAML::ExcpetionAuxEnt &AuxSym) {
 | |
|   IO.mapOptional("OffsetToExceptionTbl", AuxSym.OffsetToExceptionTbl);
 | |
|   IO.mapOptional("SizeOfFunction", AuxSym.SizeOfFunction);
 | |
|   IO.mapOptional("SymIdxOfNextBeyond", AuxSym.SymIdxOfNextBeyond);
 | |
| }
 | |
| 
 | |
| static void auxSymMapping(IO &IO, XCOFFYAML::SectAuxEntForDWARF &AuxSym) {
 | |
|   IO.mapOptional("LengthOfSectionPortion", AuxSym.LengthOfSectionPortion);
 | |
|   IO.mapOptional("NumberOfRelocEnt", AuxSym.NumberOfRelocEnt);
 | |
| }
 | |
| 
 | |
| static void auxSymMapping(IO &IO, XCOFFYAML::SectAuxEntForStat &AuxSym) {
 | |
|   IO.mapOptional("SectionLength", AuxSym.SectionLength);
 | |
|   IO.mapOptional("NumberOfRelocEnt", AuxSym.NumberOfRelocEnt);
 | |
|   IO.mapOptional("NumberOfLineNum", AuxSym.NumberOfLineNum);
 | |
| }
 | |
| 
 | |
| void MappingTraits<std::unique_ptr<XCOFFYAML::AuxSymbolEnt>>::mapping(
 | |
|     IO &IO, std::unique_ptr<XCOFFYAML::AuxSymbolEnt> &AuxSym) {
 | |
|   assert(!IO.outputting() && "We don't dump aux symbols currently.");
 | |
|   const bool Is64 =
 | |
|       static_cast<XCOFFYAML::Object *>(IO.getContext())->Header.Magic ==
 | |
|       (llvm::yaml::Hex16)XCOFF::XCOFF64;
 | |
|   XCOFFYAML::AuxSymbolType AuxType;
 | |
|   IO.mapRequired("Type", AuxType);
 | |
|   switch (AuxType) {
 | |
|   case XCOFFYAML::AUX_EXCEPT:
 | |
|     if (!Is64)
 | |
|       IO.setError("an auxiliary symbol of type AUX_EXCEPT cannot be defined in "
 | |
|                   "XCOFF32");
 | |
|     AuxSym.reset(new XCOFFYAML::ExcpetionAuxEnt());
 | |
|     auxSymMapping(IO, *cast<XCOFFYAML::ExcpetionAuxEnt>(AuxSym.get()));
 | |
|     break;
 | |
|   case XCOFFYAML::AUX_FCN:
 | |
|     AuxSym.reset(new XCOFFYAML::FunctionAuxEnt());
 | |
|     auxSymMapping(IO, *cast<XCOFFYAML::FunctionAuxEnt>(AuxSym.get()), Is64);
 | |
|     break;
 | |
|   case XCOFFYAML::AUX_SYM:
 | |
|     AuxSym.reset(new XCOFFYAML::BlockAuxEnt());
 | |
|     auxSymMapping(IO, *cast<XCOFFYAML::BlockAuxEnt>(AuxSym.get()), Is64);
 | |
|     break;
 | |
|   case XCOFFYAML::AUX_FILE:
 | |
|     AuxSym.reset(new XCOFFYAML::FileAuxEnt());
 | |
|     auxSymMapping(IO, *cast<XCOFFYAML::FileAuxEnt>(AuxSym.get()));
 | |
|     break;
 | |
|   case XCOFFYAML::AUX_CSECT:
 | |
|     AuxSym.reset(new XCOFFYAML::CsectAuxEnt());
 | |
|     auxSymMapping(IO, *cast<XCOFFYAML::CsectAuxEnt>(AuxSym.get()), Is64);
 | |
|     break;
 | |
|   case XCOFFYAML::AUX_SECT:
 | |
|     AuxSym.reset(new XCOFFYAML::SectAuxEntForDWARF());
 | |
|     auxSymMapping(IO, *cast<XCOFFYAML::SectAuxEntForDWARF>(AuxSym.get()));
 | |
|     break;
 | |
|   case XCOFFYAML::AUX_STAT:
 | |
|     if (Is64)
 | |
|       IO.setError(
 | |
|           "an auxiliary symbol of type AUX_STAT cannot be defined in XCOFF64");
 | |
|     AuxSym.reset(new XCOFFYAML::SectAuxEntForStat());
 | |
|     auxSymMapping(IO, *cast<XCOFFYAML::SectAuxEntForStat>(AuxSym.get()));
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MappingTraits<XCOFFYAML::Symbol>::mapping(IO &IO, XCOFFYAML::Symbol &S) {
 | |
|   IO.mapOptional("Name", S.SymbolName);
 | |
|   IO.mapOptional("Value", S.Value);
 | |
|   IO.mapOptional("Section", S.SectionName);
 | |
|   IO.mapOptional("SectionIndex", S.SectionIndex);
 | |
|   IO.mapOptional("Type", S.Type);
 | |
|   IO.mapOptional("StorageClass", S.StorageClass);
 | |
|   IO.mapOptional("NumberOfAuxEntries", S.NumberOfAuxEntries);
 | |
|   if (!IO.outputting())
 | |
|     IO.mapOptional("AuxEntries", S.AuxEntries);
 | |
| }
 | |
| 
 | |
| void MappingTraits<XCOFFYAML::StringTable>::mapping(IO &IO, XCOFFYAML::StringTable &Str) {
 | |
|   IO.mapOptional("ContentSize", Str.ContentSize);
 | |
|   IO.mapOptional("Length", Str.Length);
 | |
|   IO.mapOptional("Strings", Str.Strings);
 | |
|   IO.mapOptional("RawContent", Str.RawContent);
 | |
| }
 | |
| 
 | |
| void MappingTraits<XCOFFYAML::Object>::mapping(IO &IO, XCOFFYAML::Object &Obj) {
 | |
|   IO.setContext(&Obj);
 | |
|   IO.mapTag("!XCOFF", true);
 | |
|   IO.mapRequired("FileHeader", Obj.Header);
 | |
|   IO.mapOptional("AuxiliaryHeader", Obj.AuxHeader);
 | |
|   IO.mapOptional("Sections", Obj.Sections);
 | |
|   IO.mapOptional("Symbols", Obj.Symbols);
 | |
|   IO.mapOptional("StringTable", Obj.StrTbl);
 | |
|   IO.setContext(nullptr);
 | |
| }
 | |
| 
 | |
| } // namespace yaml
 | |
| } // namespace llvm
 |