1 netsniff安装与使用

首先直接下载源码包进行部署

安装一些前置包(安装完成的自动忽略)

sudo apt install pkg-config

sudo apt install libcli

sudo apt install libGeoIP

sudo apt install libsodium.c

sudo apt install ncurses

安装

cd netsniff-ng-master

./configure

make

sudo make install

进行抓包命令

sudo netsniff-ng --in any

[any]括号内是指任意网卡,当然也可以指定网卡

2 源码中的常用函数

所需要用到的函数strncmp(const char *str1, const char *str2, size_t n)

把 str1 和 str2 进行比较,最多比较前 n 个字节。

str1 -- 要进行比较的第一个字符串。

str2 -- 要进行比较的第二个字符串。

n -- 要比较的最大字符数

如果返回值 < 0,则表示 str1 小于 str2。

如果返回值 > 0,则表示 str2 小于 str1。

如果返回值 = 0,则表示 str1 等于 str2。

strcmp函数用来比较字符串(区分大小写)

int strcmp(const char *s1, const char *s2);

strcmp()首先将s1 第一个字符值减去s2 第一个字符值,若差值为0 则再继续比较下个字符,若差值不为0 则将差值返回。例如字符串"Ac"和"ba"比较则会返回字符"A"(65)和'b'(98)的差值(-33)。

例如字符串"Ac"和"ba"比较则会返回字符"A"(65)和'b'(98)的差值(-33)。

device_mtu(ctx.device_in)网卡设备输入的,获取最大输入单元

pcap_dev_to_linktype(ctx.device_in),获取物理层输入的型号

socket函数

int socket(int domain, int type, int protocol);

socket()用于创建一个socket描述符(socket descriptor),它唯一标识一个socket。这个socket描述字跟文件描述字一样,后续的操作都有用到它,把它作为参数,通过它来进行一些读写操作。

  • domain:即协议域,又称为协议族(family)。常用的协议族有,AF_INET、AF_INET6、AF_LOCAL(或称AF_UNIX,Unix域socket)、AF_ROUTE等等。协议族决定了socket的地址类型,在通信中必须采用对应的地址,如AF_INET决定了要用ipv4地址(32位的)与端口号(16位的)的组合、AF_UNIX决定了要用一个绝对路径名作为地址。
  • type:指定socket类型。常用的socket类型有,SOCK_STREAM、SOCK_DGRAM、SOCK_RAW、SOCK_PACKET、SOCK_SEQPACKET等等(socket的类型有哪些?)。
  • protocol:故名思意,就是指定协议。常用的协议有,IPPROTO_TCP、IPPTOTO_UDP、IPPROTO_SCTP、IPPROTO_TIPC等,它们分别对应TCP传输协议、UDP传输协议、STCP传输协议、TIPC传输协议。

3 初始化工作和主要的数据结构

ctx是保存用户操作的功能结构,程序运行时根据用户的输入进行初始化操作初始化。

struct ctx {
char *device_in, *device_out, *device_trans, *filter, *prefix;
int cpu, rfraw, dump, print_mode, dump_dir, packet_type, lo_ifindex;
unsigned long kpull, dump_interval, tx_bytes, tx_packets;
size_t reserve_size;
bool randomize, promiscuous, enforce, jumbo, dump_bpf, hwtimestamp, verbose;
enum pcap_ops_groups pcap;
enum dump_mode dump_mode;
uid_t uid;
gid_t gid;
uint32_t link_type, magic;
uint32_t fanout_group, fanout_type;
uint64_t pkts_seen, pkts_recvd, pkts_drops;
uint64_t pkts_recvd_last, pkts_drops_last, pkts_skipd_last;
unsigned long overwrite_interval, file_number;
};

device_in输入的网络接口或者读入的数据包名,如eth0或者xxx.pcap

device_out对应device_in,为输出

filter 是否启用过滤选项,如只显示ICMP包

cpu 指定绑定的CPU

int dump; //选择终端输出或是文件输出

/* dissector */

link_type;

print_mode; 打印类型

/* 0 for automatic, > 0 for manual */

reserve_size;

packet_type;

randomize 指定为随机模式

promiscuous 网口为混杂模式,接收所有网卡包

pcap_ops_groups pcap 包类型

kpull

jumbo_support 是否指定大容量帧数

重点函数1

在recv_only_or_dump()函数,会初始化ring的数据结构

ring的数据结构

struct ring {
struct iovec *frames; //数据帧,不同于网络术语的“帧”
struct tpacket_req layout; //frames和block的块数页数统计
struct sockaddr_ll s_ll; //常见的socket数据结构
uint8_t *mm_space; //映射的内存空间的起始地址
size_t mm_len; //所有block的字节长度 } __cacheline_aligned; //字节对齐属
struct protocol {
/* Needs to be filled out by user */
const unsigned int key;
void (*print_full)(struct pkt_buff *pkt);
void (*print_less)(struct pkt_buff *pkt);
/* Used by program logic */
struct protocol *next;
void (*process) (struct pkt_buff *pkt);
};
void ring_rx_setup(struct ring *ring, int sock, size_t size, int ifindex,
struct pollfd *poll, bool v3, bool jumbo_support,
bool verbose, uint32_t fanout_group, uint32_t fanout_type)
{
memset(ring, 0, sizeof(*ring));
setup_rx_ring_layout(sock, ring, size, jumbo_support, v3);
create_rx_ring(sock, ring, verbose);
mmap_ring_generic(sock, ring);
alloc_rx_ring_frames(sock, ring);
bind_ring_generic(sock, ring, ifindex, false);
join_fanout_group(sock, fanout_group, fanout_type);
prepare_polling(sock, poll);
}

首先是创建一个环形缓冲区static void create_rx_ring(int sock, struct ring *ring, bool verbose)

ret = setsockopt(sock, SOL_PACKET, PACKET_RX_RING, &ring->raw,layout_size);

当创建好一个原始的socket后,通过传递tpacket_req结构给通用套接字选项函数setsockopt()设置为一个环形的缓冲区,选项字段一定要是PACKET_RX_RING。



根据用户自定义或者初始化自动分配,分配的大小和数量已经在tpacket_req数据结构中。如上图所示,分配很多个block,每个block包含若干个frame,block一定是frame的整数倍,根据分配的大小和数量初始化每个指向frame的指针,下面alloc_rx_ring_frames()函数既是初始化每个指向frame的指针。

其中

static void alloc_rx_ring_frames(int sock, struct ring *ring)
{
bool v3 = is_tpacket_v3(sock);
alloc_ring_frames_generic(ring, rx_ring_get_num(ring, v3),
rx_ring_get_size(ring, v3));
}

mmap_ring_generic(sock, ring)函数则是做映射内存到用户空间

关于协议的结构体下图所示:

struct protocol {
/* Needs to be filled out by user */
const unsigned int key;
void (*print_full)(struct pkt_buff *pkt);
void (*print_less)(struct pkt_buff *pkt);
/* Used by program logic */
struct protocol *next;
void (*process) (struct pkt_buff *pkt);
};
struct pkt_buff {
/* invariant: head <= data <= tail */
uint8_t *head;
uint8_t *data;
uint8_t *tail; struct protocol *dissector;
uint32_t link_type;
struct sockaddr_ll *sll;
};

pkt_buff结构体内用了protocol类型的结构体, 并且命名为dissector。

4 数据操作部分及DPI的解析

dissector_entry_point(packet, hdr->tp_h.tp_snaplen,ctx->link_type, ctx->print_mode,&hdr->s_ll);这个函数首先通过获取linktype,从而通过使用dissector_main()函数获取协议protocol



dissector_main()该函数的代码是DPI 的核心逻辑


static void dissector_main(struct pkt_buff *pkt, struct protocol *start,struct protocol *end)
{
struct protocol *dissector; if (!start)
return;
for (pkt->dissector = start; pkt->dissector; ) {
if (unlikely(!pkt->dissector->process))
break; dissector = pkt->dissector;
pkt->dissector = NULL;
dissector->process(pkt);
}
if (end && likely(end->process))
end->process(pkt);
}

read_pcap()函数解析

frame_map是每一个frame的头部信息

struct frame_map {
struct tpacket2_hdr tp_h __aligned_tpacket;
struct sockaddr_ll s_ll __align_tpacket(sizeof(struct tpacket2_hdr));
};

所以在该函数中会定义一个struct frame_map fm; fm这样的数据结构。

__pcap_io 是一个宏定义, pcap_ops[ctx->pcap],是一个数组(数组的个数是由ctx指定)。

接下去就是根据__pcap_io 进程执行初始化等一系列操作,例如init_once_pcap、push_fhdr_pcap、prepare_access_pcap。

struct pcap_file_ops {
void (*init_once_pcap)(bool enforce_prio);
int (*pull_fhdr_pcap)(int fd, uint32_t *magic, uint32_t *linktype);
int (*push_fhdr_pcap)(int fd, uint32_t magic, uint32_t linktype);
int (*prepare_access_pcap)(int fd, enum pcap_mode mode, bool jumbo);
ssize_t (*write_pcap)(int fd, pcap_pkthdr_t *phdr, enum pcap_type type,
const uint8_t *packet, size_t len);
ssize_t (*read_pcap)(int fd, pcap_pkthdr_t *phdr, enum pcap_type type,
uint8_t *packet, size_t len);
void (*prepare_close_pcap)(int fd, enum pcap_mode mode);
void (*fsync_pcap)(int fd);
};

附件图:

netsniff使用的更多相关文章

  1. netsniff恶意流量识别和匹配解读

    代码整体框架 流量扫描函数调用 加载配置文件的代码调用 获取扫描的条数 重点匹配函数 流量eg:'\x00\x04\x00\x01\x00\x06\x00\x16>\x10\x1d>SW\ ...

  2. nodejs 下载网页及相关资源文件

    功能其实很见简单,通过 phantomjs.exe 采集 url 加载的资源,通过子进程的方式,启动nodejs 加载所有的资源,对于css的资源,匹配css内容,下载里面的url资源 当然功能还是很 ...

  3. 关于kali linux系统的简单工具

    Linux系统中关于几个重要目录的原英文解释: /etc/: Contains configuration files of the installed tools /opt/: Contains M ...

  4. [转]基于phantomJS实现web性能监控

    1.web性能监控背景描述 上期分享的<Web性能监控自动化探索之路–初识WebPageTest>从依赖webpagetest的角度给出了做性能日常检查的方案,但由于依赖结构相对复杂我们需 ...

  5. Phantomjs实现后端生成图片文件

    目录 PhantomJS简介 了解rasterize.js 使用方法 今天,给大家分享一个Java后端利用Phantomjs实现生成图片的功能,同学们使用的时候,可以参考下! PhantomJS简介 ...

  6. DDOS流量攻击

    0x01 环境 包含2台主机 attact 作为攻击方,使用Centos7.2 windows_server ,用于被攻击,同时抓包分析流量 ,任意版本均可.安装wireshark,用于抓包 0x02 ...

随机推荐

  1. Learning Attention-based Embeddings for Relation Prediction in Knowledge Graphs

    这篇论文试图将GAT应用于KG任务中,但是问题是知识图谱中实体与实体之间关系并不相同,因此结构信息不再是简单的节点与节点之间的相邻关系.这里进行了一些小的trick进行改进,即在将实体特征拼接在一起的 ...

  2. Go 包管理历史以及 Go mod 使用

    之前也写过 Go 管理依赖工具 godep 的使用,当时看 godep 使用起来还是挺方便,其原因主要在于有总比没有强.关于依赖管理工具其实还是想从头聊聊这个需求以及大家做这个功能的各种出发点. GO ...

  3. struts2学习总结

    https://www.cnblogs.com/wkrbky/p/5891237.html https://www.cnblogs.com/printN/p/6434526.html https:// ...

  4. linux下生成动态库和链接动态库

    1.生成动态库 src/test.h #ifndef _TEST_H_HH #define _TEST_H_HH void print(); #endif src/test.cpp #include ...

  5. JAVA学习第一阶段(2)

    21. ArrayList类和LinkList类 注意:add,contains,remove,retain(取交集) 都可以+all表示对集合的操作 ArrayList c = new ArrayL ...

  6. linux上性能调优常用命令及简介

    1.综合命令:nmon.top:topas(aix) d :磁盘相关 c:cpu相关 m:内存相关 2.磁盘 2.1 测试顺序写性能dd if=/dev/zero of=/cdr/test.data ...

  7. Python使用进程制作爬虫

    详情点我跳转 关注公众号"轻松学编程"了解更多. 1.进程 1.进程的概念 什么是进程->CPU在同一时刻只能处理一个任务,只是因为cpu执行速度很快. cpu在各个任务之间 ...

  8. JavaWeb中的关于html、jsp、servlet下的路径问题

    1 前言 本文将对近期项目练习中出现的关于文件路径的问题进行分析和总结,主要涉及html页面中的href及ajax指向路径.jsp页面中href指向路径及servlet转发或重定向路径等内容,本文的分 ...

  9. 10万用户一年365天的登录情况如何用redis存储,并快速检索任意时间窗内的活跃用户

    1.redsi的bitmap数据结构介绍 bitmap本质上是一个string类型,只是他操作的是string的某个位是0还是1. setbit和getbit 两条命令是对字符串的位操作.每个位只能是 ...

  10. 配置交换机基于接口划分VLAN(接入层设备作为网关)

    组网图形 简介 划分VLAN的方式有:基于接口.基于MAC地址.基于IP子网.基于协议.基于策略(MAC地址.IP地址.接口).其中基于接口划分VLAN,是最简单,最常见的划分方式. 基于接口划分VL ...