infix
A JIT-Powered FFI Library for C
Loading...
Searching...
No Matches
types.c
Go to the documentation of this file.
1
37#include "common/utility.h"
38#include <limits.h>
39#include <stdio.h>
40#include <stdlib.h>
41#include <string.h>
42// Static Descriptors for Primitive and Built-in Types
43
44#define INFIX_MAX_ALIGNMENT (1024 * 1024) // 1MB Safety Limit
45
59#define INFIX_TYPE_INIT(id, T) \
60 {.name = nullptr, \
61 .category = INFIX_TYPE_PRIMITIVE, \
62 .size = sizeof(T), \
63 .alignment = _Alignof(T), \
64 .is_arena_allocated = false, \
65 .arena = nullptr, \
66 .source_offset = 0, \
67 .meta.primitive_id = id}
73static infix_type _infix_type_void = {.name = nullptr,
74 .category = INFIX_TYPE_VOID,
75 .size = 0,
76 .alignment = 0,
77 .is_arena_allocated = false,
78 .arena = nullptr,
79 .source_offset = 0,
80 .meta = {0}};
86static infix_type _infix_type_pointer = {.name = nullptr,
87 .category = INFIX_TYPE_POINTER,
88 .size = sizeof(void *),
89 .alignment = _Alignof(void *),
90 .is_arena_allocated = false,
91 .arena = nullptr,
92 .source_offset = 0,
93 .meta.pointer_info = {.pointee_type = &_infix_type_void}};
112#if !defined(INFIX_COMPILER_MSVC)
117#endif
120 .category = INFIX_TYPE_PRIMITIVE,
121 .size = 2,
122 .alignment = 2,
123 .is_arena_allocated = false,
124 .arena = nullptr,
125 .source_offset = 0,
126 .meta.primitive_id = INFIX_PRIMITIVE_FLOAT16};
131#if defined(INFIX_COMPILER_MSVC) || (defined(INFIX_OS_WINDOWS) && defined(INFIX_COMPILER_CLANG))
132// On these platforms, long double is just an alias for double, so no separate singleton is needed.
133#else
136#endif
137// Public API: Type Creation Functions
145 switch (id) {
147 return &_infix_type_bool;
149 return &_infix_type_uint8;
151 return &_infix_type_sint8;
153 return &_infix_type_uint16;
155 return &_infix_type_sint16;
157 return &_infix_type_uint32;
159 return &_infix_type_sint32;
161 return &_infix_type_uint64;
163 return &_infix_type_sint64;
164#if !defined(INFIX_COMPILER_MSVC)
166 return &_infix_type_uint128;
168 return &_infix_type_sint128;
169#endif
171 return &_infix_type_float16;
173 return &_infix_type_float;
175 return &_infix_type_double;
177#if defined(INFIX_COMPILER_MSVC) || (defined(INFIX_OS_WINDOWS) && defined(INFIX_COMPILER_CLANG))
178 // On MSVC and macOS/Clang (sometimes), long double is an alias for double.
179 // We map to the double singleton to maintain type identity.
180 return &_infix_type_double;
181#else
182 // On MinGW and Linux, long double is distinct (16 bytes).
183 // We MUST use the distinct type to handle layout and passing correctly.
185#endif
186 default:
187 // Return null for any invalid primitive ID.
188 return nullptr;
189 }
190}
208INFIX_API infix_struct_member infix_type_create_member(const char * name, infix_type * type, size_t offset) {
209 return (infix_struct_member){name, type, offset, 0, 0, false};
210}
220 infix_type * type,
221 size_t offset,
222 uint8_t bit_width) {
223 return (infix_struct_member){name, type, offset, bit_width, 0, true};
224}
225
231static bool _layout_struct(infix_type * type) {
232 size_t current_byte_offset = 0;
233 size_t current_unit_offset = 0;
234 size_t current_unit_size = 0;
235 uint32_t current_unit_bits_used = 0;
236 size_t max_alignment = 1;
237 bool in_bitfield = false;
238
239 for (size_t i = 0; i < type->meta.aggregate_info.num_members; ++i) {
240 infix_struct_member * member = &type->meta.aggregate_info.members[i];
241 infix_type * mtype = member->type;
242
243 if (member->is_bitfield) {
244 size_t align = mtype->alignment;
245 if (align == 0)
246 align = 1;
247 if (align > max_alignment && !type->meta.aggregate_info.is_packed)
248 max_alignment = align;
249
250 // Zero-width bitfield or type mismatch or doesn't fit -> start new unit
251 if (member->bit_width == 0 || !in_bitfield || mtype->size != current_unit_size ||
252 current_unit_bits_used + member->bit_width > mtype->size * 8) {
253
254 // Align to start of new unit
255 size_t start = _infix_align_up(current_byte_offset, align);
256 if (start < current_byte_offset) {
258 return false;
259 }
260
261 current_unit_offset = start;
262 current_unit_size = mtype->size;
263 current_unit_bits_used = 0;
264 in_bitfield = true;
265
266 current_byte_offset = current_unit_offset;
267 }
268
269 member->offset = current_unit_offset + (current_unit_bits_used / 8);
270 member->bit_offset = (uint8_t)(current_unit_bits_used % 8);
271 current_unit_bits_used += member->bit_width;
272
273 size_t bytes_used = (current_unit_bits_used + 7) / 8;
274 if (current_unit_offset + bytes_used > current_byte_offset)
275 current_byte_offset = current_unit_offset + bytes_used;
276 }
277 else {
278 // Handle Flexible Array Members (FAM)
279 if (mtype->category == INFIX_TYPE_ARRAY && mtype->meta.array_info.is_flexible) {
280 in_bitfield = false;
281 size_t align = mtype->alignment;
282 if (align == 0)
283 align = 1;
284
285 if (align > max_alignment && !type->meta.aggregate_info.is_packed)
286 max_alignment = align;
287
288 size_t aligned = _infix_align_up(current_byte_offset, align);
289 if (aligned < current_byte_offset) {
291 return false;
292 }
293 current_byte_offset = aligned;
294 member->offset = current_byte_offset;
295 member->bit_offset = 0;
296 continue;
297 }
298
299 // Standard Member
300 in_bitfield = false;
301 size_t align = mtype->alignment;
302 if (align == 0)
303 align = 1;
304
305 if (align > max_alignment && !type->meta.aggregate_info.is_packed)
306 max_alignment = align;
307
308 size_t aligned = _infix_align_up(current_byte_offset, align);
309 if (aligned < current_byte_offset) {
311 return false;
312 }
313 current_byte_offset = aligned;
314 member->offset = current_byte_offset;
315 member->bit_offset = 0;
316
317 if (current_byte_offset > SIZE_MAX - mtype->size) {
319 return false;
320 }
321 current_byte_offset += mtype->size;
322 }
323 }
324
325 // If it is packed, the alignment is explicitly determined by the user.
326 if (type->meta.aggregate_info.is_packed)
327 max_alignment = type->alignment;
328
329 type->alignment = max_alignment;
330 type->size = _infix_align_up(current_byte_offset, max_alignment);
331 return true;
332}
351 infix_type ** out_type,
352 infix_struct_member ** out_arena_members,
354 size_t num_members) {
355 if (out_type == nullptr)
357 // Pre-flight check: ensure all provided member types are valid.
358 for (size_t i = 0; i < num_members; ++i) {
359 if (members[i].type == nullptr) {
360 *out_type = nullptr;
363 }
364 }
365 infix_type * type = infix_arena_calloc(arena, 1, sizeof(infix_type), _Alignof(infix_type));
366 if (type == nullptr) {
367 *out_type = nullptr;
370 }
371 infix_struct_member * arena_members = nullptr;
372 if (num_members > 0) {
373 arena_members =
374 infix_arena_alloc(arena, sizeof(infix_struct_member) * num_members, _Alignof(infix_struct_member));
375 if (arena_members == nullptr) {
376 *out_type = nullptr;
379 }
380 infix_memcpy(arena_members, members, sizeof(infix_struct_member) * num_members);
381 }
382 *out_type = type;
383 *out_arena_members = arena_members;
384 return INFIX_SUCCESS;
385}
394 infix_type ** out_type,
395 infix_type * pointee_type) {
397 if (!out_type || !pointee_type) {
400 }
401 infix_type * type = infix_arena_calloc(arena, 1, sizeof(infix_type), _Alignof(infix_type));
402 if (type == nullptr) {
403 *out_type = nullptr;
406 }
407 // Start by copying the layout of a generic pointer.
408 *type = *infix_type_create_pointer();
409 // Mark it as arena-allocated so it can be deep-copied and freed correctly.
410 type->is_arena_allocated = true;
411 // Set the specific pointee type.
412 type->meta.pointer_info.pointee_type = pointee_type;
413 *out_type = type;
414 return INFIX_SUCCESS;
415}
425 infix_type ** out_type,
426 infix_type * element_type,
427 size_t num_elements) {
429 if (out_type == nullptr || element_type == nullptr) {
432 }
433 if (element_type->size > 0 && num_elements > SIZE_MAX / element_type->size) {
434 *out_type = nullptr;
437 }
438 infix_type * type = infix_arena_calloc(arena, 1, sizeof(infix_type), _Alignof(infix_type));
439 if (type == nullptr) {
440 *out_type = nullptr;
443 }
444 type->is_arena_allocated = true;
446 type->meta.array_info.element_type = element_type;
447 type->meta.array_info.num_elements = num_elements;
448 type->meta.array_info.is_flexible = false;
449 // An array's alignment is the same as its element's alignment.
450 type->alignment = element_type->alignment;
451 type->size = element_type->size * num_elements;
452 *out_type = type;
453 return INFIX_SUCCESS;
454}
455
464 infix_type ** out_type,
465 infix_type * element_type) {
467 if (out_type == nullptr || element_type == nullptr) {
470 }
471 // Flexible arrays of incomplete types (size 0) are generally not allowed.
472 if (element_type->category == INFIX_TYPE_VOID || element_type->size == 0) {
473 *out_type = nullptr;
476 }
477
478 infix_type * type = infix_arena_calloc(arena, 1, sizeof(infix_type), _Alignof(infix_type));
479 if (type == nullptr) {
480 *out_type = nullptr;
483 }
484 type->is_arena_allocated = true;
486 type->meta.array_info.element_type = element_type;
487 type->meta.array_info.num_elements = 0;
488 type->meta.array_info.is_flexible = true; // Mark as flexible
489
490 // A flexible array member itself has size 0 within the struct (it hangs off the end).
491 // However, its alignment requirement affects the struct.
492 type->alignment = element_type->alignment;
493 type->size = 0;
494
495 *out_type = type;
496 return INFIX_SUCCESS;
497}
498
508 infix_type ** out_type,
509 infix_type * underlying_type) {
511 if (out_type == nullptr || underlying_type == nullptr) {
514 }
515 if (underlying_type->category != INFIX_TYPE_PRIMITIVE ||
516 underlying_type->meta.primitive_id > INFIX_PRIMITIVE_SINT128) {
519 }
520 infix_type * type = infix_arena_calloc(arena, 1, sizeof(infix_type), _Alignof(infix_type));
521 if (type == nullptr) {
522 *out_type = nullptr;
525 }
526 type->is_arena_allocated = true;
528 // An enum has the same memory layout as its underlying integer type.
529 type->size = underlying_type->size;
530 type->alignment = underlying_type->alignment;
531 type->meta.enum_info.underlying_type = underlying_type;
532 *out_type = type;
533 return INFIX_SUCCESS;
534}
543 infix_type ** out_type,
544 infix_type * base_type) {
546 if (out_type == nullptr || base_type == nullptr || (!is_float(base_type) && !is_double(base_type))) {
549 }
550 infix_type * type = infix_arena_calloc(arena, 1, sizeof(infix_type), _Alignof(infix_type));
551 if (type == nullptr) {
552 *out_type = nullptr;
555 }
556 type->is_arena_allocated = true;
558 // A complex number is simply two floating-point numbers back-to-back.
559 type->size = base_type->size * 2;
560 type->alignment = base_type->alignment;
561 type->meta.complex_info.base_type = base_type;
562 *out_type = type;
563 return INFIX_SUCCESS;
564}
574 infix_type ** out_type,
575 infix_type * element_type,
576 size_t num_elements) {
578 if (out_type == nullptr || element_type == nullptr || element_type->category != INFIX_TYPE_PRIMITIVE) {
581 }
582 if (element_type->size > 0 && num_elements > SIZE_MAX / element_type->size) {
583 *out_type = nullptr;
586 }
587 infix_type * type = infix_arena_calloc(arena, 1, sizeof(infix_type), _Alignof(infix_type));
588 if (type == nullptr) {
589 *out_type = nullptr;
592 }
593 type->is_arena_allocated = true;
595 type->meta.vector_info.element_type = element_type;
596 type->meta.vector_info.num_elements = num_elements;
597 type->size = element_type->size * num_elements;
598 // Vector alignment is its total size.
599 // This ensures __m128 is 16-byte aligned, __m256 is 32-byte, and __m512 is 64-byte.
600 type->alignment = type->size;
601 *out_type = type;
602 return INFIX_SUCCESS;
603}
613 infix_type ** out_type,
615 size_t num_members) {
617 infix_type * type = nullptr;
618 infix_struct_member * arena_members = nullptr;
619 infix_status status = _create_aggregate_setup(arena, &type, &arena_members, members, num_members);
620 if (status != INFIX_SUCCESS) {
621 *out_type = nullptr;
622 return status;
623 }
624 type->is_arena_allocated = true;
626 type->meta.aggregate_info.members = arena_members;
627 type->meta.aggregate_info.num_members = num_members;
628 type->meta.aggregate_info.is_packed = false; // Unions don't use this flag currently
629 // A union's size is the size of its largest member, and its alignment is the
630 // alignment of its most-aligned member.
631 size_t max_size = 0;
632 size_t max_alignment = 1;
633 for (size_t i = 0; i < num_members; ++i) {
634 arena_members[i].offset = 0; // All union members have an offset of 0.
635 if (arena_members[i].type->size > max_size)
636 max_size = arena_members[i].type->size;
637 if (arena_members[i].type->alignment > max_alignment)
638 max_alignment = arena_members[i].type->alignment;
639 }
640 type->alignment = max_alignment;
641 // The total size is the size of the largest member, padded up to the required alignment.
642 type->size = _infix_align_up(max_size, max_alignment);
643 // Overflow check
644 if (type->size < max_size) {
645 *out_type = nullptr;
648 }
649 *out_type = type;
650 return INFIX_SUCCESS;
651}
661 infix_type ** out_type,
663 size_t num_members) {
665 infix_type * type = nullptr;
666 infix_struct_member * arena_members = nullptr;
667 infix_status status = _create_aggregate_setup(arena, &type, &arena_members, members, num_members);
668 if (status != INFIX_SUCCESS) {
669 *out_type = nullptr;
670 return status;
671 }
672 type->is_arena_allocated = true;
674 type->meta.aggregate_info.members = arena_members;
675 type->meta.aggregate_info.num_members = num_members;
676 type->meta.aggregate_info.is_packed = false;
677
678 // This performs a preliminary layout calculation.
679 // Note: This layout may be incomplete if it contains unresolved named references or flexible arrays.
680 // The final, correct layout will be computed by `_infix_type_recalculate_layout`.
681 // However, we must set a preliminary size/alignment here.
682
683 // We use the recalculate logic to do the heavy lifting, assuming a temporary arena can be made.
684 // But we can't create an arena here easily if we are in a strict context.
685 // So we do a simplified pass just like the old logic, ignoring complex bitfield rules for now.
686 // The proper bitfield layout happens in `_infix_type_recalculate_layout`.
687
688 for (size_t i = 0; i < num_members; ++i) {
689 infix_struct_member * member = &arena_members[i];
690 if (member->type->alignment == 0 && member->type->category != INFIX_TYPE_NAMED_REFERENCE &&
691 !(member->type->category == INFIX_TYPE_ARRAY && member->type->meta.array_info.is_flexible)) {
692 if (member->type->category != INFIX_TYPE_ARRAY) {
693 *out_type = nullptr;
696 }
697 }
698 }
699
700 // Calculate Layout (including bitfields and FAMs)
701 if (!_layout_struct(type)) {
702 *out_type = nullptr;
704 }
705
706 *out_type = type;
707 return INFIX_SUCCESS;
708}
720 infix_type ** out_type,
721 size_t total_size,
722 size_t alignment,
724 size_t num_members) {
726 if (out_type == nullptr || (num_members > 0 && members == nullptr)) {
729 }
730 // Validate alignment is power-of-two AND within sane limits.
731 if (alignment == 0 || (alignment & (alignment - 1)) != 0) {
734 }
735 if (alignment > INFIX_MAX_ALIGNMENT) {
738 }
739 infix_type * type = infix_arena_calloc(arena, 1, sizeof(infix_type), _Alignof(infix_type));
740 if (type == nullptr) {
741 *out_type = nullptr;
744 }
745 infix_struct_member * arena_members = nullptr;
746 if (num_members > 0) {
747 arena_members =
748 infix_arena_alloc(arena, sizeof(infix_struct_member) * num_members, _Alignof(infix_struct_member));
749 if (arena_members == nullptr) {
750 *out_type = nullptr;
753 }
754 infix_memcpy(arena_members, members, sizeof(infix_struct_member) * num_members);
755 }
756 type->is_arena_allocated = true;
757 type->size = total_size;
758 type->alignment = alignment;
759 type->category = INFIX_TYPE_STRUCT; // Packed structs are still fundamentally structs.
760 type->meta.aggregate_info.members = arena_members;
761 type->meta.aggregate_info.num_members = num_members;
762 type->meta.aggregate_info.is_packed = true; // Marked as packed
763 *out_type = type;
764 return INFIX_SUCCESS;
765}
778 infix_type ** out_type,
779 const char * name,
782 if (out_type == nullptr || name == nullptr) {
785 }
786 infix_type * type = infix_arena_calloc(arena, 1, sizeof(infix_type), _Alignof(infix_type));
787 if (type == nullptr) {
788 *out_type = nullptr;
791 }
792 // The name must be copied into the arena to ensure its lifetime matches the type's.
793 size_t name_len = strlen(name) + 1;
794 char * arena_name = infix_arena_alloc(arena, name_len, 1);
795 if (arena_name == nullptr) {
796 *out_type = nullptr;
799 }
800 infix_memcpy(arena_name, name, name_len);
801 type->is_arena_allocated = true;
803 type->size = 0; // Size and alignment are unknown until resolution.
804 type->alignment = 1; // Default to 1 to be safe in preliminary layout calculations.
805 type->meta.named_reference.name = arena_name;
807 *out_type = type;
808 return INFIX_SUCCESS;
809}
810// Internal Type Graph Management
851 recalc_visited_node_t ** visited_head) {
852 if (!type || !type->is_arena_allocated)
853 return; // Base case: Don't modify static singleton types.
854 // Cycle detection: If we have already visited this node in the current recursion
855 // path, we are in a cycle. Return immediately to break the loop. The layout of
856 // this node will be calculated when the recursion unwinds to its first visit.
857 for (recalc_visited_node_t * v = *visited_head; v != nullptr; v = v->next)
858 if (v->type == type)
859 return;
860 // Allocate the memoization node from a stable temporary arena.
861 recalc_visited_node_t * visited_node =
862 infix_arena_alloc(temp_arena, sizeof(recalc_visited_node_t), _Alignof(recalc_visited_node_t));
863 if (!visited_node)
864 return; // Cannot proceed without memory.
865 visited_node->type = type;
866 visited_node->next = *visited_head;
867 *visited_head = visited_node;
868 // Recurse into child types first (post-order traversal).
869 switch (type->category) {
872 break;
873 case INFIX_TYPE_ARRAY:
875 break;
877 case INFIX_TYPE_UNION:
878 for (size_t i = 0; i < type->meta.aggregate_info.num_members; ++i) {
880 temp_arena, type->meta.aggregate_info.members[i].type, visited_head);
881 }
882 break;
883 default:
884 break; // Other types have no child types to recurse into.
885 }
886 // After children are updated, recalculate this type's layout.
889 else if (type->category == INFIX_TYPE_UNION) {
890 size_t max_size = 0;
891 size_t max_alignment = 1;
892 for (size_t i = 0; i < type->meta.aggregate_info.num_members; ++i) {
893 infix_type * member_type = type->meta.aggregate_info.members[i].type;
894 if (member_type->size > max_size)
895 max_size = member_type->size;
896 if (member_type->alignment > max_alignment)
897 max_alignment = member_type->alignment;
898 }
899 type->alignment = max_alignment;
900 type->size = _infix_align_up(max_size, max_alignment);
901 }
902 else if (type->category == INFIX_TYPE_ARRAY) {
903 // Flexible arrays have size 0 but inherit alignment.
904 // Fixed arrays calculate size normally.
907 type->size = 0;
908 }
909 else {
912 }
913 }
914}
925 // Create a temporary arena solely for the visited list's lifetime.
926 infix_arena_t * temp_arena = infix_arena_create(1024);
927 if (!temp_arena)
928 return;
929 recalc_visited_node_t * visited_head = nullptr;
930 _infix_type_recalculate_layout_recursive(temp_arena, type, &visited_head);
931 infix_arena_destroy(temp_arena);
932}
961 const infix_type * src_type,
962 memo_node_t ** memo_head) {
963 if (src_type == nullptr)
964 return nullptr;
965 // If the source type lives in the same arena as our destination, we can safely share the pointer instead of
966 // performing a deep copy.
967 if (src_type->arena == dest_arena)
968 return (infix_type *)src_type;
969 // Base case: Static types don't need to be copied; return the singleton pointer.
970 if (!src_type->is_arena_allocated)
971 return (infix_type *)src_type;
972 // Check memoization table: if we've already copied this node, return the existing copy.
973 // This correctly handles cycles and shared sub-graphs.
974 for (memo_node_t * node = *memo_head; node != NULL; node = node->next)
975 if (node->src == src_type)
976 return node->dest;
977 // Allocate the new type object in the destination arena.
978 infix_type * dest_type = infix_arena_calloc(dest_arena, 1, sizeof(infix_type), _Alignof(infix_type));
979 if (dest_type == nullptr)
980 return nullptr;
981 // Add this new pair to the memoization table BEFORE recursing. This is crucial
982 // for handling cycles: the recursive call will find this entry and return `dest_type`.
983 memo_node_t * new_memo_node = infix_arena_alloc(dest_arena, sizeof(memo_node_t), _Alignof(memo_node_t));
984 if (!new_memo_node)
985 return nullptr;
986 new_memo_node->src = src_type;
987 new_memo_node->dest = dest_type;
988 new_memo_node->next = *memo_head;
989 *memo_head = new_memo_node;
990 // Perform a shallow copy of the main struct, then recurse to deep copy child pointers.
991 *dest_type = *src_type;
992 dest_type->is_arena_allocated = true;
993 dest_type->is_incomplete = src_type->is_incomplete;
994 dest_type->arena = dest_arena; // The new type now belongs to the destination arena.
995 // Deep copy the semantic name string, if it exists.
996 if (src_type->name) {
997 size_t name_len = strlen(src_type->name) + 1;
998 char * dest_name = infix_arena_alloc(dest_arena, name_len, 1);
999 if (!dest_name)
1000 return nullptr; // Allocation failed
1001 infix_memcpy((void *)dest_name, src_type->name, name_len);
1002 dest_type->name = dest_name;
1003 }
1004 switch (src_type->category) {
1005 case INFIX_TYPE_POINTER:
1006 dest_type->meta.pointer_info.pointee_type =
1007 _copy_type_graph_to_arena_recursive(dest_arena, src_type->meta.pointer_info.pointee_type, memo_head);
1008 break;
1009 case INFIX_TYPE_ARRAY:
1010 dest_type->meta.array_info.element_type =
1011 _copy_type_graph_to_arena_recursive(dest_arena, src_type->meta.array_info.element_type, memo_head);
1012 // Explicitly copy the flexible flag to ensure it persists.
1013 dest_type->meta.array_info.is_flexible = src_type->meta.array_info.is_flexible;
1014 break;
1015 case INFIX_TYPE_STRUCT:
1016 case INFIX_TYPE_UNION:
1017 if (src_type->meta.aggregate_info.num_members > 0) {
1018 // Copy the members array itself.
1019 size_t members_size = sizeof(infix_struct_member) * src_type->meta.aggregate_info.num_members;
1020 dest_type->meta.aggregate_info.members =
1021 infix_arena_alloc(dest_arena, members_size, _Alignof(infix_struct_member));
1022 if (dest_type->meta.aggregate_info.members == nullptr)
1023 return nullptr;
1024 dest_type->meta.aggregate_info.is_packed = src_type->meta.aggregate_info.is_packed; // Copy packed flag
1025 // Now, recurse for each member's type and copy its name.
1026 for (size_t i = 0; i < src_type->meta.aggregate_info.num_members; ++i) {
1027 dest_type->meta.aggregate_info.members[i] = src_type->meta.aggregate_info.members[i];
1029 dest_arena, src_type->meta.aggregate_info.members[i].type, memo_head);
1030 const char * src_name = src_type->meta.aggregate_info.members[i].name;
1031 if (src_name) {
1032 size_t name_len = strlen(src_name) + 1;
1033 char * dest_name = infix_arena_alloc(dest_arena, name_len, 1);
1034 if (!dest_name)
1035 return nullptr;
1036 infix_memcpy((void *)dest_name, src_name, name_len);
1037 dest_type->meta.aggregate_info.members[i].name = dest_name;
1038 }
1039 // Copy bitfield properties
1040 dest_type->meta.aggregate_info.members[i].bit_width =
1041 src_type->meta.aggregate_info.members[i].bit_width;
1042 dest_type->meta.aggregate_info.members[i].bit_offset =
1044 dest_type->meta.aggregate_info.members[i].is_bitfield =
1046 }
1047 }
1048 break;
1050 {
1051 const char * src_name = src_type->meta.named_reference.name;
1052 if (src_name) {
1053 size_t name_len = strlen(src_name) + 1;
1054 char * dest_name = infix_arena_alloc(dest_arena, name_len, 1);
1055 if (!dest_name)
1056 return nullptr;
1057 infix_memcpy((void *)dest_name, src_name, name_len);
1058 dest_type->meta.named_reference.name = dest_name;
1059 }
1060 break;
1061 }
1063 dest_type->meta.func_ptr_info.return_type =
1064 _copy_type_graph_to_arena_recursive(dest_arena, src_type->meta.func_ptr_info.return_type, memo_head);
1065 if (src_type->meta.func_ptr_info.num_args > 0) {
1066 size_t args_size = sizeof(infix_function_argument) * src_type->meta.func_ptr_info.num_args;
1067 dest_type->meta.func_ptr_info.args =
1068 infix_arena_alloc(dest_arena, args_size, _Alignof(infix_function_argument));
1069 if (dest_type->meta.func_ptr_info.args == nullptr)
1070 return nullptr;
1071 for (size_t i = 0; i < src_type->meta.func_ptr_info.num_args; ++i) {
1072 dest_type->meta.func_ptr_info.args[i] = src_type->meta.func_ptr_info.args[i];
1074 dest_arena, src_type->meta.func_ptr_info.args[i].type, memo_head);
1075 const char * src_name = src_type->meta.func_ptr_info.args[i].name;
1076 if (src_name) {
1077 size_t name_len = strlen(src_name) + 1;
1078 char * dest_name = infix_arena_alloc(dest_arena, name_len, 1);
1079 if (!dest_name)
1080 return nullptr;
1081 infix_memcpy((void *)dest_name, src_name, name_len);
1082 dest_type->meta.func_ptr_info.args[i].name = dest_name;
1083 }
1084 }
1085 }
1086 break;
1087 case INFIX_TYPE_ENUM:
1088 dest_type->meta.enum_info.underlying_type =
1089 _copy_type_graph_to_arena_recursive(dest_arena, src_type->meta.enum_info.underlying_type, memo_head);
1090 break;
1091 case INFIX_TYPE_COMPLEX:
1092 dest_type->meta.complex_info.base_type =
1093 _copy_type_graph_to_arena_recursive(dest_arena, src_type->meta.complex_info.base_type, memo_head);
1094 break;
1095 case INFIX_TYPE_VECTOR:
1096 dest_type->meta.vector_info.element_type =
1097 _copy_type_graph_to_arena_recursive(dest_arena, src_type->meta.vector_info.element_type, memo_head);
1098 break;
1099 default:
1100 // Other types like primitives have no child pointers to copy.
1101 break;
1102 }
1103 return dest_type;
1104}
1113 memo_node_t * memo_head = nullptr;
1114 return _copy_type_graph_to_arena_recursive(dest_arena, src_type, &memo_head);
1115}
1138 const infix_type * type,
1139 estimate_visited_node_t ** visited_head) {
1140 if (!type || !type->is_arena_allocated)
1141 return 0;
1142 // Cycle detection: if we've seen this node, it's already accounted for.
1143 for (estimate_visited_node_t * v = *visited_head; v != NULL; v = v->next)
1144 if (v->type == type)
1145 return 0;
1146 // Add this node to the visited list before recursing.
1147 estimate_visited_node_t * visited_node =
1149 if (!visited_node) {
1150 // On allocation failure, we can't proceed with estimation. Return a large
1151 // number to ensure the caller allocates a fallback-sized arena.
1152 return 65536;
1153 }
1154 visited_node->type = type;
1155 visited_node->next = *visited_head;
1156 *visited_head = visited_node;
1157 // The size includes the type object itself, a memoization node, and the name string if it exists.
1158 size_t total_size = sizeof(infix_type) + sizeof(memo_node_t);
1159 if (type->name)
1160 total_size += strlen(type->name) + 1;
1161 switch (type->category) {
1162 case INFIX_TYPE_POINTER:
1163 total_size += _estimate_graph_size_recursive(temp_arena, type->meta.pointer_info.pointee_type, visited_head);
1164 break;
1165 case INFIX_TYPE_ARRAY:
1166 total_size += _estimate_graph_size_recursive(temp_arena, type->meta.array_info.element_type, visited_head);
1167 break;
1168 case INFIX_TYPE_STRUCT:
1169 case INFIX_TYPE_UNION:
1171 total_size += sizeof(infix_struct_member) * type->meta.aggregate_info.num_members;
1172 for (size_t i = 0; i < type->meta.aggregate_info.num_members; ++i) {
1173 const infix_struct_member * member = &type->meta.aggregate_info.members[i];
1174 if (member->name)
1175 total_size += strlen(member->name) + 1;
1176 total_size += _estimate_graph_size_recursive(temp_arena, member->type, visited_head);
1177 }
1178 }
1179 break;
1182 total_size += strlen(type->meta.named_reference.name) + 1;
1183 break;
1185 total_size += _estimate_graph_size_recursive(temp_arena, type->meta.func_ptr_info.return_type, visited_head);
1186 if (type->meta.func_ptr_info.num_args > 0) {
1187 total_size += sizeof(infix_function_argument) * type->meta.func_ptr_info.num_args;
1188 for (size_t i = 0; i < type->meta.func_ptr_info.num_args; ++i) {
1190 if (arg->name)
1191 total_size += strlen(arg->name) + 1;
1192 total_size += _estimate_graph_size_recursive(temp_arena, arg->type, visited_head);
1193 }
1194 }
1195 break;
1196 case INFIX_TYPE_ENUM:
1197 total_size += _estimate_graph_size_recursive(temp_arena, type->meta.enum_info.underlying_type, visited_head);
1198 break;
1199 case INFIX_TYPE_COMPLEX:
1200 total_size += _estimate_graph_size_recursive(temp_arena, type->meta.complex_info.base_type, visited_head);
1201 break;
1202 case INFIX_TYPE_VECTOR:
1203 total_size += _estimate_graph_size_recursive(temp_arena, type->meta.vector_info.element_type, visited_head);
1204 break;
1205 default:
1206 break;
1207 }
1208 return total_size;
1209}
1218 if (!temp_arena || !type)
1219 return 0;
1220 estimate_visited_node_t * visited_head = nullptr;
1221 return _estimate_graph_size_recursive(temp_arena, type, &visited_head);
1222}
1223// Public API: Introspection Functions
1231 if (type == nullptr)
1232 return nullptr;
1233 return type->name;
1234}
1275 return nullptr;
1276 return &type->meta.aggregate_info.members[index];
1277}
1285INFIX_API c23_nodiscard const char * infix_type_get_arg_name(const infix_type * func_type, size_t index) {
1286 if (!func_type || func_type->category != INFIX_TYPE_REVERSE_TRAMPOLINE ||
1287 index >= func_type->meta.func_ptr_info.num_args)
1288 return nullptr;
1289 return func_type->meta.func_ptr_info.args[index].name;
1290}
1297INFIX_API c23_nodiscard const infix_type * infix_type_get_arg_type(const infix_type * func_type, size_t index) {
1298 if (!func_type || func_type->category != INFIX_TYPE_REVERSE_TRAMPOLINE ||
1299 index >= func_type->meta.func_ptr_info.num_args)
1300 return nullptr;
1301 return func_type->meta.func_ptr_info.args[index].type;
1302}
1309 return trampoline ? trampoline->num_args : 0;
1310}
1317 return trampoline ? trampoline->num_fixed_args : 0;
1318}
1325 return trampoline ? trampoline->return_type : nullptr;
1326}
1334 size_t index) {
1335 if (!trampoline || index >= trampoline->num_args)
1336 return nullptr;
1337 return trampoline->arg_types[index];
1338}
1345 return trampoline ? trampoline->num_args : 0;
1346}
1353 return trampoline ? trampoline->num_fixed_args : 0;
1354}
1361 return trampoline ? trampoline->return_type : nullptr;
1362}
1370 size_t index) {
1371 if (!trampoline || index >= trampoline->num_args)
1372 return nullptr;
1373 return trampoline->arg_types[index];
1374}
infix_arena_t * arena
Definition 005_layouts.c:62
infix_status status
Definition 103_unions.c:61
infix_struct_member * members
Definition 103_unions.c:55
clock_t start
Definition 901_call_overhead.c:48
#define c23_nodiscard
Internal alias for the public INFIX_NODISCARD macro.
Definition compat_c23.h:92
@ INFIX_CODE_INVALID_MEMBER_TYPE
Definition infix.h:1388
@ INFIX_CODE_INVALID_ALIGNMENT
Definition infix.h:1372
@ INFIX_CODE_INTEGER_OVERFLOW
Definition infix.h:1379
@ INFIX_CODE_TYPE_TOO_LARGE
Definition infix.h:1386
@ INFIX_CODE_NULL_POINTER
Definition infix.h:1364
@ INFIX_CODE_OUT_OF_MEMORY
Definition infix.h:1369
@ INFIX_CATEGORY_ABI
Definition infix.h:1355
@ INFIX_CATEGORY_ALLOCATION
Definition infix.h:1353
@ INFIX_CATEGORY_GENERAL
Definition infix.h:1352
@ INFIX_CATEGORY_PARSER
Definition infix.h:1354
struct infix_type_t::@0::@1 pointer_info
Metadata for INFIX_TYPE_POINTER.
union infix_type_t::@0 meta
A union containing metadata specific to the type's category.
bool is_packed
Definition infix.h:295
struct infix_type_t::@0::@7 vector_info
Metadata for INFIX_TYPE_VECTOR.
bool is_incomplete
Definition infix.h:280
infix_type * type
Definition infix.h:348
struct infix_type_t::@0::@4 func_ptr_info
Metadata for INFIX_TYPE_REVERSE_TRAMPOLINE.
size_t num_elements
Definition infix.h:300
infix_arena_t * arena
Definition infix.h:281
size_t size
Definition infix.h:277
size_t alignment
Definition infix.h:278
uint8_t bit_offset
Definition infix.h:339
infix_struct_member * members
Definition infix.h:293
bool is_bitfield
Definition infix.h:340
struct infix_type_t::@0::@6 complex_info
Metadata for INFIX_TYPE_COMPLEX.
const char * name
Definition infix.h:275
infix_function_argument * args
Definition infix.h:306
infix_aggregate_category_t aggregate_category
Definition infix.h:326
infix_status
Enumerates the possible status codes returned by infix API functions.
Definition infix.h:434
const char * name
Definition infix.h:347
const char * name
Definition infix.h:335
infix_type_category category
Definition infix.h:276
struct infix_type_t::@0::@2 aggregate_info
Metadata for INFIX_TYPE_STRUCT and INFIX_TYPE_UNION.
struct infix_type_t::@0::@3 array_info
Metadata for INFIX_TYPE_ARRAY.
struct infix_type_t * pointee_type
Definition infix.h:289
infix_type * type
Definition infix.h:336
struct infix_type_t * element_type
Definition infix.h:299
bool is_flexible
Definition infix.h:301
struct infix_type_t * return_type
Definition infix.h:305
size_t offset
Definition infix.h:337
struct infix_type_t::@0::@5 enum_info
Metadata for INFIX_TYPE_ENUM.
struct infix_type_t * base_type
Definition infix.h:316
uint8_t bit_width
Definition infix.h:338
size_t num_members
Definition infix.h:294
struct infix_type_t * underlying_type
Definition infix.h:312
struct infix_type_t::@0::@8 named_reference
Metadata for INFIX_TYPE_NAMED_REFERENCE.
infix_primitive_type_id primitive_id
Metadata for INFIX_TYPE_PRIMITIVE.
Definition infix.h:286
size_t num_args
Definition infix.h:307
bool is_arena_allocated
Definition infix.h:279
@ INFIX_ERROR_ALLOCATION_FAILED
Definition infix.h:436
@ INFIX_SUCCESS
Definition infix.h:435
@ INFIX_ERROR_INVALID_ARGUMENT
Definition infix.h:437
INFIX_API c23_nodiscard size_t infix_forward_get_num_args(const infix_forward_t *trampoline)
Gets the total number of arguments for a forward trampoline.
Definition types.c:1308
INFIX_API INFIX_NODISCARD void * infix_arena_alloc(infix_arena_t *, size_t, size_t)
Allocates a block of memory from an arena.
Definition arena.c:117
#define infix_memcpy
A macro that can be defined to override the default memcpy function.
Definition infix.h:387
INFIX_API INFIX_NODISCARD infix_arena_t * infix_arena_create(size_t)
Creates a new memory arena.
Definition arena.c:52
INFIX_API INFIX_NODISCARD void * infix_arena_calloc(infix_arena_t *, size_t, size_t, size_t)
Allocates and zero-initializes a block of memory from an arena.
Definition arena.c:188
INFIX_API void infix_arena_destroy(infix_arena_t *)
Destroys an arena and frees all memory allocated from it.
Definition arena.c:83
struct infix_type_t infix_type
A semi-opaque object describing a C type's memory layout and calling convention. See infix_type_t for...
Definition infix.h:207
c23_nodiscard infix_status infix_type_create_pointer_to(infix_arena_t *arena, infix_type **out_type, infix_type *pointee_type)
Creates a new pointer type that points to a specific type.
Definition types.c:393
INFIX_API infix_struct_member infix_type_create_member(const char *name, infix_type *type, size_t offset)
A factory function to create an infix_struct_member.
Definition types.c:208
INFIX_API c23_nodiscard infix_status infix_type_create_packed_struct(infix_arena_t *arena, infix_type **out_type, size_t total_size, size_t alignment, infix_struct_member *members, size_t num_members)
Creates a new packed struct type with a user-specified layout.
Definition types.c:719
INFIX_API c23_nodiscard infix_status infix_type_create_complex(infix_arena_t *arena, infix_type **out_type, infix_type *base_type)
Creates a new _Complex number type.
Definition types.c:542
INFIX_API c23_nodiscard const infix_type * infix_forward_get_arg_type(const infix_forward_t *trampoline, size_t index)
Gets the type of a specific argument for a forward trampoline.
Definition types.c:1333
infix_primitive_type_id
Enumerates the supported primitive C types.
Definition infix.h:243
INFIX_API c23_nodiscard infix_type * infix_type_create_void(void)
Creates a static descriptor for the void type.
Definition types.c:200
INFIX_API c23_nodiscard const infix_struct_member * infix_type_get_member(const infix_type *type, size_t index)
Gets a specific member from a struct or union type.
Definition types.c:1272
INFIX_API c23_nodiscard infix_status infix_type_create_union(infix_arena_t *arena, infix_type **out_type, infix_struct_member *members, size_t num_members)
Creates a new union type from an array of members.
Definition types.c:612
INFIX_API c23_nodiscard const char * infix_type_get_arg_name(const infix_type *func_type, size_t index)
Gets the name of a specific argument from a function type.
Definition types.c:1285
struct infix_struct_member_t infix_struct_member
A semi-opaque object describing a member of a C struct or union. See infix_struct_member_t for detail...
Definition infix.h:209
INFIX_API infix_struct_member infix_type_create_bitfield_member(const char *name, infix_type *type, size_t offset, uint8_t bit_width)
A factory function to create a bitfield infix_struct_member.
Definition types.c:219
INFIX_API c23_nodiscard size_t infix_type_get_size(const infix_type *type)
Gets the size of a type in bytes.
Definition types.c:1248
INFIX_API c23_nodiscard const infix_type * infix_forward_get_return_type(const infix_forward_t *trampoline)
Gets the return type for a forward trampoline.
Definition types.c:1324
INFIX_API c23_nodiscard infix_status infix_type_create_named_reference(infix_arena_t *arena, infix_type **out_type, const char *name, infix_aggregate_category_t agg_cat)
Creates a placeholder for a named type that will be resolved later by a type registry.
Definition types.c:777
INFIX_API c23_nodiscard const infix_type * infix_reverse_get_return_type(const infix_reverse_t *trampoline)
Gets the return type for a reverse trampoline.
Definition types.c:1360
INFIX_API c23_nodiscard const infix_type * infix_type_get_arg_type(const infix_type *func_type, size_t index)
Gets the type of a specific argument from a function type.
Definition types.c:1297
INFIX_API c23_nodiscard size_t infix_type_get_member_count(const infix_type *type)
Gets the number of members in a struct or union type.
Definition types.c:1261
struct infix_function_argument_t infix_function_argument
A semi-opaque object describing an argument to a C function. See infix_function_argument_t for detail...
Definition infix.h:211
INFIX_API c23_nodiscard infix_type_category infix_type_get_category(const infix_type *type)
Gets the fundamental category of a type.
Definition types.c:1240
INFIX_API c23_nodiscard infix_type * infix_type_create_primitive(infix_primitive_type_id id)
Creates a static descriptor for a primitive C type.
Definition types.c:144
INFIX_API c23_nodiscard infix_type * infix_type_create_pointer(void)
Creates a static descriptor for a generic pointer (void*).
Definition types.c:195
INFIX_API c23_nodiscard size_t infix_reverse_get_num_fixed_args(const infix_reverse_t *trampoline)
Gets the number of fixed (non-variadic) arguments for a reverse trampoline.
Definition types.c:1352
infix_type_category
Enumerates the fundamental categories of types that infix can represent.
Definition infix.h:227
INFIX_API c23_nodiscard infix_status infix_type_create_vector(infix_arena_t *arena, infix_type **out_type, infix_type *element_type, size_t num_elements)
Creates a new SIMD vector type.
Definition types.c:573
INFIX_API c23_nodiscard const infix_type * infix_reverse_get_arg_type(const infix_reverse_t *trampoline, size_t index)
Gets the type of a specific argument for a reverse trampoline.
Definition types.c:1369
INFIX_API c23_nodiscard const char * infix_type_get_name(const infix_type *type)
Gets the semantic alias of a type, if one exists.
Definition types.c:1230
INFIX_API c23_nodiscard infix_status infix_type_create_struct(infix_arena_t *arena, infix_type **out_type, infix_struct_member *members, size_t num_members)
Creates a new struct type from an array of members, calculating layout automatically.
Definition types.c:660
infix_aggregate_category_t
Specifies whether a named type reference refers to a struct or a union.
Definition infix.h:264
INFIX_API c23_nodiscard size_t infix_forward_get_num_fixed_args(const infix_forward_t *trampoline)
Gets the number of fixed (non-variadic) arguments for a forward trampoline.
Definition types.c:1316
INFIX_API c23_nodiscard infix_status infix_type_create_enum(infix_arena_t *arena, infix_type **out_type, infix_type *underlying_type)
Creates a new enum type with a specified underlying integer type.
Definition types.c:507
INFIX_API c23_nodiscard size_t infix_reverse_get_num_args(const infix_reverse_t *trampoline)
Gets the total number of arguments for a reverse trampoline.
Definition types.c:1344
INFIX_API c23_nodiscard infix_status infix_type_create_flexible_array(infix_arena_t *arena, infix_type **out_type, infix_type *element_type)
Creates a new flexible array member type ([?:type]).
Definition types.c:463
INFIX_API c23_nodiscard infix_status infix_type_create_array(infix_arena_t *arena, infix_type **out_type, infix_type *element_type, size_t num_elements)
Creates a new fixed-size array type.
Definition types.c:424
INFIX_API c23_nodiscard size_t infix_type_get_alignment(const infix_type *type)
Gets the alignment requirement of a type in bytes.
Definition types.c:1254
@ INFIX_PRIMITIVE_UINT16
Definition infix.h:247
@ INFIX_PRIMITIVE_UINT32
Definition infix.h:249
@ INFIX_PRIMITIVE_LONG_DOUBLE
Definition infix.h:258
@ INFIX_PRIMITIVE_FLOAT
Definition infix.h:256
@ INFIX_PRIMITIVE_DOUBLE
Definition infix.h:257
@ INFIX_PRIMITIVE_SINT16
Definition infix.h:248
@ INFIX_PRIMITIVE_SINT64
Definition infix.h:252
@ INFIX_PRIMITIVE_SINT32
Definition infix.h:250
@ INFIX_PRIMITIVE_UINT8
Definition infix.h:245
@ INFIX_PRIMITIVE_UINT128
Definition infix.h:253
@ INFIX_PRIMITIVE_BOOL
Definition infix.h:244
@ INFIX_PRIMITIVE_UINT64
Definition infix.h:251
@ INFIX_PRIMITIVE_FLOAT16
Definition infix.h:255
@ INFIX_PRIMITIVE_SINT128
Definition infix.h:254
@ INFIX_PRIMITIVE_SINT8
Definition infix.h:246
@ INFIX_TYPE_UNION
Definition infix.h:231
@ INFIX_TYPE_PRIMITIVE
Definition infix.h:228
@ INFIX_TYPE_COMPLEX
Definition infix.h:235
@ INFIX_TYPE_ARRAY
Definition infix.h:232
@ INFIX_TYPE_VECTOR
Definition infix.h:236
@ INFIX_TYPE_VOID
Definition infix.h:238
@ INFIX_TYPE_POINTER
Definition infix.h:229
@ INFIX_TYPE_NAMED_REFERENCE
Definition infix.h:237
@ INFIX_TYPE_REVERSE_TRAMPOLINE
Definition infix.h:233
@ INFIX_TYPE_ENUM
Definition infix.h:234
@ INFIX_TYPE_STRUCT
Definition infix.h:230
#define INFIX_API
Symbol visibility macro.
Definition infix.h:114
Internal data structures, function prototypes, and constants.
static size_t _infix_align_up(size_t value, size_t alignment)
Aligns a value up to the next multiple of a power-of-two alignment.
Definition infix_internals.h:767
static bool is_double(const infix_type *type)
A fast inline check to determine if an infix_type is a double.
Definition infix_internals.h:791
INFIX_INTERNAL void _infix_clear_error(void)
Clears the thread-local error state.
Definition error.c:268
static bool is_float(const infix_type *type)
A fast inline check to determine if an infix_type is a float (32-bit).
Definition infix_internals.h:783
INFIX_INTERNAL void _infix_set_error(infix_error_category_t category, infix_error_code_t code, size_t position)
Sets the thread-local error state with detailed information.
Definition error.c:175
Definition types.c:1121
struct estimate_visited_node_t * next
Definition types.c:1123
const infix_type * type
Definition types.c:1122
Internal definition of a memory arena.
Definition infix_internals.h:143
Internal definition of a forward trampoline handle.
Definition infix_internals.h:90
size_t num_args
Definition infix_internals.h:97
size_t num_fixed_args
Definition infix_internals.h:98
infix_type ** arg_types
Definition infix_internals.h:96
infix_type * return_type
Definition infix_internals.h:95
Describes a single argument to a C function.
Definition infix.h:346
Internal definition of a reverse trampoline (callback/closure) handle.
Definition infix_internals.h:119
infix_type * return_type
Definition infix_internals.h:123
size_t num_args
Definition infix_internals.h:125
size_t num_fixed_args
Definition infix_internals.h:126
infix_type ** arg_types
Definition infix_internals.h:124
Describes a single member of a C struct or union.
Definition infix.h:334
A semi-opaque structure that describes a C type.
Definition infix.h:274
Definition types.c:941
infix_type * dest
Definition types.c:943
struct memo_node_t * next
Definition types.c:944
const infix_type * src
Definition types.c:942
Definition types.c:823
struct recalc_visited_node_t * next
Definition types.c:825
infix_type * type
Definition types.c:824
static size_t _estimate_graph_size_recursive(infix_arena_t *temp_arena, const infix_type *type, estimate_visited_node_t **visited_head)
Definition types.c:1137
static infix_type _infix_type_sint128
Definition types.c:116
void _infix_type_recalculate_layout(infix_type *type)
Recalculates the layout of a fully resolved type graph.
Definition types.c:924
static infix_type _infix_type_double
Definition types.c:130
static infix_type _infix_type_uint32
Definition types.c:105
static infix_type _infix_type_bool
Definition types.c:95
static infix_type _infix_type_float
Definition types.c:128
infix_type * _copy_type_graph_to_arena(infix_arena_t *dest_arena, const infix_type *src_type)
Performs a deep copy of a type graph into a destination arena.
Definition types.c:1112
static infix_type _infix_type_sint64
Definition types.c:111
#define INFIX_TYPE_INIT(id, T)
Definition types.c:59
size_t _infix_estimate_graph_size(infix_arena_t *temp_arena, const infix_type *type)
Estimates the total memory required to deep-copy a complete type graph.
Definition types.c:1217
static infix_type _infix_type_sint16
Definition types.c:103
#define INFIX_MAX_ALIGNMENT
Definition types.c:44
static infix_type _infix_type_uint16
Definition types.c:101
static infix_type _infix_type_long_double
Definition types.c:135
static infix_type _infix_type_pointer
Definition types.c:86
static infix_type _infix_type_uint128
Definition types.c:114
static infix_type _infix_type_void
Definition types.c:73
static infix_type _infix_type_sint32
Definition types.c:107
static infix_type * _copy_type_graph_to_arena_recursive(infix_arena_t *dest_arena, const infix_type *src_type, memo_node_t **memo_head)
Definition types.c:960
static infix_type _infix_type_uint64
Definition types.c:109
static infix_type _infix_type_float16
Definition types.c:119
static infix_status _create_aggregate_setup(infix_arena_t *arena, infix_type **out_type, infix_struct_member **out_arena_members, infix_struct_member *members, size_t num_members)
Definition types.c:350
static void _infix_type_recalculate_layout_recursive(infix_arena_t *temp_arena, infix_type *type, recalc_visited_node_t **visited_head)
Definition types.c:849
static bool _layout_struct(infix_type *type)
Definition types.c:231
static infix_type _infix_type_uint8
Definition types.c:97
static infix_type _infix_type_sint8
Definition types.c:99
A header for conditionally compiled debugging utilities.