Affix

 view release on metacpan or  search on metacpan

t/005_varargs.t  view on Meta::CPAN

            case 'd': total += (int)va_arg(ap, double); break;
            case 's': {
                char* s = va_arg(ap, char*);
                total += s ? strlen(s) : 0;
                break;
            }
            case 'P': {
                Point p = va_arg(ap, Point);
                total += p.x + p.y;
                break;
            }
        }
    }
    va_end(ap);
    return total;
}
END_C
my $lib = compile_ok( $c_code, { name => 'variadic_dynamic_lib' } );

# Bind with "Empty" Variadic Signature
# We define the fixed arguments (*char) and end with a semicolon.
# We do NOT specify any optional types here. Affix must generate them at runtime.
# Signature: (*char;)->int
isa_ok my $fn = wrap( $lib, 'var_sum', [ Pointer [Char], VarArgs ] => Int ), ['Affix'];
subtest 'Runtime Type Inference' => sub {

    # 3 Integers
    # Affix should generate JIT for: (*char; sint64, sint64, sint64)->int
    is $fn->( "iii", 10, 20, 30 ), 60, 'Inferred 3 integers';

    # Floats (promoted to double)
    # 1.5 -> 1, 2.5 -> 2. Sum = 3.
    is $fn->( "dd", 1.5, 2.5 ), 3, 'Doubles passed correctly';

    # Mixed Types
    # (*char; sint64, double)->int
    # Note: 2.5 cast to int is 2
    is $fn->( "id", 100, 2.5 ), 102, 'Inferred mixed int/double';

    # Strings (*char)
    # len("Hello") = 5
    is $fn->( "s", "Hello" ), 5, 'Strings passed correctly';

    # Edge case where we don't pass any varargs. Like calling `printf( "Just this string, please." );`
    # (*char;)->int
    is $fn->(""), 0, 'Called with no variadic arguments';
};
subtest 'Runtime Coercion (Structs)' => sub {
    typedef Point__ => Struct [ x => Int, y => Int ];
    my $p1 = { x => 1, y => 2 };    # Sum = 3
    my $p2 = { x => 3, y => 4 };    # Sum = 7

    # Coerced Struct
    # Affix should generate JIT for: (*char; {x:int,y:int}, {x:int,y:int})->int
    is $fn->( "PP", coerce( Point__(), $p1 ), coerce( Point__(), $p2 ) ), 10, 'Dynamically generated signature for coerced structs';
};

# Test Safety / Constraints
subtest 'Safety' => sub {

    # coerce() modifies the SV (attaches magic).
    # It cannot function on read-only constants.
    like dies {
        $fn->( "i", coerce( Int, 10 ) );
    }, qr/read-only/, 'coerce(Int, 10) correctly dies on read-only value';

    # Correct usage with variable
    my $v = 100;
    is $fn->( "i", coerce( Int, $v ) ), 100, 'coerce(Int, $var) works';
};
subtest coercion => sub {
    typedef Point_ => Struct [ x => Int, y => Int ];

    # Without coerce(), Affix wouldn't know to pass this as a Point_ struct by value.
    my $pt = { x => 10, y => 20 };

    # coerce() attaches magic to $pt telling Affix to treat it as a @Point.
    # The JIT generates a trampoline expecting a struct {int,int} on the stack/registers.
    # 10 + 20 = 30.
    is $fn->( 'P', coerce( Point_(), $pt ) ), 30, 'Struct passed by value using coerce()';

    # Reuse the trampoline (Cache Hit check)
    # The signature generated for ('P', Point) should be cached.
    my $pt2 = { x => 5, y => 5 };
    is $fn->( 'P', coerce( Point_(), $pt2 ) ), 10, 'Repeated call with coerced struct works';

    # Ensure previous calls didn't "lock" the signature to a specific set of types.
    # Call with 1 Integer (different from previous calls)
    is $fn->( 'i', 99 ), 99, 'Signature remains dynamic for new argument patterns';
};
subtest 'printf' => sub {    # Should be safe/standard across platforms
    ok my $printf = wrap( libc, 'printf', [ String, VarArgs ], Int ), q[wrap( libc, 'printf', [ String, VarArgs ], Int)];
    is $printf->( '# %s: %d <---', 'Test', 100 ), 16, q[->( ... )];
};
done_testing;



( run in 0.535 second using v1.01-cache-2.11-cpan-e1769b4cff6 )