1use crate::component::{
19 CanonicalAbiInfo, ComponentTypesBuilder, FixedEncoding as FE, FlatType, InterfaceType,
20 StringEncoding, Transcode, TypeComponentLocalErrorContextTableIndex, TypeEnumIndex,
21 TypeFlagsIndex, TypeFutureTableIndex, TypeListIndex, TypeOptionIndex, TypeRecordIndex,
22 TypeResourceTableIndex, TypeResultIndex, TypeStreamTableIndex, TypeTupleIndex,
23 TypeVariantIndex, VariantInfo, FLAG_MAY_ENTER, FLAG_MAY_LEAVE, MAX_FLAT_PARAMS,
24};
25use crate::fact::signature::Signature;
26use crate::fact::transcode::Transcoder;
27use crate::fact::traps::Trap;
28use crate::fact::{
29 AdapterData, Body, Context, Function, FunctionId, Helper, HelperLocation, HelperType, Module,
30 Options,
31};
32use crate::prelude::*;
33use crate::{FuncIndex, GlobalIndex};
34use std::collections::HashMap;
35use std::mem;
36use std::ops::Range;
37use wasm_encoder::{BlockType, Encode, Instruction, Instruction::*, MemArg, ValType};
38use wasmtime_component_util::{DiscriminantSize, FlagsSize};
39
40const MAX_STRING_BYTE_LENGTH: u32 = 1 << 31;
41const UTF16_TAG: u32 = 1 << 31;
42
43const INITIAL_FUEL: usize = 1_000;
46
47struct Compiler<'a, 'b> {
48 types: &'a ComponentTypesBuilder,
49 module: &'b mut Module<'a>,
50 result: FunctionId,
51
52 code: Vec<u8>,
54
55 nlocals: u32,
57
58 free_locals: HashMap<ValType, Vec<u32>>,
60
61 traps: Vec<(usize, Trap)>,
65
66 fuel: usize,
75
76 emit_resource_call: bool,
81}
82
83pub(super) fn compile(module: &mut Module<'_>, adapter: &AdapterData) {
84 fn compiler<'a, 'b>(
85 module: &'b mut Module<'a>,
86 adapter: &AdapterData,
87 ) -> (Compiler<'a, 'b>, Signature, Signature) {
88 let lower_sig = module.types.signature(&adapter.lower, Context::Lower);
89 let lift_sig = module.types.signature(&adapter.lift, Context::Lift);
90 let ty = module
91 .core_types
92 .function(&lower_sig.params, &lower_sig.results);
93 let result = module
94 .funcs
95 .push(Function::new(Some(adapter.name.clone()), ty));
96
97 let emit_resource_call = module.types.contains_borrow_resource(&adapter.lower);
102 assert_eq!(
103 emit_resource_call,
104 module.types.contains_borrow_resource(&adapter.lift)
105 );
106
107 (
108 Compiler::new(
109 module,
110 result,
111 lower_sig.params.len() as u32,
112 emit_resource_call,
113 ),
114 lower_sig,
115 lift_sig,
116 )
117 }
118
119 let async_start_adapter = |module: &mut Module| {
125 let sig = module
126 .types
127 .async_start_signature(&adapter.lower, &adapter.lift);
128 let ty = module.core_types.function(&sig.params, &sig.results);
129 let result = module.funcs.push(Function::new(
130 Some(format!("[async-start]{}", adapter.name)),
131 ty,
132 ));
133
134 Compiler::new(module, result, sig.params.len() as u32, false)
135 .compile_async_start_adapter(adapter, &sig);
136
137 result
138 };
139
140 let async_return_adapter = |module: &mut Module| {
149 let sig = module
150 .types
151 .async_return_signature(&adapter.lower, &adapter.lift);
152 let ty = module.core_types.function(&sig.params, &sig.results);
153 let result = module.funcs.push(Function::new(
154 Some(format!("[async-return]{}", adapter.name)),
155 ty,
156 ));
157
158 Compiler::new(module, result, sig.params.len() as u32, false)
159 .compile_async_return_adapter(adapter, &sig);
160
161 result
162 };
163
164 match (adapter.lower.options.async_, adapter.lift.options.async_) {
165 (false, false) => {
166 let (compiler, lower_sig, lift_sig) = compiler(module, adapter);
169 compiler.compile_sync_to_sync_adapter(adapter, &lower_sig, &lift_sig)
170 }
171 (true, true) => {
172 let start = async_start_adapter(module);
188 let return_ = async_return_adapter(module);
189 let (compiler, _, lift_sig) = compiler(module, adapter);
190 compiler.compile_async_to_async_adapter(
191 adapter,
192 start,
193 return_,
194 i32::try_from(lift_sig.params.len()).unwrap(),
195 );
196 }
197 (false, true) => {
198 let start = async_start_adapter(module);
211 let return_ = async_return_adapter(module);
212 let (compiler, lower_sig, lift_sig) = compiler(module, adapter);
213 compiler.compile_sync_to_async_adapter(
214 adapter,
215 start,
216 return_,
217 i32::try_from(lift_sig.params.len()).unwrap(),
218 &lower_sig,
219 );
220 }
221 (true, false) => {
222 let lift_sig = module.types.signature(&adapter.lift, Context::Lift);
242 let start = async_start_adapter(module);
243 let return_ = async_return_adapter(module);
244 let (compiler, ..) = compiler(module, adapter);
245 compiler.compile_async_to_sync_adapter(
246 adapter,
247 start,
248 return_,
249 i32::try_from(lift_sig.params.len()).unwrap(),
250 i32::try_from(lift_sig.results.len()).unwrap(),
251 );
252 }
253 }
254}
255
256pub(super) fn compile_helper(module: &mut Module<'_>, result: FunctionId, helper: Helper) {
263 let mut nlocals = 0;
264 let src_flat;
265 let src = match helper.src.loc {
266 HelperLocation::Stack => {
271 src_flat = module
272 .types
273 .flatten_types(&helper.src.opts, usize::MAX, [helper.src.ty])
274 .unwrap()
275 .iter()
276 .enumerate()
277 .map(|(i, ty)| (i as u32, *ty))
278 .collect::<Vec<_>>();
279 nlocals += src_flat.len() as u32;
280 Source::Stack(Stack {
281 locals: &src_flat,
282 opts: &helper.src.opts,
283 })
284 }
285 HelperLocation::Memory => {
288 nlocals += 1;
289 Source::Memory(Memory {
290 opts: &helper.src.opts,
291 addr: TempLocal::new(0, helper.src.opts.ptr()),
292 offset: 0,
293 })
294 }
295 };
296 let dst_flat;
297 let dst = match helper.dst.loc {
298 HelperLocation::Stack => {
301 dst_flat = module
302 .types
303 .flatten_types(&helper.dst.opts, usize::MAX, [helper.dst.ty])
304 .unwrap();
305 Destination::Stack(&dst_flat, &helper.dst.opts)
306 }
307 HelperLocation::Memory => {
310 nlocals += 1;
311 Destination::Memory(Memory {
312 opts: &helper.dst.opts,
313 addr: TempLocal::new(nlocals - 1, helper.dst.opts.ptr()),
314 offset: 0,
315 })
316 }
317 };
318 let mut compiler = Compiler {
319 types: module.types,
320 module,
321 code: Vec::new(),
322 nlocals,
323 free_locals: HashMap::new(),
324 traps: Vec::new(),
325 result,
326 fuel: INITIAL_FUEL,
327 emit_resource_call: false,
330 };
331 compiler.translate(&helper.src.ty, &src, &helper.dst.ty, &dst);
332 compiler.finish();
333}
334
335enum Source<'a> {
338 Stack(Stack<'a>),
344
345 Memory(Memory<'a>),
348}
349
350enum Destination<'a> {
352 Stack(&'a [ValType], &'a Options),
358
359 Memory(Memory<'a>),
361}
362
363struct Stack<'a> {
364 locals: &'a [(u32, ValType)],
370 opts: &'a Options,
372}
373
374struct Memory<'a> {
376 opts: &'a Options,
378 addr: TempLocal,
381 offset: u32,
384}
385
386impl<'a, 'b> Compiler<'a, 'b> {
387 fn new(
388 module: &'b mut Module<'a>,
389 result: FunctionId,
390 nlocals: u32,
391 emit_resource_call: bool,
392 ) -> Self {
393 Self {
394 types: module.types,
395 module,
396 result,
397 code: Vec::new(),
398 nlocals,
399 free_locals: HashMap::new(),
400 traps: Vec::new(),
401 fuel: INITIAL_FUEL,
402 emit_resource_call,
403 }
404 }
405
406 fn compile_async_to_async_adapter(
416 mut self,
417 adapter: &AdapterData,
418 start: FunctionId,
419 return_: FunctionId,
420 param_count: i32,
421 ) {
422 let enter = self.module.import_async_enter_call();
423 let exit = self
424 .module
425 .import_async_exit_call(adapter.lift.options.callback, None);
426
427 self.flush_code();
428 self.module.funcs[self.result]
429 .body
430 .push(Body::RefFunc(start));
431 self.module.funcs[self.result]
432 .body
433 .push(Body::RefFunc(return_));
434 self.instruction(I32Const(
435 i32::try_from(adapter.lower.instance.as_u32()).unwrap(),
436 ));
437 self.instruction(I32Const(
438 i32::try_from(self.types[adapter.lift.ty].results.as_u32()).unwrap(),
439 ));
440 self.instruction(LocalGet(0));
445 self.instruction(LocalGet(1));
446 self.instruction(Call(enter.as_u32()));
447
448 self.module.exports.push((
457 adapter.callee.as_u32(),
458 format!("[adapter-callee]{}", adapter.name),
459 ));
460
461 self.instruction(I32Const(
462 i32::try_from(adapter.lower.instance.as_u32()).unwrap(),
463 ));
464 self.instruction(RefFunc(adapter.callee.as_u32()));
465 self.instruction(I32Const(
466 i32::try_from(adapter.lift.instance.as_u32()).unwrap(),
467 ));
468 self.instruction(I32Const(param_count));
469 self.instruction(I32Const(1));
473 self.instruction(I32Const(super::EXIT_FLAG_ASYNC_CALLEE));
474 self.instruction(Call(exit.as_u32()));
475
476 self.finish()
477 }
478
479 fn compile_sync_to_async_adapter(
489 mut self,
490 adapter: &AdapterData,
491 start: FunctionId,
492 return_: FunctionId,
493 lift_param_count: i32,
494 lower_sig: &Signature,
495 ) {
496 let enter = self
497 .module
498 .import_sync_enter_call(&adapter.name, &lower_sig.params);
499 let exit = self.module.import_sync_exit_call(
500 &adapter.name,
501 adapter.lift.options.callback,
502 &lower_sig.results,
503 );
504
505 self.flush_code();
506 self.module.funcs[self.result]
507 .body
508 .push(Body::RefFunc(start));
509 self.module.funcs[self.result]
510 .body
511 .push(Body::RefFunc(return_));
512 self.instruction(I32Const(
513 i32::try_from(adapter.lower.instance.as_u32()).unwrap(),
514 ));
515 self.instruction(I32Const(
516 i32::try_from(self.types[adapter.lift.ty].results.as_u32()).unwrap(),
517 ));
518 self.instruction(I32Const(
519 i32::try_from(
520 self.types
521 .flatten_types(
522 &adapter.lower.options,
523 usize::MAX,
524 self.types[self.types[adapter.lower.ty].results]
525 .types
526 .iter()
527 .copied(),
528 )
529 .map(|v| v.len())
530 .unwrap_or(usize::try_from(i32::MAX).unwrap()),
531 )
532 .unwrap(),
533 ));
534
535 for index in 0..lower_sig.params.len() {
536 self.instruction(LocalGet(u32::try_from(index).unwrap()));
537 }
538
539 self.instruction(Call(enter.as_u32()));
540
541 self.module.exports.push((
550 adapter.callee.as_u32(),
551 format!("[adapter-callee]{}", adapter.name),
552 ));
553
554 self.instruction(I32Const(
555 i32::try_from(adapter.lower.instance.as_u32()).unwrap(),
556 ));
557 self.instruction(RefFunc(adapter.callee.as_u32()));
558 self.instruction(I32Const(
559 i32::try_from(adapter.lift.instance.as_u32()).unwrap(),
560 ));
561 self.instruction(I32Const(lift_param_count));
562 self.instruction(Call(exit.as_u32()));
563
564 self.finish()
565 }
566
567 fn compile_async_to_sync_adapter(
577 mut self,
578 adapter: &AdapterData,
579 start: FunctionId,
580 return_: FunctionId,
581 param_count: i32,
582 result_count: i32,
583 ) {
584 let enter = self.module.import_async_enter_call();
585 let exit = self
586 .module
587 .import_async_exit_call(None, adapter.lift.post_return);
588
589 self.flush_code();
590 self.module.funcs[self.result]
591 .body
592 .push(Body::RefFunc(start));
593 self.module.funcs[self.result]
594 .body
595 .push(Body::RefFunc(return_));
596 self.instruction(I32Const(
597 i32::try_from(adapter.lower.instance.as_u32()).unwrap(),
598 ));
599 self.instruction(I32Const(
600 i32::try_from(self.types[adapter.lift.ty].results.as_u32()).unwrap(),
601 ));
602 self.instruction(LocalGet(0));
603 self.instruction(LocalGet(1));
604 self.instruction(Call(enter.as_u32()));
605
606 self.module.exports.push((
610 adapter.callee.as_u32(),
611 format!("[adapter-callee]{}", adapter.name),
612 ));
613
614 self.instruction(I32Const(
615 i32::try_from(adapter.lower.instance.as_u32()).unwrap(),
616 ));
617 self.instruction(RefFunc(adapter.callee.as_u32()));
618 self.instruction(I32Const(
619 i32::try_from(adapter.lift.instance.as_u32()).unwrap(),
620 ));
621 self.instruction(I32Const(param_count));
622 self.instruction(I32Const(result_count));
623 self.instruction(I32Const(0));
624 self.instruction(Call(exit.as_u32()));
625
626 self.finish()
627 }
628
629 fn compile_async_start_adapter(mut self, adapter: &AdapterData, sig: &Signature) {
635 let param_locals = sig
636 .params
637 .iter()
638 .enumerate()
639 .map(|(i, ty)| (i as u32, *ty))
640 .collect::<Vec<_>>();
641
642 self.set_flag(adapter.lift.flags, FLAG_MAY_LEAVE, false);
643 self.translate_params(adapter, ¶m_locals);
644 self.set_flag(adapter.lift.flags, FLAG_MAY_LEAVE, true);
645
646 self.finish();
647 }
648
649 fn compile_async_return_adapter(mut self, adapter: &AdapterData, sig: &Signature) {
658 let param_locals = sig
659 .params
660 .iter()
661 .enumerate()
662 .map(|(i, ty)| (i as u32, *ty))
663 .collect::<Vec<_>>();
664
665 self.set_flag(adapter.lower.flags, FLAG_MAY_LEAVE, false);
666 self.translate_results(adapter, ¶m_locals, ¶m_locals);
677 self.set_flag(adapter.lower.flags, FLAG_MAY_LEAVE, true);
678
679 self.finish()
680 }
681
682 fn compile_sync_to_sync_adapter(
689 mut self,
690 adapter: &AdapterData,
691 lower_sig: &Signature,
692 lift_sig: &Signature,
693 ) {
694 self.trap_if_not_flag(adapter.lower.flags, FLAG_MAY_LEAVE, Trap::CannotLeave);
700 if adapter.called_as_export {
701 self.trap_if_not_flag(adapter.lift.flags, FLAG_MAY_ENTER, Trap::CannotEnter);
702 self.set_flag(adapter.lift.flags, FLAG_MAY_ENTER, false);
703 } else if self.module.debug {
704 self.assert_not_flag(
705 adapter.lift.flags,
706 FLAG_MAY_ENTER,
707 "may_enter should be unset",
708 );
709 }
710
711 if self.emit_resource_call {
712 let enter = self.module.import_resource_enter_call();
713 self.instruction(Call(enter.as_u32()));
714 }
715
716 self.set_flag(adapter.lift.flags, FLAG_MAY_LEAVE, false);
729 let param_locals = lower_sig
730 .params
731 .iter()
732 .enumerate()
733 .map(|(i, ty)| (i as u32, *ty))
734 .collect::<Vec<_>>();
735 self.translate_params(adapter, ¶m_locals);
736 self.set_flag(adapter.lift.flags, FLAG_MAY_LEAVE, true);
737
738 self.instruction(Call(adapter.callee.as_u32()));
742 let mut result_locals = Vec::with_capacity(lift_sig.results.len());
743 let mut temps = Vec::new();
744 for ty in lift_sig.results.iter().rev() {
745 let local = self.local_set_new_tmp(*ty);
746 result_locals.push((local.idx, *ty));
747 temps.push(local);
748 }
749 result_locals.reverse();
750
751 self.set_flag(adapter.lower.flags, FLAG_MAY_LEAVE, false);
760 self.translate_results(adapter, ¶m_locals, &result_locals);
761 self.set_flag(adapter.lower.flags, FLAG_MAY_LEAVE, true);
762
763 if let Some(func) = adapter.lift.post_return {
766 for (result, _) in result_locals.iter() {
767 self.instruction(LocalGet(*result));
768 }
769 self.instruction(Call(func.as_u32()));
770 }
771 if adapter.called_as_export {
772 self.set_flag(adapter.lift.flags, FLAG_MAY_ENTER, true);
773 }
774
775 for tmp in temps {
776 self.free_temp_local(tmp);
777 }
778
779 if self.emit_resource_call {
780 let exit = self.module.import_resource_exit_call();
781 self.instruction(Call(exit.as_u32()));
782 }
783
784 self.finish()
785 }
786
787 fn translate_params(&mut self, adapter: &AdapterData, param_locals: &[(u32, ValType)]) {
788 let src_tys = self.types[adapter.lower.ty].params;
789 let src_tys = self.types[src_tys]
790 .types
791 .iter()
792 .copied()
793 .collect::<Vec<_>>();
794 let dst_tys = self.types[adapter.lift.ty].params;
795 let dst_tys = self.types[dst_tys]
796 .types
797 .iter()
798 .copied()
799 .collect::<Vec<_>>();
800 let lift_opts = &adapter.lift.options;
801 let lower_opts = &adapter.lower.options;
802
803 assert_eq!(src_tys.len(), dst_tys.len());
805
806 let src_flat = if adapter.lower.options.async_ {
807 None
808 } else {
809 self.types
810 .flatten_types(lower_opts, MAX_FLAT_PARAMS, src_tys.iter().copied())
811 };
812 let dst_flat =
813 self.types
814 .flatten_types(lift_opts, MAX_FLAT_PARAMS, dst_tys.iter().copied());
815
816 let src = if let Some(flat) = &src_flat {
817 Source::Stack(Stack {
818 locals: ¶m_locals[..flat.len()],
819 opts: lower_opts,
820 })
821 } else {
822 let (addr, ty) = param_locals[0];
826 assert_eq!(ty, lower_opts.ptr());
827 let align = src_tys
828 .iter()
829 .map(|t| self.types.align(lower_opts, t))
830 .max()
831 .unwrap_or(1);
832 Source::Memory(self.memory_operand(lower_opts, TempLocal::new(addr, ty), align))
833 };
834
835 let dst = if let Some(flat) = &dst_flat {
836 Destination::Stack(flat, lift_opts)
837 } else {
838 let abi = CanonicalAbiInfo::record(dst_tys.iter().map(|t| self.types.canonical_abi(t)));
839 let (size, align) = if lift_opts.memory64 {
840 (abi.size64, abi.align64)
841 } else {
842 (abi.size32, abi.align32)
843 };
844
845 let size = MallocSize::Const(size);
848 Destination::Memory(self.malloc(lift_opts, size, align))
849 };
850
851 let srcs = src
852 .record_field_srcs(self.types, src_tys.iter().copied())
853 .zip(src_tys.iter());
854 let dsts = dst
855 .record_field_dsts(self.types, dst_tys.iter().copied())
856 .zip(dst_tys.iter());
857 for ((src, src_ty), (dst, dst_ty)) in srcs.zip(dsts) {
858 self.translate(&src_ty, &src, &dst_ty, &dst);
859 }
860
861 if let Destination::Memory(mem) = dst {
865 self.instruction(LocalGet(mem.addr.idx));
866 self.free_temp_local(mem.addr);
867 }
868 }
869
870 fn translate_results(
871 &mut self,
872 adapter: &AdapterData,
873 param_locals: &[(u32, ValType)],
874 result_locals: &[(u32, ValType)],
875 ) {
876 let src_tys = self.types[adapter.lift.ty].results;
877 let src_tys = self.types[src_tys]
878 .types
879 .iter()
880 .copied()
881 .collect::<Vec<_>>();
882 let dst_tys = self.types[adapter.lower.ty].results;
883 let dst_tys = self.types[dst_tys]
884 .types
885 .iter()
886 .copied()
887 .collect::<Vec<_>>();
888 let lift_opts = &adapter.lift.options;
889 let lower_opts = &adapter.lower.options;
890
891 let src_flat = self
892 .types
893 .flatten_lifting_types(lift_opts, src_tys.iter().copied());
894 let dst_flat = self
895 .types
896 .flatten_lowering_types(lower_opts, dst_tys.iter().copied());
897
898 let src = if src_flat.is_some() {
899 Source::Stack(Stack {
900 locals: result_locals,
901 opts: lift_opts,
902 })
903 } else {
904 let align = src_tys
909 .iter()
910 .map(|t| self.types.align(lift_opts, t))
911 .max()
912 .unwrap_or(1);
913 assert_eq!(
914 result_locals.len(),
915 if lower_opts.async_ || lift_opts.async_ {
916 2
917 } else {
918 1
919 }
920 );
921 let (addr, ty) = result_locals[0];
922 assert_eq!(ty, lift_opts.ptr());
923 Source::Memory(self.memory_operand(lift_opts, TempLocal::new(addr, ty), align))
924 };
925
926 let dst = if let Some(flat) = &dst_flat {
927 Destination::Stack(flat, lower_opts)
928 } else {
929 let align = dst_tys
933 .iter()
934 .map(|t| self.types.align(lower_opts, t))
935 .max()
936 .unwrap_or(1);
937 let (addr, ty) = *param_locals.last().expect("no retptr");
938 assert_eq!(ty, lower_opts.ptr());
939 Destination::Memory(self.memory_operand(lower_opts, TempLocal::new(addr, ty), align))
940 };
941
942 let srcs = src
943 .record_field_srcs(self.types, src_tys.iter().copied())
944 .zip(src_tys.iter());
945 let dsts = dst
946 .record_field_dsts(self.types, dst_tys.iter().copied())
947 .zip(dst_tys.iter());
948 for ((src, src_ty), (dst, dst_ty)) in srcs.zip(dsts) {
949 self.translate(&src_ty, &src, &dst_ty, &dst);
950 }
951 }
952
953 fn translate(
954 &mut self,
955 src_ty: &InterfaceType,
956 src: &Source<'_>,
957 dst_ty: &InterfaceType,
958 dst: &Destination,
959 ) {
960 if let Source::Memory(mem) = src {
961 self.assert_aligned(src_ty, mem);
962 }
963 if let Destination::Memory(mem) = dst {
964 self.assert_aligned(dst_ty, mem);
965 }
966
967 let cost = match src_ty {
997 InterfaceType::Bool
1001 | InterfaceType::U8
1002 | InterfaceType::S8
1003 | InterfaceType::U16
1004 | InterfaceType::S16
1005 | InterfaceType::U32
1006 | InterfaceType::S32
1007 | InterfaceType::U64
1008 | InterfaceType::S64
1009 | InterfaceType::Float32
1010 | InterfaceType::Float64 => 0,
1011
1012 InterfaceType::Char => 1,
1015
1016 InterfaceType::String => 40,
1019
1020 InterfaceType::List(_) => 40,
1023
1024 InterfaceType::Flags(i) => {
1025 let count = self.module.types[*i].names.len();
1026 match FlagsSize::from_count(count) {
1027 FlagsSize::Size0 => 0,
1028 FlagsSize::Size1 | FlagsSize::Size2 => 1,
1029 FlagsSize::Size4Plus(n) => n.into(),
1030 }
1031 }
1032
1033 InterfaceType::Record(i) => self.types[*i].fields.len(),
1034 InterfaceType::Tuple(i) => self.types[*i].types.len(),
1035 InterfaceType::Variant(i) => self.types[*i].cases.len(),
1036 InterfaceType::Enum(i) => self.types[*i].names.len(),
1037
1038 InterfaceType::Option(_) | InterfaceType::Result(_) => 2,
1040
1041 InterfaceType::Own(_)
1043 | InterfaceType::Borrow(_)
1044 | InterfaceType::Future(_)
1045 | InterfaceType::Stream(_)
1046 | InterfaceType::ErrorContext(_) => 1,
1047 };
1048
1049 match self.fuel.checked_sub(cost) {
1050 Some(n) => {
1056 self.fuel = n;
1057 match src_ty {
1058 InterfaceType::Bool => self.translate_bool(src, dst_ty, dst),
1059 InterfaceType::U8 => self.translate_u8(src, dst_ty, dst),
1060 InterfaceType::S8 => self.translate_s8(src, dst_ty, dst),
1061 InterfaceType::U16 => self.translate_u16(src, dst_ty, dst),
1062 InterfaceType::S16 => self.translate_s16(src, dst_ty, dst),
1063 InterfaceType::U32 => self.translate_u32(src, dst_ty, dst),
1064 InterfaceType::S32 => self.translate_s32(src, dst_ty, dst),
1065 InterfaceType::U64 => self.translate_u64(src, dst_ty, dst),
1066 InterfaceType::S64 => self.translate_s64(src, dst_ty, dst),
1067 InterfaceType::Float32 => self.translate_f32(src, dst_ty, dst),
1068 InterfaceType::Float64 => self.translate_f64(src, dst_ty, dst),
1069 InterfaceType::Char => self.translate_char(src, dst_ty, dst),
1070 InterfaceType::String => self.translate_string(src, dst_ty, dst),
1071 InterfaceType::List(t) => self.translate_list(*t, src, dst_ty, dst),
1072 InterfaceType::Record(t) => self.translate_record(*t, src, dst_ty, dst),
1073 InterfaceType::Flags(f) => self.translate_flags(*f, src, dst_ty, dst),
1074 InterfaceType::Tuple(t) => self.translate_tuple(*t, src, dst_ty, dst),
1075 InterfaceType::Variant(v) => self.translate_variant(*v, src, dst_ty, dst),
1076 InterfaceType::Enum(t) => self.translate_enum(*t, src, dst_ty, dst),
1077 InterfaceType::Option(t) => self.translate_option(*t, src, dst_ty, dst),
1078 InterfaceType::Result(t) => self.translate_result(*t, src, dst_ty, dst),
1079 InterfaceType::Own(t) => self.translate_own(*t, src, dst_ty, dst),
1080 InterfaceType::Borrow(t) => self.translate_borrow(*t, src, dst_ty, dst),
1081 InterfaceType::Future(t) => self.translate_future(*t, src, dst_ty, dst),
1082 InterfaceType::Stream(t) => self.translate_stream(*t, src, dst_ty, dst),
1083 InterfaceType::ErrorContext(t) => {
1084 self.translate_error_context(*t, src, dst_ty, dst)
1085 }
1086 }
1087 }
1088
1089 None => {
1095 let src_loc = match src {
1096 Source::Stack(stack) => {
1100 for (i, ty) in stack
1101 .opts
1102 .flat_types(src_ty, self.types)
1103 .unwrap()
1104 .iter()
1105 .enumerate()
1106 {
1107 let stack = stack.slice(i..i + 1);
1108 self.stack_get(&stack, (*ty).into());
1109 }
1110 HelperLocation::Stack
1111 }
1112 Source::Memory(mem) => {
1117 self.push_mem_addr(mem);
1118 HelperLocation::Memory
1119 }
1120 };
1121 let dst_loc = match dst {
1122 Destination::Stack(..) => HelperLocation::Stack,
1123 Destination::Memory(mem) => {
1124 self.push_mem_addr(mem);
1125 HelperLocation::Memory
1126 }
1127 };
1128 let helper = self.module.translate_helper(Helper {
1134 src: HelperType {
1135 ty: *src_ty,
1136 opts: *src.opts(),
1137 loc: src_loc,
1138 },
1139 dst: HelperType {
1140 ty: *dst_ty,
1141 opts: *dst.opts(),
1142 loc: dst_loc,
1143 },
1144 });
1145 self.flush_code();
1148 self.module.funcs[self.result].body.push(Body::Call(helper));
1149
1150 if let Destination::Stack(tys, opts) = dst {
1159 let flat = self
1160 .types
1161 .flatten_types(opts, usize::MAX, [*dst_ty])
1162 .unwrap();
1163 assert_eq!(flat.len(), tys.len());
1164 let locals = flat
1165 .iter()
1166 .rev()
1167 .map(|ty| self.local_set_new_tmp(*ty))
1168 .collect::<Vec<_>>();
1169 for (ty, local) in tys.iter().zip(locals.into_iter().rev()) {
1170 self.instruction(LocalGet(local.idx));
1171 self.stack_set(std::slice::from_ref(ty), local.ty);
1172 self.free_temp_local(local);
1173 }
1174 }
1175 }
1176 }
1177 }
1178
1179 fn push_mem_addr(&mut self, mem: &Memory<'_>) {
1180 self.instruction(LocalGet(mem.addr.idx));
1181 if mem.offset != 0 {
1182 self.ptr_uconst(mem.opts, mem.offset);
1183 self.ptr_add(mem.opts);
1184 }
1185 }
1186
1187 fn translate_bool(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1188 assert!(matches!(dst_ty, InterfaceType::Bool));
1190 self.push_dst_addr(dst);
1191
1192 self.instruction(I32Const(1));
1195 self.instruction(I32Const(0));
1196 match src {
1197 Source::Memory(mem) => self.i32_load8u(mem),
1198 Source::Stack(stack) => self.stack_get(stack, ValType::I32),
1199 }
1200 self.instruction(Select);
1201
1202 match dst {
1203 Destination::Memory(mem) => self.i32_store8(mem),
1204 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I32),
1205 }
1206 }
1207
1208 fn translate_u8(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1209 assert!(matches!(dst_ty, InterfaceType::U8));
1211 self.convert_u8_mask(src, dst, 0xff);
1212 }
1213
1214 fn convert_u8_mask(&mut self, src: &Source<'_>, dst: &Destination<'_>, mask: u8) {
1215 self.push_dst_addr(dst);
1216 let mut needs_mask = true;
1217 match src {
1218 Source::Memory(mem) => {
1219 self.i32_load8u(mem);
1220 needs_mask = mask != 0xff;
1221 }
1222 Source::Stack(stack) => {
1223 self.stack_get(stack, ValType::I32);
1224 }
1225 }
1226 if needs_mask {
1227 self.instruction(I32Const(i32::from(mask)));
1228 self.instruction(I32And);
1229 }
1230 match dst {
1231 Destination::Memory(mem) => self.i32_store8(mem),
1232 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I32),
1233 }
1234 }
1235
1236 fn translate_s8(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1237 assert!(matches!(dst_ty, InterfaceType::S8));
1239 self.push_dst_addr(dst);
1240 match src {
1241 Source::Memory(mem) => self.i32_load8s(mem),
1242 Source::Stack(stack) => {
1243 self.stack_get(stack, ValType::I32);
1244 self.instruction(I32Extend8S);
1245 }
1246 }
1247 match dst {
1248 Destination::Memory(mem) => self.i32_store8(mem),
1249 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I32),
1250 }
1251 }
1252
1253 fn translate_u16(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1254 assert!(matches!(dst_ty, InterfaceType::U16));
1256 self.convert_u16_mask(src, dst, 0xffff);
1257 }
1258
1259 fn convert_u16_mask(&mut self, src: &Source<'_>, dst: &Destination<'_>, mask: u16) {
1260 self.push_dst_addr(dst);
1261 let mut needs_mask = true;
1262 match src {
1263 Source::Memory(mem) => {
1264 self.i32_load16u(mem);
1265 needs_mask = mask != 0xffff;
1266 }
1267 Source::Stack(stack) => {
1268 self.stack_get(stack, ValType::I32);
1269 }
1270 }
1271 if needs_mask {
1272 self.instruction(I32Const(i32::from(mask)));
1273 self.instruction(I32And);
1274 }
1275 match dst {
1276 Destination::Memory(mem) => self.i32_store16(mem),
1277 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I32),
1278 }
1279 }
1280
1281 fn translate_s16(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1282 assert!(matches!(dst_ty, InterfaceType::S16));
1284 self.push_dst_addr(dst);
1285 match src {
1286 Source::Memory(mem) => self.i32_load16s(mem),
1287 Source::Stack(stack) => {
1288 self.stack_get(stack, ValType::I32);
1289 self.instruction(I32Extend16S);
1290 }
1291 }
1292 match dst {
1293 Destination::Memory(mem) => self.i32_store16(mem),
1294 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I32),
1295 }
1296 }
1297
1298 fn translate_u32(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1299 assert!(matches!(dst_ty, InterfaceType::U32));
1301 self.convert_u32_mask(src, dst, 0xffffffff)
1302 }
1303
1304 fn convert_u32_mask(&mut self, src: &Source<'_>, dst: &Destination<'_>, mask: u32) {
1305 self.push_dst_addr(dst);
1306 match src {
1307 Source::Memory(mem) => self.i32_load(mem),
1308 Source::Stack(stack) => self.stack_get(stack, ValType::I32),
1309 }
1310 if mask != 0xffffffff {
1311 self.instruction(I32Const(mask as i32));
1312 self.instruction(I32And);
1313 }
1314 match dst {
1315 Destination::Memory(mem) => self.i32_store(mem),
1316 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I32),
1317 }
1318 }
1319
1320 fn translate_s32(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1321 assert!(matches!(dst_ty, InterfaceType::S32));
1323 self.push_dst_addr(dst);
1324 match src {
1325 Source::Memory(mem) => self.i32_load(mem),
1326 Source::Stack(stack) => self.stack_get(stack, ValType::I32),
1327 }
1328 match dst {
1329 Destination::Memory(mem) => self.i32_store(mem),
1330 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I32),
1331 }
1332 }
1333
1334 fn translate_u64(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1335 assert!(matches!(dst_ty, InterfaceType::U64));
1337 self.push_dst_addr(dst);
1338 match src {
1339 Source::Memory(mem) => self.i64_load(mem),
1340 Source::Stack(stack) => self.stack_get(stack, ValType::I64),
1341 }
1342 match dst {
1343 Destination::Memory(mem) => self.i64_store(mem),
1344 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I64),
1345 }
1346 }
1347
1348 fn translate_s64(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1349 assert!(matches!(dst_ty, InterfaceType::S64));
1351 self.push_dst_addr(dst);
1352 match src {
1353 Source::Memory(mem) => self.i64_load(mem),
1354 Source::Stack(stack) => self.stack_get(stack, ValType::I64),
1355 }
1356 match dst {
1357 Destination::Memory(mem) => self.i64_store(mem),
1358 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I64),
1359 }
1360 }
1361
1362 fn translate_f32(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1363 assert!(matches!(dst_ty, InterfaceType::Float32));
1365 self.push_dst_addr(dst);
1366 match src {
1367 Source::Memory(mem) => self.f32_load(mem),
1368 Source::Stack(stack) => self.stack_get(stack, ValType::F32),
1369 }
1370 match dst {
1371 Destination::Memory(mem) => self.f32_store(mem),
1372 Destination::Stack(stack, _) => self.stack_set(stack, ValType::F32),
1373 }
1374 }
1375
1376 fn translate_f64(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1377 assert!(matches!(dst_ty, InterfaceType::Float64));
1379 self.push_dst_addr(dst);
1380 match src {
1381 Source::Memory(mem) => self.f64_load(mem),
1382 Source::Stack(stack) => self.stack_get(stack, ValType::F64),
1383 }
1384 match dst {
1385 Destination::Memory(mem) => self.f64_store(mem),
1386 Destination::Stack(stack, _) => self.stack_set(stack, ValType::F64),
1387 }
1388 }
1389
1390 fn translate_char(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1391 assert!(matches!(dst_ty, InterfaceType::Char));
1392 match src {
1393 Source::Memory(mem) => self.i32_load(mem),
1394 Source::Stack(stack) => self.stack_get(stack, ValType::I32),
1395 }
1396 let local = self.local_set_new_tmp(ValType::I32);
1397
1398 self.instruction(Block(BlockType::Empty));
1414 self.instruction(Block(BlockType::Empty));
1415 self.instruction(LocalGet(local.idx));
1416 self.instruction(I32Const(0xd800));
1417 self.instruction(I32Xor);
1418 self.instruction(I32Const(-0x110000));
1419 self.instruction(I32Add);
1420 self.instruction(I32Const(-0x10f800));
1421 self.instruction(I32LtU);
1422 self.instruction(BrIf(0));
1423 self.instruction(LocalGet(local.idx));
1424 self.instruction(I32Const(0x110000));
1425 self.instruction(I32Ne);
1426 self.instruction(BrIf(1));
1427 self.instruction(End);
1428 self.trap(Trap::InvalidChar);
1429 self.instruction(End);
1430
1431 self.push_dst_addr(dst);
1432 self.instruction(LocalGet(local.idx));
1433 match dst {
1434 Destination::Memory(mem) => {
1435 self.i32_store(mem);
1436 }
1437 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I32),
1438 }
1439
1440 self.free_temp_local(local);
1441 }
1442
1443 fn translate_string(&mut self, src: &Source<'_>, dst_ty: &InterfaceType, dst: &Destination) {
1444 assert!(matches!(dst_ty, InterfaceType::String));
1445 let src_opts = src.opts();
1446 let dst_opts = dst.opts();
1447
1448 match src {
1453 Source::Stack(s) => {
1454 assert_eq!(s.locals.len(), 2);
1455 self.stack_get(&s.slice(0..1), src_opts.ptr());
1456 self.stack_get(&s.slice(1..2), src_opts.ptr());
1457 }
1458 Source::Memory(mem) => {
1459 self.ptr_load(mem);
1460 self.ptr_load(&mem.bump(src_opts.ptr_size().into()));
1461 }
1462 }
1463 let src_len = self.local_set_new_tmp(src_opts.ptr());
1464 let src_ptr = self.local_set_new_tmp(src_opts.ptr());
1465 let src_str = WasmString {
1466 ptr: src_ptr,
1467 len: src_len,
1468 opts: src_opts,
1469 };
1470
1471 let dst_str = match src_opts.string_encoding {
1472 StringEncoding::Utf8 => match dst_opts.string_encoding {
1473 StringEncoding::Utf8 => self.string_copy(&src_str, FE::Utf8, dst_opts, FE::Utf8),
1474 StringEncoding::Utf16 => self.string_utf8_to_utf16(&src_str, dst_opts),
1475 StringEncoding::CompactUtf16 => {
1476 self.string_to_compact(&src_str, FE::Utf8, dst_opts)
1477 }
1478 },
1479
1480 StringEncoding::Utf16 => {
1481 self.verify_aligned(src_opts, src_str.ptr.idx, 2);
1482 match dst_opts.string_encoding {
1483 StringEncoding::Utf8 => {
1484 self.string_deflate_to_utf8(&src_str, FE::Utf16, dst_opts)
1485 }
1486 StringEncoding::Utf16 => {
1487 self.string_copy(&src_str, FE::Utf16, dst_opts, FE::Utf16)
1488 }
1489 StringEncoding::CompactUtf16 => {
1490 self.string_to_compact(&src_str, FE::Utf16, dst_opts)
1491 }
1492 }
1493 }
1494
1495 StringEncoding::CompactUtf16 => {
1496 self.verify_aligned(src_opts, src_str.ptr.idx, 2);
1497
1498 self.instruction(LocalGet(src_str.len.idx));
1501 self.ptr_uconst(src_opts, UTF16_TAG);
1502 self.ptr_and(src_opts);
1503 self.ptr_if(src_opts, BlockType::Empty);
1504
1505 self.instruction(LocalGet(src_str.len.idx));
1509 self.ptr_uconst(src_opts, UTF16_TAG);
1510 self.ptr_xor(src_opts);
1511 self.instruction(LocalSet(src_str.len.idx));
1512 let s1 = match dst_opts.string_encoding {
1513 StringEncoding::Utf8 => {
1514 self.string_deflate_to_utf8(&src_str, FE::Utf16, dst_opts)
1515 }
1516 StringEncoding::Utf16 => {
1517 self.string_copy(&src_str, FE::Utf16, dst_opts, FE::Utf16)
1518 }
1519 StringEncoding::CompactUtf16 => {
1520 self.string_compact_utf16_to_compact(&src_str, dst_opts)
1521 }
1522 };
1523
1524 self.instruction(Else);
1525
1526 let s2 = match dst_opts.string_encoding {
1530 StringEncoding::Utf16 => {
1531 self.string_copy(&src_str, FE::Latin1, dst_opts, FE::Utf16)
1532 }
1533 StringEncoding::Utf8 => {
1534 self.string_deflate_to_utf8(&src_str, FE::Latin1, dst_opts)
1535 }
1536 StringEncoding::CompactUtf16 => {
1537 self.string_copy(&src_str, FE::Latin1, dst_opts, FE::Latin1)
1538 }
1539 };
1540 self.instruction(LocalGet(s2.ptr.idx));
1543 self.instruction(LocalSet(s1.ptr.idx));
1544 self.instruction(LocalGet(s2.len.idx));
1545 self.instruction(LocalSet(s1.len.idx));
1546 self.instruction(End);
1547 self.free_temp_local(s2.ptr);
1548 self.free_temp_local(s2.len);
1549 s1
1550 }
1551 };
1552
1553 match dst {
1555 Destination::Stack(s, _) => {
1556 self.instruction(LocalGet(dst_str.ptr.idx));
1557 self.stack_set(&s[..1], dst_opts.ptr());
1558 self.instruction(LocalGet(dst_str.len.idx));
1559 self.stack_set(&s[1..], dst_opts.ptr());
1560 }
1561 Destination::Memory(mem) => {
1562 self.instruction(LocalGet(mem.addr.idx));
1563 self.instruction(LocalGet(dst_str.ptr.idx));
1564 self.ptr_store(mem);
1565 self.instruction(LocalGet(mem.addr.idx));
1566 self.instruction(LocalGet(dst_str.len.idx));
1567 self.ptr_store(&mem.bump(dst_opts.ptr_size().into()));
1568 }
1569 }
1570
1571 self.free_temp_local(src_str.ptr);
1572 self.free_temp_local(src_str.len);
1573 self.free_temp_local(dst_str.ptr);
1574 self.free_temp_local(dst_str.len);
1575 }
1576
1577 fn string_copy<'c>(
1590 &mut self,
1591 src: &WasmString<'_>,
1592 src_enc: FE,
1593 dst_opts: &'c Options,
1594 dst_enc: FE,
1595 ) -> WasmString<'c> {
1596 assert!(dst_enc.width() >= src_enc.width());
1597 self.validate_string_length(src, dst_enc);
1598
1599 let mut src_byte_len_tmp = None;
1603 let src_byte_len = if src_enc.width() == 1 {
1604 src.len.idx
1605 } else {
1606 assert_eq!(src_enc.width(), 2);
1607 self.instruction(LocalGet(src.len.idx));
1608 self.ptr_uconst(src.opts, 1);
1609 self.ptr_shl(src.opts);
1610 let tmp = self.local_set_new_tmp(src.opts.ptr());
1611 let ret = tmp.idx;
1612 src_byte_len_tmp = Some(tmp);
1613 ret
1614 };
1615
1616 self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst_opts.ptr());
1619 let dst_len = self.local_tee_new_tmp(dst_opts.ptr());
1620 if dst_enc.width() > 1 {
1621 assert_eq!(dst_enc.width(), 2);
1622 self.ptr_uconst(dst_opts, 1);
1623 self.ptr_shl(dst_opts);
1624 }
1625 let dst_byte_len = self.local_set_new_tmp(dst_opts.ptr());
1626
1627 let dst = {
1630 let dst_mem = self.malloc(
1631 dst_opts,
1632 MallocSize::Local(dst_byte_len.idx),
1633 dst_enc.width().into(),
1634 );
1635 WasmString {
1636 ptr: dst_mem.addr,
1637 len: dst_len,
1638 opts: dst_opts,
1639 }
1640 };
1641
1642 self.validate_string_inbounds(src, src_byte_len);
1647 self.validate_string_inbounds(&dst, dst_byte_len.idx);
1648
1649 let op = if src_enc == dst_enc {
1653 Transcode::Copy(src_enc)
1654 } else {
1655 assert_eq!(src_enc, FE::Latin1);
1656 assert_eq!(dst_enc, FE::Utf16);
1657 Transcode::Latin1ToUtf16
1658 };
1659 let transcode = self.transcoder(src, &dst, op);
1660 self.instruction(LocalGet(src.ptr.idx));
1661 self.instruction(LocalGet(src.len.idx));
1662 self.instruction(LocalGet(dst.ptr.idx));
1663 self.instruction(Call(transcode.as_u32()));
1664
1665 self.free_temp_local(dst_byte_len);
1666 if let Some(tmp) = src_byte_len_tmp {
1667 self.free_temp_local(tmp);
1668 }
1669
1670 dst
1671 }
1672 fn string_deflate_to_utf8<'c>(
1685 &mut self,
1686 src: &WasmString<'_>,
1687 src_enc: FE,
1688 dst_opts: &'c Options,
1689 ) -> WasmString<'c> {
1690 self.validate_string_length(src, src_enc);
1691
1692 self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst_opts.ptr());
1696 let dst_len = self.local_tee_new_tmp(dst_opts.ptr());
1697 let dst_byte_len = self.local_set_new_tmp(dst_opts.ptr());
1698
1699 let dst = {
1700 let dst_mem = self.malloc(dst_opts, MallocSize::Local(dst_byte_len.idx), 1);
1701 WasmString {
1702 ptr: dst_mem.addr,
1703 len: dst_len,
1704 opts: dst_opts,
1705 }
1706 };
1707
1708 let mut src_byte_len_tmp = None;
1710 let src_byte_len = match src_enc {
1711 FE::Latin1 => src.len.idx,
1712 FE::Utf16 => {
1713 self.instruction(LocalGet(src.len.idx));
1714 self.ptr_uconst(src.opts, 1);
1715 self.ptr_shl(src.opts);
1716 let tmp = self.local_set_new_tmp(src.opts.ptr());
1717 let ret = tmp.idx;
1718 src_byte_len_tmp = Some(tmp);
1719 ret
1720 }
1721 FE::Utf8 => unreachable!(),
1722 };
1723 self.validate_string_inbounds(src, src_byte_len);
1724 self.validate_string_inbounds(&dst, dst_byte_len.idx);
1725
1726 let op = match src_enc {
1728 FE::Latin1 => Transcode::Latin1ToUtf8,
1729 FE::Utf16 => Transcode::Utf16ToUtf8,
1730 FE::Utf8 => unreachable!(),
1731 };
1732 let transcode = self.transcoder(src, &dst, op);
1733 self.instruction(LocalGet(src.ptr.idx));
1734 self.instruction(LocalGet(src.len.idx));
1735 self.instruction(LocalGet(dst.ptr.idx));
1736 self.instruction(LocalGet(dst_byte_len.idx));
1737 self.instruction(Call(transcode.as_u32()));
1738 self.instruction(LocalSet(dst.len.idx));
1739 let src_len_tmp = self.local_set_new_tmp(src.opts.ptr());
1740
1741 self.instruction(LocalGet(src_len_tmp.idx));
1745 self.instruction(LocalGet(src.len.idx));
1746 self.ptr_ne(src.opts);
1747 self.instruction(If(BlockType::Empty));
1748
1749 self.instruction(LocalGet(dst.ptr.idx)); self.instruction(LocalGet(dst_byte_len.idx)); self.ptr_uconst(dst.opts, 1); let factor = match src_enc {
1756 FE::Latin1 => 2,
1757 FE::Utf16 => 3,
1758 _ => unreachable!(),
1759 };
1760 self.validate_string_length_u8(src, factor);
1761 self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst_opts.ptr());
1762 self.ptr_uconst(dst_opts, factor.into());
1763 self.ptr_mul(dst_opts);
1764 self.instruction(LocalTee(dst_byte_len.idx));
1765 self.instruction(Call(dst_opts.realloc.unwrap().as_u32()));
1766 self.instruction(LocalSet(dst.ptr.idx));
1767
1768 self.validate_string_inbounds(&dst, dst_byte_len.idx);
1770
1771 self.instruction(LocalGet(src.ptr.idx));
1776 self.instruction(LocalGet(src_len_tmp.idx));
1777 if let FE::Utf16 = src_enc {
1778 self.ptr_uconst(src.opts, 1);
1779 self.ptr_shl(src.opts);
1780 }
1781 self.ptr_add(src.opts);
1782 self.instruction(LocalGet(src.len.idx));
1783 self.instruction(LocalGet(src_len_tmp.idx));
1784 self.ptr_sub(src.opts);
1785 self.instruction(LocalGet(dst.ptr.idx));
1786 self.instruction(LocalGet(dst.len.idx));
1787 self.ptr_add(dst.opts);
1788 self.instruction(LocalGet(dst_byte_len.idx));
1789 self.instruction(LocalGet(dst.len.idx));
1790 self.ptr_sub(dst.opts);
1791 self.instruction(Call(transcode.as_u32()));
1792
1793 self.instruction(LocalGet(dst.len.idx));
1797 self.ptr_add(dst.opts);
1798 self.instruction(LocalSet(dst.len.idx));
1799
1800 if self.module.debug {
1803 self.instruction(LocalGet(src.len.idx));
1804 self.instruction(LocalGet(src_len_tmp.idx));
1805 self.ptr_sub(src.opts);
1806 self.ptr_ne(src.opts);
1807 self.instruction(If(BlockType::Empty));
1808 self.trap(Trap::AssertFailed("should have finished encoding"));
1809 self.instruction(End);
1810 } else {
1811 self.instruction(Drop);
1812 }
1813
1814 self.instruction(LocalGet(dst.len.idx));
1816 self.instruction(LocalGet(dst_byte_len.idx));
1817 self.ptr_ne(dst.opts);
1818 self.instruction(If(BlockType::Empty));
1819 self.instruction(LocalGet(dst.ptr.idx)); self.instruction(LocalGet(dst_byte_len.idx)); self.ptr_uconst(dst.opts, 1); self.instruction(LocalGet(dst.len.idx)); self.instruction(Call(dst.opts.realloc.unwrap().as_u32()));
1824 self.instruction(LocalSet(dst.ptr.idx));
1825 self.instruction(End);
1826
1827 if self.module.debug {
1830 self.instruction(Else);
1831
1832 self.instruction(LocalGet(dst.len.idx));
1833 self.instruction(LocalGet(dst_byte_len.idx));
1834 self.ptr_ne(dst_opts);
1835 self.instruction(If(BlockType::Empty));
1836 self.trap(Trap::AssertFailed("should have finished encoding"));
1837 self.instruction(End);
1838 }
1839
1840 self.instruction(End); self.free_temp_local(src_len_tmp);
1843 self.free_temp_local(dst_byte_len);
1844 if let Some(tmp) = src_byte_len_tmp {
1845 self.free_temp_local(tmp);
1846 }
1847
1848 dst
1849 }
1850
1851 fn string_utf8_to_utf16<'c>(
1866 &mut self,
1867 src: &WasmString<'_>,
1868 dst_opts: &'c Options,
1869 ) -> WasmString<'c> {
1870 self.validate_string_length(src, FE::Utf16);
1871 self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst_opts.ptr());
1872 let dst_len = self.local_tee_new_tmp(dst_opts.ptr());
1873 self.ptr_uconst(dst_opts, 1);
1874 self.ptr_shl(dst_opts);
1875 let dst_byte_len = self.local_set_new_tmp(dst_opts.ptr());
1876 let dst = {
1877 let dst_mem = self.malloc(dst_opts, MallocSize::Local(dst_byte_len.idx), 2);
1878 WasmString {
1879 ptr: dst_mem.addr,
1880 len: dst_len,
1881 opts: dst_opts,
1882 }
1883 };
1884
1885 self.validate_string_inbounds(src, src.len.idx);
1886 self.validate_string_inbounds(&dst, dst_byte_len.idx);
1887
1888 let transcode = self.transcoder(src, &dst, Transcode::Utf8ToUtf16);
1889 self.instruction(LocalGet(src.ptr.idx));
1890 self.instruction(LocalGet(src.len.idx));
1891 self.instruction(LocalGet(dst.ptr.idx));
1892 self.instruction(Call(transcode.as_u32()));
1893 self.instruction(LocalSet(dst.len.idx));
1894
1895 self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst.opts.ptr());
1903 self.instruction(LocalGet(dst.len.idx));
1904 self.ptr_ne(dst_opts);
1905 self.instruction(If(BlockType::Empty));
1906 self.instruction(LocalGet(dst.ptr.idx));
1907 self.instruction(LocalGet(dst_byte_len.idx));
1908 self.ptr_uconst(dst.opts, 2);
1909 self.instruction(LocalGet(dst.len.idx));
1910 self.ptr_uconst(dst.opts, 1);
1911 self.ptr_shl(dst.opts);
1912 self.instruction(Call(dst.opts.realloc.unwrap().as_u32()));
1913 self.instruction(LocalSet(dst.ptr.idx));
1914 self.instruction(End); self.free_temp_local(dst_byte_len);
1917
1918 dst
1919 }
1920
1921 fn string_compact_utf16_to_compact<'c>(
1935 &mut self,
1936 src: &WasmString<'_>,
1937 dst_opts: &'c Options,
1938 ) -> WasmString<'c> {
1939 self.validate_string_length(src, FE::Utf16);
1940 self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst_opts.ptr());
1941 let dst_len = self.local_tee_new_tmp(dst_opts.ptr());
1942 self.ptr_uconst(dst_opts, 1);
1943 self.ptr_shl(dst_opts);
1944 let dst_byte_len = self.local_set_new_tmp(dst_opts.ptr());
1945 let dst = {
1946 let dst_mem = self.malloc(dst_opts, MallocSize::Local(dst_byte_len.idx), 2);
1947 WasmString {
1948 ptr: dst_mem.addr,
1949 len: dst_len,
1950 opts: dst_opts,
1951 }
1952 };
1953
1954 self.convert_src_len_to_dst(dst_byte_len.idx, dst.opts.ptr(), src.opts.ptr());
1955 let src_byte_len = self.local_set_new_tmp(src.opts.ptr());
1956
1957 self.validate_string_inbounds(src, src_byte_len.idx);
1958 self.validate_string_inbounds(&dst, dst_byte_len.idx);
1959
1960 let transcode = self.transcoder(src, &dst, Transcode::Utf16ToCompactProbablyUtf16);
1961 self.instruction(LocalGet(src.ptr.idx));
1962 self.instruction(LocalGet(src.len.idx));
1963 self.instruction(LocalGet(dst.ptr.idx));
1964 self.instruction(Call(transcode.as_u32()));
1965 self.instruction(LocalSet(dst.len.idx));
1966
1967 if self.module.debug {
1970 self.instruction(LocalGet(dst.len.idx));
1971 self.ptr_uconst(dst.opts, !UTF16_TAG);
1972 self.ptr_and(dst.opts);
1973 self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst.opts.ptr());
1974 self.ptr_ne(dst.opts);
1975 self.instruction(If(BlockType::Empty));
1976 self.trap(Trap::AssertFailed("expected equal code units"));
1977 self.instruction(End);
1978 }
1979
1980 self.instruction(LocalGet(dst.len.idx));
1984 self.ptr_uconst(dst.opts, UTF16_TAG);
1985 self.ptr_and(dst.opts);
1986 self.ptr_br_if(dst.opts, 0);
1987
1988 self.instruction(LocalGet(dst.ptr.idx)); self.instruction(LocalGet(dst_byte_len.idx)); self.ptr_uconst(dst.opts, 2); self.instruction(LocalGet(dst.len.idx)); self.instruction(Call(dst.opts.realloc.unwrap().as_u32()));
1994 self.instruction(LocalSet(dst.ptr.idx));
1995
1996 self.free_temp_local(dst_byte_len);
1997 self.free_temp_local(src_byte_len);
1998
1999 dst
2000 }
2001
2002 fn string_to_compact<'c>(
2009 &mut self,
2010 src: &WasmString<'_>,
2011 src_enc: FE,
2012 dst_opts: &'c Options,
2013 ) -> WasmString<'c> {
2014 self.validate_string_length(src, src_enc);
2015 self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst_opts.ptr());
2016 let dst_len = self.local_tee_new_tmp(dst_opts.ptr());
2017 let dst_byte_len = self.local_set_new_tmp(dst_opts.ptr());
2018 let dst = {
2019 let dst_mem = self.malloc(dst_opts, MallocSize::Local(dst_byte_len.idx), 2);
2020 WasmString {
2021 ptr: dst_mem.addr,
2022 len: dst_len,
2023 opts: dst_opts,
2024 }
2025 };
2026
2027 self.validate_string_inbounds(src, src.len.idx);
2028 self.validate_string_inbounds(&dst, dst_byte_len.idx);
2029
2030 let (latin1, utf16) = match src_enc {
2034 FE::Utf8 => (Transcode::Utf8ToLatin1, Transcode::Utf8ToCompactUtf16),
2035 FE::Utf16 => (Transcode::Utf16ToLatin1, Transcode::Utf16ToCompactUtf16),
2036 FE::Latin1 => unreachable!(),
2037 };
2038 let transcode_latin1 = self.transcoder(src, &dst, latin1);
2039 let transcode_utf16 = self.transcoder(src, &dst, utf16);
2040 self.instruction(LocalGet(src.ptr.idx));
2041 self.instruction(LocalGet(src.len.idx));
2042 self.instruction(LocalGet(dst.ptr.idx));
2043 self.instruction(Call(transcode_latin1.as_u32()));
2044 self.instruction(LocalSet(dst.len.idx));
2045 let src_len_tmp = self.local_set_new_tmp(src.opts.ptr());
2046
2047 self.instruction(LocalGet(src_len_tmp.idx));
2050 self.instruction(LocalGet(src.len.idx));
2051 self.ptr_eq(src.opts);
2052 self.instruction(If(BlockType::Empty)); self.instruction(LocalGet(dst_byte_len.idx));
2058 self.instruction(LocalGet(dst.len.idx));
2059 self.ptr_ne(dst.opts);
2060 self.instruction(If(BlockType::Empty));
2061 self.instruction(LocalGet(dst.ptr.idx)); self.instruction(LocalGet(dst_byte_len.idx)); self.ptr_uconst(dst.opts, 2); self.instruction(LocalGet(dst.len.idx)); self.instruction(Call(dst.opts.realloc.unwrap().as_u32()));
2066 self.instruction(LocalSet(dst.ptr.idx));
2067 self.instruction(End);
2068
2069 self.instruction(Else); if src_enc.width() == 1 {
2078 self.validate_string_length_u8(src, 2);
2079 }
2080
2081 self.instruction(LocalGet(dst.ptr.idx)); self.instruction(LocalGet(dst_byte_len.idx)); self.ptr_uconst(dst.opts, 2); self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst.opts.ptr());
2087 self.ptr_uconst(dst.opts, 1);
2088 self.ptr_shl(dst.opts);
2089 self.instruction(LocalTee(dst_byte_len.idx));
2090 self.instruction(Call(dst.opts.realloc.unwrap().as_u32()));
2091 self.instruction(LocalSet(dst.ptr.idx));
2092
2093 self.instruction(LocalGet(src.ptr.idx));
2097 self.instruction(LocalGet(src_len_tmp.idx));
2098 if let FE::Utf16 = src_enc {
2099 self.ptr_uconst(src.opts, 1);
2100 self.ptr_shl(src.opts);
2101 }
2102 self.ptr_add(src.opts);
2103 self.instruction(LocalGet(src.len.idx));
2104 self.instruction(LocalGet(src_len_tmp.idx));
2105 self.ptr_sub(src.opts);
2106 self.instruction(LocalGet(dst.ptr.idx));
2107 self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst.opts.ptr());
2108 self.instruction(LocalGet(dst.len.idx));
2109 self.instruction(Call(transcode_utf16.as_u32()));
2110 self.instruction(LocalSet(dst.len.idx));
2111
2112 self.instruction(LocalGet(dst.len.idx));
2120 self.convert_src_len_to_dst(src.len.idx, src.opts.ptr(), dst.opts.ptr());
2121 self.ptr_ne(dst.opts);
2122 self.instruction(If(BlockType::Empty));
2123 self.instruction(LocalGet(dst.ptr.idx)); self.instruction(LocalGet(dst_byte_len.idx)); self.ptr_uconst(dst.opts, 2); self.instruction(LocalGet(dst.len.idx));
2127 self.ptr_uconst(dst.opts, 1);
2128 self.ptr_shl(dst.opts);
2129 self.instruction(Call(dst.opts.realloc.unwrap().as_u32()));
2130 self.instruction(LocalSet(dst.ptr.idx));
2131 self.instruction(End);
2132
2133 self.instruction(LocalGet(dst.len.idx));
2135 self.ptr_uconst(dst.opts, UTF16_TAG);
2136 self.ptr_or(dst.opts);
2137 self.instruction(LocalSet(dst.len.idx));
2138
2139 self.instruction(End); self.free_temp_local(src_len_tmp);
2142 self.free_temp_local(dst_byte_len);
2143
2144 dst
2145 }
2146
2147 fn validate_string_length(&mut self, src: &WasmString<'_>, dst: FE) {
2148 self.validate_string_length_u8(src, dst.width())
2149 }
2150
2151 fn validate_string_length_u8(&mut self, s: &WasmString<'_>, dst: u8) {
2152 self.instruction(LocalGet(s.len.idx));
2155 let max = MAX_STRING_BYTE_LENGTH / u32::from(dst);
2156 self.ptr_uconst(s.opts, max);
2157 self.ptr_ge_u(s.opts);
2158 self.instruction(If(BlockType::Empty));
2159 self.trap(Trap::StringLengthTooBig);
2160 self.instruction(End);
2161 }
2162
2163 fn transcoder(
2164 &mut self,
2165 src: &WasmString<'_>,
2166 dst: &WasmString<'_>,
2167 op: Transcode,
2168 ) -> FuncIndex {
2169 self.module.import_transcoder(Transcoder {
2170 from_memory: src.opts.memory.unwrap(),
2171 from_memory64: src.opts.memory64,
2172 to_memory: dst.opts.memory.unwrap(),
2173 to_memory64: dst.opts.memory64,
2174 op,
2175 })
2176 }
2177
2178 fn validate_string_inbounds(&mut self, s: &WasmString<'_>, byte_len: u32) {
2179 self.validate_memory_inbounds(s.opts, s.ptr.idx, byte_len, Trap::StringLengthOverflow)
2180 }
2181
2182 fn validate_memory_inbounds(
2183 &mut self,
2184 opts: &Options,
2185 ptr_local: u32,
2186 byte_len_local: u32,
2187 trap: Trap,
2188 ) {
2189 let extend_to_64 = |me: &mut Self| {
2190 if !opts.memory64 {
2191 me.instruction(I64ExtendI32U);
2192 }
2193 };
2194
2195 self.instruction(Block(BlockType::Empty));
2196 self.instruction(Block(BlockType::Empty));
2197
2198 self.instruction(MemorySize(opts.memory.unwrap().as_u32()));
2203 extend_to_64(self);
2204 self.instruction(I64Const(16));
2205 self.instruction(I64Shl);
2206
2207 self.instruction(LocalGet(ptr_local));
2212 extend_to_64(self);
2213 self.instruction(LocalGet(byte_len_local));
2214 extend_to_64(self);
2215 self.instruction(I64Add);
2216 if opts.memory64 {
2217 let tmp = self.local_tee_new_tmp(ValType::I64);
2218 self.instruction(LocalGet(ptr_local));
2219 self.ptr_lt_u(opts);
2220 self.instruction(BrIf(0));
2221 self.instruction(LocalGet(tmp.idx));
2222 self.free_temp_local(tmp);
2223 }
2224
2225 self.instruction(I64GeU);
2229 self.instruction(BrIf(1));
2230
2231 self.instruction(End);
2232 self.trap(trap);
2233 self.instruction(End);
2234 }
2235
2236 fn translate_list(
2237 &mut self,
2238 src_ty: TypeListIndex,
2239 src: &Source<'_>,
2240 dst_ty: &InterfaceType,
2241 dst: &Destination,
2242 ) {
2243 let src_element_ty = &self.types[src_ty].element;
2244 let dst_element_ty = match dst_ty {
2245 InterfaceType::List(r) => &self.types[*r].element,
2246 _ => panic!("expected a list"),
2247 };
2248 let src_opts = src.opts();
2249 let dst_opts = dst.opts();
2250 let (src_size, src_align) = self.types.size_align(src_opts, src_element_ty);
2251 let (dst_size, dst_align) = self.types.size_align(dst_opts, dst_element_ty);
2252
2253 match src {
2258 Source::Stack(s) => {
2259 assert_eq!(s.locals.len(), 2);
2260 self.stack_get(&s.slice(0..1), src_opts.ptr());
2261 self.stack_get(&s.slice(1..2), src_opts.ptr());
2262 }
2263 Source::Memory(mem) => {
2264 self.ptr_load(mem);
2265 self.ptr_load(&mem.bump(src_opts.ptr_size().into()));
2266 }
2267 }
2268 let src_len = self.local_set_new_tmp(src_opts.ptr());
2269 let src_ptr = self.local_set_new_tmp(src_opts.ptr());
2270
2271 let src_mem = self.memory_operand(src_opts, src_ptr, src_align);
2274
2275 let src_byte_len = self.calculate_list_byte_len(src_opts, src_len.idx, src_size);
2277 let dst_byte_len = if src_size == dst_size {
2278 self.convert_src_len_to_dst(src_byte_len.idx, src_opts.ptr(), dst_opts.ptr());
2279 self.local_set_new_tmp(dst_opts.ptr())
2280 } else if src_opts.ptr() == dst_opts.ptr() {
2281 self.calculate_list_byte_len(dst_opts, src_len.idx, dst_size)
2282 } else {
2283 self.convert_src_len_to_dst(src_byte_len.idx, src_opts.ptr(), dst_opts.ptr());
2284 let tmp = self.local_set_new_tmp(dst_opts.ptr());
2285 let ret = self.calculate_list_byte_len(dst_opts, tmp.idx, dst_size);
2286 self.free_temp_local(tmp);
2287 ret
2288 };
2289
2290 let dst_mem = self.malloc(dst_opts, MallocSize::Local(dst_byte_len.idx), dst_align);
2295
2296 self.validate_memory_inbounds(
2299 src_opts,
2300 src_mem.addr.idx,
2301 src_byte_len.idx,
2302 Trap::ListByteLengthOverflow,
2303 );
2304 self.validate_memory_inbounds(
2305 dst_opts,
2306 dst_mem.addr.idx,
2307 dst_byte_len.idx,
2308 Trap::ListByteLengthOverflow,
2309 );
2310
2311 self.free_temp_local(src_byte_len);
2312 self.free_temp_local(dst_byte_len);
2313
2314 if src_size > 0 || dst_size > 0 {
2318 self.instruction(Block(BlockType::Empty));
2321
2322 self.instruction(LocalGet(src_len.idx));
2324 let remaining = self.local_tee_new_tmp(src_opts.ptr());
2325 self.ptr_eqz(src_opts);
2326 self.instruction(BrIf(0));
2327
2328 self.instruction(LocalGet(src_mem.addr.idx));
2330 let cur_src_ptr = self.local_set_new_tmp(src_opts.ptr());
2331 self.instruction(LocalGet(dst_mem.addr.idx));
2332 let cur_dst_ptr = self.local_set_new_tmp(dst_opts.ptr());
2333
2334 self.instruction(Loop(BlockType::Empty));
2335
2336 let element_src = Source::Memory(Memory {
2338 opts: src_opts,
2339 offset: 0,
2340 addr: TempLocal::new(cur_src_ptr.idx, cur_src_ptr.ty),
2341 });
2342 let element_dst = Destination::Memory(Memory {
2343 opts: dst_opts,
2344 offset: 0,
2345 addr: TempLocal::new(cur_dst_ptr.idx, cur_dst_ptr.ty),
2346 });
2347 self.translate(src_element_ty, &element_src, dst_element_ty, &element_dst);
2348
2349 if src_size > 0 {
2351 self.instruction(LocalGet(cur_src_ptr.idx));
2352 self.ptr_uconst(src_opts, src_size);
2353 self.ptr_add(src_opts);
2354 self.instruction(LocalSet(cur_src_ptr.idx));
2355 }
2356 if dst_size > 0 {
2357 self.instruction(LocalGet(cur_dst_ptr.idx));
2358 self.ptr_uconst(dst_opts, dst_size);
2359 self.ptr_add(dst_opts);
2360 self.instruction(LocalSet(cur_dst_ptr.idx));
2361 }
2362
2363 self.instruction(LocalGet(remaining.idx));
2366 self.ptr_iconst(src_opts, -1);
2367 self.ptr_add(src_opts);
2368 self.instruction(LocalTee(remaining.idx));
2369 self.ptr_br_if(src_opts, 0);
2370 self.instruction(End); self.instruction(End); self.free_temp_local(cur_dst_ptr);
2374 self.free_temp_local(cur_src_ptr);
2375 self.free_temp_local(remaining);
2376 }
2377
2378 match dst {
2380 Destination::Stack(s, _) => {
2381 self.instruction(LocalGet(dst_mem.addr.idx));
2382 self.stack_set(&s[..1], dst_opts.ptr());
2383 self.convert_src_len_to_dst(src_len.idx, src_opts.ptr(), dst_opts.ptr());
2384 self.stack_set(&s[1..], dst_opts.ptr());
2385 }
2386 Destination::Memory(mem) => {
2387 self.instruction(LocalGet(mem.addr.idx));
2388 self.instruction(LocalGet(dst_mem.addr.idx));
2389 self.ptr_store(mem);
2390 self.instruction(LocalGet(mem.addr.idx));
2391 self.convert_src_len_to_dst(src_len.idx, src_opts.ptr(), dst_opts.ptr());
2392 self.ptr_store(&mem.bump(dst_opts.ptr_size().into()));
2393 }
2394 }
2395
2396 self.free_temp_local(src_len);
2397 self.free_temp_local(src_mem.addr);
2398 self.free_temp_local(dst_mem.addr);
2399 }
2400
2401 fn calculate_list_byte_len(
2402 &mut self,
2403 opts: &Options,
2404 len_local: u32,
2405 elt_size: u32,
2406 ) -> TempLocal {
2407 if elt_size == 0 {
2410 self.ptr_uconst(opts, 0);
2411 return self.local_set_new_tmp(opts.ptr());
2412 }
2413
2414 if elt_size == 1 {
2422 if let ValType::I64 = opts.ptr() {
2423 self.instruction(LocalGet(len_local));
2424 self.instruction(I64Const(32));
2425 self.instruction(I64ShrU);
2426 self.instruction(I32WrapI64);
2427 self.instruction(If(BlockType::Empty));
2428 self.trap(Trap::ListByteLengthOverflow);
2429 self.instruction(End);
2430 }
2431 self.instruction(LocalGet(len_local));
2432 return self.local_set_new_tmp(opts.ptr());
2433 }
2434
2435 self.instruction(Block(BlockType::Empty));
2440 self.instruction(Block(BlockType::Empty));
2441 self.instruction(LocalGet(len_local));
2442 match opts.ptr() {
2443 ValType::I32 => self.instruction(I64ExtendI32U),
2447
2448 ValType::I64 => {
2452 self.instruction(I64Const(32));
2453 self.instruction(I64ShrU);
2454 self.instruction(I32WrapI64);
2455 self.instruction(BrIf(0));
2456 self.instruction(LocalGet(len_local));
2457 }
2458
2459 _ => unreachable!(),
2460 }
2461
2462 self.instruction(I64Const(elt_size.into()));
2471 self.instruction(I64Mul);
2472 let tmp = self.local_tee_new_tmp(ValType::I64);
2473 self.instruction(I64Const(32));
2476 self.instruction(I64ShrU);
2477 self.instruction(I64Eqz);
2478 self.instruction(BrIf(1));
2479 self.instruction(End);
2480 self.trap(Trap::ListByteLengthOverflow);
2481 self.instruction(End);
2482
2483 if opts.ptr() == ValType::I64 {
2487 tmp
2488 } else {
2489 self.instruction(LocalGet(tmp.idx));
2490 self.instruction(I32WrapI64);
2491 self.free_temp_local(tmp);
2492 self.local_set_new_tmp(ValType::I32)
2493 }
2494 }
2495
2496 fn convert_src_len_to_dst(
2497 &mut self,
2498 src_len_local: u32,
2499 src_ptr_ty: ValType,
2500 dst_ptr_ty: ValType,
2501 ) {
2502 self.instruction(LocalGet(src_len_local));
2503 match (src_ptr_ty, dst_ptr_ty) {
2504 (ValType::I32, ValType::I64) => self.instruction(I64ExtendI32U),
2505 (ValType::I64, ValType::I32) => self.instruction(I32WrapI64),
2506 (src, dst) => assert_eq!(src, dst),
2507 }
2508 }
2509
2510 fn translate_record(
2511 &mut self,
2512 src_ty: TypeRecordIndex,
2513 src: &Source<'_>,
2514 dst_ty: &InterfaceType,
2515 dst: &Destination,
2516 ) {
2517 let src_ty = &self.types[src_ty];
2518 let dst_ty = match dst_ty {
2519 InterfaceType::Record(r) => &self.types[*r],
2520 _ => panic!("expected a record"),
2521 };
2522
2523 assert_eq!(src_ty.fields.len(), dst_ty.fields.len());
2525
2526 let mut src_fields = HashMap::new();
2530 for (i, src) in src
2531 .record_field_srcs(self.types, src_ty.fields.iter().map(|f| f.ty))
2532 .enumerate()
2533 {
2534 let field = &src_ty.fields[i];
2535 src_fields.insert(&field.name, (src, &field.ty));
2536 }
2537
2538 for (i, dst) in dst
2547 .record_field_dsts(self.types, dst_ty.fields.iter().map(|f| f.ty))
2548 .enumerate()
2549 {
2550 let field = &dst_ty.fields[i];
2551 let (src, src_ty) = &src_fields[&field.name];
2552 self.translate(src_ty, src, &field.ty, &dst);
2553 }
2554 }
2555
2556 fn translate_flags(
2557 &mut self,
2558 src_ty: TypeFlagsIndex,
2559 src: &Source<'_>,
2560 dst_ty: &InterfaceType,
2561 dst: &Destination,
2562 ) {
2563 let src_ty = &self.types[src_ty];
2564 let dst_ty = match dst_ty {
2565 InterfaceType::Flags(r) => &self.types[*r],
2566 _ => panic!("expected a record"),
2567 };
2568
2569 assert_eq!(src_ty.names, dst_ty.names);
2577 let cnt = src_ty.names.len();
2578 match FlagsSize::from_count(cnt) {
2579 FlagsSize::Size0 => {}
2580 FlagsSize::Size1 => {
2581 let mask = if cnt == 8 { 0xff } else { (1 << cnt) - 1 };
2582 self.convert_u8_mask(src, dst, mask);
2583 }
2584 FlagsSize::Size2 => {
2585 let mask = if cnt == 16 { 0xffff } else { (1 << cnt) - 1 };
2586 self.convert_u16_mask(src, dst, mask);
2587 }
2588 FlagsSize::Size4Plus(n) => {
2589 let srcs = src.record_field_srcs(self.types, (0..n).map(|_| InterfaceType::U32));
2590 let dsts = dst.record_field_dsts(self.types, (0..n).map(|_| InterfaceType::U32));
2591 let n = usize::from(n);
2592 for (i, (src, dst)) in srcs.zip(dsts).enumerate() {
2593 let mask = if i == n - 1 && (cnt % 32 != 0) {
2594 (1 << (cnt % 32)) - 1
2595 } else {
2596 0xffffffff
2597 };
2598 self.convert_u32_mask(&src, &dst, mask);
2599 }
2600 }
2601 }
2602 }
2603
2604 fn translate_tuple(
2605 &mut self,
2606 src_ty: TypeTupleIndex,
2607 src: &Source<'_>,
2608 dst_ty: &InterfaceType,
2609 dst: &Destination,
2610 ) {
2611 let src_ty = &self.types[src_ty];
2612 let dst_ty = match dst_ty {
2613 InterfaceType::Tuple(t) => &self.types[*t],
2614 _ => panic!("expected a tuple"),
2615 };
2616
2617 assert_eq!(src_ty.types.len(), dst_ty.types.len());
2619
2620 let srcs = src
2621 .record_field_srcs(self.types, src_ty.types.iter().copied())
2622 .zip(src_ty.types.iter());
2623 let dsts = dst
2624 .record_field_dsts(self.types, dst_ty.types.iter().copied())
2625 .zip(dst_ty.types.iter());
2626 for ((src, src_ty), (dst, dst_ty)) in srcs.zip(dsts) {
2627 self.translate(src_ty, &src, dst_ty, &dst);
2628 }
2629 }
2630
2631 fn translate_variant(
2632 &mut self,
2633 src_ty: TypeVariantIndex,
2634 src: &Source<'_>,
2635 dst_ty: &InterfaceType,
2636 dst: &Destination,
2637 ) {
2638 let src_ty = &self.types[src_ty];
2639 let dst_ty = match dst_ty {
2640 InterfaceType::Variant(t) => &self.types[*t],
2641 _ => panic!("expected a variant"),
2642 };
2643
2644 let src_info = variant_info(self.types, src_ty.cases.iter().map(|(_, c)| c.as_ref()));
2645 let dst_info = variant_info(self.types, dst_ty.cases.iter().map(|(_, c)| c.as_ref()));
2646
2647 let iter = src_ty
2648 .cases
2649 .iter()
2650 .enumerate()
2651 .map(|(src_i, (src_case, src_case_ty))| {
2652 let dst_i = dst_ty
2653 .cases
2654 .iter()
2655 .position(|(c, _)| c == src_case)
2656 .unwrap();
2657 let dst_case_ty = &dst_ty.cases[dst_i];
2658 let src_i = u32::try_from(src_i).unwrap();
2659 let dst_i = u32::try_from(dst_i).unwrap();
2660 VariantCase {
2661 src_i,
2662 src_ty: src_case_ty.as_ref(),
2663 dst_i,
2664 dst_ty: dst_case_ty.as_ref(),
2665 }
2666 });
2667 self.convert_variant(src, &src_info, dst, &dst_info, iter);
2668 }
2669
2670 fn translate_enum(
2671 &mut self,
2672 src_ty: TypeEnumIndex,
2673 src: &Source<'_>,
2674 dst_ty: &InterfaceType,
2675 dst: &Destination,
2676 ) {
2677 let src_ty = &self.types[src_ty];
2678 let dst_ty = match dst_ty {
2679 InterfaceType::Enum(t) => &self.types[*t],
2680 _ => panic!("expected an option"),
2681 };
2682 let src_info = variant_info(self.types, src_ty.names.iter().map(|_| None));
2683 let dst_info = variant_info(self.types, dst_ty.names.iter().map(|_| None));
2684
2685 self.convert_variant(
2686 src,
2687 &src_info,
2688 dst,
2689 &dst_info,
2690 src_ty.names.iter().enumerate().map(|(src_i, src_name)| {
2691 let dst_i = dst_ty.names.iter().position(|n| n == src_name).unwrap();
2692 let src_i = u32::try_from(src_i).unwrap();
2693 let dst_i = u32::try_from(dst_i).unwrap();
2694 VariantCase {
2695 src_i,
2696 dst_i,
2697 src_ty: None,
2698 dst_ty: None,
2699 }
2700 }),
2701 );
2702 }
2703
2704 fn translate_option(
2705 &mut self,
2706 src_ty: TypeOptionIndex,
2707 src: &Source<'_>,
2708 dst_ty: &InterfaceType,
2709 dst: &Destination,
2710 ) {
2711 let src_ty = &self.types[src_ty].ty;
2712 let dst_ty = match dst_ty {
2713 InterfaceType::Option(t) => &self.types[*t].ty,
2714 _ => panic!("expected an option"),
2715 };
2716 let src_ty = Some(src_ty);
2717 let dst_ty = Some(dst_ty);
2718
2719 let src_info = variant_info(self.types, [None, src_ty]);
2720 let dst_info = variant_info(self.types, [None, dst_ty]);
2721
2722 self.convert_variant(
2723 src,
2724 &src_info,
2725 dst,
2726 &dst_info,
2727 [
2728 VariantCase {
2729 src_i: 0,
2730 dst_i: 0,
2731 src_ty: None,
2732 dst_ty: None,
2733 },
2734 VariantCase {
2735 src_i: 1,
2736 dst_i: 1,
2737 src_ty,
2738 dst_ty,
2739 },
2740 ]
2741 .into_iter(),
2742 );
2743 }
2744
2745 fn translate_result(
2746 &mut self,
2747 src_ty: TypeResultIndex,
2748 src: &Source<'_>,
2749 dst_ty: &InterfaceType,
2750 dst: &Destination,
2751 ) {
2752 let src_ty = &self.types[src_ty];
2753 let dst_ty = match dst_ty {
2754 InterfaceType::Result(t) => &self.types[*t],
2755 _ => panic!("expected a result"),
2756 };
2757
2758 let src_info = variant_info(self.types, [src_ty.ok.as_ref(), src_ty.err.as_ref()]);
2759 let dst_info = variant_info(self.types, [dst_ty.ok.as_ref(), dst_ty.err.as_ref()]);
2760
2761 self.convert_variant(
2762 src,
2763 &src_info,
2764 dst,
2765 &dst_info,
2766 [
2767 VariantCase {
2768 src_i: 0,
2769 dst_i: 0,
2770 src_ty: src_ty.ok.as_ref(),
2771 dst_ty: dst_ty.ok.as_ref(),
2772 },
2773 VariantCase {
2774 src_i: 1,
2775 dst_i: 1,
2776 src_ty: src_ty.err.as_ref(),
2777 dst_ty: dst_ty.err.as_ref(),
2778 },
2779 ]
2780 .into_iter(),
2781 );
2782 }
2783
2784 fn convert_variant<'c>(
2785 &mut self,
2786 src: &Source<'_>,
2787 src_info: &VariantInfo,
2788 dst: &Destination,
2789 dst_info: &VariantInfo,
2790 src_cases: impl ExactSizeIterator<Item = VariantCase<'c>>,
2791 ) {
2792 let outer_block_ty = match dst {
2795 Destination::Stack(dst_flat, _) => match dst_flat.len() {
2796 0 => BlockType::Empty,
2797 1 => BlockType::Result(dst_flat[0]),
2798 _ => {
2799 let ty = self.module.core_types.function(&[], &dst_flat);
2800 BlockType::FunctionType(ty)
2801 }
2802 },
2803 Destination::Memory(_) => BlockType::Empty,
2804 };
2805 self.instruction(Block(outer_block_ty));
2806
2807 let src_cases_len = src_cases.len();
2810 for _ in 0..src_cases_len - 1 {
2811 self.instruction(Block(BlockType::Empty));
2812 }
2813
2814 self.instruction(Block(BlockType::Empty));
2816
2817 self.instruction(Block(BlockType::Empty));
2820
2821 match src {
2823 Source::Stack(s) => self.stack_get(&s.slice(0..1), ValType::I32),
2824 Source::Memory(mem) => match src_info.size {
2825 DiscriminantSize::Size1 => self.i32_load8u(mem),
2826 DiscriminantSize::Size2 => self.i32_load16u(mem),
2827 DiscriminantSize::Size4 => self.i32_load(mem),
2828 },
2829 }
2830
2831 let mut targets = Vec::new();
2834 for i in 0..src_cases_len {
2835 targets.push((i + 1) as u32);
2836 }
2837 self.instruction(BrTable(targets[..].into(), 0));
2838 self.instruction(End); self.trap(Trap::InvalidDiscriminant);
2841 self.instruction(End); let src_cases_len = u32::try_from(src_cases_len).unwrap();
2848 for case in src_cases {
2849 let VariantCase {
2850 src_i,
2851 src_ty,
2852 dst_i,
2853 dst_ty,
2854 } = case;
2855
2856 self.push_dst_addr(dst);
2859 self.instruction(I32Const(dst_i as i32));
2860 match dst {
2861 Destination::Stack(stack, _) => self.stack_set(&stack[..1], ValType::I32),
2862 Destination::Memory(mem) => match dst_info.size {
2863 DiscriminantSize::Size1 => self.i32_store8(mem),
2864 DiscriminantSize::Size2 => self.i32_store16(mem),
2865 DiscriminantSize::Size4 => self.i32_store(mem),
2866 },
2867 }
2868
2869 let src_payload = src.payload_src(self.types, src_info, src_ty);
2870 let dst_payload = dst.payload_dst(self.types, dst_info, dst_ty);
2871
2872 match (src_ty, dst_ty) {
2875 (Some(src_ty), Some(dst_ty)) => {
2876 self.translate(src_ty, &src_payload, dst_ty, &dst_payload);
2877 }
2878 (None, None) => {}
2879 _ => unimplemented!(),
2880 }
2881
2882 if let Destination::Stack(payload_results, _) = dst_payload {
2889 if let Destination::Stack(dst_results, _) = dst {
2890 let remaining = &dst_results[1..][payload_results.len()..];
2891 for ty in remaining {
2892 match ty {
2893 ValType::I32 => self.instruction(I32Const(0)),
2894 ValType::I64 => self.instruction(I64Const(0)),
2895 ValType::F32 => self.instruction(F32Const(0.0)),
2896 ValType::F64 => self.instruction(F64Const(0.0)),
2897 _ => unreachable!(),
2898 }
2899 }
2900 }
2901 }
2902
2903 if src_i != src_cases_len - 1 {
2906 self.instruction(Br(src_cases_len - src_i - 1));
2907 }
2908 self.instruction(End); }
2910 }
2911
2912 fn translate_future(
2913 &mut self,
2914 src_ty: TypeFutureTableIndex,
2915 src: &Source<'_>,
2916 dst_ty: &InterfaceType,
2917 dst: &Destination,
2918 ) {
2919 let dst_ty = match dst_ty {
2920 InterfaceType::Future(t) => *t,
2921 _ => panic!("expected a `Future`"),
2922 };
2923 let transfer = self.module.import_future_transfer();
2924 self.translate_handle(src_ty.as_u32(), src, dst_ty.as_u32(), dst, transfer);
2925 }
2926
2927 fn translate_stream(
2928 &mut self,
2929 src_ty: TypeStreamTableIndex,
2930 src: &Source<'_>,
2931 dst_ty: &InterfaceType,
2932 dst: &Destination,
2933 ) {
2934 let dst_ty = match dst_ty {
2935 InterfaceType::Stream(t) => *t,
2936 _ => panic!("expected a `Stream`"),
2937 };
2938 let transfer = self.module.import_stream_transfer();
2939 self.translate_handle(src_ty.as_u32(), src, dst_ty.as_u32(), dst, transfer);
2940 }
2941
2942 fn translate_error_context(
2943 &mut self,
2944 src_ty: TypeComponentLocalErrorContextTableIndex,
2945 src: &Source<'_>,
2946 dst_ty: &InterfaceType,
2947 dst: &Destination,
2948 ) {
2949 let dst_ty = match dst_ty {
2950 InterfaceType::ErrorContext(t) => *t,
2951 _ => panic!("expected an `ErrorContext`"),
2952 };
2953 let transfer = self.module.import_error_context_transfer();
2954 self.translate_handle(src_ty.as_u32(), src, dst_ty.as_u32(), dst, transfer);
2955 }
2956
2957 fn translate_own(
2958 &mut self,
2959 src_ty: TypeResourceTableIndex,
2960 src: &Source<'_>,
2961 dst_ty: &InterfaceType,
2962 dst: &Destination,
2963 ) {
2964 let dst_ty = match dst_ty {
2965 InterfaceType::Own(t) => *t,
2966 _ => panic!("expected an `Own`"),
2967 };
2968 let transfer = self.module.import_resource_transfer_own();
2969 self.translate_handle(src_ty.as_u32(), src, dst_ty.as_u32(), dst, transfer);
2970 }
2971
2972 fn translate_borrow(
2973 &mut self,
2974 src_ty: TypeResourceTableIndex,
2975 src: &Source<'_>,
2976 dst_ty: &InterfaceType,
2977 dst: &Destination,
2978 ) {
2979 let dst_ty = match dst_ty {
2980 InterfaceType::Borrow(t) => *t,
2981 _ => panic!("expected an `Borrow`"),
2982 };
2983
2984 let transfer = self.module.import_resource_transfer_borrow();
2985 self.translate_handle(src_ty.as_u32(), src, dst_ty.as_u32(), dst, transfer);
2986 }
2987
2988 fn translate_handle(
2996 &mut self,
2997 src_ty: u32,
2998 src: &Source<'_>,
2999 dst_ty: u32,
3000 dst: &Destination,
3001 transfer: FuncIndex,
3002 ) {
3003 self.push_dst_addr(dst);
3004 match src {
3005 Source::Memory(mem) => self.i32_load(mem),
3006 Source::Stack(stack) => self.stack_get(stack, ValType::I32),
3007 }
3008 self.instruction(I32Const(src_ty as i32));
3009 self.instruction(I32Const(dst_ty as i32));
3010 self.instruction(Call(transfer.as_u32()));
3011 match dst {
3012 Destination::Memory(mem) => self.i32_store(mem),
3013 Destination::Stack(stack, _) => self.stack_set(stack, ValType::I32),
3014 }
3015 }
3016
3017 fn trap_if_not_flag(&mut self, flags_global: GlobalIndex, flag_to_test: i32, trap: Trap) {
3018 self.instruction(GlobalGet(flags_global.as_u32()));
3019 self.instruction(I32Const(flag_to_test));
3020 self.instruction(I32And);
3021 self.instruction(I32Eqz);
3022 self.instruction(If(BlockType::Empty));
3023 self.trap(trap);
3024 self.instruction(End);
3025 }
3026
3027 fn assert_not_flag(&mut self, flags_global: GlobalIndex, flag_to_test: i32, msg: &'static str) {
3028 self.instruction(GlobalGet(flags_global.as_u32()));
3029 self.instruction(I32Const(flag_to_test));
3030 self.instruction(I32And);
3031 self.instruction(If(BlockType::Empty));
3032 self.trap(Trap::AssertFailed(msg));
3033 self.instruction(End);
3034 }
3035
3036 fn set_flag(&mut self, flags_global: GlobalIndex, flag_to_set: i32, value: bool) {
3037 self.instruction(GlobalGet(flags_global.as_u32()));
3038 if value {
3039 self.instruction(I32Const(flag_to_set));
3040 self.instruction(I32Or);
3041 } else {
3042 self.instruction(I32Const(!flag_to_set));
3043 self.instruction(I32And);
3044 }
3045 self.instruction(GlobalSet(flags_global.as_u32()));
3046 }
3047
3048 fn verify_aligned(&mut self, opts: &Options, addr_local: u32, align: u32) {
3049 if align == 1 {
3052 return;
3053 }
3054 self.instruction(LocalGet(addr_local));
3055 assert!(align.is_power_of_two());
3056 self.ptr_uconst(opts, align - 1);
3057 self.ptr_and(opts);
3058 self.ptr_if(opts, BlockType::Empty);
3059 self.trap(Trap::UnalignedPointer);
3060 self.instruction(End);
3061 }
3062
3063 fn assert_aligned(&mut self, ty: &InterfaceType, mem: &Memory) {
3064 if !self.module.debug {
3065 return;
3066 }
3067 let align = self.types.align(mem.opts, ty);
3068 if align == 1 {
3069 return;
3070 }
3071 assert!(align.is_power_of_two());
3072 self.instruction(LocalGet(mem.addr.idx));
3073 self.ptr_uconst(mem.opts, mem.offset);
3074 self.ptr_add(mem.opts);
3075 self.ptr_uconst(mem.opts, align - 1);
3076 self.ptr_and(mem.opts);
3077 self.ptr_if(mem.opts, BlockType::Empty);
3078 self.trap(Trap::AssertFailed("pointer not aligned"));
3079 self.instruction(End);
3080 }
3081
3082 fn malloc<'c>(&mut self, opts: &'c Options, size: MallocSize, align: u32) -> Memory<'c> {
3083 let realloc = opts.realloc.unwrap();
3084 self.ptr_uconst(opts, 0);
3085 self.ptr_uconst(opts, 0);
3086 self.ptr_uconst(opts, align);
3087 match size {
3088 MallocSize::Const(size) => self.ptr_uconst(opts, size),
3089 MallocSize::Local(idx) => self.instruction(LocalGet(idx)),
3090 }
3091 self.instruction(Call(realloc.as_u32()));
3092 let addr = self.local_set_new_tmp(opts.ptr());
3093 self.memory_operand(opts, addr, align)
3094 }
3095
3096 fn memory_operand<'c>(&mut self, opts: &'c Options, addr: TempLocal, align: u32) -> Memory<'c> {
3097 let ret = Memory {
3098 addr,
3099 offset: 0,
3100 opts,
3101 };
3102 self.verify_aligned(opts, ret.addr.idx, align);
3103 ret
3104 }
3105
3106 fn local_tee_new_tmp(&mut self, ty: ValType) -> TempLocal {
3112 self.gen_temp_local(ty, LocalTee)
3113 }
3114
3115 fn local_set_new_tmp(&mut self, ty: ValType) -> TempLocal {
3118 self.gen_temp_local(ty, LocalSet)
3119 }
3120
3121 fn gen_temp_local(&mut self, ty: ValType, insn: fn(u32) -> Instruction<'static>) -> TempLocal {
3122 if let Some(idx) = self.free_locals.get_mut(&ty).and_then(|v| v.pop()) {
3125 self.instruction(insn(idx));
3126 return TempLocal {
3127 ty,
3128 idx,
3129 needs_free: true,
3130 };
3131 }
3132
3133 let locals = &mut self.module.funcs[self.result].locals;
3135 match locals.last_mut() {
3136 Some((cnt, prev_ty)) if ty == *prev_ty => *cnt += 1,
3137 _ => locals.push((1, ty)),
3138 }
3139 self.nlocals += 1;
3140 let idx = self.nlocals - 1;
3141 self.instruction(insn(idx));
3142 TempLocal {
3143 ty,
3144 idx,
3145 needs_free: true,
3146 }
3147 }
3148
3149 fn free_temp_local(&mut self, mut local: TempLocal) {
3152 assert!(local.needs_free);
3153 self.free_locals
3154 .entry(local.ty)
3155 .or_insert(Vec::new())
3156 .push(local.idx);
3157 local.needs_free = false;
3158 }
3159
3160 fn instruction(&mut self, instr: Instruction) {
3161 instr.encode(&mut self.code);
3162 }
3163
3164 fn trap(&mut self, trap: Trap) {
3165 self.traps.push((self.code.len(), trap));
3166 self.instruction(Unreachable);
3167 }
3168
3169 fn flush_code(&mut self) {
3174 if self.code.is_empty() {
3175 return;
3176 }
3177 self.module.funcs[self.result].body.push(Body::Raw(
3178 mem::take(&mut self.code),
3179 mem::take(&mut self.traps),
3180 ));
3181 }
3182
3183 fn finish(mut self) {
3184 self.instruction(End);
3187 self.flush_code();
3188
3189 self.module.funcs[self.result].filled_in = true;
3192 }
3193
3194 fn stack_get(&mut self, stack: &Stack<'_>, dst_ty: ValType) {
3202 assert_eq!(stack.locals.len(), 1);
3203 let (idx, src_ty) = stack.locals[0];
3204 self.instruction(LocalGet(idx));
3205 match (src_ty, dst_ty) {
3206 (ValType::I32, ValType::I32)
3207 | (ValType::I64, ValType::I64)
3208 | (ValType::F32, ValType::F32)
3209 | (ValType::F64, ValType::F64) => {}
3210
3211 (ValType::I32, ValType::F32) => self.instruction(F32ReinterpretI32),
3212 (ValType::I64, ValType::I32) => {
3213 self.assert_i64_upper_bits_not_set(idx);
3214 self.instruction(I32WrapI64);
3215 }
3216 (ValType::I64, ValType::F64) => self.instruction(F64ReinterpretI64),
3217 (ValType::I64, ValType::F32) => {
3218 self.assert_i64_upper_bits_not_set(idx);
3219 self.instruction(I32WrapI64);
3220 self.instruction(F32ReinterpretI32);
3221 }
3222
3223 (ValType::I32, ValType::I64)
3225 | (ValType::I32, ValType::F64)
3226 | (ValType::F32, ValType::I32)
3227 | (ValType::F32, ValType::I64)
3228 | (ValType::F32, ValType::F64)
3229 | (ValType::F64, ValType::I32)
3230 | (ValType::F64, ValType::I64)
3231 | (ValType::F64, ValType::F32)
3232
3233 | (ValType::Ref(_), _)
3235 | (_, ValType::Ref(_))
3236 | (ValType::V128, _)
3237 | (_, ValType::V128) => {
3238 panic!("cannot get {dst_ty:?} from {src_ty:?} local");
3239 }
3240 }
3241 }
3242
3243 fn assert_i64_upper_bits_not_set(&mut self, local: u32) {
3244 if !self.module.debug {
3245 return;
3246 }
3247 self.instruction(LocalGet(local));
3248 self.instruction(I64Const(32));
3249 self.instruction(I64ShrU);
3250 self.instruction(I32WrapI64);
3251 self.instruction(If(BlockType::Empty));
3252 self.trap(Trap::AssertFailed("upper bits are unexpectedly set"));
3253 self.instruction(End);
3254 }
3255
3256 fn stack_set(&mut self, dst_tys: &[ValType], src_ty: ValType) {
3262 assert_eq!(dst_tys.len(), 1);
3263 let dst_ty = dst_tys[0];
3264 match (src_ty, dst_ty) {
3265 (ValType::I32, ValType::I32)
3266 | (ValType::I64, ValType::I64)
3267 | (ValType::F32, ValType::F32)
3268 | (ValType::F64, ValType::F64) => {}
3269
3270 (ValType::F32, ValType::I32) => self.instruction(I32ReinterpretF32),
3271 (ValType::I32, ValType::I64) => self.instruction(I64ExtendI32U),
3272 (ValType::F64, ValType::I64) => self.instruction(I64ReinterpretF64),
3273 (ValType::F32, ValType::I64) => {
3274 self.instruction(I32ReinterpretF32);
3275 self.instruction(I64ExtendI32U);
3276 }
3277
3278 (ValType::I64, ValType::I32)
3280 | (ValType::F64, ValType::I32)
3281 | (ValType::I32, ValType::F32)
3282 | (ValType::I64, ValType::F32)
3283 | (ValType::F64, ValType::F32)
3284 | (ValType::I32, ValType::F64)
3285 | (ValType::I64, ValType::F64)
3286 | (ValType::F32, ValType::F64)
3287
3288 | (ValType::Ref(_), _)
3290 | (_, ValType::Ref(_))
3291 | (ValType::V128, _)
3292 | (_, ValType::V128) => {
3293 panic!("cannot get {dst_ty:?} from {src_ty:?} local");
3294 }
3295 }
3296 }
3297
3298 fn i32_load8u(&mut self, mem: &Memory) {
3299 self.instruction(LocalGet(mem.addr.idx));
3300 self.instruction(I32Load8U(mem.memarg(0)));
3301 }
3302
3303 fn i32_load8s(&mut self, mem: &Memory) {
3304 self.instruction(LocalGet(mem.addr.idx));
3305 self.instruction(I32Load8S(mem.memarg(0)));
3306 }
3307
3308 fn i32_load16u(&mut self, mem: &Memory) {
3309 self.instruction(LocalGet(mem.addr.idx));
3310 self.instruction(I32Load16U(mem.memarg(1)));
3311 }
3312
3313 fn i32_load16s(&mut self, mem: &Memory) {
3314 self.instruction(LocalGet(mem.addr.idx));
3315 self.instruction(I32Load16S(mem.memarg(1)));
3316 }
3317
3318 fn i32_load(&mut self, mem: &Memory) {
3319 self.instruction(LocalGet(mem.addr.idx));
3320 self.instruction(I32Load(mem.memarg(2)));
3321 }
3322
3323 fn i64_load(&mut self, mem: &Memory) {
3324 self.instruction(LocalGet(mem.addr.idx));
3325 self.instruction(I64Load(mem.memarg(3)));
3326 }
3327
3328 fn ptr_load(&mut self, mem: &Memory) {
3329 if mem.opts.memory64 {
3330 self.i64_load(mem);
3331 } else {
3332 self.i32_load(mem);
3333 }
3334 }
3335
3336 fn ptr_add(&mut self, opts: &Options) {
3337 if opts.memory64 {
3338 self.instruction(I64Add);
3339 } else {
3340 self.instruction(I32Add);
3341 }
3342 }
3343
3344 fn ptr_sub(&mut self, opts: &Options) {
3345 if opts.memory64 {
3346 self.instruction(I64Sub);
3347 } else {
3348 self.instruction(I32Sub);
3349 }
3350 }
3351
3352 fn ptr_mul(&mut self, opts: &Options) {
3353 if opts.memory64 {
3354 self.instruction(I64Mul);
3355 } else {
3356 self.instruction(I32Mul);
3357 }
3358 }
3359
3360 fn ptr_ge_u(&mut self, opts: &Options) {
3361 if opts.memory64 {
3362 self.instruction(I64GeU);
3363 } else {
3364 self.instruction(I32GeU);
3365 }
3366 }
3367
3368 fn ptr_lt_u(&mut self, opts: &Options) {
3369 if opts.memory64 {
3370 self.instruction(I64LtU);
3371 } else {
3372 self.instruction(I32LtU);
3373 }
3374 }
3375
3376 fn ptr_shl(&mut self, opts: &Options) {
3377 if opts.memory64 {
3378 self.instruction(I64Shl);
3379 } else {
3380 self.instruction(I32Shl);
3381 }
3382 }
3383
3384 fn ptr_eqz(&mut self, opts: &Options) {
3385 if opts.memory64 {
3386 self.instruction(I64Eqz);
3387 } else {
3388 self.instruction(I32Eqz);
3389 }
3390 }
3391
3392 fn ptr_uconst(&mut self, opts: &Options, val: u32) {
3393 if opts.memory64 {
3394 self.instruction(I64Const(val.into()));
3395 } else {
3396 self.instruction(I32Const(val as i32));
3397 }
3398 }
3399
3400 fn ptr_iconst(&mut self, opts: &Options, val: i32) {
3401 if opts.memory64 {
3402 self.instruction(I64Const(val.into()));
3403 } else {
3404 self.instruction(I32Const(val));
3405 }
3406 }
3407
3408 fn ptr_eq(&mut self, opts: &Options) {
3409 if opts.memory64 {
3410 self.instruction(I64Eq);
3411 } else {
3412 self.instruction(I32Eq);
3413 }
3414 }
3415
3416 fn ptr_ne(&mut self, opts: &Options) {
3417 if opts.memory64 {
3418 self.instruction(I64Ne);
3419 } else {
3420 self.instruction(I32Ne);
3421 }
3422 }
3423
3424 fn ptr_and(&mut self, opts: &Options) {
3425 if opts.memory64 {
3426 self.instruction(I64And);
3427 } else {
3428 self.instruction(I32And);
3429 }
3430 }
3431
3432 fn ptr_or(&mut self, opts: &Options) {
3433 if opts.memory64 {
3434 self.instruction(I64Or);
3435 } else {
3436 self.instruction(I32Or);
3437 }
3438 }
3439
3440 fn ptr_xor(&mut self, opts: &Options) {
3441 if opts.memory64 {
3442 self.instruction(I64Xor);
3443 } else {
3444 self.instruction(I32Xor);
3445 }
3446 }
3447
3448 fn ptr_if(&mut self, opts: &Options, ty: BlockType) {
3449 if opts.memory64 {
3450 self.instruction(I64Const(0));
3451 self.instruction(I64Ne);
3452 }
3453 self.instruction(If(ty));
3454 }
3455
3456 fn ptr_br_if(&mut self, opts: &Options, depth: u32) {
3457 if opts.memory64 {
3458 self.instruction(I64Const(0));
3459 self.instruction(I64Ne);
3460 }
3461 self.instruction(BrIf(depth));
3462 }
3463
3464 fn f32_load(&mut self, mem: &Memory) {
3465 self.instruction(LocalGet(mem.addr.idx));
3466 self.instruction(F32Load(mem.memarg(2)));
3467 }
3468
3469 fn f64_load(&mut self, mem: &Memory) {
3470 self.instruction(LocalGet(mem.addr.idx));
3471 self.instruction(F64Load(mem.memarg(3)));
3472 }
3473
3474 fn push_dst_addr(&mut self, dst: &Destination) {
3475 if let Destination::Memory(mem) = dst {
3476 self.instruction(LocalGet(mem.addr.idx));
3477 }
3478 }
3479
3480 fn i32_store8(&mut self, mem: &Memory) {
3481 self.instruction(I32Store8(mem.memarg(0)));
3482 }
3483
3484 fn i32_store16(&mut self, mem: &Memory) {
3485 self.instruction(I32Store16(mem.memarg(1)));
3486 }
3487
3488 fn i32_store(&mut self, mem: &Memory) {
3489 self.instruction(I32Store(mem.memarg(2)));
3490 }
3491
3492 fn i64_store(&mut self, mem: &Memory) {
3493 self.instruction(I64Store(mem.memarg(3)));
3494 }
3495
3496 fn ptr_store(&mut self, mem: &Memory) {
3497 if mem.opts.memory64 {
3498 self.i64_store(mem);
3499 } else {
3500 self.i32_store(mem);
3501 }
3502 }
3503
3504 fn f32_store(&mut self, mem: &Memory) {
3505 self.instruction(F32Store(mem.memarg(2)));
3506 }
3507
3508 fn f64_store(&mut self, mem: &Memory) {
3509 self.instruction(F64Store(mem.memarg(3)));
3510 }
3511}
3512
3513impl<'a> Source<'a> {
3514 fn record_field_srcs<'b>(
3521 &'b self,
3522 types: &'b ComponentTypesBuilder,
3523 fields: impl IntoIterator<Item = InterfaceType> + 'b,
3524 ) -> impl Iterator<Item = Source<'a>> + 'b
3525 where
3526 'a: 'b,
3527 {
3528 let mut offset = 0;
3529 fields.into_iter().map(move |ty| match self {
3530 Source::Memory(mem) => {
3531 let mem = next_field_offset(&mut offset, types, &ty, mem);
3532 Source::Memory(mem)
3533 }
3534 Source::Stack(stack) => {
3535 let cnt = types.flat_types(&ty).unwrap().len() as u32;
3536 offset += cnt;
3537 Source::Stack(stack.slice((offset - cnt) as usize..offset as usize))
3538 }
3539 })
3540 }
3541
3542 fn payload_src(
3544 &self,
3545 types: &ComponentTypesBuilder,
3546 info: &VariantInfo,
3547 case: Option<&InterfaceType>,
3548 ) -> Source<'a> {
3549 match self {
3550 Source::Stack(s) => {
3551 let flat_len = match case {
3552 Some(case) => types.flat_types(case).unwrap().len(),
3553 None => 0,
3554 };
3555 Source::Stack(s.slice(1..s.locals.len()).slice(0..flat_len))
3556 }
3557 Source::Memory(mem) => {
3558 let mem = if mem.opts.memory64 {
3559 mem.bump(info.payload_offset64)
3560 } else {
3561 mem.bump(info.payload_offset32)
3562 };
3563 Source::Memory(mem)
3564 }
3565 }
3566 }
3567
3568 fn opts(&self) -> &'a Options {
3569 match self {
3570 Source::Stack(s) => s.opts,
3571 Source::Memory(mem) => mem.opts,
3572 }
3573 }
3574}
3575
3576impl<'a> Destination<'a> {
3577 fn record_field_dsts<'b, I>(
3579 &'b self,
3580 types: &'b ComponentTypesBuilder,
3581 fields: I,
3582 ) -> impl Iterator<Item = Destination<'b>> + use<'b, I>
3583 where
3584 'a: 'b,
3585 I: IntoIterator<Item = InterfaceType> + 'b,
3586 {
3587 let mut offset = 0;
3588 fields.into_iter().map(move |ty| match self {
3589 Destination::Memory(mem) => {
3590 let mem = next_field_offset(&mut offset, types, &ty, mem);
3591 Destination::Memory(mem)
3592 }
3593 Destination::Stack(s, opts) => {
3594 let cnt = types.flat_types(&ty).unwrap().len() as u32;
3595 offset += cnt;
3596 Destination::Stack(&s[(offset - cnt) as usize..offset as usize], opts)
3597 }
3598 })
3599 }
3600
3601 fn payload_dst(
3603 &self,
3604 types: &ComponentTypesBuilder,
3605 info: &VariantInfo,
3606 case: Option<&InterfaceType>,
3607 ) -> Destination {
3608 match self {
3609 Destination::Stack(s, opts) => {
3610 let flat_len = match case {
3611 Some(case) => types.flat_types(case).unwrap().len(),
3612 None => 0,
3613 };
3614 Destination::Stack(&s[1..][..flat_len], opts)
3615 }
3616 Destination::Memory(mem) => {
3617 let mem = if mem.opts.memory64 {
3618 mem.bump(info.payload_offset64)
3619 } else {
3620 mem.bump(info.payload_offset32)
3621 };
3622 Destination::Memory(mem)
3623 }
3624 }
3625 }
3626
3627 fn opts(&self) -> &'a Options {
3628 match self {
3629 Destination::Stack(_, opts) => opts,
3630 Destination::Memory(mem) => mem.opts,
3631 }
3632 }
3633}
3634
3635fn next_field_offset<'a>(
3636 offset: &mut u32,
3637 types: &ComponentTypesBuilder,
3638 field: &InterfaceType,
3639 mem: &Memory<'a>,
3640) -> Memory<'a> {
3641 let abi = types.canonical_abi(field);
3642 let offset = if mem.opts.memory64 {
3643 abi.next_field64(offset)
3644 } else {
3645 abi.next_field32(offset)
3646 };
3647 mem.bump(offset)
3648}
3649
3650impl<'a> Memory<'a> {
3651 fn memarg(&self, align: u32) -> MemArg {
3652 MemArg {
3653 offset: u64::from(self.offset),
3654 align,
3655 memory_index: self.opts.memory.unwrap().as_u32(),
3656 }
3657 }
3658
3659 fn bump(&self, offset: u32) -> Memory<'a> {
3660 Memory {
3661 opts: self.opts,
3662 addr: TempLocal::new(self.addr.idx, self.addr.ty),
3663 offset: self.offset + offset,
3664 }
3665 }
3666}
3667
3668impl<'a> Stack<'a> {
3669 fn slice(&self, range: Range<usize>) -> Stack<'a> {
3670 Stack {
3671 locals: &self.locals[range],
3672 opts: self.opts,
3673 }
3674 }
3675}
3676
3677struct VariantCase<'a> {
3678 src_i: u32,
3679 src_ty: Option<&'a InterfaceType>,
3680 dst_i: u32,
3681 dst_ty: Option<&'a InterfaceType>,
3682}
3683
3684fn variant_info<'a, I>(types: &ComponentTypesBuilder, cases: I) -> VariantInfo
3685where
3686 I: IntoIterator<Item = Option<&'a InterfaceType>>,
3687 I::IntoIter: ExactSizeIterator,
3688{
3689 VariantInfo::new(
3690 cases
3691 .into_iter()
3692 .map(|ty| ty.map(|ty| types.canonical_abi(ty))),
3693 )
3694 .0
3695}
3696
3697enum MallocSize {
3698 Const(u32),
3699 Local(u32),
3700}
3701
3702struct WasmString<'a> {
3703 ptr: TempLocal,
3704 len: TempLocal,
3705 opts: &'a Options,
3706}
3707
3708struct TempLocal {
3709 idx: u32,
3710 ty: ValType,
3711 needs_free: bool,
3712}
3713
3714impl TempLocal {
3715 fn new(idx: u32, ty: ValType) -> TempLocal {
3716 TempLocal {
3717 idx,
3718 ty,
3719 needs_free: false,
3720 }
3721 }
3722}
3723
3724impl std::ops::Drop for TempLocal {
3725 fn drop(&mut self) {
3726 if self.needs_free {
3727 panic!("temporary local not free'd");
3728 }
3729 }
3730}
3731
3732impl From<FlatType> for ValType {
3733 fn from(ty: FlatType) -> ValType {
3734 match ty {
3735 FlatType::I32 => ValType::I32,
3736 FlatType::I64 => ValType::I64,
3737 FlatType::F32 => ValType::F32,
3738 FlatType::F64 => ValType::F64,
3739 }
3740 }
3741}