2016-11-04 20:15:10 +00:00
|
|
|
/* server.cpp */
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <enet/enet.h>
|
2016-11-06 19:01:55 +00:00
|
|
|
#include "../../CommonInterfaces/CommonGUIHelperInterface.h"
|
2016-12-29 05:51:54 +00:00
|
|
|
#include "Bullet3Common/b3CommandLineArgs.h"
|
|
|
|
|
2016-11-06 19:01:55 +00:00
|
|
|
#ifdef NO_SHARED_MEMORY
|
|
|
|
#include "PhysicsServerCommandProcessor.h"
|
|
|
|
typedef PhysicsServerCommandProcessor MyCommandProcessor;
|
|
|
|
#else
|
|
|
|
#include "SharedMemoryCommandProcessor.h"
|
2016-12-29 05:51:54 +00:00
|
|
|
typedef SharedMemoryCommandProcessor MyCommandProcessor;
|
2016-11-06 19:01:55 +00:00
|
|
|
#endif //NO_SHARED_MEMORY
|
2016-12-29 05:51:54 +00:00
|
|
|
|
2016-11-04 20:15:10 +00:00
|
|
|
#include "SharedMemoryCommands.h"
|
|
|
|
#include "Bullet3Common/b3AlignedObjectArray.h"
|
|
|
|
#include "PhysicsServerCommandProcessor.h"
|
2017-01-29 06:14:45 +00:00
|
|
|
#include "../Utils/b3Clock.h"
|
2016-11-06 19:01:55 +00:00
|
|
|
|
|
|
|
bool gVerboseNetworkMessagesServer = false;
|
2016-11-04 20:15:10 +00:00
|
|
|
|
|
|
|
void MySerializeInt(unsigned int sz, unsigned char* output)
|
|
|
|
{
|
|
|
|
unsigned int tmp = sz;
|
|
|
|
output[0] = tmp & 255;
|
|
|
|
tmp = tmp >> 8;
|
|
|
|
output[1] = tmp & 255;
|
|
|
|
tmp = tmp >> 8;
|
|
|
|
output[2] = tmp & 255;
|
|
|
|
tmp = tmp >> 8;
|
|
|
|
output[3] = tmp & 255;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2016-12-29 05:51:54 +00:00
|
|
|
b3CommandLineArgs parseArgs(argc,argv);
|
2017-01-29 06:14:45 +00:00
|
|
|
b3Clock clock;
|
|
|
|
double timeOutInSeconds = 10;
|
2016-11-04 20:15:10 +00:00
|
|
|
|
2016-11-06 19:01:55 +00:00
|
|
|
DummyGUIHelper guiHelper;
|
2016-12-29 05:51:54 +00:00
|
|
|
MyCommandProcessor* sm = new MyCommandProcessor;
|
2016-11-06 19:01:55 +00:00
|
|
|
sm->setGuiHelper(&guiHelper);
|
2016-12-29 05:51:54 +00:00
|
|
|
|
|
|
|
int port = 1234;
|
|
|
|
if (parseArgs.GetCmdLineArgument("port",port))
|
|
|
|
{
|
|
|
|
printf("Using UDP port %d\n", port);
|
|
|
|
}
|
|
|
|
|
|
|
|
gVerboseNetworkMessagesServer = parseArgs.CheckCmdLineFlag("verbose");
|
|
|
|
|
|
|
|
#ifndef NO_SHARED_MEMORY
|
|
|
|
int key = 0;
|
|
|
|
if (parseArgs.GetCmdLineArgument("sharedMemoryKey",key))
|
|
|
|
{
|
|
|
|
sm->setSharedMemoryKey(key);
|
|
|
|
}
|
|
|
|
#endif//NO_SHARED_MEMORY
|
|
|
|
|
2016-11-04 20:15:10 +00:00
|
|
|
// PhysicsDirect* sm = new PhysicsDirect(sdk);
|
|
|
|
|
|
|
|
//PhysicsClientSharedMemory* sm = new PhysicsClientSharedMemory();
|
|
|
|
|
|
|
|
bool isPhysicsClientConnected = sm->connect();
|
|
|
|
|
|
|
|
if (isPhysicsClientConnected)
|
|
|
|
{
|
|
|
|
|
|
|
|
ENetAddress address;
|
|
|
|
ENetHost *server;
|
|
|
|
ENetEvent event;
|
|
|
|
int serviceResult;
|
|
|
|
|
|
|
|
puts("Starting server");
|
|
|
|
|
|
|
|
if (enet_initialize() != 0)
|
|
|
|
{
|
|
|
|
puts("Error initialising enet");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Bind the server to the default localhost. */
|
|
|
|
/* A specific host address can be specified by */
|
|
|
|
/* enet_address_set_host (& address, "x.x.x.x"); */
|
|
|
|
address.host = ENET_HOST_ANY;
|
|
|
|
/* Bind the server to port 1234. */
|
2016-12-29 05:51:54 +00:00
|
|
|
address.port = port;
|
2016-11-04 20:15:10 +00:00
|
|
|
|
|
|
|
server = enet_host_create(&address,
|
|
|
|
32, /* number of clients */
|
|
|
|
2, /* number of channels */
|
|
|
|
0, /* Any incoming bandwith */
|
|
|
|
0); /* Any outgoing bandwith */
|
|
|
|
|
|
|
|
if (server == NULL)
|
|
|
|
{
|
|
|
|
puts("Could not create server host");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
2017-01-29 06:14:45 +00:00
|
|
|
b3Clock::usleep(0);
|
|
|
|
|
2016-11-04 20:15:10 +00:00
|
|
|
serviceResult = 1;
|
|
|
|
|
|
|
|
/* Keep doing host_service until no events are left */
|
|
|
|
while (serviceResult > 0)
|
|
|
|
{
|
|
|
|
/* Wait up to 1000 milliseconds for an event. */
|
|
|
|
serviceResult = enet_host_service(server, &event, 0);
|
|
|
|
if (serviceResult > 0)
|
|
|
|
{
|
|
|
|
|
|
|
|
switch (event.type)
|
|
|
|
{
|
2016-11-14 15:39:34 +00:00
|
|
|
case ENET_EVENT_TYPE_CONNECT:
|
|
|
|
{
|
2016-11-04 20:15:10 +00:00
|
|
|
printf("A new client connected from %x:%u.\n",
|
|
|
|
event.peer->address.host,
|
|
|
|
event.peer->address.port);
|
|
|
|
|
|
|
|
/* Store any relevant client information here. */
|
|
|
|
event.peer->data = (void*)"Client information";
|
|
|
|
|
|
|
|
break;
|
2016-11-14 15:39:34 +00:00
|
|
|
}
|
2016-11-04 20:15:10 +00:00
|
|
|
case ENET_EVENT_TYPE_RECEIVE:
|
2016-11-14 15:39:34 +00:00
|
|
|
{
|
2016-11-06 19:01:55 +00:00
|
|
|
if (gVerboseNetworkMessagesServer)
|
|
|
|
{
|
2017-02-20 21:18:33 +00:00
|
|
|
int dataLen = (int)event.packet->dataLength;
|
|
|
|
|
2016-11-06 19:01:55 +00:00
|
|
|
printf("A packet of length %u containing '%s' was "
|
|
|
|
"received from %s on channel %u.\n",
|
2017-02-20 21:18:33 +00:00
|
|
|
dataLen,
|
2016-11-06 19:01:55 +00:00
|
|
|
event.packet->data,
|
|
|
|
event.peer->data,
|
|
|
|
event.channelID);
|
|
|
|
}
|
2017-01-29 06:14:45 +00:00
|
|
|
SharedMemoryCommand cmd;
|
|
|
|
|
|
|
|
SharedMemoryCommand* cmdPtr = 0;
|
|
|
|
|
|
|
|
//performance test
|
|
|
|
if (event.packet->dataLength == sizeof(int))
|
|
|
|
{
|
|
|
|
cmdPtr = &cmd;
|
|
|
|
cmd.m_type = *(int*)event.packet->data;
|
|
|
|
}
|
|
|
|
|
2016-11-04 20:15:10 +00:00
|
|
|
if (event.packet->dataLength == sizeof(SharedMemoryCommand))
|
|
|
|
{
|
2017-01-29 06:14:45 +00:00
|
|
|
cmdPtr = (SharedMemoryCommand*)event.packet->data;
|
|
|
|
}
|
|
|
|
if (cmdPtr)
|
|
|
|
{
|
2016-11-04 20:15:10 +00:00
|
|
|
SharedMemoryStatus serverStatus;
|
|
|
|
b3AlignedObjectArray<char> buffer;
|
|
|
|
buffer.resize(SHARED_MEMORY_MAX_STREAM_CHUNK_SIZE);
|
|
|
|
|
|
|
|
bool hasStatus = sm->processCommand(*cmdPtr,serverStatus, &buffer[0], buffer.size());
|
|
|
|
|
2017-01-29 06:14:45 +00:00
|
|
|
double startTimeSeconds = clock.getTimeInSeconds();
|
|
|
|
double curTimeSeconds = clock.getTimeInSeconds();
|
|
|
|
|
|
|
|
while ((!hasStatus) && ((curTimeSeconds - startTimeSeconds) <timeOutInSeconds))
|
2016-11-04 20:15:10 +00:00
|
|
|
{
|
|
|
|
hasStatus = sm->receiveStatus(serverStatus, &buffer[0], buffer.size());
|
2017-01-29 06:14:45 +00:00
|
|
|
curTimeSeconds = clock.getTimeInSeconds();
|
2016-11-04 20:15:10 +00:00
|
|
|
}
|
2016-11-06 19:01:55 +00:00
|
|
|
if (gVerboseNetworkMessagesServer)
|
|
|
|
{
|
|
|
|
printf("buffer.size = %d\n", buffer.size());
|
|
|
|
printf("serverStatus.m_numDataStreamBytes = %d\n", serverStatus.m_numDataStreamBytes);
|
|
|
|
}
|
2016-11-04 20:15:10 +00:00
|
|
|
if (hasStatus)
|
|
|
|
{
|
|
|
|
b3AlignedObjectArray<unsigned char> packetData;
|
2017-01-29 06:14:45 +00:00
|
|
|
unsigned char* statBytes = (unsigned char*)&serverStatus;
|
2016-11-04 20:15:10 +00:00
|
|
|
|
2017-01-29 06:14:45 +00:00
|
|
|
if (cmdPtr->m_type == CMD_STEP_FORWARD_SIMULATION)
|
2016-11-04 20:15:10 +00:00
|
|
|
{
|
2017-01-29 06:14:45 +00:00
|
|
|
packetData.resize(4 + sizeof(int));
|
|
|
|
int sz = packetData.size();
|
|
|
|
int curPos = 0;
|
|
|
|
|
|
|
|
MySerializeInt(sz, &packetData[curPos]);
|
|
|
|
curPos += 4;
|
|
|
|
for (int i = 0; i < sizeof(int); i++)
|
|
|
|
{
|
|
|
|
packetData[i + curPos] = statBytes[i];
|
|
|
|
}
|
|
|
|
curPos += sizeof(int);
|
|
|
|
ENetPacket *packet = enet_packet_create(&packetData[0], packetData.size(), ENET_PACKET_FLAG_RELIABLE);
|
|
|
|
enet_peer_send(event.peer, 0, packet);
|
2016-11-04 20:15:10 +00:00
|
|
|
}
|
2017-01-29 06:14:45 +00:00
|
|
|
else
|
2016-11-04 20:15:10 +00:00
|
|
|
{
|
2017-01-29 06:14:45 +00:00
|
|
|
//create packetData with [int packetSizeInBytes, status, streamBytes)
|
|
|
|
packetData.resize(4 + sizeof(SharedMemoryStatus) + serverStatus.m_numDataStreamBytes);
|
|
|
|
int sz = packetData.size();
|
|
|
|
int curPos = 0;
|
|
|
|
|
|
|
|
MySerializeInt(sz, &packetData[curPos]);
|
|
|
|
curPos += 4;
|
|
|
|
for (int i = 0; i < sizeof(SharedMemoryStatus); i++)
|
|
|
|
{
|
|
|
|
packetData[i + curPos] = statBytes[i];
|
|
|
|
}
|
|
|
|
curPos += sizeof(SharedMemoryStatus);
|
|
|
|
|
|
|
|
for (int i = 0; i < serverStatus.m_numDataStreamBytes; i++)
|
|
|
|
{
|
|
|
|
packetData[i + curPos] = buffer[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
ENetPacket *packet = enet_packet_create(&packetData[0], packetData.size(), ENET_PACKET_FLAG_RELIABLE);
|
|
|
|
enet_peer_send(event.peer, 0, packet);
|
|
|
|
//enet_host_broadcast(server, 0, packet);
|
2016-11-04 20:15:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("received packet with unknown contents\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Tell all clients about this message */
|
|
|
|
//enet_host_broadcast(server, 0, event.packet);
|
|
|
|
|
|
|
|
break;
|
2016-11-14 15:39:34 +00:00
|
|
|
}
|
2016-11-04 20:15:10 +00:00
|
|
|
case ENET_EVENT_TYPE_DISCONNECT:
|
2016-11-14 15:39:34 +00:00
|
|
|
{
|
2016-11-04 20:15:10 +00:00
|
|
|
printf("%s disconnected.\n", event.peer->data);
|
|
|
|
|
|
|
|
/* Reset the peer's client information. */
|
|
|
|
|
|
|
|
event.peer->data = NULL;
|
|
|
|
|
|
|
|
break;
|
2016-11-14 15:39:34 +00:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
2016-11-04 20:15:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (serviceResult > 0)
|
|
|
|
{
|
|
|
|
puts("Error with servicing the server");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
enet_host_destroy(server);
|
|
|
|
enet_deinitialize();
|
|
|
|
}
|
|
|
|
delete sm;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|