// Copyright (C) 2007 // Fraunhofer Institut fuer offene Kommunikationssysteme (FOKUS) // Kompetenzzentrum fuer Satelitenkommunikation (SatCom) // Stefan Bund <g0dil@berlios.de> // // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation; either version 2 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the // Free Software Foundation, Inc., // 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. /** \file \brief Packet public header */ #ifndef HH_Packet_ #define HH_Packet_ 1 // Custom includes #include <boost/operators.hpp> #include "../Utils/Exception.hh" #include "../Utils/safe_bool.hh" #include "PacketInterpreter.hh" //#include "Packet.mpp" ///////////////////////////////hh.p//////////////////////////////////////// namespace senf { /** \defgroup packet_module Packet Handling The basic groundwork of the Packet library is the packet handling: \li The packet classes provide access to a chain of packet headers (more generically called interpreters). \li They automatically manage the required memory resources and the shared packet data. \section packet_module_chain The Interpreter Chain The central data structure for a packet is the interpreter chain \image html structure.png The Interpreter Chain This image depicts a packet with several headers. Each interpreter is responsible for a specific sub-range of the complete packet. This range always \e includes the packets payload (This is, why we call the data structure interpreter and not header: The interpreter is responsible for interpreting a range of the packet according to a specific protocol), the packet interpreters are nested inside each other. For each interpreter, this structure automatically divides the packet into three areas (each of which are optional): The header, the payload and the trailer. Every packet will have either a header or a payload section while most don't have a trailer. As user of the library you always interact with the chain through one (or more) of the interpreters. The interpreter provides methods to traverse to the following or preceding header (interpreter) and provides two levels of access to the packet data: Generic low-level access in the form of an STL compatible sequence and access to the parsed fields which are provided by the parser associated with the concrete packet type. \section packet_module_management Resource Management The interface to the packet library is provided using a handle class (\ref Packet for generic, protocol agnostic access and \ref ConcretePacket derived from \ref Packet to access a specific protocol). This handle automatically manages the resources associated with the packet (the interpreter chain and the data storage holding the packet data). The resources are automatically released when the last packet handle referencing a specific packet is destroyed. \implementation The packet chain is provided on two levels: The internal representation \ref PacketInterpreterBase and \ref PacketInterpreter which are referenced by the Handle classes \ref Packet and \ref ConcretePacket. \n The internal representation classes are pertinent in the sense, that they exist regardless of the existence of a handle referencing them (as long as the packet exists). Still the interpreter chain is lazy and packet interpreters beside the first are only created dynamically when accessed (this is implemented in the handle not in the internal representation). \n The packet interpreters make use of a pool allocator. This provides extremely efficient creation and destruction of packet interpreter's and removes the dynamic memory management overhead from the packet interpreter management. The packet implementation class (\ref PacketImpl which holds the packet data itself) however is still dynamically managed (however there is only a single instance for each packet). */ template <class PackeType> class ConcretePacket; ///\addtogroup packet_module ///@{ /** \brief Main Packet class Packet is the main externally visible class of the packet library. Packet is a handle into the internal packet representation. From Packet you may access the data of that specific sub-packet/header/interpreter and navigate to the neighboring sub-packets/headers/interpreters. Packet is protocol agnostic. This class only provides non-protocol dependent members. To access the protocol specific features of a packet (like header fields) the ConcretePacket class extending Packet is provided. \section packet_semantics Semantics All operations accessing the data of \c this packet in some way will ignore any preceding packets/headers/interpreters in the chain. It does not matter, whether a given packet is taken from the middle or the beginning of the chain, all operations (except those explicitly accessing the chain of course) should work the same. This especially includes members like clone() or append(): clone() will clone \e only from \c this packet until the end of the chain, append() will append the given packet \e ignoring any possibly preceding packets/headers/interpreters. In the same way, the data() member provides an STL-sequence compatible view of the packet data. This only includes the data which is part of \c this packet including header, trailer \e and payload but \e not the headers or trailers of packets \e before \c this packet in the packet/header/interpreter chain (nonetheless, this data overlaps with the data of other packets). Several members are member templates taking an \a OtherPacket template parameter. This parameter must be the ConcretePacket instantiation associated with some concrete packet type (protocol). For each implemented protocol, typedefs should be provided for these instantiations (Example: \ref EthernetPacket is a typedef for \ref ConcretePacket < \ref EthernetPacketType >). \see \ref ConcretePacket for the type specific interface\n \ref PacketData for the sequence interface\n \ref packetparser for a specification of the parser interface */ class Packet : public safe_bool<Packet>, public boost::equality_comparable<Packet> { public: /////////////////////////////////////////////////////////////////////////// // Types typedef void type; ///< Type of the packet. typedef senf::detail::packet::size_type size_type; ///< Unsigned type to represent packet size typedef PacketInterpreterBase::factory_t factory_t; ///< Packet factory type (see below) /// Special argument flag /** Used in some ConcretePacket constructors */ enum NoInit_t { noinit }; /////////////////////////////////////////////////////////////////////////// ///\name Structors and default members ///@{ // default copy constructor // default copy assignment // default destructor Packet(); ///< Create uninitialized packet handle /**< An uninitialized handle is not valid(). It does not allow any operation except assignment and checking for validity. */ Packet clone() const; ///< Create copy packet /**< clone() will create a complete copy the packet. The returned packet will have the same data and packet chain. It does however not share any data with the original packet. */ // conversion constructors template <class PacketType> Packet(ConcretePacket<PacketType> packet); ///< Copy-construct Packet from ConcretePacket /**< This constructor allows to convert an arbitrary ConcretePacket into a general Packet, loosing the protocol specific interface. */ ///@} /////////////////////////////////////////////////////////////////////////// ///\name Interpreter chain access ///@{ Packet next() const; ///< Get next packet in chain /**< \returns in - valid() packet, if no next packet exists */ template <class OtherPacket> OtherPacket next() const; ///< Get next packet of given type in chain /**< \throws InvalidPacketChainException if no such packet is found */ template <class OtherPacket> OtherPacket next(NoThrow_t) const; ///< Get next packet of given type in chain /**< \param[in] nothrow This argument always has the value \c senf::nothrow \returns in - valid() packet, if no such packet is found */ template <class OtherPacket> OtherPacket findNext() const; ///< Find next packet of given type in chain /**< findNext() is like next(), it will however return \c *this if it is of the given type. \throws InvalidPacketChainException if no such packet is found */ template <class OtherPacket> OtherPacket findNext(NoThrow_t) const; ///< Find next packet of given type in chain /**< findNext() is like next(), it will however return \c *this if it is of the given type. \param[in] nothrow This argument always has the value \c senf::nothrow \returns in - valid() packet, if no such packet is found */ Packet prev() const; ///< Get previous packet in chain /**< \returns in - valid() packet, if no previous packet exists */ template <class OtherPacket> OtherPacket prev() const; ///< Get previous packet of given type in chain /**< \throws InvalidPacketChainException if no such packet is found */ template <class OtherPacket> OtherPacket prev(NoThrow_t) const; ///< Get previous packet of given type in chain /**< \param[in] nothrow This argument always has the value \c senf::nothrow \returns in - valid() packet, if no such packet is found */ template <class OtherPacket> OtherPacket findPrev() const; ///< Find previous packet of given type in chain /**< findPrev() is like prev(), it will however return \c *this if it is of the type \throws InvalidPacketChainException if no such packet is found */ template <class OtherPacket> OtherPacket findPrev(NoThrow_t) const; ///< Find previous packet of given type in chain /**< findPrev() is like prev(), it will however return \c *this if it is of the type \param[in] nothrow This argument always has the value \c senf::nothrow \returns in - valid() packet, if no such packet is found */ Packet first() const; ///< Return first packet in chain template <class OtherPacket> OtherPacket first() const; ///< Return first packet of given type in chain /**< \throws InvalidPacketChainException if no such packet is found */ template <class OtherPacket> OtherPacket first(NoThrow_t) const; ///< Return first packet of given type in chain /**< \param[in] nothrow This argument always has the value \c senf::nothrow \returns in - valid() packet, if no such packet is found */ Packet last() const; ///< Return last packet in chain template <class OtherPacket> OtherPacket last() const; ///< Return last packet of given type in chain /**< \throws InvalidPacketChainException if no such packet is found */ template <class OtherPacket> OtherPacket last(NoThrow_t) const; ///< Return last packet of given type in chain /**< \param[in] nothrow This argument always has the value \c senf::nothrow \returns in - valid() packet, if no such packet is found */ template <class OtherPacket> OtherPacket parseNextAs() const; ///< Parse payload as given by \a OtherPacket and add packet /**< parseNextAs() will throw away the packet chain after the current packet if necessary. It will then parse the payload section of \c this packet as given by \a OtherPacket. The new packet is added to the chain after \c this. \returns new packet instance sharing the same data and placed after \c this packet in the chain. */ Packet parseNextAs(factory_t factory) const; ///< Parse payload as given by \a factory and add packet /**< parseNextAs() will throw away the packet chain after the current packet if necessary. It will then parse the payload section of \c this packet as given by \a OtherPacket. The new packet is added to the chain after \c this. \returns new packet instance sharing the same data and placed after \c this packet in the chain. */ template <class OtherPacket> bool is() const; ///< Check, whether \c this packet is of the given type template <class OtherPacket> OtherPacket as() const; ///< Cast current packet to the given type /**< This operations returns a handle to the same packet header/interpreter however cast to the given ConcretePacket type. <b>This conversion is unchecked</b>. If the packet really is of a different type, this will wreak havoc with the packet data-structures. You can validate whether the conversion is valid using is(). */ Packet append(Packet packet) const; ///< Append the given packet to \c this packet /**< This operation will replace the payload section of \c this packet with \a packet. This operation will replace the packet chain after \c this packet with a clone of \a packet and will replace the raw data of the payload of \c this with the raw data if \a packet. \returns Packet handle to the cloned \a packet, placed after \c this in the packet/header/interpreter chain. */ ///@} ///\name Data access ///@{ PacketData & data() const; ///< Access the packets raw data container size_type size() const; ///< Return size of packet in bytes /**< This size does \e not include the size of any preceding headers/packets/interpreters. It does however include \c this packets payload. */ ///@} ///\name Other methods ///@{ bool operator==(Packet other) const; ///< Check for packet identity /**< Two packet handles compare equal if they really are the same packet header in the same packet chain. */ bool boolean_test() const; ///< Check, whether the packet is valid() /**< \see valid() */ bool valid() const; ///< Check, whether the packet is valid() /**< An in - valid() packet does not allow any operation except checking for validity and assignment. in - valid() packets serve the same role as 0-pointers. This is an alias for boolean_test() which is called when using a packet in a boolean context. */ void finalize() const; ///< Update calculated fields /**< This call will update all calculated fields of the packet after it has been created or changed. This includes checksums, payload size fields or other fields, which can be set from other information in the packet. Each concrete packet type should document, which fields are set by finalize(). finalize() will automatically process all packets/headers/interpreters from the end of the chain backwards up to \c this. */ void dump(std::ostream & os) const; ///< Write out a printable packet representation /**< This method is provided mostly to help debugging packet problems. Each concrete packet should implement a dump method writing out all fields of the packet in a readable representation. dump() will call this member for each packet/header/interpreter in the chain from \c this packet up to the end of the chain. */ TypeIdValue typeId() const; ///< Get id of \c this packet /**< This value is used e.g. in the packet registry to associate packet types with other information. \returns A type holding the same information as a type_info object, albeit assignable */ factory_t factory() const; ///< Return factory instance of \c this packet /**< The returned factory instance can be used to create new packets of the given type without knowing the concrete type of the packet. The value may be stored away for later use if needed. */ ///@} protected: explicit Packet(PacketInterpreterBase::ptr packet); PacketInterpreterBase::ptr ptr() const; private: Packet checkNext() const; Packet checkLast() const; PacketInterpreterBase::ptr packet_; template <class PacketType> friend class ConcretePacket; friend class PacketParserBase; }; /** \brief Protocol specific packet handle The ConcretePacket template class extends Packet to provide protocol/packet type specific aspects. These are packet constructors and access to the parsed packet fields. The \c PacketType template argument to ConcretePacket is a protocol specific and internal policy class which defines the protocol specific behavior. To access a specific type of packet, the library provides corresponding typedefs of ConcretePacket < \a SomePacketType > (e.g. \ref EthernetPacket as typedef for \ref ConcretePacket < \ref EthernetPacketType >). The new members provided by ConcretePacket over packet are mostly comprised of the packet constructors. These come in three major flavors: \li The create() family of constructors will create completely new packets. \li The createAfter() family of constructors will create new packets (with new data for the packet) \e after a given existing packet. \li The createBefore() family of constructors will create new packets (again with new data) \e before a given existing packet. Whereas create() will create a completely new packet with it's own chain and data storage, createAfter() and createBefore() extend a packet with additional headers/interpreters. createAfter() will set the payload of the given packet to the new packet whereas createBefore() will create a new packet with the existing packet as it's payload. createAfter() differs from Packet::parseNextAs() in that the former creates a new packet \e replacing any possibly existing data whereas the latter will interpret the already \e existing data as given by the type argument. \see \ref PacketTypeBase for a specification of the interface to be provided by the \a PacketType policy class. */ template <class PacketType> class ConcretePacket : public Packet { public: /////////////////////////////////////////////////////////////////////////// // Types typedef PacketType type; /////////////////////////////////////////////////////////////////////////// ///\name Structors and default members ///@{ // default copy constructor // default copy assignment // default destructor // no conversion constructors ConcretePacket(); ///< Create uninitialized packet handle /**< An uninitialized handle is not valid(). It does not allow any operation except assignment and checking for validity. */ static factory_t factory(); ///< Return factory for packets of specific type /**< This \e static member is like Packet::factory() for a specific packet of type \a PacketType */ // Create completely new packet static ConcretePacket create(); ///< Create default initialized packet /**< The packet will be initialized to it's default empty state. */ static ConcretePacket create(NoInit_t); ///< Create uninitialized empty packet /**< This will create a completely empty and uninitialized packet with <tt>size() == 0</tt>. \param[in] noinit This parameter must always have the value \c senf::noinit. */ static ConcretePacket create(size_type size); ///< Create default initialized packet /**< This member will create a default initialized packet with the given size. If the size parameter is smaller than the minimum allowed packet size an exception will be thrown. \param[in] size Size of the packet to create in bytes. \throws TruncatedPacketException if \a size is smaller than the smallest permissible size for this type of packet. */ static ConcretePacket create(size_type size, NoInit_t); ///< Create uninitialized packet /**< Creates an uninitialized (all-zero) packet of the exact given size. \param[in] size Size of the packet to create in bytes \param[in] noinit This parameter must always have the value \c senf::noinit. */ template <class ForwardReadableRange> static ConcretePacket create(ForwardReadableRange const & range); ///< Create packet from given data /**< The packet will be created from a copy of the given data. The data from the range will be copied directly into the packet representation. The data will \e not be validated in any way. \param[in] range <a href="http://www.boost.org/libs/range/index.html">Boost.Range</a> of data to construct packet from. */ // Create packet as new packet after a given packet static ConcretePacket createAfter(Packet packet); ///< Create default initialized packet after \a packet /**< The packet will be initialized to it's default empty state. It will be appended as next header/interpreter after \a packet in that packets interpreter chain. \param[in] packet Packet to append new packet to. */ static ConcretePacket createAfter(Packet packet, NoInit_t); ///< Create uninitialized empty packet after\a packet /**< This will create a completely empty and uninitialized packet with <tt>size() == 0</tt>. It will be appended as next header/interpreter after \a packet in that packets interpreter chain. \param[in] packet Packet to append new packet to. \param[in] noinit This parameter must always have the value \c senf::noinit. */ static ConcretePacket createAfter(Packet packet, size_type size); ///< Create default initialized packet after \a packet /**< This member will create a default initialized packet with the given size. If the size parameter is smaller than the minimum allowed packet size an exception will be thrown. It will be appended as next header/interpreter after \a packet in that packets interpreter chain. \param[in] packet Packet to append new packet to. \param[in] size Size of the packet to create in bytes. \throws TruncatedPacketException if \a size is smaller than the smallest permissible size for this type of packet. */ static ConcretePacket createAfter(Packet packet, size_type size, NoInit_t); ///< Create uninitialized packet after \a packet /**< Creates an uninitialized (all-zero) packet of the exact given size. It will be appended as next header/interpreter after \a packet in that packets interpreter chain. \param[in] packet Packet to append new packet to. \param[in] size Size of the packet to create in bytes \param[in] noinit This parameter must always have the value \c senf::noinit. */ template <class ForwardReadableRange> static ConcretePacket createAfter(Packet packet, ForwardReadableRange const & range); ///< Create packet from given data after \a packet /**< The packet will be created from a copy of the given data. The data from the range will be copied directly into the packet representation. The data will \e not be validated in any way. It will be appended as next header/interpreter after \a packet in that packets interpreter chain. \param[in] packet Packet to append new packet to. \param[in] range <a href="http://www.boost.org/libs/range/index.html">Boost.Range</a> of data to construct packet from. */ // Create packet as new packet (header) before a given packet static ConcretePacket createBefore(Packet packet); ///< Create default initialized packet before \a packet /**< The packet will be initialized to it's default empty state. It will be prepended as previous header/interpreter before \a packet in that packets interpreter chain. \param[in] packet Packet to prepend new packet to. */ static ConcretePacket createBefore(Packet packet, NoInit_t); ///< Create uninitialized empty packet before \a packet /**< Creates a completely empty and uninitialized packet. It will be prepended as previous header/interpreter before \a packet in that packets interpreter chain. \param[in] packet Packet to prepend new packet to. */ // Create a clone of the current packet ConcretePacket clone() const; ///@} /////////////////////////////////////////////////////////////////////////// // Field access typename type::parser * operator->() const; ///< Access packet fields /**< This operator allows to access the parsed fields of the packet using the notation <tt>packet->field()</tt>. The fields of the packet are specified by the PacketType's \c parser member. The members are not strictly restricted to simple field access. The parser class may have any member which is needed for full packet access (e.g. checksum validation / recreation ...) \see \ref packetparser for the parser interface. */ protected: private: typedef PacketInterpreter<PacketType> interpreter; ConcretePacket(typename interpreter::ptr packet_); typename interpreter::ptr ptr() const; friend class Packet; friend class PacketInterpreter<PacketType>; }; ///@} } ///////////////////////////////hh.e//////////////////////////////////////// #endif #if !defined(HH_Packets__decls_) && !defined(HH_Packet_i_) #define HH_Packet_i_ #include "Packet.cci" #include "Packet.ct" #include "Packet.cti" #endif // Local Variables: // mode: c++ // fill-column: 100 // c-file-style: "senf" // indent-tabs-mode: nil // ispell-local-dictionary: "american" // compile-command: "scons -u test" // comment-column: 40 // End: