diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/passes/LLVMMemoryCopyFillLowering.cpp | 4 | ||||
-rw-r--r-- | src/passes/OptimizeInstructions.cpp | 4 | ||||
-rw-r--r-- | src/tools/tool-options.h | 27 | ||||
-rw-r--r-- | src/wasm-binary.h | 2 | ||||
-rw-r--r-- | src/wasm-features.h | 19 | ||||
-rw-r--r-- | src/wasm/wasm-binary.cpp | 10 | ||||
-rw-r--r-- | src/wasm/wasm-validator.cpp | 12 | ||||
-rw-r--r-- | src/wasm/wasm.cpp | 2 |
8 files changed, 61 insertions, 19 deletions
diff --git a/src/passes/LLVMMemoryCopyFillLowering.cpp b/src/passes/LLVMMemoryCopyFillLowering.cpp index e5b940a5a..d2a7e5c48 100644 --- a/src/passes/LLVMMemoryCopyFillLowering.cpp +++ b/src/passes/LLVMMemoryCopyFillLowering.cpp @@ -49,7 +49,7 @@ struct LLVMMemoryCopyFillLowering } void run(Module* module) override { - if (!module->features.hasBulkMemory()) { + if (!module->features.hasBulkMemoryOpt()) { return; } if (module->features.hasMemory64() || module->features.hasMultiMemory()) { @@ -108,7 +108,7 @@ struct LLVMMemoryCopyFillLowering } else { module->removeFunction(memFillFuncName); } - module->features.disable(FeatureSet::BulkMemory); + module->features.setBulkMemoryOpt(false); } void createMemoryCopyFunc(Module* module) { diff --git a/src/passes/OptimizeInstructions.cpp b/src/passes/OptimizeInstructions.cpp index 881b7ea1f..792cf6235 100644 --- a/src/passes/OptimizeInstructions.cpp +++ b/src/passes/OptimizeInstructions.cpp @@ -1275,7 +1275,7 @@ struct OptimizeInstructions if (curr->type == Type::unreachable) { return; } - assert(getModule()->features.hasBulkMemory()); + assert(getModule()->features.hasBulkMemoryOpt()); if (auto* ret = optimizeMemoryCopy(curr)) { return replaceCurrent(ret); } @@ -1285,7 +1285,7 @@ struct OptimizeInstructions if (curr->type == Type::unreachable) { return; } - assert(getModule()->features.hasBulkMemory()); + assert(getModule()->features.hasBulkMemoryOpt()); if (auto* ret = optimizeMemoryFill(curr)) { return replaceCurrent(ret); } diff --git a/src/tools/tool-options.h b/src/tools/tool-options.h index f900d76ba..c7acf1d46 100644 --- a/src/tools/tool-options.h +++ b/src/tools/tool-options.h @@ -82,7 +82,16 @@ struct ToolOptions : public Options { .addFeature(FeatureSet::MutableGlobals, "mutable globals") .addFeature(FeatureSet::TruncSat, "nontrapping float-to-int operations") .addFeature(FeatureSet::SIMD, "SIMD operations and types") - .addFeature(FeatureSet::BulkMemory, "bulk memory operations") + .addFeature(FeatureSet::BulkMemory, + "bulk memory operations", + FeatureSet(FeatureSet::BulkMemoryOpt)) + .addFeature(FeatureSet::BulkMemoryOpt, + "memory.copy and memory.fill", + FeatureSet::None, + FeatureSet(FeatureSet::BulkMemory)) + .addFeature(FeatureSet::CallIndirectOverlong, + "LEB encoding of call-indirect (Ignored for compatibility as " + "it has no effect on Binaryen)") .addFeature(FeatureSet::ExceptionHandling, "exception handling operations") .addFeature(FeatureSet::TailCall, "tail call operations") @@ -200,16 +209,18 @@ struct ToolOptions : public Options { } ToolOptions& addFeature(FeatureSet::Feature feature, - const std::string& description) { + const std::string& description, + FeatureSet impliedEnable = FeatureSet::None, + FeatureSet impliedDisable = FeatureSet::None) { (*this) .add(std::string("--enable-") + FeatureSet::toString(feature), "", std::string("Enable ") + description, ToolOptionsCategory, Arguments::Zero, - [this, feature](Options*, const std::string&) { - enabledFeatures.set(feature, true); - disabledFeatures.set(feature, false); + [this, feature, impliedEnable](Options*, const std::string&) { + enabledFeatures.set(feature | impliedEnable, true); + disabledFeatures.set(feature | impliedEnable, false); }) .add(std::string("--disable-") + FeatureSet::toString(feature), @@ -217,9 +228,9 @@ struct ToolOptions : public Options { std::string("Disable ") + description, ToolOptionsCategory, Arguments::Zero, - [this, feature](Options*, const std::string&) { - enabledFeatures.set(feature, false); - disabledFeatures.set(feature, true); + [this, feature, impliedDisable](Options*, const std::string&) { + enabledFeatures.set(feature | impliedDisable, false); + disabledFeatures.set(feature | impliedDisable, true); }); return *this; } diff --git a/src/wasm-binary.h b/src/wasm-binary.h index 7c32e2169..163a62b1e 100644 --- a/src/wasm-binary.h +++ b/src/wasm-binary.h @@ -396,6 +396,8 @@ extern const char* MultiMemoryFeature; extern const char* TypedContinuationsFeature; extern const char* SharedEverythingFeature; extern const char* FP16Feature; +extern const char* BulkMemoryOptFeature; +extern const char* CallIndirectOverlongFeature; enum Subsection { NameModule = 0, diff --git a/src/wasm-features.h b/src/wasm-features.h index 92b07b547..20eec56bb 100644 --- a/src/wasm-features.h +++ b/src/wasm-features.h @@ -27,6 +27,8 @@ namespace wasm { struct FeatureSet { enum Feature : uint32_t { + // These features are intended to those documented in tool-conventions: + // https://github.com/WebAssembly/tool-conventions/blob/main/Linking.md#target-features-section None = 0, Atomics = 1 << 0, MutableGlobals = 1 << 1, @@ -47,11 +49,16 @@ struct FeatureSet { TypedContinuations = 1 << 16, SharedEverything = 1 << 17, FP16 = 1 << 18, + BulkMemoryOpt = 1 << 19, // Just the memory.copy and fill operations + // This features is a no-op for compatibility. Having it in this list means + // that we can automatically generate tool flags that set it, but otherwise + // it does nothing. Binaryen always accepts LEB call-indirect encodings. + CallIndirectOverlong = 1 << 20, MVP = None, // Keep in sync with llvm default features: // https://github.com/llvm/llvm-project/blob/c7576cb89d6c95f03968076e902d3adfd1996577/clang/lib/Basic/Targets/WebAssembly.cpp#L150-L153 Default = SignExt | MutableGlobals, - All = (1 << 19) - 1, + All = (1 << 21) - 1, }; static std::string toString(Feature f) { @@ -94,6 +101,10 @@ struct FeatureSet { return "shared-everything"; case FP16: return "fp16"; + case BulkMemoryOpt: + return "bulk-memory-opt"; + case CallIndirectOverlong: + return "call-indirect-overlong"; default: WASM_UNREACHABLE("unexpected feature"); } @@ -145,6 +156,11 @@ struct FeatureSet { return (features & SharedEverything) != 0; } bool hasFP16() const { return (features & FP16) != 0; } + bool hasBulkMemoryOpt() const { + bool has = (features & BulkMemoryOpt) != 0; + assert(has || !hasBulkMemory()); + return has; + } bool hasAll() const { return (features & All) != 0; } void set(FeatureSet f, bool v = true) { @@ -169,6 +185,7 @@ struct FeatureSet { void setTypedContinuations(bool v = true) { set(TypedContinuations, v); } void setSharedEverything(bool v = true) { set(SharedEverything, v); } void setFP16(bool v = true) { set(FP16, v); } + void setBulkMemoryOpt(bool v = true) { set(BulkMemoryOpt, v); } void setMVP() { features = MVP; } void setAll() { features = All; } diff --git a/src/wasm/wasm-binary.cpp b/src/wasm/wasm-binary.cpp index f710c8bc1..0c931f518 100644 --- a/src/wasm/wasm-binary.cpp +++ b/src/wasm/wasm-binary.cpp @@ -1354,6 +1354,10 @@ void WasmBinaryWriter::writeFeaturesSection() { return BinaryConsts::CustomSections::SharedEverythingFeature; case FeatureSet::FP16: return BinaryConsts::CustomSections::FP16Feature; + case FeatureSet::BulkMemoryOpt: + return BinaryConsts::CustomSections::BulkMemoryOptFeature; + case FeatureSet::CallIndirectOverlong: + return BinaryConsts::CustomSections::CallIndirectOverlongFeature; case FeatureSet::None: case FeatureSet::Default: case FeatureSet::All: @@ -4790,6 +4794,12 @@ void WasmBinaryReader::readFeatures(size_t payloadLen) { feature = FeatureSet::Atomics; } else if (name == BinaryConsts::CustomSections::BulkMemoryFeature) { feature = FeatureSet::BulkMemory; + if (used) { + // For backward compatibility, enable this dependent feature. + feature |= FeatureSet::BulkMemoryOpt; + } + } else if (name == BinaryConsts::CustomSections::BulkMemoryOptFeature) { + feature = FeatureSet::BulkMemoryOpt; } else if (name == BinaryConsts::CustomSections::ExceptionHandlingFeature) { feature = FeatureSet::ExceptionHandling; } else if (name == BinaryConsts::CustomSections::MutableGlobalsFeature) { diff --git a/src/wasm/wasm-validator.cpp b/src/wasm/wasm-validator.cpp index e295d3931..242e07c43 100644 --- a/src/wasm/wasm-validator.cpp +++ b/src/wasm/wasm-validator.cpp @@ -1526,10 +1526,10 @@ void FunctionValidator::visitDataDrop(DataDrop* curr) { } void FunctionValidator::visitMemoryCopy(MemoryCopy* curr) { - shouldBeTrue( - getModule()->features.hasBulkMemory(), - curr, - "Bulk memory operations require bulk memory [--enable-bulk-memory]"); + shouldBeTrue(getModule()->features.hasBulkMemoryOpt(), + curr, + "memory.copy operations require bulk memory operations " + "[--enable-bulk-memory-opt]"); shouldBeEqualOrFirstIsUnreachable( curr->type, Type(Type::none), curr, "memory.copy must have type none"); auto* destMemory = getModule()->getMemoryOrNull(curr->destMemory); @@ -1561,9 +1561,9 @@ void FunctionValidator::visitMemoryCopy(MemoryCopy* curr) { void FunctionValidator::visitMemoryFill(MemoryFill* curr) { auto* memory = getModule()->getMemoryOrNull(curr->memory); shouldBeTrue( - getModule()->features.hasBulkMemory(), + getModule()->features.hasBulkMemoryOpt(), curr, - "Bulk memory operations require bulk memory [--enable-bulk-memory]"); + "memory.fill operations require bulk memory [--enable-bulk-memory-opt]"); shouldBeEqualOrFirstIsUnreachable( curr->type, Type(Type::none), curr, "memory.fill must have type none"); shouldBeEqualOrFirstIsUnreachable( diff --git a/src/wasm/wasm.cpp b/src/wasm/wasm.cpp index f5806b184..3e04f29ec 100644 --- a/src/wasm/wasm.cpp +++ b/src/wasm/wasm.cpp @@ -57,6 +57,8 @@ const char* MultiMemoryFeature = "multimemory"; const char* TypedContinuationsFeature = "typed-continuations"; const char* SharedEverythingFeature = "shared-everything"; const char* FP16Feature = "fp16"; +const char* BulkMemoryOptFeature = "bulk-memory-opt"; +const char* CallIndirectOverlongFeature = "call-indirect-overlong"; } // namespace CustomSections } // namespace BinaryConsts |