aboutsummaryrefslogtreecommitdiff
path: root/llvm/lib/Object/WasmObjectFile.cpp
diff options
context:
space:
mode:
authorSam Clegg <sbc@chromium.org>2018-01-09 23:43:14 +0000
committerSam Clegg <sbc@chromium.org>2018-01-09 23:43:14 +0000
commitea7caceedcc8d872bc31c141515ef2e3749ef659 (patch)
tree7499d1e4ee969bda41210ab8d2e5d7c339c07b44 /llvm/lib/Object/WasmObjectFile.cpp
parent29f5f987f1b76b5c43310b2062c9f447667a4f80 (diff)
downloadllvm-ea7caceedcc8d872bc31c141515ef2e3749ef659.zip
llvm-ea7caceedcc8d872bc31c141515ef2e3749ef659.tar.gz
llvm-ea7caceedcc8d872bc31c141515ef2e3749ef659.tar.bz2
[WebAssembly] Add COMDAT support
This adds COMDAT support to the Wasm object-file format. Spec: https://github.com/WebAssembly/tool-conventions/pull/31 Corresponding LLD change: https://bugs.llvm.org/show_bug.cgi?id=35533, and D40845 Patch by Nicholas Wilson Differential Revision: https://reviews.llvm.org/D40844 llvm-svn: 322135
Diffstat (limited to 'llvm/lib/Object/WasmObjectFile.cpp')
-rw-r--r--llvm/lib/Object/WasmObjectFile.cpp54
1 files changed, 54 insertions, 0 deletions
diff --git a/llvm/lib/Object/WasmObjectFile.cpp b/llvm/lib/Object/WasmObjectFile.cpp
index c33f4f6..ec2e2bf 100644
--- a/llvm/lib/Object/WasmObjectFile.cpp
+++ b/llvm/lib/Object/WasmObjectFile.cpp
@@ -10,6 +10,7 @@
#include "llvm/ADT/ArrayRef.h"
#include "llvm/ADT/STLExtras.h"
#include "llvm/ADT/StringRef.h"
+#include "llvm/ADT/StringSet.h"
#include "llvm/ADT/Triple.h"
#include "llvm/BinaryFormat/Wasm.h"
#include "llvm/MC/SubtargetFeature.h"
@@ -422,6 +423,10 @@ Error WasmObjectFile::parseLinkingSection(const uint8_t *Ptr,
}
break;
}
+ case wasm::WASM_COMDAT_INFO:
+ if (Error Err = parseLinkingSectionComdat(Ptr, SubSectionEnd))
+ return Err;
+ break;
default:
Ptr += Size;
break;
@@ -436,6 +441,55 @@ Error WasmObjectFile::parseLinkingSection(const uint8_t *Ptr,
return Error::success();
}
+Error WasmObjectFile::parseLinkingSectionComdat(const uint8_t *&Ptr,
+ const uint8_t *End)
+{
+ uint32_t ComdatCount = readVaruint32(Ptr);
+ StringSet<> ComdatSet;
+ while (ComdatCount--) {
+ StringRef Name = readString(Ptr);
+ if (Name.empty() || !ComdatSet.insert(Name).second)
+ return make_error<GenericBinaryError>("Bad/duplicate COMDAT name " + Twine(Name),
+ object_error::parse_failed);
+ Comdats.emplace_back(Name);
+ uint32_t Flags = readVaruint32(Ptr);
+ if (Flags != 0)
+ return make_error<GenericBinaryError>("Unsupported COMDAT flags",
+ object_error::parse_failed);
+
+ uint32_t EntryCount = readVaruint32(Ptr);
+ while (EntryCount--) {
+ unsigned Kind = readVaruint32(Ptr);
+ unsigned Index = readVaruint32(Ptr);
+ switch (Kind) {
+ default:
+ return make_error<GenericBinaryError>("Invalid COMDAT entry type",
+ object_error::parse_failed);
+ case wasm::WASM_COMDAT_DATA:
+ if (Index >= DataSegments.size())
+ return make_error<GenericBinaryError>("COMDAT data index out of range",
+ object_error::parse_failed);
+ if (!DataSegments[Index].Data.Comdat.empty())
+ return make_error<GenericBinaryError>("Data segment in two COMDATs",
+ object_error::parse_failed);
+ DataSegments[Index].Data.Comdat = Name;
+ break;
+ case wasm::WASM_COMDAT_FUNCTION:
+ if (Index < NumImportedFunctions || !isValidFunctionIndex(Index))
+ return make_error<GenericBinaryError>("COMDAT function index out of range",
+ object_error::parse_failed);
+ Index -= NumImportedFunctions;
+ if (!Functions[Index].Comdat.empty())
+ return make_error<GenericBinaryError>("Function in two COMDATs",
+ object_error::parse_failed);
+ Functions[Index].Comdat = Name;
+ break;
+ }
+ }
+ }
+ return Error::success();
+}
+
WasmSection* WasmObjectFile::findCustomSectionByName(StringRef Name) {
for (WasmSection& Section : Sections) {
if (Section.Type == wasm::WASM_SEC_CUSTOM && Section.Name == Name)