line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
# -*- Perl -*- |
2
|
|
|
|
|
|
|
# |
3
|
|
|
|
|
|
|
# File::Cmp - compare two files character by character like cmp(1) |
4
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
package File::Cmp; |
6
|
|
|
|
|
|
|
|
7
|
1
|
|
|
1
|
|
235248
|
use 5.008000; |
|
1
|
|
|
|
|
4
|
|
8
|
1
|
|
|
1
|
|
5
|
use strict; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
19
|
|
9
|
1
|
|
|
1
|
|
4
|
use warnings; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
27
|
|
10
|
|
|
|
|
|
|
|
11
|
1
|
|
|
1
|
|
5
|
use Carp qw/croak/; |
|
1
|
|
|
|
|
5
|
|
|
1
|
|
|
|
|
41
|
|
12
|
1
|
|
|
1
|
|
5
|
use Scalar::Util qw/reftype/; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
761
|
|
13
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
require Exporter; |
15
|
|
|
|
|
|
|
our @ISA = qw(Exporter); |
16
|
|
|
|
|
|
|
our @EXPORT_OK = qw/&fcmp/; |
17
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
our $VERSION = '1.09'; |
19
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
# XXX 'skip' and 'limit' might be good parameters to add, to skip X |
21
|
|
|
|
|
|
|
# initial bytes, limit work to Y bytes of data to check |
22
|
|
|
|
|
|
|
sub fcmp { |
23
|
13
|
100
|
|
13
|
1
|
7924
|
croak 'fcmp needs two files' if @_ < 2; |
24
|
12
|
|
|
|
|
31
|
my @files = splice @_, 0, 2; |
25
|
12
|
100
|
100
|
|
|
94
|
my $param = ( @_ == 1 and ref $_[0] eq 'HASH' ) ? $_[0] : {@_}; |
26
|
|
|
|
|
|
|
|
27
|
12
|
100
|
|
|
|
39
|
$param->{sizecheck} = 1 unless exists $param->{sizecheck}; |
28
|
12
|
100
|
|
|
|
25
|
$param->{sizecheck} = 0 if exists $param->{tells}; |
29
|
|
|
|
|
|
|
|
30
|
12
|
100
|
|
|
|
24
|
if ( $param->{fscheck} ) { |
31
|
4
|
|
|
|
|
6
|
my @statbuf; |
32
|
4
|
|
|
|
|
7
|
for my $f (@files) { |
33
|
|
|
|
|
|
|
# stat has the handy property of chasing symlinks for us |
34
|
7
|
100
|
|
|
|
217
|
my @devino = ( stat $f )[ 0, 1 ] or croak "could not stat: $!"; |
35
|
6
|
|
|
|
|
19
|
push @statbuf, \@devino; |
36
|
|
|
|
|
|
|
} |
37
|
3
|
100
|
66
|
|
|
19
|
if ( $statbuf[0][0] == $statbuf[1][0] |
38
|
|
|
|
|
|
|
and $statbuf[0][1] == $statbuf[1][1] ) { |
39
|
2
|
100
|
|
|
|
8
|
${ $param->{reason} } = 'fscheck' if exists $param->{reason}; |
|
1
|
|
|
|
|
2
|
|
40
|
2
|
|
|
|
|
13
|
return 1; # assume files identical as both dev and inode match |
41
|
|
|
|
|
|
|
} |
42
|
|
|
|
|
|
|
} |
43
|
|
|
|
|
|
|
|
44
|
|
|
|
|
|
|
# The files are probably not identical if they differ in size; |
45
|
|
|
|
|
|
|
# however, offer means to turn this check off if -s for some reason is |
46
|
|
|
|
|
|
|
# incorrect (or if 'tells' is on so we need to find roughly where the |
47
|
|
|
|
|
|
|
# difference is in the files). |
48
|
9
|
100
|
50
|
|
|
131
|
if ( $param->{sizecheck} |
|
|
|
50
|
|
|
|
|
|
|
|
100
|
|
|
|
|
49
|
|
|
|
|
|
|
and ( ( -s $files[0] ) // -1 ) != ( ( -s $files[1] ) // -2 ) ) { |
50
|
4
|
100
|
|
|
|
14
|
${ $param->{reason} } = 'size' if exists $param->{reason}; |
|
2
|
|
|
|
|
8
|
|
51
|
4
|
|
|
|
|
32
|
return 0; |
52
|
|
|
|
|
|
|
} |
53
|
|
|
|
|
|
|
|
54
|
5
|
|
|
|
|
11
|
my @fhs; |
55
|
5
|
|
|
|
|
7
|
for my $f (@files) { |
56
|
8
|
100
|
|
|
|
27
|
if ( !defined reftype $f) { |
57
|
7
|
100
|
|
|
|
396
|
open my $fh, '<', $f or croak "could not open $f: $!"; |
58
|
6
|
|
|
|
|
26
|
push @fhs, $fh; |
59
|
|
|
|
|
|
|
} else { |
60
|
|
|
|
|
|
|
# Assume is a GLOB or something can readline on, XXX might want to |
61
|
|
|
|
|
|
|
# better check this |
62
|
1
|
|
|
|
|
4
|
push @fhs, $f; |
63
|
|
|
|
|
|
|
} |
64
|
7
|
100
|
|
|
|
21
|
if ( exists $param->{binmode} ) { |
65
|
1
|
100
|
|
1
|
|
7
|
binmode $fhs[-1], $param->{binmode} or croak "binmode failed: $!"; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
6
|
|
|
3
|
|
|
|
|
57
|
|
66
|
|
|
|
|
|
|
} |
67
|
|
|
|
|
|
|
} |
68
|
|
|
|
|
|
|
|
69
|
3
|
50
|
|
|
|
8
|
local $/ = $param->{RS} if exists $param->{RS}; |
70
|
|
|
|
|
|
|
|
71
|
3
|
|
|
|
|
4
|
while (1) { |
72
|
18
|
|
|
|
|
100
|
my $eof1 = eof $fhs[0]; |
73
|
18
|
|
|
|
|
94
|
my $eof2 = eof $fhs[1]; |
74
|
|
|
|
|
|
|
# Done if both files are at EOF; otherwise assume they differ if one |
75
|
|
|
|
|
|
|
# completes before the other (this second case would normally be |
76
|
|
|
|
|
|
|
# optimized away by the -s test, above). |
77
|
18
|
100
|
66
|
|
|
38
|
last if $eof1 and $eof2; |
78
|
17
|
100
|
50
|
|
|
48
|
if ( $eof1 xor $eof2 ) { |
79
|
1
|
50
|
|
|
|
9
|
${ $param->{reason} } = 'eof' if exists $param->{reason}; |
|
1
|
|
|
|
|
4
|
|
80
|
1
|
|
|
|
|
4
|
@{ $param->{tells} } = ( tell $fhs[0], tell $fhs[1] ) |
81
|
1
|
50
|
|
|
|
8
|
if exists $param->{tells}; |
82
|
1
|
|
|
|
|
29
|
return 0; |
83
|
|
|
|
|
|
|
} |
84
|
|
|
|
|
|
|
|
85
|
16
|
|
|
|
|
30
|
my $this = readline $fhs[0]; |
86
|
16
|
50
|
|
|
|
24
|
croak "error reading from first file: $!" if !defined $this; |
87
|
16
|
|
|
|
|
33
|
my $that = readline $fhs[1]; |
88
|
16
|
50
|
|
|
|
29
|
croak "error reading from second file: $!" if !defined $that; |
89
|
|
|
|
|
|
|
|
90
|
16
|
100
|
|
|
|
30
|
if ( $this ne $that ) { |
91
|
1
|
|
|
|
|
3
|
@{ $param->{tells} } = ( tell $fhs[0], tell $fhs[1] ) |
92
|
1
|
50
|
|
|
|
8
|
if exists $param->{tells}; |
93
|
1
|
50
|
|
|
|
5
|
${ $param->{reason} } = 'diff' if exists $param->{reason}; |
|
1
|
|
|
|
|
4
|
|
94
|
1
|
|
|
|
|
28
|
return 0; |
95
|
|
|
|
|
|
|
} |
96
|
|
|
|
|
|
|
} |
97
|
|
|
|
|
|
|
|
98
|
1
|
|
|
|
|
25
|
return 1; # assume files identical if get this far |
99
|
|
|
|
|
|
|
} |
100
|
|
|
|
|
|
|
|
101
|
|
|
|
|
|
|
1; |
102
|
|
|
|
|
|
|
__END__ |