forked from lantus/postal-nx
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathNetBrowse.cpp
277 lines (243 loc) · 9.56 KB
/
NetBrowse.cpp
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
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
////////////////////////////////////////////////////////////////////////////////
//
// Copyright 2016 RWS Inc, All Rights Reserved
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of version 2 of the GNU General Public License as published by
// the Free Software Foundation
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License along
// with this program; if not, write to the Free Software Foundation, Inc.,
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
//
// NetBrowse.cpp
// Project: RSPiX
//
// History:
// 09/01/97 MJR Nearing the end of a major overhaul.
//
// 09/02/97 MJR Changed so browse end now does the periodic broadcast
// and hosts merely respond to them. This saves bandwidth
// on the host end, and in fact cuts down overall network
// traffic because we'll only be generating these messages
// when we're browsing, as opposed to having the hosts
// constantly spew out messages, regardless of whether
// anyone is listening.
//
// 09/06/97 MJR Fixed so that it will properly drop hosts that no longer
// exist.
//
////////////////////////////////////////////////////////////////////////////////
#include "RSPiX.h"
#include "netbrowse.h"
////////////////////////////////////////////////////////////////////////////////
// Constructor
////////////////////////////////////////////////////////////////////////////////
CNetBrowse::CNetBrowse()
{
Reset();
}
////////////////////////////////////////////////////////////////////////////////
// Destructor
////////////////////////////////////////////////////////////////////////////////
CNetBrowse::~CNetBrowse()
{
Reset();
}
////////////////////////////////////////////////////////////////////////////////
// Reset
////////////////////////////////////////////////////////////////////////////////
void CNetBrowse::Reset(void)
{
m_socketBrowse.Reset();
m_lLastBroadcast = 0;
m_usBasePort = 0;
}
////////////////////////////////////////////////////////////////////////////////
// Startup
////////////////////////////////////////////////////////////////////////////////
int16_t CNetBrowse::Startup( // Returns 0 if sucessfull, non-zero otherwise
uint16_t usPort, // In: Server's base port number
RSocket::BLOCK_CALLBACK callback) // In: Blocking callback
{
int16_t sResult = 0;
// Make sure we start in a good state
Reset();
// Save base port
m_usBasePort = usPort;
// Create socket on which to broadcast
sResult = m_socketBrowse.Open(m_usBasePort + Net::BroadcastPortOffset, RSocket::typDatagram, RSocket::optDontBlock, callback);
if (sResult == 0)
{
// Set socket to broadcast mode
sResult = m_socketBrowse.Broadcast();
if (sResult == 0)
{
}
else
TRACE("CNetBrowse::StartBrowse(): Error putting socket into broadcast mode!\n");
}
else
TRACE("CNetBrowse::StartBrowse(): Couldn't open broadcast socket!\n");
return sResult;
}
////////////////////////////////////////////////////////////////////////////////
// Shutdown
////////////////////////////////////////////////////////////////////////////////
void CNetBrowse::Shutdown(void)
{
Reset();
}
////////////////////////////////////////////////////////////////////////////////
// Update (must be called regularly!)
//
// The lists are updated, if necessary. Note that only the phostsAll is
// important to this function, as it uses that list as the basis of its
// decisions to add or remove hosts. This function will simply add to the
// other two lists as needed -- it does not care what they contain. It is up
// to the caller to decide whether and when to clear those lists.
////////////////////////////////////////////////////////////////////////////////
void CNetBrowse::Update(
Hosts* phostsAll, // I/O: List of all hosts
Hosts* phostsAdded, // I/O: List of hosts that were added
Hosts* phostsRemoved) // I/O: List of hosts that were removed
{
// Check if it's time to broadcast
int32_t lTime = rspGetMilliseconds();
if ((lTime - m_lLastBroadcast) > Net::BroadcastInterval)
{
// Create message
U8 buf1[4];
buf1[0] = Net::BroadcastMagic0;
buf1[1] = Net::BroadcastMagic1;
buf1[2] = Net::BroadcastMagic2;
buf1[3] = Net::BroadcastMagic3;
// Create destination address (the address on which others will receive this message)
RSocket::Address address;
RSocket::CreateBroadcastAddress(m_usBasePort + Net::AntennaPortOffset, &address);
// Broadcast the message
int32_t lBytesSent;
int16_t serr = m_socketBrowse.SendTo(buf1, sizeof(buf1), &lBytesSent, &address);
if (serr == 0)
{
if (lBytesSent != sizeof(buf1))
TRACE("CNetBrowse::Update(): Error sending broadcast (wrong size)!\n");
}
else
{
if (serr != RSocket::errWouldBlock)
TRACE("CNetBrowse::Update(): Error sending broadcast!\n");
}
// If there was no error, reset the timer. If there was an error, we want to
// retry as soon as possible. If the error is a recurring one that won't go
// away, we'll be retrying every time this is called, but what the hell -- if
// it isn't working, what are we gonna do instead?
if (serr == 0)
m_lLastBroadcast = lTime;
}
// Check for a reply to our broadcast. If we get an incorrectly-sized message,
// we simply ignore it -- this is a datagram socket, so if the message was larger
// than we expected, the rest of it will be discarded, and if it was smaller, then
// we can ignore it as well. Bad messages could come from a foreign app that is
// using the same port as us. If we do get a message, the address of the sender
// will be recorded -- this gives us the host's address!
CHost host;
int32_t lReceived;
U8 buf[sizeof(host.m_acName) + 4 + 4];
int16_t serr = m_socketBrowse.ReceiveFrom(buf, sizeof(buf), &lReceived, &host.m_address);
if (serr == 0)
{
// Validate the message to make sure it was sent by another app of this
// type, as opposed to some unknown app that happens to use the same port.
if ((lReceived == sizeof(buf)) &&
(buf[0] == Net::BroadcastMagic0) &&
(buf[1] == Net::BroadcastMagic1) &&
(buf[2] == Net::BroadcastMagic2) &&
(buf[3] == Net::BroadcastMagic3))
{
// Copy the magic number. The endian nature will always be correct because
// the only entitity that is meant to recognize this value is the one
// that sent it, so as long as the encoding and decoding of the bytes
// is the same, that entity will get the same value that it sent. All
// other entities will see this as a meaningless value, which is fine.
host.m_lMagic =
((int32_t)buf[4] & 0x000000ff) +
(((int32_t)buf[5] << 8) & 0x0000ff00) +
(((int32_t)buf[6] << 16) & 0x00ff0000) +
(((int32_t)buf[7] << 24) & 0xff000000);
// Copy the name
strncpy(host.m_acName, (char*)&buf[8], sizeof(host.m_acName));
host.m_acName[sizeof(host.m_acName)-1] = 0;
// Init time we last heard from this host to "now"
host.m_lLastHeardFrom = rspGetMilliseconds();
// Change the host's port number from its antenna port to its base port
uint16_t usHostBasePort = RSocket::GetAddressPort(&host.m_address) - Net::AntennaPortOffset;
RSocket::SetAddressPort(usHostBasePort, &host.m_address);
// Check if this host already exists in the list
bool bExists = false;
Hosts::Pointer p;
for (p = phostsAll->GetHead(); p; p = phostsAll->GetNext(p))
{
if (host.IsSameHost(&phostsAll->GetData(p)))
{
// Update this host's time to "now"
phostsAll->GetData(p).m_lLastHeardFrom = rspGetMilliseconds();
// Set flag and stop
bExists = true;
break;
}
}
// If host doesn't already exist, add it to the list
if (!bExists)
{
phostsAll->InsertTail(host);
phostsAdded->InsertTail(host);
}
}
else
TRACE("CNetBrowse::Update(): Validation failed -- another app may be sending crap to our port!\n");
}
else
{
if (serr != RSocket::errWouldBlock)
TRACE("CNetBrowse::Update(): Error receiving broadcast!\n");
}
// Check for hosts that haven't been heard from in too long a time,
// and should therefore be dropped.
Hosts::Pointer p = phostsAll->GetHead();
while (p)
{
Hosts::Pointer pNext = phostsAll->GetNext(p);
if ((rspGetMilliseconds() - phostsAll->GetData(p).m_lLastHeardFrom) > Net::BroadcastDropTime)
{
// Drop this host by moving it from the "all" list to the "dropped" list
phostsRemoved->InsertTail(phostsAll->GetData(p));
phostsAll->Remove(p);
}
p = pNext;
}
}
////////////////////////////////////////////////////////////////////////////////
// Lookup host by name or hardwired address (like a TCP/IP dotted address).
// The specified port must be the host's "base port".
////////////////////////////////////////////////////////////////////////////////
// static
int16_t CNetBrowse::LookupHost( // Returns 0 if successfull, non-zero otherwise
char* pszName, // In: Server's name or dotted address (x.x.x.x)
uint16_t usPort, // In: Server's port number
RSocket::Address* paddress) // Out: Addresss
{
// Try to get requested address
int16_t sResult = RSocket::GetAddress(pszName, usPort, paddress);
if (sResult != 0)
TRACE("CNetBrowse::LookupHost(): Error getting host address!\n");
return sResult;
}
////////////////////////////////////////////////////////////////////////////////
// EOF
////////////////////////////////////////////////////////////////////////////////