Index: test/tools/llvm-objcopy/deterministic-archive.test =================================================================== --- /dev/null +++ test/tools/llvm-objcopy/deterministic-archive.test @@ -0,0 +1,49 @@ +# RUN: yaml2obj %s > %t.o + +# Create an archive, specifying U so that timestamps/etc. are preserved. +# We only test timestamps as a proxy for full deterministic writing; i.e. we +# assume UID/GIDs are preserved if timestamps are preserved. +# RUN: touch -t 199505050555.55 %t.o +# RUN: rm -f %t.a +# RUN: llvm-ar crsU %t.a %t.o + +# Test short flags. +# RUN: llvm-objcopy -D %t.a %t.2D.a +# RUN: env TZ=GMT llvm-ar tv %t.2D.a | FileCheck %s --check-prefix=CHECK-DETERMINISTIC +# RUN: llvm-objcopy -U %t.a %t.2U.a +# RUN: env TZ=GMT llvm-ar tv %t.2U.a | FileCheck %s --check-prefix=CHECK-NONDETERMINISTIC + +# RUN: llvm-strip -D %t.a -o %t.3D.a +# RUN: env TZ=GMT llvm-ar tv %t.3D.a | FileCheck %s --check-prefix=CHECK-DETERMINISTIC +# RUN: llvm-strip -U %t.a -o %t.3U.a +# RUN: env TZ=GMT llvm-ar tv %t.3U.a | FileCheck %s --check-prefix=CHECK-NONDETERMINISTIC + +# Test long flags. +# RUN: llvm-objcopy --enable-deterministic-archives %t.a %t.4D.a +# RUN: env TZ=GMT llvm-ar tv %t.4D.a | FileCheck %s --check-prefix=CHECK-DETERMINISTIC +# RUN: llvm-objcopy --disable-deterministic-archives %t.a %t.4U.a +# RUN: env TZ=GMT llvm-ar tv %t.4U.a | FileCheck %s --check-prefix=CHECK-NONDETERMINISTIC + +# Test various forms of invalid flag combintations. +# RUN: not llvm-objcopy -U -D %t.a 2>&1 \ +# RUN: | FileCheck %s --check-prefix=NO-MIXING +# RUN: not llvm-objcopy --enable-deterministic-archives \ +# RUN: --disable-deterministic-archives %t.a 2>&1 \ +# RUN: | FileCheck %s --check-prefix=NO-MIXING +# RUN: not llvm-objcopy --disable-deterministic-archives -D %t.a 2>&1 \ +# RUN: | FileCheck %s --check-prefix=NO-MIXING + +# CHECK-DETERMINISTIC: {{[[:space:]]1970[[:space:]]}} +# CHECK-NONDETERMINISTIC: {{[[:space:]]1995[[:space:]]}} +# NO-MIXING: Specifying both --enable-deterministic-archives and --disable-deterministic-archives is ambiguous. + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .text + Type: SHT_PROGBITS + Flags: [ SHF_ALLOC, SHF_EXECINSTR ] Index: tools/llvm-objcopy/CopyConfig.h =================================================================== --- tools/llvm-objcopy/CopyConfig.h +++ tools/llvm-objcopy/CopyConfig.h @@ -72,6 +72,7 @@ StringMap SymbolsToRename; // Boolean options + bool DeterministicArchives = true; bool DiscardAll = false; bool ExtractDWO = false; bool KeepFileSymbols = false; Index: tools/llvm-objcopy/CopyConfig.cpp =================================================================== --- tools/llvm-objcopy/CopyConfig.cpp +++ tools/llvm-objcopy/CopyConfig.cpp @@ -343,6 +343,15 @@ for (auto Arg : InputArgs.filtered(OBJCOPY_keep_symbol)) Config.SymbolsToKeep.push_back(Arg->getValue()); + if (InputArgs.hasArg(OBJCOPY_enable_deterministic_archives) && + InputArgs.hasArg(OBJCOPY_disable_deterministic_archives)) + error("Specifying both --enable-deterministic-archives and " + "--disable-deterministic-archives is ambiguous"); + if (InputArgs.hasArg(OBJCOPY_enable_deterministic_archives)) + Config.DeterministicArchives = true; + else if (InputArgs.hasArg(OBJCOPY_disable_deterministic_archives)) + Config.DeterministicArchives = false; + Config.PreserveDates = InputArgs.hasArg(OBJCOPY_preserve_dates); if (Config.DecompressDebugSections && @@ -411,6 +420,15 @@ for (auto Arg : InputArgs.filtered(STRIP_keep_symbol)) Config.SymbolsToKeep.push_back(Arg->getValue()); + if (InputArgs.hasArg(STRIP_enable_deterministic_archives) && + InputArgs.hasArg(STRIP_disable_deterministic_archives)) + error("Specifying both --enable-deterministic-archives and " + "--disable-deterministic-archives is ambiguous"); + if (InputArgs.hasArg(STRIP_enable_deterministic_archives)) + Config.DeterministicArchives = true; + else if (InputArgs.hasArg(STRIP_disable_deterministic_archives)) + Config.DeterministicArchives = false; + Config.PreserveDates = InputArgs.hasArg(STRIP_preserve_dates); DriverConfig DC; Index: tools/llvm-objcopy/ObjcopyOpts.td =================================================================== --- tools/llvm-objcopy/ObjcopyOpts.td +++ tools/llvm-objcopy/ObjcopyOpts.td @@ -35,6 +35,18 @@ MetaVarName<"dwo-file">, HelpText<"Equivalent to extract-dwo on the input file to , then strip-dwo on the input file">; +def enable_deterministic_archives + : Flag<["-", "--"], "enable-deterministic-archives">, + HelpText<"Enable deterministic mode when stripping archives (use zero " + "for UIDs, GIDs, and timestamps).">; +def D : Flag<[ "-" ], "D">, Alias; + +def disable_deterministic_archives + : Flag<["-", "--"], "disable-deterministic-archives">, + HelpText<"Disable deterministic mode when stripping archives (use real " + "values for UIDs, GIDs, and timestamps).">; +def U : Flag<[ "-" ], "U">, Alias; + def preserve_dates : Flag<[ "-", "--" ], "preserve-dates">, HelpText<"Preserve access and modification timestamps">; Index: tools/llvm-objcopy/StripOpts.td =================================================================== --- tools/llvm-objcopy/StripOpts.td +++ tools/llvm-objcopy/StripOpts.td @@ -7,6 +7,18 @@ def help : Flag<["-", "--"], "help">; +def enable_deterministic_archives + : Flag<["-", "--"], "enable-deterministic-archives">, + HelpText<"Enable deterministic mode when stripping archives (use zero " + "for UIDs, GIDs, and timestamps).">; +def D : Flag<[ "-" ], "D">, Alias; + +def disable_deterministic_archives + : Flag<["-", "--"], "disable-deterministic-archives">, + HelpText<"Disable deterministic mode when stripping archives (use real " + "values for UIDs, GIDs, and timestamps).">; +def U : Flag<[ "-" ], "U">, Alias; + defm output : Eq<"o">, MetaVarName<"output">, HelpText<"Write output to ">; Index: tools/llvm-objcopy/llvm-objcopy.cpp =================================================================== --- tools/llvm-objcopy/llvm-objcopy.cpp +++ tools/llvm-objcopy/llvm-objcopy.cpp @@ -147,7 +147,7 @@ executeObjcopyOnBinary(Config, *Bin, MB); Expected Member = - NewArchiveMember::getOldMember(Child, true); + NewArchiveMember::getOldMember(Child, Config.DeterministicArchives); if (!Member) reportError(Ar.getFileName(), Member.takeError()); Member->Buf = MB.releaseMemoryBuffer(); @@ -157,9 +157,9 @@ if (Err) reportError(Config.InputFilename, std::move(Err)); - if (Error E = - deepWriteArchive(Config.OutputFilename, NewArchiveMembers, - Ar.hasSymbolTable(), Ar.kind(), true, Ar.isThin())) + if (Error E = deepWriteArchive(Config.OutputFilename, NewArchiveMembers, + Ar.hasSymbolTable(), Ar.kind(), + Config.DeterministicArchives, Ar.isThin())) reportError(Config.OutputFilename, std::move(E)); }