Why not adopt me?
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"',
}],
);
LOADING XS MODULE SYNOPSIS
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');
TYPEMAP CAST SYNOPSIS
bool
filter (AV* users, const char* what)
CODE:
for (int i = 0; i <= av_len(users); ++i) {
User* user = typemap_incast<User*>(av_fetch(users, i, 0));
if (...) XSRETURN_TRUE;
}
XSRETURN_FALSE;
OUTPUT:
RETVAL
AV*
MyStorage::get_sites ()
CODE:
RETVAL = newAV();
for (int i = 0; i < THIS->urls.length(); ++i) {
SV* uri_perl_obj = typemap_outcast<URI*, const char* CLASS>(THIS->urls[i], "My::URI");
av_push(RETVAL, uri_perl_obj);
}
OUTPUT:
RETVAL
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's 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 and PREREQ_PM sections.
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.
Receives HASHREF, format is the same as for PAYLOAD, the only difference is that it scans folders for header files only.
- 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;
TYPEMAP CAST
Sometimes the type of data you receive or return depends on something and therefore you cannot use certain input or output typemap. To help you dealing with this, there are typemap input and output cast operators (for XS code).
template <class T> T typemap_incast (SV* input)
Does what INPUT typemap "T" would do. Returns T.
Can ONLY be used inside XS functions.
template <class T> SV* typemap_outcast (T output)
Does what OUTPUT typemap "T" would do. Returns SV*.
Can ONLY be used inside XS functions.
template <class T, arg def1, arg def2, ...> SV* typemap_outcast (T output, arg1, arg2, ...)
This is an extended version of typemap_outcast, which is useful if typemap "T" requires additional variables to be predefined. For example, typemaps which create objects, often require "const char* CLASS = ..." variable to be defined. In this case you need to define these variables right after typemap type and pass all of them as a parameters to typemap cast function:
... = typemap_outcast<MyClass*, const char* CLASS, bool do_checks, SV* extra>(new MyClass(), "My::Class", true, myextra);
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);
Note that writing
int myfunc (int a)
will result in default ParseXS behaviour (calling C function myfunc(a) and returning its result). That's because it has no body.
However this function has a body (empty) and therefore prevents default behaviour
int myfunc (int a) {}
AUTHOR
Pronin Oleg <syber@crazypanda.ru>, Crazy Panda, CP Decision LTD
LICENSE
You may distribute this code under the same terms as Perl itself.