[llvm] [llvm-objcopy] MachO: Fix section finding policy for object files (PR #127604)

Dmitry Nechitaev via llvm-commits llvm-commits at lists.llvm.org
Thu Feb 20 01:29:33 PST 2025


https://github.com/Nechda updated https://github.com/llvm/llvm-project/pull/127604

>From 24331ec175410f92c5ef4c1eee518e22847c79d5 Mon Sep 17 00:00:00 2001
From: Dmitry Nechitaev <nechda6 at gmail.com>
Date: Tue, 18 Feb 2025 12:29:56 +0300
Subject: [PATCH 1/5] llvm-objcopy MachO: scan each section when
 --update-section option is used

---
 llvm/lib/ObjCopy/MachO/MachOObjcopy.cpp | 6 ++++++
 1 file changed, 6 insertions(+)

diff --git a/llvm/lib/ObjCopy/MachO/MachOObjcopy.cpp b/llvm/lib/ObjCopy/MachO/MachOObjcopy.cpp
index 682edffc84f34..b5ddaa5666229 100644
--- a/llvm/lib/ObjCopy/MachO/MachOObjcopy.cpp
+++ b/llvm/lib/ObjCopy/MachO/MachOObjcopy.cpp
@@ -360,6 +360,12 @@ static Error addSection(const NewSectionInfo &NewSection, Object &Obj) {
 static Expected<Section &> findSection(StringRef SecName, Object &O) {
   StringRef SegName;
   std::tie(SegName, SecName) = SecName.split(",");
+  if (O.Header.FileType == MachO::HeaderFileType::MH_OBJECT) {
+    for (const auto& LC : O.LoadCommands)
+      for (const auto& Sec : LC.Sections)
+        if (Sec->Segname == SegName && Sec->Sectname == SecName)
+          return *Sec;
+  }
   auto FoundSeg =
       llvm::find_if(O.LoadCommands, [SegName](const LoadCommand &LC) {
         return LC.getSegmentName() == SegName;

>From dc6cbcb1ae917e6be165c1382a1aa9e8bca4776e Mon Sep 17 00:00:00 2001
From: Dmitry Nechitaev <nechda6 at gmail.com>
Date: Thu, 20 Feb 2025 00:01:48 +0300
Subject: [PATCH 2/5] Add test

---
 .../MachO/Inputs/macho_sections.s             | 30 ++++++++++++
 .../MachO/update-section-object.test          | 48 +++++++++++++++++++
 2 files changed, 78 insertions(+)
 create mode 100644 llvm/test/tools/llvm-objcopy/MachO/Inputs/macho_sections.s
 create mode 100644 llvm/test/tools/llvm-objcopy/MachO/update-section-object.test

diff --git a/llvm/test/tools/llvm-objcopy/MachO/Inputs/macho_sections.s b/llvm/test/tools/llvm-objcopy/MachO/Inputs/macho_sections.s
new file mode 100644
index 0000000000000..0818d3544e8fc
--- /dev/null
+++ b/llvm/test/tools/llvm-objcopy/MachO/Inputs/macho_sections.s
@@ -0,0 +1,30 @@
+	.section	__TEXT,__text,regular,pure_instructions
+	.build_version macos, 11, 0
+	.globl	_main                           ; -- Begin function main
+	.p2align	2
+_main:                                  ; @main
+	.cfi_startproc
+; %bb.0:
+	sub	sp, sp, #32
+	stp	x29, x30, [sp, #16]             ; 16-byte Folded Spill
+	add	x29, sp, #16
+	.cfi_def_cfa w29, 16
+	.cfi_offset w30, -8
+	.cfi_offset w29, -16
+	mov	w8, #0                          ; =0x0
+	str	w8, [sp, #8]                    ; 4-byte Folded Spill
+	stur	wzr, [x29, #-4]
+	adrp	x0, __ZL7storage at PAGE
+	add	x0, x0, __ZL7storage at PAGEOFF
+	bl	__Z3fooPKc
+	ldr	w0, [sp, #8]                    ; 4-byte Folded Reload
+	ldp	x29, x30, [sp, #16]             ; 16-byte Folded Reload
+	add	sp, sp, #32
+	ret
+	.cfi_endproc
+                                        ; -- End function
+	.section	__DATA,__storage
+__ZL7storage:                           ; @_ZL7storage
+	.space	1383                        ; 0x567
+
+.subsections_via_symbols
diff --git a/llvm/test/tools/llvm-objcopy/MachO/update-section-object.test b/llvm/test/tools/llvm-objcopy/MachO/update-section-object.test
new file mode 100644
index 0000000000000..8db7c821ff447
--- /dev/null
+++ b/llvm/test/tools/llvm-objcopy/MachO/update-section-object.test
@@ -0,0 +1,48 @@
+# REQUIRES: aarch64-registered-target
+
+# RUN: llvm-mc -assemble -triple=arm64-apple-macos11 -filetype=obj %p/Inputs/macho_sections.s -o %t.o
+# RUN: llvm-otool -l %t.o | FileCheck %s --check-prefix=ORIG
+
+
+# RUN: llvm-objcopy %t.o --update-section __DATA,__storage=%p/Inputs/macho_sections.s %t.new.o
+# RUN: llvm-otool -l %t.new.o | FileCheck %s --check-prefix=UPDATED
+
+
+# ORIG:      cmd LC_SEGMENT_64
+# ORIG-NEXT: cmdsize 312
+# ORIG-NEXT: segname
+# ORIG-NEXT: vmaddr 0x0000000000000000
+# ORIG-NEXT: vmsize 0x00000000000005c0
+# ORIG-NEXT: fileoff 472
+# ORIG-NEXT: filesize 1472
+# ORIG-NEXT: maxprot 0x00000007
+# ORIG-NEXT: initprot 0x00000007
+# ORIG-NEXT: nsects 3
+# ORIG-NEXT: flags 0x0
+
+# ORIG:      Section
+# ORIG:      sectname __storage
+# ORIG-NEXT: segname __DATA
+# ORIG-NEXT: addr 0x0000000000000034
+# ORIG-NEXT: size 0x0000000000000567
+
+
+### Make sure the file size and segment size have changed
+# UPDATED:      cmd LC_SEGMENT_64
+# UPDATED-NEXT: cmdsize 312
+# UPDATED-NEXT: segname
+# UPDATED-NEXT: vmaddr 0x0000000000000000
+# UPDATED-NEXT: vmsize 0x00000000000005c0
+# UPDATED-NEXT: fileoff 472
+# UPDATED-NEXT: filesize 1048
+# UPDATED-NEXT: maxprot 0x00000007
+# UPDATED-NEXT: initprot 0x00000007
+# UPDATED-NEXT: nsects 3
+# UPDATED-NEXT: flags 0x0
+
+# UPDATED:      Section
+# UPDATED:      sectname __storage
+# UPDATED-NEXT: segname __DATA
+# UPDATED-NEXT: addr 0x0000000000000034
+# UPDATED-NEXT: size 0x00000000000003be
+

>From 4ad313ca31c89ca83937c00528a9f09f5916ec69 Mon Sep 17 00:00:00 2001
From: Dmitry Nechitaev <nechda6 at gmail.com>
Date: Thu, 20 Feb 2025 11:59:09 +0300
Subject: [PATCH 3/5] Add err-msg & ref to doc

---
 llvm/lib/ObjCopy/MachO/MachOObjcopy.cpp | 12 ++++++++++++
 1 file changed, 12 insertions(+)

diff --git a/llvm/lib/ObjCopy/MachO/MachOObjcopy.cpp b/llvm/lib/ObjCopy/MachO/MachOObjcopy.cpp
index b5ddaa5666229..cbc8e471b3090 100644
--- a/llvm/lib/ObjCopy/MachO/MachOObjcopy.cpp
+++ b/llvm/lib/ObjCopy/MachO/MachOObjcopy.cpp
@@ -360,11 +360,23 @@ static Error addSection(const NewSectionInfo &NewSection, Object &Obj) {
 static Expected<Section &> findSection(StringRef SecName, Object &O) {
   StringRef SegName;
   std::tie(SegName, SecName) = SecName.split(",");
+  // For compactness, intermediate object files (MH_OBJECT) contain
+  // only one segment in which all sections are placed.
+  // The static linker places each section in the named segment when building
+  // the final product (any file that is not of type MH_OBJECT).
+  //
+  // Source:
+  // https://math-atlas.sourceforge.net/devel/assembly/MachORuntime.pdf
+  // page 57
   if (O.Header.FileType == MachO::HeaderFileType::MH_OBJECT) {
     for (const auto& LC : O.LoadCommands)
       for (const auto& Sec : LC.Sections)
         if (Sec->Segname == SegName && Sec->Sectname == SecName)
           return *Sec;
+
+    StringRef ErrMsg = "could not find section with name '%s' in '%s' segment";
+    return createStringError(errc::invalid_argument, ErrMsg.str().c_str(),
+                             SecName.str().c_str(), SegName.str().c_str());
   }
   auto FoundSeg =
       llvm::find_if(O.LoadCommands, [SegName](const LoadCommand &LC) {

>From 1a7dc1ab5af601ab31cee070cf6923f442ff695e Mon Sep 17 00:00:00 2001
From: Dmitry Nechitaev <nechda6 at gmail.com>
Date: Thu, 20 Feb 2025 12:03:30 +0300
Subject: [PATCH 4/5] Test err msg

---
 llvm/test/tools/llvm-objcopy/MachO/update-section-object.test | 1 -
 llvm/test/tools/llvm-objcopy/MachO/update-section.test        | 2 +-
 2 files changed, 1 insertion(+), 2 deletions(-)

diff --git a/llvm/test/tools/llvm-objcopy/MachO/update-section-object.test b/llvm/test/tools/llvm-objcopy/MachO/update-section-object.test
index 8db7c821ff447..92c21be49f1ed 100644
--- a/llvm/test/tools/llvm-objcopy/MachO/update-section-object.test
+++ b/llvm/test/tools/llvm-objcopy/MachO/update-section-object.test
@@ -7,7 +7,6 @@
 # RUN: llvm-objcopy %t.o --update-section __DATA,__storage=%p/Inputs/macho_sections.s %t.new.o
 # RUN: llvm-otool -l %t.new.o | FileCheck %s --check-prefix=UPDATED
 
-
 # ORIG:      cmd LC_SEGMENT_64
 # ORIG-NEXT: cmdsize 312
 # ORIG-NEXT: segname
diff --git a/llvm/test/tools/llvm-objcopy/MachO/update-section.test b/llvm/test/tools/llvm-objcopy/MachO/update-section.test
index d9fb23d41474e..4ac7fc72d7f4d 100644
--- a/llvm/test/tools/llvm-objcopy/MachO/update-section.test
+++ b/llvm/test/tools/llvm-objcopy/MachO/update-section.test
@@ -16,7 +16,7 @@
 # RUN: not llvm-objcopy --update-section __TEXT,__text=%t.noexist %t /dev/null
 
 # RUN: not llvm-objcopy --update-section __NOEXIST,__text=%t.diff %t /dev/null 2>&1 | FileCheck %s --check-prefix=NO-SEGMENT
-# NO-SEGMENT: error: {{.*}}could not find segment with name '__NOEXIST'
+# NO-SEGMENT: error: {{.*}}could not find section with name '__text' in '__NOEXIST' segment
 
 # RUN: not llvm-objcopy --update-section __TEXT,__noexist=%t.diff %t /dev/null 2>&1 | FileCheck %s --check-prefix=NO-SECTION
 # NO-SECTION: error: {{.*}}could not find section with name '__noexist'

>From 08e746ad645afb9870f26565e33f08088d87f1c7 Mon Sep 17 00:00:00 2001
From: Dmitry Nechitaev <nechda6 at gmail.com>
Date: Thu, 20 Feb 2025 12:28:47 +0300
Subject: [PATCH 5/5] Add release notes

---
 llvm/docs/ReleaseNotes.md | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/llvm/docs/ReleaseNotes.md b/llvm/docs/ReleaseNotes.md
index 75638d75e70f2..9203e22a9c886 100644
--- a/llvm/docs/ReleaseNotes.md
+++ b/llvm/docs/ReleaseNotes.md
@@ -146,6 +146,8 @@ Changes to the Debug Info
 Changes to the LLVM tools
 ---------------------------------
 
+* llvm-objcopy now supports the `--update-section` flag for intermediate Mach-O object files.
+
 Changes to LLDB
 ---------------------------------
 



More information about the llvm-commits mailing list