| line | stmt | bran | cond | sub | pod | time | code | 
| 1 |  |  |  |  |  |  | # | 
| 2 |  |  |  |  |  |  | # $Id: IPv4.pm,v 7609c9d085d3 2018/03/15 15:17:19 gomor $ | 
| 3 |  |  |  |  |  |  | # | 
| 4 |  |  |  |  |  |  | package Net::Frame::Layer::IPv4; | 
| 5 | 2 |  |  | 2 |  | 5124 | use strict; | 
|  | 2 |  |  |  |  | 9 |  | 
|  | 2 |  |  |  |  | 52 |  | 
| 6 | 2 |  |  | 2 |  | 9 | use warnings; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 59 |  | 
| 7 |  |  |  |  |  |  |  | 
| 8 | 2 |  |  | 2 |  | 369 | use Net::Frame::Layer qw(:consts :subs); | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 477 |  | 
| 9 |  |  |  |  |  |  | require Exporter; | 
| 10 |  |  |  |  |  |  | our @ISA = qw(Net::Frame::Layer Exporter); | 
| 11 |  |  |  |  |  |  |  | 
| 12 |  |  |  |  |  |  | our %EXPORT_TAGS = ( | 
| 13 |  |  |  |  |  |  | consts => [qw( | 
| 14 |  |  |  |  |  |  | NF_IPv4_HDR_LEN | 
| 15 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_ICMPv4 | 
| 16 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IGMP | 
| 17 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IPIP | 
| 18 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_TCP | 
| 19 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_EGP | 
| 20 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IGRP | 
| 21 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_CHAOS | 
| 22 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_UDP | 
| 23 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IDP | 
| 24 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_DCCP | 
| 25 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IPv6 | 
| 26 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IPv6ROUTING | 
| 27 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IPv6FRAGMENT | 
| 28 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IDRP | 
| 29 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_RSVP | 
| 30 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_GRE | 
| 31 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_ESP | 
| 32 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_AH | 
| 33 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_ICMPv6 | 
| 34 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_EIGRP | 
| 35 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_OSPF | 
| 36 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_ETHERIP | 
| 37 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_PIM | 
| 38 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_VRRP | 
| 39 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_STP | 
| 40 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_SCTP | 
| 41 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_UDPLITE | 
| 42 |  |  |  |  |  |  | NF_IPv4_MORE_FRAGMENT | 
| 43 |  |  |  |  |  |  | NF_IPv4_DONT_FRAGMENT | 
| 44 |  |  |  |  |  |  | NF_IPv4_RESERVED_FRAGMENT | 
| 45 |  |  |  |  |  |  | )], | 
| 46 |  |  |  |  |  |  | ); | 
| 47 |  |  |  |  |  |  | our @EXPORT_OK = ( | 
| 48 |  |  |  |  |  |  | @{$EXPORT_TAGS{consts}}, | 
| 49 |  |  |  |  |  |  | ); | 
| 50 |  |  |  |  |  |  |  | 
| 51 | 2 |  |  | 2 |  | 14 | use constant NF_IPv4_HDR_LEN           => 20; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 119 |  | 
| 52 | 2 |  |  | 2 |  | 11 | use constant NF_IPv4_PROTOCOL_ICMPv4       => 0x01; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 73 |  | 
| 53 | 2 |  |  | 2 |  | 8 | use constant NF_IPv4_PROTOCOL_IGMP         => 0x02; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 64 |  | 
| 54 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_IPIP         => 0x04; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 74 |  | 
| 55 | 2 |  |  | 2 |  | 9 | use constant NF_IPv4_PROTOCOL_TCP          => 0x06; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 79 |  | 
| 56 | 2 |  |  | 2 |  | 8 | use constant NF_IPv4_PROTOCOL_EGP          => 0x08; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 79 |  | 
| 57 | 2 |  |  | 2 |  | 11 | use constant NF_IPv4_PROTOCOL_IGRP         => 0x09; | 
|  | 2 |  |  |  |  | 2 |  | 
|  | 2 |  |  |  |  | 107 |  | 
| 58 | 2 |  |  | 2 |  | 11 | use constant NF_IPv4_PROTOCOL_CHAOS        => 0x10; | 
|  | 2 |  |  |  |  | 2 |  | 
|  | 2 |  |  |  |  | 79 |  | 
| 59 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_UDP          => 0x11; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 64 |  | 
| 60 | 2 |  |  | 2 |  | 8 | use constant NF_IPv4_PROTOCOL_IDP          => 0x16; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 78 |  | 
| 61 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_DCCP         => 0x21; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 68 |  | 
| 62 | 2 |  |  | 2 |  | 9 | use constant NF_IPv4_PROTOCOL_IPv6         => 0x29; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 76 |  | 
| 63 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_IPv6ROUTING  => 0x2b; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 72 |  | 
| 64 | 2 |  |  | 2 |  | 8 | use constant NF_IPv4_PROTOCOL_IPv6FRAGMENT => 0x2c; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 79 |  | 
| 65 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_IDRP         => 0x2d; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 67 |  | 
| 66 | 2 |  |  | 2 |  | 8 | use constant NF_IPv4_PROTOCOL_RSVP         => 0x2e; | 
|  | 2 |  |  |  |  | 2 |  | 
|  | 2 |  |  |  |  | 90 |  | 
| 67 | 2 |  |  | 2 |  | 11 | use constant NF_IPv4_PROTOCOL_GRE          => 0x2f; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 72 |  | 
| 68 | 2 |  |  | 2 |  | 9 | use constant NF_IPv4_PROTOCOL_ESP          => 0x32; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 76 |  | 
| 69 | 2 |  |  | 2 |  | 9 | use constant NF_IPv4_PROTOCOL_AH           => 0x33; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 83 |  | 
| 70 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_ICMPv6       => 0x3a; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 73 |  | 
| 71 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_EIGRP        => 0x58; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 83 |  | 
| 72 | 2 |  |  | 2 |  | 9 | use constant NF_IPv4_PROTOCOL_OSPF         => 0x59; | 
|  | 2 |  |  |  |  | 10 |  | 
|  | 2 |  |  |  |  | 85 |  | 
| 73 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_ETHERIP      => 0x61; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 72 |  | 
| 74 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_PIM          => 0x67; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 89 |  | 
| 75 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_VRRP         => 0x70; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 72 |  | 
| 76 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_STP          => 0x76; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 71 |  | 
| 77 | 2 |  |  | 2 |  | 9 | use constant NF_IPv4_PROTOCOL_SCTP         => 0x84; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 75 |  | 
| 78 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_PROTOCOL_UDPLITE      => 0x88; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 107 |  | 
| 79 | 2 |  |  | 2 |  | 9 | use constant NF_IPv4_MORE_FRAGMENT     => 1; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 91 |  | 
| 80 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_DONT_FRAGMENT     => 2; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 83 |  | 
| 81 | 2 |  |  | 2 |  | 10 | use constant NF_IPv4_RESERVED_FRAGMENT => 4; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 293 |  | 
| 82 |  |  |  |  |  |  |  | 
| 83 |  |  |  |  |  |  | our @AS = qw( | 
| 84 |  |  |  |  |  |  | id | 
| 85 |  |  |  |  |  |  | ttl | 
| 86 |  |  |  |  |  |  | src | 
| 87 |  |  |  |  |  |  | dst | 
| 88 |  |  |  |  |  |  | protocol | 
| 89 |  |  |  |  |  |  | checksum | 
| 90 |  |  |  |  |  |  | flags | 
| 91 |  |  |  |  |  |  | offset | 
| 92 |  |  |  |  |  |  | version | 
| 93 |  |  |  |  |  |  | tos | 
| 94 |  |  |  |  |  |  | length | 
| 95 |  |  |  |  |  |  | hlen | 
| 96 |  |  |  |  |  |  | options | 
| 97 |  |  |  |  |  |  | noFixLen | 
| 98 |  |  |  |  |  |  | ); | 
| 99 |  |  |  |  |  |  | __PACKAGE__->cgBuildIndices; | 
| 100 |  |  |  |  |  |  | __PACKAGE__->cgBuildAccessorsScalar(\@AS); | 
| 101 |  |  |  |  |  |  |  | 
| 102 |  |  |  |  |  |  | BEGIN { | 
| 103 | 2 |  |  | 2 |  | 22 | my $osname = { | 
| 104 |  |  |  |  |  |  | freebsd => [ \&_fixLenBsd, ], | 
| 105 |  |  |  |  |  |  | netbsd  => [ \&_fixLenBsd, ], | 
| 106 |  |  |  |  |  |  | openbsd => [ \&_fixLenBsd, ], | 
| 107 |  |  |  |  |  |  | darwin  => [ \&_fixLenBsd, ], | 
| 108 |  |  |  |  |  |  | }; | 
| 109 |  |  |  |  |  |  |  | 
| 110 | 2 |  | 50 |  |  | 65 | *_fixLen = $osname->{$^O}->[0] || \&_fixLenOther; | 
| 111 |  |  |  |  |  |  | } | 
| 112 |  |  |  |  |  |  |  | 
| 113 | 2 |  |  | 2 |  | 11 | no strict 'vars'; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 93 |  | 
| 114 |  |  |  |  |  |  |  | 
| 115 | 2 |  |  | 2 |  | 10 | use Carp; | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 95 |  | 
| 116 | 2 |  |  | 2 |  | 854 | use Bit::Vector; | 
|  | 2 |  |  |  |  | 1709 |  | 
|  | 2 |  |  |  |  | 2264 |  | 
| 117 |  |  |  |  |  |  |  | 
| 118 | 0 |  |  | 0 |  | 0 | sub _fixLenBsd   { pack('v', shift) } | 
| 119 | 1 |  |  | 1 |  | 6 | sub _fixLenOther { pack('n', shift) } | 
| 120 |  |  |  |  |  |  |  | 
| 121 |  |  |  |  |  |  | sub new { | 
| 122 |  |  |  |  |  |  | shift->SUPER::new( | 
| 123 | 1 |  |  | 1 | 1 | 14 | version  => 4, | 
| 124 |  |  |  |  |  |  | tos      => 0, | 
| 125 |  |  |  |  |  |  | id       => getRandom16bitsInt(), | 
| 126 |  |  |  |  |  |  | length   => NF_IPv4_HDR_LEN, | 
| 127 |  |  |  |  |  |  | hlen     => 5, | 
| 128 |  |  |  |  |  |  | flags    => 0, | 
| 129 |  |  |  |  |  |  | offset   => 0, | 
| 130 |  |  |  |  |  |  | ttl      => 128, | 
| 131 |  |  |  |  |  |  | protocol => NF_IPv4_PROTOCOL_TCP, | 
| 132 |  |  |  |  |  |  | checksum => 0, | 
| 133 |  |  |  |  |  |  | src      => '127.0.0.1', | 
| 134 |  |  |  |  |  |  | dst      => '127.0.0.1', | 
| 135 |  |  |  |  |  |  | options  => '', | 
| 136 |  |  |  |  |  |  | noFixLen => 0, | 
| 137 |  |  |  |  |  |  | @_, | 
| 138 |  |  |  |  |  |  | ); | 
| 139 |  |  |  |  |  |  | } | 
| 140 |  |  |  |  |  |  |  | 
| 141 |  |  |  |  |  |  | sub pack { | 
| 142 | 1 |  |  | 1 | 1 | 430 | my $self = shift; | 
| 143 |  |  |  |  |  |  |  | 
| 144 |  |  |  |  |  |  | # Here, we pack in this order: version, hlen (4 bits each) | 
| 145 | 1 |  |  |  |  | 15 | my $version = Bit::Vector->new_Dec(4, $self->[$__version]); | 
| 146 | 1 |  |  |  |  | 4 | my $hlen    = Bit::Vector->new_Dec(4, $self->[$__hlen]); | 
| 147 | 1 |  |  |  |  | 6 | my $v8      = $version->Concat_List($hlen); | 
| 148 |  |  |  |  |  |  |  | 
| 149 |  |  |  |  |  |  | # Here, we pack in this order: flags (3 bits), offset (13 bits) | 
| 150 | 1 |  |  |  |  | 5 | my $flags  = Bit::Vector->new_Dec(3,  $self->[$__flags]); | 
| 151 | 1 |  |  |  |  | 3 | my $offset = Bit::Vector->new_Dec(13, $self->[$__offset]); | 
| 152 | 1 |  |  |  |  | 3 | my $v16    = $flags->Concat_List($offset); | 
| 153 |  |  |  |  |  |  |  | 
| 154 | 1 | 50 |  |  |  | 5 | my $len = ($self->[$__noFixLen] ? _fixLenOther($self->[$__length]) | 
| 155 |  |  |  |  |  |  | : _fixLen($self->[$__length])); | 
| 156 |  |  |  |  |  |  |  | 
| 157 | 1 | 50 |  |  |  | 9 | $self->[$__raw] = $self->SUPER::pack('CCa*nnCCna4a4', | 
| 158 |  |  |  |  |  |  | $v8->to_Dec, | 
| 159 |  |  |  |  |  |  | $self->[$__tos], | 
| 160 |  |  |  |  |  |  | $len, | 
| 161 |  |  |  |  |  |  | $self->[$__id], | 
| 162 |  |  |  |  |  |  | $v16->to_Dec, | 
| 163 |  |  |  |  |  |  | $self->[$__ttl], | 
| 164 |  |  |  |  |  |  | $self->[$__protocol], | 
| 165 |  |  |  |  |  |  | $self->[$__checksum], | 
| 166 |  |  |  |  |  |  | inetAton($self->[$__src]), | 
| 167 |  |  |  |  |  |  | inetAton($self->[$__dst]), | 
| 168 |  |  |  |  |  |  | ) or return undef; | 
| 169 |  |  |  |  |  |  |  | 
| 170 | 1 |  |  |  |  | 2 | my $opt; | 
| 171 | 1 | 50 |  |  |  | 3 | if ($self->[$__options]) { | 
| 172 | 0 | 0 |  |  |  | 0 | $opt = $self->SUPER::pack('a*', $self->[$__options]) | 
| 173 |  |  |  |  |  |  | or return undef; | 
| 174 | 0 |  |  |  |  | 0 | $self->[$__raw] = $self->[$__raw].$opt; | 
| 175 |  |  |  |  |  |  | } | 
| 176 |  |  |  |  |  |  |  | 
| 177 | 1 |  |  |  |  | 7 | $self->[$__raw]; | 
| 178 |  |  |  |  |  |  | } | 
| 179 |  |  |  |  |  |  |  | 
| 180 |  |  |  |  |  |  | sub unpack { | 
| 181 | 1 |  |  | 1 | 1 | 5 | my $self = shift; | 
| 182 |  |  |  |  |  |  |  | 
| 183 | 1 | 50 |  |  |  | 5 | my ($verHlen, $tos, $len, $id, $flagsOffset, $ttl, $proto, $cksum, $src, | 
| 184 |  |  |  |  |  |  | $dst, $payload) = $self->SUPER::unpack('CCnnnCCna4a4 a*', $self->[$__raw]) | 
| 185 |  |  |  |  |  |  | or return undef; | 
| 186 |  |  |  |  |  |  |  | 
| 187 | 1 |  |  |  |  | 5 | my $v8  = Bit::Vector->new_Dec(8,  $verHlen); | 
| 188 | 1 |  |  |  |  | 3 | my $v16 = Bit::Vector->new_Dec(16, $flagsOffset); | 
| 189 |  |  |  |  |  |  |  | 
| 190 |  |  |  |  |  |  | # Here, we unpack in this order: hlen, version (4 bits each) | 
| 191 | 1 |  |  |  |  | 5 | $self->[$__hlen]     = $v8->Chunk_Read(4, 0); | 
| 192 | 1 |  |  |  |  | 3 | $self->[$__version]  = $v8->Chunk_Read(4, 4); | 
| 193 | 1 |  |  |  |  | 2 | $self->[$__tos]      = $tos; | 
| 194 | 1 |  |  |  |  | 2 | $self->[$__length]   = $len; | 
| 195 | 1 |  |  |  |  | 2 | $self->[$__id]       = $id; | 
| 196 |  |  |  |  |  |  | # Here, we unpack in this order: offset (13 bits), flags (3 bits) | 
| 197 | 1 |  |  |  |  | 2 | $self->[$__offset]   = $v16->Chunk_Read(13,  0); | 
| 198 | 1 |  |  |  |  | 3 | $self->[$__flags]    = $v16->Chunk_Read( 3, 13); | 
| 199 | 1 |  |  |  |  | 2 | $self->[$__ttl]      = $ttl; | 
| 200 | 1 |  |  |  |  | 2 | $self->[$__protocol] = $proto; | 
| 201 | 1 |  |  |  |  | 2 | $self->[$__checksum] = $cksum; | 
| 202 | 1 |  |  |  |  | 3 | $self->[$__src]      = inetNtoa($src); | 
| 203 | 1 |  |  |  |  | 3 | $self->[$__dst]      = inetNtoa($dst); | 
| 204 | 1 |  |  |  |  | 2 | $self->[$__payload]  = $payload; | 
| 205 |  |  |  |  |  |  |  | 
| 206 | 1 | 50 |  |  |  | 3 | my ($options, $payload2) = $self->SUPER::unpack( | 
| 207 |  |  |  |  |  |  | 'a'. $self->getOptionsLength. 'a*', $self->[$__payload] | 
| 208 |  |  |  |  |  |  | ) or return undef; | 
| 209 |  |  |  |  |  |  |  | 
| 210 | 1 |  |  |  |  | 2 | $self->[$__options] = $options; | 
| 211 | 1 |  |  |  |  | 10 | $self->[$__payload] = $payload2; | 
| 212 |  |  |  |  |  |  |  | 
| 213 | 1 |  |  |  |  | 6 | $self; | 
| 214 |  |  |  |  |  |  | } | 
| 215 |  |  |  |  |  |  |  | 
| 216 |  |  |  |  |  |  | sub getLength { | 
| 217 | 1 |  |  | 1 | 1 | 2 | my $self = shift; | 
| 218 | 1 | 50 |  |  |  | 4 | $self->[$__hlen] > 0 ? $self->[$__hlen] * 4 : 0; | 
| 219 |  |  |  |  |  |  | } | 
| 220 |  |  |  |  |  |  |  | 
| 221 |  |  |  |  |  |  | sub getPayloadLength { | 
| 222 | 0 |  |  | 0 | 1 | 0 | my $self = shift; | 
| 223 | 0 |  |  |  |  | 0 | my $gLen = $self->getLength; | 
| 224 | 0 | 0 |  |  |  | 0 | $self->[$__length] > $gLen ? $self->[$__length] - $gLen : 0; | 
| 225 |  |  |  |  |  |  | } | 
| 226 |  |  |  |  |  |  |  | 
| 227 |  |  |  |  |  |  | sub getOptionsLength { | 
| 228 | 1 |  |  | 1 | 1 | 2 | my $self = shift; | 
| 229 | 1 |  |  |  |  | 3 | my $gLen = $self->getLength; | 
| 230 | 1 |  |  |  |  | 2 | my $hLen = NF_IPv4_HDR_LEN; | 
| 231 | 1 | 50 |  |  |  | 7 | $gLen > $hLen ? $gLen - $hLen : 0; | 
| 232 |  |  |  |  |  |  | } | 
| 233 |  |  |  |  |  |  |  | 
| 234 |  |  |  |  |  |  | sub computeLengths { | 
| 235 | 0 |  |  | 0 | 1 | 0 | my $self = shift; | 
| 236 | 0 |  |  |  |  | 0 | my ($layers) = @_; | 
| 237 |  |  |  |  |  |  |  | 
| 238 | 0 |  |  |  |  | 0 | my $hLen = NF_IPv4_HDR_LEN; | 
| 239 | 0 | 0 |  |  |  | 0 | $hLen   += length($self->[$__options]) if $self->[$__options]; | 
| 240 | 0 |  |  |  |  | 0 | $self->[$__hlen] = $hLen / 4; | 
| 241 |  |  |  |  |  |  |  | 
| 242 | 0 |  |  |  |  | 0 | my $len = $hLen; | 
| 243 | 0 |  |  |  |  | 0 | my $last; | 
| 244 |  |  |  |  |  |  | my $start; | 
| 245 | 0 |  |  |  |  | 0 | for my $l (@$layers) { | 
| 246 | 0 | 0 |  |  |  | 0 | if (! $start) { | 
| 247 | 0 | 0 |  |  |  | 0 | $start++ if $l->layer eq 'IPv4'; | 
| 248 | 0 |  |  |  |  | 0 | next; | 
| 249 |  |  |  |  |  |  | } | 
| 250 | 0 |  |  |  |  | 0 | $len += $l->getLength; | 
| 251 | 0 |  |  |  |  | 0 | $last = $l; | 
| 252 |  |  |  |  |  |  | } | 
| 253 | 0 | 0 |  |  |  | 0 | if (defined($last->payload)) { | 
| 254 | 0 |  |  |  |  | 0 | $len += length($last->payload); | 
| 255 |  |  |  |  |  |  | } | 
| 256 |  |  |  |  |  |  |  | 
| 257 | 0 |  |  |  |  | 0 | $self->length($len); | 
| 258 |  |  |  |  |  |  |  | 
| 259 | 0 |  |  |  |  | 0 | return 1; | 
| 260 |  |  |  |  |  |  | } | 
| 261 |  |  |  |  |  |  |  | 
| 262 |  |  |  |  |  |  | sub computeChecksums { | 
| 263 | 0 |  |  | 0 | 1 | 0 | my $self = shift; | 
| 264 | 0 |  |  |  |  | 0 | my ($layers) = @_; | 
| 265 |  |  |  |  |  |  |  | 
| 266 |  |  |  |  |  |  | # Reset the checksum if already filled by a previous pack | 
| 267 | 0 | 0 |  |  |  | 0 | if ($self->[$__checksum]) { | 
| 268 | 0 |  |  |  |  | 0 | $self->[$__checksum] = 0; | 
| 269 |  |  |  |  |  |  | } | 
| 270 |  |  |  |  |  |  |  | 
| 271 | 0 |  |  |  |  | 0 | $self->[$__checksum] = inetChecksum($self->pack); | 
| 272 |  |  |  |  |  |  |  | 
| 273 | 0 |  |  |  |  | 0 | return 1; | 
| 274 |  |  |  |  |  |  | } | 
| 275 |  |  |  |  |  |  |  | 
| 276 |  |  |  |  |  |  | our $Next = { | 
| 277 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_ICMPv4()       => 'ICMPv4', | 
| 278 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IGMP()         => 'IGMP', | 
| 279 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IPIP()         => 'IPv4', | 
| 280 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_TCP()          => 'TCP', | 
| 281 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_EGP()          => 'EGP', | 
| 282 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IGRP()         => 'IGRP', | 
| 283 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_CHAOS()        => 'CHAOS', | 
| 284 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_UDP()          => 'UDP', | 
| 285 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IDP()          => 'IDP', | 
| 286 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_DCCP()         => 'DCCP', | 
| 287 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IPv6()         => 'IPv6', | 
| 288 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IPv6ROUTING()  => 'IPv6Routing', | 
| 289 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IPv6FRAGMENT() => 'IPv6Fragment', | 
| 290 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_IDRP()         => 'IDRP', | 
| 291 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_RSVP()         => 'RSVP', | 
| 292 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_GRE()          => 'GRE', | 
| 293 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_ESP()          => 'ESP', | 
| 294 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_AH()           => 'AH', | 
| 295 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_ICMPv6()       => 'ICMPv6', | 
| 296 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_EIGRP()        => 'EIGRP', | 
| 297 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_OSPF()         => 'OSPF', | 
| 298 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_ETHERIP()      => 'ETHERIP', | 
| 299 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_PIM()          => 'PIM', | 
| 300 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_VRRP()         => 'VRRP', | 
| 301 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_STP()          => 'STP', | 
| 302 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_SCTP()         => 'SCTP', | 
| 303 |  |  |  |  |  |  | NF_IPv4_PROTOCOL_UDPLITE()      => 'UDPLite', | 
| 304 |  |  |  |  |  |  | }; | 
| 305 |  |  |  |  |  |  |  | 
| 306 |  |  |  |  |  |  | sub encapsulate { | 
| 307 | 1 |  |  | 1 | 1 | 6 | my $self = shift; | 
| 308 |  |  |  |  |  |  |  | 
| 309 | 1 | 50 |  |  |  | 4 | return $self->[$__nextLayer] if $self->[$__nextLayer]; | 
| 310 |  |  |  |  |  |  |  | 
| 311 | 1 |  | 50 |  |  | 7 | return $Next->{$self->[$__protocol]} || NF_LAYER_UNKNOWN; | 
| 312 |  |  |  |  |  |  | } | 
| 313 |  |  |  |  |  |  |  | 
| 314 |  |  |  |  |  |  | sub print { | 
| 315 | 1 |  |  | 1 | 1 | 5 | my $self = shift; | 
| 316 |  |  |  |  |  |  |  | 
| 317 | 1 |  |  |  |  | 6 | my $l = $self->layer; | 
| 318 | 1 |  |  |  |  | 13 | my $buf = sprintf | 
| 319 |  |  |  |  |  |  | "$l: version:%d  hlen:%d  tos:0x%02x  length:%d  id:%d\n". | 
| 320 |  |  |  |  |  |  | "$l: flags:0x%02x  offset:%d  ttl:%d  protocol:0x%02x  checksum:0x%04x\n". | 
| 321 |  |  |  |  |  |  | "$l: src:%s  dst:%s", | 
| 322 |  |  |  |  |  |  | $self->[$__version], $self->[$__hlen], $self->[$__tos], | 
| 323 |  |  |  |  |  |  | $self->[$__length], $self->[$__id], $self->[$__flags], | 
| 324 |  |  |  |  |  |  | $self->[$__offset], $self->[$__ttl], $self->[$__protocol], | 
| 325 |  |  |  |  |  |  | $self->[$__checksum], $self->[$__src], $self->[$__dst]; | 
| 326 |  |  |  |  |  |  |  | 
| 327 | 1 | 50 |  |  |  | 3 | if ($self->[$__options]) { | 
| 328 | 0 |  |  |  |  | 0 | $buf .= sprintf "\n$l: optionsLength:%d  options:%s", | 
| 329 |  |  |  |  |  |  | $self->getOptionsLength, | 
| 330 |  |  |  |  |  |  | CORE::unpack('H*', $self->[$__options]); | 
| 331 |  |  |  |  |  |  | } | 
| 332 |  |  |  |  |  |  |  | 
| 333 | 1 |  |  |  |  | 43 | $buf; | 
| 334 |  |  |  |  |  |  | } | 
| 335 |  |  |  |  |  |  |  | 
| 336 |  |  |  |  |  |  | 1; | 
| 337 |  |  |  |  |  |  |  | 
| 338 |  |  |  |  |  |  | __END__ |