| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643 | {    Copyright (c) 2016-2017 by Karoly Balogh    Contains the base types for the WebAssembly    This program is free software; you can redistribute it and/or modify    it under the terms of the GNU General Public License as published by    the Free Software Foundation; either version 2 of the License, or    (at your option) any later version.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the    GNU General Public License for more details.    You should have received a copy of the GNU General Public License    along with this program; if not, write to the Free Software    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. ****************************************************************************}{ This Unit contains the base types for WebAssembly}unit cpubase;{$i fpcdefs.inc}interfaceuses  globtype,  aasmbase,cpuinfo,cgbase;{*****************************************************************************                                Assembler Opcodes*****************************************************************************}    type      TAsmOp=(A_None,      // control flow      a_block, a_loop, a_br, a_br_if, a_br_table, a_if, a_else, a_end_block,      a_end_loop, a_end_if, a_end_function, a_return, a_unreachable,      // basic      a_nop, a_drop, a_i32_const, a_i64_const, a_f32_const, a_f64_const,      a_local_get, a_local_set, a_local_tee, a_global_get, a_global_set,      a_select, a_call, a_call_indirect,      // integer      a_i32_add, a_i64_add, a_i32_sub, a_i64_sub, a_i32_mul, a_i64_mul,      a_i32_div_s, a_i64_div_s, a_i32_div_u, a_i64_div_u, a_i32_rem_s, a_i64_rem_s,      a_i32_rem_u, a_i64_rem_u, a_i32_and, a_i64_and, a_i32_or, a_i64_or,      a_i32_xor, a_i64_xor, a_i32_shl, a_i64_shl, a_i32_shr_s, a_i64_shr_s,      a_i32_shr_u, a_i64_shr_u, a_i32_rotl, a_i64_rotl, a_i32_rotr, a_i64_rotr,      a_i32_clz, a_i64_clz, a_i32_ctz, a_i64_ctz, a_i32_popcnt, a_i64_popcnt,      a_i32_eqz, a_i64_eqz,      // floating point      a_f32_add, a_f64_add, a_f32_sub, a_f64_sub, a_f32_mul, a_f64_mul,      a_f32_div, a_f64_div, a_f32_sqrt, a_f64_sqrt, a_f32_min, a_f64_min,      a_f32_max, a_f64_max, a_f32_ceil, a_f64_ceil, a_f32_floor, a_f64_floor,      a_f32_trunc, a_f64_trunc, a_f32_nearest, a_f64_nearest, a_f32_abs, a_f64_abs,      a_f32_neg, a_f64_neg, a_f32_copysign, a_f64_copysign,      // integer compare      a_i32_eq, a_i64_eq, a_i32_ne, a_i64_ne, a_i32_lt_s, a_i64_lt_s,      a_i32_lt_u, a_i64_lt_u, a_i32_le_s, a_i64_le_s, a_i32_le_u, a_i64_le_u,      a_i32_gt_s, a_i64_gt_s, a_i32_gt_u, a_i64_gt_u, a_i32_ge_s, a_i64_ge_s,      a_i32_ge_u, a_i64_ge_u,      // floating point compare      a_f32_eq, a_f64_eq, a_f32_ne, a_f64_ne, a_f32_lt, a_f64_lt,      a_f32_le, a_f64_le, a_f32_gt, a_f64_gt, a_f32_ge, a_f64_ge,      // conversion      a_i32_wrap_i64, a_i64_extend_i32_s, a_i64_extend_i32_u,      a_i32_extend8_s,a_i32_extend16_s,a_i64_extend8_s,a_i64_extend16_s,a_i64_extend32_s,      a_i32_trunc_f32_s, a_i32_trunc_f64_s, a_i64_trunc_f32_s, a_i64_trunc_f64_s,      a_i32_trunc_f32_u, a_i32_trunc_f64_u, a_i64_trunc_f32_u, a_i64_trunc_f64_u,      a_f32_demote_f64, a_f64_promote_f32,      a_f32_convert_i32_s, a_f32_convert_i64_s,a_f64_convert_i32_s,a_f64_convert_i64_s,      a_f32_convert_i32_u, a_f32_convert_i64_u,a_f64_convert_i32_u,a_f64_convert_i64_u,      a_i32_reinterpret_f32, a_i64_reinterpret_f64, a_f32_reinterpret_i32, a_f64_reinterpret_i64,      // load/store      a_i32_load, a_i64_load, a_f32_load, a_f64_load,      a_i32_store, a_i64_store, a_f32_store, a_f64_store,      a_i32_load8_s, a_i32_load16_s, a_i64_load8_s, a_i64_load16_s, a_i64_load32_s,      a_i32_load8_u, a_i32_load16_u, a_i64_load8_u, a_i64_load16_u, a_i64_load32_u,      a_i32_store8, a_i32_store16, a_i64_store8, a_i64_store16, a_i64_store32,      // additional memory      a_memory_grow, a_memory_size,      // bulk memory operations      a_memory_copy, a_memory_fill, a_memory_init, a_data_drop,      // reference instructions      a_ref_null, a_ref_is_null, a_ref_func,      // table instructions      a_table_get, a_table_set, a_table_size, a_table_grow, a_table_fill, a_table_copy, a_table_init, a_elem_drop,      // saturating truncation instructions      a_i32_trunc_sat_f32_s,      a_i32_trunc_sat_f32_u,      a_i32_trunc_sat_f64_s,      a_i32_trunc_sat_f64_u,      a_i64_trunc_sat_f32_s,      a_i64_trunc_sat_f32_u,      a_i64_trunc_sat_f64_s,      a_i64_trunc_sat_f64_u,      // exceptions      a_try,a_catch,a_catch_all,a_delegate,a_throw,a_rethrow,a_end_try,      // atomic memory accesses - load/store      a_i32_atomic_load8_u, a_i32_atomic_load16_u, a_i32_atomic_load,      a_i64_atomic_load8_u, a_i64_atomic_load16_u, a_i64_atomic_load32_u,      a_i64_atomic_load, a_i32_atomic_store8, a_i32_atomic_store16,      a_i32_atomic_store, a_i64_atomic_store8, a_i64_atomic_store16,      a_i64_atomic_store32, a_i64_atomic_store,      // atomic memory accesses - read-modify-write      a_i32_atomic_rmw8_add_u, a_i32_atomic_rmw16_add_u, a_i32_atomic_rmw_add,      a_i64_atomic_rmw8_add_u, a_i64_atomic_rmw16_add_u, a_i64_atomic_rmw32_add_u,      a_i64_atomic_rmw_add, a_i32_atomic_rmw8_sub_u, a_i32_atomic_rmw16_sub_u,      a_i32_atomic_rmw_sub, a_i64_atomic_rmw8_sub_u, a_i64_atomic_rmw16_sub_u,      a_i64_atomic_rmw32_sub_u, a_i64_atomic_rmw_sub, a_i32_atomic_rmw8_and_u,      a_i32_atomic_rmw16_and_u, a_i32_atomic_rmw_and, a_i64_atomic_rmw8_and_u,      a_i64_atomic_rmw16_and_u, a_i64_atomic_rmw32_and_u, a_i64_atomic_rmw_and,      a_i32_atomic_rmw8_or_u, a_i32_atomic_rmw16_or_u, a_i32_atomic_rmw_or,      a_i64_atomic_rmw8_or_u, a_i64_atomic_rmw16_or_u, a_i64_atomic_rmw32_or_u,      a_i64_atomic_rmw_or, a_i32_atomic_rmw8_xor_u, a_i32_atomic_rmw16_xor_u,      a_i32_atomic_rmw_xor, a_i64_atomic_rmw8_xor_u, a_i64_atomic_rmw16_xor_u,      a_i64_atomic_rmw32_xor_u, a_i64_atomic_rmw_xor, a_i32_atomic_rmw8_xchg_u,      a_i32_atomic_rmw16_xchg_u, a_i32_atomic_rmw_xchg, a_i64_atomic_rmw8_xchg_u,      a_i64_atomic_rmw16_xchg_u, a_i64_atomic_rmw32_xchg_u, a_i64_atomic_rmw_xchg,      // atomic memory accesses - compare exchange      a_i32_atomic_rmw8_cmpxchg_u, a_i32_atomic_rmw16_cmpxchg_u, a_i32_atomic_rmw_cmpxchg,      a_i64_atomic_rmw8_cmpxchg_u, a_i64_atomic_rmw16_cmpxchg_u, a_i64_atomic_rmw32_cmpxchg_u,      a_i64_atomic_rmw_cmpxchg,      // atomic memory accesses - wait and notify operators      a_memory_atomic_wait32, a_memory_atomic_wait64, a_memory_atomic_notify, a_atomic_fence      );      TWasmBasicType = (        { number types }        wbt_i32, wbt_i64, wbt_f32, wbt_f64,        { reference types }        wbt_funcref, wbt_externref,        { vector types }        wbt_v128      );      TWasmResultType = array of TWasmBasicType;      { TWasmFuncType }      PWasmFuncType = ^TWasmFuncType;      TWasmFuncType = class        params: TWasmResultType;        results: TWasmResultType;        constructor Create(aparams, aresults: TWasmResultType);        constructor Create(afunctype: TWasmFuncType);        procedure add_param(param: TWasmBasicType);        procedure add_result(res: TWasmBasicType);        function Equals(Obj: TObject): boolean; override;      end;      {# This should define the array of instructions as string }      op2strtable=array[tasmop] of string[31];    Const      {# First value of opcode enumeration }      firstop = low(tasmop);      {# Last value of opcode enumeration  }      lastop  = high(tasmop);      AsmOp_Store = [         a_i32_store, a_i32_store16, a_i32_store8        ,a_i64_store, a_i64_store16, a_i64_store8, a_i64_store32        ,a_f32_store, a_f64_store      ];      AsmOp_Load = [         a_i32_load,         a_i32_load8_s,  a_i32_load8_u,         a_i32_load16_s, a_i32_load16_u,         a_i64_load,         a_i64_load8_s,  a_i64_load8_u,         a_i64_load16_s, a_i64_load16_u,         a_i64_load32_s, a_i64_load32_u,         a_f32_load, a_f64_load      ];      AsmOp_LoadStore = AsmOp_Load + AsmOp_Store;{*****************************************************************************                                  Registers*****************************************************************************}    type      { Number of registers used for indexing in tables }      tregisterindex=0..{$i rwasmnor.inc}-1; // no registers in wasm      totherregisterset = set of tregisterindex;    const      { Available Superregisters }      // there's no registers in wasm      {$i rwasmsup.inc}      { No Subregisters }      R_SUBWHOLE = R_SUBNONE;      { Available Registers }      // there's no registers in wasm      {$i rwasmcon.inc}      { aliases }      { used as base register in references for parameters passed to        subroutines: these are passed on the evaluation stack, but this way we        can use the offset field to indicate the order, which is used by ncal        to sort the parameters }      NR_EVAL_STACK_BASE = NR_R0;      RS_EVAL_STACK_BASE = RS_R0;      { used as base register in references to indicate that it's a local }      NR_LOCAL_STACK_POINTER_REG = NR_R1;      RS_LOCAL_STACK_POINTER_REG = RS_R1;      { fake register, representing the local frame pointer. Used for accessing        address-taken local variables on the linear stack: (localframeptr+offset). }      NR_LOCAL_FRAME_POINTER_REG = NR_R3;      RS_LOCAL_FRAME_POINTER_REG = RS_R3;      maxvarregs = 1;      maxfpuvarregs = 1;      { Integer Super registers first and last }      first_int_imreg = 4;      { Float Super register first and last }      first_fpu_imreg     = 4;      { MM Super register first and last }      first_mm_imreg     = 4;      regnumber_table : array[tregisterindex] of tregister = (        {$i rwasmnum.inc}      );     EVALSTACKLOCS = [LOC_REGISTER,LOC_CREGISTER,LOC_FPUREGISTER,LOC_CFPUREGISTER,       LOC_MMREGISTER,LOC_CMMREGISTER,LOC_SUBSETREG,LOC_CSUBSETREG];{*****************************************************************************                                Conditions*****************************************************************************}   type     // not used by wasm target     TAsmCond=(C_None);{*****************************************************************************                                 Constants*****************************************************************************}    const      max_operands = 2;{*****************************************************************************                          Default generic sizes*****************************************************************************}{$ifdef cpu64bitaddr}      {# Defines the default address size for a processor,        -- fake for JVM, only influences default width of           arithmetic calculations }      OS_ADDR = OS_64;      {# the natural int size for a processor,         has to match osuinttype/ossinttype as initialized in psystem }      OS_INT = OS_64;      OS_SINT = OS_S64;{$else}      {# Defines the default address size for a processor,        -- fake for wasm, only influences default width of           arithmetic calculations }      OS_ADDR = OS_32;      {# the natural int size for a processor,         has to match osuinttype/ossinttype as initialized in psystem }      OS_INT = OS_32;      OS_SINT = OS_S32;{$endif}      {# the maximum float size for a processor,           }      OS_FLOAT = OS_F64;      {# the size of a vector register for a processor     }      OS_VECTOR = OS_M128;{*****************************************************************************                          Generic Register names*****************************************************************************}      { dummies, not used for Wasm }      {# Stack pointer register }      { used as base register in references to indicate that it's a local }      NR_STACK_POINTER_REG = NR_R1;      RS_STACK_POINTER_REG = RS_R1;      {# Frame pointer register }      NR_FRAME_POINTER_REG = NR_LOCAL_FRAME_POINTER_REG;      RS_FRAME_POINTER_REG = RS_LOCAL_FRAME_POINTER_REG;      { WebAssembly results are returned on the evaluation stack, not via a register }      { Results are returned in this register (32-bit values) }      NR_FUNCTION_RETURN_REG = NR_NO;      RS_FUNCTION_RETURN_REG = RS_NO;      { Low part of 64bit return value }      NR_FUNCTION_RETURN64_LOW_REG = NR_NO;      RS_FUNCTION_RETURN64_LOW_REG = RS_NO;      { High part of 64bit return value }      NR_FUNCTION_RETURN64_HIGH_REG = NR_NO;      RS_FUNCTION_RETURN64_HIGH_REG = RS_NO;      { The value returned from a function is available in this register }      NR_FUNCTION_RESULT_REG = NR_FUNCTION_RETURN_REG;      RS_FUNCTION_RESULT_REG = RS_FUNCTION_RETURN_REG;      { The lowh part of 64bit value returned from a function }      NR_FUNCTION_RESULT64_LOW_REG = NR_FUNCTION_RETURN64_LOW_REG;      RS_FUNCTION_RESULT64_LOW_REG = RS_FUNCTION_RETURN64_LOW_REG;      { The high part of 64bit value returned from a function }      NR_FUNCTION_RESULT64_HIGH_REG = NR_FUNCTION_RETURN64_HIGH_REG;      RS_FUNCTION_RESULT64_HIGH_REG = RS_FUNCTION_RETURN64_HIGH_REG;      NR_FPU_RESULT_REG = NR_NO;      NR_MM_RESULT_REG = NR_NO;      { No default flags }      NR_DEFAULTFLAGS = NR_NO;      RS_DEFAULTFLAGS = RS_NO;{*****************************************************************************                       GCC /ABI linking information*****************************************************************************}      { dummies, not used for Wasm }      {# Required parameter alignment when calling a routine      }      std_param_align = 1;{*****************************************************************************                            CPU Dependent Constants*****************************************************************************}      maxfpuregs = 0;      { Global variable, that acts as the stack pointer in linear memory        (also called the "linear stack"). This stack is used for address-taken        local variables. This separate stack is needed, because the WASM        implementation's runtime call stack (which includes return addresses and        function parameters) is not visible in linear memory. }      STACK_POINTER_SYM = '__stack_pointer';      { The exception tag symbol, used for FPC exceptions }      FPC_EXCEPTION_TAG_SYM = '__FPC_exception';      { Immutable global variable, created by the linker, when multithreading        support is enabled. Contains the total size (plus padding) of all the        threadvars, used by the program. }      TLS_SIZE_SYM = '__tls_size';      { Immutable global variable, created by the linker, when multithreading        support is enabled. Contains the alignment requirement for the thread        local block. }      TLS_ALIGN_SYM = '__tls_align';      { Mutable global variable, created by the linker, when multithreading        support is enabled. }      TLS_BASE_SYM = '__tls_base';{*****************************************************************************                                  Helpers*****************************************************************************}    function cgsize2subreg(regtype: tregistertype; s:Tcgsize):Tsubregister;    function reg_cgsize(const reg: tregister) : tcgsize;    function std_regnum_search(const s:string):Tregister;    function std_regname(r:Tregister):string;    function findreg_by_number(r:Tregister):tregisterindex;    function dwarf_reg(r:tregister):byte;    function dwarf_reg_no_error(r:tregister):shortint;    function eh_return_data_regno(nr: longint): longint;    { since we don't use tasmconds, don't call this routine      (it will internalerror). We need it anyway to get aoptobj      to compile (but it won't execute it).    }    function inverse_cond(const c: TAsmCond): Tasmcond; {$ifdef USEINLINE}inline;{$endif USEINLINE}    function natural_alignment_for_load_store(op: TAsmOp): shortint;    function encode_wasm_basic_type(wbt: TWasmBasicType): Byte;implementationuses  verbose,  rgbase;{*****************************************************************************                                  Helpers*****************************************************************************}    const      std_regname_table : array[tregisterindex] of string[15] = (        {$i rwasmstd.inc}      );      regnumber_index : array[tregisterindex] of tregisterindex = (        {$i rwasmrni.inc}      );      std_regname_index : array[tregisterindex] of tregisterindex = (        {$i rwasmsri.inc}      );    function reg_cgsize(const reg: tregister): tcgsize;      begin        result:=OS_NO;      end;    function cgsize2subreg(regtype: tregistertype; s:Tcgsize):Tsubregister;      begin        cgsize2subreg:=R_SUBNONE;      end;    function std_regnum_search(const s:string):Tregister;      begin        result:=NR_NO;      end;    function findreg_by_number(r:Tregister):tregisterindex;      begin        result:=findreg_by_number_table(r,regnumber_index);      end;    function std_regname(r:Tregister):string;      var        p : tregisterindex;      begin        p:=findreg_by_number_table(r,regnumber_index);        if p<>0 then          result:=std_regname_table[p]        else          result:=generic_regname(r);      end;    function dwarf_reg(r:tregister):byte;      begin        result:=0;        internalerror(200603251);      end;    function dwarf_reg_no_error(r:tregister):shortint;      begin        result:=-1;      end;    function eh_return_data_regno(nr: longint): longint;      begin        result:=-1;      end;    function inverse_cond(const c: TAsmCond): Tasmcond; {$ifdef USEINLINE}inline;{$endif USEINLINE}      begin        result:=C_None;        internalerror(2015082701);      end;    function natural_alignment_for_load_store(op: TAsmOp): shortint;      begin        case op of          a_i32_load8_s,          a_i32_load8_u,          a_i64_load8_s,          a_i64_load8_u,          a_i32_store8,          a_i64_store8,          a_i32_atomic_load8_u,          a_i64_atomic_load8_u,          a_i32_atomic_store8,          a_i64_atomic_store8,          a_i32_atomic_rmw8_add_u,          a_i64_atomic_rmw8_add_u,          a_i32_atomic_rmw8_sub_u,          a_i64_atomic_rmw8_sub_u,          a_i32_atomic_rmw8_and_u,          a_i64_atomic_rmw8_and_u,          a_i32_atomic_rmw8_or_u,          a_i64_atomic_rmw8_or_u,          a_i32_atomic_rmw8_xor_u,          a_i64_atomic_rmw8_xor_u,          a_i32_atomic_rmw8_xchg_u,          a_i64_atomic_rmw8_xchg_u,          a_i32_atomic_rmw8_cmpxchg_u,          a_i64_atomic_rmw8_cmpxchg_u:            result:=0;          a_i32_load16_s,          a_i32_load16_u,          a_i64_load16_s,          a_i64_load16_u,          a_i32_store16,          a_i64_store16,          a_i32_atomic_load16_u,          a_i64_atomic_load16_u,          a_i32_atomic_store16,          a_i64_atomic_store16,          a_i32_atomic_rmw16_add_u,          a_i64_atomic_rmw16_add_u,          a_i32_atomic_rmw16_sub_u,          a_i64_atomic_rmw16_sub_u,          a_i32_atomic_rmw16_and_u,          a_i64_atomic_rmw16_and_u,          a_i32_atomic_rmw16_or_u,          a_i64_atomic_rmw16_or_u,          a_i32_atomic_rmw16_xor_u,          a_i64_atomic_rmw16_xor_u,          a_i32_atomic_rmw16_xchg_u,          a_i64_atomic_rmw16_xchg_u,          a_i32_atomic_rmw16_cmpxchg_u,          a_i64_atomic_rmw16_cmpxchg_u:            result:=1;          a_i32_load,          a_f32_load,          a_i64_load32_s,          a_i64_load32_u,          a_i32_store,          a_f32_store,          a_i64_store32,          a_memory_atomic_notify,          a_memory_atomic_wait32,          a_i32_atomic_load,          a_i64_atomic_load32_u,          a_i32_atomic_store,          a_i64_atomic_store32,          a_i32_atomic_rmw_add,          a_i64_atomic_rmw32_add_u,          a_i32_atomic_rmw_sub,          a_i64_atomic_rmw32_sub_u,          a_i32_atomic_rmw_and,          a_i64_atomic_rmw32_and_u,          a_i32_atomic_rmw_or,          a_i64_atomic_rmw32_or_u,          a_i32_atomic_rmw_xor,          a_i64_atomic_rmw32_xor_u,          a_i32_atomic_rmw_xchg,          a_i64_atomic_rmw32_xchg_u,          a_i32_atomic_rmw_cmpxchg,          a_i64_atomic_rmw32_cmpxchg_u:            result:=2;          a_i64_load,          a_f64_load,          a_i64_store,          a_f64_store,          a_memory_atomic_wait64,          a_i64_atomic_load,          a_i64_atomic_store,          a_i64_atomic_rmw_add,          a_i64_atomic_rmw_sub,          a_i64_atomic_rmw_and,          a_i64_atomic_rmw_or,          a_i64_atomic_rmw_xor,          a_i64_atomic_rmw_xchg,          a_i64_atomic_rmw_cmpxchg:            result:=3;          else            internalerror(2021092614);        end;      end;    function encode_wasm_basic_type(wbt: TWasmBasicType): Byte;      begin        case wbt of          wbt_i32:            result:=$7F;          wbt_i64:            result:=$7E;          wbt_f32:            result:=$7D;          wbt_f64:            result:=$7C;          wbt_funcref:            result:=$70;          wbt_externref:            result:=$6F;          wbt_v128:            result:=$7B;        end;      end;{*****************************************************************************                                  TWasmFuncType*****************************************************************************}    constructor TWasmFuncType.Create(aparams, aresults: TWasmResultType);      begin        inherited Create;        params:=aparams;        results:=aresults;      end;    constructor TWasmFuncType.Create(afunctype: TWasmFuncType);      begin        inherited Create;        params:=afunctype.params;        results:=afunctype.results;      end;    procedure TWasmFuncType.add_param(param: TWasmBasicType);      begin        SetLength(params,Length(params)+1);        params[High(params)]:=param;      end;    procedure TWasmFuncType.add_result(res: TWasmBasicType);      begin        SetLength(results,Length(results)+1);        results[High(results)]:=res;      end;    function TWasmFuncType.Equals(Obj: TObject): boolean;      var        O: TWasmFuncType;      begin        if Obj=Self then          exit(true)        else if (Obj<>nil) and (Obj is TWasmFuncType) then          begin            O:=TWasmFuncType(Obj);            if (Length(params)<>Length(O.params)) or (Length(results)<>Length(O.results)) then              exit(false);            if (Length(params)>0) and (CompareByte(params[0],O.params[0],Length(params)*SizeOf(params[0]))<>0) then              exit(false);            if (Length(results)>0) and (CompareByte(results[0],O.results[0],Length(results)*SizeOf(results[0]))<>0) then              exit(false);            Result:=true;          end        else          Result:=inherited Equals(Obj);      end;end.
 |