blob: e1381f5312d202a38cc9aa83fa141747a739ad09 [file] [log] [blame]
/*
**************************************************************************
* Copyright (c) 2014, The Linux Foundation. All rights reserved.
* Permission to use, copy, modify, and/or distribute this software for
* any purpose with or without fee is hereby granted, provided that the
* above copyright notice and this permission notice appear in all copies.
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
* OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
**************************************************************************
*/
struct ecm_tracker_instance;
/*
* Data tracking data limits - system global and per-connection - defaults.
*/
#define ECM_TRACKER_GLOBAL_DATA_LIMIT_DEFAULT (1024 * 1024 * 8)
#define ECM_TRACKER_GLOBAL_DATA_BUFFER_LIMIT_DEFAULT (1024 * 1024 * 64)
#define ECM_TRACKER_CONNECTION_TRACKING_LIMIT_DEFAULT (1024 * 1024)
#define ECM_TRACKER_CONNECTION_TRACKING_LIMIT_MAX ECM_TRACKER_GLOBAL_DATA_LIMIT_DEFAULT
enum ecm_tracker_sender_types {
ECM_TRACKER_SENDER_TYPE_SRC = 0, /* Sender of tracked data is the source of the connection (who established the connection) */
ECM_TRACKER_SENDER_TYPE_DEST = 1, /* Sender of tracked data is the destination of the connection (to whom connection was established) */
ECM_TRACKER_SENDER_MAX, /* MUST BE LAST */
};
typedef enum ecm_tracker_sender_types ecm_tracker_sender_type_t;
/*
* enum ecm_tracker_sender_states
* Notional states of senders of a tracker
*
* Order is important here - don't change them as logic depends on their numerical value.
*/
enum ecm_tracker_sender_states {
ECM_TRACKER_SENDER_STATE_UNKNOWN = 0, /* Endpoint has not sent any packets yet */
ECM_TRACKER_SENDER_STATE_ESTABLISHING, /* Endpoint has not yet given any indication it is established */
ECM_TRACKER_SENDER_STATE_ESTABLISHED, /* Endpoint has indicated that it is established */
ECM_TRACKER_SENDER_STATE_CLOSING, /* Endpoint has indicated that it wants to close down its side of the connection */
ECM_TRACKER_SENDER_STATE_CLOSED, /* Endpoint has closed, connection remains to service any late packets */
ECM_TRACKER_SENDER_STATE_FAULT, /* Endpoint experienced a fault */
ECM_TRACKER_SENDER_STATE_MAX, /* MUST BE LAST */
};
typedef enum ecm_tracker_sender_states ecm_tracker_sender_state_t;
/*
* ecm_tracker_sender_state_to_string[]
* Convert a sender state to a string
*/
const char *
ecm_tracker_sender_state_to_string(enum ecm_tracker_sender_states);
/*
* enum ecm_tracker_connection_states
* Notional states of connection being monitored by the tracker
*
* Order is important here - don't change them as logic depends on their numerical value.
*/
enum ecm_tracker_connection_states {
ECM_TRACKER_CONNECTION_STATE_ESTABLISHING = 0, /* Not yet given any indication it is established */
ECM_TRACKER_CONNECTION_STATE_ESTABLISHED, /* It is established */
ECM_TRACKER_CONNECTION_STATE_CLOSING, /* Connection has begun process of closing */
ECM_TRACKER_CONNECTION_STATE_CLOSED, /* Has closed, connection remains to service any late packets */
ECM_TRACKER_CONNECTION_STATE_FAULT, /* Experienced a fault */
ECM_TRACKER_CONNECTION_STATE_MAX, /* MUST BE LAST */
};
typedef enum ecm_tracker_connection_states ecm_tracker_connection_state_t;
/*
* ecm_tracker_connection_state_to_string
* Convert a connection state to a string
*/
const char *
ecm_tracker_connection_state_to_string(enum ecm_tracker_connection_states);
/*
* enum ecm_tracker_ip_protocol_types
* A list of protocol types that can be recorded in the ecm_ip_header
*
* This is especially useful for IPv6 where the ip header can contain many sub headers.
* But it is also useful for IPv4 where you might have IP following a GRE header, for example.
* An ECM IP header may record only ONE of each type of header, if more are found the header is considered invalid.
*
* These constants are used to index into the ecm_tracker_ip_header.headers[]
*/
enum ecm_tracker_ip_protocol_types {
ECM_TRACKER_IP_PROTOCOL_TYPE_IPV6_HBH,
ECM_TRACKER_IP_PROTOCOL_TYPE_ICMP,
ECM_TRACKER_IP_PROTOCOL_TYPE_UDP,
ECM_TRACKER_IP_PROTOCOL_TYPE_TCP,
ECM_TRACKER_IP_PROTOCOL_TYPE_GRE,
ECM_TRACKER_IP_PROTOCOL_TYPE_IPV6_ROUTING,
ECM_TRACKER_IP_PROTOCOL_TYPE_IPV6_FRAGMENT,
ECM_TRACKER_IP_PROTOCOL_TYPE_AH,
ECM_TRACKER_IP_PROTOCOL_TYPE_IPV6_ICMP,
ECM_TRACKER_IP_PROTOCOL_TYPE_IPV6_DO,
ECM_TRACKER_IP_PROTOCOL_TYPE_UNKNOWN, /* A protocol that is unrecognised */
ECM_TRACKER_IP_PROTOCOL_TYPE_COUNT /* Must be last, do not use */
};
typedef enum ecm_tracker_ip_protocol_types ecm_tracker_ip_protocol_type_t;
/*
* struct ecm_tracker_ip_protocol_header
* Records a protocol header as stored within an IP datagram
*/
struct ecm_tracker_ip_protocol_header {
uint8_t protocol_number; /* IP protocol number */
uint16_t header_size; /* Size of the protocol header */
uint16_t size; /* Size of the header_size + its payload */
uint16_t offset; /* Offset from the start of the skb where this header is located */
};
/*
* struct ecm_tracker_ip_header
* An IP header in the ECM can deal with V4 and V6 headers.
*
* WARNING: An ecm_ip_header is ONLY VALID while the skb from which it was initialised remains untouched.
*/
struct ecm_tracker_ip_header {
/*
* h is a union of v4 and v6 headers.
* This only works as far as the version field goes, but that's enough to know what we are dealing with.
* These are also used as buffers where skn_header_pointer() needs them to perform a skb_copy_bits() operation.
*/
union {
struct iphdr v4_hdr;
struct ipv6hdr v6_hdr;
} h;
struct sk_buff *skb; /* COPY of POINTER to the skb this header relates to. This ecm_ip_header is ONLY VALID for as long as the skb it relates to remains UNTOUCHED */
bool is_v4; /* True when v4, else v6 */
ip_addr_t src_addr; /* ECM ip address equivalent */
ip_addr_t dest_addr; /* ECM ip address equivalent */
int protocol; /* The upper layer transport protocol */
bool fragmented; /* True when fragmented */
uint32_t ip_header_length; /* Length of the IP header plus any variable sized intrinsically attached options */
uint32_t total_length; /* total length of IP header including all extensions and payload. For v4 this is total_len, for v6 this is payload_len + size of the IP 6 header */
uint32_t payload_length; /* total_length - ip_header_length */
struct ecm_tracker_ip_protocol_header headers[ECM_TRACKER_IP_PROTOCOL_TYPE_COUNT];
/* Use one of the ECM_TRACKER_IP_PROTOCOL_TYPE_XYZ constants to index into this to locate the header you want to inspect. If the size is zero then the header was not found. */
};
typedef int32_t (*ecm_tracker_datagram_count_get_method_t)(struct ecm_tracker_instance *ti, ecm_tracker_sender_type_t sender);
/* Return number of available datagrams sent by the sender */
typedef void (*ecm_tracker_datagram_discard_method_t)(struct ecm_tracker_instance *ti, ecm_tracker_sender_type_t sender, int32_t n);
/* Discard n number of datagrams at the head of the datagram list that were sent by the sender */
typedef int32_t (*ecm_tracker_datagram_size_get_method_t)(struct ecm_tracker_instance *ti, ecm_tracker_sender_type_t sender, int32_t i);
/* Return size in bytes of datagram at index i that was sent by the sender */
typedef int (*ecm_tracker_datagram_read_method_t)(struct ecm_tracker_instance *ti, ecm_tracker_sender_type_t sender, int32_t i, int32_t offset, int32_t size, void *buffer);
/* Read size bytes from datagram at index i into the buffer */
typedef bool (*ecm_tracker_datagram_add_method_t)(struct ecm_tracker_instance *ti, ecm_tracker_sender_type_t sender, struct sk_buff *skb);
/* Add (append) the datagram into the tracker */
typedef void (*ecm_tracker_ref_method_t)(struct ecm_tracker_instance *ti);
typedef int (*ecm_tracker_deref_method_t)(struct ecm_tracker_instance *ti);
typedef void (*ecm_tracker_discard_all_method_t)(struct ecm_tracker_instance *ti);
/* Discard all tracked data */
typedef int32_t (*ecm_tracker_data_total_get_method_t)(struct ecm_tracker_instance *ti);
/* Return number of bytes of tracked data in total */
typedef int32_t (*ecm_tracker_data_limit_get_method_t)(struct ecm_tracker_instance *ti);
/* Return the limit on the number of bytes we can track */
typedef void (*ecm_tracker_data_limit_set_method_t)(struct ecm_tracker_instance *ti, int32_t data_limit);
/* Set the limit on the number of bytes we can track */
typedef void (*ecm_tracker_state_update_method_t)(struct ecm_tracker_instance *ti, ecm_tracker_sender_type_t sender, struct ecm_tracker_ip_header *ip_hdr, struct sk_buff *skb);
/* Update state of the sender */
typedef void (*ecm_tracker_state_get_method_t)(struct ecm_tracker_instance *ti, ecm_tracker_sender_state_t *src_state, ecm_tracker_sender_state_t *dest_state, ecm_tracker_connection_state_t *state, ecm_db_timer_group_t *tg);
/* State of the connection */
typedef int (*ecm_tracker_xml_state_get_callback_t)(struct ecm_tracker_instance *ti, char *buf, int buf_sz);
/* Get XML state output, buf has buf_sz bytes available. Returns number of bytes written.
* Function has failed if the return is (<= 0) || (return value == buf_sz).
* The return code is compatible with snprintf().
*/
/*
* struct ecm_tracker_instance
* Base class of all trackers
*
* ALL trackers must implement these features in addition to their own.
* ALL trackers must be castable to a type of this, i.e. this structure must be the first element of their own data type.
*/
struct ecm_tracker_instance {
ecm_tracker_data_total_get_method_t data_total_get;
ecm_tracker_data_limit_get_method_t data_limit_get;
ecm_tracker_data_limit_set_method_t data_limit_set;
ecm_tracker_datagram_count_get_method_t datagram_count_get;
ecm_tracker_datagram_discard_method_t datagram_discard;
ecm_tracker_datagram_size_get_method_t datagram_size_get;
ecm_tracker_datagram_read_method_t datagram_read;
ecm_tracker_datagram_add_method_t datagram_add;
ecm_tracker_discard_all_method_t discard_all;
ecm_tracker_state_update_method_t state_update;
ecm_tracker_state_get_method_t state_get;
ecm_tracker_xml_state_get_callback_t xml_state_get; /* Return an XML element containing its state */
ecm_tracker_ref_method_t ref;
ecm_tracker_deref_method_t deref;
};
bool ecm_tracker_ip_check_header_and_read(struct ecm_tracker_ip_header *ip_hdr, struct sk_buff *skb);
uint32_t ecm_tracker_data_limit_get(void);
void ecm_tracker_data_limit_set(uint32_t limit);
uint32_t ecm_tracker_data_total_get(void);
uint32_t ecm_tracker_data_buffer_total_get(void);
bool ecm_tracker_data_total_increase(uint32_t n, uint32_t data_bufer_size);
void ecm_tracker_data_total_decrease(uint32_t n, uint32_t data_bufer_size);