SOCK_RAW 写一个简单的网络嗅探器

2024-01-09 21:08
文章标签 简单 网络 sock raw 嗅探器

本文主要是介绍SOCK_RAW 写一个简单的网络嗅探器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

之前有一篇写UDP反射放大攻击的
为什么IP头和UDP头可以伪造,看下面的TCP自己想一想就知道了
一般情况下我们使用的都是封装好的TCP或UDP协议。但事实上我们可以自己封装和解析。
注意:SOCK_RAW 程序必须有管理员权限

简单实现代码

/*
通常的套接字只能响应与自己MAC地址相匹配的或是以广播行事发出的数据帧。
对于其他形式的数据帧网路接口采取的动作是直接丢弃。为了使网卡接收所有经过它的封包,
要将其设为混杂模式。在用户模式下,对网卡混杂模式的设置是通过原始套接字来实现的。创建原始套接字之后,将它绑定到一个明确的本地地址,然后向套接字发送SIO_RCVALL控制命令,让它接收所有的IP包,这样网卡便进入了混杂模式。*/#include "pch.h"
#include <iostream>
#include <WinSock2.h>
#include <ws2ipdef.h>
#include <mstcpip.h> //SIO_RCVALL
#include <process.h>
#pragma warning(disable:4996)
#pragma comment(lib,"ws2_32.lib")typedef struct _IPHeader {u_char VIHL; //版本和首部长度 各占4bitu_char ToS; //服务类型u_short TotalLen; //总长度u_short ID; //标识号u_short Frag_Flags; //片偏移量u_char TTL; //生存时间u_char Protocol; //协议u_short Checksum; //首部校验和//ULONG SrcIP;//ULONG DestIP;struct in_addr SrcIP; //源IP地址struct in_addr DestIP; //目的地址
}IPHDR, *PIPHDR;/*TCP头*/
typedef struct _tcpheader {USHORT sourcePort; //来源端口USHORT destinationPort;//目标端口ULONG sequenceNumber; // 32位序列号ULONG acknowledgeNumber; //32位确认号//下面2个 共占16位UCHAR dataOffset;//4位首部长度/6位保留字UCHAR flags;//6位标志位USHORT windows;//16位窗口大小USHORT checksum;//16位校验和USHORT urgentPointer;//16位紧急数据偏移量}TCP_HDR, *PTCP_HDR;//解析TCP封包
void DecodeTCPPacket(char* pData) {PTCP_HDR pTcphdr = (PTCP_HDR)pData;//取出端口printf_s("Port: %d -> %d \n", ntohs(pTcphdr->sourcePort), ntohs(pTcphdr->destinationPort));//下面还可以根据端口进一步解析应用层协议:switch (ntohs(pTcphdr->destinationPort)) {case 80: {printf_s("这是一个80端口 具体信息:%s\n", pData+sizeof(TCP_HDR));break;}case 21: {printf_s("这是一个21端口 具体信息:%s\n", pData + sizeof(TCP_HDR));break;}case 8080: {printf_s("这是一个8080端口 具体信息:%s\n", pData + sizeof(TCP_HDR));break;}}
}//解析IP封包
void DecodeIPPacket(char * pData) {PIPHDR pIphdr = (PIPHDR)pData;char szSourceIP[32] = { 0 }, szDestIP[32] = {0};printf_s("---------------------\n");//从IP头中取出源IP地址和目的IP地址strcpy_s(szSourceIP, inet_ntoa(pIphdr->SrcIP));strcpy_s(szDestIP, inet_ntoa(pIphdr->DestIP));printf_s("%s -> %s \n", szSourceIP, szDestIP);//IP头长度//因为IP头 版本号和首部长度各占4位,先取低4位长度//IP头我们封装的时候是这样://pIphdr->VIHL = (4 << 4 | (sizeof(IPHDR) / sizeof(ULONG)));int nHeaderLen = (pIphdr->VIHL & 0xf) * sizeof(ULONG);switch (pIphdr->Protocol) {case IPPROTO_TCP:{//解析TCP封包DecodeTCPPacket(pData + nHeaderLen);break;}case IPPROTO_UDP:{break;}case IPPROTO_ICMP:{break;}}
}int main()
{WSADATA wsaData;WSAStartup(MAKEWORD(2, 2), &wsaData);SOCKET s = socket(AF_INET, SOCK_RAW, IPPROTO_IP);if (s == INVALID_SOCKET) {printf_s("请管理员运行\n");return -1;}//获取本地IP地址char szHostName[56];SOCKADDR_IN sin;struct hostent* pHost;gethostname(szHostName,56);if( (pHost=gethostbyname((char*)szHostName)) == NULL){printf_s("gethostbyname失败\n");return -1;}sin.sin_family = AF_INET;sin.sin_port = htons(0);//因为我有一大堆网卡,所以我这里是第三个是我需要的for (ULONG i = 0; pHost->h_addr_list[i] + pHost->h_length < pHost->h_name;) {printf_s("我的IP[%d]:%s\n", ++i , inet_ntoa(*((in_addr *)pHost->h_addr_list[i])));}memcpy_s(&sin.sin_addr.S_un.S_addr, 4U, pHost->h_addr_list[3], pHost->h_length);printf_s("准备绑定到IP:%s\n", inet_ntoa(sin.sin_addr));//在调用ioctl之前,套接字必须绑定if (bind(s, (sockaddr*)&sin, sizeof(sin)) == SOCKET_ERROR){printf_s("Bind失败\n");return -1;}//设置SIO_RCVALLDWORD dwValue = 1;if (ioctlsocket(s, SIO_RCVALL, &dwValue) != 0) {printf_s("SIO_RCVALL 失败\n");return -1;}//开始接收封包char buff[1024];int nRet;while (TRUE) {nRet = recv(s, buff, 1024, 0);if (nRet > 0) {//解析IP封包DecodeIPPacket(buff);}}closesocket(s);WSACleanup();
}

这篇关于SOCK_RAW 写一个简单的网络嗅探器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/588436

相关文章

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要

MyBatis框架实现一个简单的数据查询操作

《MyBatis框架实现一个简单的数据查询操作》本文介绍了MyBatis框架下进行数据查询操作的详细步骤,括创建实体类、编写SQL标签、配置Mapper、开启驼峰命名映射以及执行SQL语句等,感兴趣的... 基于在前面几章我们已经学习了对MyBATis进行环境配置,并利用SqlSessionFactory核

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

hdu2289(简单二分)

虽说是简单二分,但是我还是wa死了  题意:已知圆台的体积,求高度 首先要知道圆台体积怎么求:设上下底的半径分别为r1,r2,高为h,V = PI*(r1*r1+r1*r2+r2*r2)*h/3 然后以h进行二分 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#includ

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

usaco 1.3 Prime Cryptarithm(简单哈希表暴搜剪枝)

思路: 1. 用一个 hash[ ] 数组存放输入的数字,令 hash[ tmp ]=1 。 2. 一个自定义函数 check( ) ,检查各位是否为输入的数字。 3. 暴搜。第一行数从 100到999,第二行数从 10到99。 4. 剪枝。 代码: /*ID: who jayLANG: C++TASK: crypt1*/#include<stdio.h>bool h

uva 10387 Billiard(简单几何)

题意是一个球从矩形的中点出发,告诉你小球与矩形两条边的碰撞次数与小球回到原点的时间,求小球出发时的角度和小球的速度。 简单的几何问题,小球每与竖边碰撞一次,向右扩展一个相同的矩形;每与横边碰撞一次,向上扩展一个相同的矩形。 可以发现,扩展矩形的路径和在当前矩形中的每一段路径相同,当小球回到出发点时,一条直线的路径刚好经过最后一个扩展矩形的中心点。 最后扩展的路径和横边竖边恰好组成一个直

ASIO网络调试助手之一:简介

多年前,写过几篇《Boost.Asio C++网络编程》的学习文章,一直没机会实践。最近项目中用到了Asio,于是抽空写了个网络调试助手。 开发环境: Win10 Qt5.12.6 + Asio(standalone) + spdlog 支持协议: UDP + TCP Client + TCP Server 独立的Asio(http://www.think-async.com)只包含了头文件,不依

poj 1113 凸包+简单几何计算

题意: 给N个平面上的点,现在要在离点外L米处建城墙,使得城墙把所有点都包含进去且城墙的长度最短。 解析: 韬哥出的某次训练赛上A出的第一道计算几何,算是大水题吧。 用convexhull算法把凸包求出来,然后加加减减就A了。 计算见下图: 好久没玩画图了啊好开心。 代码: #include <iostream>#include <cstdio>#inclu

uva 10130 简单背包

题意: 背包和 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#include <stack>#include <vector>#include <queue>#include <map>