- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
准确地说,我已经创建了这个简单的 TCP 程序来通过网络将一个简单的 TCP 数据包发送到目标机器,但不知何故,我真的不知道我做错了什么,但它似乎没有发送任何数据包到目的主机。我也无法在我的 wireshark 中找到它。
rawtcp.c:
//---cat rawtcp.c---
// Run as root or SUID 0, just datagram no data/payload
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <stdio.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/ip.h>
#include <netinet/in.h>
#include <netinet/tcp.h>
// Packet length
#define PCKT_LEN 8192
#pragma pack(push, 1)
// May create separate header file (.h) for all
// headers' structures
// IP header's structure
struct ipheader {
unsigned char iph_ihl:4, /* Little-endian */
iph_ver:4;
unsigned char iph_tos;
unsigned short int iph_len;
unsigned short int iph_ident;
unsigned short int iph_offset;
unsigned char iph_ttl;
unsigned char iph_protocol;
unsigned short int iph_chksum;
unsigned int iph_sourceip;
unsigned int iph_destip;
};
/* Structure of a TCP header */
struct tcpheader {
unsigned short int tcph_srcport;
unsigned short int tcph_destport;
unsigned int tcph_seqnum;
unsigned int tcph_acknum;
unsigned char tcph_reserved:4, tcph_offset:4;
// unsigned char tcph_flags;
unsigned int
tcp_res1:4, /*little-endian*/
tcph_hlen:4, /*length of tcp header in 32-bit words*/
tcph_fin:1, /*Finish flag "fin"*/
tcph_syn:1, /*Synchronize sequence numbers to start a connection*/
tcph_rst:1, /*Reset flag */
tcph_psh:1, /*Push, sends data to the application*/
tcph_ack:1, /*acknowledge*/
tcph_urg:1, /*urgent pointer*/
tcph_res2:2;
unsigned short int tcph_win;
unsigned short int tcph_chksum;
unsigned short int tcph_urgptr;
};
#pragma pack(pop)
// Simple checksum function, may use others such as Cyclic Redundancy Check, CRC
unsigned short csum(unsigned short *buf, int len)
{
unsigned long sum;
for(sum=0; len>0; len--)
sum += *buf++;
sum = (sum >> 16) + (sum &0xffff);
sum += (sum >> 16);
return (unsigned short)(~sum);
}
int main(int argc, char *argv[])
{
int sd;
// No data, just datagram
char buffer[PCKT_LEN];
// The size of the headers
struct ipheader *ip = (struct ipheader *) buffer;
struct tcpheader *tcp = (struct tcpheader *) (buffer + sizeof(struct ipheader));
struct sockaddr_in sin, din;
int one = 1;
const int *val = &one;
memset(buffer, 0, PCKT_LEN);
if(argc != 5)
{
printf("- Invalid parameters!!!\n");
printf("- Usage: %s <source hostname/IP> <source port> <target hostname/IP> <target port>\n", argv[0]);
exit(-1);
}
sd = socket(PF_INET, SOCK_RAW, IPPROTO_TCP);
if(sd < 0)
{
perror("socket() error");
exit(-1);
} else
printf("socket()-SOCK_RAW and tcp protocol is OK.\n");
// The source is redundant, may be used later if needed
// Address family
sin.sin_family = AF_INET;
din.sin_family = AF_INET;
// Source port, can be any, modify as needed
sin.sin_port = htons(atoi(argv[2]));
din.sin_port = htons(atoi(argv[4]));
// Source IP, can be any, modify as needed
sin.sin_addr.s_addr = inet_addr(argv[1]);
din.sin_addr.s_addr = inet_addr(argv[3]);
// IP structure
ip->iph_ihl = 5;
ip->iph_ver = 4;
ip->iph_tos = 16;
ip->iph_len = htons(sizeof(struct ipheader) + sizeof(struct tcpheader));
ip->iph_ident = htons(54321);
ip->iph_offset = 0;
ip->iph_ttl = 64;
ip->iph_protocol = 6; // TCP
ip->iph_chksum = 0; // Done by kernel
// Source IP, modify as needed, spoofed, we accept through command line argument
ip->iph_sourceip = inet_addr(argv[1]);
// Destination IP, modify as needed, but here we accept through command line argument
ip->iph_destip = inet_addr(argv[3]);
// The TCP structure. The source port, spoofed, we accept through the command line
tcp->tcph_srcport = htons(atoi(argv[2]));
// The destination port, we accept through command line
tcp->tcph_destport = htons(atoi(argv[4]));
tcp->tcph_seqnum = htonl(1);
tcp->tcph_acknum = 0;
tcp->tcph_offset = 5;
tcp->tcph_syn = 1;
tcp->tcph_ack = 0;
tcp->tcph_win = htons(32767);
tcp->tcph_chksum = 0; // Done by kernel
tcp->tcph_urgptr = 0;
// IP checksum calculation
ip->iph_chksum = htons(csum((unsigned short *) buffer, (sizeof(struct ipheader) + sizeof(struct tcpheader))));
// Inform the kernel do not fill up the headers' structure, we fabricated our own
if(setsockopt(sd, IPPROTO_IP, IP_HDRINCL, val, sizeof(one)) < 0)
{
perror("setsockopt() error");
exit(-1);
} else
printf("setsockopt() is OK\n");
printf("Using:::::Destination IP: %s port: %u, Source IP: %s port: %u.\n", argv[1], atoi(argv[2]), argv[3], atoi(argv[4]));
// sendto() loop, send every 2 second for 50 counts
unsigned int count;
for(count = 0; count < 20; count++)
{
if(sendto(sd, buffer, ip->iph_len, 0, (struct sockaddr *)&din, sizeof(din)) < 0)
// Verify
{
perror("sendto() error");
exit(-1);
} else
printf("Count #%u - sendto() is OK\n", count);
sleep(2);
}
close(sd);
return 0;
}
编译时,它编译得很好。当我运行程序时,它应该是 ./rawtcp 192.168.1.152 1000 192.168.1.151 1000
它说:
socket()-SOCK_RAW and tcp protocol is OK.
setsockopt() is OK
Using:::::Destination IP: 192.168.1.151 port: 1000, Source IP: 192.168.1.152 port: 1000.
Count #0 - sendto() is OK
Count #1 - sendto() is OK
但是当将 wireshark 过滤为 tcp.port == 1000
时,它永远不会显示(注意:源 ip 是 192.168.1.152,目标 ip 是 192.168.1.151)。但是如果我做一个简单的 hping3 -p 1000 192.168.1.151
它工作得很好,我也可以在 wireshark 中看到它。
如果您能告诉我我的代码中哪里做错了,我将不胜感激:)
最佳答案
首先,您缺少 <arpa/inet.h>
header ,它提供了 inet_addr
您正在使用的功能。由于隐式函数声明,您的程序仍应在 C89 下编译而无需此,但依赖它是不好的做法,它可能会导致细微的错误。
一个问题是您滥用了 #pragma pack
.
在你声明你的结构之前,你应该做 #pragma pack(push, 1)
,这两者都将 struct packing alignment 设置为 1,并将先前的 packing alignment 状态推送到堆栈。完成声明结构后,您可以执行 #pragma pack(pop)
在任何以下声明中将结构包装重置为正常。对于您的程序,省略 #pragma pack(pop)
在你的声明之后应该仍然有效,但最好重新设置它,以防你声明任何其他你不想被紧密打包的结构,或者你之后包含任何其他头文件。
#pragma pack(push, 1)
struct a {
/* ... */
};
struct b {
/* ... */
};
#pragma pack(pop)
你的 struct ipheader
的另一个问题是你有两个 8 位 iph_flags
和一个 16 位 iph_offset
,而 IP header 中的标志字段应为 3 位,偏移字段应为 13 位(两者加起来为 16 位)。这意味着您的 header 中有额外的 8 位,不应该存在。由于您无论如何都要用零填充这两个字段,因此您可以通过删除 iph_flags
来解决此问题完全从你的结构中,只保留一个 16 位 iph_offset
字段,用0填充,实际上跨越了IP头中3位和13位字段的空间。
通过这些修复,您的程序对我有效,并且可以在 Wireshark 中看到一个数据包。
关于c - 在 c 中使用 sendto() 发送自定义 tcp 数据包?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48338190/
我在 udp 套接字客户端中调用 sendto,如下所示: if((num_w = sendto(cli_udp_sock, buffer, strlen(buffer), 0, (struct so
在 C 中使用 sendto() 时,我遇到了一个奇怪的问题。我有以下代码: struct sockaddr_in currSocket; int currSocketLength = sizeof(
我目前正在尝试在两台计算机之间设置 UDP 协议(protocol)。第一个是 Windows 驱动的计算机,第二个是带有 Raspbian 操作系统的 Raspberry Pi。 问题是我能够将信息
我正在为一个大学项目创建一个并发 UDP 套接字。客户端等待 stdin 命令,然后使用函数 child_job 创建一个新进程和一个新套接字。服务器接收来自客户端的命令,创建一个新进程来初始化新套接
我正在尝试在没有 tcp 有效负载的情况下进行客户端-服务器通信。我只是使用 tcp 的 header 作为有效负载字段,默认情况下为 20 字节。我刚刚意识到传输大文件时 CPU 使用率非常高,我尝
我尝试使用以下代码通过 UDP 发送,但得到了奇怪的结果。 if((sendto(newSocket, sendBuf, totalLength, 0, (SOCKADDR *)&sendAd
我有两个相互通信的程序。 客户端:先发送消息,然后监听回复。服务器:监听回复然后发送消息。 我能够完美地从客户端发送消息并在服务器中监听。但是当我尝试从服务器发送消息时出现问题。 struct hos
我目前正在处理一项网络作业。我们希望客户端自动获得分配给 TCP 套接字的 IP 和端口,并将 UDP 套接字绑定(bind)到与 TCP 套接字相同的地址和端口。这样,UDP 和 TCP 共享相同的
我有服务器和客户端通过 Windows 套接字发送/接收字符串。 客户端代码(发件人): string c = "Haha"; sprintf(temp.Buffer, c.c_str()); tem
您好,我有以下代码: if ((fd = socket(PF_INET, SOCK_DGRAM, 0)) sin_family = AF_INET; sin->sin_port = htons(
我是 python(3.3) 网络编程的新手,所以一开始我尝试编写一个基本的跟踪路由程序。其中一行代码是: send_socket.sendto(512, '', (dest_name, port))
在我最近的项目中,我需要使用UDP协议(protocol)来传输数据。如果我使用 size = s.sendto(data, (, )) 发送数据,UDP协议(protocol)会保证数据打包成一个U
我有 2 个程序在运行(程序 1、程序 2),它们都使用我初始化的套接字相互发送和接收 UDP 数据包。 我的问题是,如果我从 program1 的 TRANSMIT(TX) 套接字发送广播,prog
ioctl(socketFd_, SIOCOUTQ, &outstandingBytes); getsockopt(socketFd_, SOL_SOCKET, SO_SNDBUF, &sendBu
我正在编写一个通过套接字(从客户端到服务器)发送数据的应用程序。我观察到非常奇怪的行为:当我杀死服务器时,先发送然后杀死就像服务器存在一样。下一个返回 EPIPE。你能解释一下为什么吗? 这是来自客户
我有一个程序,它通过 UDP 接受坐标,移动一些设备,然后在工作完成时进行回复。 我似乎和这个人有同样的问题: Python sendto doesn't seem to send 我的代码在这里:
ssize_t sendto(int socket, const void *message, size_t length, int flags, const struct sockaddr *des
我如何以编程方式 (.net) 调用 Windows 资源管理器 SendTo 上下文菜单选项上的命令? 最佳答案 sendto 项是文件系统对象这一事实是一个实现细节,菜单由 sendto shel
我是 Python 的新手,正在尝试使用套接字将字节数组作为原始数据包发送。我的 IP 是 192.168.0.116,我要发送到的设备是 192.168.0.64。这个客户端设备是一个基于微 Con
在SendTo文件夹里加上一文件夹的快捷方式后,在右键发送到这个文件夹的是这些文件的一个副本,实际上是一个复制的过程,有时候我们只希望是快捷方式,那就得另想办法了。 把如下代码内容保存为sho
我是一名优秀的程序员,十分优秀!