Probieren Sie es aus
(module
;; Define function type
(type $ret_i32 (func (result i32)))
;; table with 2 function slots
(table $return_values 2 funcref)
;; Define functions of that type
(func $f1 (type $ret_i32)
(i32.const 42)
)
(func $f2 (type $ret_i32)
(i32.const 100)
)
;; initialize table slots
(elem (i32.const 0) $f1 $f2)
(func (export "accessTable") (param $index i32) (result i32)
(local.get $index)
(call_indirect (type $ret_i32))
)
)
WebAssembly.instantiateStreaming(fetch("{%wasm-url%}")).then((result) => {
const value = result.instance.exports.accessTable(1);
console.log(value);
});
Syntax
table name initial_size max_size type
table-
Der
table-Defintionstyp. Muss immer zuerst angegeben werden. nameOptional-
Ein optionaler Name zur Identifizierung der Tabelle. Dieser muss mit einem
$-Symbol beginnen, zum Beispiel$my_table. Wenn dieser weggelassen wird, kann die Tabelle durch ihren Index identifiziert werden, zum Beispiel0für die erste Tabelle im Wasm-Skript,1für die zweite usw. initial_size-
Eine ganze Zahl, die die anfängliche Größe der Tabelle darstellt.
max_sizeOptional-
Eine ganze Zahl, die die maximale Größe angibt, die die Tabelle erreichen darf. Wenn diese nicht angegeben wird, hat die Tabelle keine maximale Größe, und ihr Wachstum ist nur durch Systembeschränkungen wie verfügbaren Speicher begrenzt.
type-
Der Name des zu speichernden Funktionstyps. Mögliche Werte sind:
Beschreibung
WebAssembly-Tabellen ermöglichen die Speicherung von Referenzwerten, die nicht im byte-orientierten WebAssembly-Speicher liegen. Der Hauptanwendungsfall ist das Speichern von Funktionsreferenzen, die mit call_indirect verwendet werden können, um indirekte Funktionsaufrufe für Sprachen zu unterstützen, die solche bieten. Die table-Definition erstellt eine neue Tabelle.
Einer Tabelle muss eine anfängliche Größe und ein Speichertyp zugewiesen werden. Dieses Beispiel erstellt eine Tabelle mit zwei Speicherplätzen, die nur Referenzen auf Funktionen speichern, die innerhalb von Wasm erstellt wurden (gekennzeichnet durch funcref):
(table 2 funcref)
Optional können Sie auch einen Bezeichner mit angeben, der verwendet werden kann, um die Tabelle an anderer Stelle zu identifizieren, sowie eine maximale Wachstumsgröße. Zum Beispiel:
(table $mytable 2 10 funcref)
Im Folgenden wird ein Funktionstyp definiert, eine grundlegende Funktion mit diesem Typ, die einen i32 zurückgibt, wird definiert, und sie wird mithilfe von (elem declare func $f1) vorangemeldet, um später darauf zuzugreifen.
(type $ret_i32 (func (result i32)))
(func $f1 (type $ret_i32)
(i32.const 42)
)
(elem declare func $f1)
Um eine in einer Tabelle referenzierte Funktion aufzurufen, müssen Sie die Tabelle und den Indexwert, bei dem die Funktionsreferenz gespeichert ist, angeben. Das folgende Beispiel verwendet call_indirect:
(call_indirect (type $ret_i32) (local.get $index))
Es ist möglich, Tabellen zur Laufzeit mit Anweisungen wie table.set und table.fill zu ändern und Werte mit table.get abzurufen.
Externe Referenzen
Sie können auch externe in JavaScript definierte Referenzen in einer Wasm-Tabelle speichern, indem Sie das Schlüsselwort externref spezifizieren. Zum Beispiel:
(table $mytable 2 10 externref)
Mehrere Tabellen
Sie können mehrere Tabellen im selben Wasm-Modul erstellen, zum Beispiel:
(table $table_1 1 2 funcref)
(table $table_2 1 2 funcref)
Sie könnten eine Funktion wie diese verwenden, um jede Tabelle mit einer anderen Funktion zu befüllen:
(func $populate
(table.set $table_1
(i32.const 0)
(ref.func $f1)
)
(table.set $table_2
(i32.const 0)
(ref.func $f2)
)
)
Dann könnten Sie die $populate-Funktion aufrufen und call_indirect verwenden, um die in jeder Tabelle referenzierten Funktionen aufzurufen. Das folgende Snippet verweist auf die Tabellen anhand ihres Bezeichnernamens:
(func (export "accessTable")
(call $populate)
(call_indirect $table_1 (type $ret_i32) (i32.const 0))
(call_indirect $table_2 (type $ret_i32) (i32.const 0))
...
)
Aber Sie könnten stattdessen die Tabellen anhand ihrer Indexwerte referenzieren (0 bezeichnet die erste Tabelle im Modul, 1 die zweite Tabelle usw.):
(call_indirect 0 (type $ret_i32) (i32.const 0))
(call_indirect 1 (type $ret_i32) (i32.const 0))
Wenn Sie keinen Bezeichnernamen oder einen Index angeben, wird angenommen, dass der Index 0 ist:
;; Accesses the table with index 0
(call_indirect (type $ret_i32) (i32.const 0))
Beispiele
>Erstellung einer Basistabelle
Dieses Beispiel zeigt, wie man eine Basistabelle erstellt, ein paar Funktionen darin speichert und dann eine Funktion aus der Tabelle aufruft.
JavaScript
In unserem Skript beginnen wir mit dem Abrufen einer Referenz zu einem <p>-Element, in das wir Ergebnisse ausgeben werden. Dann kompilieren und instanziieren wir unser Wasm-Modul mit der Methode WebAssembly.instantiateStreaming(). Wenn das Ergebnis zurückgegeben wird, rufen wir die exportierte Wasm-Funktion accessTable() auf, die im WebAssembly-Objekt Instance exports verfügbar ist, und übergeben ihr die Zahl 0 als Parameter. Schließlich setzen wir den Rückgabewert der accessTable()-Funktion als textContent-Wert des <p>-Elements, damit wir ihn untersuchen können.
const output = document.querySelector("p");
WebAssembly.instantiateStreaming(fetch("{%wasm-url%}")).then((result) => {
const value = result.instance.exports.accessTable(0);
output.textContent = value;
});
Wasm
In unserem Wasm-Modul definieren wir zunächst einen Funktionstyp namens $ret_i32, der einen i32-Wert zurückgibt. Dann definieren wir zwei auf diesem Typ basierende Funktionen namens $f1 und $f2, die die innerhalb definierten Werte zurückgeben. Anschließend definieren wir eine table namens $return_values mit zwei Slots, die Funktionsreferenzen speichert (deshalb wird funcref angegeben), und initialisieren diese, indem wir die beiden Slots mit Referenzen auf die $f1 und $f2 Funktionen füllen.
Schließlich exportieren wir die accessTable()-Funktion, die einen i32-Wert namens $index als Parameter akzeptiert und einen i32 zurückgibt. Im Funktionskörper verwenden wir call_indirect, um die Funktion aufzurufen, die in der Tabelle am Indexwert $index referenziert wird.
(module
(type $ret_i32 (func (result i32)))
(func $f1 (type $ret_i32)
(i32.const 42)
)
(func $f2 (type $ret_i32)
(i32.const 100)
)
(table $return_values 2 funcref)
(elem (i32.const 0) $f1 $f2)
(func (export "accessTable") (param $index i32) (result i32)
(call_indirect (type $ret_i32) (local.get $index))
)
)
Ergebnis
Der ausgegebene Wert ist wie folgt:
Das ergibt Sinn, da der exportierte accessTable()-Funktion ein Indexwert übergeben wird. Innerhalb des Wasm-Moduls rufen wir die Funktion auf, die bei diesem Index in der definierten Tabelle verfügbar ist, was den Wert zurückgibt, den wir ausgegeben sehen.
Spezifikationen
| Specification |
|---|
| Unknown specification> # syntax-table> |