原始套接字-自定义IP首部和TCP首部
/* =====================================================================================
*
* Filename: raw.c
* Description: 使用原始套接字发送TCP协议,并外带自己的数据。
*
* ====================================================================================*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <netinet/tcp.h>
#include <arpa/inet.h>
#define DATA "hello"
#define PACKET_SIZE sizeof(struct iphdr) + sizeof(struct tcphdr) + sizeof(DATA) /*---------------------------------------------------------
Function Name : check_sum()
Descrypthon : 校验和计算,摘自UNP源码
------------------------------------------------------------*/
unsigned short check_sum(unsigned short *addr, int len)
{
int nleft = len;
int sum = ;
unsigned short *w = addr;
short answer = ;
while (nleft > )
{
sum += *w++;
nleft -=;
}
if (nleft == )
{
*(unsigned char *)(&answer) = *(unsigned char *)w;
sum += answer;
}
sum = (sum >> ) + (sum & 0xffff);
sum += (sum >> );
answer = ~sum;
return answer;
} /*---------------------------------------------------------
Function Name : init_socket()
Descrypthon : 初始化socket,使用原始套接字
parameter : P1 一个待初始化的原始套接字,P2 待初始化的目标地址结构,P3 目标地址,P4 目标端口
return : 返回一个原始套接字,在函数体内将目标地址结构进行初始
------------------------------------------------------------*/
int init_socket(int sockfd, struct sockaddr_in *target,const char *dst_addr, const char *dst_port)
{
const int flag = ;
//目标协议簇
target->sin_family = AF_INET;
//目标端口
target->sin_port = htons(atoi(dst_port)); //将dst_addr中的ASCII-IP地址更新到target->sin_addr结构中
if (inet_aton(dst_addr, &target->sin_addr) == )
{
perror("inet_aton fail\n");
exit(-);
}
//初始化原始套接字
if((sockfd = socket(AF_INET, SOCK_RAW, IPPROTO_TCP)) < )
{
perror("error");
exit(-);
}
//设置套接字×××
if (setsockopt(sockfd,IPPROTO_IP, IP_HDRINCL, &flag, sizeof(flag)) < )
{
perror("setsockopt fail \n");
exit(-);
}
return sockfd;
} /*---------------------------------------------------------------
Function Name : buile_iphdr()
Descrypthon : 构建IP头部数据, 源地址使用伪随机地址
-----------------------------------------------------------------*/
void buile_iphdr(struct sockaddr_in *target, char *buffer)
{
struct iphdr *ip = (struct iphdr *)(buffer);
ip->version = ;//版本
ip->ihl = ;//首部长度 5*4 = 20
ip->tos = ;//8位服务类型
ip->tot_len = htons(PACKET_SIZE);//16位总长度
ip->id = ;//16位标识符
ip->frag_off = ;//3位标志
ip->ttl = ;//生存时间
ip->protocol = IPPROTO_TCP;//协议
ip->check = check_sum((unsigned short *)ip, sizeof(struct iphdr) + sizeof(DATA));//16位首部校验和
ip->saddr = random();//源ip地址
ip->daddr = target->sin_addr.s_addr;//目标ip地址
} /*---------------------------------------------------------------
Function Name : buile_tcphdr()
Descrypthon : 构建TCP头部信息,并加入一些自己的数据,然后进行
校验计算。
-----------------------------------------------------------------*/
void buile_tcphdr(struct sockaddr_in *target, const char *src_port, char *buffer)
{
struct tcphdr *tcp = (struct tcphdr *)(buffer);
tcp->source = htons(atoi(src_port));//16位源端口号
tcp->dest = target->sin_port;//16位目的端口号
tcp->seq = random();//32位序号
tcp->doff = ;//
tcp->syn = ;//同步序号
buffer += sizeof(struct tcphdr);
tcp->check = check_sum((unsigned short *)tcp, sizeof(struct tcphdr) + sizeof(DATA));//16位检验和
memcpy(buffer, DATA, sizeof(DATA));//将DATA中的数据拷贝sizeof(DATA)字节到buffer所指的地址中
}
int main(int argc, const char *argv[])
{
char *buffer;
char *buffer_head = NULL;
int sockfd = ;
struct sockaddr_in *target;
if (argc != )
{
printf("usage: destination addresss, destination port, source port \n");
exit(-);
}
const char *dst_addr = argv[];
const char *dst_port = argv[];
const char *src_port = argv[]; target = calloc(sizeof(struct sockaddr_in),);
//calloc在动态分配完内存后,自动初始化该内存空间为零,而malloc不初始化,里边数据是随机的垃圾数据。
buffer = calloc(PACKET_SIZE, );
buffer_head = buffer; //初始化套接字
sockfd = init_socket(sockfd, target, dst_addr, dst_port);
//创建IP首部
buile_iphdr(target, buffer);
buffer += sizeof(struct iphdr);//指针下移
//创建TCP首部
buile_tcphdr(target, src_port, buffer);
//发送
sendto(sockfd, buffer_head, PACKET_SIZE, ,(struct sockaddr *)target, sizeof(struct sockaddr_in)); //下两行是对calloc申请的释放
free(buffer_head);
free(target);
return ;
}
原始套接字-自定义IP首部和TCP首部的更多相关文章
- UNP——原始套接字
1.原始套接字的用处 使用原始套接字可以构造或读取网际层及其以上报文. 具体来说,可以构造 ICMP, IGMP 协议报文,通过开启 IP_HDRINCL 套接字选项,进而自定义 IPv4首部. 2. ...
- Linux系统编程(37)—— socket编程之原始套接字
原始套接字的特点 原始套接字(SOCK_RAW)可以用来自行组装IP数据包,然后将数据包发送到其他终端.也就是说原始套接字是基于IP数据包的编程(SOCK_PACKET是基于数据链路层的编程).另外, ...
- UNIX网络编程——原始套接字SOCK_RAW
实际上,我们常用的网络编程都是在应用层的报文的收发操作,也就是大多数程序员接触到的流式套接字(SOCK_STREAM)和数据包式套接字(SOCK_DGRAM).而这些数据包都是由系统提供的协议栈实现, ...
- raw_socket(原始套接字)以及普通socket使用终极总结
一.传输层socket(四层socket,普通socket) 可参考本人以下博客: Windows Socket编程之UDP实现大文件的传输:http://blog.csdn.net/luchen ...
- 005.TCP--拼接TCP头部IP头部,实现TCP三次握手的第一步(Linux,原始套接字)
一.目的: 自己拼接IP头,TCP头,计算效验和,将生成的报文用原始套接字发送出去. 若使用tcpdump能监听有对方服务器的包回应,则证明TCP报文是正确的! 二.数据结构: TCP首部结构图: s ...
- 004.UDP--拼接UDP数据包,构造ip头和udp头通信(使用原始套接字)
一.大致流程: 建立一个client端,一个server端,自己构建IP头和UDP头,写入数据(hello,world!)后通过原始套接字(SOCK_RAW)将包发出去. server端收到数据后,打 ...
- python使用原始套接字 解析原始ip头数据
使用底层套接字解码底层流量,是这次做的重点工作. 首先来捕获第一个包 # coding:utf-8import socket # 监听的主机IP host = "192.168.1.100& ...
- C++ Win 32 使用原始套接字获取所有ip数据包并分析(包括ping包)
/*页面编码:GBK 开发环境 VS2019 */ #define _WINSOCK_DEPRECATED_NO_WARNINGS#include <iostream>#include&l ...
- linux原始套接字(4)-构造IP_UDP
一.概述 同上一篇tcp一样,udp也是封装在ip报文里面.创建UDP的原始套接字如下: (soc ...
随机推荐
- http to https
https://www.cnblogs.com/powertoolsteam/p/http2https.html
- JWT总结
Json web token (JWT) 什么是JWT? Json web token (JWT), 是为了在网络应用环境间传递声明而执行的一种基于JSON的开放标准((RFC 7519).该toke ...
- Arcengine效率探究之二——属性的更新(转载)
http://blog.csdn.net/lk103852503/article/details/6570748 修改一批要素的属性有多种方法,当数据量较大时,若选择不当可能会大大影响速度. 一.IR ...
- Python学习-将Python3.3.4还原成Python2.6版本
http://blog.csdn.net/dream_angel_z/article/details/45724515
- 20135202闫佳歆--week3 课本1-2章学习笔记
第一章 Linux内核简介 一.Unix Unix是一个强大.健壮和稳定的操作系统. 简洁 绝大部分东西都被当做文件对待.这种抽象使对数据和对设备的操作都是通过一套相同的系统调用借口来进行的:open ...
- HashMap相关总结
1.HashMap:根据键值hashCode值存储数据,大多数情况下可以直接定位到它的值,但是遍历顺序不确定.所有哈希值相同的值存储到同一个链表中 Ha ...
- win10下装上virtualbox 以及在virtualbox上装上 ubuntu 12.04
首先要下载virtual 在win10下可能第一步你就遇到了麻烦 首先刚开始我装的是最新版本的virtualbox 5.0.24.8355 (直接百度就可搜到) 然后可以按照这个教程 http://j ...
- VS2015安装与单元测试
很久之前就听说微软有一款强大的编程软件——Visual Stdio系列,也许是满足于VC和CB的小巧一直都没有去尝试,借这次软件工程的机会终于可以一睹其真容,第一感觉是高大上,一改VC和CB的简洁,看 ...
- HTTP协议(1)
HTTP 协议笔记 1. HTTP 传输机制 HTTP 是一个应用层协议,位于 TCP 的层次之上,并且是基于 TCP 协议进行通信的,也就是说 HTTP 在每一次通信之前都要先建立 TCP 连接来保 ...
- 『编程题全队』alpha阶段项目复审
小组的名字和链接 优点 缺点,bug 报告 最终名次 Gakki赛高 (1)支持注册账号和账号管理(2) 支持自动登录,提供便捷性(3)题目不重复且题目答案准确(4)支持排行榜统计功能(5)自己设计算 ...