Index: llvm/lib/ObjCopy/ELF/ELFObjcopy.cpp =================================================================== --- llvm/lib/ObjCopy/ELF/ELFObjcopy.cpp +++ llvm/lib/ObjCopy/ELF/ELFObjcopy.cpp @@ -97,6 +97,14 @@ return (OldFlags & PreserveMask) | (NewFlags & ~PreserveMask); } +static void setSectionType(SectionBase &Sec, uint64_t Type) { + // If Sec's type is changed from SHT_NOBITS due to --set-section-flags, + // Offset may not be aligned. Align it to max(Align, 1). + if (Sec.Type == ELF::SHT_NOBITS && Type != ELF::SHT_NOBITS) + Sec.Offset = alignTo(Sec.Offset, std::max(Sec.Align, uint64_t(1))); + Sec.Type = Type; +} + static void setSectionFlagsAndType(SectionBase &Sec, SectionFlag Flags) { Sec.Flags = getSectionFlagsPreserveMask(Sec.Flags, getNewShfFlags(Flags)); @@ -106,7 +114,7 @@ if (Sec.Type == SHT_NOBITS && (!(Sec.Flags & ELF::SHF_ALLOC) || Flags & (SectionFlag::SecContents | SectionFlag::SecLoad))) - Sec.Type = SHT_PROGBITS; + setSectionType(Sec, ELF::SHT_PROGBITS); } static ElfType getOutputElfType(const Binary &Bin) { @@ -684,7 +692,7 @@ } auto It2 = Config.SetSectionType.find(Sec.Name); if (It2 != Config.SetSectionType.end()) - Sec.Type = It2->second; + setSectionType(Sec, It2->second); } } Index: llvm/lib/ObjCopy/ELF/ELFObject.cpp =================================================================== --- llvm/lib/ObjCopy/ELF/ELFObject.cpp +++ llvm/lib/ObjCopy/ELF/ELFObject.cpp @@ -2631,12 +2631,9 @@ // MinAddr will be skipped. uint64_t MinAddr = UINT64_MAX; for (SectionBase &Sec : Obj.allocSections()) { - // If Sec's type is changed from SHT_NOBITS due to --set-section-flags, - // Offset may not be aligned. Align it to max(Align, 1). if (Sec.ParentSegment != nullptr) - Sec.Addr = alignTo(Sec.Offset - Sec.ParentSegment->Offset + - Sec.ParentSegment->PAddr, - std::max(Sec.Align, uint64_t(1))); + Sec.Addr = + Sec.Offset - Sec.ParentSegment->Offset + Sec.ParentSegment->PAddr; if (Sec.Type != SHT_NOBITS && Sec.Size > 0) MinAddr = std::min(MinAddr, Sec.Addr); } Index: llvm/test/tools/llvm-objcopy/ELF/binary-paddr.test =================================================================== --- llvm/test/tools/llvm-objcopy/ELF/binary-paddr.test +++ llvm/test/tools/llvm-objcopy/ELF/binary-paddr.test @@ -254,3 +254,68 @@ FileSize: 0x2 MemSize: 0x13 Align: 0x1000 + +## Test that sections do not move relative to their physical addresses if +## the physical address is not aligned. This behavior matches GNU objcopy +## and is important for embedded images where the physical address is +## used to store the initial data image. The loader typically will copy +## this image using a start symbol created by the linker. Since copying is +## done by memcpy, the source alignment is not important but the symbol +## address must match the initial image location. + +# RUN: yaml2obj --docnum=8 %s -o %t8 +# RUN: llvm-objcopy -O binary --only-section=.text --only-section=.data %t8 %t8.out +# RUN: od -A x -t x2 %t8.out | FileCheck %s --check-prefix=PHYSUNALIGNED --ignore-case + +# PHYSUNALIGNED: 000000 2211 4433 6655 8877 aa99 ccbb eedd 00ff +# PHYSUNALIGNED-NEXT: 000010 2211 4433 acbd abcd + +--- !ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_EXEC + Machine: EM_X86_64 +ProgramHeaders: + - Type: PT_LOAD + Flags: [ PF_X, PF_R ] + FirstSec: .text + LastSec: .text + VAddr: 0x200000 + PAddr: 0x200000 + Align: 0x1000 + - Type: PT_LOAD + Flags: [ PF_W, PF_R ] + FirstSec: .data + LastSec: .data + VAddr: 0x400000 + PAddr: 0x200014 + Align: 0x1000 +Sections: + - Name: .text + Type: SHT_PROGBITS + Flags: [ SHF_ALLOC, SHF_EXECINSTR ] + Address: 0x200000 + AddressAlign: 0x1 + Offset: 0x1000 + Content: 112233445566778899aabbccddeeff0011223344 + - Name: .data + Type: SHT_PROGBITS + Flags: [ SHF_WRITE, SHF_ALLOC ] + Address: 0x400000 + AddressAlign: 0x8 + Offset: 0x2000 + Content: BDACCDAB +Symbols: + - Name: INIT_DATA_START + Index: SHN_ABS + Binding: STB_GLOBAL + Value: 0x200014 + - Name: RAM_START + Index: SHN_ABS + Binding: STB_GLOBAL + Value: 0x400000 + - Name: ROM_START + Index: SHN_ABS + Binding: STB_GLOBAL + Value: 0x200000