Index: ELF/Config.h =================================================================== --- ELF/Config.h +++ ELF/Config.h @@ -30,7 +30,7 @@ ELF64BEKind }; -enum class BuildIdKind { None, Fnv1, Md5, Sha1, Hexstring }; +enum class BuildIdKind { None, Fnv1, Md5, Sha1, Hexstring, Uuid }; enum class UnresolvedPolicy { NoUndef, Error, Warn, Ignore }; Index: ELF/Driver.cpp =================================================================== --- ELF/Driver.cpp +++ ELF/Driver.cpp @@ -427,6 +427,8 @@ Config->BuildId = BuildIdKind::Md5; } else if (S == "sha1") { Config->BuildId = BuildIdKind::Sha1; + } else if (S == "uuid") { + Config->BuildId = BuildIdKind::Uuid; } else if (S == "none") { Config->BuildId = BuildIdKind::None; } else if (S.startswith("0x")) { Index: ELF/OutputSections.h =================================================================== --- ELF/OutputSections.h +++ ELF/OutputSections.h @@ -611,6 +611,12 @@ void writeBuildId(ArrayRef> Bufs) override; }; +template class BuildIdUuid final : public BuildIdSection { +public: + BuildIdUuid() : BuildIdSection(16) {} + void writeBuildId(ArrayRef> Bufs) override; +}; + template class BuildIdHexstring final : public BuildIdSection { public: Index: ELF/OutputSections.cpp =================================================================== --- ELF/OutputSections.cpp +++ ELF/OutputSections.cpp @@ -20,6 +20,14 @@ #include "llvm/Support/MathExtras.h" #include "llvm/Support/SHA1.h" #include +#ifdef LLVM_ON_WIN32 +#define NOMINMAX +#include +#pragma comment(lib, "rpcrt4.lib") +#else +#include +#include +#endif using namespace llvm; using namespace llvm::dwarf; @@ -1676,6 +1684,21 @@ } template +void BuildIdUuid::writeBuildId(ArrayRef> Bufs) { +#ifdef LLVM_ON_WIN32 + UUID Uuid; + UuidCreate(&Uuid); + memcpy(this->HashBuf, &Uuid, 16); +#else + int Fd = open("/dev/urandom", O_RDONLY); + if (Fd == -1) + error("failed to open /dev/urandom"); + else + ::read(Fd, this->HashBuf, 16); +#endif +} + +template BuildIdHexstring::BuildIdHexstring() : BuildIdSection(Config->BuildIdVector.size()) {} @@ -1939,6 +1962,11 @@ template class BuildIdSha1; template class BuildIdSha1; +template class BuildIdUuid; +template class BuildIdUuid; +template class BuildIdUuid; +template class BuildIdUuid; + template class BuildIdHexstring; template class BuildIdHexstring; template class BuildIdHexstring; Index: ELF/Writer.cpp =================================================================== --- ELF/Writer.cpp +++ ELF/Writer.cpp @@ -153,6 +153,8 @@ BuildId.reset(new BuildIdMd5); else if (Config->BuildId == BuildIdKind::Sha1) BuildId.reset(new BuildIdSha1); + else if (Config->BuildId == BuildIdKind::Uuid) + BuildId.reset(new BuildIdUuid); else if (Config->BuildId == BuildIdKind::Hexstring) BuildId.reset(new BuildIdHexstring); Index: test/ELF/build-id.s =================================================================== --- test/ELF/build-id.s +++ test/ELF/build-id.s @@ -7,6 +7,8 @@ # RUN: llvm-objdump -s %t2 | FileCheck -check-prefix=MD5 %s # RUN: ld.lld --build-id=sha1 %t -o %t2 # RUN: llvm-objdump -s %t2 | FileCheck -check-prefix=SHA1 %s +# RUN: ld.lld --build-id=uuid %t -o %t2 +# RUN: llvm-objdump -s %t2 | FileCheck -check-prefix=UUID %s # RUN: ld.lld --build-id=0x12345678 %t -o %t2 # RUN: llvm-objdump -s %t2 | FileCheck -check-prefix=HEX %s # RUN: ld.lld %t -o %t2 @@ -31,6 +33,9 @@ # SHA1: Contents of section .note.gnu.build-id: # SHA1-NEXT: 04000000 14000000 03000000 474e5500 ............GNU. +# UUID: Contents of section .note.gnu.build-id: +# UUID-NEXT: 04000000 10000000 03000000 474e5500 ............GNU. + # HEX: Contents of section .note.gnu.build-id: # HEX-NEXT: 04000000 04000000 03000000 474e5500 ............GNU. # HEX-NEXT: 12345678