-
Notifications
You must be signed in to change notification settings - Fork 134
/
Copy pathtcp.h
197 lines (177 loc) · 6.23 KB
/
tcp.h
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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
#ifndef __TCP_H
#define __TCP_H
#include "sock.h"
#include "list.h"
#include "tcp_timer.h"
#define TCP_DEFAULT_WINDOW 4096
#define TCP_DEFAULT_TTL 64
#define TCP_LITTLE_ENDIAN
struct tcp {
unsigned short src; /* source port */
unsigned short dst; /* dest port */
unsigned int seq; /* sequence number */
unsigned int ackn; /* acknowledgment number */
#ifdef TCP_LITTLE_ENDIAN
unsigned short reserved:4,
doff:4, /* data offset(head length)in 32 bits long */
/* control bits */
fin:1,
syn:1,
rst:1,
psh:1, /* push */
ack:1, /* acknowlegment */
urg:1, /* urgent */
ece:1, /* See RFC 3168 */
cwr:1; /* See RFC 3168 */
#else
unsigned short doff:4, /* data offset(head length)in 32 bits long */
reserved:4,
/* control bits */
cwr:1, /* See RFC 3168 */
ece:1, /* See RFC 3168 */
urg:1, /* urgent */
ack:1, /* acknowlegment */
psh:1, /* push */
rst:1,
syn:1,
fin:1;
#endif
unsigned short window;
unsigned short checksum;
unsigned short urgptr; /* urgent pointer */
unsigned char data[0];
} __attribute__((packed));
#define pkb2tcp(pkb) ((struct tcp *)((pkb)->pk_data + ETH_HRD_SZ + IP_HRD_SZ))
#define ip2tcp(ip) ((struct tcp *)ipdata(ip))
#define TCP_HRD_SZ (sizeof(struct tcp))
#define TCP_HRD_DOFF (TCP_HRD_SZ >> 2)
#define tcphlen(tcp) ((tcp)->doff << 2)
#define tcptext(tcp) ((unsigned char *)(tcp) + tcphlen(tcp))
enum tcp_state {
TCP_CLOSED = 1,
TCP_LISTEN,
TCP_SYN_RECV,
TCP_SYN_SENT,
TCP_ESTABLISHED,
TCP_CLOSE_WAIT,
TCP_LAST_ACK,
TCP_FIN_WAIT1,
TCP_FIN_WAIT2,
TCP_CLOSING,
TCP_TIME_WAIT,
TCP_MAX_STATE
};
struct tcp_sock {
struct sock sk;
struct hlist_node bhash_list; /* for bind hash table, e/lhash node is in sk */
unsigned int bhash; /* bind hash value */
int accept_backlog; /* current entries of accept queue */
int backlog; /* size of accept queue */
struct list_head listen_queue; /* waiting for second SYN+ACK of three-way handshake */
struct list_head accept_queue; /* waiting for third ACK of three-way handshake */
struct list_head list;
struct tcp_timer timewait; /* TIME-WAIT TIMEOUT */
struct tapip_wait *wait_accept;
struct tapip_wait *wait_connect;
struct tcp_sock *parent;
unsigned int flags;
struct cbuf *rcv_buf;
struct list_head rcv_reass; /* list head of unordered reassembled tcp segments */
/* transmission control block (RFC 793) */
unsigned int snd_una; /* send unacknowledged */
unsigned int snd_nxt; /* send next */
unsigned int snd_wnd; /* send window */
unsigned int snd_up; /* send urgent pointer */
unsigned int snd_wl1; /* seq for last window update */
unsigned int snd_wl2; /* ack for last window update */
unsigned int iss; /* initial send sequence number */
unsigned int rcv_nxt; /* receive next */
unsigned int rcv_wnd; /* receive window */
unsigned int rcv_up; /* receive urgent pointer */
unsigned int irs; /* initial receive sequence number */
unsigned int state; /* connection state */
};
#define tcpsk(sk) ((struct tcp_sock *)sk)
#define TCP_MAX_BACKLOG 128
#define TCP_DEAD_PARENT ((struct tcp_sock *)0xffffdaed)
#define TCP_F_PUSH 0x00000001 /* text pushing to user */
#define TCP_F_ACKNOW 0x00000002 /* ack at right */
#define TCP_F_ACKDELAY 0x00000004 /* ack at right */
/* host-order tcp current segment (RFC 793) */
struct tcp_segment {
unsigned int seq; /* segment sequence number */
unsigned int ack; /* segment acknowledgment number */
unsigned int lastseq; /* segment last sequence number */
unsigned int len; /* segment length */
unsigned int dlen; /* segment text length */
unsigned int wnd; /* segment window */
unsigned int up; /* segment urgent point */
unsigned int prc; /* segment precedence value(no used) */
unsigned char *text; /* segment text */
struct ip *iphdr;
struct tcp *tcphdr;
};
static _inline int tcp_accept_queue_full(struct tcp_sock *tsk)
{
return (tsk->accept_backlog >= tsk->backlog);
}
static _inline void tcp_accept_enqueue(struct tcp_sock *tsk)
{
/* move it from listen queue to accept queue */
if (!list_empty(&tsk->list))
list_del(&tsk->list);
list_add(&tsk->list, &tsk->parent->accept_queue);
tsk->accept_backlog++;
}
static _inline struct tcp_sock *tcp_accept_dequeue(struct tcp_sock *tsk)
{
struct tcp_sock *newtsk;
newtsk = list_first_entry(&tsk->accept_queue, struct tcp_sock, list);
list_del_init(&newtsk->list);
tsk->accept_backlog--;
return newtsk;
}
extern void tcp_in(struct pkbuf *);
extern struct sock *tcp_lookup_sock(unsigned int, unsigned int, unsigned int, unsigned int);
extern void tcp_process(struct pkbuf *, struct tcp_segment *, struct sock *);
extern struct sock *tcp_alloc_sock(int);
extern int tcp_hash(struct sock *);
extern void tcp_unhash(struct sock *);
extern void tcp_unbhash(struct tcp_sock *);
extern void tcp_init(void);
extern struct tcp_sock *get_tcp_sock(struct tcp_sock *);
extern void tcp_send_reset(struct tcp_sock *, struct tcp_segment *);
extern void tcp_send_synack(struct tcp_sock *, struct tcp_segment *);
extern void tcp_send_ack(struct tcp_sock *, struct tcp_segment *);
extern void tcp_send_syn(struct tcp_sock *, struct tcp_segment *);
extern void tcp_send_fin(struct tcp_sock *);
extern void tcp_recv_text(struct tcp_sock *, struct tcp_segment *, struct pkbuf *);
extern void tcp_free_buf(struct tcp_sock *);
extern int tcp_write_buf(struct tcp_sock *, void *, unsigned int);
extern void tcp_free_reass_head(struct tcp_sock *);
extern void tcp_segment_reass(struct tcp_sock *, struct tcp_segment *, struct pkbuf *);
extern void tcp_send_out(struct tcp_sock *, struct pkbuf *, struct tcp_segment *);
extern int tcp_send_text(struct tcp_sock *, void *, int);
extern unsigned int alloc_new_iss(void);
extern int tcp_id;
extern const char *tcp_state_string[];
#define for_each_tcp_sock(tsk, node, head)\
hlist_for_each_entry(tsk, node, head, bhash_list)
/* used by tcp_recv_text() & tcp_reass() */
#define ADJACENT_SEGMENT_HEAD(nseq) \
do { \
if ((nseq) > seg->seq) { \
if (seg->dlen <= (nseq) - seg->seq) \
goto out; \
seg->dlen -= (nseq) - seg->seq; \
seg->text += (nseq) - seg->seq; \
seg->seq = (nseq); \
} \
} while (0)
static _inline void tcp_set_state(struct tcp_sock *tsk, enum tcp_state state)
{
tcpsdbg("State from %s to %s", tcp_state_string[tsk->state],
tcp_state_string[state]);
tsk->state = state;
}
#endif /* tcp.h */