mirror of
https://github.com/zeek/zeek.git
synced 2025-10-02 06:38:20 +00:00

Given IP-aware ConnKeys, ConnTuples aren't really required any more. ConnTuple had two benefits: - It preserved the original src/dst orientation from the packet headers it was based on, which IPBasedConnKey now tracks and provides accessor methods for. - In IPBasedAnalyzer::AnalyzePacket() its instance survived past the std:move() of the key into NewConn(), which we sidestep by keeping the original src address and port around until we need after the connection is obtained.
214 lines
8 KiB
C++
214 lines
8 KiB
C++
// See the file "COPYING" in the main distribution directory for copyright.
|
|
|
|
#pragma once
|
|
|
|
#include <map>
|
|
#include <set>
|
|
|
|
#include "zeek/Tag.h"
|
|
#include "zeek/packet_analysis/Analyzer.h"
|
|
#include "zeek/packet_analysis/protocol/ip/conn_key/IPBasedConnKey.h"
|
|
|
|
namespace zeek::analyzer::pia {
|
|
class PIA;
|
|
}
|
|
|
|
namespace zeek::packet_analysis::IP {
|
|
|
|
class SessionAdapter;
|
|
|
|
/**
|
|
* A base class for reuse by packet analyzers based on IP. This is used by default
|
|
* by the TCP, UDP, and ICMP analyzers to reduce a large amount of duplicated code
|
|
* that those plugins have in common.
|
|
*/
|
|
class IPBasedAnalyzer : public Analyzer {
|
|
public:
|
|
~IPBasedAnalyzer() override;
|
|
|
|
bool AnalyzePacket(size_t len, const uint8_t* data, Packet* packet) override;
|
|
|
|
/**
|
|
* Returns true if the analyzer determines that in fact a new
|
|
* connection has started without the connection statement having
|
|
* terminated the previous one, i.e., the new data is arriving at
|
|
* what's the analyzer for the previous instance. This is used only
|
|
* for TCP.
|
|
*/
|
|
virtual bool IsReuse(double t, const u_char* pkt) { return false; }
|
|
|
|
/**
|
|
* Registers a well-known port for an analyzer. Once registered,
|
|
* connection on that port will start with a corresponding analyzer
|
|
* assigned.
|
|
*
|
|
* @param tag The analyzer's tag.
|
|
* @param port The port's number.
|
|
* @return True if successful.
|
|
*/
|
|
bool RegisterAnalyzerForPort(const zeek::Tag& tag, uint32_t port);
|
|
|
|
/**
|
|
* Unregisters a well-known port for an analyzer.
|
|
*
|
|
* @param tag The analyzer's tag.
|
|
* @param port The port's number.
|
|
* @param tag The analyzer's tag as an enum of script type \c
|
|
* Tag.
|
|
*/
|
|
bool UnregisterAnalyzerForPort(const zeek::Tag& tag, uint32_t port);
|
|
|
|
/**
|
|
* Dumps information about the registered session analyzers per port.
|
|
* Used by analyzer::Manager.
|
|
*/
|
|
void DumpPortDebug();
|
|
|
|
/**
|
|
* Updates the internal pointer to the script-level variable `ignore_checksums_nets`.
|
|
* This is used to prevent repeated (costly) lookup of the script-level variable
|
|
* by IP-based analyzers.
|
|
*
|
|
* @param t New value of ignore_checksums_nets
|
|
*/
|
|
static void SetIgnoreChecksumsNets(TableValPtr t);
|
|
|
|
/**
|
|
* Gets the internal pointer to the script-level variable `ignore_checksums_nets`.
|
|
* This is used to prevent repeated (costly) lookup of the script-level variable
|
|
* by IP-based analyzers.
|
|
*
|
|
* @return Current value of `ignore_checksums_nets`.
|
|
*/
|
|
static TableValPtr GetIgnoreChecksumsNets();
|
|
|
|
protected:
|
|
/**
|
|
* Construct a new IP-based analyzer.
|
|
*
|
|
* @param name The name for the type of analyzer. The name must match
|
|
* the one the corresponding Component registers.
|
|
* @param proto The transport protocol implemented by this analyzer.
|
|
* @param mask The mask used to determine if a port is a server port
|
|
* for this protocol. This is used by IsLikelyServerPort().
|
|
* @param report_unknown_protocols Flag for whether to report unknown
|
|
* protocols during packet forwarding. This is typically false for IP
|
|
* protocols since packets may go into the session analysis framework
|
|
* as well.
|
|
*/
|
|
IPBasedAnalyzer(const char* name, TransportProto proto, uint32_t mask, bool report_unknown_protocols);
|
|
|
|
/**
|
|
* Initialize the given ConnKey from the packet header & data.
|
|
*
|
|
* @param len Remaining length of data.
|
|
* @param data Remaining packet data.
|
|
* @param packet The packet being processed.
|
|
* @param key The ConnKey instance to initialize.
|
|
*
|
|
* @return True if initialization succeeded, false otherwise (e.g. because
|
|
* there wasn't enough data available).
|
|
*/
|
|
virtual bool InitConnKey(size_t len, const uint8_t* data, Packet* packet, IPBasedConnKey& key) {
|
|
// Given deprecation of BuildConnTuple below, make this pure virtual in 8.1.
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Parse the header from the packet into a ConnTuple object.
|
|
*/
|
|
[[deprecated("Remove in v8.1. Switch to InitConnKey() and key-only initialization.")]]
|
|
virtual bool BuildConnTuple(size_t len, const uint8_t* data, Packet* packet, ConnTuple& tuple) {
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Continues process of packet after the connection has been inserted into the
|
|
* session manager. This should be implemented by all child classes.
|
|
*
|
|
* @param conn The connection currently being processed.
|
|
* @param t The timestamp for the current packet.
|
|
* @param is_orig Flag denoting whether this packet is from the originator of
|
|
* the connection.
|
|
* @param remaining The remaining about of data in the packet.
|
|
* @param pkt The packet being processed.
|
|
*/
|
|
virtual void DeliverPacket(Connection* conn, double t, bool is_orig, int remaining, Packet* pkt) {}
|
|
|
|
/**
|
|
* Upon seeing the first packet of a connection, checks whether we want
|
|
* to analyze it (e.g. we may not want to look at partial connections)
|
|
* and, if yes, whether we should flip the roles of originator and
|
|
* responder based on known ports and such.
|
|
*
|
|
* @param src_port The source port of the connection.
|
|
* @param dst_port The destination port of the connection.
|
|
* @param data The payload data for the packet being processed.
|
|
* @param flip_roles Return value if the roles should be flipped.
|
|
* @return True if the connection is wanted. False otherwise.
|
|
*/
|
|
virtual bool WantConnection(uint16_t src_port, uint16_t dst_port, const u_char* data, bool& flip_roles) const {
|
|
flip_roles = false;
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Returns an analyzer adapter appropriate for this IP-based analyzer. This adapter
|
|
* is used to hook into the session analyzer framework. This function can also be used
|
|
* to do any extra initialization of connection timers, etc.
|
|
*/
|
|
virtual SessionAdapter* MakeSessionAdapter(Connection* conn) = 0;
|
|
|
|
/**
|
|
* Returns a PIA appropriate for this IP-based analyzer. This method is optional to
|
|
* override in child classes, as not all analyzers need a PIA.
|
|
*/
|
|
virtual analyzer::pia::PIA* MakePIA(Connection* conn) { return nullptr; }
|
|
|
|
/**
|
|
* Verifies that there is enough data in the packet to process the header
|
|
* length requested.
|
|
*
|
|
* @param min_hdr_len The minimum data in bytes that needs to exist.
|
|
* @param remaining The remaining number of bytes in the packet reported by
|
|
* previous analyzer.
|
|
* @param packet The packet being processed. This will be used to pull out the
|
|
* number of bytes the IP header says we have remaining.
|
|
*/
|
|
bool CheckHeaderTrunc(size_t min_hdr_len, size_t remaining, Packet* packet);
|
|
|
|
/**
|
|
* Returns true if the port corresponds to an application for which there
|
|
* is a Zeek analyzer (even if it might not be used by the present policy
|
|
* script) or if it's generally a likely server port.
|
|
*
|
|
* @param port The port number to check, in host order.
|
|
*/
|
|
bool IsLikelyServerPort(uint32_t port) const;
|
|
|
|
private:
|
|
// While this is storing session analyzer tags, we store it here since packet analyzers
|
|
// are persistent objects. We can't do this in the adapters because those get created
|
|
// and destroyed for each connection.
|
|
using tag_set = std::set<zeek::Tag>;
|
|
using analyzer_map_by_port = std::map<uint32_t, tag_set*>;
|
|
analyzer_map_by_port analyzers_by_port;
|
|
|
|
tag_set* LookupPort(uint32_t port, bool add_if_not_found);
|
|
|
|
/**
|
|
* Creates a new Connection object from data gleaned from the current packet.
|
|
*
|
|
* @param key A ConnKey with common 5-tuple information.
|
|
* @param pkt The packet associated with the new connection, for additional connection info.
|
|
*/
|
|
zeek::Connection* NewConn(IPBasedConnKeyPtr key, const Packet* pkt);
|
|
|
|
void BuildSessionAnalyzerTree(Connection* conn);
|
|
|
|
TransportProto transport;
|
|
uint32_t server_port_mask;
|
|
static TableValPtr ignore_checksums_nets_table;
|
|
};
|
|
|
|
} // namespace zeek::packet_analysis::IP
|