1use crate::stack::{Stack, pop, push};
7use crate::value::Value;
8use std::sync::Arc;
9
10#[unsafe(no_mangle)]
17pub unsafe extern "C" fn patch_seq_variant_field_count(stack: Stack) -> Stack {
18 unsafe {
19 let (stack, value) = pop(stack);
20
21 match value {
22 Value::Variant(variant_data) => {
23 let count = variant_data.fields.len() as i64;
24 push(stack, Value::Int(count))
25 }
26 _ => panic!("variant-field-count: expected Variant, got {:?}", value),
27 }
28 }
29}
30
31#[unsafe(no_mangle)]
38pub unsafe extern "C" fn patch_seq_variant_tag(stack: Stack) -> Stack {
39 unsafe {
40 let (stack, value) = pop(stack);
41
42 match value {
43 Value::Variant(variant_data) => {
44 push(stack, Value::Symbol(variant_data.tag.clone()))
46 }
47 _ => panic!("variant-tag: expected Variant, got {:?}", value),
48 }
49 }
50}
51
52#[unsafe(no_mangle)]
64pub unsafe extern "C" fn patch_seq_symbol_eq_cstr(stack: Stack, c_str: *const i8) -> Stack {
65 use std::ffi::CStr;
66
67 unsafe {
68 let (stack, value) = pop(stack);
69 let symbol_str = match value {
70 Value::Symbol(s) => s,
71 _ => panic!("symbol_eq_cstr: expected Symbol, got {:?}", value),
72 };
73
74 let expected = CStr::from_ptr(c_str)
75 .to_str()
76 .expect("Invalid UTF-8 in variant name");
77
78 let is_equal = symbol_str.as_str() == expected;
79 push(stack, Value::Bool(is_equal))
80 }
81}
82
83#[unsafe(no_mangle)]
94pub unsafe extern "C" fn patch_seq_variant_field_at(stack: Stack) -> Stack {
95 unsafe {
96 let (stack, index_val) = pop(stack);
97 let index = match index_val {
98 Value::Int(i) => i,
99 _ => panic!(
100 "variant-field-at: expected Int (index), got {:?}",
101 index_val
102 ),
103 };
104
105 if index < 0 {
106 panic!("variant-field-at: index cannot be negative: {}", index);
107 }
108
109 let (stack, variant_val) = pop(stack);
110
111 match variant_val {
112 Value::Variant(variant_data) => {
113 let idx = index as usize;
114 if idx >= variant_data.fields.len() {
115 panic!(
116 "variant-field-at: index {} out of bounds (variant has {} fields)",
117 index,
118 variant_data.fields.len()
119 );
120 }
121
122 let field = variant_data.fields[idx].clone();
124 push(stack, field)
125 }
126 _ => panic!("variant-field-at: expected Variant, got {:?}", variant_val),
127 }
128 }
129}
130
131#[unsafe(no_mangle)]
143pub unsafe extern "C" fn patch_seq_make_variant_0(stack: Stack) -> Stack {
144 use crate::value::VariantData;
145
146 unsafe {
147 let (stack, tag_val) = pop(stack);
148 let tag = match tag_val {
149 Value::Symbol(s) => s,
150 _ => panic!("make-variant-0: expected Symbol (tag), got {:?}", tag_val),
151 };
152
153 let variant = Value::Variant(Arc::new(VariantData::new(tag, vec![])));
154 push(stack, variant)
155 }
156}
157
158#[unsafe(no_mangle)]
165pub unsafe extern "C" fn patch_seq_make_variant_1(stack: Stack) -> Stack {
166 use crate::value::VariantData;
167
168 unsafe {
169 let (stack, tag_val) = pop(stack);
170 let tag = match tag_val {
171 Value::Symbol(s) => s,
172 _ => panic!("make-variant-1: expected Symbol (tag), got {:?}", tag_val),
173 };
174
175 let (stack, field1) = pop(stack);
176 let variant = Value::Variant(Arc::new(VariantData::new(tag, vec![field1])));
177 push(stack, variant)
178 }
179}
180
181#[unsafe(no_mangle)]
188pub unsafe extern "C" fn patch_seq_make_variant_2(stack: Stack) -> Stack {
189 use crate::value::VariantData;
190
191 unsafe {
192 let (stack, tag_val) = pop(stack);
193 let tag = match tag_val {
194 Value::Symbol(s) => s,
195 _ => panic!("make-variant-2: expected Symbol (tag), got {:?}", tag_val),
196 };
197
198 let (stack, field2) = pop(stack);
199 let (stack, field1) = pop(stack);
200 let variant = Value::Variant(Arc::new(VariantData::new(tag, vec![field1, field2])));
201 push(stack, variant)
202 }
203}
204
205#[unsafe(no_mangle)]
212pub unsafe extern "C" fn patch_seq_make_variant_3(stack: Stack) -> Stack {
213 use crate::value::VariantData;
214
215 unsafe {
216 let (stack, tag_val) = pop(stack);
217 let tag = match tag_val {
218 Value::Symbol(s) => s,
219 _ => panic!("make-variant-3: expected Symbol (tag), got {:?}", tag_val),
220 };
221
222 let (stack, field3) = pop(stack);
223 let (stack, field2) = pop(stack);
224 let (stack, field1) = pop(stack);
225 let variant = Value::Variant(Arc::new(VariantData::new(
226 tag,
227 vec![field1, field2, field3],
228 )));
229 push(stack, variant)
230 }
231}
232
233#[unsafe(no_mangle)]
240pub unsafe extern "C" fn patch_seq_make_variant_4(stack: Stack) -> Stack {
241 use crate::value::VariantData;
242
243 unsafe {
244 let (stack, tag_val) = pop(stack);
245 let tag = match tag_val {
246 Value::Symbol(s) => s,
247 _ => panic!("make-variant-4: expected Symbol (tag), got {:?}", tag_val),
248 };
249
250 let (stack, field4) = pop(stack);
251 let (stack, field3) = pop(stack);
252 let (stack, field2) = pop(stack);
253 let (stack, field1) = pop(stack);
254 let variant = Value::Variant(Arc::new(VariantData::new(
255 tag,
256 vec![field1, field2, field3, field4],
257 )));
258 push(stack, variant)
259 }
260}
261
262#[unsafe(no_mangle)]
269pub unsafe extern "C" fn patch_seq_make_variant_5(stack: Stack) -> Stack {
270 use crate::value::VariantData;
271
272 unsafe {
273 let (stack, tag_val) = pop(stack);
274 let tag = match tag_val {
275 Value::Symbol(s) => s,
276 _ => panic!("make-variant-5: expected Symbol (tag), got {:?}", tag_val),
277 };
278
279 let (stack, field5) = pop(stack);
280 let (stack, field4) = pop(stack);
281 let (stack, field3) = pop(stack);
282 let (stack, field2) = pop(stack);
283 let (stack, field1) = pop(stack);
284 let variant = Value::Variant(Arc::new(VariantData::new(
285 tag,
286 vec![field1, field2, field3, field4, field5],
287 )));
288 push(stack, variant)
289 }
290}
291
292#[unsafe(no_mangle)]
299pub unsafe extern "C" fn patch_seq_make_variant_6(stack: Stack) -> Stack {
300 use crate::value::VariantData;
301
302 unsafe {
303 let (stack, tag_val) = pop(stack);
304 let tag = match tag_val {
305 Value::Symbol(s) => s,
306 _ => panic!("make-variant-6: expected Symbol (tag), got {:?}", tag_val),
307 };
308
309 let (stack, field6) = pop(stack);
310 let (stack, field5) = pop(stack);
311 let (stack, field4) = pop(stack);
312 let (stack, field3) = pop(stack);
313 let (stack, field2) = pop(stack);
314 let (stack, field1) = pop(stack);
315 let variant = Value::Variant(Arc::new(VariantData::new(
316 tag,
317 vec![field1, field2, field3, field4, field5, field6],
318 )));
319 push(stack, variant)
320 }
321}
322
323#[unsafe(no_mangle)]
330pub unsafe extern "C" fn patch_seq_make_variant_7(stack: Stack) -> Stack {
331 use crate::value::VariantData;
332
333 unsafe {
334 let (stack, tag_val) = pop(stack);
335 let tag = match tag_val {
336 Value::Symbol(s) => s,
337 _ => panic!("make-variant-7: expected Symbol (tag), got {:?}", tag_val),
338 };
339
340 let (stack, field7) = pop(stack);
341 let (stack, field6) = pop(stack);
342 let (stack, field5) = pop(stack);
343 let (stack, field4) = pop(stack);
344 let (stack, field3) = pop(stack);
345 let (stack, field2) = pop(stack);
346 let (stack, field1) = pop(stack);
347 let variant = Value::Variant(Arc::new(VariantData::new(
348 tag,
349 vec![field1, field2, field3, field4, field5, field6, field7],
350 )));
351 push(stack, variant)
352 }
353}
354
355#[unsafe(no_mangle)]
362pub unsafe extern "C" fn patch_seq_make_variant_8(stack: Stack) -> Stack {
363 use crate::value::VariantData;
364
365 unsafe {
366 let (stack, tag_val) = pop(stack);
367 let tag = match tag_val {
368 Value::Symbol(s) => s,
369 _ => panic!("make-variant-8: expected Symbol (tag), got {:?}", tag_val),
370 };
371
372 let (stack, field8) = pop(stack);
373 let (stack, field7) = pop(stack);
374 let (stack, field6) = pop(stack);
375 let (stack, field5) = pop(stack);
376 let (stack, field4) = pop(stack);
377 let (stack, field3) = pop(stack);
378 let (stack, field2) = pop(stack);
379 let (stack, field1) = pop(stack);
380 let variant = Value::Variant(Arc::new(VariantData::new(
381 tag,
382 vec![
383 field1, field2, field3, field4, field5, field6, field7, field8,
384 ],
385 )));
386 push(stack, variant)
387 }
388}
389
390#[unsafe(no_mangle)]
397pub unsafe extern "C" fn patch_seq_make_variant_9(stack: Stack) -> Stack {
398 use crate::value::VariantData;
399
400 unsafe {
401 let (stack, tag_val) = pop(stack);
402 let tag = match tag_val {
403 Value::Symbol(s) => s,
404 _ => panic!("make-variant-9: expected Symbol (tag), got {:?}", tag_val),
405 };
406
407 let (stack, field9) = pop(stack);
408 let (stack, field8) = pop(stack);
409 let (stack, field7) = pop(stack);
410 let (stack, field6) = pop(stack);
411 let (stack, field5) = pop(stack);
412 let (stack, field4) = pop(stack);
413 let (stack, field3) = pop(stack);
414 let (stack, field2) = pop(stack);
415 let (stack, field1) = pop(stack);
416 let variant = Value::Variant(Arc::new(VariantData::new(
417 tag,
418 vec![
419 field1, field2, field3, field4, field5, field6, field7, field8, field9,
420 ],
421 )));
422 push(stack, variant)
423 }
424}
425
426#[unsafe(no_mangle)]
433pub unsafe extern "C" fn patch_seq_make_variant_10(stack: Stack) -> Stack {
434 use crate::value::VariantData;
435
436 unsafe {
437 let (stack, tag_val) = pop(stack);
438 let tag = match tag_val {
439 Value::Symbol(s) => s,
440 _ => panic!("make-variant-10: expected Symbol (tag), got {:?}", tag_val),
441 };
442
443 let (stack, field10) = pop(stack);
444 let (stack, field9) = pop(stack);
445 let (stack, field8) = pop(stack);
446 let (stack, field7) = pop(stack);
447 let (stack, field6) = pop(stack);
448 let (stack, field5) = pop(stack);
449 let (stack, field4) = pop(stack);
450 let (stack, field3) = pop(stack);
451 let (stack, field2) = pop(stack);
452 let (stack, field1) = pop(stack);
453 let variant = Value::Variant(Arc::new(VariantData::new(
454 tag,
455 vec![
456 field1, field2, field3, field4, field5, field6, field7, field8, field9, field10,
457 ],
458 )));
459 push(stack, variant)
460 }
461}
462
463#[unsafe(no_mangle)]
470pub unsafe extern "C" fn patch_seq_make_variant_11(stack: Stack) -> Stack {
471 use crate::value::VariantData;
472
473 unsafe {
474 let (stack, tag_val) = pop(stack);
475 let tag = match tag_val {
476 Value::Symbol(s) => s,
477 _ => panic!("make-variant-11: expected Symbol (tag), got {:?}", tag_val),
478 };
479
480 let (stack, field11) = pop(stack);
481 let (stack, field10) = pop(stack);
482 let (stack, field9) = pop(stack);
483 let (stack, field8) = pop(stack);
484 let (stack, field7) = pop(stack);
485 let (stack, field6) = pop(stack);
486 let (stack, field5) = pop(stack);
487 let (stack, field4) = pop(stack);
488 let (stack, field3) = pop(stack);
489 let (stack, field2) = pop(stack);
490 let (stack, field1) = pop(stack);
491 let variant = Value::Variant(Arc::new(VariantData::new(
492 tag,
493 vec![
494 field1, field2, field3, field4, field5, field6, field7, field8, field9, field10,
495 field11,
496 ],
497 )));
498 push(stack, variant)
499 }
500}
501
502#[unsafe(no_mangle)]
509pub unsafe extern "C" fn patch_seq_make_variant_12(stack: Stack) -> Stack {
510 use crate::value::VariantData;
511
512 unsafe {
513 let (stack, tag_val) = pop(stack);
514 let tag = match tag_val {
515 Value::Symbol(s) => s,
516 _ => panic!("make-variant-12: expected Symbol (tag), got {:?}", tag_val),
517 };
518
519 let (stack, field12) = pop(stack);
520 let (stack, field11) = pop(stack);
521 let (stack, field10) = pop(stack);
522 let (stack, field9) = pop(stack);
523 let (stack, field8) = pop(stack);
524 let (stack, field7) = pop(stack);
525 let (stack, field6) = pop(stack);
526 let (stack, field5) = pop(stack);
527 let (stack, field4) = pop(stack);
528 let (stack, field3) = pop(stack);
529 let (stack, field2) = pop(stack);
530 let (stack, field1) = pop(stack);
531 let variant = Value::Variant(Arc::new(VariantData::new(
532 tag,
533 vec![
534 field1, field2, field3, field4, field5, field6, field7, field8, field9, field10,
535 field11, field12,
536 ],
537 )));
538 push(stack, variant)
539 }
540}
541
542pub use patch_seq_make_variant_0 as make_variant_0;
544pub use patch_seq_make_variant_1 as make_variant_1;
545pub use patch_seq_make_variant_2 as make_variant_2;
546pub use patch_seq_make_variant_3 as make_variant_3;
547pub use patch_seq_make_variant_4 as make_variant_4;
548pub use patch_seq_make_variant_5 as make_variant_5;
549pub use patch_seq_make_variant_6 as make_variant_6;
550pub use patch_seq_make_variant_7 as make_variant_7;
551pub use patch_seq_make_variant_8 as make_variant_8;
552pub use patch_seq_make_variant_9 as make_variant_9;
553pub use patch_seq_make_variant_10 as make_variant_10;
554pub use patch_seq_make_variant_11 as make_variant_11;
555pub use patch_seq_make_variant_12 as make_variant_12;
556
557#[unsafe(no_mangle)]
571pub unsafe extern "C" fn patch_seq_variant_append(stack: Stack) -> Stack {
572 use crate::value::VariantData;
573
574 unsafe {
575 let (stack, value) = pop(stack);
577
578 let (stack, variant_val) = pop(stack);
580
581 match variant_val {
582 Value::Variant(mut arc) => {
583 if let Some(data) = Arc::get_mut(&mut arc) {
585 data.fields.push(value);
586 push(stack, Value::Variant(arc))
587 } else {
588 let mut new_fields = Vec::with_capacity(arc.fields.len() + 1);
590 new_fields.extend(arc.fields.iter().cloned());
591 new_fields.push(value);
592 let new_variant =
593 Value::Variant(Arc::new(VariantData::new(arc.tag.clone(), new_fields)));
594 push(stack, new_variant)
595 }
596 }
597 _ => panic!("variant-append: expected Variant, got {:?}", variant_val),
598 }
599 }
600}
601
602#[unsafe(no_mangle)]
612pub unsafe extern "C" fn patch_seq_variant_last(stack: Stack) -> Stack {
613 unsafe {
614 let (stack, variant_val) = pop(stack);
615
616 match variant_val {
617 Value::Variant(variant_data) => {
618 if variant_data.fields.is_empty() {
619 panic!("variant-last: variant has no fields");
620 }
621
622 let last = variant_data.fields.last().unwrap().clone();
623 push(stack, last)
624 }
625 _ => panic!("variant-last: expected Variant, got {:?}", variant_val),
626 }
627 }
628}
629
630#[unsafe(no_mangle)]
641pub unsafe extern "C" fn patch_seq_variant_init(stack: Stack) -> Stack {
642 use crate::value::VariantData;
643
644 unsafe {
645 let (stack, variant_val) = pop(stack);
646
647 match variant_val {
648 Value::Variant(variant_data) => {
649 if variant_data.fields.is_empty() {
650 panic!("variant-init: variant has no fields");
651 }
652
653 let new_fields: Vec<Value> =
655 variant_data.fields[..variant_data.fields.len() - 1].to_vec();
656
657 let new_variant = Value::Variant(Arc::new(VariantData::new(
658 variant_data.tag.clone(),
659 new_fields,
660 )));
661
662 push(stack, new_variant)
663 }
664 _ => panic!("variant-init: expected Variant, got {:?}", variant_val),
665 }
666 }
667}
668
669#[unsafe(no_mangle)]
682pub unsafe extern "C" fn patch_seq_unpack_variant(stack: Stack, field_count: i64) -> Stack {
683 unsafe {
684 let (mut stack, variant_val) = pop(stack);
685
686 match variant_val {
687 Value::Variant(variant_data) => {
688 let count = field_count as usize;
689 if count > variant_data.fields.len() {
690 panic!(
691 "unpack-variant: requested {} fields but variant only has {}",
692 count,
693 variant_data.fields.len()
694 );
695 }
696
697 for i in 0..count {
699 stack = push(stack, variant_data.fields[i].clone());
700 }
701
702 stack
703 }
704 _ => panic!("unpack-variant: expected Variant, got {:?}", variant_val),
705 }
706 }
707}
708
709pub use patch_seq_unpack_variant as unpack_variant;
711pub use patch_seq_variant_append as variant_append;
712pub use patch_seq_variant_field_at as variant_field_at;
713pub use patch_seq_variant_field_count as variant_field_count;
714pub use patch_seq_variant_init as variant_init;
715pub use patch_seq_variant_last as variant_last;
716pub use patch_seq_variant_tag as variant_tag;
717
718#[cfg(test)]
719mod tests {
720 use super::*;
721 use crate::seqstring::global_string;
722 use crate::value::VariantData;
723
724 #[test]
725 fn test_variant_field_count() {
726 unsafe {
727 let variant = Value::Variant(Arc::new(VariantData::new(
729 global_string("TestTag".to_string()),
730 vec![Value::Int(10), Value::Int(20), Value::Int(30)],
731 )));
732
733 let stack = crate::stack::alloc_test_stack();
734 let stack = push(stack, variant);
735 let stack = variant_field_count(stack);
736
737 let (_stack, result) = pop(stack);
738 assert_eq!(result, Value::Int(3));
739 }
740 }
741
742 #[test]
743 fn test_variant_tag() {
744 unsafe {
745 let variant = Value::Variant(Arc::new(VariantData::new(
747 global_string("MyTag".to_string()),
748 vec![Value::Int(10)],
749 )));
750
751 let stack = crate::stack::alloc_test_stack();
752 let stack = push(stack, variant);
753 let stack = variant_tag(stack);
754
755 let (_stack, result) = pop(stack);
756 assert_eq!(result, Value::Symbol(global_string("MyTag".to_string())));
757 }
758 }
759
760 #[test]
761 fn test_variant_field_at() {
762 unsafe {
763 let str1 = global_string("hello".to_string());
764 let str2 = global_string("world".to_string());
765
766 let variant = Value::Variant(Arc::new(VariantData::new(
768 global_string("TestTag".to_string()),
769 vec![
770 Value::String(str1.clone()),
771 Value::Int(42),
772 Value::String(str2.clone()),
773 ],
774 )));
775
776 let stack = crate::stack::alloc_test_stack();
778 let stack = push(stack, variant.clone());
779 let stack = push(stack, Value::Int(0));
780 let stack = variant_field_at(stack);
781
782 let (_stack, result) = pop(stack);
783 assert_eq!(result, Value::String(str1.clone()));
784
785 let stack = push(stack, variant.clone());
787 let stack = push(stack, Value::Int(1));
788 let stack = variant_field_at(stack);
789
790 let (_stack, result) = pop(stack);
791 assert_eq!(result, Value::Int(42));
792
793 let stack = push(stack, variant.clone());
795 let stack = push(stack, Value::Int(2));
796 let stack = variant_field_at(stack);
797
798 let (_stack, result) = pop(stack);
799 assert_eq!(result, Value::String(str2));
800 }
801 }
802
803 #[test]
804 fn test_variant_field_count_empty() {
805 unsafe {
806 let variant = Value::Variant(Arc::new(VariantData::new(
808 global_string("Empty".to_string()),
809 vec![],
810 )));
811
812 let stack = crate::stack::alloc_test_stack();
813 let stack = push(stack, variant);
814 let stack = variant_field_count(stack);
815
816 let (_stack, result) = pop(stack);
817 assert_eq!(result, Value::Int(0));
818 }
819 }
820
821 #[test]
822 fn test_make_variant_with_fields() {
823 unsafe {
824 let stack = crate::stack::alloc_test_stack();
827 let stack = push(stack, Value::Int(10)); let stack = push(stack, Value::Int(20)); let stack = push(stack, Value::Int(30)); let stack = push(stack, Value::Symbol(global_string("Tag".to_string()))); let stack = make_variant_3(stack);
833
834 let (_stack, result) = pop(stack);
835
836 match result {
837 Value::Variant(v) => {
838 assert_eq!(v.tag.as_str(), "Tag");
839 assert_eq!(v.fields.len(), 3);
840 assert_eq!(v.fields[0], Value::Int(10));
841 assert_eq!(v.fields[1], Value::Int(20));
842 assert_eq!(v.fields[2], Value::Int(30));
843 }
844 _ => panic!("Expected Variant"),
845 }
846 }
847 }
848
849 #[test]
850 fn test_make_variant_empty() {
851 unsafe {
852 let stack = crate::stack::alloc_test_stack();
855 let stack = push(stack, Value::Symbol(global_string("None".to_string()))); let stack = make_variant_0(stack);
858
859 let (_stack, result) = pop(stack);
860
861 match result {
862 Value::Variant(v) => {
863 assert_eq!(v.tag.as_str(), "None");
864 assert_eq!(v.fields.len(), 0);
865 }
866 _ => panic!("Expected Variant"),
867 }
868 }
869 }
870
871 #[test]
872 fn test_make_variant_with_mixed_types() {
873 unsafe {
874 let s = global_string("hello".to_string());
875
876 let stack = crate::stack::alloc_test_stack();
878 let stack = push(stack, Value::Int(42));
879 let stack = push(stack, Value::String(s.clone()));
880 let stack = push(stack, Value::Float(3.5));
881 let stack = push(stack, Value::Symbol(global_string("Mixed".to_string()))); let stack = make_variant_3(stack);
884
885 let (_stack, result) = pop(stack);
886
887 match result {
888 Value::Variant(v) => {
889 assert_eq!(v.tag.as_str(), "Mixed");
890 assert_eq!(v.fields.len(), 3);
891 assert_eq!(v.fields[0], Value::Int(42));
892 assert_eq!(v.fields[1], Value::String(s));
893 assert_eq!(v.fields[2], Value::Float(3.5));
894 }
895 _ => panic!("Expected Variant"),
896 }
897 }
898 }
899
900 #[test]
901 fn test_variant_append() {
902 unsafe {
903 let stack = crate::stack::alloc_test_stack();
905 let stack = push(stack, Value::Symbol(global_string("Array".to_string()))); let stack = make_variant_0(stack);
907
908 let stack = push(stack, Value::Int(42));
910 let stack = variant_append(stack);
911
912 let (_stack, result) = pop(stack);
914 match result {
915 Value::Variant(v) => {
916 assert_eq!(v.tag.as_str(), "Array");
917 assert_eq!(v.fields.len(), 1);
918 assert_eq!(v.fields[0], Value::Int(42));
919 }
920 _ => panic!("Expected Variant"),
921 }
922 }
923 }
924
925 #[test]
926 fn test_variant_append_multiple() {
927 unsafe {
928 let stack = crate::stack::alloc_test_stack();
930 let stack = push(stack, Value::Symbol(global_string("Object".to_string()))); let stack = make_variant_0(stack);
932
933 let key = global_string("name".to_string());
935 let stack = push(stack, Value::String(key.clone()));
936 let stack = variant_append(stack);
937
938 let val = global_string("John".to_string());
940 let stack = push(stack, Value::String(val.clone()));
941 let stack = variant_append(stack);
942
943 let (_stack, result) = pop(stack);
945 match result {
946 Value::Variant(v) => {
947 assert_eq!(v.tag.as_str(), "Object");
948 assert_eq!(v.fields.len(), 2);
949 assert_eq!(v.fields[0], Value::String(key));
950 assert_eq!(v.fields[1], Value::String(val));
951 }
952 _ => panic!("Expected Variant"),
953 }
954 }
955 }
956
957 #[test]
958 fn test_variant_last() {
959 unsafe {
960 let variant = Value::Variant(Arc::new(VariantData::new(
962 global_string("List".to_string()),
963 vec![Value::Int(10), Value::Int(20), Value::Int(30)],
964 )));
965
966 let stack = crate::stack::alloc_test_stack();
967 let stack = push(stack, variant);
968 let stack = variant_last(stack);
969
970 let (_stack, result) = pop(stack);
971 assert_eq!(result, Value::Int(30));
972 }
973 }
974
975 #[test]
976 fn test_variant_init() {
977 unsafe {
978 let variant = Value::Variant(Arc::new(VariantData::new(
980 global_string("Custom".to_string()),
981 vec![Value::Int(10), Value::Int(20), Value::Int(30)],
982 )));
983
984 let stack = crate::stack::alloc_test_stack();
985 let stack = push(stack, variant);
986 let stack = variant_init(stack);
987
988 let (_stack, result) = pop(stack);
989 match result {
990 Value::Variant(v) => {
991 assert_eq!(v.tag.as_str(), "Custom"); assert_eq!(v.fields.len(), 2);
993 assert_eq!(v.fields[0], Value::Int(10));
994 assert_eq!(v.fields[1], Value::Int(20));
995 }
996 _ => panic!("Expected Variant"),
997 }
998 }
999 }
1000
1001 #[test]
1002 fn test_variant_stack_operations() {
1003 unsafe {
1005 let stack = crate::stack::alloc_test_stack();
1007 let stack = push(stack, Value::Symbol(global_string("Stack".to_string()))); let stack = make_variant_0(stack);
1009
1010 let stack = push(stack, Value::Int(10));
1012 let stack = variant_append(stack);
1013
1014 let stack = push(stack, Value::Int(20));
1016 let stack = variant_append(stack);
1017
1018 let (stack, variant) = pop(stack);
1021 let stack = push(stack, variant.clone());
1022 let stack = push(stack, variant);
1023 let stack = variant_last(stack);
1024 let (stack, top) = pop(stack);
1025 assert_eq!(top, Value::Int(20));
1026
1027 let stack = variant_init(stack);
1029
1030 let (stack, variant) = pop(stack);
1032 let stack = push(stack, variant.clone());
1033 let stack = push(stack, variant);
1034 let stack = variant_last(stack);
1035 let (stack, top) = pop(stack);
1036 assert_eq!(top, Value::Int(10));
1037
1038 let (_stack, result) = pop(stack);
1040 match result {
1041 Value::Variant(v) => {
1042 assert_eq!(v.fields.len(), 1);
1043 assert_eq!(v.fields[0], Value::Int(10));
1044 }
1045 _ => panic!("Expected Variant"),
1046 }
1047 }
1048 }
1049
1050 #[test]
1051 fn test_variant_clone_is_o1() {
1052 let mut variant = Value::Variant(Arc::new(VariantData::new(
1055 global_string("Level0".to_string()),
1056 vec![],
1057 )));
1058
1059 for i in 0..100 {
1061 variant = Value::Variant(Arc::new(VariantData::new(
1062 global_string(format!("Level{}", i)),
1063 vec![variant.clone()],
1064 )));
1065 }
1066
1067 let start = std::time::Instant::now();
1069 for _ in 0..1000 {
1070 let _copy = variant.clone();
1071 }
1072 let elapsed = start.elapsed();
1073
1074 assert!(
1077 elapsed.as_millis() < 10,
1078 "Clone took {:?} - should be O(1) with Arc",
1079 elapsed
1080 );
1081 }
1082
1083 #[test]
1084 fn test_variant_arc_sharing() {
1085 let inner = Value::Variant(Arc::new(VariantData::new(
1087 global_string("Inner".to_string()),
1088 vec![Value::Int(42)],
1089 )));
1090 let outer = Value::Variant(Arc::new(VariantData::new(
1091 global_string("Outer".to_string()),
1092 vec![inner.clone()],
1093 )));
1094
1095 let outer_clone = outer.clone();
1097
1098 match (&outer, &outer_clone) {
1100 (Value::Variant(a), Value::Variant(b)) => {
1101 assert_eq!(a.tag, b.tag);
1104 assert_eq!(a.fields.len(), b.fields.len());
1105 }
1106 _ => panic!("Expected Variants"),
1107 }
1108 }
1109
1110 #[test]
1111 fn test_variant_thread_safe_sharing() {
1112 use std::sync::Arc as StdArc;
1115 use std::thread;
1116
1117 let variant = Value::Variant(Arc::new(VariantData::new(
1118 global_string("ThreadSafe".to_string()),
1119 vec![Value::Int(1), Value::Int(2), Value::Int(3)],
1120 )));
1121
1122 let shared = StdArc::new(variant);
1124
1125 let handles: Vec<_> = (0..4)
1126 .map(|_| {
1127 let v = StdArc::clone(&shared);
1128 thread::spawn(move || {
1129 match &*v {
1131 Value::Variant(data) => {
1132 assert_eq!(data.tag.as_str(), "ThreadSafe");
1133 assert_eq!(data.fields.len(), 3);
1134 }
1135 _ => panic!("Expected Variant"),
1136 }
1137 })
1138 })
1139 .collect();
1140
1141 for h in handles {
1142 h.join().expect("Thread panicked");
1143 }
1144 }
1145}