NAME

Panda::Install - ExtUtils::MakeMaker based module installer for XS modules.

DESCRIPTION

Panda::Install makes it much easier to write MakeMaker's makefiles especially for XS modules. It provides dependecies support between XS modules, so that one could easily use header files, code, compilation options, ...etc of another. Panda::Install also lets you put source files in subdirectories any level deep (MakeMaker doesn't handle that) and easily compile-in external C libraries.

The params for Panda::Install are compatible with MakeMaker with some additions.

Also it supports typemap inheritance and C-like XS synopsis.

SYNOPSIS

# Makefile.PL
use Panda::Install;

write_makefile(
    NAME    => 'My::XS',
    INC     => '-Isrc -I/usr/local/libevent/include',
    LIBS    => '-levent',
    SRC     => 'src', # all source files (code,header,xs) under 'src' are included in build
    C       => 'src2/foo.cc src2/bar.cc src3/baz/*.c', # plus src2/foo.cc, src2/bar.cc, and first-level c files in src3/baz/
    CPLUS   => 1,
    PAYLOAD => {
        # implements File::ShareDir functionality
        'data.txt'   => '/data.txt',
        'list.txt'   => '/',
        'abc.dat'    => '/mydir/bca.dat',
        'payloaddir' => '/',
    },
    BIN_DEPS  => ['XS::Module1', 'XS::Module2'],
    BIN_SHARE => {
        # modules that depend on My::XS will compile with this INC, LIBS, etc set.
        TYPEMAPS    => {'typemap1.map' => '/typemap.map'},
        INC         => '-I/usr/local/libevent/include', 
        INCLUDE     => {'src' => '/'},
        LIBS        => '-levent',
        DEFINE      => '-DHELLO_FROM_MYXS',
        CCFLAGS     => 'something',
    },
    postamble => 'mytarget: blah-blah; echo "hello"',
    CLIB => [{
        DIR    => 'libuv',
        FILE   => 'libuv.a',
        TARGET => 'libuv.a',
        FLAGS  => 'CFLAGS="-fPIC -O2"',
    }],
);
package MyXSModule;
use Panda::XSLoader;

our $VERSION = '1.0.0';
Panda::XSLoader::load(); # same as Panda::XSLoader::load('MyXSModule', $VERSION, 0x01);

see Panda::XSLoader

TYPEMAP INHERITANCE SYNOPSIS

T_TYPE1
    mycode1;
    
T_TYPE2 : T_TYPE1
    mycode2;

C-LIKE XS SYNOPSIS

char* my_xs_sub (SV* sv) { // CODE
    if (badsv(sv)) XSRETURN_UNDEF;
    RETVAL = process(sv);
}

void other_xs_sub (SV* sv) : ALIAS(other_name=1, yet_another=2) { // PPCODE
    xPUSHi(1);
    xPUSHi(2);
}

GETTING PAYLOAD SYNOPSIS

my $payload_dir = Panda::Install::Payload::payload_dir('My::Module');

see Panda::Install::Payload

FUNCTIONS

write_makefile(%params)

Same as WriteMakefile(makemaker_args(%params))

makemaker_args(%params)

Processes %params, does all the neccessary job and returns final parameters for passing to MakeMaker's WriteMakefile.

PARAMETERS

Only differences from MakeMaker params are listed.

ALL_FROM [default: NAME]

Sets ABSTRACT_FROM and VERSION_FROM to value of ALL_FROM.

If not defined, defaults to NAME. That means that if you have version and abstract in your module's main package, then you don't need to define anything.

XS [*.xs]

Sets source files for xsubpp. If you define this param, defaults are aborted.

XS => 'myxs/*.xs'
XS => 'file1.xs folder/file2.xs folder2/*.xs'
XS => ['file1.xs', 'folder/file2.xs folder2/*.xs']
C [*.c, *.cc, *.cxx, *.cpp, <xsubpp's output files>]

Sets source files to compile. If you define this param, defaults are aborted, however C files created by xsubpp are still included.

Usage: see "XS".

H [*.h *.hh *.hxx *.hpp]

Sets header files for makefile's dependencies (forces module to recompile if any of these changes). Useful during development. If you define this param, defaults are aborted.

Usage: see "XS".

SRC

Scans specified folder(s), finds all XS, C and H files and includes them in build. No matter whether you define XS/C/H parameters or not, SRCs are always added to them.

SRC => 'src'
SRC => 'src src2 src3',
SRC => ['src src2', 'src3'],
CPLUS

If true, will use c++ to build current extension.

postamble

Passed unchanged to Makefile. Can be HASHREF for your convenience, in which case keys are ignored, values are concatenated.

postamble => 'sayhello: ; echo "hello"'
postamble => {
    memd_dep   => 'linkext:: libmemd/libmemd.a; cd libmemd && $(MAKE) static',
    memd_clean => 'clean:: ; cd libmemd && $(MAKE) clean',
}
MIN_PERL_VERSION [5.10.0]

Is set to 5.10.0 if you don't provide it.

PAYLOAD

Implements File::ShareDir functionality. Specified files are installed together with module and can later be accessed at runtime by the module itself or by other modules (via Panda::Install::payload_dir()).

Value is a HASHREF where key is a file or directory path relative to module's dist dir and value is relative to payload's installation dir. If key is a directory then all content of that directory is installed to the destination path. If value is not specified (undef, '') then dest path is the same as source path.

Examples (given that $payload is a directory where payload is installed and $dist is a module's dist dir):

'file.txt' => ''       # $dist/file.txt => $payload/file.txt
'file.txt' => 'a.txt'  # $dist/file.txt => $payload/a.txt
'mydir'    => '',      # $dist/mydir    => $payload/mydir
'mydir'    => 'a/b/c', # $dist/mydir/*  => $payload/a/b/c/*
'mydir'    => '/',     # $dist/mydir/*  => $payload/*
BIN_DEPS

List of modules current module binary depends on. That means all that those modules specified in BIN_SHARE section will be applied while building current module. It also adds those modules to CONFIGURE_REQUIRES section.

Also if your module has BIN_SHARE section then all modules in BIN_DEPS goes to BIN_SHARE/PASSTHROUGH unless module name is prefixed with '-' (minus).

Examples:

BIN_DEPS => 'Module1'
BIN_DEPS => ['Module1', '-Module2']
BIN_SHARE

In this section you put values that you want to be applied to any module which specified your module as a dependency.

BIN_SHARE/TYPEMAPS

Installs specified typemaps and also adds it to the list of typemaps when building descendant modules.

Receives HASHREF, format is the same as for PAYLOAD, the only difference is that it scans folders for *.map files only.

BIN_SHARE/INC

Adds include file dirs to INC when building descendant modules.

BIN_SHARE/INCLUDE

Installs specified include files/dirs into module's installation include directory and adds that directory to INC when building descendant modules.

BIN_SHARE/LIBS

Added to LIBS when building descendant modules.

BIN_SHARE/DEFINE

Added to DEFINE when building descendant modules.

BIN_SHARE/CCFLAGS

Added to CCFLAGS when building descendant modules.

BIN_SHARE/XSOPT

Added to XSOPT when building descendant modules.

BIN_SHARE/PASSTHROUGH

Merge 'BIN_SHARE' of this module with 'BIN_SHARE' of specified modules. Everything gets concatenated (strings, arrays, etc) while merging. You don't need to manually manage this setting as it's managed automatically (see BIN_DEPS section).

CLIB

List of external C libraries that need to be built and compiled into the extension.

CLIB/DIR

Directory where external library is. Makefile must present in that directory!

CLIB/FILE

Static library file which is built by the library (relative to CLIB/DIR).

CLIB/TARGET

Name of the target for Makefile to built static library.

CLIB/FLAGS

Flags to build external library with.

TYPEMAP FEATURES

TYPEMAP INHERITANCE

Output typemaps

T_TYPE1
    mycode1;
    
T_TYPE2 : T_TYPE1
    mycode2;
    

T_TYPE2 will have mycode1 inserted after mycode2 as if it was written

T_TYPE2
    mycode2;
    mycode1;
    

Input typemaps

T_TYPE1
    mycode1;
    
T_TYPE2 : T_TYPE1
    mycode2;
    

T_TYPE2 will have mycode1 inserted before mycode2 as if it was written

T_TYPE2
    mycode1;
    mycode2;

Passing params

You can pass params when inheriting typemaps. These params can be accessed in parent typemap via %p hash.

T_TYPE1
    int $p{varname} = 150;
    mycode1;
    $p{expr};

T_TYPE2 : T_TYPE1(varname=myvar, expr="myvar = a + b")
    mycode2;

will result in (for input typemap)

T_TYPE2
    int myvar = 150;
    mycode1;
    myvar = a + b;
    mycode2;

TYPEMAP INIT CODE

In OUTPUT typemaps you can use 'INIT: expr;' expressions. These expressions later will be moved to the top of the XS function like INIT: section of XS function itself. It is useful for typemaps which want to predefine some variable, so that user has a chance to change it. Such typemaps then use this variable in its code. For example:

TYPEMAP

int MY_TYPE

OUTPUT

MY_TYPE
    INIT: int lolo = 0;
    sv_setiv($arg, $var + lolo);
    

#XS

int
myfunc () 
CODE:
    lolo = 10;
    RETVAL = 20; // returns 30
OUTPUT:
    RETVAL;

C-LIKE XS

If you're using Panda::Install then all of your XS files support C-like XS. It means that code

char* my_xs_sub (SV* sv) { // CODE
    if (badsv(sv)) XSRETURN_UNDEF;
    RETVAL = process(sv);
}

void other_xs_sub (SV* sv) : ALIAS(other_name=1, yet_another=2) { // PPCODE
    xPUSHi(1);
    xPUSHi(2);
}
    

is replaced with code

char*
my_xs_sub (SV* sv)
CODE:
    if (badsv(sv)) XSRETURN_UNDEF;
    RETVAL = process(sv);
OUTPUT:
    RETVAL

void
other_xs_sub (SV* sv)
ALIAS:
    other_name=1
    yet_another=2
PPCODE:
    xPUSHi(1);
    xPUSHi(2);

AUTHOR

Pronin Oleg <syber@crazypanda.ru>, Crazy Panda, CP Decision LTD

LICENSE

You may distribute this code under the same terms as Perl itself.

1 POD Error

The following errors were encountered while parsing the POD:

Around line 575:

Unknown directive: =head