当前位置: 首页 > article >正文

【网络入侵检测】基于Suricata源码分析运行模式(Runmode)

【作者主页】只道当时是寻常

【专栏介绍】Suricata入侵检测。专注网络、主机安全,欢迎关注与评论。


1. 概要

👋 在 Suricata 中抽象出线程、线程模块和队列三个概念:线程类似进程,可多线程并行执行操作;监听、解码、检测等网络操作被抽象为模块,由线程执行,而不同的线程可以执行一个或多个不同的模块;队列实现线程间数据包流转,三者的不同组合构成 Suricata 的运行模式。

2. 运行模式

Suricata的运行模式可通过参数 --list-runmodes 显示,执行后结果汇总成表格如下所示,RunMode Type 和 Custom Mode 配置同时指定一种运行模式。

RunMode TypeCustom ModeDescription
PCAP_DEVsingle单线程pcap实时模式
autofp多线程pcap实时模式。来自每个流的数据包被分配到一个一致的检测线程
workers工作者pcap实时模式,每个线程执行从采集到日志记录的所有任务
PCAP_FILEsingle单线程pcap文件模式
autofp多线程pcap文件模式。来自每个流的数据包被分配到一个一致的检测线程
PFRINGautofp多线程pfring模式。来自每个流的数据包被分配到一个检测线程,不同于"pfring_auto"模式,在"pfring_auto"模式中,同一流的数据包可以由任何检测线程处理
single单线程pfring模式
workers工作者pfring模式,每个线程执行从采集到日志记录的所有任务
NFQautofp考虑流的多线程NFQ IPS模式
workers每队列一个线程的多队列NFQ IPS模式
NFLOGautofp多线程nflog模式
single单线程nflog模式
workers工作者nflog模式
IPFWautofp考虑流的多线程IPFW IPS模式
workers每队列一个线程的多队列IPFW IPS模式
ERF_FILEsingle单线程ERF文件模式
autofp多线程ERF文件模式。来自每个流的数据包被分配到一个检测线程
ERF_DAGautofp多线程DAG模式。来自每个流的数据包被分配到一个检测线程,不同于"dag_auto"模式,在"dag_auto"模式中,同一流的数据包可以由任何检测线程处理
single单线程DAG模式
workers工作者DAG模式,每个线程执行从采集到日志记录的所有任务
AF_PACKET_DEVsingle单线程af-packet模式
workers工作者af-packet模式,每个线程执行从采集到日志记录的所有任务
autofp多套接字AF_PACKET模式。来自每个流的数据包被分配到一个检测线程。
AF_XDP_DEVsingle单线程af-xdp模式
workers工作者af-xdp模式,每个线程执行从采集到日志记录的所有任务
NETMAP(DISABLED)single单线程netmap模式
workers工作者netmap模式,每个线程执行从采集到日志记录的所有任务
autofp多线程netmap模式。来自每个流的数据包被分配到一个检测线程。
DPDK(DISABLED)workers工作者DPDK模式,每个线程执行从采集到日志记录的所有任务
UNIX_SOCKETsingleUnix套接字模式
autofpUnix套接字模式
WINDIVERTautofp按流负载均衡的多线程WinDivert IPS模式

2.1 single模式

Suricata 里,single 模式被限制为只能监听单个网卡设备,并且仅仅创建一个线程来执行数据包嗅探解码以及 flow 解析等操作。由于这种单线程的处理方式,开发者在调试和测试过程中能够更方便地进行操作,所以该模式在开发过程中十分实用。

2.2 autofp模式

Suricata autofp 模式中,捕获解码线程负责数据包的捕获与解码,并通过队列将其传递至 "flow worker" 线程进行处理和响应。此模式支持一个或者多个网卡的同时监听,每个网卡设备可通过配置文件设定数据包嗅探线程的数量(例如pcap-interface:ens33-threads配置),而"flow worker"线程处理数量取决于配置文件中threading-cpu-affinity-worker-cpu-set-threads数量。

2.3 workers模式

通常情况下,workers 运行模式的性能表现最佳。在此模式中,每个数据包处理线程都包含完整的数据包处理流程(即完整的数据包管道)。此模式支持同时监听多个网卡设备,且可通过配置文件配置指定每个网卡设备线程数量。

3. 源码分析

3.1 注册运行模式

Suricata 中,RunModeRegisterRunModes 函数是各种运行模式注册函数,该函数会将章节2中通过--list-runmodes 显示的所有运行模式注册到一个全局变量 runmodes 中,该变量的结构和定义如下所示:

typedef struct RunModes_ {int cnt;RunMode *runmodes;
} RunModes;static RunModes runmodes[RUNMODE_USER_MAX];

下面是 RunModeRegisterRunModes 函数示意图:

3.2 注册线程模块

Suricata 中将不同的功能封装成模块的形式,通过将这些模块分配给不同的线程从而显示对线程所实现功能的控制。RegisterAllModules 函数是注册模块的入口函数,将所有的模块功能存储在一个名为 tmm_modules 的全局变量中,该变量的声明与定义如下所示:

typedef struct TmModule_ {const char *name;/** thread handling */TmEcode (*ThreadInit)(ThreadVars *, const void *, void **);void (*ThreadExitPrintStats)(ThreadVars *, void *);TmEcode (*ThreadDeinit)(ThreadVars *, void *);/** the packet processing function */TmEcode (*Func)(ThreadVars *, Packet *, void *);TmEcode (*PktAcqLoop)(ThreadVars *, void *, void *);/** terminates the capture loop in PktAcqLoop */TmEcode (*PktAcqBreakLoop)(ThreadVars *, void *);/** does a thread still have tasks to complete before it can be killed?*  \retval bool*  \param tv threadvars*  \param thread_data thread module thread data (e.g. FlowWorkerThreadData for FlowWorker) */bool (*ThreadBusy)(ThreadVars *tv, void *thread_data);TmEcode (*Management)(ThreadVars *, void *);/** global Init/DeInit */TmEcode (*Init)(void);TmEcode (*DeInit)(void);
#ifdef UNITTESTSvoid (*RegisterTests)(void);
#endifuint8_t cap_flags; /**< Flags to indicate the capability requirement ofthe given TmModule *//* Other flags used by the module */uint8_t flags;
} TmModule;TmModule tmm_modules[TMM_SIZE];

RegisterAllModules 函数实现如下所示:

TmModuleReceivePcapRegister 为例,Suricata 会将网络报文的嗅探功能封装成一个模块,模块中包含初始化函数、销毁函数、主函数等信息,并通过注册函数将其添加到tmm_modules数组中。 TmModuleReceivePcapRegister 函数实现如下所示:

/*** \brief Registration Function for ReceivePcap.*/
void TmModuleReceivePcapRegister (void)
{tmm_modules[TMM_RECEIVEPCAP].name = "ReceivePcap";tmm_modules[TMM_RECEIVEPCAP].ThreadInit = ReceivePcapThreadInit;tmm_modules[TMM_RECEIVEPCAP].ThreadDeinit = ReceivePcapThreadDeinit;tmm_modules[TMM_RECEIVEPCAP].PktAcqLoop = ReceivePcapLoop;tmm_modules[TMM_RECEIVEPCAP].PktAcqBreakLoop = ReceivePcapBreakLoop;tmm_modules[TMM_RECEIVEPCAP].ThreadExitPrintStats = ReceivePcapThreadExitStats;tmm_modules[TMM_RECEIVEPCAP].cap_flags = SC_CAP_NET_RAW;tmm_modules[TMM_RECEIVEPCAP].flags = TM_FLAG_RECEIVE_TM;
#ifdef UNITTESTStmm_modules[TMM_RECEIVEPCAP].RegisterTests = SourcePcapRegisterTests;
#endif
}

http://www.lryc.cn/news/2379183.html

相关文章:

  • AI日报 - 2025年05月19日
  • Spring源码主线全链路拆解:从启动到关闭的完整生命周期
  • Linux常用命令(十四)
  • 规则联动引擎GoRules初探
  • 基于OpenCV中的图像拼接方法详解
  • AI大模型学习二十六、使用 Dify + awesome-digital-human-live2d + ollama + ChatTTS打造数字人
  • HTML-3.2 表格的跨行跨列(课表制作实例)
  • Spring Cloud Sentinel 快速入门与生产实践指南
  • 系统架构设计(十):结构化编程
  • 标准差和方差是什么
  • 【周输入】510周阅读推荐-3
  • Android平台GB28181设备接入与功能支持详解
  • mvc-ioc实现
  • Windows 11 C:\Windows\Web\Wallpaper
  • 医疗信息系统安全防护体系的深度构建与理论实践融合
  • Compose笔记(二十五)--Brush
  • Spring Web MVC————入门(3)
  • HTML 属性详解
  • NC61 两数之和【牛客网】
  • Java生成可控的Word表格功能开发
  • Node.js中的洋葱模型
  • Spring Boot 中 MyBatis 与 Spring Data JPA 的对比介绍
  • 如何分析动态采样引起的计划不稳定 | OceanBase SQL 调优实践
  • 如何实现RTSP和RTMP低至100-200ms的延迟:直播SDK的技术突破
  • symfonos: 2靶场
  • 【图像生成大模型】Step-Video-T2V:下一代文本到视频生成技术
  • C语言中的指针:从基础到进阶实战
  • 深度学习推理引擎---ONNX Runtime
  • JAVA Spring MVC+Mybatis Spring MVC的工作流程*,多表连查
  • ctr查看镜像