forked from Lukasa/gopcap
-
Notifications
You must be signed in to change notification settings - Fork 0
/
transport.go
158 lines (133 loc) · 4.34 KB
/
transport.go
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
package gopcap
//-----------------------------------------------------------------------------
// Unknown Transport
//-----------------------------------------------------------------------------
// UnknownTransport represents the data for a Transport-Layer packet that gopcap doesn't
// understand. It simply provides uninterpreted data representing the entire transport-layer
// packet.
type UnknownTransport struct {
data []byte
}
func (u *UnknownTransport) TransportData() []byte {
return u.data
}
func (u *UnknownTransport) FromBytes(data []byte) error {
u.data = data
return nil
}
//-----------------------------------------------------------------------------
// TCPSegment
//-----------------------------------------------------------------------------
// TCPSegment represents the data for a single Transmission Control Protocol segment. This method of
// storing a TCPSegment is less efficient than storing the binary representation on the wire.
type TCPSegment struct {
SourcePort uint16
DestinationPort uint16
SequenceNumber uint32
AckNumber uint32
HeaderSize uint8
NS bool // This should be viewed as a temporary solution for flags: it's hugely space-inefficient,
CWR bool // and so I'll probably update the TCPSegment to handle flags differently.
ECE bool
URG bool
ACK bool
PSH bool
RST bool
SYN bool
FIN bool
WindowSize uint16
Checksum uint16
UrgentOffset uint16
OptionData []byte // This is temporary. We should handle TCP options properly.
data []byte
}
func (t *TCPSegment) TransportData() []byte {
return t.data
}
func (t *TCPSegment) FromBytes(data []byte) error {
// Begin by confirming that we have enough data for a complete TCP header.
if len(data) < 20 {
return InsufficientLength
}
// The first four fields are really easy.
t.SourcePort = getUint16(data[0:2], false)
t.DestinationPort = getUint16(data[2:4], false)
t.SequenceNumber = getUint32(data[4:8], false)
t.AckNumber = getUint32(data[8:12], false)
// The header size is the top four bits of the next byte.
t.HeaderSize = uint8(data[12]) >> 4
// Now we have all the flag fields. First, the NS flag.
if (uint8(data[12]) & 0x01) != 0 {
t.NS = true
}
// The next eight flags are all in the next byte.
flags := uint8(data[13])
if (flags & 0x80) != 0 {
t.CWR = true
}
if (flags & 0x40) != 0 {
t.ECE = true
}
if (flags & 0x20) != 0 {
t.URG = true
}
if (flags & 0x10) != 0 {
t.ACK = true
}
if (flags & 0x08) != 0 {
t.PSH = true
}
if (flags & 0x04) != 0 {
t.RST = true
}
if (flags & 0x02) != 0 {
t.SYN = true
}
if (flags & 0x01) != 0 {
t.FIN = true
}
// Now we're back to sane things.
t.WindowSize = getUint16(data[14:16], false)
t.Checksum = getUint16(data[16:18], false)
t.UrgentOffset = getUint16(data[18:20], false)
// If the header size is larger than 5 (it's measured in 32-bit words for reasons that escape me),
// we have some number of extra bytes that form the TCP options.
extraBytes := (t.HeaderSize - 5) * 4
data = data[20:]
if len(data) < int(extraBytes) {
return InsufficientLength
}
t.OptionData = data[:extraBytes]
// All that remains is the contained data.
t.data = data[extraBytes:]
return nil
}
//-----------------------------------------------------------------------------
// UDPDatagram
//-----------------------------------------------------------------------------
// UDPDatagram represents the data for a single User Datagram Protocol datagram. This method of
// storing a UDPDatagram is less efficient than storing the binary representation on the wire.
type UDPDatagram struct {
SourcePort uint16
DestinationPort uint16
Length uint16
Checksum uint16
data []byte
}
func (u *UDPDatagram) TransportData() []byte {
return u.data
}
func (u *UDPDatagram) FromBytes(data []byte) error {
// Begin by confirming that we have enough data to actually represent a UDP datagram.
if len(data) < 8 {
return InsufficientLength
}
// Happily, UDP is super simple. This makes this code equally simple.
u.SourcePort = getUint16(data[0:2], false)
u.DestinationPort = getUint16(data[2:4], false)
u.Length = getUint16(data[4:6], false)
u.Checksum = getUint16(data[6:8], false)
// All that remains is data.
u.data = data[8:]
return nil
}