/* =====================================================================================
*
* 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. javascript source map 的使用

    之前发现VS.NET会为压缩的js文添加一个与文件名同名的.map文件,一直没有搞懂他是用来做什么的,直接删除掉运行时浏览器又会报错,后来google了一直才真正搞懂了这个小小的map文件背后的巨大意 ...

  2. 微信小程序选择并上传图片

      上传图片 API: wx.chooseImage() 和 wx.uploadFile() wx.chooseImage({ count: 1, // 默认9 sizeType: ['origina ...

  3. FME Cloud 账号申请流程

    第一步,访问SAFE的FME Cloud注册页,官网明确表态,如果你是一个新的FME Cloud用户,你可以免费获得一个初级版.地址:https://console.fmecloud.safe.com ...

  4. Google C++ 编码规范

    刚刚看到一位博主的文章分享Google C++ 编码规范 本人做一下记录,方便以后学习.. 中文在线版本地址: http://zh-google-styleguide.readthedocs.io/e ...

  5. Scrum Meeting NO.4

    Scrum Meeting No.4 1.会议内容 今天我们开始着手重构前端通讯模块的代码.用封装性较好的HttpClient代替了源代码中比较底层的访问方式,并按照约定的json格式处理HttpRe ...

  6. Linux第二章读书笔记

    1.获取内核源码 1.1Git 分布式的:下载和管理Linux内核源代码: - 获取最新提交到版本树的一个副本 $ git clone git://git.kernel.org/pub/scm/lin ...

  7. Sprint 1 Review & Daily Scrum - 11/18

    今天我们组利用课后的时间对Sprint 1阶段进行了回顾,并对接下来的工作进行了安排. Sprint 1阶段我们开始定的计划是完成最基础的背单词功能,可以让用户可以完整地走一遍背单词流程.回顾上周,我 ...

  8. C与C++的编程风格区别

    c和c++都是在实践中发展起来的语言.实用性极强.c是与UNIX/Linux的发展相辅相成的.而C++是B.S为了摆脱c与硬件以及底层过于紧密的苦恼,而开发的语言.C++可以说就是c语言的超集.任何c ...

  9. Photoshop一些常用的快捷键

    1.按住Alt键,点击图层上的小眼睛,只显示当前图层 2.新建纯色图层,抠图 3.Fn+ num 调整不透明度 4.Shift + num 调整流量 5.调整图层透明度,在移动工具状态下,输入数字:0 ...

  10. sql server获取当前月的天数

    方法1 SELECT 32-DAY(getdate()+32-DAY(getdate())) 方法2 CREATE FUNCTION dbo.fn_getMonthDayAll ---自定义函数名称 ...