Wasmtime: C++
All Classes Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
wasmtime.hh
Go to the documentation of this file.
1
32#ifndef WASMTIME_HH
33#define WASMTIME_HH
34
35#include <any>
36#include <array>
37#include <cstdio>
38#include <initializer_list>
39#include <iosfwd>
40#include <limits>
41#include <memory>
42#include <optional>
43#include <ostream>
44#include <variant>
45#include <vector>
46#ifdef __has_include
47#if __has_include(<span>)
48#include <span>
49#endif
50#endif
51
52#include "wasmtime.h"
53
54namespace wasmtime {
55
56#ifdef __cpp_lib_span
57
59template <typename T, std::size_t Extent = std::dynamic_extent>
60using Span = std::span<T, Extent>;
61
62#else
63
65inline constexpr size_t dynamic_extent =
66 std::numeric_limits<std::size_t>::max();
67
73template <typename T, std::size_t Extent = dynamic_extent> class Span;
74
76template <typename T> struct IsSpan : std::false_type {};
77
78template <typename T, std::size_t Extent>
79struct IsSpan<Span<T, Extent>> : std::true_type {};
80
81template <typename T, std::size_t Extent> class Span {
82 static_assert(Extent == dynamic_extent,
83 "The current implementation supports dynamic-extent span only");
84
85public:
87 using iterator = T *;
88
90 Span(T *t, std::size_t n) : ptr_{t}, size_{n} {}
91
93 template <typename C,
94 std::enable_if_t<
96 std::is_pointer_v<decltype(std::declval<C &>().data())> &&
97 std::is_convertible_v<
98 std::remove_pointer_t<
99 decltype(std::declval<C &>().data())> (*)[],
100 T (*)[]> &&
101 std::is_convertible_v<decltype(std::declval<C>().size()),
102 std::size_t>,
103 int> = 0>
104 Span(C &range) : ptr_{range.data()}, size_{range.size()} {}
105
107 T &operator[](ptrdiff_t idx) const {
108 return ptr_[idx]; // NOLINT
109 }
110
112 T *data() const { return ptr_; }
113
115 std::size_t size() const { return size_; }
116
118 iterator begin() const { return ptr_; }
119
121 iterator end() const {
122 return ptr_ + size_; // NOLINT
123 }
124
126 std::size_t size_bytes() const { return sizeof(T) * size_; }
127
128private:
129 T *ptr_;
130 std::size_t size_;
131};
132
133#endif
134
135class Trace;
136
143class Error {
144 struct deleter {
145 void operator()(wasmtime_error_t *p) const { wasmtime_error_delete(p); }
146 };
147
148 std::unique_ptr<wasmtime_error_t, deleter> ptr;
149
150public:
154 Error(wasmtime_error_t *error) : ptr(error) {}
155
157 std::string message() const {
158 wasm_byte_vec_t msg_bytes;
159 wasmtime_error_message(ptr.get(), &msg_bytes);
160 auto ret = std::string(msg_bytes.data, msg_bytes.size);
161 wasm_byte_vec_delete(&msg_bytes);
162 return ret;
163 }
164
167 std::optional<int32_t> i32_exit() const {
168 int32_t status = 0;
169 if (wasmtime_error_exit_status(ptr.get(), &status)) {
170 return status;
171 }
172 return std::nullopt;
173 }
174
178 Trace trace() const;
179};
180
182inline std::ostream &operator<<(std::ostream &os, const Error &e) {
183 os << e.message();
184 return os;
185}
186
194template <typename T, typename E = Error> class [[nodiscard]] Result {
195 std::variant<T, E> data;
196
197public:
199 Result(T t) : data(std::move(t)) {}
201 Result(E e) : data(std::move(e)) {}
202
205 explicit operator bool() const { return data.index() == 0; }
206
208 E &&err() { return std::get<E>(std::move(data)); }
210 const E &&err() const { return std::get<E>(std::move(data)); }
211
213 T &&ok() { return std::get<T>(std::move(data)); }
215 const T &&ok() const { return std::get<T>(std::move(data)); }
216
218 T unwrap() {
219 if (*this) {
220 return this->ok();
221 }
222 unwrap_failed();
223 }
224
225private:
226 [[noreturn]] void unwrap_failed() {
227 fprintf(stderr, "error: %s\n", this->err().message().c_str()); // NOLINT
228 std::abort();
229 }
230};
231
233enum class Strategy {
235 Auto = WASMTIME_STRATEGY_AUTO,
237 Cranelift = WASMTIME_STRATEGY_CRANELIFT,
238};
239
241enum class OptLevel {
243 None = WASMTIME_OPT_LEVEL_NONE,
245 Speed = WASMTIME_OPT_LEVEL_SPEED,
247 SpeedAndSize = WASMTIME_OPT_LEVEL_SPEED_AND_SIZE,
248};
249
253 None = WASMTIME_PROFILING_STRATEGY_NONE,
255 Jitdump = WASMTIME_PROFILING_STRATEGY_JITDUMP,
257 Vtune = WASMTIME_PROFILING_STRATEGY_VTUNE,
258};
259
267 friend class Config;
268
269 struct deleter {
270 void operator()(wasmtime_pooling_allocation_config_t *p) const {
271 wasmtime_pooling_allocation_config_delete(p);
272 }
273 };
274
275 std::unique_ptr<wasmtime_pooling_allocation_config_t, deleter> ptr;
276
277public:
278 PoolAllocationConfig() : ptr(wasmtime_pooling_allocation_config_new()) {}
279
284 void max_unused_warm_slots(uint32_t max) {
285 wasmtime_pooling_allocation_config_max_unused_warm_slots_set(ptr.get(),
286 max);
287 }
288
292 void decommit_batch_size(size_t batch_size) {
293 wasmtime_pooling_allocation_config_decommit_batch_size_set(ptr.get(),
294 batch_size);
295 }
296
301 void async_stack_keep_resident(size_t size) {
302 wasmtime_pooling_allocation_config_async_stack_keep_resident_set(ptr.get(),
303 size);
304 }
305
310 void linear_memory_keep_resident(size_t size) {
311 wasmtime_pooling_allocation_config_linear_memory_keep_resident_set(
312 ptr.get(), size);
313 }
314
319 void table_keep_resident(size_t size) {
320 wasmtime_pooling_allocation_config_table_keep_resident_set(ptr.get(), size);
321 }
322
327 void total_component_instances(uint32_t count) {
328 wasmtime_pooling_allocation_config_total_component_instances_set(ptr.get(),
329 count);
330 }
331
336 void max_component_instance_size(size_t size) {
337 wasmtime_pooling_allocation_config_max_component_instance_size_set(
338 ptr.get(), size);
339 }
340
345 void max_core_instances_per_component(uint32_t count) {
346 wasmtime_pooling_allocation_config_max_core_instances_per_component_set(
347 ptr.get(), count);
348 }
349
354 void max_memories_per_component(uint32_t count) {
355 wasmtime_pooling_allocation_config_max_memories_per_component_set(ptr.get(),
356 count);
357 }
358
363 void max_tables_per_component(uint32_t count) {
364 wasmtime_pooling_allocation_config_max_tables_per_component_set(ptr.get(),
365 count);
366 }
367
372 void total_memories(uint32_t count) {
373 wasmtime_pooling_allocation_config_total_memories_set(ptr.get(), count);
374 }
375
380 void total_tables(uint32_t count) {
381 wasmtime_pooling_allocation_config_total_tables_set(ptr.get(), count);
382 }
383
388 void total_stacks(uint32_t count) {
389 wasmtime_pooling_allocation_config_total_stacks_set(ptr.get(), count);
390 }
391
396 void total_core_instances(uint32_t count) {
397 wasmtime_pooling_allocation_config_total_core_instances_set(ptr.get(),
398 count);
399 }
400
405 void max_core_instance_size(size_t size) {
406 wasmtime_pooling_allocation_config_max_core_instance_size_set(ptr.get(),
407 size);
408 }
409
414 void max_tables_per_module(uint32_t tables) {
415 wasmtime_pooling_allocation_config_max_tables_per_module_set(ptr.get(),
416 tables);
417 }
418
423 void table_elements(size_t elements) {
424 wasmtime_pooling_allocation_config_table_elements_set(ptr.get(), elements);
425 }
426
431 void max_memories_per_module(uint32_t memories) {
432 wasmtime_pooling_allocation_config_max_memories_per_module_set(ptr.get(),
433 memories);
434 }
435
440 void max_memory_size(size_t bytes) {
441 wasmtime_pooling_allocation_config_max_memory_size_set(ptr.get(), bytes);
442 }
443
448 void total_gc_heaps(uint32_t count) {
449 wasmtime_pooling_allocation_config_total_gc_heaps_set(ptr.get(), count);
450 }
451};
452
462class Config {
463 friend class Engine;
464
465 struct deleter {
466 void operator()(wasm_config_t *p) const { wasm_config_delete(p); }
467 };
468
469 std::unique_ptr<wasm_config_t, deleter> ptr;
470
471public:
473 Config() : ptr(wasm_config_new()) {}
474
479 void debug_info(bool enable) {
480 wasmtime_config_debug_info_set(ptr.get(), enable);
481 }
482
487 void epoch_interruption(bool enable) {
488 wasmtime_config_epoch_interruption_set(ptr.get(), enable);
489 }
490
495 void consume_fuel(bool enable) {
496 wasmtime_config_consume_fuel_set(ptr.get(), enable);
497 }
498
502 void max_wasm_stack(size_t stack) {
503 wasmtime_config_max_wasm_stack_set(ptr.get(), stack);
504 }
505
509 void wasm_threads(bool enable) {
510 wasmtime_config_wasm_threads_set(ptr.get(), enable);
511 }
512
517 void wasm_reference_types(bool enable) {
518 wasmtime_config_wasm_reference_types_set(ptr.get(), enable);
519 }
520
524 void wasm_simd(bool enable) {
525 wasmtime_config_wasm_simd_set(ptr.get(), enable);
526 }
527
531 void wasm_bulk_memory(bool enable) {
532 wasmtime_config_wasm_bulk_memory_set(ptr.get(), enable);
533 }
534
538 void wasm_multi_value(bool enable) {
539 wasmtime_config_wasm_multi_value_set(ptr.get(), enable);
540 }
541
545 void wasm_gc(bool enable) {
546 wasmtime_config_wasm_gc_set(ptr.get(), enable);
547 }
548
552 void wasm_function_references(bool enable) {
553 wasmtime_config_wasm_function_references_set(ptr.get(), enable);
554 }
555
559 void parallel_compilation(bool enable) {
560 wasmtime_config_parallel_compilation_set(ptr.get(), enable);
561 }
562
566 void strategy(Strategy strategy) {
567 wasmtime_config_strategy_set(ptr.get(),
568 static_cast<wasmtime_strategy_t>(strategy));
569 }
570
574 void cranelift_debug_verifier(bool enable) {
575 wasmtime_config_cranelift_debug_verifier_set(ptr.get(), enable);
576 }
577
582 wasmtime_config_cranelift_opt_level_set(
583 ptr.get(), static_cast<wasmtime_opt_level_t>(level));
584 }
585
590 wasmtime_config_profiler_set(
591 ptr.get(), static_cast<wasmtime_profiling_strategy_t>(profiler));
592 }
593
597 void memory_reservation(size_t size) {
598 wasmtime_config_memory_reservation_set(ptr.get(), size);
599 }
600
604 void memory_guard_size(size_t size) {
605 wasmtime_config_memory_guard_size_set(ptr.get(), size);
606 }
607
612 auto *error = wasmtime_config_cache_config_load(ptr.get(), nullptr);
613 if (error != nullptr) {
614 return Error(error);
615 }
616 return std::monostate();
617 }
618
622 Result<std::monostate> cache_load(const std::string &path) {
623 auto *error = wasmtime_config_cache_config_load(ptr.get(), path.c_str());
624 if (error != nullptr) {
625 return Error(error);
626 }
627 return std::monostate();
628 }
629
634 wasmtime_pooling_allocation_strategy_set(ptr.get(), config.ptr.get());
635 }
636};
637
645class Engine {
646 friend class Store;
647 friend class Module;
648 friend class Linker;
649
650 struct deleter {
651 void operator()(wasm_engine_t *p) const { wasm_engine_delete(p); }
652 };
653
654 std::unique_ptr<wasm_engine_t, deleter> ptr;
655
656public:
658 Engine() : ptr(wasm_engine_new()) {}
660 explicit Engine(Config config)
661 : ptr(wasm_engine_new_with_config(config.ptr.release())) {}
662
666 void increment_epoch() const { wasmtime_engine_increment_epoch(ptr.get()); }
667};
668
680inline Result<std::vector<uint8_t>> wat2wasm(std::string_view wat) {
681 wasm_byte_vec_t ret;
682 auto *error = wasmtime_wat2wasm(wat.data(), wat.size(), &ret);
683 if (error != nullptr) {
684 return Error(error);
685 }
686 std::vector<uint8_t> vec;
687 // NOLINTNEXTLINE TODO can this be done without triggering lints?
688 Span<uint8_t> raw(reinterpret_cast<uint8_t *>(ret.data), ret.size);
689 vec.assign(raw.begin(), raw.end());
690 wasm_byte_vec_delete(&ret);
691 return vec;
692}
693
695enum class ValKind {
697 I32,
699 I64,
701 F32,
703 F64,
705 V128,
707 ExternRef,
709 FuncRef,
710};
711
714#define WASMTIME_FOR_EACH_VAL_KIND(X) \
715 X(I32, "i32", WASM_I32) \
716 X(I64, "i64", WASM_I64) \
717 X(F32, "f32", WASM_F32) \
718 X(F64, "f64", WASM_F64) \
719 X(ExternRef, "externref", WASM_EXTERNREF) \
720 X(FuncRef, "funcref", WASM_FUNCREF) \
721 X(V128, "v128", WASMTIME_V128)
722
724inline std::ostream &operator<<(std::ostream &os, const ValKind &e) {
725 switch (e) {
726#define CASE_KIND_PRINT_NAME(kind, name, ignore) \
727 case ValKind::kind: \
728 os << name; \
729 break;
730 WASMTIME_FOR_EACH_VAL_KIND(CASE_KIND_PRINT_NAME)
731#undef CASE_KIND_PRINT_NAME
732 default:
733 abort();
734 }
735 return os;
736}
737
743class ValType {
744 friend class TableType;
745 friend class GlobalType;
746 friend class FuncType;
747
748 struct deleter {
749 void operator()(wasm_valtype_t *p) const { wasm_valtype_delete(p); }
750 };
751
752 std::unique_ptr<wasm_valtype_t, deleter> ptr;
753
754 static wasm_valkind_t kind_to_c(ValKind kind) {
755 switch (kind) {
756#define CASE_KIND_TO_C(kind, ignore, ckind) \
757 case ValKind::kind: \
758 return ckind;
759 WASMTIME_FOR_EACH_VAL_KIND(CASE_KIND_TO_C)
760#undef CASE_KIND_TO_C
761 default:
762 abort();
763 }
764 }
765
766public:
769 class Ref {
770 friend class ValType;
771
772 const wasm_valtype_t *ptr;
773
774 public:
776 Ref(const wasm_valtype_t *ptr) : ptr(ptr) {}
778 Ref(const ValType &ty) : Ref(ty.ptr.get()) {}
779
781 ValKind kind() const {
782 switch (wasm_valtype_kind(ptr)) {
783#define CASE_C_TO_KIND(kind, ignore, ckind) \
784 case ckind: \
785 return ValKind::kind;
786 WASMTIME_FOR_EACH_VAL_KIND(CASE_C_TO_KIND)
787#undef CASE_C_TO_KIND
788 }
789 std::abort();
790 }
791 };
792
795 class ListRef {
796 const wasm_valtype_vec_t *list;
797
798 public:
800 ListRef(const wasm_valtype_vec_t *list) : list(list) {}
801
803 typedef const Ref *iterator;
804
806 iterator begin() const {
807 return reinterpret_cast<Ref *>(&list->data[0]); // NOLINT
808 }
809
811 iterator end() const {
812 return reinterpret_cast<Ref *>(&list->data[list->size]); // NOLINT
813 }
814
816 size_t size() const { return list->size; }
817 };
818
819private:
820 Ref ref;
821 ValType(wasm_valtype_t *ptr) : ptr(ptr), ref(ptr) {}
822
823public:
825 ValType(ValKind kind) : ValType(wasm_valtype_new(kind_to_c(kind))) {}
827 ValType(Ref other) : ValType(wasm_valtype_copy(other.ptr)) {}
829 ValType(const ValType &other) : ValType(wasm_valtype_copy(other.ptr.get())) {}
831 ValType &operator=(const ValType &other) {
832 ptr.reset(wasm_valtype_copy(other.ptr.get()));
833 ref = other.ref;
834 return *this;
835 }
836 ~ValType() = default;
838 ValType(ValType &&other) = default;
840 ValType &operator=(ValType &&other) = default;
841
844 Ref *operator->() { return &ref; }
847 Ref *operator*() { return &ref; }
848};
849
854 friend class Memory;
855
856 struct deleter {
857 void operator()(wasm_memorytype_t *p) const { wasm_memorytype_delete(p); }
858 };
859
860 std::unique_ptr<wasm_memorytype_t, deleter> ptr;
861
862public:
865 class Ref {
866 friend class MemoryType;
867
868 const wasm_memorytype_t *ptr;
869
870 public:
872 Ref(const wasm_memorytype_t *ptr) : ptr(ptr) {}
874 Ref(const MemoryType &ty) : Ref(ty.ptr.get()) {}
875
877 uint64_t min() const { return wasmtime_memorytype_minimum(ptr); }
878
881 std::optional<uint64_t> max() const {
882 uint64_t max = 0;
883 auto present = wasmtime_memorytype_maximum(ptr, &max);
884 if (present) {
885 return max;
886 }
887 return std::nullopt;
888 }
889
891 bool is_64() const { return wasmtime_memorytype_is64(ptr); }
892
894 bool is_shared() const { return wasmtime_memorytype_isshared(ptr); }
895 };
896
897private:
898 Ref ref;
899 MemoryType(wasm_memorytype_t *ptr) : ptr(ptr), ref(ptr) {}
900
901public:
905 explicit MemoryType(uint32_t min)
906 : MemoryType(wasmtime_memorytype_new(min, false, 0, false, false)) {}
909 MemoryType(uint32_t min, uint32_t max)
910 : MemoryType(wasmtime_memorytype_new(min, true, max, false, false)) {}
911
913 static MemoryType New64(uint64_t min) {
914 return MemoryType(wasmtime_memorytype_new(min, false, 0, true, false));
915 }
916
918 static MemoryType New64(uint64_t min, uint64_t max) {
919 return MemoryType(wasmtime_memorytype_new(min, true, max, true, false));
920 }
921
924 MemoryType(Ref other) : MemoryType(wasm_memorytype_copy(other.ptr)) {}
926 MemoryType(const MemoryType &other)
927 : MemoryType(wasm_memorytype_copy(other.ptr.get())) {}
930 ptr.reset(wasm_memorytype_copy(other.ptr.get()));
931 return *this;
932 }
933 ~MemoryType() = default;
935 MemoryType(MemoryType &&other) = default;
937 MemoryType &operator=(MemoryType &&other) = default;
938
941 Ref *operator->() { return &ref; }
944 Ref *operator*() { return &ref; }
945};
946
951 friend class Table;
952
953 struct deleter {
954 void operator()(wasm_tabletype_t *p) const { wasm_tabletype_delete(p); }
955 };
956
957 std::unique_ptr<wasm_tabletype_t, deleter> ptr;
958
959public:
962 class Ref {
963 friend class TableType;
964
965 const wasm_tabletype_t *ptr;
966
967 public:
969 Ref(const wasm_tabletype_t *ptr) : ptr(ptr) {}
971 Ref(const TableType &ty) : Ref(ty.ptr.get()) {}
972
974 uint32_t min() const { return wasm_tabletype_limits(ptr)->min; }
975
977 std::optional<uint32_t> max() const {
978 const auto *limits = wasm_tabletype_limits(ptr);
979 if (limits->max == wasm_limits_max_default) {
980 return std::nullopt;
981 }
982 return limits->max;
983 }
984
986 ValType::Ref element() const { return wasm_tabletype_element(ptr); }
987 };
988
989private:
990 Ref ref;
991 TableType(wasm_tabletype_t *ptr) : ptr(ptr), ref(ptr) {}
992
993public:
996 TableType(ValType ty, uint32_t min) : ptr(nullptr), ref(nullptr) {
997 wasm_limits_t limits;
998 limits.min = min;
999 limits.max = wasm_limits_max_default;
1000 ptr.reset(wasm_tabletype_new(ty.ptr.release(), &limits));
1001 ref = ptr.get();
1002 }
1005 TableType(ValType ty, uint32_t min, uint32_t max) // NOLINT
1006 : ptr(nullptr), ref(nullptr) {
1007 wasm_limits_t limits;
1008 limits.min = min;
1009 limits.max = max;
1010 ptr.reset(wasm_tabletype_new(ty.ptr.release(), &limits));
1011 ref = ptr.get();
1012 }
1014 TableType(Ref other) : TableType(wasm_tabletype_copy(other.ptr)) {}
1016 TableType(const TableType &other)
1017 : TableType(wasm_tabletype_copy(other.ptr.get())) {}
1020 ptr.reset(wasm_tabletype_copy(other.ptr.get()));
1021 return *this;
1022 }
1023 ~TableType() = default;
1025 TableType(TableType &&other) = default;
1027 TableType &operator=(TableType &&other) = default;
1028
1031 Ref *operator->() { return &ref; }
1034 Ref *operator*() { return &ref; }
1035};
1036
1041 friend class Global;
1042
1043 struct deleter {
1044 void operator()(wasm_globaltype_t *p) const { wasm_globaltype_delete(p); }
1045 };
1046
1047 std::unique_ptr<wasm_globaltype_t, deleter> ptr;
1048
1049public:
1052 class Ref {
1053 friend class GlobalType;
1054 const wasm_globaltype_t *ptr;
1055
1056 public:
1058 Ref(const wasm_globaltype_t *ptr) : ptr(ptr) {}
1060 Ref(const GlobalType &ty) : Ref(ty.ptr.get()) {}
1061
1063 bool is_mutable() const {
1064 return wasm_globaltype_mutability(ptr) == WASM_VAR;
1065 }
1066
1068 ValType::Ref content() const { return wasm_globaltype_content(ptr); }
1069 };
1070
1071private:
1072 Ref ref;
1073 GlobalType(wasm_globaltype_t *ptr) : ptr(ptr), ref(ptr) {}
1074
1075public:
1077 GlobalType(ValType ty, bool mut)
1078 : GlobalType(wasm_globaltype_new(
1079 ty.ptr.release(),
1080 (wasm_mutability_t)(mut ? WASM_VAR : WASM_CONST))) {}
1082 GlobalType(Ref other) : GlobalType(wasm_globaltype_copy(other.ptr)) {}
1085 : GlobalType(wasm_globaltype_copy(other.ptr.get())) {}
1088 ptr.reset(wasm_globaltype_copy(other.ptr.get()));
1089 return *this;
1090 }
1091 ~GlobalType() = default;
1093 GlobalType(GlobalType &&other) = default;
1095 GlobalType &operator=(GlobalType &&other) = default;
1096
1099 Ref *operator->() { return &ref; }
1102 Ref *operator*() { return &ref; }
1103};
1104
1109 friend class Func;
1110 friend class Linker;
1111
1112 struct deleter {
1113 void operator()(wasm_functype_t *p) const { wasm_functype_delete(p); }
1114 };
1115
1116 std::unique_ptr<wasm_functype_t, deleter> ptr;
1117
1118public:
1121 class Ref {
1122 friend class FuncType;
1123 const wasm_functype_t *ptr;
1124
1125 public:
1127 Ref(const wasm_functype_t *ptr) : ptr(ptr) {}
1129 Ref(const FuncType &ty) : Ref(ty.ptr.get()) {}
1130
1132 ValType::ListRef params() const { return wasm_functype_params(ptr); }
1133
1135 ValType::ListRef results() const { return wasm_functype_results(ptr); }
1136 };
1137
1138private:
1139 Ref ref;
1140 FuncType(wasm_functype_t *ptr) : ptr(ptr), ref(ptr) {}
1141
1142public:
1144 FuncType(std::initializer_list<ValType> params,
1145 std::initializer_list<ValType> results)
1146 : ref(nullptr) {
1147 *this = FuncType::from_iters(params, results);
1148 }
1149
1151 FuncType(Ref other) : FuncType(wasm_functype_copy(other.ptr)) {}
1153 FuncType(const FuncType &other)
1154 : FuncType(wasm_functype_copy(other.ptr.get())) {}
1156 FuncType &operator=(const FuncType &other) {
1157 ptr.reset(wasm_functype_copy(other.ptr.get()));
1158 return *this;
1159 }
1160 ~FuncType() = default;
1162 FuncType(FuncType &&other) = default;
1164 FuncType &operator=(FuncType &&other) = default;
1165
1167 template <typename P, typename R>
1168 static FuncType from_iters(P params, R results) {
1169 wasm_valtype_vec_t param_vec;
1170 wasm_valtype_vec_t result_vec;
1171 wasm_valtype_vec_new_uninitialized(&param_vec, params.size());
1172 wasm_valtype_vec_new_uninitialized(&result_vec, results.size());
1173 size_t i = 0;
1174
1175 for (auto val : params) {
1176 param_vec.data[i++] = val.ptr.release(); // NOLINT
1177 }
1178 i = 0;
1179 for (auto val : results) {
1180 result_vec.data[i++] = val.ptr.release(); // NOLINT
1181 }
1182
1183 return wasm_functype_new(&param_vec, &result_vec);
1184 }
1185
1188 Ref *operator->() { return &ref; }
1191 Ref *operator*() { return &ref; }
1192};
1193
1198public:
1201 class Ref {
1202 friend class ExternType;
1203
1204 const wasm_importtype_t *ptr;
1205
1206 // TODO: can this circle be broken another way?
1207 const wasm_externtype_t *raw_type() { return wasm_importtype_type(ptr); }
1208
1209 public:
1211 Ref(const wasm_importtype_t *ptr) : ptr(ptr) {}
1212
1214 std::string_view module() {
1215 const auto *name = wasm_importtype_module(ptr);
1216 return std::string_view(name->data, name->size);
1217 }
1218
1220 std::string_view name() {
1221 const auto *name = wasm_importtype_name(ptr);
1222 return std::string_view(name->data, name->size);
1223 }
1224 };
1225
1227 class List {
1228 friend class Module;
1229 wasm_importtype_vec_t list;
1230
1231 public:
1233 List() : list{} {
1234 list.size = 0;
1235 list.data = nullptr;
1236 }
1237 List(const List &other) = delete;
1239 List(List &&other) noexcept : list(other.list) { other.list.size = 0; }
1240 ~List() {
1241 if (list.size > 0) {
1242 wasm_importtype_vec_delete(&list);
1243 }
1244 }
1245
1246 List &operator=(const List &other) = delete;
1248 List &operator=(List &&other) noexcept {
1249 std::swap(list, other.list);
1250 return *this;
1251 }
1252
1255 typedef const Ref *iterator;
1257 iterator begin() const {
1258 return reinterpret_cast<iterator>(&list.data[0]); // NOLINT
1259 }
1261 iterator end() const {
1262 return reinterpret_cast<iterator>(&list.data[list.size]); // NOLINT
1263 }
1265 size_t size() const { return list.size; }
1266 };
1267};
1268
1273
1274public:
1278 class Ref {
1279 friend class ExternType;
1280
1281 const wasm_exporttype_t *ptr;
1282
1283 const wasm_externtype_t *raw_type() { return wasm_exporttype_type(ptr); }
1284
1285 public:
1287 Ref(const wasm_exporttype_t *ptr) : ptr(ptr) {}
1288
1290 std::string_view name() {
1291 const auto *name = wasm_exporttype_name(ptr);
1292 return std::string_view(name->data, name->size);
1293 }
1294 };
1295
1297 class List {
1298 friend class Module;
1299 wasm_exporttype_vec_t list;
1300
1301 public:
1303 List() : list{} {
1304 list.size = 0;
1305 list.data = nullptr;
1306 }
1307 List(const List &other) = delete;
1309 List(List &&other) noexcept : list(other.list) { other.list.size = 0; }
1310 ~List() {
1311 if (list.size > 0) {
1312 wasm_exporttype_vec_delete(&list);
1313 }
1314 }
1315
1316 List &operator=(const List &other) = delete;
1318 List &operator=(List &&other) noexcept {
1319 std::swap(list, other.list);
1320 return *this;
1321 }
1322
1325 typedef const Ref *iterator;
1327 iterator begin() const {
1328 return reinterpret_cast<iterator>(&list.data[0]); // NOLINT
1329 }
1331 iterator end() const {
1332 return reinterpret_cast<iterator>(&list.data[list.size]); // NOLINT
1333 }
1335 size_t size() const { return list.size; }
1336 };
1337};
1338
1343 friend class ExportType;
1344 friend class ImportType;
1345
1346public:
1353 typedef std::variant<FuncType::Ref, GlobalType::Ref, TableType::Ref,
1356
1359 // TODO: this would ideally be some sort of implicit constructor, unsure how
1360 // to do that though...
1361 return ref_from_c(ty.raw_type());
1362 }
1363
1366 // TODO: this would ideally be some sort of implicit constructor, unsure how
1367 // to do that though...
1368 return ref_from_c(ty.raw_type());
1369 }
1370
1371private:
1372 static Ref ref_from_c(const wasm_externtype_t *ptr) {
1373 switch (wasm_externtype_kind(ptr)) {
1374 case WASM_EXTERN_FUNC:
1375 return wasm_externtype_as_functype_const(ptr);
1376 case WASM_EXTERN_GLOBAL:
1377 return wasm_externtype_as_globaltype_const(ptr);
1378 case WASM_EXTERN_TABLE:
1379 return wasm_externtype_as_tabletype_const(ptr);
1380 case WASM_EXTERN_MEMORY:
1381 return wasm_externtype_as_memorytype_const(ptr);
1382 }
1383 std::abort();
1384 }
1385};
1386
1395 wasm_frame_t *frame;
1396
1397public:
1400 uint32_t func_index() const { return wasm_frame_func_index(frame); }
1403 size_t func_offset() const { return wasm_frame_func_offset(frame); }
1406 size_t module_offset() const { return wasm_frame_module_offset(frame); }
1407
1412 std::optional<std::string_view> func_name() const {
1413 const auto *name = wasmtime_frame_func_name(frame);
1414 if (name != nullptr) {
1415 return std::string_view(name->data, name->size);
1416 }
1417 return std::nullopt;
1418 }
1419
1424 std::optional<std::string_view> module_name() const {
1425 const auto *name = wasmtime_frame_module_name(frame);
1426 if (name != nullptr) {
1427 return std::string_view(name->data, name->size);
1428 }
1429 return std::nullopt;
1430 }
1431};
1432
1440class Trace {
1441 friend class Trap;
1442 friend class Error;
1443
1444 wasm_frame_vec_t vec;
1445
1446 Trace(wasm_frame_vec_t vec) : vec(vec) {}
1447
1448public:
1449 ~Trace() { wasm_frame_vec_delete(&vec); }
1450
1451 Trace(const Trace &other) = delete;
1452 Trace(Trace &&other) = delete;
1453 Trace &operator=(const Trace &other) = delete;
1454 Trace &operator=(Trace &&other) = delete;
1455
1457 typedef const FrameRef *iterator;
1458
1460 iterator begin() const {
1461 return reinterpret_cast<FrameRef *>(&vec.data[0]); // NOLINT
1462 }
1464 iterator end() const {
1465 return reinterpret_cast<FrameRef *>(&vec.data[vec.size]); // NOLINT
1466 }
1468 size_t size() const { return vec.size; }
1469};
1470
1471inline Trace Error::trace() const {
1472 wasm_frame_vec_t frames;
1473 wasmtime_error_wasm_trace(ptr.get(), &frames);
1474 return Trace(frames);
1475}
1476
1487class Trap {
1488 friend class Linker;
1489 friend class Instance;
1490 friend class Func;
1491 template <typename Params, typename Results> friend class TypedFunc;
1492
1493 struct deleter {
1494 void operator()(wasm_trap_t *p) const { wasm_trap_delete(p); }
1495 };
1496
1497 std::unique_ptr<wasm_trap_t, deleter> ptr;
1498
1499 Trap(wasm_trap_t *ptr) : ptr(ptr) {}
1500
1501public:
1503 explicit Trap(std::string_view msg)
1504 : Trap(wasmtime_trap_new(msg.data(), msg.size())) {}
1505
1507 std::string message() const {
1508 wasm_byte_vec_t msg;
1509 wasm_trap_message(ptr.get(), &msg);
1510 std::string ret(msg.data, msg.size - 1);
1511 wasm_byte_vec_delete(&msg);
1512 return ret;
1513 }
1514
1518 Trace trace() const {
1519 wasm_frame_vec_t frames;
1520 wasm_trap_trace(ptr.get(), &frames);
1521 return Trace(frames);
1522 }
1523};
1524
1528 std::variant<Trap, Error> data;
1529
1531 TrapError(Trap t) : data(std::move(t)) {}
1533 TrapError(Error e) : data(std::move(e)) {}
1534
1536 std::string message() const {
1537 if (const auto *trap = std::get_if<Trap>(&data)) {
1538 return trap->message();
1539 }
1540 if (const auto *error = std::get_if<Error>(&data)) {
1541 return std::string(error->message());
1542 }
1543 std::abort();
1544 }
1545};
1546
1549template <typename T> using TrapResult = Result<T, TrapError>;
1550
1559class Module {
1560 friend class Store;
1561 friend class Instance;
1562 friend class Linker;
1563
1564 struct deleter {
1565 void operator()(wasmtime_module_t *p) const { wasmtime_module_delete(p); }
1566 };
1567
1568 std::unique_ptr<wasmtime_module_t, deleter> ptr;
1569
1570 Module(wasmtime_module_t *raw) : ptr(raw) {}
1571
1572public:
1574 Module(const Module &other) : ptr(wasmtime_module_clone(other.ptr.get())) {}
1576 Module &operator=(const Module &other) {
1577 ptr.reset(wasmtime_module_clone(other.ptr.get()));
1578 return *this;
1579 }
1580 ~Module() = default;
1582 Module(Module &&other) = default;
1584 Module &operator=(Module &&other) = default;
1585
1592 static Result<Module> compile(Engine &engine, std::string_view wat) {
1593 auto wasm = wat2wasm(wat);
1594 if (!wasm) {
1595 return wasm.err();
1596 }
1597 auto bytes = wasm.ok();
1598 return compile(engine, bytes);
1599 }
1600
1612 wasmtime_module_t *ret = nullptr;
1613 auto *error =
1614 wasmtime_module_new(engine.ptr.get(), wasm.data(), wasm.size(), &ret);
1615 if (error != nullptr) {
1616 return Error(error);
1617 }
1618 return Module(ret);
1619 }
1620
1628 auto *error =
1629 wasmtime_module_validate(engine.ptr.get(), wasm.data(), wasm.size());
1630 if (error != nullptr) {
1631 return Error(error);
1632 }
1633 return std::monostate();
1634 }
1635
1649 wasmtime_module_t *ret = nullptr;
1650 auto *error = wasmtime_module_deserialize(engine.ptr.get(), wasm.data(),
1651 wasm.size(), &ret);
1652 if (error != nullptr) {
1653 return Error(error);
1654 }
1655 return Module(ret);
1656 }
1657
1671 const std::string &path) {
1672 wasmtime_module_t *ret = nullptr;
1673 auto *error =
1674 wasmtime_module_deserialize_file(engine.ptr.get(), path.c_str(), &ret);
1675 if (error != nullptr) {
1676 return Error(error);
1677 }
1678 return Module(ret);
1679 }
1680
1683 ImportType::List list;
1684 wasmtime_module_imports(ptr.get(), &list.list);
1685 return list;
1686 }
1687
1690 ExportType::List list;
1691 wasmtime_module_exports(ptr.get(), &list.list);
1692 return list;
1693 }
1694
1702 wasm_byte_vec_t bytes;
1703 auto *error = wasmtime_module_serialize(ptr.get(), &bytes);
1704 if (error != nullptr) {
1705 return Error(error);
1706 }
1707 std::vector<uint8_t> ret;
1708 // NOLINTNEXTLINE TODO can this be done without triggering lints?
1709 Span<uint8_t> raw(reinterpret_cast<uint8_t *>(bytes.data), bytes.size);
1710 ret.assign(raw.begin(), raw.end());
1711 wasm_byte_vec_delete(&bytes);
1712 return ret;
1713 }
1714};
1715
1722 friend class Store;
1723
1724 struct deleter {
1725 void operator()(wasi_config_t *p) const { wasi_config_delete(p); }
1726 };
1727
1728 std::unique_ptr<wasi_config_t, deleter> ptr;
1729
1730public:
1732 WasiConfig() : ptr(wasi_config_new()) {}
1733
1735 void argv(const std::vector<std::string> &args) {
1736 std::vector<const char *> ptrs;
1737 ptrs.reserve(args.size());
1738 for (const auto &arg : args) {
1739 ptrs.push_back(arg.c_str());
1740 }
1741
1742 wasi_config_set_argv(ptr.get(), (int)args.size(), ptrs.data());
1743 }
1744
1746 void inherit_argv() { wasi_config_inherit_argv(ptr.get()); }
1747
1751 void env(const std::vector<std::pair<std::string, std::string>> &env) {
1752 std::vector<const char *> names;
1753 std::vector<const char *> values;
1754 for (const auto &[name, value] : env) {
1755 names.push_back(name.c_str());
1756 values.push_back(value.c_str());
1757 }
1758 wasi_config_set_env(ptr.get(), (int)env.size(), names.data(),
1759 values.data());
1760 }
1761
1764 void inherit_env() { wasi_config_inherit_env(ptr.get()); }
1765
1768 [[nodiscard]] bool stdin_file(const std::string &path) {
1769 return wasi_config_set_stdin_file(ptr.get(), path.c_str());
1770 }
1771
1774 void inherit_stdin() { return wasi_config_inherit_stdin(ptr.get()); }
1775
1778 [[nodiscard]] bool stdout_file(const std::string &path) {
1779 return wasi_config_set_stdout_file(ptr.get(), path.c_str());
1780 }
1781
1784 void inherit_stdout() { return wasi_config_inherit_stdout(ptr.get()); }
1785
1788 [[nodiscard]] bool stderr_file(const std::string &path) {
1789 return wasi_config_set_stderr_file(ptr.get(), path.c_str());
1790 }
1791
1794 void inherit_stderr() { return wasi_config_inherit_stderr(ptr.get()); }
1795
1797 [[nodiscard]] bool preopen_dir(const std::string &path,
1798 const std::string &guest_path,
1799 size_t dir_perms,
1800 size_t file_perms) {
1801 return wasi_config_preopen_dir(ptr.get(), path.c_str(), guest_path.c_str(), dir_perms, file_perms);
1802 }
1803};
1804
1805class Caller;
1806template <typename Params, typename Results> class TypedFunc;
1807
1820class Store {
1821 struct deleter {
1822 void operator()(wasmtime_store_t *p) const { wasmtime_store_delete(p); }
1823 };
1824
1825 std::unique_ptr<wasmtime_store_t, deleter> ptr;
1826
1827 static void finalizer(void *ptr) {
1828 std::unique_ptr<std::any> _ptr(static_cast<std::any *>(ptr));
1829 }
1830
1831public:
1833 explicit Store(Engine &engine)
1834 : ptr(wasmtime_store_new(engine.ptr.get(), nullptr, finalizer)) {}
1835
1847 class Context {
1848 friend class Global;
1849 friend class Table;
1850 friend class Memory;
1851 friend class Func;
1852 friend class Instance;
1853 friend class Linker;
1854 friend class ExternRef;
1855 friend class Val;
1856 wasmtime_context_t *ptr;
1857
1858 Context(wasmtime_context_t *ptr) : ptr(ptr) {}
1859
1860 public:
1862 Context(Store &store) : Context(wasmtime_store_context(store.ptr.get())) {}
1864 Context(Store *store) : Context(*store) {}
1866 Context(Caller &caller);
1868 Context(Caller *caller);
1869
1872 void gc() { wasmtime_context_gc(ptr); }
1873
1881 auto *error = wasmtime_context_set_fuel(ptr, fuel);
1882 if (error != nullptr) {
1883 return Error(error);
1884 }
1885 return std::monostate();
1886 }
1887
1892 uint64_t fuel = 0;
1893 auto *error = wasmtime_context_get_fuel(ptr, &fuel);
1894 if (error != nullptr) {
1895 return Error(error);
1896 }
1897 return fuel;
1898 }
1899
1901 void set_data(std::any data) const {
1902 finalizer(static_cast<std::any *>(wasmtime_context_get_data(ptr)));
1903 wasmtime_context_set_data(
1904 ptr, std::make_unique<std::any>(std::move(data)).release());
1905 }
1906
1908 std::any &get_data() const {
1909 return *static_cast<std::any *>(wasmtime_context_get_data(ptr));
1910 }
1911
1918 auto *error = wasmtime_context_set_wasi(ptr, config.ptr.release());
1919 if (error != nullptr) {
1920 return Error(error);
1921 }
1922 return std::monostate();
1923 }
1924
1931 void set_epoch_deadline(uint64_t ticks_beyond_current) {
1932 wasmtime_context_set_epoch_deadline(ptr, ticks_beyond_current);
1933 }
1934
1936 wasmtime_context_t *raw_context() { return ptr; }
1937 };
1938
1969 void limiter(int64_t memory_size, int64_t table_elements, int64_t instances,
1970 int64_t tables, int64_t memories) {
1971 wasmtime_store_limiter(ptr.get(), memory_size, table_elements, instances,
1972 tables, memories);
1973 }
1974
1976 Context context() { return this; }
1977};
1978
1993 friend class Val;
1994
1995 wasmtime_externref_t val;
1996
1997 static void finalizer(void *ptr) {
1998 std::unique_ptr<std::any> _ptr(static_cast<std::any *>(ptr));
1999 }
2000
2001public:
2003 explicit ExternRef(wasmtime_externref_t val) : val(val) {}
2004
2010 template <typename T> explicit ExternRef(Store::Context cx, T val) {
2011 void *ptr = std::make_unique<std::any>(std::move(val)).release();
2012 bool ok = wasmtime_externref_new(cx.ptr, ptr, finalizer, &this->val);
2013 if (!ok) {
2014 fprintf(stderr, "failed to allocate a new externref");
2015 abort();
2016 }
2017 }
2018
2021 wasmtime_externref_t other;
2022 wasmtime_externref_clone(cx.ptr, &val, &other);
2023 return ExternRef(other);
2024 }
2025
2027 std::any &data(Store::Context cx) {
2028 return *static_cast<std::any *>(wasmtime_externref_data(cx.ptr, &val));
2029 }
2030
2034 wasmtime_externref_unroot(cx.ptr, &val);
2035 }
2036
2040 const wasmtime_externref_t *raw() const { return &val; }
2041};
2042
2043class Func;
2044class Global;
2045class Instance;
2046class Memory;
2047class Table;
2048
2051typedef std::variant<Func, Global, Memory, Table> Extern;
2052
2054struct V128 {
2056 wasmtime_v128 v128;
2057
2059 V128() : v128{} { memset(&v128[0], 0, sizeof(wasmtime_v128)); }
2060
2062 V128(const wasmtime_v128 &v) : v128{} {
2063 memcpy(&v128[0], &v[0], sizeof(wasmtime_v128));
2064 }
2065};
2066
2077class Val {
2078 friend class Global;
2079 friend class Table;
2080 friend class Func;
2081
2082 wasmtime_val_t val;
2083
2084 Val() : val{} {
2085 val.kind = WASMTIME_I32;
2086 val.of.i32 = 0;
2087 }
2088 Val(wasmtime_val_t val) : val(val) {}
2089
2090public:
2092 Val(int32_t i32) : val{} {
2093 val.kind = WASMTIME_I32;
2094 val.of.i32 = i32;
2095 }
2097 Val(int64_t i64) : val{} {
2098 val.kind = WASMTIME_I64;
2099 val.of.i64 = i64;
2100 }
2102 Val(float f32) : val{} {
2103 val.kind = WASMTIME_F32;
2104 val.of.f32 = f32;
2105 }
2107 Val(double f64) : val{} {
2108 val.kind = WASMTIME_F64;
2109 val.of.f64 = f64;
2110 }
2112 Val(const V128 &v128) : val{} {
2113 val.kind = WASMTIME_V128;
2114 memcpy(&val.of.v128[0], &v128.v128[0], sizeof(wasmtime_v128));
2115 }
2117 Val(std::optional<Func> func);
2119 Val(Func func);
2121 Val(std::optional<ExternRef> ptr) : val{} {
2122 val.kind = WASMTIME_EXTERNREF;
2123 if (ptr) {
2124 val.of.externref = ptr->val;
2125 } else {
2126 wasmtime_externref_set_null(&val.of.externref);
2127 }
2128 }
2131 Val(ExternRef ptr);
2132
2134 ValKind kind() const {
2135 switch (val.kind) {
2136 case WASMTIME_I32:
2137 return ValKind::I32;
2138 case WASMTIME_I64:
2139 return ValKind::I64;
2140 case WASMTIME_F32:
2141 return ValKind::F32;
2142 case WASMTIME_F64:
2143 return ValKind::F64;
2144 case WASMTIME_FUNCREF:
2145 return ValKind::FuncRef;
2146 case WASMTIME_EXTERNREF:
2147 return ValKind::ExternRef;
2148 case WASMTIME_V128:
2149 return ValKind::V128;
2150 }
2151 std::abort();
2152 }
2153
2156 int32_t i32() const {
2157 if (val.kind != WASMTIME_I32) {
2158 std::abort();
2159 }
2160 return val.of.i32;
2161 }
2162
2165 int64_t i64() const {
2166 if (val.kind != WASMTIME_I64) {
2167 std::abort();
2168 }
2169 return val.of.i64;
2170 }
2171
2174 float f32() const {
2175 if (val.kind != WASMTIME_F32) {
2176 std::abort();
2177 }
2178 return val.of.f32;
2179 }
2180
2183 double f64() const {
2184 if (val.kind != WASMTIME_F64) {
2185 std::abort();
2186 }
2187 return val.of.f64;
2188 }
2189
2192 V128 v128() const {
2193 if (val.kind != WASMTIME_V128) {
2194 std::abort();
2195 }
2196 return val.of.v128;
2197 }
2198
2204 std::optional<ExternRef> externref(Store::Context cx) const {
2205 if (val.kind != WASMTIME_EXTERNREF) {
2206 std::abort();
2207 }
2208 if (val.of.externref.store_id == 0) {
2209 return std::nullopt;
2210 }
2211 wasmtime_externref_t other;
2212 wasmtime_externref_clone(cx.ptr, &val.of.externref, &other);
2213 return ExternRef(other);
2214 }
2215
2221 std::optional<Func> funcref() const;
2222
2225 wasmtime_val_unroot(cx.ptr, &val);
2226 }
2227};
2228
2236class Caller {
2237 friend class Func;
2238 friend class Store;
2239 wasmtime_caller_t *ptr;
2240 Caller(wasmtime_caller_t *ptr) : ptr(ptr) {}
2241
2242public:
2247 std::optional<Extern> get_export(std::string_view name);
2248
2250 Store::Context context() { return this; }
2251};
2252
2253inline Store::Context::Context(Caller &caller)
2254 : Context(wasmtime_caller_context(caller.ptr)) {}
2255inline Store::Context::Context(Caller *caller) : Context(*caller) {}
2256
2257namespace detail {
2258
2261template <typename T> struct WasmType { static const bool valid = false; };
2262
2265// NOLINTNEXTLINE
2266#define NATIVE_WASM_TYPE(native, valkind, field) \
2267 template <> struct WasmType<native> { \
2268 static const bool valid = true; \
2269 static const ValKind kind = ValKind::valkind; \
2270 static void store(Store::Context cx, wasmtime_val_raw_t *p, \
2271 const native &t) { \
2272 p->field = t; \
2273 } \
2274 static native load(Store::Context cx, wasmtime_val_raw_t *p) { \
2275 return p->field; \
2276 } \
2277 };
2278
2279NATIVE_WASM_TYPE(int32_t, I32, i32)
2280NATIVE_WASM_TYPE(uint32_t, I32, i32)
2281NATIVE_WASM_TYPE(int64_t, I64, i64)
2282NATIVE_WASM_TYPE(uint64_t, I64, i64)
2283NATIVE_WASM_TYPE(float, F32, f32)
2284NATIVE_WASM_TYPE(double, F64, f64)
2285
2286#undef NATIVE_WASM_TYPE
2287
2290template <> struct WasmType<std::optional<ExternRef>> {
2291 static const bool valid = true;
2292 static const ValKind kind = ValKind::ExternRef;
2293 static void store(Store::Context cx, wasmtime_val_raw_t *p,
2294 const std::optional<ExternRef> &ref) {
2295 if (ref) {
2296 p->externref = wasmtime_externref_to_raw(cx.raw_context(), ref->raw());
2297 } else {
2298 p->externref = 0;
2299 }
2300 }
2301 static std::optional<ExternRef> load(Store::Context cx,
2302 wasmtime_val_raw_t *p) {
2303 if (p->externref == 0) {
2304 return std::nullopt;
2305 }
2306 wasmtime_externref_t val;
2307 wasmtime_externref_from_raw(cx.raw_context(), p->externref, &val);
2308 return ExternRef(val);
2309 }
2310};
2311
2313template <> struct WasmType<V128> {
2314 static const bool valid = true;
2315 static const ValKind kind = ValKind::V128;
2316 static void store(Store::Context cx, wasmtime_val_raw_t *p, const V128 &t) {
2317 memcpy(&p->v128[0], &t.v128[0], sizeof(wasmtime_v128));
2318 }
2319 static V128 load(Store::Context cx, wasmtime_val_raw_t *p) { return p->v128; }
2320};
2321
2326template <typename T> struct WasmTypeList {
2327 static const bool valid = WasmType<T>::valid;
2328 static const size_t size = 1;
2329 static bool matches(ValType::ListRef types) {
2330 return WasmTypeList<std::tuple<T>>::matches(types);
2331 }
2332 static void store(Store::Context cx, wasmtime_val_raw_t *storage,
2333 const T &t) {
2334 WasmType<T>::store(cx, storage, t);
2335 }
2336 static T load(Store::Context cx, wasmtime_val_raw_t *storage) {
2337 return WasmType<T>::load(cx, storage);
2338 }
2339 static std::vector<ValType> types() { return {WasmType<T>::kind}; }
2340};
2341
2343template <> struct WasmTypeList<std::monostate> {
2344 static const bool valid = true;
2345 static const size_t size = 0;
2346 static bool matches(ValType::ListRef types) { return types.size() == 0; }
2347 static void store(Store::Context cx, wasmtime_val_raw_t *storage,
2348 const std::monostate &t) {}
2349 static std::monostate load(Store::Context cx, wasmtime_val_raw_t *storage) {
2350 return std::monostate();
2351 }
2352 static std::vector<ValType> types() { return {}; }
2353};
2354
2356template <typename... T> struct WasmTypeList<std::tuple<T...>> {
2357 static const bool valid = (WasmType<T>::valid && ...);
2358 static const size_t size = sizeof...(T);
2359 static bool matches(ValType::ListRef types) {
2360 if (types.size() != size) {
2361 return false;
2362 }
2363 size_t n = 0;
2364 return ((WasmType<T>::kind == types.begin()[n++].kind()) && ...);
2365 }
2366 static void store(Store::Context cx, wasmtime_val_raw_t *storage,
2367 const std::tuple<T...> &t) {
2368 size_t n = 0;
2369 std::apply(
2370 [&](const auto &...val) {
2371 (WasmType<T>::store(cx, &storage[n++], val), ...); // NOLINT
2372 },
2373 t);
2374 }
2375 static std::tuple<T...> load(Store::Context cx, wasmtime_val_raw_t *storage) {
2376 size_t n = 0;
2377 return std::tuple<T...>{WasmType<T>::load(cx, &storage[n++])...}; // NOLINT
2378 }
2379 static std::vector<ValType> types() { return {WasmType<T>::kind...}; }
2380};
2381
2385template <typename R> struct WasmHostRet {
2386 using Results = WasmTypeList<R>;
2387
2388 template <typename F, typename... A>
2389 static std::optional<Trap> invoke(F f, Caller cx, wasmtime_val_raw_t *raw,
2390 A... args) {
2391 auto ret = f(args...);
2392 Results::store(cx, raw, ret);
2393 return std::nullopt;
2394 }
2395};
2396
2398template <> struct WasmHostRet<void> {
2400
2401 template <typename F, typename... A>
2402 static std::optional<Trap> invoke(F f, Caller cx, wasmtime_val_raw_t *raw,
2403 A... args) {
2404 f(args...);
2405 return std::nullopt;
2406 }
2407};
2408
2409// Alternative method of returning "nothing" (also enables `std::monostate` in
2410// the `R` type of `Result` below)
2411template <> struct WasmHostRet<std::monostate> : public WasmHostRet<void> {};
2412
2415template <typename R> struct WasmHostRet<Result<R, Trap>> {
2416 using Results = WasmTypeList<R>;
2417
2418 template <typename F, typename... A>
2419 static std::optional<Trap> invoke(F f, Caller cx, wasmtime_val_raw_t *raw,
2420 A... args) {
2421 Result<R, Trap> ret = f(args...);
2422 if (!ret) {
2423 return ret.err();
2424 }
2425 Results::store(cx, raw, ret.ok());
2426 return std::nullopt;
2427 }
2428};
2429
2430template <typename F, typename = void> struct WasmHostFunc;
2431
2434template <typename R, typename... A> struct WasmHostFunc<R (*)(A...)> {
2435 using Params = WasmTypeList<std::tuple<A...>>;
2436 using Results = typename WasmHostRet<R>::Results;
2437
2438 template <typename F>
2439 static std::optional<Trap> invoke(F &f, Caller cx, wasmtime_val_raw_t *raw) {
2440 auto params = Params::load(cx, raw);
2441 return std::apply(
2442 [&](const auto &...val) {
2443 return WasmHostRet<R>::invoke(f, cx, raw, val...);
2444 },
2445 params);
2446 }
2447};
2448
2450template <typename R, typename... A>
2451struct WasmHostFunc<R (*)(Caller, A...)> : public WasmHostFunc<R (*)(A...)> {
2452 // Override `invoke` here to pass the `cx` as the first parameter
2453 template <typename F>
2454 static std::optional<Trap> invoke(F &f, Caller cx, wasmtime_val_raw_t *raw) {
2455 auto params = WasmTypeList<std::tuple<A...>>::load(cx, raw);
2456 return std::apply(
2457 [&](const auto &...val) {
2458 return WasmHostRet<R>::invoke(f, cx, raw, cx, val...);
2459 },
2460 params);
2461 }
2462};
2463
2465template <typename R, typename C, typename... A>
2466struct WasmHostFunc<R (C::*)(A...)> : public WasmHostFunc<R (*)(A...)> {};
2467
2469template <typename R, typename C, typename... A>
2470struct WasmHostFunc<R (C::*)(A...) const> : public WasmHostFunc<R (*)(A...)> {};
2471
2474template <typename R, typename C, typename... A>
2475struct WasmHostFunc<R (C::*)(Caller, A...)>
2476 : public WasmHostFunc<R (*)(Caller, A...)> {};
2477
2480template <typename R, typename C, typename... A>
2481struct WasmHostFunc<R (C::*)(Caller, A...) const>
2482 : public WasmHostFunc<R (*)(Caller, A...)> {};
2483
2486template <typename T>
2487struct WasmHostFunc<T, std::void_t<decltype(&T::operator())>>
2488 : public WasmHostFunc<decltype(&T::operator())> {};
2489
2490} // namespace detail
2491
2492using namespace detail;
2493
2505class Func {
2506 friend class Val;
2507 friend class Instance;
2508 friend class Linker;
2509 template <typename Params, typename Results> friend class TypedFunc;
2510
2511 wasmtime_func_t func;
2512
2513 template <typename F>
2514 static wasm_trap_t *raw_callback(void *env, wasmtime_caller_t *caller,
2515 const wasmtime_val_t *args, size_t nargs,
2516 wasmtime_val_t *results, size_t nresults) {
2517 static_assert(alignof(Val) == alignof(wasmtime_val_t));
2518 static_assert(sizeof(Val) == sizeof(wasmtime_val_t));
2519 F *func = reinterpret_cast<F *>(env); // NOLINT
2520 Span<const Val> args_span(reinterpret_cast<const Val *>(args), // NOLINT
2521 nargs);
2522 Span<Val> results_span(reinterpret_cast<Val *>(results), // NOLINT
2523 nresults);
2525 (*func)(Caller(caller), args_span, results_span);
2526 if (!result) {
2527 return result.err().ptr.release();
2528 }
2529 return nullptr;
2530 }
2531
2532 template <typename F>
2533 static wasm_trap_t *
2534 raw_callback_unchecked(void *env, wasmtime_caller_t *caller,
2535 wasmtime_val_raw_t *args_and_results,
2536 size_t nargs_and_results) {
2537 using HostFunc = WasmHostFunc<F>;
2538 Caller cx(caller);
2539 F *func = reinterpret_cast<F *>(env); // NOLINT
2540 auto trap = HostFunc::invoke(*func, cx, args_and_results);
2541 if (trap) {
2542 return trap->ptr.release();
2543 }
2544 return nullptr;
2545 }
2546
2547 template <typename F> static void raw_finalize(void *env) {
2548 std::unique_ptr<F> ptr(reinterpret_cast<F *>(env)); // NOLINT
2549 }
2550
2551public:
2553 Func(wasmtime_func_t func) : func(func) {}
2554
2586 template <typename F,
2587 std::enable_if_t<
2588 std::is_invocable_r_v<Result<std::monostate, Trap>, F, Caller,
2590 bool> = true>
2591 Func(Store::Context cx, const FuncType &ty, F f) : func({}) {
2592 wasmtime_func_new(cx.ptr, ty.ptr.get(), raw_callback<F>,
2593 std::make_unique<F>(f).release(), raw_finalize<F>, &func);
2594 }
2595
2634 template <typename F,
2635 std::enable_if_t<WasmHostFunc<F>::Params::valid, bool> = true,
2636 std::enable_if_t<WasmHostFunc<F>::Results::valid, bool> = true>
2637 static Func wrap(Store::Context cx, F f) {
2638 using HostFunc = WasmHostFunc<F>;
2639 auto params = HostFunc::Params::types();
2640 auto results = HostFunc::Results::types();
2641 auto ty = FuncType::from_iters(params, results);
2642 wasmtime_func_t func;
2643 wasmtime_func_new_unchecked(cx.ptr, ty.ptr.get(), raw_callback_unchecked<F>,
2644 std::make_unique<F>(f).release(),
2645 raw_finalize<F>, &func);
2646 return func;
2647 }
2648
2669 template <typename I>
2671 const I &end) const {
2672 std::vector<wasmtime_val_t> raw_params;
2673 raw_params.reserve(end - begin);
2674 for (auto i = begin; i != end; i++) {
2675 raw_params.push_back(i->val);
2676 }
2677 size_t nresults = this->type(cx)->results().size();
2678 std::vector<wasmtime_val_t> raw_results(nresults);
2679
2680 wasm_trap_t *trap = nullptr;
2681 auto *error =
2682 wasmtime_func_call(cx.ptr, &func, raw_params.data(), raw_params.size(),
2683 raw_results.data(), raw_results.capacity(), &trap);
2684 if (error != nullptr) {
2685 return TrapError(Error(error));
2686 }
2687 if (trap != nullptr) {
2688 return TrapError(Trap(trap));
2689 }
2690
2691 std::vector<Val> results;
2692 results.reserve(nresults);
2693 for (size_t i = 0; i < nresults; i++) {
2694 results.push_back(raw_results[i]);
2695 }
2696 return results;
2697 }
2698
2700 const std::vector<Val> &params) const {
2701 return this->call(cx, params.begin(), params.end());
2702 }
2703
2704 TrapResult<std::vector<Val>>
2705 call(Store::Context cx, const std::initializer_list<Val> &params) const {
2706 return this->call(cx, params.begin(), params.end());
2707 }
2708
2711 return wasmtime_func_type(cx.ptr, &func);
2712 }
2713
2731 template <typename Params, typename Results,
2732 std::enable_if_t<WasmTypeList<Params>::valid, bool> = true,
2733 std::enable_if_t<WasmTypeList<Results>::valid, bool> = true>
2735 auto ty = this->type(cx);
2736 if (!WasmTypeList<Params>::matches(ty->params()) ||
2737 !WasmTypeList<Results>::matches(ty->results())) {
2738 return Trap("static type for this function does not match actual type");
2739 }
2740 TypedFunc<Params, Results> ret(*this);
2741 return ret;
2742 }
2743
2745 const wasmtime_func_t &raw_func() const { return func; }
2746};
2747
2752template <typename Params, typename Results> class TypedFunc {
2753 friend class Func;
2754 Func f;
2755 TypedFunc(Func func) : f(func) {}
2756
2757public:
2768 TrapResult<Results> call(Store::Context cx, Params params) const {
2769 std::array<wasmtime_val_raw_t, std::max(WasmTypeList<Params>::size,
2771 storage;
2772 wasmtime_val_raw_t *ptr = storage.data();
2773 if (ptr == nullptr)
2774 ptr = reinterpret_cast<wasmtime_val_raw_t*>(alignof(wasmtime_val_raw_t));
2775 WasmTypeList<Params>::store(cx, ptr, params);
2776 wasm_trap_t *trap = nullptr;
2777 auto *error = wasmtime_func_call_unchecked(
2778 cx.raw_context(), &f.func, ptr, storage.size(), &trap);
2779 if (error != nullptr) {
2780 return TrapError(Error(error));
2781 }
2782 if (trap != nullptr) {
2783 return TrapError(Trap(trap));
2784 }
2785 return WasmTypeList<Results>::load(cx, ptr);
2786 }
2787
2789 const Func &func() const { return f; }
2790};
2791
2792inline Val::Val(std::optional<Func> func) : val{} {
2793 val.kind = WASMTIME_FUNCREF;
2794 if (func) {
2795 val.of.funcref = (*func).func;
2796 } else {
2797 wasmtime_funcref_set_null(&val.of.funcref);
2798 }
2799}
2800
2801inline Val::Val(Func func) : Val(std::optional(func)) {}
2802inline Val::Val(ExternRef ptr) : Val(std::optional(ptr)) {}
2803
2804inline std::optional<Func> Val::funcref() const {
2805 if (val.kind != WASMTIME_FUNCREF) {
2806 std::abort();
2807 }
2808 if (val.of.funcref.store_id == 0) {
2809 return std::nullopt;
2810 }
2811 return Func(val.of.funcref);
2812}
2813
2815template <> struct detail::WasmType<std::optional<Func>> {
2817 static const bool valid = true;
2819 static const ValKind kind = ValKind::FuncRef;
2821 static void store(Store::Context cx, wasmtime_val_raw_t *p,
2822 const std::optional<Func> func) {
2823 if (func) {
2824 p->funcref = wasmtime_func_to_raw(cx.raw_context(), &func->raw_func());
2825 } else {
2826 p->funcref = 0;
2827 }
2828 }
2830 static std::optional<Func> load(Store::Context cx, wasmtime_val_raw_t *p) {
2831 if (p->funcref == 0) {
2832 return std::nullopt;
2833 }
2834 wasmtime_func_t ret;
2835 wasmtime_func_from_raw(cx.raw_context(), p->funcref, &ret);
2836 return ret;
2837 }
2838};
2839
2852class Global {
2853 friend class Instance;
2854 wasmtime_global_t global;
2855
2856public:
2858 Global(wasmtime_global_t global) : global(global) {}
2859
2870 const Val &init) {
2871 wasmtime_global_t global;
2872 auto *error = wasmtime_global_new(cx.ptr, ty.ptr.get(), &init.val, &global);
2873 if (error != nullptr) {
2874 return Error(error);
2875 }
2876 return Global(global);
2877 }
2878
2881 return wasmtime_global_type(cx.ptr, &global);
2882 }
2883
2885 Val get(Store::Context cx) const;
2886
2891 auto *error = wasmtime_global_set(cx.ptr, &global, &val.val);
2892 if (error != nullptr) {
2893 return Error(error);
2894 }
2895 return std::monostate();
2896 }
2897};
2898
2911class Table {
2912 friend class Instance;
2913 wasmtime_table_t table;
2914
2915public:
2917 Table(wasmtime_table_t table) : table(table) {}
2918
2929 const Val &init) {
2930 wasmtime_table_t table;
2931 auto *error = wasmtime_table_new(cx.ptr, ty.ptr.get(), &init.val, &table);
2932 if (error != nullptr) {
2933 return Error(error);
2934 }
2935 return Table(table);
2936 }
2937
2940 return wasmtime_table_type(cx.ptr, &table);
2941 }
2942
2944 uint64_t size(Store::Context cx) const {
2945 return wasmtime_table_size(cx.ptr, &table);
2946 }
2947
2951 std::optional<Val> get(Store::Context cx, uint64_t idx) const {
2952 Val val;
2953 if (wasmtime_table_get(cx.ptr, &table, idx, &val.val)) {
2954 return val;
2955 }
2956 return std::nullopt;
2957 }
2958
2963 const Val &val) const {
2964 auto *error = wasmtime_table_set(cx.ptr, &table, idx, &val.val);
2965 if (error != nullptr) {
2966 return Error(error);
2967 }
2968 return std::monostate();
2969 }
2970
2980 const Val &init) const {
2981 uint64_t prev = 0;
2982 auto *error = wasmtime_table_grow(cx.ptr, &table, delta, &init.val, &prev);
2983 if (error != nullptr) {
2984 return Error(error);
2985 }
2986 return prev;
2987 }
2988};
2989
2990// gcc 8.3.0 seems to require that this comes after the definition of `Table`. I
2991// don't know why...
2993 Val val;
2994 wasmtime_global_get(cx.ptr, &global, &val.val);
2995 return val;
2996}
2997
3009class Memory {
3010 friend class Instance;
3011 wasmtime_memory_t memory;
3012
3013public:
3015 Memory(wasmtime_memory_t memory) : memory(memory) {}
3016
3019 wasmtime_memory_t memory;
3020 auto *error = wasmtime_memory_new(cx.ptr, ty.ptr.get(), &memory);
3021 if (error != nullptr) {
3022 return Error(error);
3023 }
3024 return Memory(memory);
3025 }
3026
3029 return wasmtime_memory_type(cx.ptr, &memory);
3030 }
3031
3033 uint64_t size(Store::Context cx) const {
3034 return wasmtime_memory_size(cx.ptr, &memory);
3035 }
3036
3043 auto *base = wasmtime_memory_data(cx.ptr, &memory);
3044 auto size = wasmtime_memory_data_size(cx.ptr, &memory);
3045 return {base, size};
3046 }
3047
3052 Result<uint64_t> grow(Store::Context cx, uint64_t delta) const {
3053 uint64_t prev = 0;
3054 auto *error = wasmtime_memory_grow(cx.ptr, &memory, delta, &prev);
3055 if (error != nullptr) {
3056 return Error(error);
3057 }
3058 return prev;
3059 }
3060};
3061
3075 friend class Linker;
3076 friend class Caller;
3077
3078 wasmtime_instance_t instance;
3079
3080 static Extern cvt(wasmtime_extern_t &e) {
3081 switch (e.kind) {
3082 case WASMTIME_EXTERN_FUNC:
3083 return Func(e.of.func);
3084 case WASMTIME_EXTERN_GLOBAL:
3085 return Global(e.of.global);
3086 case WASMTIME_EXTERN_MEMORY:
3087 return Memory(e.of.memory);
3088 case WASMTIME_EXTERN_TABLE:
3089 return Table(e.of.table);
3090 }
3091 std::abort();
3092 }
3093
3094 static void cvt(const Extern &e, wasmtime_extern_t &raw) {
3095 if (const auto *func = std::get_if<Func>(&e)) {
3096 raw.kind = WASMTIME_EXTERN_FUNC;
3097 raw.of.func = func->func;
3098 } else if (const auto *global = std::get_if<Global>(&e)) {
3099 raw.kind = WASMTIME_EXTERN_GLOBAL;
3100 raw.of.global = global->global;
3101 } else if (const auto *table = std::get_if<Table>(&e)) {
3102 raw.kind = WASMTIME_EXTERN_TABLE;
3103 raw.of.table = table->table;
3104 } else if (const auto *memory = std::get_if<Memory>(&e)) {
3105 raw.kind = WASMTIME_EXTERN_MEMORY;
3106 raw.of.memory = memory->memory;
3107 } else {
3108 std::abort();
3109 }
3110 }
3111
3112public:
3114 Instance(wasmtime_instance_t instance) : instance(instance) {}
3115
3133 const std::vector<Extern> &imports) {
3134 std::vector<wasmtime_extern_t> raw_imports;
3135 for (const auto &item : imports) {
3136 raw_imports.push_back(wasmtime_extern_t{});
3137 auto &last = raw_imports.back();
3138 Instance::cvt(item, last);
3139 }
3140 wasmtime_instance_t instance;
3141 wasm_trap_t *trap = nullptr;
3142 auto *error = wasmtime_instance_new(cx.ptr, m.ptr.get(), raw_imports.data(),
3143 raw_imports.size(), &instance, &trap);
3144 if (error != nullptr) {
3145 return TrapError(Error(error));
3146 }
3147 if (trap != nullptr) {
3148 return TrapError(Trap(trap));
3149 }
3150 return Instance(instance);
3151 }
3152
3159 std::optional<Extern> get(Store::Context cx, std::string_view name) {
3160 wasmtime_extern_t e;
3161 if (!wasmtime_instance_export_get(cx.ptr, &instance, name.data(),
3162 name.size(), &e)) {
3163 return std::nullopt;
3164 }
3165 return Instance::cvt(e);
3166 }
3167
3174 std::optional<std::pair<std::string_view, Extern>> get(Store::Context cx,
3175 size_t idx) {
3176 wasmtime_extern_t e;
3177 // I'm not sure why clang-tidy thinks this is using va_list or anything
3178 // related to that...
3179 // NOLINTNEXTLINE(cppcoreguidelines-pro-type-vararg)
3180 char *name = nullptr;
3181 size_t len = 0;
3182 if (!wasmtime_instance_export_nth(cx.ptr, &instance, idx, &name, &len,
3183 &e)) {
3184 return std::nullopt;
3185 }
3186 std::string_view n(name, len);
3187 return std::pair(n, Instance::cvt(e));
3188 }
3189};
3190
3191inline std::optional<Extern> Caller::get_export(std::string_view name) {
3192 wasmtime_extern_t item;
3193 if (wasmtime_caller_export_get(ptr, name.data(), name.size(), &item)) {
3194 return Instance::cvt(item);
3195 }
3196 return std::nullopt;
3197}
3198
3206class Linker {
3207 struct deleter {
3208 void operator()(wasmtime_linker_t *p) const { wasmtime_linker_delete(p); }
3209 };
3210
3211 std::unique_ptr<wasmtime_linker_t, deleter> ptr;
3212
3213public:
3215 explicit Linker(Engine &engine)
3216 : ptr(wasmtime_linker_new(engine.ptr.get())) {}
3217
3221 void allow_shadowing(bool allow) {
3222 wasmtime_linker_allow_shadowing(ptr.get(), allow);
3223 }
3224
3227 std::string_view name, const Extern &item) {
3228 wasmtime_extern_t raw;
3229 Instance::cvt(item, raw);
3230 auto *error =
3231 wasmtime_linker_define(ptr.get(), cx.ptr, module.data(), module.size(),
3232 name.data(), name.size(), &raw);
3233 if (error != nullptr) {
3234 return Error(error);
3235 }
3236 return std::monostate();
3237 }
3238
3244 auto *error = wasmtime_linker_define_wasi(ptr.get());
3245 if (error != nullptr) {
3246 return Error(error);
3247 }
3248 return std::monostate();
3249 }
3250
3254 define_instance(Store::Context cx, std::string_view name, Instance instance) {
3255 auto *error = wasmtime_linker_define_instance(
3256 ptr.get(), cx.ptr, name.data(), name.size(), &instance.instance);
3257 if (error != nullptr) {
3258 return Error(error);
3259 }
3260 return std::monostate();
3261 }
3262
3266 wasmtime_instance_t instance;
3267 wasm_trap_t *trap = nullptr;
3268 auto *error = wasmtime_linker_instantiate(ptr.get(), cx.ptr, m.ptr.get(),
3269 &instance, &trap);
3270 if (error != nullptr) {
3271 return TrapError(Error(error));
3272 }
3273 if (trap != nullptr) {
3274 return TrapError(Trap(trap));
3275 }
3276 return Instance(instance);
3277 }
3278
3282 const Module &m) {
3283 auto *error = wasmtime_linker_module(ptr.get(), cx.ptr, name.data(),
3284 name.size(), m.ptr.get());
3285 if (error != nullptr) {
3286 return Error(error);
3287 }
3288 return std::monostate();
3289 }
3290
3293 [[nodiscard]] std::optional<Extern>
3294 get(Store::Context cx, std::string_view module, std::string_view name) {
3295 wasmtime_extern_t item;
3296 if (wasmtime_linker_get(ptr.get(), cx.ptr, module.data(), module.size(),
3297 name.data(), name.size(), &item)) {
3298 return Instance::cvt(item);
3299 }
3300 return std::nullopt;
3301 }
3302
3305 template <typename F,
3306 std::enable_if_t<
3307 std::is_invocable_r_v<Result<std::monostate, Trap>, F, Caller,
3309 bool> = true>
3311 std::string_view name, const FuncType &ty,
3312 F&& f) {
3313
3314 auto *error = wasmtime_linker_define_func(
3315 ptr.get(), module.data(), module.length(), name.data(), name.length(),
3316 ty.ptr.get(), Func::raw_callback<std::remove_reference_t<F>>, std::make_unique<std::remove_reference_t<F>>(std::forward<F>(f)).release(),
3317 Func::raw_finalize<std::remove_reference_t<F>>);
3318
3319 if (error != nullptr) {
3320 return Error(error);
3321 }
3322
3323 return std::monostate();
3324 }
3325
3328 template <typename F,
3329 std::enable_if_t<WasmHostFunc<F>::Params::valid, bool> = true,
3330 std::enable_if_t<WasmHostFunc<F>::Results::valid, bool> = true>
3332 std::string_view name, F&& f) {
3333 using HostFunc = WasmHostFunc<F>;
3334 auto params = HostFunc::Params::types();
3335 auto results = HostFunc::Results::types();
3336 auto ty = FuncType::from_iters(params, results);
3337 auto *error = wasmtime_linker_define_func_unchecked(
3338 ptr.get(), module.data(), module.length(), name.data(), name.length(),
3339 ty.ptr.get(), Func::raw_callback_unchecked<std::remove_reference_t<F>>,
3340 std::make_unique<std::remove_reference_t<F>>(std::forward<F>(f)).release(), Func::raw_finalize<std::remove_reference_t<F>>);
3341
3342 if (error != nullptr) {
3343 return Error(error);
3344 }
3345
3346 return std::monostate();
3347 }
3348
3351 Result<Func> get_default(Store::Context cx, std::string_view name) {
3352 wasmtime_func_t item;
3353 auto *error = wasmtime_linker_get_default(ptr.get(), cx.ptr, name.data(),
3354 name.size(), &item);
3355 if (error != nullptr) {
3356 return Error(error);
3357 }
3358 return Func(item);
3359 }
3360};
3361
3362} // namespace wasmtime
3363
3364#endif // WASMTIME_HH
Structure provided to host functions to lookup caller information or acquire a Store::Context.
Definition: wasmtime.hh:2236
std::optional< Extern > get_export(std::string_view name)
Definition: wasmtime.hh:3191
Store::Context context()
Explicitly acquire a Store::Context from this Caller.
Definition: wasmtime.hh:2250
Configuration for Wasmtime.
Definition: wasmtime.hh:462
void memory_guard_size(size_t size)
Configures the size of memory's guard region.
Definition: wasmtime.hh:604
Result< std::monostate > cache_load(const std::string &path)
Loads cache configuration from the specified filename.
Definition: wasmtime.hh:622
void wasm_bulk_memory(bool enable)
Configures whether the WebAssembly bulk memory proposal is enabled.
Definition: wasmtime.hh:531
void cranelift_debug_verifier(bool enable)
Configures whether cranelift's debug verifier is enabled.
Definition: wasmtime.hh:574
Config()
Creates configuration with all the default settings.
Definition: wasmtime.hh:473
void debug_info(bool enable)
Configures whether dwarf debuginfo is emitted for assisting in-process debugging.
Definition: wasmtime.hh:479
void wasm_simd(bool enable)
Configures whether the WebAssembly simd proposal is enabled.
Definition: wasmtime.hh:524
void wasm_function_references(bool enable)
Configures whether the WebAssembly function references proposal will be enabled.
Definition: wasmtime.hh:552
void wasm_threads(bool enable)
Configures whether the WebAssembly threads proposal is enabled.
Definition: wasmtime.hh:509
void strategy(Strategy strategy)
Configures compilation strategy for wasm code.
Definition: wasmtime.hh:566
void memory_reservation(size_t size)
Configures the size of the initial linear memory allocation.
Definition: wasmtime.hh:597
void wasm_reference_types(bool enable)
Configures whether the WebAssembly reference types proposal is enabled.
Definition: wasmtime.hh:517
void cranelift_opt_level(OptLevel level)
Configures cranelift's optimization level.
Definition: wasmtime.hh:581
void consume_fuel(bool enable)
Configures whether WebAssembly code will consume fuel and trap when it runs out.
Definition: wasmtime.hh:495
void pooling_allocation_strategy(const PoolAllocationConfig &config)
Enables and configures the pooling allocation strategy.
Definition: wasmtime.hh:633
void wasm_multi_value(bool enable)
Configures whether the WebAssembly multi value proposal is enabled.
Definition: wasmtime.hh:538
void max_wasm_stack(size_t stack)
Configures the maximum amount of native stack wasm can consume.
Definition: wasmtime.hh:502
Result< std::monostate > cache_load_default()
Loads the default cache configuration present on the system.
Definition: wasmtime.hh:611
void parallel_compilation(bool enable)
Configure whether wasmtime should compile a module using multiple threads.
Definition: wasmtime.hh:559
void epoch_interruption(bool enable)
Configures whether epochs are enabled which can be used to interrupt currently executing WebAssembly.
Definition: wasmtime.hh:487
void wasm_gc(bool enable)
Configures whether the WebAssembly Garbage Collection proposal will be enabled.
Definition: wasmtime.hh:545
void profiler(ProfilingStrategy profiler)
Configures an active wasm profiler.
Definition: wasmtime.hh:589
Global compilation state in Wasmtime.
Definition: wasmtime.hh:645
Engine(Config config)
Creates an engine with the specified compilation settings.
Definition: wasmtime.hh:660
Engine()
Creates an engine with default compilation settings.
Definition: wasmtime.hh:658
void increment_epoch() const
Increments the current epoch which may result in interrupting currently executing WebAssembly in conn...
Definition: wasmtime.hh:666
Errors coming from Wasmtime.
Definition: wasmtime.hh:143
Error(wasmtime_error_t *error)
Creates an error from the raw C API representation.
Definition: wasmtime.hh:154
Trace trace() const
Definition: wasmtime.hh:1471
std::optional< int32_t > i32_exit() const
Definition: wasmtime.hh:167
std::string message() const
Returns the error message associated with this error.
Definition: wasmtime.hh:157
An owned list of ExportType instances.
Definition: wasmtime.hh:1297
List()
Creates an empty list.
Definition: wasmtime.hh:1303
iterator begin() const
Returns the start of iteration.
Definition: wasmtime.hh:1327
const Ref * iterator
Definition: wasmtime.hh:1325
size_t size() const
Returns the size of this list.
Definition: wasmtime.hh:1335
iterator end() const
Returns the end of iteration.
Definition: wasmtime.hh:1331
List & operator=(List &&other) noexcept
Moves another list into this one.
Definition: wasmtime.hh:1318
List(List &&other) noexcept
Moves another list into this one.
Definition: wasmtime.hh:1309
Non-owning reference to an ExportType.
Definition: wasmtime.hh:1278
Ref(const wasm_exporttype_t *ptr)
Creates a new reference from the raw underlying C API representation.
Definition: wasmtime.hh:1287
std::string_view name()
Returns the name of this export.
Definition: wasmtime.hh:1290
Type information about a WebAssembly export.
Definition: wasmtime.hh:1272
Representation of a WebAssembly externref value.
Definition: wasmtime.hh:1992
ExternRef(wasmtime_externref_t val)
Creates a new ExternRef directly from its C-API representation.
Definition: wasmtime.hh:2003
const wasmtime_externref_t * raw() const
Definition: wasmtime.hh:2040
ExternRef clone(Store::Context cx)
Creates a new ExternRef which is separately rooted from this one.
Definition: wasmtime.hh:2020
void unroot(Store::Context cx)
Definition: wasmtime.hh:2033
ExternRef(Store::Context cx, T val)
Definition: wasmtime.hh:2010
std::any & data(Store::Context cx)
Returns the underlying host data associated with this ExternRef.
Definition: wasmtime.hh:2027
Generic type of a WebAssembly item.
Definition: wasmtime.hh:1342
static Ref from_import(ImportType::Ref ty)
Extract the type of the item imported by the provided type.
Definition: wasmtime.hh:1358
std::variant< FuncType::Ref, GlobalType::Ref, TableType::Ref, MemoryType::Ref > Ref
Non-owning reference to an item's type.
Definition: wasmtime.hh:1355
static Ref from_export(ExportType::Ref ty)
Extract the type of the item exported by the provided type.
Definition: wasmtime.hh:1365
Non-owning reference to a WebAssembly function frame as part of a Trace
Definition: wasmtime.hh:1394
size_t module_offset() const
Definition: wasmtime.hh:1406
std::optional< std::string_view > module_name() const
Definition: wasmtime.hh:1424
uint32_t func_index() const
Definition: wasmtime.hh:1400
std::optional< std::string_view > func_name() const
Definition: wasmtime.hh:1412
size_t func_offset() const
Definition: wasmtime.hh:1403
Representation of a WebAssembly function.
Definition: wasmtime.hh:2505
TrapResult< std::vector< Val > > call(Store::Context cx, const I &begin, const I &end) const
Invoke a WebAssembly function.
Definition: wasmtime.hh:2670
Func(wasmtime_func_t func)
Creates a new function from the raw underlying C API representation.
Definition: wasmtime.hh:2553
FuncType type(Store::Context cx) const
Returns the type of this function.
Definition: wasmtime.hh:2710
const wasmtime_func_t & raw_func() const
Returns the raw underlying C API function this is using.
Definition: wasmtime.hh:2745
static Func wrap(Store::Context cx, F f)
Creates a new host function from the provided callback f, inferring the WebAssembly function type fro...
Definition: wasmtime.hh:2637
Func(Store::Context cx, const FuncType &ty, F f)
Creates a new host-defined function.
Definition: wasmtime.hh:2591
Result< TypedFunc< Params, Results >, Trap > typed(Store::Context cx) const
Statically checks this function against the provided types.
Definition: wasmtime.hh:2734
Definition: wasmtime.hh:1121
Ref(const wasm_functype_t *ptr)
Creates a new reference from the underlying C API representation.
Definition: wasmtime.hh:1127
ValType::ListRef results() const
Returns the list of types this function type returns.
Definition: wasmtime.hh:1135
Ref(const FuncType &ty)
Creates a new reference to the given type.
Definition: wasmtime.hh:1129
ValType::ListRef params() const
Returns the list of types this function type takes as parameters.
Definition: wasmtime.hh:1132
Type information for a WebAssembly function.
Definition: wasmtime.hh:1108
static FuncType from_iters(P params, R results)
Creates a new function type from the given list of parameters and results.
Definition: wasmtime.hh:1168
FuncType(const FuncType &other)
Copies another type's information into this one.
Definition: wasmtime.hh:1153
Ref * operator->()
Returns the underlying Ref, a non-owning reference pointing to this instance.
Definition: wasmtime.hh:1188
Ref * operator*()
Returns the underlying Ref, a non-owning reference pointing to this instance.
Definition: wasmtime.hh:1191
FuncType & operator=(const FuncType &other)
Copies another type's information into this one.
Definition: wasmtime.hh:1156
FuncType(Ref other)
Copies a reference into a uniquely owned function type.
Definition: wasmtime.hh:1151
FuncType(FuncType &&other)=default
Moves type information from another type into this one.
FuncType(std::initializer_list< ValType > params, std::initializer_list< ValType > results)
Creates a new function type from the given list of parameters and results.
Definition: wasmtime.hh:1144
FuncType & operator=(FuncType &&other)=default
Moves type information from another type into this one.
A WebAssembly global.
Definition: wasmtime.hh:2852
Global(wasmtime_global_t global)
Creates as global from the raw underlying C API representation.
Definition: wasmtime.hh:2858
Result< std::monostate > set(Store::Context cx, const Val &val) const
Definition: wasmtime.hh:2890
Val get(Store::Context cx) const
Returns the current value of this global.
Definition: wasmtime.hh:2992
GlobalType type(Store::Context cx) const
Returns the type of this global.
Definition: wasmtime.hh:2880
static Result< Global > create(Store::Context cx, const GlobalType &ty, const Val &init)
Create a new WebAssembly global.
Definition: wasmtime.hh:2869
Definition: wasmtime.hh:1052
bool is_mutable() const
Returns whether or not this global type is mutable.
Definition: wasmtime.hh:1063
Ref(const GlobalType &ty)
Creates a new reference to the specified type.
Definition: wasmtime.hh:1060
Ref(const wasm_globaltype_t *ptr)
Creates a new reference from the raw underlying C API representation.
Definition: wasmtime.hh:1058
ValType::Ref content() const
Returns the type of value stored within this global type.
Definition: wasmtime.hh:1068
Type information about a WebAssembly global.
Definition: wasmtime.hh:1040
GlobalType & operator=(GlobalType &&other)=default
Moves the underlying type information from another global into this one.
Ref * operator->()
Returns the underlying Ref, a non-owning reference pointing to this instance.
Definition: wasmtime.hh:1099
GlobalType(Ref other)
Clones a reference into a uniquely owned global type.
Definition: wasmtime.hh:1082
GlobalType & operator=(const GlobalType &other)
Copies other type information into this one.
Definition: wasmtime.hh:1087
GlobalType(ValType ty, bool mut)
Creates a new global type from the specified value type and mutability.
Definition: wasmtime.hh:1077
Ref * operator*()
Returns the underlying Ref, a non-owning reference pointing to this instance.
Definition: wasmtime.hh:1102
GlobalType(const GlobalType &other)
Copies other type information into this one.
Definition: wasmtime.hh:1084
GlobalType(GlobalType &&other)=default
Moves the underlying type information from another global into this one.
An owned list of ImportType instances.
Definition: wasmtime.hh:1227
size_t size() const
Returns the size of this list.
Definition: wasmtime.hh:1265
List & operator=(List &&other) noexcept
Moves another list into this one.
Definition: wasmtime.hh:1248
const Ref * iterator
Definition: wasmtime.hh:1255
List(List &&other) noexcept
Moves another list into this one.
Definition: wasmtime.hh:1239
iterator begin() const
Returns the start of iteration.
Definition: wasmtime.hh:1257
iterator end() const
Returns the end of iteration.
Definition: wasmtime.hh:1261
List()
Creates an empty list.
Definition: wasmtime.hh:1233
Definition: wasmtime.hh:1201
Ref(const wasm_importtype_t *ptr)
Creates a new reference from the raw underlying C API representation.
Definition: wasmtime.hh:1211
std::string_view module()
Returns the module name associated with this import.
Definition: wasmtime.hh:1214
std::string_view name()
Returns the field name associated with this import.
Definition: wasmtime.hh:1220
Type information about a WebAssembly import.
Definition: wasmtime.hh:1197
A WebAssembly instance.
Definition: wasmtime.hh:3074
Instance(wasmtime_instance_t instance)
Creates a new instance from the raw underlying C API representation.
Definition: wasmtime.hh:3114
std::optional< std::pair< std::string_view, Extern > > get(Store::Context cx, size_t idx)
Load an instance's export by index.
Definition: wasmtime.hh:3174
std::optional< Extern > get(Store::Context cx, std::string_view name)
Load an instance's export by name.
Definition: wasmtime.hh:3159
static TrapResult< Instance > create(Store::Context cx, const Module &m, const std::vector< Extern > &imports)
Instantiates the module m with the provided imports
Definition: wasmtime.hh:3132
Helper class for linking modules together with name-based resolution.
Definition: wasmtime.hh:3206
Result< std::monostate > define(Store::Context cx, std::string_view module, std::string_view name, const Extern &item)
Defines the provided item into this linker with the given name.
Definition: wasmtime.hh:3226
Result< std::monostate > func_wrap(std::string_view module, std::string_view name, F &&f)
Definition: wasmtime.hh:3331
Linker(Engine &engine)
Creates a new linker which will instantiate in the given engine.
Definition: wasmtime.hh:3215
Result< std::monostate > func_new(std::string_view module, std::string_view name, const FuncType &ty, F &&f)
Definition: wasmtime.hh:3310
void allow_shadowing(bool allow)
Definition: wasmtime.hh:3221
Result< std::monostate > module(Store::Context cx, std::string_view name, const Module &m)
Definition: wasmtime.hh:3281
std::optional< Extern > get(Store::Context cx, std::string_view module, std::string_view name)
Definition: wasmtime.hh:3294
Result< Func > get_default(Store::Context cx, std::string_view name)
Definition: wasmtime.hh:3351
TrapResult< Instance > instantiate(Store::Context cx, const Module &m)
Definition: wasmtime.hh:3265
Result< std::monostate > define_instance(Store::Context cx, std::string_view name, Instance instance)
Definition: wasmtime.hh:3254
Result< std::monostate > define_wasi()
Definition: wasmtime.hh:3243
A WebAssembly linear memory.
Definition: wasmtime.hh:3009
Memory(wasmtime_memory_t memory)
Creates a new memory from the raw underlying C API representation.
Definition: wasmtime.hh:3015
static Result< Memory > create(Store::Context cx, const MemoryType &ty)
Creates a new host-defined memory with the type specified.
Definition: wasmtime.hh:3018
Span< uint8_t > data(Store::Context cx) const
Definition: wasmtime.hh:3042
MemoryType type(Store::Context cx) const
Returns the type of this memory.
Definition: wasmtime.hh:3028
uint64_t size(Store::Context cx) const
Returns the size, in WebAssembly pages, of this memory.
Definition: wasmtime.hh:3033
Result< uint64_t > grow(Store::Context cx, uint64_t delta) const
Definition: wasmtime.hh:3052
Non-owning reference to a MemoryType, must not be used after the original owner has been deleted.
Definition: wasmtime.hh:865
Ref(const wasm_memorytype_t *ptr)
Creates a reference from the raw C API representation.
Definition: wasmtime.hh:872
bool is_64() const
Returns whether or not this is a 64-bit memory type.
Definition: wasmtime.hh:891
bool is_shared() const
Returns whether or not this is a shared memory type.
Definition: wasmtime.hh:894
std::optional< uint64_t > max() const
Definition: wasmtime.hh:881
uint64_t min() const
Returns the minimum size, in WebAssembly pages, of this memory.
Definition: wasmtime.hh:877
Ref(const MemoryType &ty)
Creates a reference from an original MemoryType.
Definition: wasmtime.hh:874
Type information about a WebAssembly linear memory.
Definition: wasmtime.hh:853
Ref * operator*()
Returns the underlying Ref, a non-owning reference pointing to this instance.
Definition: wasmtime.hh:944
MemoryType & operator=(const MemoryType &other)
Copies the provided type into a new type.
Definition: wasmtime.hh:929
MemoryType(const MemoryType &other)
Copies the provided type into a new type.
Definition: wasmtime.hh:926
MemoryType(Ref other)
Definition: wasmtime.hh:924
static MemoryType New64(uint64_t min, uint64_t max)
Same as the MemoryType constructor, except creates a 64-bit memory.
Definition: wasmtime.hh:918
static MemoryType New64(uint64_t min)
Same as the MemoryType constructor, except creates a 64-bit memory.
Definition: wasmtime.hh:913
MemoryType & operator=(MemoryType &&other)=default
Moves the type information from another type into this one.
MemoryType(MemoryType &&other)=default
Moves the type information from another type into this one.
MemoryType(uint32_t min, uint32_t max)
Definition: wasmtime.hh:909
Ref * operator->()
Returns the underlying Ref, a non-owning reference pointing to this instance.
Definition: wasmtime.hh:941
MemoryType(uint32_t min)
Definition: wasmtime.hh:905
Representation of a compiled WebAssembly module.
Definition: wasmtime.hh:1559
Result< std::vector< uint8_t > > serialize() const
Serializes this module to a list of bytes.
Definition: wasmtime.hh:1701
Module & operator=(Module &&other)=default
Moves resources from another module into this one.
ImportType::List imports() const
Returns the list of types imported by this module.
Definition: wasmtime.hh:1682
Module(Module &&other)=default
Moves resources from another module into this one.
static Result< Module > deserialize_file(Engine &engine, const std::string &path)
Deserializes a module from an on-disk file.
Definition: wasmtime.hh:1670
static Result< Module > compile(Engine &engine, std::string_view wat)
Compiles a module from the WebAssembly text format.
Definition: wasmtime.hh:1592
ExportType::List exports() const
Returns the list of types exported by this module.
Definition: wasmtime.hh:1689
static Result< Module > compile(Engine &engine, Span< uint8_t > wasm)
Compiles a module from the WebAssembly binary format.
Definition: wasmtime.hh:1611
static Result< std::monostate > validate(Engine &engine, Span< uint8_t > wasm)
Validates the provided WebAssembly binary without compiling it.
Definition: wasmtime.hh:1627
static Result< Module > deserialize(Engine &engine, Span< uint8_t > wasm)
Deserializes a previous list of bytes created with serialize.
Definition: wasmtime.hh:1648
Module(const Module &other)
Copies another module into this one.
Definition: wasmtime.hh:1574
Module & operator=(const Module &other)
Copies another module into this one.
Definition: wasmtime.hh:1576
Pool allocation configuration for Wasmtime.
Definition: wasmtime.hh:266
void max_tables_per_module(uint32_t tables)
The maximum number of defined tables for a core module (default is 1).
Definition: wasmtime.hh:414
void total_tables(uint32_t count)
The maximum number of concurrent tables supported (default is 1000).
Definition: wasmtime.hh:380
void max_component_instance_size(size_t size)
The maximum size, in bytes, allocated for a component instance’s VMComponentContext metadata.
Definition: wasmtime.hh:336
void max_memories_per_component(uint32_t count)
The maximum number of Wasm linear memories that a single component may transitively contain (default ...
Definition: wasmtime.hh:354
void decommit_batch_size(size_t batch_size)
The target number of decommits to do per batch.
Definition: wasmtime.hh:292
void total_gc_heaps(uint32_t count)
The maximum number of concurrent GC heaps supported (default is 1000).
Definition: wasmtime.hh:448
void max_core_instances_per_component(uint32_t count)
The maximum number of core instances a single component may contain (default is unlimited).
Definition: wasmtime.hh:345
void async_stack_keep_resident(size_t size)
How much memory, in bytes, to keep resident for async stacks allocated with the pooling allocator.
Definition: wasmtime.hh:301
void max_core_instance_size(size_t size)
The maximum size, in bytes, allocated for a core instance’s VMContext metadata.
Definition: wasmtime.hh:405
void total_core_instances(uint32_t count)
The maximum number of concurrent core instances supported (default is 1000).
Definition: wasmtime.hh:396
void table_elements(size_t elements)
The maximum table elements for any table defined in a module (default is 20000).
Definition: wasmtime.hh:423
void max_unused_warm_slots(uint32_t max)
Configures the maximum number of “unused warm slots” to retain in the pooling allocator.
Definition: wasmtime.hh:284
void max_memory_size(size_t bytes)
The maximum byte size that any WebAssembly linear memory may grow to.
Definition: wasmtime.hh:440
void total_memories(uint32_t count)
The maximum number of concurrent Wasm linear memories supported (default is 1000).
Definition: wasmtime.hh:372
void linear_memory_keep_resident(size_t size)
How much memory, in bytes, to keep resident for each linear memory after deallocation.
Definition: wasmtime.hh:310
void total_component_instances(uint32_t count)
The maximum number of concurrent component instances supported (default is 1000).
Definition: wasmtime.hh:327
void max_memories_per_module(uint32_t memories)
The maximum number of defined linear memories for a module (default is 1).
Definition: wasmtime.hh:431
void table_keep_resident(size_t size)
How much memory, in bytes, to keep resident for each table after deallocation.
Definition: wasmtime.hh:319
void total_stacks(uint32_t count)
The maximum number of execution stacks allowed for asynchronous execution, when enabled (default is 1...
Definition: wasmtime.hh:388
void max_tables_per_component(uint32_t count)
The maximum number of tables that a single component may transitively contain (default is unlimited).
Definition: wasmtime.hh:363
Fallible result type used for Wasmtime.
Definition: wasmtime.hh:194
const T && ok() const
Returns the success, if present, aborts if this is an error.
Definition: wasmtime.hh:215
E && err()
Returns the error, if present, aborts if this is not an error.
Definition: wasmtime.hh:208
T unwrap()
Returns the success, if present, aborts if this is an error.
Definition: wasmtime.hh:218
T && ok()
Returns the success, if present, aborts if this is an error.
Definition: wasmtime.hh:213
Result(T t)
Creates a Result from its successful value.
Definition: wasmtime.hh:199
const E && err() const
Returns the error, if present, aborts if this is not an error.
Definition: wasmtime.hh:210
Result(E e)
Creates a Result from an error value.
Definition: wasmtime.hh:201
Span class used when c++20 is not available.
Definition: wasmtime.hh:81
std::size_t size_bytes() const
Returns size in bytes.
Definition: wasmtime.hh:126
std::size_t size() const
Returns number of data that referred by Span class.
Definition: wasmtime.hh:115
iterator end() const
Returns end iterator.
Definition: wasmtime.hh:121
T * iterator
Type used to iterate over this span (a raw pointer)
Definition: wasmtime.hh:87
T * data() const
Returns pointer to data.
Definition: wasmtime.hh:112
Span(T *t, std::size_t n)
Constructor of Span class.
Definition: wasmtime.hh:90
iterator begin() const
Returns begin iterator.
Definition: wasmtime.hh:118
T & operator[](ptrdiff_t idx) const
Returns item by index.
Definition: wasmtime.hh:107
Span(C &range)
Constructor of Span class for containers.
Definition: wasmtime.hh:104
An interior pointer into a Store.
Definition: wasmtime.hh:1847
void gc()
Definition: wasmtime.hh:1872
void set_epoch_deadline(uint64_t ticks_beyond_current)
Definition: wasmtime.hh:1931
wasmtime_context_t * raw_context()
Returns the raw context pointer for the C API.
Definition: wasmtime.hh:1936
Result< std::monostate > set_wasi(WasiConfig config)
Definition: wasmtime.hh:1917
Result< std::monostate > set_fuel(uint64_t fuel)
Definition: wasmtime.hh:1880
Context(Store &store)
Creates a context referencing the provided Store.
Definition: wasmtime.hh:1862
void set_data(std::any data) const
Set user specified data associated with this store.
Definition: wasmtime.hh:1901
Result< uint64_t > get_fuel() const
Definition: wasmtime.hh:1891
std::any & get_data() const
Get user specified data associated with this store.
Definition: wasmtime.hh:1908
Context(Store *store)
Creates a context referencing the provided Store.
Definition: wasmtime.hh:1864
Owner of all WebAssembly objects.
Definition: wasmtime.hh:1820
Store(Engine &engine)
Creates a new Store within the provided Engine.
Definition: wasmtime.hh:1833
Context context()
Explicit function to acquire a Context from this store.
Definition: wasmtime.hh:1976
void limiter(int64_t memory_size, int64_t table_elements, int64_t instances, int64_t tables, int64_t memories)
Provides limits for a store. Used by hosts to limit resource consumption of instances....
Definition: wasmtime.hh:1969
A WebAssembly table.
Definition: wasmtime.hh:2911
Table(wasmtime_table_t table)
Creates a new table from the raw underlying C API representation.
Definition: wasmtime.hh:2917
Result< std::monostate > set(Store::Context cx, uint64_t idx, const Val &val) const
Definition: wasmtime.hh:2962
static Result< Table > create(Store::Context cx, const TableType &ty, const Val &init)
Creates a new host-defined table.
Definition: wasmtime.hh:2928
Result< uint64_t > grow(Store::Context cx, uint64_t delta, const Val &init) const
Definition: wasmtime.hh:2979
uint64_t size(Store::Context cx) const
Returns the size, in elements, that the table currently has.
Definition: wasmtime.hh:2944
TableType type(Store::Context cx) const
Returns the type of this table.
Definition: wasmtime.hh:2939
std::optional< Val > get(Store::Context cx, uint64_t idx) const
Definition: wasmtime.hh:2951
Definition: wasmtime.hh:962
Ref(const wasm_tabletype_t *ptr)
Creates a reference from the raw underlying C API representation.
Definition: wasmtime.hh:969
Ref(const TableType &ty)
Creates a reference to the provided TableType.
Definition: wasmtime.hh:971
std::optional< uint32_t > max() const
Returns the maximum size of this table type, in elements, if present.
Definition: wasmtime.hh:977
ValType::Ref element() const
Returns the type of value that is stored in this table.
Definition: wasmtime.hh:986
uint32_t min() const
Returns the minimum size of this table type, in elements.
Definition: wasmtime.hh:974
Type information about a WebAssembly table.
Definition: wasmtime.hh:950
TableType(const TableType &other)
Copies another table type into this one.
Definition: wasmtime.hh:1016
TableType & operator=(TableType &&other)=default
Moves the table type resources from another type to this one.
TableType(ValType ty, uint32_t min, uint32_t max)
Definition: wasmtime.hh:1005
TableType(Ref other)
Clones the given reference into a new table type.
Definition: wasmtime.hh:1014
Ref * operator->()
Returns the underlying Ref, a non-owning reference pointing to this instance.
Definition: wasmtime.hh:1031
TableType(ValType ty, uint32_t min)
Definition: wasmtime.hh:996
TableType & operator=(const TableType &other)
Copies another table type into this one.
Definition: wasmtime.hh:1019
TableType(TableType &&other)=default
Moves the table type resources from another type to this one.
Ref * operator*()
Returns the underlying Ref, a non-owning reference pointing to this instance.
Definition: wasmtime.hh:1034
An owned vector of FrameRef instances representing the WebAssembly call-stack on a trap.
Definition: wasmtime.hh:1440
iterator begin() const
Returns the start of iteration.
Definition: wasmtime.hh:1460
const FrameRef * iterator
Iterator used to iterate over this trace.
Definition: wasmtime.hh:1457
size_t size() const
Returns the size of this trace, or how many frames it contains.
Definition: wasmtime.hh:1468
iterator end() const
Returns the end of iteration.
Definition: wasmtime.hh:1464
Information about a WebAssembly trap.
Definition: wasmtime.hh:1487
Trap(std::string_view msg)
Creates a new host-defined trap with the specified message.
Definition: wasmtime.hh:1503
std::string message() const
Returns the descriptive message associated with this trap.
Definition: wasmtime.hh:1507
Trace trace() const
Definition: wasmtime.hh:1518
A version of a WebAssembly Func where the type signature of the function is statically known.
Definition: wasmtime.hh:2752
const Func & func() const
Returns the underlying un-typed Func for this function.
Definition: wasmtime.hh:2789
TrapResult< Results > call(Store::Context cx, Params params) const
Calls this function with the provided parameters.
Definition: wasmtime.hh:2768
Representation of a generic WebAssembly value.
Definition: wasmtime.hh:2077
Val(int32_t i32)
Creates a new i32 WebAssembly value.
Definition: wasmtime.hh:2092
Val(float f32)
Creates a new f32 WebAssembly value.
Definition: wasmtime.hh:2102
float f32() const
Definition: wasmtime.hh:2174
int32_t i32() const
Definition: wasmtime.hh:2156
double f64() const
Definition: wasmtime.hh:2183
int64_t i64() const
Definition: wasmtime.hh:2165
void unroot(Store::Context cx)
Unroots any GC references this Val points to within the cx provided.
Definition: wasmtime.hh:2224
V128 v128() const
Definition: wasmtime.hh:2192
ValKind kind() const
Returns the kind of value that this value has.
Definition: wasmtime.hh:2134
Val(int64_t i64)
Creates a new i64 WebAssembly value.
Definition: wasmtime.hh:2097
std::optional< ExternRef > externref(Store::Context cx) const
Definition: wasmtime.hh:2204
Val(const V128 &v128)
Creates a new v128 WebAssembly value.
Definition: wasmtime.hh:2112
Val(std::optional< ExternRef > ptr)
Creates a new externref value.
Definition: wasmtime.hh:2121
Val(double f64)
Creates a new f64 WebAssembly value.
Definition: wasmtime.hh:2107
std::optional< Func > funcref() const
Definition: wasmtime.hh:2804
Non-owning reference to a list of ValType instances. Must not be used after the original owner is del...
Definition: wasmtime.hh:795
ListRef(const wasm_valtype_vec_t *list)
Creates a list from the raw underlying C API.
Definition: wasmtime.hh:800
size_t size() const
Returns how many types are in this list.
Definition: wasmtime.hh:816
iterator begin() const
Pointer to the beginning of iteration.
Definition: wasmtime.hh:806
iterator end() const
Pointer to the end of iteration.
Definition: wasmtime.hh:811
const Ref * iterator
This list iterates over a list of ValType::Ref instances.
Definition: wasmtime.hh:803
Non-owning reference to a ValType, must not be used after the original ValType is deleted.
Definition: wasmtime.hh:769
Ref(const ValType &ty)
Copy constructor.
Definition: wasmtime.hh:778
Ref(const wasm_valtype_t *ptr)
Instantiates from the raw C API representation.
Definition: wasmtime.hh:776
ValKind kind() const
Returns the corresponding "kind" for this type.
Definition: wasmtime.hh:781
Type information about a WebAssembly value.
Definition: wasmtime.hh:743
ValType & operator=(ValType &&other)=default
Moves the memory owned by another value type into this one.
ValType & operator=(const ValType &other)
Copies the contents of another type into this one.
Definition: wasmtime.hh:831
Ref * operator->()
Returns the underlying Ref, a non-owning reference pointing to this instance.
Definition: wasmtime.hh:844
Ref * operator*()
Returns the underlying Ref, a non-owning reference pointing to this instance.
Definition: wasmtime.hh:847
ValType(ValType &&other)=default
Moves the memory owned by another value type into this one.
ValType(const ValType &other)
Copies one type to a new one.
Definition: wasmtime.hh:829
ValType(ValKind kind)
Creates a new type from its kind.
Definition: wasmtime.hh:825
ValType(Ref other)
Copies a Ref to a new owned value.
Definition: wasmtime.hh:827
Configuration for an instance of WASI.
Definition: wasmtime.hh:1721
void inherit_stderr()
Definition: wasmtime.hh:1794
void inherit_env()
Definition: wasmtime.hh:1764
bool stdin_file(const std::string &path)
Definition: wasmtime.hh:1768
void inherit_stdout()
Definition: wasmtime.hh:1784
bool stderr_file(const std::string &path)
Definition: wasmtime.hh:1788
bool preopen_dir(const std::string &path, const std::string &guest_path, size_t dir_perms, size_t file_perms)
Opens path to be opened as guest_path in the WASI pseudo-filesystem.
Definition: wasmtime.hh:1797
bool stdout_file(const std::string &path)
Definition: wasmtime.hh:1778
void env(const std::vector< std::pair< std::string, std::string > > &env)
Definition: wasmtime.hh:1751
WasiConfig()
Creates a new configuration object with default settings.
Definition: wasmtime.hh:1732
void argv(const std::vector< std::string > &args)
Configures the argv explicitly with the given string array.
Definition: wasmtime.hh:1735
void inherit_argv()
Configures the argv for wasm to be inherited from this process itself.
Definition: wasmtime.hh:1746
void inherit_stdin()
Definition: wasmtime.hh:1774
Check whether a type is Span
Definition: wasmtime.hh:76
Structure used to represent either a Trap or an Error.
Definition: wasmtime.hh:1526
TrapError(Trap t)
Creates a new TrapError from a Trap
Definition: wasmtime.hh:1531
TrapError(Error e)
Creates a new TrapError from an Error
Definition: wasmtime.hh:1533
std::string message() const
Dispatches internally to return the message associated with this error.
Definition: wasmtime.hh:1536
std::variant< Trap, Error > data
Storage for what this trap represents.
Definition: wasmtime.hh:1528
Container for the v128 WebAssembly type.
Definition: wasmtime.hh:2054
V128()
Creates a new zero-value v128.
Definition: wasmtime.hh:2059
wasmtime_v128 v128
The little-endian bytes of the v128 value.
Definition: wasmtime.hh:2056
V128(const wasmtime_v128 &v)
Creates a new V128 from its C API representation.
Definition: wasmtime.hh:2062
Definition: wasmtime.hh:2430
Definition: wasmtime.hh:2385
Definition: wasmtime.hh:2261
Definition: wasmtime.hh:2326
Result< std::vector< uint8_t > > wat2wasm(std::string_view wat)
Converts the WebAssembly text format into the WebAssembly binary format.
Definition: wasmtime.hh:680
Strategy
Strategies passed to Config::strategy
Definition: wasmtime.hh:233
@ Auto
Automatically selects the compilation strategy.
@ Cranelift
Requires Cranelift to be used for compilation.
constexpr size_t dynamic_extent
Means number of elements determined at runtime.
Definition: wasmtime.hh:65
ValKind
Different kinds of types accepted by Wasmtime.
Definition: wasmtime.hh:695
@ F64
WebAssembly's f64 type.
@ F32
WebAssembly's f32 type.
@ FuncRef
WebAssembly's funcref type from the reference types.
@ I32
WebAssembly's i32 type.
@ I64
WebAssembly's i64 type.
OptLevel
Values passed to Config::cranelift_opt_level
Definition: wasmtime.hh:241
@ Speed
Optimize for speed.
@ None
No extra optimizations performed.
@ SpeedAndSize
Optimize for speed and generated code size.
ProfilingStrategy
Values passed to Config::profiler
Definition: wasmtime.hh:251
@ Vtune
Profiling hooks via VTune.
@ Jitdump
Profiling hooks via perf's jitdump.
std::ostream & operator<<(std::ostream &os, const Error &e)
Used to print an error.
Definition: wasmtime.hh:182
#define NATIVE_WASM_TYPE(native, valkind, field)
Definition: wasmtime.hh:2266
std::variant< Func, Global, Memory, Table > Extern
Representation of an external WebAssembly item.
Definition: wasmtime.hh:2051
#define WASMTIME_FOR_EACH_VAL_KIND(X)
Definition: wasmtime.hh:714