-
Notifications
You must be signed in to change notification settings - Fork 333
/
sendtcp.c
101 lines (83 loc) · 2.62 KB
/
sendtcp.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
/*
* $smu-mark$
* $name: sendtcp.c$
* $author: Salvatore Sanfilippo <antirez@invece.org>$
* $copyright: Copyright (C) 1999 by Salvatore Sanfilippo$
* $license: This software is under GPL version 2 of license$
* $date: Fri Nov 5 11:55:49 MET 1999$
* $rev: 8$
*/
/* $Id: sendtcp.c,v 1.2 2003/09/01 00:22:06 antirez Exp $ */
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <time.h>
#include <sys/time.h>
#include <unistd.h>
#include <signal.h>
#include "hping2.h"
#include "globals.h"
void send_tcp(void)
{
int packet_size;
int tcp_opt_size = 0;
char *packet, *data;
struct mytcphdr *tcp;
struct pseudohdr *pseudoheader;
unsigned char *tstamp;
if (opt_tcp_timestamp)
tcp_opt_size = 12;
packet_size = TCPHDR_SIZE + tcp_opt_size + data_size;
packet = malloc(PSEUDOHDR_SIZE + packet_size);
if (packet == NULL) {
perror("[send_tcphdr] malloc()");
return;
}
pseudoheader = (struct pseudohdr*) packet;
tcp = (struct mytcphdr*) (packet+PSEUDOHDR_SIZE);
tstamp = (unsigned char*) (packet+PSEUDOHDR_SIZE+TCPHDR_SIZE);
data = (char*) (packet+PSEUDOHDR_SIZE+TCPHDR_SIZE+tcp_opt_size);
memset(packet, 0, PSEUDOHDR_SIZE+packet_size);
/* tcp pseudo header */
memcpy(&pseudoheader->saddr, &local.sin_addr.s_addr, 4);
memcpy(&pseudoheader->daddr, &remote.sin_addr.s_addr, 4);
pseudoheader->protocol = 6; /* tcp */
pseudoheader->lenght = htons(TCPHDR_SIZE+tcp_opt_size+data_size);
/* tcp header */
tcp->th_dport = htons(dst_port);
tcp->th_sport = htons(src_port);
/* sequence number and ack are random if not set */
tcp->th_seq = (set_seqnum) ? htonl(tcp_seqnum) : htonl(rand());
tcp->th_ack = (set_ack) ? htonl(tcp_ack) : htonl(rand());
tcp->th_off = src_thoff + (tcp_opt_size >> 2);
tcp->th_win = htons(src_winsize);
tcp->th_flags = tcp_th_flags;
/* tcp timestamp option */
if (opt_tcp_timestamp) {
__u32 randts = rand() ^ (rand() << 16);
tstamp[0] = tstamp[1] = 1; /* NOOP */
tstamp[2] = 8;
tstamp[3] = 10; /* 10 bytes, kind+len+T1+T2 */
memcpy(tstamp+4, &randts, 4); /* random */
memset(tstamp+8, 0, 4); /* zero */
}
/* data */
data_handler(data, data_size);
/* compute checksum */
#ifdef STUPID_SOLARIS_CHECKSUM_BUG
tcp->th_sum = packet_size;
#else
tcp->th_sum = cksum((u_short*) packet, PSEUDOHDR_SIZE +
packet_size);
#endif
/* adds this pkt in delaytable */
delaytable_add(sequence, src_port, time(NULL), get_usec(), S_SENT);
/* send packet */
send_ip_handler(packet+PSEUDOHDR_SIZE, packet_size);
free(packet);
sequence++; /* next sequence number */
if (!opt_keepstill)
src_port = (sequence + initsport) % 65536;
if (opt_force_incdport)
dst_port++;
}