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(variant_data) => {
583 let mut new_fields = variant_data.fields.to_vec();
585 new_fields.push(value);
586
587 let new_variant = Value::Variant(Arc::new(VariantData::new(
589 variant_data.tag.clone(),
590 new_fields,
591 )));
592
593 push(stack, new_variant)
594 }
595 _ => panic!("variant-append: expected Variant, got {:?}", variant_val),
596 }
597 }
598}
599
600#[unsafe(no_mangle)]
610pub unsafe extern "C" fn patch_seq_variant_last(stack: Stack) -> Stack {
611 unsafe {
612 let (stack, variant_val) = pop(stack);
613
614 match variant_val {
615 Value::Variant(variant_data) => {
616 if variant_data.fields.is_empty() {
617 panic!("variant-last: variant has no fields");
618 }
619
620 let last = variant_data.fields.last().unwrap().clone();
621 push(stack, last)
622 }
623 _ => panic!("variant-last: expected Variant, got {:?}", variant_val),
624 }
625 }
626}
627
628#[unsafe(no_mangle)]
639pub unsafe extern "C" fn patch_seq_variant_init(stack: Stack) -> Stack {
640 use crate::value::VariantData;
641
642 unsafe {
643 let (stack, variant_val) = pop(stack);
644
645 match variant_val {
646 Value::Variant(variant_data) => {
647 if variant_data.fields.is_empty() {
648 panic!("variant-init: variant has no fields");
649 }
650
651 let new_fields: Vec<Value> =
653 variant_data.fields[..variant_data.fields.len() - 1].to_vec();
654
655 let new_variant = Value::Variant(Arc::new(VariantData::new(
656 variant_data.tag.clone(),
657 new_fields,
658 )));
659
660 push(stack, new_variant)
661 }
662 _ => panic!("variant-init: expected Variant, got {:?}", variant_val),
663 }
664 }
665}
666
667#[unsafe(no_mangle)]
680pub unsafe extern "C" fn patch_seq_unpack_variant(stack: Stack, field_count: i64) -> Stack {
681 unsafe {
682 let (mut stack, variant_val) = pop(stack);
683
684 match variant_val {
685 Value::Variant(variant_data) => {
686 let count = field_count as usize;
687 if count > variant_data.fields.len() {
688 panic!(
689 "unpack-variant: requested {} fields but variant only has {}",
690 count,
691 variant_data.fields.len()
692 );
693 }
694
695 for i in 0..count {
697 stack = push(stack, variant_data.fields[i].clone());
698 }
699
700 stack
701 }
702 _ => panic!("unpack-variant: expected Variant, got {:?}", variant_val),
703 }
704 }
705}
706
707pub use patch_seq_unpack_variant as unpack_variant;
709pub use patch_seq_variant_append as variant_append;
710pub use patch_seq_variant_field_at as variant_field_at;
711pub use patch_seq_variant_field_count as variant_field_count;
712pub use patch_seq_variant_init as variant_init;
713pub use patch_seq_variant_last as variant_last;
714pub use patch_seq_variant_tag as variant_tag;
715
716#[cfg(test)]
717mod tests {
718 use super::*;
719 use crate::seqstring::global_string;
720 use crate::value::VariantData;
721
722 #[test]
723 fn test_variant_field_count() {
724 unsafe {
725 let variant = Value::Variant(Arc::new(VariantData::new(
727 global_string("TestTag".to_string()),
728 vec![Value::Int(10), Value::Int(20), Value::Int(30)],
729 )));
730
731 let stack = crate::stack::alloc_test_stack();
732 let stack = push(stack, variant);
733 let stack = variant_field_count(stack);
734
735 let (_stack, result) = pop(stack);
736 assert_eq!(result, Value::Int(3));
737 }
738 }
739
740 #[test]
741 fn test_variant_tag() {
742 unsafe {
743 let variant = Value::Variant(Arc::new(VariantData::new(
745 global_string("MyTag".to_string()),
746 vec![Value::Int(10)],
747 )));
748
749 let stack = crate::stack::alloc_test_stack();
750 let stack = push(stack, variant);
751 let stack = variant_tag(stack);
752
753 let (_stack, result) = pop(stack);
754 assert_eq!(result, Value::Symbol(global_string("MyTag".to_string())));
755 }
756 }
757
758 #[test]
759 fn test_variant_field_at() {
760 unsafe {
761 let str1 = global_string("hello".to_string());
762 let str2 = global_string("world".to_string());
763
764 let variant = Value::Variant(Arc::new(VariantData::new(
766 global_string("TestTag".to_string()),
767 vec![
768 Value::String(str1.clone()),
769 Value::Int(42),
770 Value::String(str2.clone()),
771 ],
772 )));
773
774 let stack = crate::stack::alloc_test_stack();
776 let stack = push(stack, variant.clone());
777 let stack = push(stack, Value::Int(0));
778 let stack = variant_field_at(stack);
779
780 let (_stack, result) = pop(stack);
781 assert_eq!(result, Value::String(str1.clone()));
782
783 let stack = push(stack, variant.clone());
785 let stack = push(stack, Value::Int(1));
786 let stack = variant_field_at(stack);
787
788 let (_stack, result) = pop(stack);
789 assert_eq!(result, Value::Int(42));
790
791 let stack = push(stack, variant.clone());
793 let stack = push(stack, Value::Int(2));
794 let stack = variant_field_at(stack);
795
796 let (_stack, result) = pop(stack);
797 assert_eq!(result, Value::String(str2));
798 }
799 }
800
801 #[test]
802 fn test_variant_field_count_empty() {
803 unsafe {
804 let variant = Value::Variant(Arc::new(VariantData::new(
806 global_string("Empty".to_string()),
807 vec![],
808 )));
809
810 let stack = crate::stack::alloc_test_stack();
811 let stack = push(stack, variant);
812 let stack = variant_field_count(stack);
813
814 let (_stack, result) = pop(stack);
815 assert_eq!(result, Value::Int(0));
816 }
817 }
818
819 #[test]
820 fn test_make_variant_with_fields() {
821 unsafe {
822 let stack = crate::stack::alloc_test_stack();
825 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);
831
832 let (_stack, result) = pop(stack);
833
834 match result {
835 Value::Variant(v) => {
836 assert_eq!(v.tag.as_str(), "Tag");
837 assert_eq!(v.fields.len(), 3);
838 assert_eq!(v.fields[0], Value::Int(10));
839 assert_eq!(v.fields[1], Value::Int(20));
840 assert_eq!(v.fields[2], Value::Int(30));
841 }
842 _ => panic!("Expected Variant"),
843 }
844 }
845 }
846
847 #[test]
848 fn test_make_variant_empty() {
849 unsafe {
850 let stack = crate::stack::alloc_test_stack();
853 let stack = push(stack, Value::Symbol(global_string("None".to_string()))); let stack = make_variant_0(stack);
856
857 let (_stack, result) = pop(stack);
858
859 match result {
860 Value::Variant(v) => {
861 assert_eq!(v.tag.as_str(), "None");
862 assert_eq!(v.fields.len(), 0);
863 }
864 _ => panic!("Expected Variant"),
865 }
866 }
867 }
868
869 #[test]
870 fn test_make_variant_with_mixed_types() {
871 unsafe {
872 let s = global_string("hello".to_string());
873
874 let stack = crate::stack::alloc_test_stack();
876 let stack = push(stack, Value::Int(42));
877 let stack = push(stack, Value::String(s.clone()));
878 let stack = push(stack, Value::Float(3.5));
879 let stack = push(stack, Value::Symbol(global_string("Mixed".to_string()))); let stack = make_variant_3(stack);
882
883 let (_stack, result) = pop(stack);
884
885 match result {
886 Value::Variant(v) => {
887 assert_eq!(v.tag.as_str(), "Mixed");
888 assert_eq!(v.fields.len(), 3);
889 assert_eq!(v.fields[0], Value::Int(42));
890 assert_eq!(v.fields[1], Value::String(s));
891 assert_eq!(v.fields[2], Value::Float(3.5));
892 }
893 _ => panic!("Expected Variant"),
894 }
895 }
896 }
897
898 #[test]
899 fn test_variant_append() {
900 unsafe {
901 let stack = crate::stack::alloc_test_stack();
903 let stack = push(stack, Value::Symbol(global_string("Array".to_string()))); let stack = make_variant_0(stack);
905
906 let stack = push(stack, Value::Int(42));
908 let stack = variant_append(stack);
909
910 let (_stack, result) = pop(stack);
912 match result {
913 Value::Variant(v) => {
914 assert_eq!(v.tag.as_str(), "Array");
915 assert_eq!(v.fields.len(), 1);
916 assert_eq!(v.fields[0], Value::Int(42));
917 }
918 _ => panic!("Expected Variant"),
919 }
920 }
921 }
922
923 #[test]
924 fn test_variant_append_multiple() {
925 unsafe {
926 let stack = crate::stack::alloc_test_stack();
928 let stack = push(stack, Value::Symbol(global_string("Object".to_string()))); let stack = make_variant_0(stack);
930
931 let key = global_string("name".to_string());
933 let stack = push(stack, Value::String(key.clone()));
934 let stack = variant_append(stack);
935
936 let val = global_string("John".to_string());
938 let stack = push(stack, Value::String(val.clone()));
939 let stack = variant_append(stack);
940
941 let (_stack, result) = pop(stack);
943 match result {
944 Value::Variant(v) => {
945 assert_eq!(v.tag.as_str(), "Object");
946 assert_eq!(v.fields.len(), 2);
947 assert_eq!(v.fields[0], Value::String(key));
948 assert_eq!(v.fields[1], Value::String(val));
949 }
950 _ => panic!("Expected Variant"),
951 }
952 }
953 }
954
955 #[test]
956 fn test_variant_last() {
957 unsafe {
958 let variant = Value::Variant(Arc::new(VariantData::new(
960 global_string("List".to_string()),
961 vec![Value::Int(10), Value::Int(20), Value::Int(30)],
962 )));
963
964 let stack = crate::stack::alloc_test_stack();
965 let stack = push(stack, variant);
966 let stack = variant_last(stack);
967
968 let (_stack, result) = pop(stack);
969 assert_eq!(result, Value::Int(30));
970 }
971 }
972
973 #[test]
974 fn test_variant_init() {
975 unsafe {
976 let variant = Value::Variant(Arc::new(VariantData::new(
978 global_string("Custom".to_string()),
979 vec![Value::Int(10), Value::Int(20), Value::Int(30)],
980 )));
981
982 let stack = crate::stack::alloc_test_stack();
983 let stack = push(stack, variant);
984 let stack = variant_init(stack);
985
986 let (_stack, result) = pop(stack);
987 match result {
988 Value::Variant(v) => {
989 assert_eq!(v.tag.as_str(), "Custom"); assert_eq!(v.fields.len(), 2);
991 assert_eq!(v.fields[0], Value::Int(10));
992 assert_eq!(v.fields[1], Value::Int(20));
993 }
994 _ => panic!("Expected Variant"),
995 }
996 }
997 }
998
999 #[test]
1000 fn test_variant_stack_operations() {
1001 unsafe {
1003 let stack = crate::stack::alloc_test_stack();
1005 let stack = push(stack, Value::Symbol(global_string("Stack".to_string()))); let stack = make_variant_0(stack);
1007
1008 let stack = push(stack, Value::Int(10));
1010 let stack = variant_append(stack);
1011
1012 let stack = push(stack, Value::Int(20));
1014 let stack = variant_append(stack);
1015
1016 let (stack, variant) = pop(stack);
1019 let stack = push(stack, variant.clone());
1020 let stack = push(stack, variant);
1021 let stack = variant_last(stack);
1022 let (stack, top) = pop(stack);
1023 assert_eq!(top, Value::Int(20));
1024
1025 let stack = variant_init(stack);
1027
1028 let (stack, variant) = pop(stack);
1030 let stack = push(stack, variant.clone());
1031 let stack = push(stack, variant);
1032 let stack = variant_last(stack);
1033 let (stack, top) = pop(stack);
1034 assert_eq!(top, Value::Int(10));
1035
1036 let (_stack, result) = pop(stack);
1038 match result {
1039 Value::Variant(v) => {
1040 assert_eq!(v.fields.len(), 1);
1041 assert_eq!(v.fields[0], Value::Int(10));
1042 }
1043 _ => panic!("Expected Variant"),
1044 }
1045 }
1046 }
1047
1048 #[test]
1049 fn test_variant_clone_is_o1() {
1050 let mut variant = Value::Variant(Arc::new(VariantData::new(
1053 global_string("Level0".to_string()),
1054 vec![],
1055 )));
1056
1057 for i in 0..100 {
1059 variant = Value::Variant(Arc::new(VariantData::new(
1060 global_string(format!("Level{}", i)),
1061 vec![variant.clone()],
1062 )));
1063 }
1064
1065 let start = std::time::Instant::now();
1067 for _ in 0..1000 {
1068 let _copy = variant.clone();
1069 }
1070 let elapsed = start.elapsed();
1071
1072 assert!(
1075 elapsed.as_millis() < 10,
1076 "Clone took {:?} - should be O(1) with Arc",
1077 elapsed
1078 );
1079 }
1080
1081 #[test]
1082 fn test_variant_arc_sharing() {
1083 let inner = Value::Variant(Arc::new(VariantData::new(
1085 global_string("Inner".to_string()),
1086 vec![Value::Int(42)],
1087 )));
1088 let outer = Value::Variant(Arc::new(VariantData::new(
1089 global_string("Outer".to_string()),
1090 vec![inner.clone()],
1091 )));
1092
1093 let outer_clone = outer.clone();
1095
1096 match (&outer, &outer_clone) {
1098 (Value::Variant(a), Value::Variant(b)) => {
1099 assert_eq!(a.tag, b.tag);
1102 assert_eq!(a.fields.len(), b.fields.len());
1103 }
1104 _ => panic!("Expected Variants"),
1105 }
1106 }
1107
1108 #[test]
1109 fn test_variant_thread_safe_sharing() {
1110 use std::sync::Arc as StdArc;
1113 use std::thread;
1114
1115 let variant = Value::Variant(Arc::new(VariantData::new(
1116 global_string("ThreadSafe".to_string()),
1117 vec![Value::Int(1), Value::Int(2), Value::Int(3)],
1118 )));
1119
1120 let shared = StdArc::new(variant);
1122
1123 let handles: Vec<_> = (0..4)
1124 .map(|_| {
1125 let v = StdArc::clone(&shared);
1126 thread::spawn(move || {
1127 match &*v {
1129 Value::Variant(data) => {
1130 assert_eq!(data.tag.as_str(), "ThreadSafe");
1131 assert_eq!(data.fields.len(), 3);
1132 }
1133 _ => panic!("Expected Variant"),
1134 }
1135 })
1136 })
1137 .collect();
1138
1139 for h in handles {
1140 h.join().expect("Thread panicked");
1141 }
1142 }
1143}