diff options
author | Soni L. <EnderMoneyMod@gmail.com> | 2024-09-23 17:01:18 -0300 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-09-23 13:01:18 -0700 |
commit | 0a2a59fae573c2d793501720a2faf986118ee4f4 (patch) | |
tree | a95fd371a30945944121f1ae816491b1daf1031e /src | |
parent | b287470b53ef074332c4b0056cfc05486a85c4a7 (diff) | |
download | wabt-0a2a59fae573c2d793501720a2faf986118ee4f4.tar.gz wabt-0a2a59fae573c2d793501720a2faf986118ee4f4.tar.bz2 wabt-0a2a59fae573c2d793501720a2faf986118ee4f4.zip |
wasm2c: Use wrappers for function references (#2465)
Clang 17(?) tightened UBSAN checks, so that you now get this:
```
- test/wasm2c/spec/call_indirect.txt
expected error code 0, got 1.
STDERR MISMATCH:
--- expected
+++ actual
@@ -0,0 +1,3 @@
+out/test/wasm2c/spec/call_indirect/call_indirect.0.c:2144:12: runtime error: call to function w2c_call__indirect__0__wasm_f0 through pointer to incorrect function type 'unsigned int (*)(void *)'
+/home/runner/work/wabt/wabt/out/test/wasm2c/spec/call_indirect/call_indirect.0.c:1925: note: w2c_call__indirect__0__wasm_f0 defined here
+SUMMARY: UndefinedBehaviorSanitizer: undefined-behavior out/test/wasm2c/spec/call_indirect/call_indirect.0.c:2144:12
STDOUT MISMATCH:
--- expected
+++ actual
@@ -1 +0,0 @@
-134/134 tests passed.
```
This happens because emitted functions use a typed module instance,
while function references use a `void*` instance. It is UB in C to call
the former with the latter, so clang is correct here.
We had to pick one of two ways to fix this: either emit `void*` wrapper
functions that do the appropriate downcasting for any module functions
that go into a table (potentially including imported functions), or the
approach that takes significantly less effort of changing everything to
`void*` and downcasting internally. ~~We obviously chose the latter.~~
We eventually started emitting wrapper functions.
Diffstat (limited to 'src')
-rw-r--r-- | src/binary-reader-ir.cc | 1 | ||||
-rw-r--r-- | src/c-writer.cc | 70 |
2 files changed, 65 insertions, 6 deletions
diff --git a/src/binary-reader-ir.cc b/src/binary-reader-ir.cc index 8c79cd52..0b3e6f4d 100644 --- a/src/binary-reader-ir.cc +++ b/src/binary-reader-ir.cc @@ -1119,6 +1119,7 @@ Result BinaryReaderIR::OnTableFillExpr(Index table_index) { } Result BinaryReaderIR::OnRefFuncExpr(Index func_index) { + module_->used_func_refs.insert(func_index); return AppendExpr( std::make_unique<RefFuncExpr>(Var(func_index, GetLocation()))); } diff --git a/src/c-writer.cc b/src/c-writer.cc index cbffa1d4..9704f0e2 100644 --- a/src/c-writer.cc +++ b/src/c-writer.cc @@ -103,6 +103,11 @@ struct ExternalRef : GlobalName { using GlobalName::GlobalName; }; +struct WrapperRef : GlobalName { + explicit WrapperRef(const std::string& name) + : GlobalName(ModuleFieldType::Func, name) {} +}; + struct TailCallRef : GlobalName { explicit TailCallRef(const std::string& name) : GlobalName(ModuleFieldType::Func, name) {} @@ -342,6 +347,7 @@ class CWriter { void Write(const GlobalName&); void Write(const TagSymbol&); void Write(const ExternalRef&); + void Write(const WrapperRef&); void Write(const TailCallRef&); void Write(const ExternalInstancePtr&); void Write(const ExternalInstanceRef&); @@ -379,6 +385,7 @@ class CWriter { void WriteImportFuncDeclaration(const FuncDeclaration&, const std::string& module_name, const std::string&); + void WriteFuncRefWrapper(const Func* func); void WriteCallIndirectFuncDeclaration(const FuncDeclaration&, const std::string&); void ComputeSimdScope(); @@ -402,6 +409,7 @@ class CWriter { void WriteDataInitializers(); void WriteElemInitializerDecls(); void WriteElemInitializers(); + void WriteFuncRefWrappers(); void WriteElemTableInit(bool, const ElemSegment*, const Table*); bool IsSingleUnsharedMemory(); void InstallSegueBase(Memory* memory, bool save_old_value); @@ -544,6 +552,7 @@ static constexpr char kParamSuffix = static constexpr char kLabelSuffix = kParamSuffix + 1; static constexpr char kGlobalSymbolPrefix[] = "w2c_"; +static constexpr char kWrapperSymbolPrefix[] = "wrap_"; static constexpr char kLocalSymbolPrefix[] = "var_"; static constexpr char kAdminSymbolPrefix[] = "wasm2c_"; static constexpr char kTailCallSymbolPrefix[] = "wasm_tailcall_"; @@ -1125,6 +1134,10 @@ void CWriter::Write(const ExternalRef& name) { } } +void CWriter::Write(const WrapperRef& name) { + Write(kWrapperSymbolPrefix, GlobalName(name)); +} + void CWriter::Write(const TailCallRef& name) { Write(GetTailCallRef(name.name)); } @@ -1446,8 +1459,7 @@ void CWriter::WriteInitExprTerminal(const Expr* expr) { const FuncType* func_type = module_->GetFuncType(decl.type_var); Write("(wasm_rt_funcref_t){", FuncTypeExpr(func_type), ", ", - "(wasm_rt_function_ptr_t)", - ExternalRef(ModuleFieldType::Func, func->name), ", {"); + "(wasm_rt_function_ptr_t)", WrapperRef(func->name), ", {"); if (options_.features.tail_call_enabled() && (IsImport(func->name) || func->features_used.tailcall)) { Write(TailCallRef(func->name)); @@ -2286,6 +2298,53 @@ void CWriter::WriteElemInitializerDecls() { } } +void CWriter::WriteFuncRefWrapper(const Func* func) { + Write("static ", func->decl.sig.result_types, " ", WrapperRef(func->name)); + Write("(void *instance"); + if (func->GetNumParams() != 0) { + Indent(4); + for (Index i = 0; i < func->GetNumParams(); ++i) { + Write(", "); + if (i != 0 && (i % 8) == 0) { + Write(Newline()); + } + Write(func->GetParamType(i), " ", kLocalSymbolPrefix); + Writef("%d", i); + } + Dedent(4); + } + Write(") ", OpenBrace()); + Write("return "); + Write(ExternalRef(ModuleFieldType::Func, func->name)); + Write("(instance"); + if (func->GetNumParams() != 0) { + Indent(4); + for (Index i = 0; i < func->GetNumParams(); ++i) { + Write(", "); + if (i != 0 && (i % 8) == 0) { + Write(Newline()); + } + Write(kLocalSymbolPrefix); + Writef("%d", i); + } + Dedent(4); + } + Write(");", Newline(), CloseBrace(), Newline()); +} + +void CWriter::WriteFuncRefWrappers() { + std::set<std::string> unique_func_wrappers; + + for (Index index : module_->used_func_refs) { + assert(index < module_->funcs.size()); + const Func* func = module_->funcs[index]; + if (unique_func_wrappers.count(func->name) == 0) { + WriteFuncRefWrapper(func); + unique_func_wrappers.insert(func->name); + } + } +} + void CWriter::WriteElemInitializers() { if (module_->tables.empty()) { return; @@ -2315,8 +2374,7 @@ void CWriter::WriteElemInitializers() { const Func* func = module_->GetFunc(cast<RefFuncExpr>(&expr)->var); const FuncType* func_type = module_->GetFuncType(func->decl.type_var); Write("{RefFunc, ", FuncTypeExpr(func_type), - ", (wasm_rt_function_ptr_t)", - ExternalRef(ModuleFieldType::Func, func->name), ", {"); + ", (wasm_rt_function_ptr_t)", WrapperRef(func->name), ", {"); if (options_.features.tail_call_enabled() && (IsImport(func->name) || func->features_used.tailcall)) { Write(TailCallRef(func->name)); @@ -3740,8 +3798,7 @@ void CWriter::Write(const ExprList& exprs) { const FuncType* func_type = module_->GetFuncType(decl.type_var); Write(StackVar(0), " = (wasm_rt_funcref_t){", FuncTypeExpr(func_type), - ", (wasm_rt_function_ptr_t)", - ExternalRef(ModuleFieldType::Func, func->name), ", {"); + ", (wasm_rt_function_ptr_t)", WrapperRef(func->name), ", {"); if (options_.features.tail_call_enabled() && (IsImport(func->name) || func->features_used.tailcall)) { Write(TailCallRef(func->name)); @@ -5858,6 +5915,7 @@ void CWriter::WriteCSource() { WriteMultiCTop(); WriteFuncTypes(); WriteTags(); + WriteFuncRefWrappers(); WriteGlobalInitializers(); WriteDataInitializers(); WriteElemInitializers(); |