NAME

PDL::Core::Dev - PDL development module

SYNOPSIS

use PDL::Core::Dev;

DESCRIPTION

This module encapsulates most of the stuff useful for PDL development and is often used from within Makefile.PL's.

BUILD MODES

Old Skool

The original scheme, which still works, was one PDL module per ExtUtils::MakeMaker (EUMM) "module", i.e. directory. There has been work with Module::Build, but as that is now deprecated, it will not be further discussed. That module would generate a .xs and .pm file in its directory, according to its Makefile.PL, such as linking extra objects, or other libraries.

To have several distinct PDL modules in a CPAN distribution, you made several directories, each with a Makefile.PL. Parallel building was not normally possible between modules, though see PDL's top-level Makefile.PL as of about 2015, which gained a coretest target that did build in parallel despite subdirectories and "recursive make". Any change to any C code caused a rebuild of the whole .xs file (for Slices this was upwards of 40,000 lines), and being one compilation unit, it could not be parallelised.

Multi-C files

As of 2.058, a new "multi-C" mode was added. For all internal PDL modules, and external ones that opted in (by adding a 5th, true, element to the array-ref describing the package). This creates one C file per operation, so parallel building is possible. This makes for quicker builds, both for those installing the package, and those developing it.

It can also avoid unnecessary rebuilds if only one operation got changed - it only recompiles that C file. This is possible due to some trickiness by PDL::PP, which detects if each C file that it would output is the same as the one already on disk, first removing any #line directives so that line renumbering will not force a rebuild, and not writing anything unless a real change happened.

It is opt-in because if the module adds C functions with pp_addhdr without scoping them appropriately, they get incorporated in each generated C file, which causes linking problems. Moving those to separate C files solves that.

But parallel building (without the "cleverness" of the coretest work) is only possible within each module.

Deep mode

EUMM pure-Perl distributions in the modern era have a lib directory, whose structure matches the hierarchy of modules. PDL now uses this in its Basic subdirectory, so there is e.g. lib/PDL/Core.pm under that. As of EUMM 7.12 (shipped with Perl 5.26), it's also possible to put .xs files next to their respective .pm files, by giving a true value for XSMULTI.

As of 2.096, another new mode was added, which automatically engages "multi-C" mode. This allows you to place your .pd file under a lib directory, whose location tells the build system its package name, which means the previous schemes' need to communicate that in each Makefile.PL is no more. Now, the configuration is communicated by each .pd file by setting values in a hash, e.g.:

{ no warnings 'once'; # pass info back to Makefile.PL
$PDL::Core::Dev::EXTRAS{$::PDLMOD}{OBJECT} .= join '', map " $::PDLBASE/$_\$(OBJ_EXT)", qw(fftn);
$PDL::Core::Dev::EXTRAS{$::PDLMOD}{DEFINE} .= qq{ -DFFT_FLOAT -DFFT_DOUBLE -DFFT_LDOUBLE};
$PDL::Core::Dev::EXTRAS{$::PDLMOD}{INC} .= qq{ "-I$::PDLBASE"};
}

This works because PDL needs to make an entry in the Makefile for each operation defined, which it does by loading the .pd file, and making its version of pp_def just record the operation name. As a side effect, the setting of the EXTRAS value can be seen by the build process.

To have the only Makefile.PL work in this new scheme, converting it from the previous one(s), you need to add this key to the WriteMakefile call:

VERSION_FROM => 'lib/PDL/GSL/CDF.pd',

Note that the ones supplied by pdlpp_stdargs are added for you. You do need to provide overrides for postamble as before, and also init_PM:

{
my @pd_srcs;
package MY; # so that "SUPER" works right
sub init_PM {
  my ($self) = @_;
  $self->SUPER::init_PM;
  @pd_srcs = ::pdlpp_eumm_update_deep($self);
}
sub postamble { ::pdlpp_postamble(@pd_srcs) }
}

FUNCTIONS

isbigendian

Is the machine big or little endian?

print "Your machins is big endian.\n" if isbigendian();

returns 1 if the machine is big endian, 0 if little endian, or dies if neither. It uses the byteorder element of perl's %Config array.

my $retval = isbigendian();

a perl configure clone

if (trylink 'libGL', '', 'char glBegin(); glBegin();', '-lGL') {
  $libs = '-lGLU -lGL';
  $have_GL = 1;
} else {
  $have_GL = 0;
}
$maybe =
  trylink 'libwhatever', '', $body, $libs, $cflags,
      {MakeMaker=>1, Hide=>0, Clean=>1};

Try to link some C-code making up the body of a function with a given set of library specifiers

return 1 if successful, 0 otherwise

trylink $infomsg, $include, $progbody, $libs [,$cflags,{OPTIONS}];

Takes 4 + 2 optional arguments.

  • an informational message to print (can be empty)

  • any commands to be included at the top of the generated C program (typically something like #include "mylib.h")

  • the body of the program (in function main)

  • library flags to use for linking. Preprocessing by MakeMaker should be performed as needed (see options and example).

  • compilation flags. For example, something like -I/usr/local/lib. Optional argument. Empty if omitted.

  • OPTIONS

    MakeMaker

    Preprocess library strings in the way MakeMaker does things. This is advisable to ensure that your code will actually work after the link specs have been processed by MakeMaker.

    Hide

    Controls if linking output etc is hidden from the user or not. On by default but overridable with environment variable HIDE_TRYLINK if set.

    Clean

    Remove temporary files. Enabled by default. You might want to switch it off during debugging.

generate_core_flags

prints on STDOUT XS text with core flags, for Core.xs.

got_complex_version

PDL::Core::Dev::got_complex_version($func_name, $num_params)

For a given function appearing in C99's complex.h, will return a boolean of whether the system being compiled on has the complex version of that. E.g. for sin, will test whether csinl exists (before 2.069, would only check for csin, causing build failures on non-C99 compliant libc which mandates long-double versions).