diff options
author | Jérôme Vouillon <jerome.vouillon@gmail.com> | 2023-06-29 22:20:53 +0200 |
---|---|---|
committer | GitHub <noreply@github.com> | 2023-06-29 13:20:53 -0700 |
commit | 5f6594c52d1a9a285f7bfcfd31c46f5062c24b19 (patch) | |
tree | d8bdb42ea6dc47c507a58bf25e69bd1107aca96f | |
parent | 7b2e06aa3a0b6cef9568c3eab010a7187b9b6803 (diff) | |
download | binaryen-5f6594c52d1a9a285f7bfcfd31c46f5062c24b19.tar.gz binaryen-5f6594c52d1a9a285f7bfcfd31c46f5062c24b19.tar.bz2 binaryen-5f6594c52d1a9a285f7bfcfd31c46f5062c24b19.zip |
[wasm-metadce] Note ref.func connections + fix rooting of segment offsets (#5791)
-rw-r--r-- | src/tools/wasm-metadce.cpp | 41 | ||||
-rw-r--r-- | test/metadce/ref-func.wast | 11 | ||||
-rw-r--r-- | test/metadce/ref-func.wast.dced | 13 | ||||
-rw-r--r-- | test/metadce/ref-func.wast.dced.stdout | 0 | ||||
-rw-r--r-- | test/metadce/ref-func.wast.graph.txt | 23 | ||||
-rw-r--r-- | test/metadce/segments.wast | 14 | ||||
-rw-r--r-- | test/metadce/segments.wast.dced | 18 | ||||
-rw-r--r-- | test/metadce/segments.wast.dced.stdout | 0 | ||||
-rw-r--r-- | test/metadce/segments.wast.graph.txt | 19 |
9 files changed, 126 insertions, 13 deletions
diff --git a/src/tools/wasm-metadce.cpp b/src/tools/wasm-metadce.cpp index 029d4aae8..e5e7d8706 100644 --- a/src/tools/wasm-metadce.cpp +++ b/src/tools/wasm-metadce.cpp @@ -195,6 +195,17 @@ struct MetaDCEGraph { void visitGlobalGet(GlobalGet* curr) { handleGlobal(curr->name); } void visitGlobalSet(GlobalSet* curr) { handleGlobal(curr->name); } + void visitRefFunc(RefFunc* curr) { + Name dceName; + if (!getModule()->getFunction(curr->func)->imported()) { + dceName = parent->functionToDCENode[curr->func]; + } else { + dceName = + parent->importIdToDCENode[parent->getFunctionImportId(curr->func)]; + } + assert(!parentDceName.isNull()); + parent->nodes[parentDceName].reaches.push_back(dceName); + } private: MetaDCEGraph* parent; @@ -209,7 +220,9 @@ struct MetaDCEGraph { // it's an import. dceName = parent->importIdToDCENode[parent->getGlobalImportId(name)]; } - if (!parentDceName.isNull()) { + if (parentDceName.isNull()) { + parent->roots.insert(dceName); + } else { parent->nodes[parentDceName].reaches.push_back(dceName); } } @@ -248,18 +261,8 @@ struct MetaDCEGraph { return std::make_unique<Scanner>(parent); } - void visitCall(Call* curr) { - if (!getModule()->getFunction(curr->target)->imported()) { - parent->nodes[parent->functionToDCENode[getFunction()->name]] - .reaches.push_back(parent->functionToDCENode[curr->target]); - } else { - assert(parent->functionToDCENode.count(getFunction()->name) > 0); - parent->nodes[parent->functionToDCENode[getFunction()->name]] - .reaches.push_back( - parent - ->importIdToDCENode[parent->getFunctionImportId(curr->target)]); - } - } + void visitCall(Call* curr) { handleFunction(curr->target); } + void visitRefFunc(RefFunc* curr) { handleFunction(curr->func); } void visitGlobalGet(GlobalGet* curr) { handleGlobal(curr->name); } void visitGlobalSet(GlobalSet* curr) { handleGlobal(curr->name); } void visitThrow(Throw* curr) { handleTag(curr->tag); } @@ -272,6 +275,18 @@ struct MetaDCEGraph { private: MetaDCEGraph* parent; + void handleFunction(Name name) { + if (!getModule()->getFunction(name)->imported()) { + parent->nodes[parent->functionToDCENode[getFunction()->name]] + .reaches.push_back(parent->functionToDCENode[name]); + } else { + assert(parent->functionToDCENode.count(getFunction()->name) > 0); + parent->nodes[parent->functionToDCENode[getFunction()->name]] + .reaches.push_back( + parent->importIdToDCENode[parent->getFunctionImportId(name)]); + } + } + void handleGlobal(Name name) { if (!getFunction()) { return; // non-function stuff (initializers) are handled separately diff --git a/test/metadce/ref-func.wast b/test/metadce/ref-func.wast new file mode 100644 index 000000000..b7d2b8ac7 --- /dev/null +++ b/test/metadce/ref-func.wast @@ -0,0 +1,11 @@ +(module + (import "env" "f1" (func $f1)) + (import "env" "f2" (func $f2)) + + (export "g" (global $g)) + (export "f" (func $f)) + + (global $g funcref (ref.func $f1)) + + (func $f (result funcref) (ref.func $f2)) +) diff --git a/test/metadce/ref-func.wast.dced b/test/metadce/ref-func.wast.dced new file mode 100644 index 000000000..3117c4978 --- /dev/null +++ b/test/metadce/ref-func.wast.dced @@ -0,0 +1,13 @@ +(module + (type $none_=>_none (func)) + (type $none_=>_funcref (func (result funcref))) + (import "env" "f2" (func $f2 (type $none_=>_none))) + (import "env" "f1" (func $f1 (type $none_=>_none))) + (global $g funcref (ref.func $f1)) + (elem declare func $f2) + (export "g" (global $g)) + (export "f" (func $f)) + (func $f (type $none_=>_funcref) (result funcref) + (ref.func $f2) + ) +) diff --git a/test/metadce/ref-func.wast.dced.stdout b/test/metadce/ref-func.wast.dced.stdout new file mode 100644 index 000000000..e69de29bb --- /dev/null +++ b/test/metadce/ref-func.wast.dced.stdout diff --git a/test/metadce/ref-func.wast.graph.txt b/test/metadce/ref-func.wast.graph.txt new file mode 100644 index 000000000..89adf046f --- /dev/null +++ b/test/metadce/ref-func.wast.graph.txt @@ -0,0 +1,23 @@ +[ + { + "name": "root", + "reaches": ["f", "g"], + "root": true + }, + { + "name": "f", + "export": "f" + }, + { + "name": "g", + "export": "g" + }, + { + "name": "f1", + "import": ["env", "f1"] + }, + { + "name": "f2", + "import": ["env", "f2"] + }, +] diff --git a/test/metadce/segments.wast b/test/metadce/segments.wast new file mode 100644 index 000000000..96e0088e8 --- /dev/null +++ b/test/metadce/segments.wast @@ -0,0 +1,14 @@ +(module + (import "env" "g1" (global $g1 i32)) + (import "env" "g2" (global $g2 i32)) + + (table $tbl funcref) + (elem (offset (global.get $g1)) funcref (ref.func $f)) + + (memory 3) + (data (offset (global.get $g2)) "xxx") + + (export "f" (func $f)) + (func $f (param i32) + (call_indirect (param i32) (local.get 0) (i32.load8_u (i32.const 0)))) +) diff --git a/test/metadce/segments.wast.dced b/test/metadce/segments.wast.dced new file mode 100644 index 000000000..944614bc2 --- /dev/null +++ b/test/metadce/segments.wast.dced @@ -0,0 +1,18 @@ +(module + (type $i32_=>_none (func (param i32))) + (import "env" "g1" (global $g1 i32)) + (import "env" "g2" (global $g2 i32)) + (memory $0 3) + (data $0 (global.get $g2) "xxx") + (table $tbl 0 funcref) + (elem $0 (global.get $g1) $f) + (export "f" (func $f)) + (func $f (type $i32_=>_none) (param $0 i32) + (call_indirect $tbl (type $i32_=>_none) + (local.get $0) + (i32.load8_u + (i32.const 0) + ) + ) + ) +) diff --git a/test/metadce/segments.wast.dced.stdout b/test/metadce/segments.wast.dced.stdout new file mode 100644 index 000000000..e69de29bb --- /dev/null +++ b/test/metadce/segments.wast.dced.stdout diff --git a/test/metadce/segments.wast.graph.txt b/test/metadce/segments.wast.graph.txt new file mode 100644 index 000000000..eed62cced --- /dev/null +++ b/test/metadce/segments.wast.graph.txt @@ -0,0 +1,19 @@ +[ + { + "name": "root", + "reaches": ["f"], + "root": true + }, + { + "name": "f", + "export": "f" + }, + { + "name": "g1", + "import": ["env", "g1"] + }, + { + "name": "g2", + "import": ["env", "g2"] + }, +] |