grow: Wasm-Tabellenanweisung
Baseline
Weitgehend verfügbar
Diese Funktion ist gut etabliert und funktioniert auf vielen Geräten und in vielen Browserversionen. Sie ist seit November 2021 browserübergreifend verfügbar.
Die table.grow Tabellenanweisung erhöht die Größe einer Tabelle um eine angegebene Anzahl von Elementen.
Probieren Sie es aus
(module
;; Define function type
(type $ret_i32 (func (result i32)))
;; table with 0 function slots
(table $return_funcs 0 funcref)
;; Define a function of the specified type
(func $f1 (type $ret_i32)
(i32.const 42)
)
(elem declare func $f1)
(func $populate
(table.grow $return_funcs
ref.null func
(i32.const 1)
)
(drop)
(table.set $return_funcs
(i32.const 0)
(ref.func $f1)
)
)
(func (export "run") (result i32)
(call $populate)
(call_indirect (type $ret_i32) (i32.const 0))
)
)
WebAssembly.instantiateStreaming(fetch("{%wasm-url%}")).then((result) => {
const value = result.instance.exports.run();
console.log(value);
});
Syntax
table.grow identifier
table.grow-
Der
table.growAnweisungstyp. Muss immer zuerst angegeben werden. identifierOptional-
Ein Bezeichner für die Tabelle, die Sie vergrößern möchten. Dies kann eines der folgenden sein:
name-
Ein identifizierender Name für die Tabelle festgelegt, als sie zuerst erstellt wurde. Dieser muss mit einem
$-Symbol beginnen, z.B.$my_table. index-
Die Indexnummer der Tabelle, z.B.
0für die erste Tabelle im Wasm-Skript,1für die zweite, usw.
Wenn der
identifierweggelassen wird, wird er standardmäßig auf0gesetzt.
Typ
[initial_value, grow_amount] -> [previous_length]
initial_value-
Der Anfangswert, der für die neuen Tabellenelemente festgelegt werden soll. Sein Typ muss derselbe sein wie der
Typ, der ursprünglich bei der Erstellung der Tabelle festgelegt wurde. grow_amount-
Die Anzahl der Elemente, um die die Tabelle vergrößert werden soll. Dies muss ein
i32Wert sein, z.B.(i32.const 1). previous_length-
Ein
i32, das der Größe der Tabelle entspricht, bevor diegrowAnweisung darauf angewendet wird, oder-1, wenn das Wachstum der Tabelle fehlgeschlagen ist, z.B. aufgrund eines Speicherfehlers (OOM) oder weil die neue Größe größer als die maximale Größe der Tabelle ist.
Um die neue Tabellengröße zu ermitteln, nachdem die grow Anweisung darauf angewendet wurde, verwenden Sie die table.size Anweisung.
Opcodes
| Anweisung | Binärformat | Beispieltext => binär |
|---|---|---|
table.grow |
0xfc 15:u32 𝑥:table_idx |
(table.grow (i32.const 1) (ref.null func)) => 0xfc 0x0f 0x00 0x41 0x01 0xd0 0x70 |
Beschreibung
table.grow wird verwendet, um die Größe einer Tabelle um eine angegebene Anzahl von Elementen zu erhöhen.
Eine Wasm-Tabelle kann von JavaScript aus mit der table.grow() Methode vergrößert werden.
Beispiele
>Erstellen, Vergrößern und Setzen einer Tabelle
Dieses Beispiel zeigt, wie man eine Tabelle erstellt, ihre Größe vergrößert, die darin gespeicherten Funktionen dynamisch ändert und dann die Funktion aufruft, die zu jedem Zeitpunkt in der Tabelle gespeichert ist.
JavaScript
In unserem Skript beginnen wir damit, eine Referenz zu einem <p> Element zu erhalten, zu dem wir Ergebnisse ausgeben werden. Wir definieren dann ein obj-Objekt, das eine Funktion namens output() enthält, die einen gegebenen Wert zum textContent eines gegebenen Elements hinzufügt.
Wir kompilieren und instanziieren dann unser Wasm-Modul mit der WebAssembly.instantiateStreaming() Methode und importieren dabei das obj Objekt.
Wenn das Ergebnis zurückgegeben wird, rufen wir die exportierte Wasm run() Funktion auf, die im WebAssembly Instance exports Objekt verfügbar ist, und übergeben ihr das outputElem Element als Parameter.
const outputElem = document.querySelector("p");
const obj = {
output(elem, val) {
elem.textContent += `${val} `;
},
};
WebAssembly.instantiateStreaming(fetch("{%wasm-url%}"), {
obj,
}).then((result) => {
value = result.instance.exports.run(outputElem);
});
Wasm
In unserem Wasm-Modul importieren wir zunächst die JavaScript output() Funktion und stellen sicher, dass sie zwei Parameter hat, einen externref und einen i32.
Als nächstes definieren wir einen Funktionstyp namens $ret_i32, der einen i32 Wert zurückgibt. Wir definieren dann zwei Funktionen auf Basis dieses Typs namens $f1 und $f2, die die darin definierten Werte zurückgeben und deklarieren sie mit (elem declare func $f1 $f2), damit sie später referenziert werden können. Danach definieren wir eine table namens $func_table, die Funktionsreferenzen (daher funcref) speichert und zunächst leer ist.
Schließlich exportieren wir die run() Funktion, die einen externref namens $elem als Parameter nimmt. In der Funktionskörper:
- Verwenden wir
table.grow, um die Tabellengröße um1zu erhöhen, mit einem initialenref.nullWert, und überprüfen, ob das Operationsergebnis-1ist, was auf ein Scheitern hinweisen würde. - Setzen wir unser Tabellenelement, um die
$f1Funktion mittable.setzu enthalten, und rufen dann die importierte$outputFunktion auf, indem wir als Parameter den$elemexternref, der in dieoutput()Funktion übergeben wurde, und den Wert, der von der$f1Funktion zurückgegeben wird und aus der Tabelle durch Verwendung von(call_indirect (type $ret_i32) (i32.const 0))referenziert wird, übergeben. - Setzen wir unser Tabellenelement, um die
$f2Funktion mittable.setzu enthalten, und rufen dann dieoutput()Funktion erneut auf.
(module
;; Import output function
(import "obj" "output" (func $output (param externref) (param i32)))
;; Define function type
(type $ret_i32 (func (result i32)))
;; Define basic functions that return i32s
(func $f1 (result i32)
(i32.const 42)
)
(func $f2 (result i32)
(i32.const 100)
)
(elem declare func $f1 $f2)
;; Define an initially empty table of funcrefs
(table $func_table 0 funcref)
(func (export "run") (param $elem externref)
;; Grow the table by 1, setting the initial value to null.
;; Check the result for -1, which indicates failure.
(if
(i32.eq
(table.grow $func_table
ref.null func
(i32.const 1)
)
i32.const -1
)
;; Trap if we failed to grow the table
(then unreachable)
)
;; Set the first function in the table to f1
(table.set $func_table
(i32.const 0)
(ref.func $f1)
)
;; Call the output function, to output the table
;; function's return value to the DOM
(call $output
(local.get $elem)
(call_indirect (type $ret_i32) (i32.const 0))
)
;; Set the first function in the table to f2
(table.set $func_table
(i32.const 0)
(ref.func $f2)
)
;; Call the output function, to output the table
;; function's return value to the DOM
(call $output
(local.get $elem)
(call_indirect (type $ret_i32) (i32.const 0))
)
)
)
Ergebnis
Die Ausgabe ist wie folgt:
Dies macht Sinn, da jedes Mal, wenn die output() Funktion von innerhalb des Wasm-Moduls ausgeführt wird, der Wert, der als zweiter Parameter übergeben wird, in unserem Ergebnis-<p> im DOM gedruckt wird. Jeder Wert ist der von den $f1 und $f2 Funktionen zurückgegebene Wert — 42 und 100 jeweils.
Spezifikationen
| Spezifikation |
|---|
| Unknown specification> # syntax-instr-table> |