Affix

 view release on metacpan or  search on metacpan

t/019_fileio.t  view on Meta::CPAN

        // Define a struct that contains a file pointer
        typedef struct {
            FILE* log_file;
            int   counter;
        } Logger;

        // Initialize logger with a file
        DLLEXPORT void init_logger(Logger* logger, FILE* fp) {
            if (!fp) fprintf(stderr, "C-side Warning: fp is NULL\n");
            logger->log_file = fp;
            logger->counter = 0;
        }

        // Write to a file retrieved from the struct
        DLLEXPORT void log_message(Logger* logger, const char* msg) {
            if (logger->log_file) {
                fprintf(logger->log_file, "[%d] %s\n", ++logger->counter, msg);
                fflush(logger->log_file);
            }
        }

        // Return a struct containing a file pointer
        DLLEXPORT Logger create_logger(FILE* fp) {
            Logger l;
            l.log_file = fp;
            l.counter = 100;
            return l;
        }
        END_C

    # Define the struct type in Perl.
    # Use Pointer[File] because the C struct member is FILE*.
    typedef Logger => Struct [ log_file => Pointer [File], counter => Int ];

    # Bind functions
    affix $lib, 'init_logger',   [ Pointer [ Logger() ], Pointer [File] ] => Void;
    affix $lib, 'log_message',   [ Pointer [ Logger() ], String ]         => Void;
    affix $lib, 'create_logger', [ Pointer [File] ] => Logger();
    subtest 'File inside Struct (Pointer)' => sub {
        my ( $fh, $filename ) = tempfile();
        my $old_fh = select($fh);
        $| = 1;
        select($old_fh);

        # Allocate struct memory
        my $logger = malloc( sizeof( Logger() ) );

        # Pass filehandle to C to store in struct
        init_logger( $logger, $fh );

        # Verify via C function
        log_message( $logger, 'First message' );
        log_message( $logger, 'Second message' );

        # Verify Perl side struct access
        # Note: Pulling a File handle usually creates a new GLOB wrapper around the FILE*
        # Since we own $fh, let's verify checking against undef works
        my $logger_struct = cast( $logger, Logger() );    # View as struct
        my $retrieved_fh  = $logger_struct->{log_file};
        ok $retrieved_fh, 'Retrieved filehandle from struct';
        is ref($retrieved_fh), 'GLOB', 'It is a glob';

        # Write from Perl using retrieved handle
        # print {$retrieved_fh} "From Perl\n"; # Careful, might double-close if not careful
        # Check file content
        open my $check, '<', $filename;
        my @lines = <$check>;
        close $check;
        is scalar(@lines), 2, 'File has 2 lines';
        like $lines[0], qr/\[1\] First message/,  'Line 1 matches';
        like $lines[1], qr/\[2\] Second message/, 'Line 2 matches';
        free($logger);

        # Keep $fh alive until test end to avoid closing underneath C
        close $fh;
    };
    subtest 'File inside Struct (Value Return)' => sub {
        my ( $fh, $filename ) = tempfile();
        my $old_fh = select($fh);
        $| = 1;
        select($old_fh);

        # Call C function returning a struct by value
        my $logger_hash = create_logger($fh);
        is $logger_hash->{counter}, 100, 'Counter is correct';
        ok $logger_hash->{log_file}, 'Got filehandle back';
        is ref( $logger_hash->{log_file} ), 'GLOB', 'It is a glob';

        # Write using the returned handle to verify it works
        # Note: $logger_hash->{log_file} wraps the same FILE* as $fh.
        ok syswrite( $logger_hash->{log_file}, "Direct write from Perl\n" ), 'syswrite to the handle from Perl';

        # To avoid double-close warnings, we let Perl handle cleanup of the glob
        # but be careful about explicit closes.
        undef $logger_hash;

        # Check
        open my $check, '<', $filename;
        my $content = <$check>;
        close $check;
        is $content, "Direct write from Perl\n", 'Handle returned in struct is usable';
        close $fh;
    };
    subtest 'File in Array' => sub {
        my $lib2 = compile_ok(<<~'END_C2');
        #include "std.h"
        //ext: .c

        #include <stdio.h>

        DLLEXPORT void write_all(FILE* files[3], const char* msg) {
            for(int i=0; i<3; i++) {
                if(files[i]) fprintf(files[i], "%s", msg);
            }
        }
        END_C2

        # Array of Pointers to Files (FILE* files[3])
        affix $lib2, 'write_all', [ Array [ Pointer [File], 3 ], String ] => Void;
        my ( $fh1, $f1 ) = tempfile();
        my ( $fh2, $f2 ) = tempfile();
        my ( $fh3, $f3 ) = tempfile();

        # Flush buffers
        for my $h ( $fh1, $fh2, $fh3 ) { my $o = select($h); $| = 1; select($o); }

        # Pass array of handles
        write_all( [ $fh1, $fh2, $fh3 ], 'Broadcast' );
        close $_ for ( $fh1, $fh2, $fh3 );

        # Verify
        for my $f ( $f1, $f2, $f3 ) {
            open my $in, '<', $f;
            is <$in>, 'Broadcast', "File $f written to";
            close $in;
        }
    };
    subtest PerlIO => sub {

        # Define C code.
        # NOTE: We use void* and PerlIO types to avoid Windows CRT mismatch crashes.
        # This proves that we can store ANY pointer-sized handle in a struct and retrieve it.
        my $C_CODE = <<'END_C';
#include "std.h"
//ext: .c
// Define a struct that contains a handle (void* to allow PerlIO or FILE)
typedef struct {
    void* handle;
    int   counter;
} Logger;

// 1. Initialize logger with a handle
DLLEXPORT void init_logger2(Logger* logger, void* fp) {
    logger->handle = fp;
    logger->counter = 0;
}

// 2. Return a struct containing a handle
DLLEXPORT Logger create_logger2(void* fp) {
    Logger l;
    l.handle = fp;
    l.counter = 100;
    return l;
}

END_C
        my $lib = compile_ok($C_CODE);

        # Define the struct type in Perl using PerlIO for the handle
        # Struct member is PerlIO* so use Pointer[PerlIO]
        typedef Logger2 => Struct [ handle => Pointer [PerlIO], counter => Int ];

        # Bind functions using the defined struct
        affix $lib, 'init_logger2',   [ Pointer [ Logger2() ], Pointer [PerlIO] ] => Void;
        affix $lib, 'create_logger2', [ Pointer [PerlIO] ]                        => Logger2();
        subtest 'PerlIO inside Struct (Pointer)' => sub {
            my ( $fh, $filename ) = tempfile();
            syswrite $fh, "Original Content\n";

            # Allocate struct memory
            my $logger = malloc( sizeof( Logger2() ) );

            # Pass Perl filehandle to C. C stores the PerlIO* address.
            init_logger2( $logger, $fh );

            # Verify we can retrieve it back as a Glob
            my $logger_struct = cast( $logger, Logger2() );
            my $retrieved_fh  = $logger_struct->{handle};
            ok $retrieved_fh, 'Retrieved filehandle from struct';
            is ref($retrieved_fh), 'GLOB', 'It is a glob';

            # Verify it points to the same stream by writing to it
            syswrite $retrieved_fh, "Appended via Struct\n";
            close $fh;

            # Check file content
            open my $check, '<', $filename;
            my @lines = <$check>;
            close $check;
            is scalar(@lines), 2, 'File has 2 lines';
            like $lines[0], qr/Original Content/,    'Line 1 matches';
            like $lines[1], qr/Appended via Struct/, 'Line 2 matches';
            free($logger);
        };
        subtest 'PerlIO inside Struct (Value Return)' => sub {
            my ( $fh, $filename ) = tempfile();

            # Call C function returning a struct by value
            my $logger_hash = create_logger2($fh);
            is $logger_hash->{counter}, 100, 'Counter is correct';
            ok $logger_hash->{handle}, 'Got filehandle back';
            is ref( $logger_hash->{handle} ), 'GLOB', 'It is a glob';

            # Write using the returned handle
            syswrite $logger_hash->{handle}, "Write via Value Return\n";
            close $fh;

            # Verify content
            open my $check, '<', $filename;
            my $content = <$check>;
            close $check;
            is $content, "Write via Value Return\n", 'Handle returned in struct is usable';
        };
        subtest 'PerlIO in Array' => sub {

            # Quick dynamic test for array of handles
            my $C_CODE_ARRAY = <<'END_C2';
    #include "std.h"
    //ext: .c

    // Swap the first two handles in the array
    DLLEXPORT void swap_handles(void* handles[3]) {
        void* temp = handles[0];
        handles[0] = handles[1];
        handles[1] = temp;
    }
END_C2
            my $lib2 = compile_ok($C_CODE_ARRAY);

            # Array of PerlIO*
            affix $lib2, 'swap_handles', [ Array [ Pointer [PerlIO], 3 ] ] => Void;
            my ( $fh1, $f1 ) = tempfile();
            my ( $fh2, $f2 ) = tempfile();
            my ( $fh3, $f3 ) = tempfile();

            # Write distinct markers
            syswrite $fh1, 'File 1';
            syswrite $fh2, 'File 2';
            syswrite $fh3, 'File 3';

            # Pass array. C swaps 0 and 1.
            my $list = [ $fh1, $fh2, $fh3 ];
            swap_handles($list);

            # $list now reflects the C-side modification (Array Writeback)
            my $swapped_1 = $list->[0];
            my $swapped_2 = $list->[1];

            # Verify contents
            seek $swapped_1, 0, 0;
            seek $swapped_2, 0, 0;
            my $c1 = <$swapped_1>;
            my $c2 = <$swapped_2>;
            is $c1, 'File 2', 'Index 0 now contains File 2';
            is $c2, 'File 1', 'Index 1 now contains File 1';
            close $_ for ( $fh1, $fh2, $fh3 );
        }
    }
};
#
done_testing;



( run in 2.381 seconds using v1.01-cache-2.11-cpan-ceb78f64989 )