7b9b36bd2e
Signed-off-by: Chris Cormack <crc@liblime.com> Signed-off-by: Joshua Ferraro <jmf@liblime.com>
55 lines
1 KiB
Perl
55 lines
1 KiB
Perl
package Sip::Checksum;
|
|
|
|
use Exporter;
|
|
use strict;
|
|
use warnings;
|
|
|
|
our @ISA = qw(Exporter);
|
|
our @EXPORT_OK = qw(checksum verify_cksum);
|
|
|
|
sub checksum {
|
|
my $pkt = shift;
|
|
|
|
return (-unpack('%16U*', $pkt) & 0xFFFF);
|
|
}
|
|
|
|
sub verify_cksum {
|
|
my $pkt = shift;
|
|
my $cksum;
|
|
my $shortsum;
|
|
|
|
return 0 if (substr($pkt, -6, 2) ne "AZ"); # No checksum at end
|
|
|
|
# Convert the checksum back to hex and calculate the sum of the
|
|
# pack without the checksum.
|
|
$cksum = hex(substr($pkt, -4));
|
|
$shortsum = unpack("%16U*", substr($pkt, 0, -4));
|
|
|
|
# The checksum is valid if the hex sum, plus the checksum of the
|
|
# base packet short when truncated to 16 bits.
|
|
return (($cksum + $shortsum) & 0xFFFF) == 0;
|
|
}
|
|
|
|
{
|
|
no warnings qw(once);
|
|
eval join('',<main::DATA>) || die $@ unless caller();
|
|
}
|
|
__END__
|
|
|
|
#
|
|
# Some simple test data
|
|
#
|
|
sub test {
|
|
my $testpkt = shift;
|
|
my $cksum = checksum($testpkt);
|
|
my $fullpkt = sprintf("%s%4X", $testpkt, $cksum);
|
|
|
|
print $fullpkt, "\n";
|
|
}
|
|
|
|
while (<>) {
|
|
chomp;
|
|
test($_);
|
|
}
|
|
|
|
1;
|