所有相同元组(协议,源IP,目的IP,源端口,目的端口)的包分组属于同一流。使用flow管理一个会话。
通常我们也指从一端到另一端的一次网络数据传输过程。其包含连接的建立,数据的传输。一条流往往指的是一次完成的数据传输,其包含了该过程中所有的数据包。但是一个连接可以有多个流,即在连接建立之后,可以有多次的数据传输。
suricata实现了流管理机制来回收与重复利用Flow。suricata使用三个线程维护这三个队列(Flow哈希表,Flow备用队列,Flow回收队列)。
其中:
①FlowManager线程:从流表中摘除超时的流放入回收队列,检查空闲队列长度是否为预设值,若过长则释放一部分,若过短则申请一部分。
// 根据配置创建flow_managers线程个数
void FlowManagerThreadSpawn()
{intmax_t setting = 1;(void)ConfGetInt("flow.managers", &setting);if (setting < 1 || setting > 1024) {FatalError(SC_ERR_INVALID_ARGUMENTS,"invalid flow.managers setting %"PRIdMAX, setting);}flowmgr_number = (uint32_t)setting;SCLogConfig("using %u flow manager threads", flowmgr_number);StatsRegisterGlobalCounter("flow.memuse", FlowGetMemuse);// 注册一个全局的原子的flow计数器flow_memusefor (uint32_t u = 0; u < flowmgr_number; u++) {char name[TM_THREAD_NAME_MAX];snprintf(name, sizeof(name), "%s#%02u", thread_name_flow_mgr, u+1);ThreadVars *tv_flowmgr = TmThreadCreateMgmtThreadByName(name,"FlowManager", 0);// 创建线程变量结构体BUG_ON(tv_flowmgr == NULL);if (tv_flowmgr == NULL) {FatalError(SC_ERR_FATAL, "flow manager thread creation failed");}if (TmThreadSpawn(tv_flowmgr) != TM_ECODE_OK) { // 根据生成的线程名创建线程FatalError(SC_ERR_FATAL, "flow manager thread spawn failed");}}return;
}
②FlowRecyler线程:从回收队列中摘除Flow,调用注册流输出模块的回掉,清空Flow信息,插入回收队列。
③FlowWorker线程:
Flow哈希表: 这个是FlowBucket *flow_hash 数组,数据大小为配置中的 flow_config 的 hash_size,其在数组中的每一个FlowBucket元素中都挂了一串之前数据包对应的Flow链队列。
Flow备用队列: static FlowSparePool *flow_spare_pool 备用池链队列,其每一个链pool中都挂载一个备用Flow队列。
Flow回收队列: FlowQueue flow_recycle_q 回收队列,挂载一个备用Flow队列。
①当一个数据包来到时,计算数据包对应的hash值
②在后续的模块中将使用这个hash 在Flow哈希表中找到 FlowBucket ,不同会话的hash值可能相同,所以不是比较hash相同来定位Flow,而是在篮子里顺序查找具有相同五元组和vlan的Flow,找到则利用此Flow
③若Flow哈希表中没有,则在Flow备用队列取出一个Flow, 或是新建一个Flow使用,同时把这个Flow插入到Flow哈希表