Go to the documentation of this file.
75 static inline int FlowHashRawAddressIPv6GtU32(
const uint32_t *a,
const uint32_t *b)
77 for (uint8_t i = 0; i < 4; i++) {
98 const uint32_t
u32[6];
149 if (PacketIsIPv4(
p)) {
154 int ai = (
p->
src.addr_data32[0] >
p->
dst.addr_data32[0]);
155 fhk.
addrs[1 - ai] =
p->
src.addr_data32[0];
158 fhk.
ports[0] = 0xfedc;
159 fhk.
ports[1] = 0xba98;
161 FlowHashIp4Fill(&fhk,
p);
164 }
else if (PacketIsIPv6(
p)) {
168 if (FlowHashRawAddressIPv6GtU32(
p->
src.addr_data32,
p->
dst.addr_data32)) {
169 fhk.
src[0] =
p->
src.addr_data32[0];
170 fhk.
src[1] =
p->
src.addr_data32[1];
171 fhk.
src[2] =
p->
src.addr_data32[2];
172 fhk.
src[3] =
p->
src.addr_data32[3];
173 fhk.
dst[0] =
p->
dst.addr_data32[0];
174 fhk.
dst[1] =
p->
dst.addr_data32[1];
175 fhk.
dst[2] =
p->
dst.addr_data32[2];
176 fhk.
dst[3] =
p->
dst.addr_data32[3];
178 fhk.
src[0] =
p->
dst.addr_data32[0];
179 fhk.
src[1] =
p->
dst.addr_data32[1];
180 fhk.
src[2] =
p->
dst.addr_data32[2];
181 fhk.
src[3] =
p->
dst.addr_data32[3];
182 fhk.
dst[0] =
p->
src.addr_data32[0];
183 fhk.
dst[1] =
p->
src.addr_data32[1];
184 fhk.
dst[2] =
p->
src.addr_data32[2];
185 fhk.
dst[3] =
p->
src.addr_data32[3];
188 fhk.
ports[0] = 0xfedc;
189 fhk.
ports[1] = 0xba98;
191 FlowHashIp6Fill(&fhk,
p);
211 static inline uint32_t FlowGetHash(
const Packet *
p)
215 if (PacketIsIPv4(
p)) {
216 if (PacketIsTCP(
p) || PacketIsUDP(
p)) {
219 int ai = (
p->
src.addr_data32[0] >
p->
dst.addr_data32[0]);
223 const int pi = (
p->
sp >
p->
dp);
227 FlowHashIp4Fill(&fhk,
p);
236 const int ai = (psrc > pdst);
237 fhk.
addrs[1-ai] = psrc;
238 fhk.
addrs[ai] = pdst;
244 FlowHashIp4Fill(&fhk,
p);
251 const int ai = (
p->
src.addr_data32[0] >
p->
dst.addr_data32[0]);
254 fhk.
ports[0] = 0xfeed;
255 fhk.
ports[1] = 0xbeef;
256 FlowHashIp4Fill(&fhk,
p);
260 }
else if (PacketIsIPv6(
p)) {
262 if (FlowHashRawAddressIPv6GtU32(
p->
src.addr_data32,
p->
dst.addr_data32)) {
263 fhk.
src[0] =
p->
src.addr_data32[0];
264 fhk.
src[1] =
p->
src.addr_data32[1];
265 fhk.
src[2] =
p->
src.addr_data32[2];
266 fhk.
src[3] =
p->
src.addr_data32[3];
267 fhk.
dst[0] =
p->
dst.addr_data32[0];
268 fhk.
dst[1] =
p->
dst.addr_data32[1];
269 fhk.
dst[2] =
p->
dst.addr_data32[2];
270 fhk.
dst[3] =
p->
dst.addr_data32[3];
272 fhk.
src[0] =
p->
dst.addr_data32[0];
273 fhk.
src[1] =
p->
dst.addr_data32[1];
274 fhk.
src[2] =
p->
dst.addr_data32[2];
275 fhk.
src[3] =
p->
dst.addr_data32[3];
276 fhk.
dst[0] =
p->
src.addr_data32[0];
277 fhk.
dst[1] =
p->
src.addr_data32[1];
278 fhk.
dst[2] =
p->
src.addr_data32[2];
279 fhk.
dst[3] =
p->
src.addr_data32[3];
282 const int pi = (
p->
sp >
p->
dp);
285 FlowHashIp6Fill(&fhk,
p);
313 const int pi = (fk->
sp > fk->
dp);
350 const int pi = (fk->
sp > fk->
dp);
365 static inline bool CmpAddrs(
const uint32_t addr1[4],
const uint32_t addr2[4])
367 return addr1[0] == addr2[0] && addr1[1] == addr2[1] &&
368 addr1[2] == addr2[2] && addr1[3] == addr2[3];
371 static inline bool CmpAddrsAndPorts(
const uint32_t src1[4],
372 const uint32_t dst1[4],
Port src_port1,
Port dst_port1,
373 const uint32_t src2[4],
const uint32_t dst2[4],
Port src_port2,
379 return (CmpAddrs(src1, src2) && CmpAddrs(dst1, dst2) &&
380 src_port1 == src_port2 && dst_port1 == dst_port2) ||
381 (CmpAddrs(src1, dst2) && CmpAddrs(dst1, src2) &&
382 src_port1 == dst_port2 && dst_port1 == src_port2);
385 static inline bool CmpVlanIds(
388 return ((vlan_id1[0] ^ vlan_id2[0]) &
g_vlan_mask) == 0 &&
389 ((vlan_id1[1] ^ vlan_id2[1]) &
g_vlan_mask) == 0 &&
393 static inline bool CmpLiveDevIds(
const uint16_t id1,
const uint16_t id2)
398 #define CmpFlowMisc(x, y) \
399 (((x)->proto == (y)->proto) && \
400 ((x)->recursion_level == (y)->recursion_level || g_recurlvl_mask == 0) && \
401 CmpVlanIds((x)->vlan_id, (y)->vlan_id))
405 static inline bool CmpFlowPacket(
const Flow *f,
const Packet *
p)
411 return CmpAddrsAndPorts(f_src, f_dst, f->
sp, f->
dp, p_src, p_dst,
p->
sp,
p->
dp) &&
415 static inline bool CmpFlowKey(
const Flow *f,
const FlowKey *k)
421 return CmpAddrsAndPorts(f_src, f_dst, f->
sp, f->
dp, k_src, k_dst, k->
sp, k->
dp) &&
425 static inline bool CmpAddrsAndICMPTypes(
const uint32_t src1[4],
426 const uint32_t dst1[4], uint8_t icmp_s_type1, uint8_t icmp_d_type1,
427 const uint32_t src2[4],
const uint32_t dst2[4], uint8_t icmp_s_type2,
428 uint8_t icmp_d_type2)
433 return (CmpAddrs(src1, src2) && CmpAddrs(dst1, dst2) &&
434 icmp_s_type1 == icmp_s_type2 && icmp_d_type1 == icmp_d_type2) ||
435 (CmpAddrs(src1, dst2) && CmpAddrs(dst1, src2) &&
436 icmp_s_type1 == icmp_d_type2 && icmp_d_type1 == icmp_s_type2);
439 static inline bool CmpFlowICMPPacket(
const Flow *f,
const Packet *
p)
445 return CmpAddrsAndICMPTypes(f_src, f_dst, f->
icmp_s.type, f->
icmp_d.type, p_src, p_dst,
460 static inline int FlowCompareICMPv4(
Flow *f,
const Packet *
p)
489 return CmpFlowICMPPacket(f,
p);
504 static inline int FlowCompareESP(
Flow *f,
const Packet *
p)
511 return CmpAddrs(f_src, p_src) && CmpAddrs(f_dst, p_dst) &&
CmpFlowMisc(f,
p) &&
522 static inline int FlowCompare(
Flow *f,
const Packet *
p)
524 if (
p->
proto == IPPROTO_ICMP) {
525 return FlowCompareICMPv4(f,
p);
526 }
else if (PacketIsESP(
p)) {
527 return FlowCompareESP(f,
p);
529 return CmpFlowPacket(f,
p);
543 static inline bool FlowCreateCheck(
const Packet *
p,
const bool emerg)
548 if (PacketIsTCP(
p)) {
549 const TCPHdr *tcph = PacketGetTCP(
p);
559 if (PacketIsICMPv4(
p)) {
601 const Packet *
p,
const bool emerg)
604 bool spare_sync =
false;
621 if (
tv && fls->
dtv) {
645 static void FlowExceptionPolicyStatsIncr(
649 if (
tv == NULL || fls->
dtv == NULL) {
681 if (g_eps_flow_memcap != UINT64_MAX && g_eps_flow_memcap ==
PcapPacketCntGet(
p)) {
682 NoFlowHandleIPS(
tv, fls,
p);
687 if (!FlowCreateCheck(
p, emerg)) {
694 f = FlowSpareSync(
tv, fls,
p, emerg);
706 f = FlowGetUsedFlow(
tv, fls->
dtv,
p->
ts);
708 NoFlowHandleIPS(
tv, fls,
p);
710 if (
tv != NULL && fls->
dtv != NULL) {
719 if (
tv != NULL && fls->
dtv != NULL) {
734 if (
tv != NULL && fls->
dtv != NULL) {
740 NoFlowHandleIPS(
tv, fls,
p);
757 const uint32_t hash,
Packet *
p)
760 if (
tv != NULL && fls->
dtv != NULL) {
773 Flow *f = FlowGetNew(
tv, fls,
p);
806 FlowBucket *fb,
Flow *f,
Flow *prev_f)
818 if (f->
proto != IPPROTO_TCP || FlowBelongsToUs(
tv, f)) {
825 f->
next = fb->evicted;
833 static inline bool FlowIsTimedOut(
866 tv_id = (uint16_t)
tv->
id;
869 tv_id = (uint16_t)
tv->
id;
900 SCLogDebug(
"fb %p fb->head %p", fb, fb->head);
903 if (fb->head == NULL) {
904 f = FlowGetNew(
tv, fls,
p);
919 FlowReference(dest, f);
925 const uint16_t tv_id = GetTvId(
tv);
927 const uint32_t fb_nextts = !emerg ?
SC_ATOMIC_GET(fb->next_ts) : 0;
928 const bool timeout_check = (fb_nextts <= (uint32_t)
SCTIME_SECS(
p->
ts));
934 const bool our_flow = FlowCompare(f,
p) != 0;
935 if (our_flow || timeout_check) {
937 const bool timedout = (timeout_check && FlowIsTimedOut(tv_id, f,
p->
ts, emerg));
940 MoveToWorkQueue(
tv, fls, fb, f, prev_f);
943 }
else if (our_flow) {
946 Flow *new_f = TcpReuseReplace(
tv, fls, fb, f, hash,
p);
949 MoveToWorkQueue(
tv, fls, fb, f, prev_f);
958 FlowReference(dest, f);
971 if (next_f == NULL) {
972 f = FlowGetNew(
tv, fls,
p);
988 FlowReference(dest, f);
1005 static inline bool FlowCompareKey(
Flow *f,
FlowKey *key)
1007 if ((f->
proto != IPPROTO_TCP) && (f->
proto != IPPROTO_UDP))
1009 return CmpFlowKey(f, key);
1023 uint32_t hash = flow_id & 0x0000FFFF;
1026 SCLogDebug(
"fb %p fb->head %p", fb, fb->head);
1028 for (
Flow *f = fb->head; f != NULL; f = f->
next) {
1029 if (FlowGetId(f) == flow_id) {
1050 static Flow *FlowGetExistingFlowFromHash(
FlowKey *key,
const uint32_t hash)
1055 SCLogDebug(
"fb %p fb->head %p", fb, fb->head);
1057 for (
Flow *f = fb->head; f != NULL; f = f->
next) {
1059 if (FlowCompareKey(f, key)) {
1087 Flow *f = FlowGetExistingFlowFromHash(key, hash);
1096 SCLogDebug(
"Can't get a spare flow at start");
1102 f->
src.addr_data32[0] = key->
src.addr_data32[0];
1103 f->
src.addr_data32[1] = key->
src.addr_data32[1];
1104 f->
src.addr_data32[2] = key->
src.addr_data32[2];
1105 f->
src.addr_data32[3] = key->
src.addr_data32[3];
1106 f->
dst.addr_data32[0] = key->
dst.addr_data32[0];
1107 f->
dst.addr_data32[1] = key->
dst.addr_data32[1];
1108 f->
dst.addr_data32[2] = key->
dst.addr_data32[2];
1109 f->
dst.addr_data32[3] = key->
dst.addr_data32[3];
1117 }
else if (key->
src.
family == AF_INET6) {
1136 #define FLOW_GET_NEW_TRIES 5
1140 static inline int GetUsedTryLockBucket(FlowBucket *fb)
1145 static inline int GetUsedTryLockFlow(
Flow *f)
1150 static inline uint32_t GetUsedAtomicUpdate(
const uint32_t val)
1187 #define STATSADDUI64(cnt, value) \
1189 StatsCounterAddI64(&tv->stats, dtv->cnt, (value)); \
1192 #define STATSADDUI64(cnt, value) StatsCounterAddI64(&tv->stats, dtv->cnt, (value));
1228 if (GetUsedTryLockBucket(fb) != 0) {
1239 if (GetUsedTryLockFlow(f) != 0) {
1245 if (StillAlive(f,
ts)) {
#define ESP_GET_SPI(esph)
Get the spi field off a packet.
FlowQueuePrivate work_queue
StatsCounterId counter_flow_icmp6
TmEcode OutputFlowLog(ThreadVars *tv, void *thread_data, Flow *f)
Run flow logger(s)
void ExceptionPolicyApply(Packet *p, enum ExceptionPolicy policy, enum PacketDropReason drop_reason)
#define ICMPV4_GET_EMB_PROTO(p)
#define IPV4_GET_RAW_IPDST_U32(ip4h)
uint32_t hashword(const uint32_t *k, size_t length, uint32_t initval)
bool TcpSessionPacketSsnReuse(const Packet *p, const Flow *f, const void *tcp_ssn)
struct Packet_::@34::@40 icmp_d
#define SC_ATOMIC_SET(name, val)
Set the value for the atomic variable.
@ PKT_DROP_REASON_FLOW_MEMCAP
uint32_t address_un_data32[4]
FlowQueuePrivate FlowSpareGetFromPool(void)
uint64_t PcapPacketCntGet(const Packet *p)
struct Flow_::@121::@128 esp
StatsCounterId counter_flow_spare_sync_empty
uint32_t FlowKeyGetHash(FlowKey *fk)
#define ICMPV4_DEST_UNREACH_IS_VALID(p)
uint16_t vlan_id[VLAN_MAX_LAYERS]
unsigned int SCFlowStorageSize(void)
#define SC_ATOMIC_ADD(name, val)
add a value to our atomic variable
#define FLOWLOCK_TRYWRLOCK(fb)
StatsCounterId counter_flow_get_used
struct FlowHashKey4_ FlowHashKey4
#define FBLOCK_TRYLOCK(fb)
uint32_t address_un_data32[4]
TcpStreamCnf stream_config
StatsCounterId counter_flow_spare_sync
SCTime_t TmThreadsGetThreadTime(const int idx)
#define FLOWLOCK_UNLOCK(fb)
void FlowWakeupFlowManagerThread(void)
struct Packet_::@32::@39 icmp_s
union Address_::@29 address
#define FLOW_CHECK_MEMCAP(size)
check if a memory alloc would fit in the memcap
uint32_t emerg_spare_sync_stamp
void StatsCounterAvgAddI64(StatsThreadContext *stats, StatsCounterAvgId id, int64_t x)
StatsCounterId counter_flow_memcap
void SCFlowRunFinishCallbacks(ThreadVars *tv, Flow *f)
#define FLOW_GET_NEW_TRIES
#define STATSADDUI64(cnt, value)
#define SCTIME_FROM_TIMESPEC(ts)
#define FLOWLOCK_WRLOCK(fb)
SC_ATOMIC_EXTERN(unsigned int, flow_prune_idx)
Per thread variable structure.
StatsCounterId counter_flow_active
void StatsCounterIncr(StatsThreadContext *stats, StatsCounterId id)
Increments the local counter.
struct FlowHashKey6_ FlowHashKey6
Flow * FlowQueuePrivateGetFromTop(FlowQueuePrivate *fqc)
StatsCounterAvgId counter_flow_spare_sync_avg
FlowProtoTimeout flow_timeouts_emerg[FLOW_PROTO_MAX]
StatsCounterId counter_flow_icmp4
bool TimeModeIsLive(void)
void FlowTimeoutsEmergency(void)
uint8_t FlowGetProtoMapping(uint8_t proto)
Function to map the protocol to the defined FLOW_PROTO_* enumeration.
#define FLOW_END_FLAG_TCPREUSE
#define FLOW_END_FLAG_EMERGENCY
#define FBLOCK_UNLOCK(fb)
int FlowClearMemory(Flow *f, uint8_t proto_map)
Function clear the flow memory before queueing it to spare flow queue.
#define STREAM_PKT_FLAG_TCP_SESSION_REUSE
Data structures and function prototypes for keeping state for the detection engine.
int RunmodeIsUnittests(void)
void FlowUpdateState(Flow *f, const enum FlowState s)
FlowQueuePrivate spare_queue
uint32_t FlowGetIpPairProtoHash(const Packet *p)
#define SCTIME_CMP_LT(a, b)
#define IPV4_GET_RAW_IPSRC_U32(ip4h)
struct Flow_::@123::@129 icmp_d
StatsCounterId counter_flow_udp
ExceptionPolicyCounters counter_flow_memcap_eps
Flow * FlowGetFlowFromHash(ThreadVars *tv, FlowLookupStruct *fls, Packet *p, Flow **dest)
Get Flow for packet.
enum ExceptionPolicy memcap_policy
union FlowAddress_::@120 address
StatsCounterId counter_flow_spare_sync_incomplete
Structure to hold thread specific data for all decode modules.
Flow * FlowGetFromFlowKey(FlowKey *key, struct timespec *ttime, const uint32_t hash)
Get or create a Flow using a FlowKey.
void * output_flow_thread_data
void FlowInit(ThreadVars *tv, Flow *f, const Packet *p)
uint16_t vlan_id[VLAN_MAX_LAYERS]
#define STREAM_PKT_FLAG_SET(p, f)
uint16_t vlan_id[VLAN_MAX_LAYERS]
#define FLOW_END_FLAG_TIMEOUT
uint16_t vlan_id[VLAN_MAX_LAYERS]
StatsCounterId counter_flow_total
struct Flow_::@121::@127 icmp_s
#define SC_ATOMIC_GET(name)
Get the value from the atomic variable.
uint16_t vlan_id[VLAN_MAX_LAYERS]
void FlowQueuePrivateAppendFlow(FlowQueuePrivate *fqc, Flow *f)
Flow * FlowAlloc(void)
allocate a flow
#define SCTIME_ADD_SECS(ts, s)
Flow * FlowGetExistingFlowFromFlowId(uint64_t flow_id)
Look for existing Flow using a flow id value.
#define FLOW_END_FLAG_FORCED
#define ICMPV4_IS_ERROR_MSG(type)
StatsCounterId eps_id[EXCEPTION_POLICY_MAX]
void FlowSetupPacket(Packet *p)
prepare packet for a life with flow Set PKT_WANTS_FLOW flag to indicate workers should do a flow look...
union PacketL4::L4Vars vars
FlowThreadId thread_id[2]
#define SC_ATOMIC_OR(name, val)
Bitwise OR a value to our atomic variable.
StatsCounterId counter_flow_tcp_reuse
#define CmpFlowMisc(x, y)
StatsCounterId counter_flow_tcp