/* * 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. * */ #include #if !defined(TINS_DOT11_DOT11_CONTROL_H) && defined(TINS_HAVE_DOT11) #define TINS_DOT11_DOT11_CONTROL_H #include #include namespace Tins { /** * \brief Represents an IEEE 802.11 control frame. */ class TINS_API Dot11Control : public Dot11 { public: /** * \brief This PDU's flag. */ static const PDU::PDUType pdu_flag = PDU::DOT11_CONTROL; /** * \brief Constructor for creating a 802.11 control frame PDU * * Constructs a 802.11 Control PDU taking the destination and * source hardware addresses. * * \param dst_addr The destination hardware address. */ Dot11Control(const address_type& dst_addr = address_type()); /** * \brief Constructs a Dot11Control object from a buffer and * adds all identifiable PDUs found in the buffer as children * of this one. * * If the next PDU is not recognized, then a RawPDU is used. * * If there is not enough size for the header in the buffer * or the input data is malformed, a malformed_packet exception * is thrown. * * \param buffer The buffer from which this PDU will be constructed. * \param total_sz The total size of the buffer. */ Dot11Control(const uint8_t* buffer, uint32_t total_sz); /** * \brief Getter for the PDU's type. * \sa PDU::pdu_type */ PDUType pdu_type() const { return pdu_flag; } /** * \brief Check whether this PDU matches the specified flag. * \param flag The flag to match * \sa PDU::matches_flag */ bool matches_flag(PDUType flag) const { return flag == pdu_flag || Dot11::matches_flag(flag); } }; /** * \brief Class that represents an abstraction of the 802.11 control frames * that contain a target address. */ class TINS_API Dot11ControlTA : public Dot11Control { public: /** * \brief This PDU's flag. */ static const PDU::PDUType pdu_flag = PDU::DOT11_CONTROL_TA; /** * \brief Getter for the target address field. */ address_type target_addr() const { return taddr_; } /** * \brief Setter for the target address field. * \param addr The new target address. */ void target_addr(const address_type& addr); /** * \brief Check whether this PDU matches the specified flag. * \param flag The flag to match * \sa PDU::matches_flag */ bool matches_flag(PDUType flag) const { return flag == pdu_flag || Dot11::matches_flag(flag); } protected: /** * \brief Constructor for creating a 802.11 control frame TA PDU * * Constructs a 802.11 PDU taking the destination and source * hardware addresses. * * \param dst_addr The destination hardware address. * \param target_addr The source hardware address. */ Dot11ControlTA(const address_type& dst_addr = address_type(), const address_type& target_addr = address_type()); /** * \brief Constructs a Dot11ControlTA object from a buffer and * adds all identifiable PDUs found in the buffer as children * of this one. * * If the next PDU is not recognized, then a RawPDU is used. * * If there is not enough size for the header in the buffer * or the input data is malformed, a malformed_packet exception * is thrown. * * \param buffer The buffer from which this PDU will be constructed. * \param total_sz The total size of the buffer. */ Dot11ControlTA(const uint8_t* buffer, uint32_t total_sz); /** * \brief Returns the 802.11 frame's header length. * * \return An uint32_t with the header's size. * \sa PDU::header_size() */ uint32_t header_size() const; protected: /** * \brief Getter for the control ta additional fields size. */ uint32_t controlta_size() const { return static_cast(taddr_.size() + sizeof(dot11_header)); } void write_ext_header(Memory::OutputMemoryStream& stream); private: address_type taddr_; }; /** * \brief IEEE 802.11 RTS frame. */ class TINS_API Dot11RTS : public Dot11ControlTA { public: /** * \brief This PDU's flag. */ static const PDU::PDUType pdu_flag = PDU::DOT11_RTS; /** * \brief Constructor for creating a 802.11 RTS frame PDU * * Constructs a 802.11 RTS PDU taking the destination and source * hardware addresses. * * \param dst_addr The destination hardware address. * \param target_addr The source hardware address. */ Dot11RTS(const address_type& dst_addr = address_type(), const address_type& target_addr = address_type()); /** * \brief Constructs a Dot11RTS object from a buffer and adds all * identifiable PDUs found in the buffer as children of this one. * * If the next PDU is not recognized, then a RawPDU is used. * * If there is not enough size for the header in the buffer * or the input data is malformed, a malformed_packet exception * is thrown. * * \param buffer The buffer from which this PDU will be constructed. * \param total_sz The total size of the buffer. */ Dot11RTS(const uint8_t* buffer, uint32_t total_sz); /** * \brief Clones this PDU. * * \sa PDU::clone */ Dot11RTS* clone() const { return new Dot11RTS(*this); } /** * \brief Getter for the PDU's type. * \sa PDU::pdu_type */ PDUType pdu_type() const { return pdu_flag; } /** * \brief Check whether this PDU matches the specified flag. * \param flag The flag to match * \sa PDU::matches_flag */ bool matches_flag(PDUType flag) const { return flag == pdu_flag || Dot11Control::matches_flag(flag); } }; class TINS_API Dot11PSPoll : public Dot11ControlTA { public: /** * \brief This PDU's flag. */ static const PDU::PDUType pdu_flag = PDU::DOT11_PS_POLL; /** * \brief Constructor for creating a 802.11 PS-Poll frame PDU * * Constructs a 802.11 PDU taking the destination and source * hardware addresses. * * \param dst_addr The destination hardware address. * \param target_addr The source hardware address. */ Dot11PSPoll(const address_type& dst_addr = address_type(), const address_type& target_addr = address_type()); /** * \brief Constructs a Dot11PSPoll object from a buffer and * adds all identifiable PDUs found in the buffer as children of * this one. * * If the next PDU is not recognized, then a RawPDU is used. * * If there is not enough size for the header in the buffer * or the input data is malformed, a malformed_packet exception * is thrown. * * \param buffer The buffer from which this PDU will be constructed. * \param total_sz The total size of the buffer. */ Dot11PSPoll(const uint8_t* buffer, uint32_t total_sz); /** * \brief Clones this PDU. * * \sa PDU::clone */ Dot11PSPoll* clone() const { return new Dot11PSPoll(*this); } /** * \brief Getter for the PDU's type. * \sa PDU::pdu_type */ PDUType pdu_type() const { return pdu_flag; } /** * \brief Check whether this PDU matches the specified flag. * \param flag The flag to match * \sa PDU::matches_flag */ bool matches_flag(PDUType flag) const { return flag == pdu_flag || Dot11Control::matches_flag(flag); } }; class TINS_API Dot11CFEnd : public Dot11ControlTA { public: /** * \brief This PDU's flag. */ static const PDU::PDUType pdu_flag = PDU::DOT11_CF_END; /** * \brief Constructor for creating a 802.11 CF-End frame PDU * * Constructs a 802.11 PDU taking the destination and source * hardware addresses. * * \param dst_addr The destination hardware address. * \param target_addr The source hardware address. */ Dot11CFEnd(const address_type& dst_addr = address_type(), const address_type& target_addr = address_type()); /** * \brief Constructs a Dot11CFEnd object from a buffer and adds * all identifiable PDUs found in the buffer as children of this * one. * * If the next PDU is not recognized, then a RawPDU is used. * * If there is not enough size for the header in the buffer * or the input data is malformed, a malformed_packet exception * is thrown. * * \param buffer The buffer from which this PDU will be constructed. * \param total_sz The total size of the buffer. */ Dot11CFEnd(const uint8_t* buffer, uint32_t total_sz); /** * \brief Clones this PDU. * * \sa PDU::clone */ Dot11CFEnd* clone() const { return new Dot11CFEnd(*this); } /** * \brief Getter for the PDU's type. * \sa PDU::pdu_type */ PDUType pdu_type() const { return pdu_flag; } /** * \brief Check whether this PDU matches the specified flag. * \param flag The flag to match * \sa PDU::matches_flag */ bool matches_flag(PDUType flag) const { return flag == pdu_flag || Dot11Control::matches_flag(flag); } }; class TINS_API Dot11EndCFAck : public Dot11ControlTA { public: /** * \brief This PDU's flag. */ static const PDU::PDUType pdu_flag = PDU::DOT11_END_CF_ACK; /** * \brief Constructor for creating a 802.11 End-CF-Ack frame PDU * * Constructs a 802.11 PDU taking the destination and source * hardware addresses. * * \param dst_addr The destination hardware address. * \param target_addr The source hardware address. */ Dot11EndCFAck(const address_type& dst_addr = address_type(), const address_type& target_addr = address_type()); /** * \brief Constructs a Dot11EndCFAck frame object from a buffer * and adds all identifiable PDUs found in the buffer as children * of this one. * * If the next PDU is not recognized, then a RawPDU is used. * * If there is not enough size for the header in the buffer * or the input data is malformed, a malformed_packet exception * is thrown. * * \param buffer The buffer from which this PDU will be constructed. * \param total_sz The total size of the buffer. */ Dot11EndCFAck(const uint8_t* buffer, uint32_t total_sz); /** * \brief Clones this PDU. * * \sa PDU::clone */ Dot11EndCFAck* clone() const { return new Dot11EndCFAck(*this); } /** * \brief Getter for the PDU's type. * \sa PDU::pdu_type */ PDUType pdu_type() const { return pdu_flag; } /** * \brief Check whether this PDU matches the specified flag. * \param flag The flag to match * \sa PDU::matches_flag */ bool matches_flag(PDUType flag) const { return flag == pdu_flag || Dot11Control::matches_flag(flag); } }; class TINS_API Dot11Ack : public Dot11Control { public: /** * \brief This PDU's flag. */ static const PDU::PDUType pdu_flag = PDU::DOT11_ACK; /** * \brief Constructor for creating a 802.11 Ack frame PDU * * Constructs a 802.11 PDU taking the destination and source * hardware addresses. * * \param dst_addr The destination hardware address. */ Dot11Ack(const address_type& dst_addr = address_type()); /** * \brief Constructs a Dot11Ack frame object from a buffer and * adds all identifiable PDUs found in the buffer as children of * this one. * * If the next PDU is not recognized, then a RawPDU is used. * * If there is not enough size for the header in the buffer * or the input data is malformed, a malformed_packet exception * is thrown. * * \param buffer The buffer from which this PDU will be constructed. * \param total_sz The total size of the buffer. */ Dot11Ack(const uint8_t* buffer, uint32_t total_sz); /** * \brief Clones this PDU. * * \sa PDU::clone */ Dot11Ack* clone() const { return new Dot11Ack(*this); } /** * \brief Getter for the PDU's type. * \sa PDU::pdu_type */ PDUType pdu_type() const { return pdu_flag; } /** * \brief Check whether this PDU matches the specified flag. * \param flag The flag to match * \sa PDU::matches_flag */ bool matches_flag(PDUType flag) const { return flag == pdu_flag || Dot11Control::matches_flag(flag); } }; /** * \brief Class that represents an 802.11 Block Ack Request PDU. */ class TINS_API Dot11BlockAckRequest : public Dot11ControlTA { public: /** * \brief This PDU's flag. */ static const PDU::PDUType pdu_flag = PDU::DOT11_BLOCK_ACK_REQ; /** * \brief Constructor for creating a 802.11 Block Ack request frame PDU * * Constructs a 802.11 PDU taking the destination and source * hardware addresses. * * \param dst_addr The destination hardware address. * \param target_addr The source hardware address. */ Dot11BlockAckRequest(const address_type& dst_addr = address_type(), const address_type& target_addr = address_type()); /** * \brief Constructs a Dot11BlockAckRequest object from a buffer * and adds all identifiable PDUs found in the buffer as children * of this one. * * If the next PDU is not recognized, then a RawPDU is used. * * If there is not enough size for the header in the buffer * or the input data is malformed, a malformed_packet exception * is thrown. * * \param buffer The buffer from which this PDU will be constructed. * \param total_sz The total size of the buffer. */ Dot11BlockAckRequest(const uint8_t* buffer, uint32_t total_sz); /* Getter */ /** * \brief Getter for the bar control field. * \return The stored bar control field. */ small_uint<4> bar_control() const { #if TINS_IS_LITTLE_ENDIAN return bar_control_ & 0xf; #else return (bar_control_ >> 8) & 0xf; #endif } /** * \brief Getter for the start sequence field. * \return The stored start sequence. */ small_uint<12> start_sequence() const { #if TINS_IS_LITTLE_ENDIAN return (start_sequence_ >> 4) & 0xfff; #else return (Endian::le_to_host(start_sequence_) >> 4) & 0xfff; #endif } /** * \brief Getter for the fragment number field. * \return The stored fragment number field. */ small_uint<4> fragment_number() const { #if TINS_IS_LITTLE_ENDIAN return start_sequence_ & 0xf; #else return (start_sequence_ >> 8) & 0xf; #endif } /** * \brief Returns the 802.11 frame's header length. * * \return The header's size. * \sa PDU::header_size() */ uint32_t header_size() const; /* Setter */ /** * \brief Setter for the bar control field. * \param bar The bar control field to be set. */ void bar_control(small_uint<4> bar); /** * \brief Setter for the start sequence field. * \param bar The start sequence field to be set. */ void start_sequence(small_uint<12> seq); /** * \brief Setter for the fragment number field. * \param frag The fragment number field to be set. */ void fragment_number(small_uint<4> frag); /** * \brief Clones this PDU. * * \sa PDU::clone */ Dot11BlockAckRequest* clone() const { return new Dot11BlockAckRequest(*this); } /** * \brief Getter for the PDU's type. * \sa PDU::pdu_type */ PDUType pdu_type() const { return pdu_flag; } /** * \brief Check whether this PDU matches the specified flag. * \param flag The flag to match * \sa PDU::matches_flag */ bool matches_flag(PDUType flag) const { return flag == pdu_flag || Dot11Control::matches_flag(flag); } protected: void write_ext_header(Memory::OutputMemoryStream& stream); private: uint16_t bar_control_; uint16_t start_sequence_; }; /** * \brief Class that represents an 802.11 block ack frame. */ class TINS_API Dot11BlockAck : public Dot11ControlTA { public: /** * \brief This PDU's flag. */ static const PDU::PDUType pdu_flag = PDU::DOT11_BLOCK_ACK; /** * The size of the bitmap field. */ static const size_t bitmap_size = 8; /** * \brief Constructor for creating a 802.11 Block Ack frame PDU * * Constructs a 802.11 PDU taking the destination and source * hardware addresses. * * \param dst_addr The destination hardware address. * \param target_addr The source hardware address. */ Dot11BlockAck(const address_type& dst_addr = address_type(), const address_type& target_addr = address_type()); /** * \brief Constructs a Dot11BlockAck frame object from a buffer * and adds all identifiable PDUs found in the buffer as children * of this one. * * If the next PDU is not recognized, then a RawPDU is used. * * If there is not enough size for the header in the buffer * or the input data is malformed, a malformed_packet exception * is thrown. * * \param buffer The buffer from which this PDU will be constructed. * \param total_sz The total size of the buffer. */ Dot11BlockAck(const uint8_t* buffer, uint32_t total_sz); /* Getters */ /** * \brief Getter for the bar control field. * \return The stored bar control field. */ small_uint<4> bar_control() const { #if TINS_IS_LITTLE_ENDIAN return bar_control_ & 0xf; #else return (bar_control_ >> 8) & 0xf; #endif } /** * \brief Getter for the start sequence field. * \return The stored start sequence. */ small_uint<12> start_sequence() const { #if TINS_IS_LITTLE_ENDIAN return (start_sequence_ >> 4) & 0xfff; #else return (Endian::le_to_host(start_sequence_) >> 4) & 0xfff; #endif } /** * \brief Getter for the fragment number field. * \return The stored fragment number field. */ small_uint<4> fragment_number() const { #if TINS_IS_LITTLE_ENDIAN return start_sequence_ & 0xf; #else return (start_sequence_ >> 8) & 0xf; #endif } /** * \brief Returns the 802.11 frame's header length. * * \return An uint32_t with the header's size. * \sa PDU::header_size() */ uint32_t header_size() const; /* Setters */ /** * \brief Setter for the bar control field. * \param bar The bar control field to be set. */ void bar_control(small_uint<4> bar); /** * \brief Setter for the start sequence field. * \param bar The start sequence field to be set. */ void start_sequence(small_uint<12> seq); /** * \brief Setter for the fragment number field. * \param frag The fragment number field to be set. */ void fragment_number(small_uint<4> frag); /** * \brief Getter for the bitmap field. * * The returned pointer must not be free'd. * * \return The bitmap field. */ const uint8_t* bitmap() const { return bitmap_; } /** * \brief Setter for the bitmap field. * \param bit The new bitmap field to be set. */ void bitmap(const uint8_t* bit); /** * \brief Getter for the PDU's type. * \sa PDU::pdu_type */ PDUType pdu_type() const { return pdu_flag; } /** * \brief Check whether this PDU matches the specified flag. * \param flag The flag to match * \sa PDU::matches_flag */ bool matches_flag(PDUType flag) const { return flag == pdu_flag || Dot11Control::matches_flag(flag); } /** * \brief Clones this PDU. * * \sa PDU::clone */ Dot11BlockAck* clone() const { return new Dot11BlockAck(*this); } private: void write_ext_header(Memory::OutputMemoryStream& stream); uint16_t bar_control_, start_sequence_; uint8_t bitmap_[bitmap_size]; }; } // namespace Tins #endif // TINS_DOT11_DOT11_CONTROL_H