diff --git a/lld/ELF/Config.h b/lld/ELF/Config.h --- a/lld/ELF/Config.h +++ b/lld/ELF/Config.h @@ -154,6 +154,8 @@ bool demangle = true; bool dependentLibraries; bool disableVerify; + bool eb = false; + bool el = false; bool ehFrameHdr; bool emitLLVM; bool emitRelocs; diff --git a/lld/ELF/Driver.cpp b/lld/ELF/Driver.cpp --- a/lld/ELF/Driver.cpp +++ b/lld/ELF/Driver.cpp @@ -1125,6 +1125,13 @@ config->zWxneeded = hasZOption(args, "wxneeded"); setUnresolvedSymbolPolicy(args); + if (opt::Arg *arg = args.getLastArg(OPT_eb, OPT_el)) { + if (arg->getOption().matches(OPT_eb)) + config->eb = true; + else + config->el = true; + } + for (opt::Arg *arg : args.filtered(OPT_z)) { std::pair option = StringRef(arg->getValue()).split('='); diff --git a/lld/ELF/Options.td b/lld/ELF/Options.td --- a/lld/ELF/Options.td +++ b/lld/ELF/Options.td @@ -159,6 +159,9 @@ "shared object. Implies -Bsymbolic but does not set DF_SYMBOLIC">, MetaVarName<"">; +def el: F<"EL">; +def eb: F<"EB">; + defm eh_frame_hdr: B<"eh-frame-hdr", "Request creation of .eh_frame_hdr section and PT_GNU_EH_FRAME segment header", "Do not create .eh_frame_hdr section">; @@ -687,8 +690,6 @@ def: F<"warn-execstack">; def: F<"warn-once">; def: F<"warn-shared-textrel">; -def: F<"EB">; -def: F<"EL">; def: JoinedOrSeparate<["-"], "G">; def: F<"Qy">; diff --git a/lld/ELF/ScriptParser.cpp b/lld/ELF/ScriptParser.cpp --- a/lld/ELF/ScriptParser.cpp +++ b/lld/ELF/ScriptParser.cpp @@ -431,13 +431,25 @@ .Default({ELFNoneKind, EM_NONE}); } -// Parse OUTPUT_FORMAT(bfdname) or OUTPUT_FORMAT(bfdname, big, little). -// Currently we ignore big and little parameters. +// Parse OUTPUT_FORMAT(bfdname) or OUTPUT_FORMAT(default, big, little). If the +// second form is used, choose big if -EB is specified, little if -EL is +// specified. void ScriptParser::readOutputFormat() { expect("("); + StringRef s; config->bfdname = unquote(next()); - StringRef s = config->bfdname; + if (consume(",")) { + s = unquote(next()); + if (config->eb) + config->bfdname = s; + expect(","); + s = unquote(next()); + if (config->el) + config->bfdname = s; + } + expect(")"); + s = config->bfdname; if (s.consume_back("-freebsd")) config->osabi = ELFOSABI_FREEBSD; @@ -448,14 +460,6 @@ config->mipsN32Abi = true; if (config->emachine == EM_MSP430) config->osabi = ELFOSABI_STANDALONE; - - if (consume(")")) - return; - expect(","); - skip(); - expect(","); - skip(); - expect(")"); } void ScriptParser::readPhdrs() { diff --git a/lld/test/ELF/emulation-aarch64.s b/lld/test/ELF/emulation-aarch64.s --- a/lld/test/ELF/emulation-aarch64.s +++ b/lld/test/ELF/emulation-aarch64.s @@ -23,6 +23,20 @@ # RUN: ld.lld %t.script %t.be.o -o %t3.be # RUN: llvm-readobj --file-headers %t3.be | FileCheck --check-prefixes=AARCH64,BE %s +## Test OUTPUT_FORMAT(default, big, little). +# RUN: echo 'OUTPUT_FORMAT("elf64-littleaarch64", "elf64-bigaarch64", "elf64-littleaarch64")' > %t.script +# RUN: ld.lld -EL -T %t.script %t.o -o %t4.le +# RUN: llvm-readobj --file-headers %t4.le | FileCheck --check-prefixes=AARCH64,LE %s +# RUN: not ld.lld -EB -T %t.script %t.o -o /dev/null 2>&1 | FileCheck --check-prefix=ERR_BE %s + +# RUN: not ld.lld -T %t.script %t.be.o -o /dev/null 2>&1 | FileCheck --check-prefix=ERR_LE %s +# RUN: not ld.lld -EL -T %t.script %t.be.o -o /dev/null 2>&1 | FileCheck --check-prefix=ERR_LE %s +# RUN: ld.lld -EB -T %t.script %t.be.o -o %t4.be +# RUN: llvm-readobj --file-headers %t4.be | FileCheck --check-prefixes=AARCH64,BE %s + +# ERR_LE: error: {{.*}}.o is incompatible with elf64-littleaarch64 +# ERR_BE: error: {{.*}}.o is incompatible with elf64-bigaarch64 + # AARCH64: ElfHeader { # AARCH64-NEXT: Ident { # AARCH64-NEXT: Magic: (7F 45 4C 46) diff --git a/lld/test/ELF/invalid-linkerscript.test b/lld/test/ELF/invalid-linkerscript.test --- a/lld/test/ELF/invalid-linkerscript.test +++ b/lld/test/ELF/invalid-linkerscript.test @@ -51,10 +51,10 @@ # RUN: echo "OUTPUT_FORMAT(x y z)" > %t8 # RUN: not ld.lld %t8 no-such-file 2>&1 | FileCheck -check-prefix=ERR8 %s # RUN: not ld.lld -m elf_amd64 %t8 no-such-file 2>&1 | FileCheck -check-prefix=ERR8 %s -# ERR8: unknown output format name: x +# ERR8: ) expected, but got y # ERR8: cannot open no-such-file: # RUN: echo "OUTPUT_FORMAT(elf64-x86-64 y z)" > %t9 # RUN: not ld.lld %t9 no-such-file 2>&1 | FileCheck -check-prefix=ERR9 %s -# ERR9: , expected, but got y +# ERR9: ) expected, but got y # ERR9: cannot open no-such-file: