#
# $Id: CDP.pm,v 1.1.2.2 2006/11/14 19:13:31 gomor Exp $
#
package Net::Packet::CDP;
use strict;
use warnings;
require Net::Packet::Layer4;
our @ISA = qw(Net::Packet::Layer4);
use Net::Packet::Consts qw(:cdp :layer);
require Net::Packet::CDP::TypeDeviceId;
require Net::Packet::CDP::TypeAddresses;
require Net::Packet::CDP::TypePortId;
require Net::Packet::CDP::TypeCapabilities;
require Net::Packet::CDP::TypeSoftwareVersion;
our @AS = qw(
version
ttl
checksum
typeDeviceId
typeAddresses
typePortId
typeCapabilities
typeSoftwareVersion
);
__PACKAGE__->cgBuildIndices;
__PACKAGE__->cgBuildAccessorsScalar(\@AS);
no strict 'vars';
sub new {
shift->SUPER::new(
version => 2,
ttl => 180,
checksum => 0,
@_,
);
}
sub getLength { NP_CDP_HDR_LEN }
sub pack {
my $self = shift;
$self->[$__raw] = $self->SUPER::pack('CCn',
$self->[$__version],
$self->[$__ttl],
$self->[$__checksum],
) or return undef;
if ($self->[$__typeDeviceId]) {
$self->[$__raw] .= $self->[$__typeDeviceId]->pack
or return undef;
}
if ($self->[$__typeAddresses]) {
$self->[$__raw] .= $self->[$__typeAddresses]->pack
or return undef;
}
if ($self->[$__typePortId]) {
$self->[$__raw] .= $self->[$__typePortId]->pack
or return undef;
}
if ($self->[$__typeCapabilities]) {
$self->[$__raw] .= $self->[$__typeCapabilities]->pack
or return undef;
}
if ($self->[$__typeSoftwareVersion]) {
$self->[$__raw] .= $self->[$__typeSoftwareVersion]->pack
or return undef;
}
1;
}
sub unpack {
my $self = shift;
my ($version, $ttl, $checksum, $payload) =
$self->SUPER::unpack('CCn a*', $self->[$__raw])
or return undef;
$self->[$__version] = $version;
$self->[$__ttl] = $ttl;
$self->[$__checksum] = $checksum;
my $tail = CORE::unpack('H*', $payload);
if ($tail =~ /^0001/) {
my $type = Net::Packet::CDP::TypeDeviceId->new(raw => $payload);
$self->[$__typeDeviceId] = $type;
$payload = $type->payload;
$tail = CORE::unpack('H*', $payload);
}
if ($tail =~ /^0002/) {
my $type = Net::Packet::CDP::TypeAddresses->new(raw => $payload);
$self->[$__typeAddresses] = $type;
$payload = $type->payload;
$tail = CORE::unpack('H*', $payload);
}
if ($tail =~ /^0003/) {
my $type = Net::Packet::CDP::TypePortId->new(raw => $payload);
$self->[$__typePortId] = $type;
$payload = $type->payload;
$tail = CORE::unpack('H*', $payload);
}
if ($tail =~ /^0004/) {
my $type = Net::Packet::CDP::TypeCapabilities->new(raw => $payload);
$self->[$__typeCapabilities] = $type;
$payload = $type->payload;
$tail = CORE::unpack('H*', $payload);
}
if ($tail =~ /^0005/) {
my $type = Net::Packet::CDP::TypeSoftwareVersion->new(raw => $payload);
$self->[$__typeSoftwareVersion] = $type;
$payload = $type->payload;
$tail = CORE::unpack('H*', $payload);
}
$self->[$__payload] = $payload;
1;
}
sub encapsulate {
my $types = {
NP_LAYER_NONE() => NP_LAYER_NONE(),
};
$types->{NP_LAYER_NONE()} || NP_LAYER_UNKNOWN();
}
sub print {
my $self = shift;
my $buf = '';
my $l = $self->layer;
my $i = $self->is;
$buf .= sprintf "$l:+$i: version:%d ttl:%d checksum:0x%04x",
$self->[$__version], $self->[$__ttl], $self->[$__checksum];
if ($self->[$__typeDeviceId]) {
$buf .= "\n".$self->[$__typeDeviceId]->print;
}
if ($self->[$__typeAddresses]) {
$buf .= "\n".$self->[$__typeAddresses]->print;
}
if ($self->[$__typePortId]) {
$buf .= "\n".$self->[$__typePortId]->print;
}
if ($self->[$__typeCapabilities]) {
$buf .= "\n".$self->[$__typeCapabilities]->print;
}
if ($self->[$__typeSoftwareVersion]) {
$buf .= "\n".$self->[$__typeSoftwareVersion]->print;
}
$buf;
}
1;
__END__
=head1 NAME
Net::Packet::CDP - Cisco Discovery Protocol layer 4 object
=head1 SYNOPSIS
use Net::Packet::Consts qw(:cdp);
require Net::Packet::CDP;
# Build a layer
my $layer = Net::Packet::CDP->new(
version => 2,
ttl => 180,
checksum => 0,
);
$layer->pack;
print 'RAW: '.unpack('H*', $layer->raw)."\n";
# Read a raw layer
my $layer = Net::Packet::CDP->new(raw => $raw);
print $layer->print."\n";
print 'PAYLOAD: '.unpack('H*', $layer->payload)."\n"
if $layer->payload;
=head1 DESCRIPTION
This modules implements the encoding and decoding of the Cisco Discovery Protocol layer.
See also B<Net::Packet::Layer> and B<Net::Packet::Layer4> for other attributes and methods.
=head1 ATTRIBUTES
=over 4
=item B<version> - 8 bits
=item B<ttl> - 8 bits
=item B<checksum> - 16 bits
=item B<typeDeviceId>
=item B<typeAddresses>
=item B<typePortId>
=item B<typeCapabilities>
=item B<typeSoftwareVersion>
All these type attributes keep a pointer to the respective object. That is, for typeDeviceId, you have a pointer to a B<Net::Packet::CDP::TypeDeviceId> object, if applicable.
=back
=head1 METHODS
=over 4
=item B<new>
Object constructor. You can pass attributes that will overwrite default ones. Default values:
version: 2
ttl: 180
checksum: 0
=item B<pack>
Packs all attributes into a raw format, in order to inject to network. Returns 1 on success, undef otherwise.
=item B<unpack>
Unpacks raw data from network and stores attributes into the object. Returns 1 on success, undef otherwise.
=back
=head1 CONSTANTS
Load them: use Net::Packet::Consts qw(:cdp);
=over 4
=item B<NP_CDP_HDR_LEN>
CDP header length.
=item B<NP_CDP_TYPE_DEVICE_ID>
=item B<NP_CDP_TYPE_ADDRESSES>
=item B<NP_CDP_TYPE_PORT_ID>
=item B<NP_CDP_TYPE_CAPABILITIES>
=item B<NP_CDP_TYPE_SOFTWARE_VERSION>
=item B<NP_CDP_TYPE_PLATFORM>
=item B<NP_CDP_TYPE_VTP_MANAGEMENT_DOMAIN>
=item B<NP_CDP_TYPE_DUPLEX>
=item B<NP_CDP_TYPE_VOIP_VLAN_REPLY>
=item B<NP_CDP_TYPE_TRUST_BITMAP>
=item B<NP_CDP_TYPE_UNTRUSTED_PORT_COS>
=item B<NP_CDP_TYPE_SYSTEM_NAME>
=item B<NP_CDP_TYPE_SYSTEM_OBJECT_ID>
=item B<NP_CDP_TYPE_LOCATION>
Various supported CDP types.
=back
=head1 AUTHOR
Patrice E<lt>GomoRE<gt> Auffret
=head1 COPYRIGHT AND LICENSE
Copyright (c) 2004-2006, Patrice E<lt>GomoRE<gt> Auffret
You may distribute this module under the terms of the Artistic license.
See LICENSE.Artistic file in the source distribution archive.
=head1 RELATED MODULES
L<NetPacket>, L<Net::RawIP>, L<Net::RawSock>
=cut
syntax highlighted by Code2HTML, v. 0.9.1