Go to the documentation of this file.
65 #define FLOW_DEFAULT_EMERGENCY_RECOVERY 30
68 #define FLOW_DEFAULT_HASHSIZE 65536
70 #define FLOW_DEFAULT_MEMCAP (32 * 1024 * 1024)
72 #define FLOW_DEFAULT_PREALLOC 10000
142 if (f == NULL || f->
proto == 0)
206 static inline void FlowSwapFlags(
Flow *f)
219 static inline void FlowSwapFileFlags(
Flow *f)
228 static inline void TcpStreamFlowSwap(
Flow *f)
254 FlowSwapFileFlags(f);
258 if (f->
proto == IPPROTO_TCP) {
259 TcpStreamFlowSwap(f);
305 }
else if (p->
proto == IPPROTO_ICMP || p->
proto == IPPROTO_ICMPV6) {
325 static inline int FlowUpdateSeenFlag(
const Packet *p)
327 if (PacketIsICMPv4(p)) {
336 static inline void FlowUpdateTtlTS(
Flow *f, uint8_t ttl)
346 static inline void FlowUpdateTtlTC(
Flow *f, uint8_t ttl)
356 static inline void FlowUpdateFlowRate(
370 SCLogDebug(
"Flow rate for flow %p exceeds the configured values, marking it as an "
395 static inline void FlowUpdateEthernet(
399 const EthernetHdr *ethh = PacketGetEthernet(p);
429 #ifdef CAPTURE_OFFLOAD
432 if (state != FLOW_STATE_CAPTURE_BYPASSED) {
438 #ifdef CAPTURE_OFFLOAD
444 SCLogDebug(
"Downgrading flow to local bypass");
463 if (FlowUpdateSeenFlag(p)) {
474 FlowUpdateEthernet(
tv,
dtv, f, p,
true);
476 if (PacketIsIPv4(p)) {
477 const IPV4Hdr *ip4h = PacketGetIPv4(p);
479 }
else if (PacketIsIPv6(p)) {
480 const IPV6Hdr *ip6h = PacketGetIPv6(p);
489 if (FlowUpdateSeenFlag(p)) {
500 FlowUpdateEthernet(
tv,
dtv, f, p,
false);
502 if (PacketIsIPv4(p)) {
503 const IPV4Hdr *ip4h = PacketGetIPv4(p);
505 }
else if (PacketIsIPv6(p)) {
506 const IPV6Hdr *ip6h = PacketGetIPv6(p);
518 }
else if (f->
proto == IPPROTO_TCP) {
529 #ifdef CAPTURE_OFFLOAD
530 (f->
flow_state != FLOW_STATE_CAPTURE_BYPASSED) &&
542 SCLogDebug(
"setting FLOW_NOPAYLOAD_INSPECTION flag on flow %p", f);
543 DecodeSetNoPayloadInspectionFlag(p);
591 if (
SCConfGetInt(
"flow.emergency-recovery", &val) == 1) {
592 if (val <= 100 && val >= 1) {
595 SCLogError(
"flow.emergency-recovery must be in the range of "
596 "1 and 100 (as percentage)");
600 SCLogDebug(
"flow.emergency-recovery, using default value");
605 const char *conf_val;
606 uint32_t configval = 0;
609 uint64_t flow_memcap_copy = 0;
610 if ((
SCConfGet(
"flow.memcap", &conf_val)) == 1) {
611 if (conf_val == NULL) {
612 FatalError(
"Invalid value for flow.memcap: NULL");
617 "from conf file - %s. Killing engine",
624 if ((
SCConfGet(
"flow.hash-size", &conf_val)) == 1) {
625 if (conf_val == NULL) {
626 FatalError(
"Invalid value for flow.hash-size: NULL");
629 if (
StringParseUint32(&configval, 10, strlen(conf_val), conf_val) && configval != 0) {
632 FatalError(
"Invalid value for flow.hash-size. Must be a numeric value in the range "
636 if ((
SCConfGet(
"flow.prealloc", &conf_val)) == 1) {
637 if (conf_val == NULL) {
638 FatalError(
"Invalid value for flow.prealloc: NULL");
649 SCLogDebug(
"Flow config from suricata.yaml: memcap: %"PRIu64
", hash-size: "
657 "max flow memcap is smaller than projected hash size. "
658 "Memcap: %" PRIu64
", Hash table size %" PRIu64
". Calculate "
659 "total hash size by multiplying \"flow.hash-size\" with %" PRIuMAX
", "
660 "which is the hash bucket size.",
666 FatalError(
"Fatal error encountered in FlowInitConfig. Exiting...");
678 SCLogConfig(
"allocated %"PRIu64
" bytes of memory for the flow hash... "
679 "%" PRIu32
" buckets of size %" PRIuMAX
"",
681 (uintmax_t)
sizeof(FlowBucket));
685 SCLogConfig(
"flow memory usage: %"PRIu64
" bytes, maximum: %"PRIu64,
692 SCLogConfig(
"flow size %u, memcap allows for %" PRIu64
" flows. Per hash row in perfect "
693 "conditions %" PRIu64,
763 #define SET_DEFAULTS(p, n, e, c, b, ne, ee, ce, be) \
764 flow_timeouts_normal[(p)].new_timeout = (n); \
765 flow_timeouts_normal[(p)].est_timeout = (e); \
766 flow_timeouts_normal[(p)].closed_timeout = (c); \
767 flow_timeouts_normal[(p)].bypassed_timeout = (b); \
768 flow_timeouts_emerg[(p)].new_timeout = (ne); \
769 flow_timeouts_emerg[(p)].est_timeout = (ee); \
770 flow_timeouts_emerg[(p)].closed_timeout = (ce); \
771 flow_timeouts_emerg[(p)].bypassed_timeout = (be); \
800 const char *
new = NULL;
801 const char *established = NULL;
802 const char *closed = NULL;
803 const char *bypassed = NULL;
804 const char *emergency_new = NULL;
805 const char *emergency_established = NULL;
806 const char *emergency_closed = NULL;
807 const char *emergency_bypassed = NULL;
810 if (flow_timeouts != NULL) {
812 uint32_t configval = 0;
831 if (established != NULL &&
837 if (closed != NULL &&
843 if (bypassed != NULL &&
850 if (emergency_new != NULL &&
852 emergency_new) > 0) {
856 if (emergency_established != NULL &&
858 strlen(emergency_established),
859 emergency_established) > 0) {
863 if (emergency_closed != NULL &&
865 strlen(emergency_closed),
866 emergency_closed) > 0) {
870 if (emergency_bypassed != NULL &&
872 strlen(emergency_bypassed),
873 emergency_bypassed) > 0) {
896 if (established != NULL &&
902 if (closed != NULL &&
908 if (bypassed != NULL &&
915 if (emergency_new != NULL &&
917 emergency_new) > 0) {
921 if (emergency_established != NULL &&
923 strlen(emergency_established),
924 emergency_established) > 0) {
928 if (emergency_closed != NULL &&
930 strlen(emergency_closed),
931 emergency_closed) > 0) {
935 if (emergency_bypassed != NULL &&
937 strlen(emergency_bypassed),
938 emergency_bypassed) > 0) {
959 if (established != NULL &&
965 if (bypassed != NULL &&
972 if (emergency_new != NULL &&
974 emergency_new) > 0) {
978 if (emergency_established != NULL &&
980 strlen(emergency_established),
981 emergency_established) > 0) {
985 if (emergency_bypassed != NULL &&
987 strlen(emergency_bypassed),
988 emergency_bypassed) > 0) {
1009 if (established != NULL &&
1015 if (bypassed != NULL &&
1022 if (emergency_new != NULL &&
1024 emergency_new) > 0) {
1028 if (emergency_established != NULL &&
1030 strlen(emergency_established),
1031 emergency_established) > 0) {
1035 if (emergency_bypassed != NULL &&
1037 strlen(emergency_bypassed),
1038 emergency_bypassed) > 0) {
1051 SCLogWarning(
"emergency timeout value %u for \'established\' "
1052 "must be below regular value %u",
1058 SCLogWarning(
"emergency timeout value %u for \'new\' must be "
1059 "below regular value %u",
1065 SCLogWarning(
"emergency timeout value %u for \'closed\' must "
1066 "be below regular value %u",
1072 SCLogWarning(
"emergency timeout value %u for \'bypassed\' "
1073 "must be below regular value %u",
1085 SCLogWarning(
"emergency timeout value for \'established\' must be below normal value");
1091 SCLogWarning(
"emergency timeout value for \'new\' must be below normal value");
1097 SCLogWarning(
"emergency timeout value for \'closed\' must be below normal value");
1103 SCLogWarning(
"emergency timeout value for \'bypassed\' must be below normal value");
1108 SCLogDebug(
"deltas: new: -%u est: -%u closed: -%u bypassed: -%u",
1167 if (f->
proto != IPPROTO_TCP) {
1174 uint8_t newflags =
flags;
1179 newflags |= STREAM_DEPTH;
1194 const uint32_t timeout_policy = FlowGetTimeoutPolicy(f);
1200 if (f->
fb != NULL) {
1267 static int FlowTest01 (
void)
1301 static void test(
void *f) {}
1310 static int FlowTest02 (
void)
1332 static int FlowTest07 (
void)
1378 static int FlowTest08 (
void)
1425 static int FlowTest09 (
void)
1472 UtRegisterTest(
"FlowTest01 -- Protocol Specific Timeouts", FlowTest01);
1473 UtRegisterTest(
"FlowTest02 -- Setting Protocol Specific Free Function",
1475 UtRegisterTest(
"FlowTest07 -- Test flow Allocations when it reach memcap",
1477 UtRegisterTest(
"FlowTest08 -- Test flow Allocations when it reach memcap",
1479 UtRegisterTest(
"FlowTest09 -- Test flow Allocations when it reach memcap",
#define FLOWFILE_NO_MD5_TS
void FlowUnsetChangeProtoFlag(Flow *f)
Unset flag to indicate to change proto for the flow.
#define FLOWFILE_NO_MD5_TC
#define FBLOCK_DESTROY(fb)
#define FLOW_DEFAULT_NEW_TIMEOUT
void MacSetAddWithCtr(MacSet *ms, const uint8_t *src_addr, const uint8_t *dst_addr, ThreadVars *tv, StatsCounterMaxId ctr_src, StatsCounterMaxId ctr_dst)
#define FLOW_DEFAULT_EMERG_BYPASSED_TIMEOUT
#define FLOW_HAS_EXPECTATION
void FlowSetHasAlertsFlag(Flow *f)
Set flag to indicate that flow has alerts.
void FlowCleanupAppLayer(Flow *f)
void FlowSetChangeProtoFlag(Flow *f)
Set flag to indicate to change proto for the flow.
#define IPV6_GET_RAW_HLIM(ip6h)
uint32_t FlowSpareGetPoolSize(void)
int FlowGetPacketDirection(const Flow *f, const Packet *p)
determine the direction of the packet compared to the flow
#define SC_ATOMIC_INIT(name)
wrapper for initializing an atomic variable.
#define FLOW_DEFAULT_HASHSIZE
SC_ATOMIC_DECLARE(FlowProtoTimeoutPtr, flow_timeouts)
uint32_t emergency_recovery
#define FLOW_DEFAULT_PREALLOC
#define SC_ATOMIC_SET(name, val)
Set the value for the atomic variable.
void UtRegisterTest(const char *name, int(*TestFn)(void))
Register unit test.
uint64_t PcapPacketCntGet(const Packet *p)
#define FLOW_IS_ELEPHANT_TOCLIENT
void FlowRegisterTests(void)
Function to register the Flow Unitests.
#define FLOW_IPPROTO_UDP_EMERG_NEW_TIMEOUT
#define FLOW_SGH_TOCLIENT
int ParseSizeStringU64(const char *size, uint64_t *res)
#define FLOWFILE_NO_MAGIC_TS
#define SC_ATOMIC_ADD(name, val)
add a value to our atomic variable
int SCConfGet(const char *name, const char **vptr)
Retrieve the value of a configuration node.
#define FLOW_NOPAYLOAD_INSPECTION
#define FLOW_TS_PM_ALPROTO_DETECT_DONE
FlowStorageId FlowRateGetStorageID(void)
#define FLOW_DEFAULT_EMERG_NEW_TIMEOUT
#define FLOW_IPPROTO_TCP_EMERG_NEW_TIMEOUT
void BypassedFlowUpdate(Flow *f, Packet *p)
uint32_t bypassed_timeout
#define FLOW_DEFAULT_EST_TIMEOUT
#define FLOW_PKT_TOSERVER
uint64_t FlowGetMemuse(void)
uint64_t FlowGetMemcap(void)
Return memcap value.
void FlowQueueDestroy(FlowQueue *q)
Destroy a flow queue.
void FlowHandlePacket(ThreadVars *tv, FlowLookupStruct *fls, Packet *p)
Entry point for packet flow handling.
void SCFlowRunUpdateCallbacks(ThreadVars *tv, Flow *f, Packet *p)
#define FLOW_TOSERVER_DROP_LOGGED
const char * SCConfNodeLookupChildValue(const SCConfNode *node, const char *name)
Lookup the value of a child configuration node by name.
#define FLOW_TC_PE_ALPROTO_DETECT_DONE
#define FLOW_DEFAULT_BYPASSED_TIMEOUT
uint32_t SCFlowGetFlags(const Flow *flow)
Get flow flags.
enum ExceptionPolicy ExceptionPolicyParse(const char *option, bool support_flow)
#define FLOW_IPPROTO_UDP_BYPASSED_TIMEOUT
#define STREAMTCP_STREAM_FLAG_DEPTH_REACHED
#define FLOW_IPPROTO_TCP_EMERG_CLOSED_TIMEOUT
#define FLOW_IPPROTO_ICMP_EST_TIMEOUT
uint32_t UTHBuildPacketOfFlows(uint32_t start, uint32_t end, uint8_t dir)
void AppLayerExpectationClean(Flow *f)
#define PKT_PROTO_DETECT_TS_DONE
struct Packet_::@32::@39 icmp_s
const struct SigGroupHead_ * sgh_toserver
#define FLOW_CHECK_MEMCAP(size)
check if a memory alloc would fit in the memcap
#define FLOW_RECYCLE(f)
macro to recycle a flow before it goes into the spare queue for reuse.
AppLayerParserState * alparser
void FlowRateStoreUpdate(FlowRateStore *frs, SCTime_t p_ts, uint32_t pkt_len, int direction)
StatsCounterId counter_flow_elephant
#define FLOWFILE_NO_SHA1_TC
#define FLOW_IPPROTO_ICMP_EMERG_EST_TIMEOUT
void FlowInitConfig(bool quiet)
initialize the configuration
#define FLOW_IPPROTO_UDP_EST_TIMEOUT
@ FLOW_STATE_LOCAL_BYPASSED
#define PASS
Pass the test.
#define FLOWFILE_NO_STORE_TS
#define FLOW_IPPROTO_TCP_CLOSED_TIMEOUT
void FlowHandlePacketUpdate(Flow *f, Packet *p, ThreadVars *tv, DecodeThreadVars *dtv)
Update Packet and Flow.
int SCConfGetInt(const char *name, intmax_t *val)
Retrieve a configuration value as an integer.
#define FLOW_CHANGE_PROTO
Per thread variable structure.
unsigned int FlowStorageSize(void)
#define FLOWFILE_NO_SHA256_TS
#define FLOW_PKT_TOCLIENT_FIRST
void StatsCounterIncr(StatsThreadContext *stats, StatsCounterId id)
Increments the local counter.
FlowQueue * FlowQueueInit(FlowQueue *q)
int StringParseUint32(uint32_t *res, int base, size_t len, const char *str)
#define SCLogWarning(...)
Macro used to log WARNING messages.
void AppLayerParserStateCleanup(const Flow *f, void *alstate, AppLayerParserState *pstate)
#define FLOW_PROTO_DETECT_TC_DONE
#define FLOW_TC_PP_ALPROTO_DETECT_DONE
const struct SigGroupHead_ * sgh_toclient
bool MacSetFlowStorageEnabled(void)
#define SC_ATOMIC_SUB(name, val)
sub a value from our atomic variable
void TimeSetIncrementTime(uint32_t tv_sec)
increment the time in the engine
uint8_t FlowGetProtoMapping(uint8_t proto)
Function to map the protocol to the defined FLOW_PROTO_* enumeration.
#define IPV4_GET_RAW_IPTTL(ip4h)
#define FLOW_DEFAULT_MEMCAP
#define FLOW_IPPROTO_TCP_NEW_TIMEOUT
int FlowClearMemory(Flow *f, uint8_t proto_map)
Function clear the flow memory before queueing it to spare flow queue.
FlowProtoTimeout flow_timeouts_delta[FLOW_PROTO_MAX]
void SCFlowGetLastTimeAsParts(const Flow *flow, uint64_t *secs, uint64_t *usecs)
Get flow last time as individual values.
FlowStorageId MacSetGetFlowStorageID(void)
uint32_t probing_parser_toclient_alproto_masks
#define FLOW_PKT_TOCLIENT
uint16_t SCFlowGetDestinationPort(const Flow *flow)
Get flow destination port.
void * FlowGetStorageById(const Flow *f, FlowStorageId id)
void FlowUpdateState(Flow *f, const enum FlowState s)
StatsCounterMaxId counter_max_mac_addrs_dst
uint32_t probing_parser_toserver_alproto_masks
#define FLOWFILE_NO_SHA256_TC
StatsCounterMaxId counter_max_mac_addrs_src
SCConfNode * SCConfNodeLookupChild(const SCConfNode *node, const char *name)
Lookup a child configuration node by name.
Flow * FlowDequeue(FlowQueue *q)
remove a flow from the queue
#define FLOW_TS_PE_ALPROTO_DETECT_DONE
#define FLOW_IPPROTO_UDP_NEW_TIMEOUT
int FlowSetProtoFreeFunc(uint8_t, void(*Free)(void *))
Function to set the function to get protocol specific flow state.
#define FLOW_IPPROTO_ICMP_NEW_TIMEOUT
#define FAIL_IF(expr)
Fail a test if expression evaluates to true.
FlowProtoTimeout flow_timeouts_normal[FLOW_PROTO_MAX]
unsigned short FlowStateType
#define FLOWFILE_NO_MAGIC_TC
enum ExceptionPolicy FlowGetMemcapExceptionPolicy(void)
void FlowFree(Flow *f)
cleanup & free the memory of a flow
#define FLOW_IPPROTO_TCP_BYPASSED_TIMEOUT
uint16_t SCFlowGetSourcePort(const Flow *flow)
Get flow source port.
void FlowShutdown(void)
shutdown the flow engine
void FlowSparePoolDestroy(void)
#define SCMallocAligned(size, align)
#define SWAP_VARS(type, a, b)
void FlowTimeoutsInit(void)
#define FLOW_PROTO_DETECT_TS_DONE
bool FlowRateIsExceeding(FlowRateStore *frs, int direction)
#define FLOW_TS_PP_ALPROTO_DETECT_DONE
#define PKT_PROTO_DETECT_TC_DONE
#define FLOWFILE_NO_STORE_TC
Flow * FlowGetFlowFromHash(ThreadVars *tv, FlowLookupStruct *fls, Packet *p, Flow **dest)
Get Flow for packet.
#define FLOW_IPPROTO_TCP_EMERG_EST_TIMEOUT
void FlowSwap(Flow *f)
swap the flow's direction
enum ExceptionPolicy memcap_policy
void MacSetSwap(MacSet *ms)
struct SCLogConfig_ SCLogConfig
Holds the config state used by the logging api.
void FlowInitFlowProto(void)
Function to set the default timeout, free function and flow state function for all supported flow_pro...
#define SCTIME_CMP_GT(a, b)
#define FLOWFILE_NO_SHA1_TS
SCConfNode * SCConfGetNode(const char *name)
Get a SCConfNode by name.
#define SCLogError(...)
Macro used to log ERROR messages.
#define SWAP_FLAGS(flags, a, b)
#define FLOW_SGH_TOSERVER
Structure to hold thread specific data for all decode modules.
FlowProtoFreeFunc flow_freefuncs[FLOW_PROTO_MAX]
FlowProtoTimeout flow_timeouts_emerg[FLOW_PROTO_MAX]
#define FLOW_IPPROTO_UDP_EMERG_EST_TIMEOUT
void FlowFreeStorage(Flow *f)
#define FLOW_IPPROTO_ICMP_EMERG_NEW_TIMEOUT
#define FLOW_TOCLIENT_DROP_LOGGED
#define FLOW_PKT_ESTABLISHED
bool FlowRateStorageEnabled(void)
void PacketDrop(Packet *p, const uint8_t action, enum PacketDropReason r)
issue drop action
int EngineModeIsIPS(void)
@ SIGNATURE_HOOK_PKT_FLOW_START
#define FLOW_TC_PM_ALPROTO_DETECT_DONE
#define FLOW_DEFAULT_EMERGENCY_RECOVERY
int FlowHasAlerts(const Flow *f)
Check if flow has alerts.
void FlowSparePoolInit(void)
#define FLOW_DEFAULT_EMERG_EST_TIMEOUT
#define FLOW_IPPROTO_ICMP_BYPASSED_TIMEOUT
int FlowChangeProto(Flow *f)
Check if change proto flag is set for flow.
#define SC_ATOMIC_GET(name)
Get the value from the atomic variable.
StatsCounterId counter_flow_elephant_toclient
StatsCounterId counter_flow_elephant_toserver
uint8_t FlowGetDisruptionFlags(const Flow *f, uint8_t flags)
get 'disruption' flags: GAP/DEPTH/PASS
int8_t data_first_seen_dir
#define FLOW_DIR_REVERSED
#define ICMPV4_IS_ERROR_MSG(type)
#define FLOW_IPPROTO_TCP_EST_TIMEOUT
void RegisterFlowStorageTests(void)
#define SET_DEFAULTS(p, n, e, c, b, ne, ee, ce, be)
#define FLOW_PKT_TOSERVER_FIRST
#define DEBUG_VALIDATE_BUG_ON(exp)
@ PKT_DROP_REASON_FLOW_DROP
#define FLOW_IS_ELEPHANT_TOSERVER
FlowThreadId thread_id[2]
int FlowSetMemcap(uint64_t size)
Update memcap value.