llvm-project/lld/test/MachO/start-end.s

419 lines
15 KiB
ArmAsm

# REQUIRES: x86
# RUN: rm -rf %t; split-file %s %t
# RUN: llvm-mc -filetype=obj -triple=x86_64-apple-darwin %t/main.s -o %t/main.o
# RUN: llvm-mc -filetype=obj -triple=x86_64-apple-darwin %t/foo.s -o %t/foo.o
# RUN: llvm-mc -filetype=obj -triple=x86_64-apple-darwin %t/seg.s -o %t/seg.o
# RUN: %lld -lSystem %t/main.o %t/foo.o -o %t.out \
# RUN: -rename_section __FOO __bar __BAZ __quux \
# RUN: -rename_section __WHAT __ever __FOO __bar \
# RUN: -u 'section$start$__UFLAG_SEG$__uflag_sect' \
# RUN: -U 'section$start$__DYNAMIC$__lookup' \
# RUN: -U 'section$start$__DYNAMIC$__unref' \
# RUN: -e 'section$start$__TEXT$__text'
# RUN: llvm-objdump --macho --syms --section-headers %t.out > %t-dump.txt
# RUN: llvm-objdump --macho -d --no-symbolic-operands --no-show-raw-insn %t.out >> %t-dump.txt
# RUN: llvm-objdump --macho --function-starts %t.out >> %t-dump.txt
# RUN: FileCheck %s < %t-dump.txt
## Setting the entry point to the start of the __text section should
## set it to _main, since that's the first function in that section.
# RUN: llvm-objdump --macho --syms --all-headers %t.out \
# RUN: | FileCheck --check-prefix=MAINENTRY %s
# MAINENTRY: [[#%x, MAINADDR:]] g F __TEXT,__text _main
# MAINENTRY: LC_MAIN
# MAINENTRY-NEXT: cmdsize
# MAINENTRY-NEXT: entryoff [[#%d, MAINADDR - 0x100000000]]
## Nothing should change if we reorder two functions in the text segment.
## (Reorder some section$start/end symbols too for good measure.)
# RUN: %lld -lSystem %t/main.o %t/foo.o -o %t.ordered.out \
# RUN: -order_file %t/order.txt \
# RUN: -rename_section __FOO __bar __BAZ __quux \
# RUN: -rename_section __WHAT __ever __FOO __bar \
# RUN: -u 'section$start$__UFLAG_SEG$__uflag_sect' \
# RUN: -U 'section$start$__DYNAMIC$__lookup' \
# RUN: -U 'section$start$__DYNAMIC$__unref' \
# RUN: -e 'section$start$__TEXT$__text'
# RUN: llvm-objdump --macho --syms --section-headers %t.ordered.out > %t-ordered-dump.txt
# RUN: llvm-objdump --macho -d --no-symbolic-operands --no-show-raw-insn %t.ordered.out >> %t-ordered-dump.txt
# RUN: llvm-objdump --macho --function-starts %t.out >> %t-ordered-dump.txt
# RUN: FileCheck %s < %t-ordered-dump.txt
## `-undefined dynamic_lookup` also shouldn't change anything.
# RUN: %lld -lSystem %t/main.o %t/foo.o -o %t.dl.out -undefined dynamic_lookup \
# RUN: -rename_section __FOO __bar __BAZ __quux \
# RUN: -rename_section __WHAT __ever __FOO __bar \
# RUN: -u 'section$start$__UFLAG_SEG$__uflag_sect' \
# RUN: -U 'section$start$__DYNAMIC$__lookup' \
# RUN: -U 'section$start$__DYNAMIC$__unref' \
# RUN: -e 'section$start$__TEXT$__text'
# RUN: llvm-objdump --macho --syms --section-headers %t.dl.out > %t-dump.dl.txt
# RUN: llvm-objdump --macho -d --no-symbolic-operands --no-show-raw-insn %t.dl.out >> %t-dump.dl.txt
# RUN: llvm-objdump --macho --function-starts %t.out >> %t-dump.dl.txt
# RUN: FileCheck %s < %t-dump.dl.txt
## ...except that the entry point is now _otherfun instead of _main since
## _otherfun is now at the start of the __text section.
# RUN: llvm-objdump --macho --syms --all-headers %t.ordered.out \
# RUN: | FileCheck --check-prefix=OTHERENTRY %s
# OTHERENTRY: [[#%x, OTHERADDR:]] g F __TEXT,__text _otherfun
# OTHERENTRY: LC_MAIN
# OTHERENTRY-NEXT: cmdsize
# OTHERENTRY-NEXT: entryoff [[#%d, OTHERADDR - 0x100000000]]
## Test that the link succeeds with dead-stripping enabled too.
# RUN: %lld -dead_strip -lSystem %t/main.o -o %t/stripped.out
## (Fun fact: `-e 'section$start$__TEXT$__text -dead_strip` strips
## everything in the text section because markLive runs well before
## section$start symbols are replaced, so the entry point is just
## an undefined symbol that keeps nothing alive, and then later it
## sets the entry point to the start of the now-empty text section
## and the output program crashes when running. This matches ld64's
## behavior.)
# CHECK-LABEL: Sections:
# CHECK-NEXT: Idx Name Size VMA Type
# CHECK: 0 __text {{[0-9a-f]*}} [[#%x, TEXTSTART:]] TEXT
# CHECK: 1 __aftertext {{[0-9a-f]*}} [[#%x, TEXTEND:]]
# CHECK: 2 __cstring {{[0-9a-f]*}} [[#%x, CSTRINGSTART:]] DATA
# CHECK: 3 __aftercstring {{[0-9a-f]*}} [[#%x, CSTRINGEND:]]
# CHECK: 4 __data 00000008 [[#%x, DATASTART:]] DATA
# CHECK: 5 __llvm_orderfile 00000000 [[#%x, LLVMORDERFILESTART:]] DATA
# CHECK: 6 __mybss 00008000 [[#%x, MYBSSSTART:]] BSS
# CHECK: 7 __quux 0000002a [[#%x, QUUXSTART:]]
# CHECK: 8 __bar 00000059 [[#%x, BARSTART:]]
# CHECK: 9 __uflag_sect 00000000
# CHECK: 10 __lookup 00000000
# CHECK-NOT: symbol
# CHECK-NOT: __unref
# CHECK-LABEL: SYMBOL TABLE:
# CHECK-NOT: section$start$__TEXT$__text
# CHECK-NOT: section$end$__TEXT$__text
# CHECK-NOT: section$start$__TEXT$__cstring
# CHECK-NOT: section$end$__TEXT$__cstring
# CHECK-NOT: section$start$__DATA$__data
# CHECK-NOT: section$end$__DATA$__data
# CHECK-NOT: section$start$__DATA$__llvm_orderfile
# CHECK-NOT: section$end$__DATA$__llvm_orderfile
# CHECK-NOT: section$start$__DYNAMIC$__lookup
# CHECK-NOT: section$start$__DYNAMIC$__unref
# CHECK: section$end$ACTUAL$symbol
# CHECK: section$start$ACTUAL$symbol
# CHECK-LABEL: _main:
## The CHECK-SAMEs work around FileCheck's
## "error: numeric variable 'PC2' defined earlier in the same CHECK directive"
## limitation.
## The 7s are the length of a leaq instruction.
## section$start$__TEXT$__text / section$end$__TEXT$__text
# CHECK: [[#%x, PC1:]]:
# CHECK-SAME: leaq [[#%d, TEXTSTART - PC1 - 7]](%rip), %rax
# CHECK-NEXT: [[#%x, PC2:]]:
# CHECK-SAME: leaq [[#%d, TEXTEND - PC2 - 7]](%rip), %rbx
## section$start$__TEXT$__cstring / section$end$__TEXT$__cstring
# CHECK: [[#%x, PC3:]]:
# CHECK-SAME: leaq [[#%d, CSTRINGSTART - PC3 - 7]](%rip), %rax
# CHECK-NEXT: [[#%x, PC4:]]:
# CHECK-SAME: leaq [[#%d, CSTRINGEND - PC4 - 7]](%rip), %rbx
## section$start$__DATA$__data / section$end$__DATA$__data
# CHECK: [[#%x, PC5:]]:
# CHECK-SAME: leaq [[#%d, DATASTART - PC5 - 7]](%rip), %rax
# CHECK-NEXT: [[#%x, PC6:]]:
# CHECK-SAME: leaq [[#%d, DATASTART + 8 - PC6 - 7]](%rip), %rbx
## section$start$__MYBSS$__mybss / section$end$__MYBSS$__mybss
# CHECK: [[#%x, PC7:]]:
# CHECK-SAME: leaq [[#%d, MYBSSSTART - PC7 - 7]](%rip), %rax
# CHECK-NEXT: [[#%x, PC8:]]:
# CHECK-SAME: leaq [[#%d, MYBSSSTART + 0x8000 - PC8 - 7]](%rip), %rbx
## section$start$__DATA$__llvm_orderfile / section$end$__DATA$__llvm_orderfile
## This section has size 0.
# CHECK: [[#%x, PC9:]]:
# CHECK-SAME: leaq [[#%d, LLVMORDERFILESTART - PC9 - 7]](%rip), %rax
# CHECK-NEXT: [[#%x, PC10:]]:
# CHECK-SAME: leaq [[#%d, LLVMORDERFILESTART - PC10 - 7]](%rip), %rbx
## Section-rename tests.
## Input section __FOO/__bar is renamed to output section
## __BAZ/__quux by a -rename_section flag.
## section$start$__FOO$__bar ends up referring to the __BAZ/__quux section.
# CHECK: [[#%x, PC11:]]:
# CHECK-SAME: leaq [[#%d, QUUXSTART - PC11 - 7]](%rip), %rax
# CHECK-NEXT: [[#%x, PC12:]]:
# CHECK-SAME: leaq [[#%d, QUUXSTART + 42 - PC12 - 7]](%rip), %rbx
## section$start$__BAZ$__quux also refers to the __BAZ/__quux section.
# CHECK: [[#%x, PC13:]]:
# CHECK-SAME: leaq [[#%d, QUUXSTART - PC13 - 7]](%rip), %rax
# CHECK-NEXT: [[#%x, PC14:]]:
# CHECK-SAME: leaq [[#%d, QUUXSTART + 42 - PC14 - 7]](%rip), %rbx
## Input section __WHAT/__ever is renamed to output section
## __FOO/__bar by a -rename_section flag.
## section$start$__WHAT$__ever ends up referring to the __FOO/__bar section.
# CHECK: [[#%x, PC15:]]:
# CHECK-SAME: leaq [[#%d, BARSTART - PC15 - 7]](%rip), %rax
# CHECK-NEXT: [[#%x, PC16:]]:
# CHECK-SAME: leaq [[#%d, BARSTART + 89 - PC16 - 7]](%rip), %rbx
## The function_starts section should not have an entry for the
## section$end$__TEXT$__text symbol.
# CHECK: [[#%.16x, TEXTSTART]]
# CHECK-NOT: [[#%.16x, TEXTEND]]
###############################################################################
## Test segment$start and segment$end.
# RUN: %lld -lSystem %t/seg.o -o %t.seg.out \
# RUN: -rename_segment __FOO __BAZ \
# RUN: -rename_segment __WHAT __FOO \
# RUN: -u 'segment$start$__UFLAG_SEG' \
# RUN: -e 'segment$start$__TEXT'
# RUN: llvm-objdump --macho --syms %t.seg.out > %t-seg-dump.txt
## llvm-objdump can't dump segment names; use lld-otool for this.
# RUN: llvm-otool -l %t.seg.out | grep -A6 LC_SEGMENT >> %t-seg-dump.txt
# RUN: llvm-objdump --macho -d --no-symbolic-operands --no-show-raw-insn %t.seg.out >> %t-seg-dump.txt
# RUN: llvm-objdump --macho --function-starts %t.out >> %t-seg-dump.txt
# RUN: FileCheck %s --check-prefix=SEG < %t-seg-dump.txt
# SEG-LABEL: SYMBOL TABLE:
# SEG-NOT: segment$start$__TEXT
# SEG-NOT: segment$end$__TEXT
# SEG-NOT: segment$start$__FOO
# SEG-NOT: segment$end$__FOO
# SEG-NOT: segment$start$__BAZ
# SEG-NOT: segment$end$__BAZ
# SEG-NOT: segment$start$__WHAT
# SEG-NOT: segment$end$__WHAT
# SEG-NOT: segment$start$__UFLAG_SEG
# SEG-NOT: segment$start$__UFLAG_SEG
# SEG: segment$start$REGULAR
# SEG: segment$end$REGULAR
# SEG: cmd LC_SEGMENT_64
# SEG-NEXT: cmdsize
# SEG-NEXT: segname __PAGEZERO
# SEG: cmd LC_SEGMENT_64
# SEG-NEXT: cmdsize
# SEG-NEXT: segname __TEXT
# SEG-NEXT: vmaddr 0x[[#%x, TEXTSTART:]]
# SEG-NEXT: vmsize 0x[[#%x, TEXTSIZE:]]
# SEG: cmd LC_SEGMENT_64
# SEG-NEXT: cmdsize
# SEG-NEXT: segname __BAZ
# SEG-NEXT: vmaddr 0x[[#%x, BAZSTART:]]
# SEG-NEXT: vmsize 0x[[#%x, BAZSIZE:]]
# SEG: cmd LC_SEGMENT_64
# SEG-NEXT: cmdsize
# SEG-NEXT: segname __FOO
# SEG-NEXT: vmaddr 0x[[#%x, FOOSTART:]]
# SEG-NEXT: vmsize 0x[[#%x, FOOSIZE:]]
# SEG: cmd LC_SEGMENT_64
# SEG-NEXT: cmdsize
# SEG-NEXT: segname __UFLAG_SEG
# SEG-NEXT: vmaddr 0x[[#%x, UFLAGSTART:]]
# SEG-NEXT: vmsize 0x0000000000000000
# SEG: cmd LC_SEGMENT_64
# SEG-NEXT: cmdsize
# SEG-NEXT: segname ASDF
# SEG-NEXT: vmaddr 0x[[#%x, ASDFSTART:]]
# SEG-NEXT: vmsize 0x0000000000000000
# SEG: _main
## segment$start$__TEXT / segment$end$__TEXT
# SEG: [[#%x, PC1:]]:
# SEG-SAME: leaq [[#%d, TEXTSTART - PC1 - 7]](%rip), %rax
# SEG-NEXT: [[#%x, PC2:]]:
# SEG-SAME: leaq [[#%d, TEXTSTART + TEXTSIZE - PC2 - 7]](%rip), %rbx
## segment$start$__FOO / segment$end$__FOO, which is renamed to __BAZ
# SEG: [[#%x, PC3:]]:
# SEG-SAME: leaq [[#%d, BAZSTART - PC3 - 7]](%rip), %rax
# SEG-NEXT: [[#%x, PC4:]]:
# SEG-SAME: leaq [[#%d, BAZSTART + BAZSIZE - PC4 - 7]](%rip), %rbx
## segment$start$__BAZ / segment$end$__BAZ
# SEG: [[#%x, PC5:]]:
# SEG-SAME: leaq [[#%d, BAZSTART - PC5 - 7]](%rip), %rax
# SEG-NEXT: [[#%x, PC6:]]:
# SEG-SAME: leaq [[#%d, BAZSTART + BAZSIZE - PC6 - 7]](%rip), %rbx
## segment$start$__WHAT / segment$end$__WHAT, which is renamed to __FOO
# SEG: [[#%x, PC7:]]:
# SEG-SAME: leaq [[#%d, FOOSTART - PC7 - 7]](%rip), %rax
# SEG-NEXT: [[#%x, PC8:]]:
# SEG-SAME: leaq [[#%d, FOOSTART + FOOSIZE - PC8 - 7]](%rip), %rbx
## segment$start$ASDF / segment$end$ASDF
# SEG: [[#%x, PC9:]]:
# SEG-SAME: leaq [[#%d, ASDFSTART - PC9 - 7]](%rip), %rax
# SEG-NEXT: [[#%x, PC10:]]:
# SEG-SAME: leaq [[#%d, ASDFSTART - PC10 - 7]](%rip), %rbx
#--- order.txt
_otherfun
_main
section$end$__TEXT$__text
section$start$__TEXT$__text
#--- main.s
.zerofill __MYBSS,__mybss,_zero_foo,0x8000
.globl section$start$ACTUAL$symbol
.globl section$end$ACTUAL$symbol
## Renamed to __BAZ,__quux by -rename_section
.section __FOO,__bar
.space 42
## Renamed to __FOO,__bar by -rename_section
.section __WHAT,__ever
.space 89
.text
.globl _main
_main:
# Basics: start/end of existing, normal sections.
# For __TEXT/__text, these magic symbols shouldn't be
# included in __function_starts
movq section$start$__TEXT$__text@GOTPCREL(%rip), %rax
movq section$end$__TEXT$__text@GOTPCREL(%rip), %rbx
# __TEXT/__cstring are interesting because they're not ConcatInputSections.
movq section$start$__TEXT$__cstring@GOTPCREL(%rip), %rax
movq section$end$__TEXT$__cstring@GOTPCREL(%rip), %rbx
# Vanilla __DATA/__data
movq section$start$__DATA$__data@GOTPCREL(%rip), %rax
movq section$end$__DATA$__data@GOTPCREL(%rip), %rbx
# Vanilla zerofill.
movq section$start$__MYBSS$__mybss@GOTPCREL(%rip), %rax
movq section$end$__MYBSS$__mybss@GOTPCREL(%rip), %rbx
# Referring to a non-existent section wills it into existence.
# This is needed for e.g. __DATA/__llvm_orderfile in libclang_rt.profile.
# This means `-u` can be used as a janky `-sectcreate`.
movq section$start$__DATA$__llvm_orderfile@GOTPCREL(%rip), %rax
movq section$end$__DATA$__llvm_orderfile@GOTPCREL(%rip), %rbx
# Section-rename tests.
movq section$start$__FOO$__bar@GOTPCREL(%rip), %rax
movq section$end$__FOO$__bar@GOTPCREL(%rip), %rbx
movq section$start$__BAZ$__quux@GOTPCREL(%rip), %rax
movq section$end$__BAZ$__quux@GOTPCREL(%rip), %rbx
movq section$start$__WHAT$__ever@GOTPCREL(%rip), %rax
movq section$end$__WHAT$__ever@GOTPCREL(%rip), %rbx
# If there are actual symbols with the magic names, the magic
# names lose their magic and just refer to those symbols (and
# no section is implicitly created for them).
movq section$start$ACTUAL$symbol@GOTPCREL(%rip), %rax
movq section$end$ACTUAL$symbol@GOTPCREL(%rip), %rbx
# -U section$start is not exported as dynamic_lookup, it just
# creates a section like -u.
movq section$start$__DYNAMIC$__lookup@GOTPCREL(%rip), %rax
movq section$end$__DYNAMIC$__lookup@GOTPCREL(%rip), %rbx
ret
.globl _otherfun
_otherfun:
ret
.section __TEXT,__aftertext
.fill 1
.cstring
.asciz "foo"
.asciz "barbaz"
.section __TEXT,__aftercstring
.fill 1
.data
.quad 0x1234
.subsections_via_symbols
#--- foo.s
.text
.globl section$start$ACTUAL$symbol
section$start$ACTUAL$symbol:
.fill 1
.globl section$end$ACTUAL$symbol
section$end$ACTUAL$symbol:
.fill 1
.subsections_via_symbols
#--- seg.s
## Renamed to __BAZ,__bar by -rename_segment
.section __FOO,__bar
.space 1
## Renamed to __FOO,__ever by -rename_segment
.section __WHAT,__ever
.space 1
.text
.globl segment$start$REGULAR
segment$start$REGULAR:
retq
.globl segment$end$REGULAR
segment$end$REGULAR:
retq
.globl _main
_main:
movq segment$start$__TEXT@GOTPCREL(%rip), %rax
movq segment$end$__TEXT@GOTPCREL(%rip), %rbx
movq segment$start$__FOO@GOTPCREL(%rip), %rax
movq segment$end$__FOO@GOTPCREL(%rip), %rbx
movq segment$start$__BAZ@GOTPCREL(%rip), %rax
movq segment$end$__BAZ@GOTPCREL(%rip), %rbx
# (These two lines make ld64 crash linking the .o file.
# The rest of the test works with ld64 too.)
movq segment$start$__WHAT@GOTPCREL(%rip), %rax
movq segment$end$__WHAT@GOTPCREL(%rip), %rbx
# References to non-existing segments create that segment,
# without any sections in it.
movq segment$start$ASDF@GOTPCREL(%rip), %rax
movq segment$end$ASDF@GOTPCREL(%rip), %rbx
# Non-undefined symbols don't create segments.
callq segment$start$REGULAR
callq segment$end$REGULAR
ret
.section __TEXT,__fill
.space 578
.subsections_via_symbols