Index: lld/COFF/Config.h =================================================================== --- lld/COFF/Config.h +++ lld/COFF/Config.h @@ -181,6 +181,7 @@ uint32_t MinorImageVersion = 0; uint32_t MajorOSVersion = 6; uint32_t MinorOSVersion = 0; + uint32_t Timestamp = 0; bool DynamicBase = true; bool AllowBind = true; bool NxCompat = true; @@ -194,6 +195,7 @@ bool WarnLocallyDefinedImported = true; bool Incremental = true; bool KillAt = false; + bool TimestampIsHash = false; }; extern Configuration *Config; Index: lld/COFF/Driver.h =================================================================== --- lld/COFF/Driver.h +++ lld/COFF/Driver.h @@ -148,7 +148,7 @@ // Parses a string in the form of "[,[.]]". void parseSubsystem(StringRef Arg, WindowsSubsystem *Sys, uint32_t *Major, uint32_t *Minor); - +void parseTimestamp(StringRef Arg, uint32_t *Timestamp, bool *TimestampIsHash); void parseAlternateName(StringRef); void parseMerge(StringRef); void parseSection(StringRef); Index: lld/COFF/Driver.cpp =================================================================== --- lld/COFF/Driver.cpp +++ lld/COFF/Driver.cpp @@ -1033,6 +1033,15 @@ parseSubsystem(Arg->getValue(), &Config->Subsystem, &Config->MajorOSVersion, &Config->MinorOSVersion); + // Handle /timestamp + if (auto *Arg = Args.getLastArg(OPT_timestamp)) + parseTimestamp(Arg->getValue(), &Config->Timestamp, + &Config->TimestampIsHash); + else { + Config->TimestampIsHash = false; + Config->Timestamp = time(nullptr); + } + // Handle /alternatename for (auto *Arg : Args.filtered(OPT_alternatename)) parseAlternateName(Arg->getValue()); Index: lld/COFF/DriverUtils.cpp =================================================================== --- lld/COFF/DriverUtils.cpp +++ lld/COFF/DriverUtils.cpp @@ -165,6 +165,18 @@ parseVersion(Ver, Major, Minor); } +void parseTimestamp(StringRef Arg, uint32_t *Timestamp, bool *TimestampIsHash) { + if (Arg.equals_lower("hash")) { + *Timestamp = 0; + *TimestampIsHash = true; + return; + } + + *TimestampIsHash = false; + if (Arg.getAsInteger(0, *Timestamp)) + fatal("invalid timestamp: " + Arg); +} + // Parse a string of the form of "=". // Results are directly written to Config. void parseAlternateName(StringRef S) { Index: lld/COFF/Options.td =================================================================== --- lld/COFF/Options.td +++ lld/COFF/Options.td @@ -58,6 +58,7 @@ def stack : P<"stack", "Size of the stack">; def stub : P<"stub", "Specify DOS stub file">; def subsystem : P<"subsystem", "Specify subsystem">; +def timestamp : P<"timestamp", "Specify the timestamp to be written to the PE header">; def version : P<"version", "Specify a version number in the PE header">; def wholearchive_file : P<"wholearchive", "Include all object files from this archive">; Index: lld/COFF/Writer.cpp =================================================================== --- lld/COFF/Writer.cpp +++ lld/COFF/Writer.cpp @@ -1192,16 +1192,18 @@ reinterpret_cast(Buffer->getBufferStart()), Buffer->getBufferSize()); - uint32_t Hash = static_cast(xxHash64(OutputFileData)); + uint32_t Timestamp = Config->Timestamp; + if (Config->TimestampIsHash) + Timestamp = static_cast(xxHash64(OutputFileData)); if (DebugDirectory) - DebugDirectory->setTimeDateStamp(Hash); + DebugDirectory->setTimeDateStamp(Timestamp); uint8_t *Buf = Buffer->getBufferStart(); Buf += DOSStubSize + sizeof(PEMagic); object::coff_file_header *CoffHeader = reinterpret_cast(Buf); - CoffHeader->TimeDateStamp = Hash; + CoffHeader->TimeDateStamp = Timestamp; } // Sort .pdata section contents according to PE/COFF spec 5.5. Index: lld/test/COFF/timestamp.test =================================================================== --- /dev/null +++ lld/test/COFF/timestamp.test @@ -0,0 +1,18 @@ +rm %t.*.exe +RUN: yaml2obj %p/Inputs/generic.yaml > %t.obj +RUN: lld-link %t.obj /debug /timestamp:hash /entry:main /nodefaultlib /out:%t.1.exe +RUN: lld-link %t.obj /debug /timestamp:hash /entry:main /nodefaultlib /out:%t.2.exe +RUN: lld-link %t.obj /debug /timestamp:0 /entry:main /nodefaultlib /out:%t.3.exe +RUN: llvm-readobj -file-headers -coff-debug-directory %t.1.exe | FileCheck %s --check-prefix=HASH +RUN: llvm-readobj -file-headers -coff-debug-directory %t.2.exe | FileCheck %s --check-prefix=HASH +RUN: llvm-readobj -file-headers -coff-debug-directory %t.3.exe | FileCheck %s --check-prefix=ZERO + +HASH: ImageFileHeader { +HASH: TimeDateStamp: [[STAMP:.*]] +HASH: DebugDirectory [ +HASH: TimeDateStamp: [[STAMP]] + +ZERO: ImageFileHeader { +ZERO: TimeDateStamp: 1970-01-01 00:00:00 (0x0) +ZERO: DebugDirectory [ +ZERO: TimeDateStamp: 1970-01-01 00:00:00 (0x0)