Cheeze/lib/libtins/include/tins/packet_writer.h
2024-02-21 14:52:47 +03:00

230 lines
6.9 KiB
C++

/*
* Copyright (c) 2017, Matias Fontanini
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
#ifndef TINS_PACKET_WRITER_H
#define TINS_PACKET_WRITER_H
#include <string>
#include <tins/macros.h>
#include <tins/cxxstd.h>
#include <tins/utils/pdu_utils.h>
#ifdef TINS_HAVE_PCAP
#include <pcap.h>
#include <tins/data_link_type.h>
struct timeval;
namespace Tins {
class PDU;
class Packet;
/**
* \class PacketWriter
* \brief Writes PDUs to a pcap format file.
*
* This class can be used to write packets into a <i>pcap</i> format
* file. It supports both writing packets one by one, or writing all
* packets in a range (provided by iterators), so you can use it
* to dump all packets in a vector.
*
* Since you might use both PDU objects and pointers to them,
* both the PacketWriter::write overload that takes a single object
* or the one that takes an iterator range accept a PDU reference type
* as well as any type that can be dereferenced until a PDU type is found.
* This means you can use both raw and smart pointers.
*
* For example:
*
* \code
* // Differents types holding PDUs
* EthernetII object;
* std::shared_ptr<PDU> smart_ptr = ...;
* std::vector<std::shared_ptr<PDU>> vt = ....;
*
* // The writer we'll use
* PacketWriter writer("/tmp/file.pcap", DataLinkType<EthernetII>());
*
* // Now write all of them
* writer.write(object);
* writer.write(smart_ptr);
* writer.write(vt.begin(), vt.end());
* \endcode
*/
class TINS_API PacketWriter {
public:
/**
* \brief The type of PDUs that will be written to this file (deprecated).
* \deprecated Use DataLinkType instead of this enum.
*
* This flag should match the type of the lowest layer PDU to be
* written.
*/
enum LinkType {
RADIOTAP = DLT_IEEE802_11_RADIO,
DOT11 = DLT_IEEE802_11,
ETH2 = DLT_EN10MB,
DOT3 = DLT_EN10MB,
SLL = DLT_LINUX_SLL
};
/**
* \brief Constructs a PacketWriter.
*
* This method takes a DataLinkType, which indicates the link
* layer protocol that will be used on the packets to write.
*
* For example, you can write packets that contain an
* EthernetII link layer type by doing:
*
* \code
* // Construct a PacketWriter
* PacketWriter writer("/tmp/test.pcap", DataLinkType<EthernetII>());
* // Write some packet
* writer.write(packet);
* \endcode
*
* \param file_name The file in which to store the written PDUs.
* \param lt A DataLinkType that represents the link layer
* protocol to use.
* \sa PcapIdentifier.
*/
template<typename T>
PacketWriter(const std::string& file_name, const DataLinkType<T>& lt) {
init(file_name, lt.get_type());
}
/**
* \brief Constructs a PacketWriter.
*
* \deprecated Use the PacketWriter(const std::string&, const DataLinkType<T>&)
* constructor.
*
* \param file_name The file in which to store the written PDUs.
* \param lt The link type which will be written to this file.
* \sa LinkType.
*/
PacketWriter(const std::string& file_name, LinkType lt);
#if TINS_IS_CXX11
/**
* \brief Move constructor.
*
* Note that calling PacketWriter::write on an previously moved
* object will lead to undefined behaviour.
*
* \param rhs The PacketWriter to be moved.
*/
PacketWriter(PacketWriter &&rhs) TINS_NOEXCEPT {
*this = std::move(rhs);
}
/**
* \brief Move assignment operator.
*
* Note that calling PacketWriter::write on an previously moved
* object will lead to undefined behaviour.
*
* \param rhs The PacketWriter to be moved.
*/
PacketWriter& operator=(PacketWriter &&rhs) TINS_NOEXCEPT {
handle_ = 0;
dumper_ = 0;
std::swap(handle_, rhs.handle_);
std::swap(dumper_, rhs.dumper_);
return* this;
}
#endif
/**
* \brief Destructor.
*
* Gracefully closes the output file.
*/
~PacketWriter();
/**
* \brief Writes a PDU to this file.
* \param pdu The PDU to be written.
*/
void write(PDU& pdu);
/**
* \brief Writes a Packet to this file.
*
* The timestamp used on the entry for this packet will be the Timestamp
* object associated with this packet.
*
* \param packet The packet to be written.
*/
void write(Packet& packet);
/**
* \brief Writes a PDU to this file.
*
* The template parameter T must at some point yield a PDU& after
* applying operator* one or more than one time. This accepts both
* raw and smart pointers.
*/
template<typename T>
void write(T& pdu) {
write(Utils::dereference_until_pdu(pdu));
}
/**
* \brief Writes all the PDUs in the range [start, end)
* \param start A forward iterator pointing to the first PDU
* to be written.
* \param end A forward iterator pointing to one past the last
* PDU in the range.
*/
template<typename ForwardIterator>
void write(ForwardIterator start, ForwardIterator end) {
while (start != end) {
write(Utils::dereference_until_pdu(*start++));
}
}
private:
// You shall not copy
PacketWriter(const PacketWriter&);
PacketWriter& operator=(const PacketWriter&);
void init(const std::string& file_name, int link_type);
void write(PDU& pdu, const struct timeval& tv);
pcap_t* handle_;
pcap_dumper_t* dumper_;
};
} // Tins
#endif // TINS_HAVE_PCAP
#endif // TINS_PACKET_WRITER_H