| Index: third_party/dpkg-dev/scripts/Dpkg/Arch.pm | 
| diff --git a/third_party/dpkg-dev/scripts/Dpkg/Arch.pm b/third_party/dpkg-dev/scripts/Dpkg/Arch.pm | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..33a669d83bf9aa1a8e1e82e186c4c359592bd716 | 
| --- /dev/null | 
| +++ b/third_party/dpkg-dev/scripts/Dpkg/Arch.pm | 
| @@ -0,0 +1,445 @@ | 
| +# Copyright © 2006-2013 Guillem Jover <guillem@debian.org> | 
| +# | 
| +# 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 2 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 <https://www.gnu.org/licenses/>. | 
| + | 
| +package Dpkg::Arch; | 
| + | 
| +use strict; | 
| +use warnings; | 
| + | 
| +our $VERSION = '0.01'; | 
| + | 
| +use Exporter qw(import); | 
| +our @EXPORT_OK = qw(get_raw_build_arch get_raw_host_arch | 
| +                    get_build_arch get_host_arch get_gcc_host_gnu_type | 
| +                    get_valid_arches debarch_eq debarch_is debarch_is_wildcard | 
| +                    debarch_to_cpuattrs | 
| +                    debarch_to_gnutriplet gnutriplet_to_debarch | 
| +                    debtriplet_to_gnutriplet gnutriplet_to_debtriplet | 
| +                    debtriplet_to_debarch debarch_to_debtriplet | 
| +                    gnutriplet_to_multiarch debarch_to_multiarch); | 
| + | 
| +use POSIX qw(:errno_h); | 
| +use Dpkg (); | 
| +use Dpkg::Gettext; | 
| +use Dpkg::ErrorHandling; | 
| +use Dpkg::Util qw(:list); | 
| +use Dpkg::BuildEnv; | 
| + | 
| +my (@cpu, @os); | 
| +my (%cputable, %ostable); | 
| +my (%cputable_re, %ostable_re); | 
| +my (%cpubits, %cpuendian); | 
| +my %abibits; | 
| + | 
| +my %debtriplet_to_debarch; | 
| +my %debarch_to_debtriplet; | 
| + | 
| +{ | 
| +    my $build_arch; | 
| +    my $host_arch; | 
| +    my $gcc_host_gnu_type; | 
| + | 
| +    sub get_raw_build_arch() | 
| +    { | 
| +	return $build_arch if defined $build_arch; | 
| + | 
| +	# Note: We *always* require an installed dpkg when inferring the | 
| +	# build architecture. The bootstrapping case is handled by | 
| +	# dpkg-architecture itself, by avoiding computing the DEB_BUILD_ | 
| +	# variables when they are not requested. | 
| + | 
| +	$build_arch = `dpkg --print-architecture`; | 
| +	syserr('dpkg --print-architecture failed') if $? >> 8; | 
| + | 
| +	chomp $build_arch; | 
| +	return $build_arch; | 
| +    } | 
| + | 
| +    sub get_build_arch() | 
| +    { | 
| +	return Dpkg::BuildEnv::get('DEB_BUILD_ARCH') || get_raw_build_arch(); | 
| +    } | 
| + | 
| +    sub get_gcc_host_gnu_type() | 
| +    { | 
| +	return $gcc_host_gnu_type if defined $gcc_host_gnu_type; | 
| + | 
| +	$gcc_host_gnu_type = `\${CC:-gcc} -dumpmachine`; | 
| +	if ($? >> 8) { | 
| +	    $gcc_host_gnu_type = ''; | 
| +	} else { | 
| +	    chomp $gcc_host_gnu_type; | 
| +	} | 
| + | 
| +	return $gcc_host_gnu_type; | 
| +    } | 
| + | 
| +    sub get_raw_host_arch() | 
| +    { | 
| +	return $host_arch if defined $host_arch; | 
| + | 
| +	$gcc_host_gnu_type = get_gcc_host_gnu_type(); | 
| + | 
| +	if ($gcc_host_gnu_type eq '') { | 
| +	    warning(_g("couldn't determine gcc system type, falling back to " . | 
| +	               'default (native compilation)')); | 
| +	} else { | 
| +	    my (@host_archtriplet) = gnutriplet_to_debtriplet($gcc_host_gnu_type); | 
| +	    $host_arch = debtriplet_to_debarch(@host_archtriplet); | 
| + | 
| +	    if (defined $host_arch) { | 
| +		$gcc_host_gnu_type = debtriplet_to_gnutriplet(@host_archtriplet); | 
| +	    } else { | 
| +		warning(_g('unknown gcc system type %s, falling back to ' . | 
| +		           'default (native compilation)'), $gcc_host_gnu_type); | 
| +		$gcc_host_gnu_type = ''; | 
| +	    } | 
| +	} | 
| + | 
| +	if (!defined($host_arch)) { | 
| +	    # Switch to native compilation. | 
| +	    $host_arch = get_raw_build_arch(); | 
| +	} | 
| + | 
| +	return $host_arch; | 
| +    } | 
| + | 
| +    sub get_host_arch() | 
| +    { | 
| +	return Dpkg::BuildEnv::get('DEB_HOST_ARCH') || get_raw_host_arch(); | 
| +    } | 
| +} | 
| + | 
| +sub get_valid_arches() | 
| +{ | 
| +    read_cputable(); | 
| +    read_ostable(); | 
| + | 
| +    my @arches; | 
| + | 
| +    foreach my $os (@os) { | 
| +	foreach my $cpu (@cpu) { | 
| +	    my $arch = debtriplet_to_debarch(split(/-/, $os, 2), $cpu); | 
| +	    push @arches, $arch if defined($arch); | 
| +	} | 
| +    } | 
| + | 
| +    return @arches; | 
| +} | 
| + | 
| +my $cputable_loaded = 0; | 
| +sub read_cputable | 
| +{ | 
| +    return if ($cputable_loaded); | 
| + | 
| +    local $_; | 
| +    local $/ = "\n"; | 
| + | 
| +    open my $cputable_fh, '<', "$Dpkg::DATADIR/cputable" | 
| +	or syserr(_g('cannot open %s'), 'cputable'); | 
| +    while (<$cputable_fh>) { | 
| +	if (m/^(?!\#)(\S+)\s+(\S+)\s+(\S+)\s+(\S+)\s+(\S+)/) { | 
| +	    $cputable{$1} = $2; | 
| +	    $cputable_re{$1} = $3; | 
| +	    $cpubits{$1} = $4; | 
| +	    $cpuendian{$1} = $5; | 
| +	    push @cpu, $1; | 
| +	} | 
| +    } | 
| +    close $cputable_fh; | 
| + | 
| +    $cputable_loaded = 1; | 
| +} | 
| + | 
| +my $ostable_loaded = 0; | 
| +sub read_ostable | 
| +{ | 
| +    return if ($ostable_loaded); | 
| + | 
| +    local $_; | 
| +    local $/ = "\n"; | 
| + | 
| +    open my $ostable_fh, '<', "$Dpkg::DATADIR/ostable" | 
| +	or syserr(_g('cannot open %s'), 'ostable'); | 
| +    while (<$ostable_fh>) { | 
| +	if (m/^(?!\#)(\S+)\s+(\S+)\s+(\S+)/) { | 
| +	    $ostable{$1} = $2; | 
| +	    $ostable_re{$1} = $3; | 
| +	    push @os, $1; | 
| +	} | 
| +    } | 
| +    close $ostable_fh; | 
| + | 
| +    $ostable_loaded = 1; | 
| +} | 
| + | 
| +my $abitable_loaded = 0; | 
| +sub abitable_load() | 
| +{ | 
| +    return if ($abitable_loaded); | 
| + | 
| +    local $_; | 
| +    local $/ = "\n"; | 
| + | 
| +    # Because the abitable is only for override information, do not fail if | 
| +    # it does not exist, as that will only mean the other tables do not have | 
| +    # an entry needing to be overridden. This way we do not require a newer | 
| +    # dpkg by libdpkg-perl. | 
| +    if (open my $abitable_fh, '<', "$Dpkg::DATADIR/abitable") { | 
| +        while (<$abitable_fh>) { | 
| +            if (m/^(?!\#)(\S+)\s+(\S+)/) { | 
| +                $abibits{$1} = $2; | 
| +            } | 
| +        } | 
| +        close $abitable_fh; | 
| +    } elsif ($! != ENOENT) { | 
| +        syserr(_g('cannot open %s'), 'abitable'); | 
| +    } | 
| + | 
| +    $abitable_loaded = 1; | 
| +} | 
| + | 
| +my $triplettable_loaded = 0; | 
| +sub read_triplettable() | 
| +{ | 
| +    return if ($triplettable_loaded); | 
| + | 
| +    read_cputable(); | 
| + | 
| +    local $_; | 
| +    local $/ = "\n"; | 
| + | 
| +    open my $triplettable_fh, '<', "$Dpkg::DATADIR/triplettable" | 
| +	or syserr(_g('cannot open %s'), 'triplettable'); | 
| +    while (<$triplettable_fh>) { | 
| +	if (m/^(?!\#)(\S+)\s+(\S+)/) { | 
| +	    my $debtriplet = $1; | 
| +	    my $debarch = $2; | 
| + | 
| +	    if ($debtriplet =~ /<cpu>/) { | 
| +		foreach my $_cpu (@cpu) { | 
| +		    (my $dt = $debtriplet) =~ s/<cpu>/$_cpu/; | 
| +		    (my $da = $debarch) =~ s/<cpu>/$_cpu/; | 
| + | 
| +		    next if exists $debarch_to_debtriplet{$da} | 
| +		         or exists $debtriplet_to_debarch{$dt}; | 
| + | 
| +		    $debarch_to_debtriplet{$da} = $dt; | 
| +		    $debtriplet_to_debarch{$dt} = $da; | 
| +		} | 
| +	    } else { | 
| +		$debarch_to_debtriplet{$2} = $1; | 
| +		$debtriplet_to_debarch{$1} = $2; | 
| +	    } | 
| +	} | 
| +    } | 
| +    close $triplettable_fh; | 
| + | 
| +    $triplettable_loaded = 1; | 
| +} | 
| + | 
| +sub debtriplet_to_gnutriplet(@) | 
| +{ | 
| +    my ($abi, $os, $cpu) = @_; | 
| + | 
| +    read_cputable(); | 
| +    read_ostable(); | 
| + | 
| +    return unless defined($abi) && defined($os) && defined($cpu) && | 
| +        exists($cputable{$cpu}) && exists($ostable{"$abi-$os"}); | 
| +    return join('-', $cputable{$cpu}, $ostable{"$abi-$os"}); | 
| +} | 
| + | 
| +sub gnutriplet_to_debtriplet($) | 
| +{ | 
| +    my ($gnu) = @_; | 
| +    return unless defined($gnu); | 
| +    my ($gnu_cpu, $gnu_os) = split(/-/, $gnu, 2); | 
| +    return unless defined($gnu_cpu) && defined($gnu_os); | 
| + | 
| +    read_cputable(); | 
| +    read_ostable(); | 
| + | 
| +    my ($os, $cpu); | 
| + | 
| +    foreach my $_cpu (@cpu) { | 
| +	if ($gnu_cpu =~ /^$cputable_re{$_cpu}$/) { | 
| +	    $cpu = $_cpu; | 
| +	    last; | 
| +	} | 
| +    } | 
| + | 
| +    foreach my $_os (@os) { | 
| +	if ($gnu_os =~ /^(.*-)?$ostable_re{$_os}$/) { | 
| +	    $os = $_os; | 
| +	    last; | 
| +	} | 
| +    } | 
| + | 
| +    return if !defined($cpu) || !defined($os); | 
| +    return (split(/-/, $os, 2), $cpu); | 
| +} | 
| + | 
| +sub gnutriplet_to_multiarch($) | 
| +{ | 
| +    my ($gnu) = @_; | 
| +    my ($cpu, $cdr) = split(/-/, $gnu, 2); | 
| + | 
| +    if ($cpu =~ /^i[456]86$/) { | 
| +	return "i386-$cdr"; | 
| +    } else { | 
| +	return $gnu; | 
| +    } | 
| +} | 
| + | 
| +sub debarch_to_multiarch($) | 
| +{ | 
| +    my ($arch) = @_; | 
| + | 
| +    return gnutriplet_to_multiarch(debarch_to_gnutriplet($arch)); | 
| +} | 
| + | 
| +sub debtriplet_to_debarch(@) | 
| +{ | 
| +    my ($abi, $os, $cpu) = @_; | 
| + | 
| +    read_triplettable(); | 
| + | 
| +    if (!defined($abi) || !defined($os) || !defined($cpu)) { | 
| +	return; | 
| +    } elsif (exists $debtriplet_to_debarch{"$abi-$os-$cpu"}) { | 
| +	return $debtriplet_to_debarch{"$abi-$os-$cpu"}; | 
| +    } else { | 
| +	return; | 
| +    } | 
| +} | 
| + | 
| +sub debarch_to_debtriplet($) | 
| +{ | 
| +    local ($_) = @_; | 
| +    my $arch; | 
| + | 
| +    read_triplettable(); | 
| + | 
| +    if (/^linux-([^-]*)/) { | 
| +	# XXX: Might disappear in the future, not sure yet. | 
| +	$arch = $1; | 
| +    } else { | 
| +	$arch = $_; | 
| +    } | 
| + | 
| +    my $triplet = $debarch_to_debtriplet{$arch}; | 
| + | 
| +    if (defined($triplet)) { | 
| +	return split(/-/, $triplet, 3); | 
| +    } else { | 
| +	return; | 
| +    } | 
| +} | 
| + | 
| +sub debarch_to_gnutriplet($) | 
| +{ | 
| +    my ($arch) = @_; | 
| + | 
| +    return debtriplet_to_gnutriplet(debarch_to_debtriplet($arch)); | 
| +} | 
| + | 
| +sub gnutriplet_to_debarch($) | 
| +{ | 
| +    my ($gnu) = @_; | 
| + | 
| +    return debtriplet_to_debarch(gnutriplet_to_debtriplet($gnu)); | 
| +} | 
| + | 
| +sub debwildcard_to_debtriplet($) | 
| +{ | 
| +    my ($arch) = @_; | 
| +    my @tuple = split /-/, $arch, 3; | 
| + | 
| +    if (any { $_ eq 'any' } @tuple) { | 
| +	if (scalar @tuple == 3) { | 
| +	    return @tuple; | 
| +	} elsif (scalar @tuple == 2) { | 
| +	    return ('any', @tuple); | 
| +	} else { | 
| +	    return ('any', 'any', 'any'); | 
| +	} | 
| +    } else { | 
| +	return debarch_to_debtriplet($arch); | 
| +    } | 
| +} | 
| + | 
| +sub debarch_to_cpuattrs($) | 
| +{ | 
| +    my ($arch) = @_; | 
| +    my ($abi, $os, $cpu) = debarch_to_debtriplet($arch); | 
| + | 
| +    if (defined($cpu)) { | 
| +        abitable_load(); | 
| + | 
| +        return ($abibits{$abi} || $cpubits{$cpu}, $cpuendian{$cpu}); | 
| +    } else { | 
| +        return; | 
| +    } | 
| +} | 
| + | 
| +sub debarch_eq($$) | 
| +{ | 
| +    my ($a, $b) = @_; | 
| + | 
| +    return 1 if ($a eq $b); | 
| + | 
| +    my @a = debarch_to_debtriplet($a); | 
| +    my @b = debarch_to_debtriplet($b); | 
| + | 
| +    return 0 if scalar @a != 3 or scalar @b != 3; | 
| + | 
| +    return ($a[0] eq $b[0] && $a[1] eq $b[1] && $a[2] eq $b[2]); | 
| +} | 
| + | 
| +sub debarch_is($$) | 
| +{ | 
| +    my ($real, $alias) = @_; | 
| + | 
| +    return 1 if ($alias eq $real or $alias eq 'any'); | 
| + | 
| +    my @real = debarch_to_debtriplet($real); | 
| +    my @alias = debwildcard_to_debtriplet($alias); | 
| + | 
| +    return 0 if scalar @real != 3 or scalar @alias != 3; | 
| + | 
| +    if (($alias[0] eq $real[0] || $alias[0] eq 'any') && | 
| +        ($alias[1] eq $real[1] || $alias[1] eq 'any') && | 
| +        ($alias[2] eq $real[2] || $alias[2] eq 'any')) { | 
| +	return 1; | 
| +    } | 
| + | 
| +    return 0; | 
| +} | 
| + | 
| +sub debarch_is_wildcard($) | 
| +{ | 
| +    my ($arch) = @_; | 
| + | 
| +    return 0 if $arch eq 'all'; | 
| + | 
| +    my @triplet = debwildcard_to_debtriplet($arch); | 
| + | 
| +    return 0 if scalar @triplet != 3; | 
| +    return 1 if any { $_ eq 'any' } @triplet; | 
| +    return 0; | 
| +} | 
| + | 
| +1; | 
|  |