summaryrefslogtreecommitdiff
path: root/src/passes/SeparateDataSegments.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/passes/SeparateDataSegments.cpp')
-rw-r--r--src/passes/SeparateDataSegments.cpp78
1 files changed, 78 insertions, 0 deletions
diff --git a/src/passes/SeparateDataSegments.cpp b/src/passes/SeparateDataSegments.cpp
new file mode 100644
index 000000000..3e48d14bc
--- /dev/null
+++ b/src/passes/SeparateDataSegments.cpp
@@ -0,0 +1,78 @@
+/*
+ * Copyright 2023 WebAssembly Community Group participants
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+//
+// Emits the data segments to a file. The file contains data from address base
+// onwards (we must pass in base, as we can't tell it from the wasm - the
+// first segment may start after a run of zeros, but we need those zeros in
+// the file.
+//
+
+#include "pass.h"
+#include "support/file.h"
+#include "wasm-features.h"
+#include "wasm.h"
+
+namespace wasm {
+
+struct SeparateDataSegments : public Pass {
+ bool modifiesBinaryenIR() override { return false; }
+
+ void run(Module* module) override {
+ std::string outfileName =
+ getPassOptions().getArgument("separate-data-segments",
+ "SeparateDataSegments usage: wasm-opt "
+ "--separate-data-segments@FILENAME");
+ Output outfile(outfileName, Flags::Binary);
+ std::string baseStr = getPassOptions().getArgument(
+ "separate-data-segments-global-base",
+ "SeparateDataSegments usage: wasm-opt "
+ "--pass-arg=separate-data-segments-global-base@NUMBER");
+ Address base = std::stoi(baseStr);
+ size_t lastEnd = 0;
+ for (auto& seg : module->dataSegments) {
+ if (seg->isPassive) {
+ Fatal() << "separating passive segments not implemented";
+ }
+ if (!seg->offset->is<Const>()) {
+ Fatal() << "separating relocatable segments not implemented";
+ }
+ size_t offset = seg->offset->cast<Const>()->value.getInteger();
+ offset -= base;
+ size_t fill = offset - lastEnd;
+ if (fill > 0) {
+ std::vector<char> buf(fill);
+ outfile.write(buf.data(), fill);
+ }
+ outfile.write(seg->data.data(), seg->data.size());
+ lastEnd = offset + seg->data.size();
+ }
+ module->dataSegments.clear();
+ // Remove the start/stop symbols that the PostEmscripten uses to remove
+ // em_asm/em_js data. Since we just removed all the data segments from the
+ // file there is nothing more for that pass to do.
+ // TODO(sbc): Fix the ordering so that the removal the EM_ASM/EM_JS data
+ // comes before this pass.
+ module->removeExport("__start_em_asm");
+ module->removeExport("__stop_em_asm");
+ module->removeExport("__start_em_js");
+ module->removeExport("__stop_em_js");
+ }
+};
+
+Pass* createSeparateDataSegmentsPass() { return new SeparateDataSegments(); }
+
+} // namespace wasm