%PDF- %PDF-
Direktori : /usr/share/perl5/Dpkg/Vendor/ |
Current File : //usr/share/perl5/Dpkg/Vendor/Ubuntu.pm |
# Copyright © 2008 Ian Jackson <ijackson@chiark.greenend.org.uk> # Copyright © 2008 Canonical, Ltd. # written by Colin Watson <cjwatson@ubuntu.com> # Copyright © 2008 James Westby <jw+debian@jameswestby.net> # Copyright © 2009 Raphaël Hertzog <hertzog@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/>. =encoding utf8 =head1 NAME Dpkg::Vendor::Ubuntu - Ubuntu vendor class =head1 DESCRIPTION This vendor class customizes the behavior of dpkg scripts for Ubuntu specific behavior and policies. B<Note>: This is a private module, its API can change at any time. =cut package Dpkg::Vendor::Ubuntu 0.01; use strict; use warnings; use List::Util qw(any); use Dpkg::ErrorHandling; use Dpkg::Gettext; use Dpkg::Control::Types; use parent qw(Dpkg::Vendor::Debian); sub run_hook { my ($self, $hook, @params) = @_; if ($hook eq 'before-source-build') { my $src = shift @params; my $fields = $src->{fields}; # check that Maintainer/XSBC-Original-Maintainer comply to # https://wiki.ubuntu.com/DebianMaintainerField if (defined($fields->{'Version'}) and defined($fields->{'Maintainer'}) and $fields->{'Version'} =~ /ubuntu/) { if ($fields->{'Maintainer'} !~ /(?:ubuntu|canonical)/i) { if (length $ENV{DEBEMAIL} and $ENV{DEBEMAIL} =~ /\@(?:ubuntu|canonical)\.com/) { error(g_('Version number suggests Ubuntu changes, but Maintainer: does not have Ubuntu address')); } else { warning(g_('Version number suggests Ubuntu changes, but Maintainer: does not have Ubuntu address')); } } unless ($fields->{'Original-Maintainer'}) { warning(g_('Version number suggests Ubuntu changes, but there is no XSBC-Original-Maintainer field')); } } } elsif ($hook eq 'package-keyrings') { return ($self->SUPER::run_hook($hook), '/usr/share/keyrings/ubuntu-archive-keyring.gpg'); } elsif ($hook eq 'archive-keyrings') { return ($self->SUPER::run_hook($hook), '/usr/share/keyrings/ubuntu-archive-keyring.gpg'); } elsif ($hook eq 'archive-keyrings-historic') { return ($self->SUPER::run_hook($hook), '/usr/share/keyrings/ubuntu-archive-removed-keys.gpg'); } elsif ($hook eq 'register-custom-fields') { my @field_ops = $self->SUPER::run_hook($hook); push @field_ops, [ 'register', 'Launchpad-Bugs-Fixed', CTRL_FILE_CHANGES | CTRL_CHANGELOG, ], [ 'insert_after', CTRL_FILE_CHANGES, 'Closes', 'Launchpad-Bugs-Fixed', ], [ 'insert_after', CTRL_CHANGELOG, 'Closes', 'Launchpad-Bugs-Fixed', ]; return @field_ops; } elsif ($hook eq 'post-process-changelog-entry') { my $fields = shift @params; # Add Launchpad-Bugs-Fixed field my $bugs = find_launchpad_closes($fields->{'Changes'} // ''); if (scalar(@$bugs)) { $fields->{'Launchpad-Bugs-Fixed'} = join(' ', @$bugs); } } elsif ($hook eq 'update-buildopts') { my $build_opts = shift @params; require Dpkg::Arch; my $arch = Dpkg::Arch::get_host_arch(); if (Dpkg::Arch::debarch_eq($arch, 'riscv64')) { $build_opts->set('nocheck', 1, 'riscv64'); } } elsif ($hook eq 'update-buildprofiles') { my $build_profiles_ref = shift @params; unless(grep $_ =~ /^!?noudeb$/, @$build_profiles_ref) { unshift(@$build_profiles_ref, 'noudeb'); } else { # Strip otherwise invalid profile name @$build_profiles_ref = grep { $_ ne "!noudeb" } @$build_profiles_ref; } } else { return $self->SUPER::run_hook($hook, @params); } } sub _lto_disabled { my $fn = "/usr/share/lto-disabled-list/lto-disabled-list"; open(LIST, "<", $fn) or return; # get source name -r "debian/control" or return; require Dpkg::Control::Info; my $ctrl = Dpkg::Control::Info->new(); my $src_fields = $ctrl->get_source(); return unless defined $src_fields; my $src = ""; foreach (keys %{$src_fields}) { my $v = $src_fields->{$_}; if (m/^Source$/i) { $src = $v; last; } } return unless $src ne ""; my $arch = Dpkg::Arch::get_host_arch(); # read disabled-list while (<LIST>) { if (m/^$src\s/) { if (m/^$src\s.*(any|$arch)\s/) { close(LIST); return 1; } else { close(LIST); return; } } } close(LIST); return; } # Override Debian default features. sub init_build_features { my ($self, $use_feature, $builtin_feature) = @_; $self->SUPER::init_build_features($use_feature, $builtin_feature); require Dpkg::Arch; my $arch = Dpkg::Arch::get_host_arch(); if (any { $_ eq $arch } qw(amd64 arm64 ppc64el s390x)) { $use_feature->{optimize}{lto} = 1; if (_lto_disabled()) { $use_feature->{optimize}{lto} = 0; } } if (any { $_ eq $arch } qw(amd64 arm64 riscv64 s390x)) { $use_feature->{qa}{framepointer} = 1; } } sub set_build_features { my ($self, $flags) = @_; $self->SUPER::set_build_features($flags); require Dpkg::Arch; my $arch = Dpkg::Arch::get_host_arch(); if ($arch eq 'ppc64el' && $flags->get_option_value('optimize-level') != 0) { $flags->set_option_value('optimize-level', 3); } $flags->set_option_value('fortify-level', 3); # Debian enables -fstack-clash-protection but it causes troubles on armhf # (which are not fully analyzed as of writing this); disable it there in # order to have time to investigate if ($arch eq 'armhf') { # Stack clash protector only available on amd64 and arm. $flags->set_feature('hardening', 'stackclash', 0); } } sub add_build_flags { my ($self, $flags) = @_; my @compile_flags = qw( CFLAGS CXXFLAGS OBJCFLAGS OBJCXXFLAGS FFLAGS FCFLAGS ); $self->SUPER::add_build_flags($flags); # Per https://wiki.ubuntu.com/DistCompilerFlags $flags->prepend('LDFLAGS', '-Wl,-Bsymbolic-functions'); # In Ubuntu these flags are set by the compiler, so when disabling the # features we need to pass appropriate flags to disable them. if (!$flags->use_feature('hardening', 'stackprotectorstrong') && !$flags->use_feature('hardening', 'stackprotector')) { my $flag = '-fno-stack-protector'; $flags->append($_, $flag) foreach @compile_flags; } if (!$flags->use_feature('hardening', 'stackclash')) { my $flag = '-fno-stack-clash-protection'; $flags->append($_, $flag) foreach @compile_flags; } if (!$flags->use_feature('hardening', 'fortify')) { $flags->append('CPPFLAGS', '-D_FORTIFY_SOURCE=0'); } if (!$flags->use_feature('hardening', 'format')) { my $flag = '-Wno-format -Wno-error=format-security'; $flags->append('CFLAGS', $flag); $flags->append('CXXFLAGS', $flag); $flags->append('OBJCFLAGS', $flag); $flags->append('OBJCXXFLAGS', $flag); } if (!$flags->use_feature('hardening', 'branch')) { my $cpu = $flags->get_option_value('hardening-branch-cpu'); my $flag; if ($cpu eq 'arm64') { $flag = '-mbranch-protection=none'; } elsif ($cpu eq 'amd64') { $flag = '-fcf-protection=none'; } if (defined $flag) { $flags->append($_, $flag) foreach @compile_flags; } } # We always enable fdebug-prefix-map and (ideally) set the new # path as full (unless ${DEB_BUILD_DEBUGPATH} points to a relative # path), because (a) the DWARF spec forbids relative paths (which # gets mapped to DW_AT_comp_dir), and (b) it makes our debuginfod # service happy when indexing source code. if ($flags->use_feature('reproducible', 'fixfilepath')) { my $build_path = $flags->get_option_value('build-path'); my $stripflag = '-fdebug-prefix-map=' . $build_path . '=.'; if (defined $ENV{DEB_BUILD_DEBUGPATH}) { my $debugprefixmap = '-fdebug-prefix-map=' . $build_path . '=' . $ENV{DEB_BUILD_DEBUGPATH}; # Strip any existing -fdebug-prefix-map flag. $flags->strip($_, $stripflag) foreach @compile_flags; $flags->append($_, $debugprefixmap) foreach @compile_flags; } elsif (-r 'debian/changelog') { require Dpkg::Changelog::Debian; my $pkgchangelog = Dpkg::Changelog::Debian->new(range => { "count" => 1 }); $pkgchangelog->load('debian/changelog'); my $chgentry = @{$pkgchangelog}[0]; my $pkgver = $chgentry->get_version(); my $pkgsrc = $chgentry->get_source(); if ($pkgver ne "" && $pkgsrc ne "") { my $debugprefixmap = '-fdebug-prefix-map=' . $build_path . '=/usr/src/' . $pkgsrc . '-' . $pkgver; # Strip any existing -fdebug-prefix-map flag. $flags->strip($_, $stripflag) foreach @compile_flags; $flags->append($_, $debugprefixmap) foreach @compile_flags; } } } return; } =head1 PUBLIC FUNCTIONS =over =item $bugs = Dpkg::Vendor::Ubuntu::find_launchpad_closes($changes) Takes one string as argument and finds "LP: #123456, #654321" statements, which are references to bugs on Launchpad. Returns all closed bug numbers in an array reference. =cut sub find_launchpad_closes { my $changes = shift; my %closes; while ($changes && ($changes =~ /lp:\s+\#\d+(?:,\s*\#\d+)*/pig)) { $closes{$_} = 1 foreach (${^MATCH} =~ /\#?\s?(\d+)/g); } my @closes = sort { $a <=> $b } keys %closes; return \@closes; } =back =head1 CHANGES =head2 Version 0.xx This is a semi-private module. Only documented functions are public. =cut 1;