package ExtUtils::Liblist::Kid;
use 5.006;
use strict;
our $VERSION = 6.56;
use Config;
use Cwd 'cwd';
use File::Basename;
use File::Spec;
sub ext {
if ($^O eq 'VMS') { return &_vms_ext; }
elsif($^O eq 'MSWin32') { return &_win32_ext; }
else { return &_unix_os2_ext; }
}
sub _unix_os2_ext {
my($self,$potential_libs, $verbose, $give_libs) = @_;
$verbose ||= 0;
if ($^O =~ 'os2' and $Config{perllibs}) {
$potential_libs .= " " if $potential_libs;
$potential_libs .= $Config{perllibs};
}
return ("", "", "", "", ($give_libs ? [] : ())) unless $potential_libs;
warn "Potential libraries are '$potential_libs':\n" if $verbose;
my($so) = $Config{so};
my($libs) = defined $Config{perllibs} ? $Config{perllibs} : $Config{libs};
my $Config_libext = $Config{lib_ext} || ".a";
my(@searchpath); my(@libpath) = split " ", $Config{'libpth'};
my(@ldloadlibs, @bsloadlibs, @extralibs, @ld_run_path, %ld_run_path_seen);
my(@libs, %libs_seen);
my($fullname, @fullname);
my($pwd) = cwd(); my($found) = 0;
foreach my $thislib (split ' ', $potential_libs) {
if ($thislib =~ s/^(-[LR]|-Wl,-R)//){ # save path flag type
my($ptype) = $1;
unless (-d $thislib){
warn "$ptype$thislib ignored, directory does not exist\n"
if $verbose;
next;
}
my($rtype) = $ptype;
if (($ptype eq '-R') or ($ptype eq '-Wl,-R')) {
if ($Config{'lddlflags'} =~ /-Wl,-R/) {
$rtype = '-Wl,-R';
} elsif ($Config{'lddlflags'} =~ /-R/) {
$rtype = '-R';
}
}
unless (File::Spec->file_name_is_absolute($thislib)) {
warn "Warning: $ptype$thislib changed to $ptype$pwd/$thislib\n";
$thislib = $self->catdir($pwd,$thislib);
}
push(@searchpath, $thislib);
push(@extralibs, "$ptype$thislib");
push(@ldloadlibs, "$rtype$thislib");
next;
}
unless ($thislib =~ s/^-l//){
warn "Unrecognized argument in LIBS ignored: '$thislib'\n";
next;
}
my($found_lib)=0;
foreach my $thispth (@searchpath, @libpath) {
if (@fullname =
$self->lsdir($thispth,"^\Qlib$thislib.$so.\E[0-9]+")){
$fullname = "$thispth/" .
(sort { my($ma) = $a;
my($mb) = $b;
$ma =~ tr/A-Za-z/0/s;
$ma =~ s/\b(\d)\b/0$1/g;
$mb =~ tr/A-Za-z/0/s;
$mb =~ s/\b(\d)\b/0$1/g;
while (length($ma) < length($mb)) { $ma .= ".00"; }
while (length($mb) < length($ma)) { $mb .= ".00"; }
$mb cmp $ma;} @fullname)[0];
} elsif (-f ($fullname="$thispth/lib$thislib.$so")
&& (($Config{'dlsrc'} ne "dl_dld.xs") || ($thislib eq "m"))){
} elsif (-f ($fullname="$thispth/lib${thislib}_s$Config_libext")
&& ($Config{'archname'} !~ /RM\d\d\d-svr4/)
&& ($thislib .= "_s") ){ } elsif (-f ($fullname="$thispth/lib$thislib$Config_libext")){
} elsif (-f ($fullname="$thispth/$thislib$Config_libext")){
} elsif (-f ($fullname="$thispth/lib$thislib.dll$Config_libext")){
} elsif (-f ($fullname="$thispth/Slib$thislib$Config_libext")){
} elsif ($^O eq 'dgux'
&& -l ($fullname="$thispth/lib$thislib$Config_libext")
&& readlink($fullname) =~ /^elink:/s) {
} else {
warn "$thislib not found in $thispth\n" if $verbose;
next;
}
warn "'-l$thislib' found at $fullname\n" if $verbose;
push @libs, $fullname unless $libs_seen{$fullname}++;
$found++;
$found_lib++;
my $is_dyna = ($fullname !~ /\Q$Config_libext\E\z/);
my $in_perl = ($libs =~ /\B-l\Q${thislib}\E\b/s);
my($fullnamedir) = dirname($fullname);
push @ld_run_path, $fullnamedir
if $is_dyna && !$in_perl &&
!$ld_run_path_seen{$fullnamedir}++;
unless ($in_perl ||
($Config{'osname'} eq 'next' &&
($thislib eq 'm' || $thislib eq 'ndbm')) ){
push(@extralibs, "-l$thislib");
}
if ( ($Config{'dlsrc'} =~ /dl_next/ && $Config{'osvers'} lt '4_0')
|| ($Config{'dlsrc'} =~ /dl_dld/) )
{
push(@bsloadlibs, "$fullname");
} else {
if ($is_dyna){
push(@ldloadlibs, "-l$thislib")
unless ($in_perl and $^O eq 'sunos');
} else {
push(@ldloadlibs, "-l$thislib");
}
}
last; }
warn "Note (probably harmless): "
."No library found for -l$thislib\n"
unless $found_lib>0;
}
unless( $found ) {
return ('','','','', ($give_libs ? \@libs : ()));
}
else {
return ("@extralibs", "@bsloadlibs", "@ldloadlibs",
join(":",@ld_run_path), ($give_libs ? \@libs : ()));
}
}
sub _win32_ext {
require Text::ParseWords;
my($self, $potential_libs, $verbose, $give_libs) = @_;
$verbose ||= 0;
return ("", "", "", "", ($give_libs ? [] : ())) unless $potential_libs;
my $cc = $Config{cc};
my $VC = $cc =~ /^cl/i;
my $BC = $cc =~ /^bcc/i;
my $GC = $cc =~ /^gcc/i;
my $so = $Config{'so'};
my $libs = $Config{'perllibs'};
my $libpth = $Config{'libpth'};
my $libext = $Config{'lib_ext'} || ".lib";
my(@libs, %libs_seen);
if ($libs and $potential_libs !~ /:nodefault/i) {
$potential_libs .= " " if $potential_libs;
$potential_libs .= $libs;
}
warn "Potential libraries are '$potential_libs':\n" if $verbose;
$libpth =~ s,\\,/,g;
$potential_libs =~ s,\\,/,g;
my @searchpath; my @libpath = Text::ParseWords::quotewords('\s+', 0, $libpth);
my @extralibs;
my $pwd = cwd(); my $lib = '';
my $found = 0;
my $search = 1;
my($fullname);
push @libpath, "$Config{installarchlib}/CORE";
if ($VC and exists $ENV{LIB} and $ENV{LIB}) {
push @libpath, split /;/, $ENV{LIB};
}
foreach (Text::ParseWords::quotewords('\s+', 0, $potential_libs)){
my $thislib = $_;
if (/^:\w+$/) {
$search = 0 if lc eq ':nosearch';
$search = 1 if lc eq ':search';
warn "Ignoring unknown flag '$thislib'\n"
if $verbose and !/^:(no)?(search|default)$/i;
next;
}
unless ($search) {
s/^-l(.+)$/$1.lib/ unless $GC;
s/^-L/-libpath:/ if $VC;
push(@extralibs, $_);
$found++;
next;
}
if (s/^-L// and not -d) {
warn "$thislib ignored, directory does not exist\n"
if $verbose;
next;
}
elsif (-d) {
unless (File::Spec->file_name_is_absolute($_)) {
warn "Warning: '$thislib' changed to '-L$pwd/$_'\n";
$_ = $self->catdir($pwd,$_);
}
push(@searchpath, $_);
next;
}
if (s/^-l// and $GC and !/^lib/i) {
$_ = "lib$_";
}
$_ .= $libext if !/\Q$libext\E$/i;
my $secondpass = 0;
LOOKAGAIN:
if (-f) {
warn "'$thislib' found as '$_'\n" if $verbose;
$found++;
push(@extralibs, $_);
next;
}
my $found_lib = 0;
foreach my $thispth (@searchpath, @libpath){
unless (-f ($fullname="$thispth\\$_")) {
warn "'$thislib' not found as '$fullname'\n" if $verbose;
next;
}
warn "'$thislib' found as '$fullname'\n" if $verbose;
$found++;
$found_lib++;
push(@extralibs, $fullname);
push @libs, $fullname unless $libs_seen{$fullname}++;
last;
}
if (!$found_lib and $thislib =~ /^-l/ and !$secondpass++) {
if ($GC) {
goto LOOKAGAIN if s/^lib//i;
}
elsif (!/^lib/i) {
$_ = "lib$_";
goto LOOKAGAIN;
}
}
warn "Note (probably harmless): "
."No library found for $thislib\n"
unless $found_lib>0;
}
return ('','','','', ($give_libs ? \@libs : ())) unless $found;
@extralibs = map { (/\s/ && !/^".*"$/) ? qq["$_"] : $_ } @extralibs;
@libs = map { (/\s/ && !/^".*"$/) ? qq["$_"] : $_ } @libs;
$lib = join(' ',@extralibs);
$lib =~ s,/,\\,g;
warn "Result: $lib\n" if $verbose;
wantarray ? ($lib, '', $lib, '', ($give_libs ? \@libs : ())) : $lib;
}
sub _vms_ext {
my($self, $potential_libs, $verbose, $give_libs) = @_;
$verbose ||= 0;
my(@crtls,$crtlstr);
@crtls = ( ($Config{'ldflags'} =~ m-/Debug-i ? $Config{'dbgprefix'} : '')
. 'PerlShr/Share' );
push(@crtls, grep { not /\(/ } split /\s+/, $Config{'perllibs'});
push(@crtls, grep { not /\(/ } split /\s+/, $Config{'libc'});
if ($self->{PERL_SRC}) {
my($locspec,$type);
foreach my $lib (@crtls) {
if (($locspec,$type) = $lib =~ m{^([\w\$-]+)(/\w+)?} and $locspec =~ /perl/i) {
if (lc $type eq '/share') { $locspec .= $Config{'exe_ext'}; }
elsif (lc $type eq '/library') { $locspec .= $Config{'lib_ext'}; }
else { $locspec .= $Config{'obj_ext'}; }
$locspec = $self->catfile($self->{PERL_SRC},$locspec);
$lib = "$locspec$type" if -e $locspec;
}
}
}
$crtlstr = @crtls ? join(' ',@crtls) : '';
unless ($potential_libs) {
warn "Result:\n\tEXTRALIBS: \n\tLDLOADLIBS: $crtlstr\n" if $verbose;
return ('', '', $crtlstr, '', ($give_libs ? [] : ()));
}
my(%found,@fndlibs,$ldlib);
my $cwd = cwd();
my($so,$lib_ext,$obj_ext) = @Config{'so','lib_ext','obj_ext'};
my(@flibs, %libs_seen);
my %libmap = ( 'm' => '', 'f77' => '', 'F77' => '', 'V77' => '', 'c' => '',
'malloc' => '', 'crypt' => '', 'resolv' => '', 'c_s' => '',
'socket' => '', 'X11' => 'DECW$XLIBSHR',
'Xt' => 'DECW$XTSHR', 'Xm' => 'DECW$XMLIBSHR',
'Xmu' => 'DECW$XMULIBSHR');
if ($Config{'vms_cc_type'} ne 'decc') { $libmap{'curses'} = 'VAXCCURSE'; }
warn "Potential libraries are '$potential_libs'\n" if $verbose;
my(@dirs, @libs);
foreach my $lib (split ' ',$potential_libs) {
push(@dirs,$1), next if $lib =~ /^-L(.*)/;
push(@dirs,$lib), next if $lib =~ /[:>\]]$/;
push(@dirs,$lib), next if -d $lib;
push(@libs,$1), next if $lib =~ /^-l(.*)/;
push(@libs,$lib);
}
push(@dirs,split(' ',$Config{'libpth'}));
foreach my $dir (@dirs) {
unless (-d $dir) {
warn "Skipping nonexistent Directory $dir\n" if $verbose > 1;
$dir = '';
next;
}
warn "Resolving directory $dir\n" if $verbose;
if (File::Spec->file_name_is_absolute($dir)) {
$dir = $self->fixpath($dir,1);
}
else {
$dir = $self->catdir($cwd,$dir);
}
}
@dirs = grep { length($_) } @dirs;
unshift(@dirs,'');
LIB: foreach my $lib (@libs) {
if (exists $libmap{$lib}) {
next unless length $libmap{$lib};
$lib = $libmap{$lib};
}
my(@variants,$cand);
my($ctype) = '';
if ($lib !~ /\.[^:>\]]*$/) {
push(@variants,"${lib}shr","${lib}rtl","${lib}lib");
push(@variants,"lib$lib") if $lib !~ /[:>\]]/;
}
push(@variants,$lib);
warn "Looking for $lib\n" if $verbose;
foreach my $variant (@variants) {
my($fullname, $name);
foreach my $dir (@dirs) {
my($type);
$name = "$dir$variant";
warn "\tChecking $name\n" if $verbose > 2;
$fullname = VMS::Filespec::rmsexpand($name);
if (defined $fullname and -f $fullname) {
if ($fullname =~ /(?:$so|exe)$/i) { $type = 'SHR'; }
elsif ($fullname =~ /(?:$lib_ext|olb)$/i) { $type = 'OLB'; }
elsif ($fullname =~ /(?:$obj_ext|obj)$/i) {
warn "Note (probably harmless): "
."Plain object file $fullname found in library list\n";
$type = 'OBJ';
}
else {
warn "Note (probably harmless): "
."Unknown library type for $fullname; assuming shared\n";
$type = 'SHR';
}
}
elsif (-f ($fullname = VMS::Filespec::rmsexpand($name,$so)) or
-f ($fullname = VMS::Filespec::rmsexpand($name,'.exe'))) {
$type = 'SHR';
$name = $fullname unless $fullname =~ /exe;?\d*$/i;
}
elsif (not length($ctype) and ( -f ($fullname = VMS::Filespec::rmsexpand($name,$lib_ext)) or
-f ($fullname = VMS::Filespec::rmsexpand($name,'.olb')))) {
$type = 'OLB';
$name = $fullname unless $fullname =~ /olb;?\d*$/i;
}
elsif (not length($ctype) and ( -f ($fullname = VMS::Filespec::rmsexpand($name,$obj_ext)) or
-f ($fullname = VMS::Filespec::rmsexpand($name,'.obj')))) {
warn "Note (probably harmless): "
."Plain object file $fullname found in library list\n";
$type = 'OBJ';
$name = $fullname unless $fullname =~ /obj;?\d*$/i;
}
if (defined $type) {
$ctype = $type; $cand = $name;
last if $ctype eq 'SHR';
}
}
if ($ctype) {
if ($cand eq 'VAXCCURSE') { unshift @{$found{$ctype}}, $cand; }
else { push @{$found{$ctype}}, $cand; }
warn "\tFound as $cand (really $fullname), type $ctype\n"
if $verbose > 1;
push @flibs, $name unless $libs_seen{$fullname}++;
next LIB;
}
}
warn "Note (probably harmless): "
."No library found for $lib\n";
}
push @fndlibs, @{$found{OBJ}} if exists $found{OBJ};
push @fndlibs, map { "$_/Library" } @{$found{OLB}} if exists $found{OLB};
push @fndlibs, map { "$_/Share" } @{$found{SHR}} if exists $found{SHR};
my $lib = join(' ',@fndlibs);
$ldlib = $crtlstr ? "$lib $crtlstr" : $lib;
warn "Result:\n\tEXTRALIBS: $lib\n\tLDLOADLIBS: $ldlib\n" if $verbose;
wantarray ? ($lib, '', $ldlib, '', ($give_libs ? \@flibs : ())) : $lib;
}
1;