summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorSoni L. <EnderMoneyMod@gmail.com>2024-09-23 17:01:18 -0300
committerGitHub <noreply@github.com>2024-09-23 13:01:18 -0700
commit0a2a59fae573c2d793501720a2faf986118ee4f4 (patch)
treea95fd371a30945944121f1ae816491b1daf1031e /src
parentb287470b53ef074332c4b0056cfc05486a85c4a7 (diff)
downloadwabt-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.cc1
-rw-r--r--src/c-writer.cc70
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();