# ex:ts=8 sw=4: # $OpenBSD: Signature.pm,v 1.13 2023/05/06 05:20:31 espie Exp $ # # Copyright (c) 2010-2013 Marc Espie # # Permission to use, copy, modify, and distribute this software for any # purpose with or without fee is hereby granted, provided that the above # copyright notice and this permission notice appear in all copies. # # THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES # WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR # ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF # OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. use v5.36; use OpenBSD::LibSpec; use OpenBSD::LibSpec::Build; package DPB::Signature::Dir; sub best($h, $lib) { if ($lib->is_static) { $h->{a}->{$lib->stem} = 1; return; } my $old = $h->{$lib->stem} //= $lib; return if $old eq $lib; return if $old->major > $lib->major; return if $old->major == $lib->major && $old->minor > $lib->minor; $h->{$lib->stem} = $lib; } sub is_empty($h) { return keys %$h == 0; } sub new($class) { bless {}, $class; } sub compare1($s1, $s2, $h1, $h2, $full) { my $r = ''; while (my ($stem, $lib) = each %$s1) { next if $stem eq 'la'; next if $stem eq 'a'; if (!defined $s2->{$stem}) { $r .= " can't find ".$lib->to_string." on $h2\n"; } elsif ($full && $s2->{$stem}->to_string ne $lib->to_string) { $r .= " mismatch ". $s2->{$stem}->to_string." vs ". $lib->to_string."\n"; } } for my $k (keys %{$s1->{la}}) { if (!defined $s2->{la}{$k}) { $r .= " Can't find $k.la on $h2\n"; } } for my $k (keys %{$s1->{a}}) { if (!defined $s2->{a}{$k}) { $r .= " can't find lib$k.a on $h2\n"; } } return $r; } sub print_out($self, $dir, $fh) { for my $k (sort keys %$self) { next if $k eq 'la'; next if $k eq 'a'; next if !defined $self->{$k}; print $fh "\t", $self->{$k}->to_string, "\n"; } if (defined $self->{la}) { for my $v (sort keys %{$self->{la}}) { print $fh "\t$dir/$v.la\n"; } } if (defined $self->{a}) { for my $v (sort keys %{$self->{a}}) { print $fh "\t$dir/lib$v.a\n"; } } } sub compare($s1, $s2, $h1, $h2) { return compare1($s1, $s2, $h1, $h2, 0) . compare1($s2, $s1, $h2, $h1, 1); } package DPB::Signature::Task; our @ISA = qw(DPB::Task::Pipe); sub new($class, $o, $base) { my $repo = $o->{$base} = DPB::Signature::Dir->new; bless {repo => $repo, dir => "$base/lib"}, $class; } sub run($self, $core) { $core->shell->exec("/bin/ls", $self->{dir}); } sub process($self, $core) { my $fh = $core->fh; my $repo = $self->{repo}; while (<$fh>) { if ($_ =~ m/(.*).la/) { $repo->{la}->{$1} = 1; } else { my $lib = OpenBSD::Library::Build->from_string("$self->{dir}/$_"); next unless $lib->is_valid; $repo->best($lib); } } } package DPB::Signature; sub new($class, $state) { return bless { reporter => $state->{reporter}}, $class; } sub library_dirs($self) { return OpenBSD::Paths->library_dirs; } sub add_tasks($self, $job) { for my $base ($self->library_dirs) { $job->add_tasks(DPB::Signature::Task->new($self, $base)); } return $self; } sub compare($s1, $s2) { my $r = ''; for my $dir ($s1->library_dirs) { $r .= $s1->{$dir}->compare($s2->{$dir}, $s1->{host}, $s2->{host}); } if ($r) { $s1->{reporter}->myprint( "Error between $s1->{host} and $s2->{host}:\n$r"); } return $r; } my $ref; sub matches($self, $core, $logger) { $self->{host} = $core->hostname; if (!defined $ref) { # couldn't read system dir, can't possibly be okay. if ($self->{'/usr'}->is_empty) { return 0; } $ref = $self; return 1; } else { my $r = $self->compare($ref); if ($r ne '') { my $log = $logger->append('signature'); print $log "$r\n"; return 0; close $log; } else { return 1; } } } sub print_out($self, $core, $logger) { my $log = $logger->create($core->hostname.".sig"); for my $dir ($self->library_dirs) { print $log "$dir: \n"; $self->{$dir}->print_out("$dir/lib", $log); } } 1;