forked from OSchip/llvm-project
[lld][WebAssembly] Allow element sections for nonzero table numbers
This patch fixes LLD to allow element sections for tables whose number is nonzero. We also add a test for linking multiple tables, showing that nonzero table numbers for the indirect function table, user-declared imported tables, and local user table definitions work. Differential Revision: https://reviews.llvm.org/D92321
This commit is contained in:
parent
202ae987d3
commit
51f1ddf8cb
|
@ -0,0 +1,152 @@
|
|||
# RUN: not llvm-mc -filetype=obj -triple=wasm32-unknown-unknown -o %t.a1.o %s 2>&1 | FileCheck %s --check-prefix=MVP
|
||||
# RUN: llvm-mc -filetype=obj -triple=wasm32-unknown-unknown -mattr=+reference-types -o %t.a1.rt.o %s
|
||||
# RUN: llvm-mc -filetype=obj -triple=wasm32-unknown-unknown %p/Inputs/call-indirect.s -o %t.a2.o
|
||||
# RUN: llvm-mc -filetype=obj -triple=wasm32-unknown-unknown -mattr=+reference-types %p/Inputs/call-indirect.s -o %t.a2.rt.o
|
||||
# RUN: not wasm-ld --allow-undefined --export-dynamic --no-entry -o %t.wasm %t.a1.rt.o %t.a2.o 2>&1 | FileCheck %s --check-prefix=RT-MVP
|
||||
# RUN: wasm-ld --allow-undefined --export-dynamic --no-entry -o- %t.a1.rt.o %t.a2.rt.o | obj2yaml | FileCheck %s
|
||||
|
||||
.globl table_a
|
||||
.tabletype table_a, funcref
|
||||
|
||||
.globl table_b
|
||||
table_b:
|
||||
.tabletype table_b, funcref
|
||||
|
||||
.globl table_c
|
||||
table_c:
|
||||
.tabletype table_c, externref
|
||||
|
||||
.globl call_indirect_explicit_tables
|
||||
call_indirect_explicit_tables:
|
||||
.functype call_indirect_explicit_tables () -> ()
|
||||
i32.const 0
|
||||
call_indirect table_a, () -> ()
|
||||
# MVP: error: Expected (, instead got: table_a
|
||||
i32.const 0
|
||||
call_indirect table_b, () -> ()
|
||||
end_function
|
||||
|
||||
# RT-MVP: wasm-ld: error: object file not built with 'reference-types' feature conflicts with import of table table_a by file
|
||||
|
||||
# CHECK: --- !WASM
|
||||
# CHECK-NEXT: FileHeader:
|
||||
# CHECK-NEXT: Version: 0x1
|
||||
# CHECK-NEXT: Sections:
|
||||
# CHECK-NEXT: - Type: TYPE
|
||||
# CHECK-NEXT: Signatures:
|
||||
# CHECK-NEXT: - Index: 0
|
||||
# CHECK-NEXT: ParamTypes: []
|
||||
# CHECK-NEXT: ReturnTypes: []
|
||||
# CHECK-NEXT: - Index: 1
|
||||
# CHECK-NEXT: ParamTypes: []
|
||||
# CHECK-NEXT: ReturnTypes:
|
||||
# CHECK-NEXT: - I64
|
||||
# CHECK-NEXT: - Index: 2
|
||||
# CHECK-NEXT: ParamTypes: []
|
||||
# CHECK-NEXT: ReturnTypes:
|
||||
# CHECK-NEXT: - I32
|
||||
# CHECK-NEXT: - Type: IMPORT
|
||||
# CHECK-NEXT: Imports:
|
||||
# CHECK-NEXT: - Module: env
|
||||
# CHECK-NEXT: Field: table_a
|
||||
# CHECK-NEXT: Kind: TABLE
|
||||
# CHECK-NEXT: Table:
|
||||
# CHECK-NEXT: Index: 0
|
||||
# CHECK-NEXT: ElemType: FUNCREF
|
||||
# CHECK-NEXT: Limits:
|
||||
# CHECK-NEXT: Initial: 0x0
|
||||
# CHECK-NEXT: - Module: env
|
||||
# CHECK-NEXT: Field: foo
|
||||
# CHECK-NEXT: Kind: FUNCTION
|
||||
# CHECK-NEXT: SigIndex: 2
|
||||
# CHECK-NEXT: - Type: FUNCTION
|
||||
# CHECK-NEXT: FunctionTypes: [ 0, 1, 0 ]
|
||||
# CHECK-NEXT: - Type: TABLE
|
||||
# CHECK-NEXT: Tables:
|
||||
# CHECK-NEXT: - Index: 1
|
||||
# CHECK-NEXT: ElemType: FUNCREF
|
||||
# CHECK-NEXT: Limits:
|
||||
# CHECK-NEXT: Initial: 0x0
|
||||
# CHECK-NEXT: - Index: 2
|
||||
# CHECK-NEXT: ElemType: EXTERNREF
|
||||
# CHECK-NEXT: Limits:
|
||||
# CHECK-NEXT: Initial: 0x0
|
||||
# CHECK-NEXT: - Index: 3
|
||||
# CHECK-NEXT: ElemType: FUNCREF
|
||||
# CHECK-NEXT: Limits:
|
||||
# CHECK-NEXT: Flags: [ HAS_MAX ]
|
||||
# CHECK-NEXT: Initial: 0x3
|
||||
# CHECK-NEXT: Maximum: 0x3
|
||||
# CHECK-NEXT: - Type: MEMORY
|
||||
# CHECK-NEXT: Memories:
|
||||
# CHECK-NEXT: - Initial: 0x2
|
||||
# CHECK-NEXT: - Type: GLOBAL
|
||||
# CHECK-NEXT: Globals:
|
||||
# CHECK-NEXT: - Index: 0
|
||||
# CHECK-NEXT: Type: I32
|
||||
# CHECK-NEXT: Mutable: true
|
||||
# CHECK-NEXT: InitExpr:
|
||||
# CHECK-NEXT: Opcode: I32_CONST
|
||||
# CHECK-NEXT: Value: 66576
|
||||
# CHECK-NEXT: - Type: EXPORT
|
||||
# CHECK-NEXT: Exports:
|
||||
# CHECK-NEXT: - Name: memory
|
||||
# CHECK-NEXT: Kind: MEMORY
|
||||
# CHECK-NEXT: Index: 0
|
||||
# CHECK-NEXT: - Name: table_b
|
||||
# CHECK-NEXT: Kind: TABLE
|
||||
# CHECK-NEXT: Index: 1
|
||||
# CHECK-NEXT: - Name: table_c
|
||||
# CHECK-NEXT: Kind: TABLE
|
||||
# CHECK-NEXT: Index: 2
|
||||
# CHECK-NEXT: - Name: call_indirect_explicit_tables
|
||||
# CHECK-NEXT: Kind: FUNCTION
|
||||
# CHECK-NEXT: Index: 1
|
||||
# CHECK-NEXT: - Name: bar
|
||||
# CHECK-NEXT: Kind: FUNCTION
|
||||
# CHECK-NEXT: Index: 2
|
||||
# CHECK-NEXT: - Name: call_bar_indirect
|
||||
# CHECK-NEXT: Kind: FUNCTION
|
||||
# CHECK-NEXT: Index: 3
|
||||
# CHECK-NEXT: - Type: ELEM
|
||||
# CHECK-NEXT: Segments:
|
||||
# CHECK-NEXT: - Flags: 2
|
||||
# CHECK-NEXT: TableNumber: 3
|
||||
# CHECK-NEXT: ElemKind: FUNCREF
|
||||
# CHECK-NEXT: Offset:
|
||||
# CHECK-NEXT: Opcode: I32_CONST
|
||||
# CHECK-NEXT: Value: 1
|
||||
# CHECK-NEXT: Functions: [ 2, 0 ]
|
||||
# CHECK-NEXT: - Type: CODE
|
||||
# CHECK-NEXT: Functions:
|
||||
# CHECK-NEXT: - Index: 1
|
||||
# CHECK-NEXT: Locals: []
|
||||
# CHECK-NEXT: Body: 41001180808080008080808000410011808080800081808080000B
|
||||
# CHECK-NEXT: - Index: 2
|
||||
# CHECK-NEXT: Locals: []
|
||||
# CHECK-NEXT: Body: 42010B
|
||||
# CHECK-NEXT: - Index: 3
|
||||
# CHECK-NEXT: Locals: []
|
||||
# CHECK-NEXT: Body: 41002802808880800011818080800083808080001A41002802848880800011828080800083808080001A0B
|
||||
# CHECK-NEXT: - Type: DATA
|
||||
# CHECK-NEXT: Segments:
|
||||
# CHECK-NEXT: - SectionOffset: 7
|
||||
# CHECK-NEXT: InitFlags: 0
|
||||
# CHECK-NEXT: Offset:
|
||||
# CHECK-NEXT: Opcode: I32_CONST
|
||||
# CHECK-NEXT: Value: 1024
|
||||
# CHECK-NEXT: Content: '0100000002000000'
|
||||
# CHECK-NEXT: - Type: CUSTOM
|
||||
# CHECK-NEXT: Name: name
|
||||
# CHECK-NEXT: FunctionNames:
|
||||
# CHECK-NEXT: - Index: 0
|
||||
# CHECK-NEXT: Name: foo
|
||||
# CHECK-NEXT: - Index: 1
|
||||
# CHECK-NEXT: Name: call_indirect_explicit_tables
|
||||
# CHECK-NEXT: - Index: 2
|
||||
# CHECK-NEXT: Name: bar
|
||||
# CHECK-NEXT: - Index: 3
|
||||
# CHECK-NEXT: Name: call_bar_indirect
|
||||
# CHECK-NEXT: GlobalNames:
|
||||
# CHECK-NEXT: - Index: 0
|
||||
# CHECK-NEXT: Name: __stack_pointer
|
|
@ -230,7 +230,8 @@ void TableSection::addTable(InputTable *table) {
|
|||
if (isa<UndefinedTable>(culprit)) {
|
||||
error("object file not built with 'reference-types' feature "
|
||||
"conflicts with import of table " +
|
||||
culprit->getName() + "by file " + toString(culprit->getFile()));
|
||||
culprit->getName() + " by file " +
|
||||
toString(culprit->getFile()));
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
@ -429,8 +430,16 @@ void ElemSection::addEntry(FunctionSymbol *sym) {
|
|||
void ElemSection::writeBody() {
|
||||
raw_ostream &os = bodyOutputStream;
|
||||
|
||||
assert(WasmSym::indirectFunctionTable);
|
||||
writeUleb128(os, 1, "segment count");
|
||||
writeUleb128(os, 0, "table index");
|
||||
uint32_t tableNumber = WasmSym::indirectFunctionTable->getTableNumber();
|
||||
uint32_t flags = 0;
|
||||
if (tableNumber)
|
||||
flags |= WASM_ELEM_SEGMENT_HAS_TABLE_NUMBER;
|
||||
writeUleb128(os, flags, "elem segment flags");
|
||||
if (flags & WASM_ELEM_SEGMENT_HAS_TABLE_NUMBER)
|
||||
writeUleb128(os, tableNumber, "table number");
|
||||
|
||||
WasmInitExpr initExpr;
|
||||
if (config->isPic) {
|
||||
initExpr.Opcode = WASM_OPCODE_GLOBAL_GET;
|
||||
|
@ -440,8 +449,15 @@ void ElemSection::writeBody() {
|
|||
initExpr.Value.Int32 = config->tableBase;
|
||||
}
|
||||
writeInitExpr(os, initExpr);
|
||||
writeUleb128(os, indirectFunctions.size(), "elem count");
|
||||
|
||||
if (flags & WASM_ELEM_SEGMENT_MASK_HAS_ELEM_KIND) {
|
||||
// We only write active function table initializers, for which the elem kind
|
||||
// is specified to be written as 0x00 and interpreted to mean "funcref".
|
||||
const uint8_t elemKind = 0;
|
||||
writeU8(os, elemKind, "elem kind");
|
||||
}
|
||||
|
||||
writeUleb128(os, indirectFunctions.size(), "elem count");
|
||||
uint32_t tableIndex = config->tableBase;
|
||||
for (const FunctionSymbol *sym : indirectFunctions) {
|
||||
assert(sym->getTableIndex() == tableIndex);
|
||||
|
|
Loading…
Reference in New Issue