Affix
view release on metacpan or search on metacpan
lib/Affix.h view on Meta::CPAN
#pragma once
#define DEBUG 0
// Disables the implicit 'pTHX_' context pointer argument, which is good practice for
// modern Perl XS code that uses the 'aTHX_' macro explicitly.
#define PERL_NO_GET_CONTEXT 1
#include <EXTERN.h>
#include <perl.h>
// Disables Perl's internal locking mechanisms for certain structures.
// This is often used when the XS module manages its own thread safety.
#define NO_XSLOCKS
#include <XSUB.h>
// Redirect infix's internal memory allocation to use Perl's safe allocation functions.
// This ensures all memory is tracked by Perl's memory manager, which is safer and
// helps with leak detection tools like valgrind.
#define infix_malloc safemalloc
#define infix_calloc safecalloc
#define infix_free safefree
#define infix_realloc saferealloc
#include "common/infix_internals.h"
#include <infix/infix.h>
// This structure defines the thread-local storage for our module. Under ithreads,
// each Perl thread will get its own private instance of this struct.
typedef struct {
/// A per-thread hash table to store loaded libraries.
/// Maps library path -> LibRegistryEntry*.
/// This prevents reloading the same .so/.dll and manages its lifecycle.
HV * lib_registry;
// A per-thread hash table to cache callback trampolines, preventing re-creation and leaks.
// Maps the memory address of a Perl CV* to its corresponding Implicit_Callback_Magic* struct.
HV * callback_registry;
/// Type alias for an infix type registry. Represents a collection of named types.
infix_registry_t * registry;
/// // Smart enums
HV * enum_registry;
// Cache for coercion strings to avoid re-fetching from SV objects
HV * coercion_cache;
HV * stash_pointer; // Cache for Affix::Pointer stash
} my_cxt_t;
START_MY_CXT;
// Helper macro to fetch a value from a hash if it exists, otherwise return a default.
#define hv_existsor(hv, key, _or) hv_exists(hv, key, strlen(key)) ? *hv_fetch(hv, key, strlen(key), 0) : _or
// Macros to handle passing the Perl interpreter context ('THX') explicitly,
// which is necessary for thread-safe code.
#ifdef MULTIPLICITY
#define storeTHX(var) (var) = aTHX
#define dTHXfield(var) tTHX var;
#else
#define storeTHX(var) dNOOP
#define dTHXfield(var)
#endif
// Forward-declare the primary structures.
typedef struct Affix Affix;
typedef struct Affix_Backend Affix_Backend;
typedef struct Affix_Plan_Step Affix_Plan_Step;
typedef struct OutParamInfo OutParamInfo;
/**
* The single, homogeneous function pointer signature for all steps in the execution plan.
* @param pTHX_ The Perl interpreter context.
* @param affix The main Affix context object.
* @param step A pointer to the current plan step, containing its pre-calculated data.
* @param perl_stack_frame A pointer to the base of the Perl stack frame (&ST(0)).
* @param c_args The array of pointers to be passed to the C function.
* @param ret_buffer A pointer to the memory allocated for the C function's return value.
*/
typedef void (*Affix_Step_Executor)(pTHX_ Affix * affix,
Affix_Plan_Step * step,
SV ** perl_stack_frame,
void * args_buffer,
void ** c_args,
void * ret_buffer);
/// Function pointer type for a "pull" operation: marshalling from C (void*) to Perl (SV).
typedef void (*Affix_Pull)(pTHX_ Affix * affix, SV *, const infix_type *, void *);
/// Function pointer type for a "push" operation: marshalling from Perl (SV) to C (void*).
typedef void (*Affix_Push_Handler)(pTHX_ Affix * affix, SV *, void *);
/**
* Function pointer for a specialized out-parameter write-back handler.
* By pre-resolving this function, we avoid conditional logic in the hot path.
* @param pTHX_ The Perl interpreter context.
* @param affix The main Affix context object.
* @param info A pointer to the OutParamInfo struct for this parameter.
* @param perl_sv The referenced SV* to be modified (e.g., the scalar backing `$$foo`).
* @param c_arg_ptr The pointer from the c_args array (e.g., `T**` for a `T*` out-param).
*/
typedef void (*Affix_Out_Param_Writer)(pTHX_ Affix * affix, const OutParamInfo * info, SV * perl_sv, void * c_arg_ptr);
/// Stores the pre-calculated information needed to write back an "out" parameter.
struct OutParamInfo {
size_t perl_stack_index; // Index of the SV* in the perl_stack_frame
const infix_type * pointee_type; // The type of the data pointed to (e.g., 'int' for 'int*')
Affix_Out_Param_Writer writer; // Pre-resolved handler to perform the write-back.
};
/// The data payload for a single step in the execution plan.
typedef struct {
const infix_type * type; // Type info for this step (arg or ret).
size_t index; // Index into perl_stack_frame for args, or c_args for out-params.
Affix_Pull pull_handler; // Pre-resolved pull handler for the return step.
size_t c_arg_offset; // Pre-calculated offset into the C arguments buffer.
} Affix_Step_Data;
typedef enum {
// argument marshalling opcodes
OP_PUSH_BOOL,
lib/Affix.h view on Meta::CPAN
char * lib_path;
dTHXfield(owner_perl)
};
// Trigger function for the experimental backend (shh!)
extern void Affix_trigger_backend(pTHX_ CV *);
// Main execution trigger
extern void Affix_trigger_stack(pTHX_ CV *);
extern void Affix_trigger_arena(pTHX_ CV *);
extern void Affix_trigger_variadic(pTHX_ CV *);
// Marshalling (Perl -> C)
void sv2ptr(pTHX_ Affix * affix, SV * perl_sv, void * c_ptr, const infix_type * type);
void push_struct(pTHX_ Affix * affix, const infix_type * type, SV * sv, void * p);
void push_array(pTHX_ Affix * affix, const infix_type * type, SV * sv, void * p);
void push_reverse_trampoline(pTHX_ Affix * affix, const infix_type * type, SV * sv, void * p);
// Marshalling (Perl <- C)
void ptr2sv(pTHX_ Affix * affix, void * c_ptr, SV * perl_sv, const infix_type * type);
void _populate_hv_from_c_struct(
pTHX_ Affix * affix, HV * hv, const infix_type * type, void * p, bool live, SV * owner_sv);
// Handler resolution
Affix_Step_Executor get_plan_step_executor(const infix_type * type);
Affix_Pull get_pull_handler(pTHX_ const infix_type * type);
Affix_Out_Param_Writer get_out_param_writer(const infix_type * type);
// Pin management
void _pin_sv(pTHX_ SV * sv,
const infix_type * type,
void * pointer,
bool managed,
SV * owner_sv,
size_t bit_offset,
size_t bit_width);
bool is_pin(pTHX_ SV * sv);
Affix_Pin * _get_pin_from_sv(pTHX_ SV * sv);
SV * _new_pointer_obj(pTHX_ Affix_Pin * pin);
// Reverse trampolines
void _affix_callback_handler_entry(infix_context_t *, void *, void **);
// Misc.
void _export_function(pTHX_ HV *, const char *, const char *);
// XS Bootstrap
void boot_Affix(pTHX_ CV *);
// 'Portable' XS MACROS
#ifdef newXS_flags
#define newXSproto_portable(name, c_impl, file, proto) newXS_flags(name, c_impl, file, proto, 0)
#else
#define newXSproto_portable(name, c_impl, file, proto) \
(PL_Sv = (SV *)newXS(name, c_impl, file), sv_setpv(PL_Sv, proto), (CV *)PL_Sv)
#endif
#define newXS_deffile(a, b) Perl_newXS_deffile(aTHX_ a, b)
#define export_function(package, what, tag) \
_export_function(aTHX_ get_hv(form("%s::EXPORT_TAGS", package), GV_ADD), what, tag)
// Debugging Macros
#if DEBUG > 1
#define PING warn("Ping at %s line %d", __FILE__, __LINE__);
#else
#define PING
#endif
#define DumpHex(addr, len) _DumpHex(aTHX_ addr, len, __FILE__, __LINE__)
void _DumpHex(pTHX_ const void *, size_t, const char *, int);
#define DD(scalar) _DD(aTHX_ scalar, __FILE__, __LINE__)
void _DD(pTHX_ SV *, const char *, int);
#include <string.h>
#include <wchar.h>
( run in 1.899 second using v1.01-cache-2.11-cpan-99c4e6809bf )