/* =====================================================================================
*
* 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首部的更多相关文章

  1. UNP——原始套接字

    1.原始套接字的用处 使用原始套接字可以构造或读取网际层及其以上报文. 具体来说,可以构造 ICMP, IGMP 协议报文,通过开启 IP_HDRINCL 套接字选项,进而自定义 IPv4首部. 2. ...

  2. Linux系统编程(37)—— socket编程之原始套接字

    原始套接字的特点 原始套接字(SOCK_RAW)可以用来自行组装IP数据包,然后将数据包发送到其他终端.也就是说原始套接字是基于IP数据包的编程(SOCK_PACKET是基于数据链路层的编程).另外, ...

  3. UNIX网络编程——原始套接字SOCK_RAW

    实际上,我们常用的网络编程都是在应用层的报文的收发操作,也就是大多数程序员接触到的流式套接字(SOCK_STREAM)和数据包式套接字(SOCK_DGRAM).而这些数据包都是由系统提供的协议栈实现, ...

  4. raw_socket(原始套接字)以及普通socket使用终极总结

      一.传输层socket(四层socket,普通socket) 可参考本人以下博客: Windows Socket编程之UDP实现大文件的传输:http://blog.csdn.net/luchen ...

  5. 005.TCP--拼接TCP头部IP头部,实现TCP三次握手的第一步(Linux,原始套接字)

    一.目的: 自己拼接IP头,TCP头,计算效验和,将生成的报文用原始套接字发送出去. 若使用tcpdump能监听有对方服务器的包回应,则证明TCP报文是正确的! 二.数据结构: TCP首部结构图: s ...

  6. 004.UDP--拼接UDP数据包,构造ip头和udp头通信(使用原始套接字)

    一.大致流程: 建立一个client端,一个server端,自己构建IP头和UDP头,写入数据(hello,world!)后通过原始套接字(SOCK_RAW)将包发出去. server端收到数据后,打 ...

  7. python使用原始套接字 解析原始ip头数据

    使用底层套接字解码底层流量,是这次做的重点工作. 首先来捕获第一个包 # coding:utf-8import socket # 监听的主机IP host = "192.168.1.100& ...

  8. C++ Win 32 使用原始套接字获取所有ip数据包并分析(包括ping包)

    /*页面编码:GBK 开发环境 VS2019 */ #define _WINSOCK_DEPRECATED_NO_WARNINGS#include <iostream>#include&l ...

  9. linux原始套接字(4)-构造IP_UDP

    一.概述                                                    同上一篇tcp一样,udp也是封装在ip报文里面.创建UDP的原始套接字如下: (soc ...

随机推荐

  1. 如何在百度云虚拟机中配置thinkphp5,并且url去掉index.php

    第一步:将public目录下的index.php移到和public同级目录下,[或者直接在public同级目录下新建一个index.php] 第二步:那么这个新的index.php文件的内容如下: & ...

  2. FINAUNCE金融业增速反弹信贷投放创新高叠加股市回暖

    FINAUNCE金融业增速反弹信贷投放创新高叠加股市回暖,金融业增加值增速回暖,不过难以回到2015年的巅峰. 国家统计局4月18日发布的数据显示,今年一季度,国内生产总值21.34万亿元,按可比价格 ...

  3. keycode值对照表

    转载自:https://segmentfault.com/a/1190000005828048 字母和数字键的键码值(keyCode) 按键 键码 按键 键码 按键 键码 按键 键码 A 65 J 7 ...

  4. ini_set的用法介绍

    https://www.cnblogs.com/xieqian111/p/5367732.html

  5. 11.5 Daily Scrum

    请把现在当成11月5日······   Today's tasks  Tomorrow's tasks 丁辛 餐厅列表数据结构设计 餐厅列表UI设计             李承晗           ...

  6. 课堂讨论——Alpha版总结会议

    我们在课堂上针对第一阶段冲刺过程中存在的问题,展开了激烈的讨论,并投票选出需要改进的最主要三个问题.

  7. (Alpha)Let's-展示博客

    Let's Alpha 项目答辩 ·选题由来     手机端——用户相对较多,使用环境限制相对宽松     手机游戏?校园p2p应用?线下交流!(滴滴打水?)     模式的选择:发起——加入活动   ...

  8. Office处理

    1.NPOI:一个开源项目,不需要安装Microsoft Office,支持对Office 97-2003,2007文件格式,功能比较强大. http://npoi.codeplex.com/ 2.a ...

  9. PAT 甲级 1096 Consecutive Factors

    https://pintia.cn/problem-sets/994805342720868352/problems/994805370650738688 Among all the factors ...

  10. mybatis之接口方法多参数的三种实现方式

    关键代码举例: DaoMapper.xml <!-- 传入多个参数时,自动转换为map形式 --> <insert id="insertByColumns" us ...