123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273 |
- # Copyright 2014, 2015, 2016 Free Software Foundation, Inc.
- #
- # This program is free software; you can redistribute it and/or modify
- # it under the terms of the GNU General Public License as published by
- # the Free Software Foundation; either version 3 of the License,
- # or (at your option) any later version.
- #
- # This program is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- # GNU General Public License for more details.
- #
- # You should have received a copy of the GNU General Public License
- # along with this program. If not, see <http://www.gnu.org/licenses/>.
- package Texinfo::XSLoader;
- use DynaLoader;
- use 5.00405;
- use strict;
- use warnings;
- our $TEXINFO_XS;
- our $VERSION = '6.3.90';
- our $disable_XS;
- # For verbose information about what's being done
- sub _debug($) {
- if ($TEXINFO_XS eq 'debug') {
- my $msg = shift;
- warn $msg . "\n";
- }
- }
- # For messages to say that XS module couldn't be loaded
- sub _fatal($) {
- if ($TEXINFO_XS eq 'debug'
- or $TEXINFO_XS eq 'required'
- or $TEXINFO_XS eq 'warn') {
- my $msg = shift;
- warn $msg . "\n";
- }
- }
- # We look for the .la and .so files in @INC because this allows us to override
- # which modules are used using -I flags to "perl".
- sub _find_file($) {
- my $file = shift;
- for my $dir (@INC) {
- _debug "checking $dir/$file";
- if (-f "$dir/$file") {
- _debug "found $dir/$file";
- return ($dir, "$dir/$file");
- }
- }
- return undef;
- }
-
- # Make symbols accessible under
- # namespace $FULL_MODULE_NAME (e.g. Texinfo::Convert::Paragraph),
- # either from XS implementation in $MODULE, or non-XS implementation
- # $FALLBACK_MODULE. $MODULE_NAME is the name of a Libtool file used for
- # loading the XS subroutines.
- # $INTERFACE_VERSION is a module interface number, to be changed when the XS
- # interface changes.
- sub init {
- my ($full_module_name,
- $module,
- $fallback_module,
- $module_name,
- $interface_version,
- $warning_message,
- $fatal_message
- ) = @_;
-
- # Possible values for TEXINFO_XS environment variable:
- #
- # TEXINFO_XS=omit # don't try loading xs at all
- # TEXINFO_XS=default # try xs, libtool and then perl paths,
- # # silent fallback
- # TEXINFO_XS=libtool # try xs, libtool only, silent fallback
- # TEXINFO_XS=standalone # try xs, perl paths only, silent fallback
- # TEXINFO_XS=warn # try xs, libtool and then perl paths, warn
- # # on failure
- # TEXINFO_XS=required # abort if not loadable, no fallback
- # TEXINFO_XS=debug # voluminuous debugging
- #
- # Other values are treated at the moment as 'default'.
-
- $TEXINFO_XS = $ENV{'TEXINFO_XS'};
- if (!defined($TEXINFO_XS)) {
- $TEXINFO_XS = '';
- }
-
- if ($TEXINFO_XS eq 'omit') {
- # Don't try to use the XS module
- goto FALLBACK;
- }
-
- if ($disable_XS) {
- _fatal "use of XS modules was disabled when Texinfo was built";
- goto FALLBACK;
- }
- if ($warning_message) {
- _debug $warning_message;
- }
- if ($fatal_message) {
- _fatal $fatal_message;
- goto FALLBACK;
- }
- if (!$module) {
- goto FALLBACK;
- }
-
- my ($libtool_dir, $libtool_archive);
- if ($TEXINFO_XS ne 'standalone') {
- ($libtool_dir, $libtool_archive) = _find_file("$module_name.la");
- if (!$libtool_archive) {
- if ($TEXINFO_XS eq 'libtool') {
- _fatal "$module_name: couldn't find Libtool archive file";
- goto FALLBACK;
- }
- _debug "$module_name: couldn't find Libtool archive file";
- }
- }
-
- my $dlname = undef;
- my $dlpath = undef;
-
- # Try perl paths
- if (!$libtool_archive) {
- my @modparts = split(/::/,$module);
- my $dlname = $modparts[-1];
- my $modpname = join('/',@modparts);
- # the directories with -L prepended setup directories to
- # be in the search path. Then $dlname is prepended as it is
- # the name really searched for.
- $dlpath = DynaLoader::dl_findfile(map("-L$_/auto/$modpname", @INC), $dlname);
- if (!$dlpath) {
- _fatal "$module_name: couldn't find $module";
- goto FALLBACK;
- }
- goto LOAD;
- }
-
- my $fh;
- open $fh, $libtool_archive;
- if (!$fh) {
- _fatal "$module_name: couldn't open Libtool archive file";
- goto FALLBACK;
- }
-
- # Look for the line in XS*.la giving the name of the loadable object.
- while (my $line = <$fh>) {
- if ($line =~ /^\s*dlname\s*=\s*'([^']+)'\s$/) {
- $dlname = $1;
- last;
- }
- }
- if (!$dlname) {
- _fatal "$module_name: couldn't find name of shared object";
- goto FALLBACK;
- }
-
- # The *.so file is under .libs in the source directory.
- push @DynaLoader::dl_library_path, $libtool_dir;
- push @DynaLoader::dl_library_path, "$libtool_dir/.libs";
-
- $dlpath = DynaLoader::dl_findfile($dlname);
- if (!$dlpath) {
- _fatal "$module_name: couldn't find $dlname";
- goto FALLBACK;
- }
-
- LOAD:
-
- #my $flags = dl_load_flags $module; # This is 0 in DynaLoader
- my $flags = 0;
- my $libref = DynaLoader::dl_load_file($dlpath, $flags);
- if (!$libref) {
- _fatal "$module_name: couldn't load file $dlpath";
- goto FALLBACK;
- }
- _debug "$dlpath loaded";
- my @undefined_symbols = DynaLoader::dl_undef_symbols();
- if ($#undefined_symbols+1 != 0) {
- _fatal "$module_name: still have undefined symbols after dl_load_file";
- }
- my $bootname = "boot_$module";
- $bootname =~ s/:/_/g;
- _debug "looking for $bootname";
- my $symref = DynaLoader::dl_find_symbol($libref, $bootname);
- if (!$symref) {
- _fatal "$module_name: couldn't find $bootname symbol";
- goto FALLBACK;
- }
- _debug "trying to call $bootname...";
- my $boot_fn = DynaLoader::dl_install_xsub("${module}::bootstrap",
- $symref, $dlname);
-
- if (!$boot_fn) {
- _fatal "$module_name: couldn't bootstrap";
- goto FALLBACK;
- }
- _debug " ...succeeded";
-
- push @DynaLoader::dl_shared_objects, $dlpath; # record files loaded
-
- # This is the module bootstrap function, which causes all the other
- # functions (XSUB's) provided by the module to become available to
- # be called from Perl code.
- &$boot_fn($module, $interface_version);
-
- # This makes it easier to refer to packages and symbols by name.
- no strict 'refs';
-
- if (defined &{"${module}::init"} and !&{"${module}::init"} ()) {
- _fatal "$module_name: error initializing";
- goto FALLBACK;
- }
-
- *{"${full_module_name}::"} = \%{"${module}::"};
-
- return $module;
-
- FALLBACK:
- if ($TEXINFO_XS eq 'required') {
- die "unset the TEXINFO_XS environment variable to use the "
- ."pure Perl modules\n";
- } elsif ($TEXINFO_XS eq 'warn' or $TEXINFO_XS eq 'debug') {
- warn "falling back to pure Perl module\n";
- }
- if (!defined $fallback_module) {
- die "no fallback module for $full_module_name";
- }
- # Fall back to using the Perl code.
- # Use eval here to interpret :: properly in module name.
- eval "require $fallback_module";
- *{"${full_module_name}::"} = \%{"${fallback_module}::"};
- return $fallback_module;
- } # end init
- # Override subroutine $TARGET with $SOURCE.
- sub override {
- my ($target, $source) = @_;
- _debug "attempting to override $target with $source...";
- no strict 'refs'; # access modules and symbols by name.
- no warnings 'redefine'; # do not warn about redefining a function.
- if (defined &{"${source}"}) {
- *{"${target}"} = \&{"${source}"};
- _debug " ...succeeded";
- } else {
- _debug " ...failed";
- }
- }
- 1;
- __END__
|