2019-04-22 21:03:00 +03:00
|
|
|
/*******************************************************************************************
|
2020-02-20 14:42:37 +03:00
|
|
|
*
|
|
|
|
* raylib [network] example - Client/Server ping-pong
|
2020-02-19 20:16:20 +03:00
|
|
|
*
|
|
|
|
* This example has been created using raylib 3.0 (www.raylib.com)
|
|
|
|
* raylib is licensed under an unmodified zlib/libpng license (View raylib.h for details)
|
|
|
|
*
|
|
|
|
* Copyright (c) 2019-2020 Jak Barnes (@syphonx) and Ramon Santamaria (@raysan5)
|
|
|
|
*
|
|
|
|
********************************************************************************************/
|
2019-04-22 21:03:00 +03:00
|
|
|
|
|
|
|
#include "raylib.h"
|
2020-02-19 20:16:20 +03:00
|
|
|
|
|
|
|
#define RNET_IMPLEMENTATION
|
2021-07-28 20:48:20 +03:00
|
|
|
#include "extras/rnet.h"
|
2019-04-22 21:03:00 +03:00
|
|
|
|
2020-02-20 14:42:37 +03:00
|
|
|
float elapsed = 0.0f;
|
|
|
|
float delay = 1.0f;
|
|
|
|
bool ping = false;
|
|
|
|
bool pong = false;
|
|
|
|
bool connected = false;
|
|
|
|
bool clientConnected = false;
|
|
|
|
const char *pingmsg = "Ping!";
|
|
|
|
const char *pongmsg = "Pong!";
|
|
|
|
int msglen = 0;
|
|
|
|
SocketConfig serverConfig = { .host = "127.0.0.1", .port = "4950", .type = SOCKET_TCP, .server = true, .nonblocking = true };
|
|
|
|
SocketConfig clientConfig = { .host = "127.0.0.1", .port = "4950", .type = SOCKET_TCP, .nonblocking = true };
|
|
|
|
SocketConfig connectionConfig = { .nonblocking = true };
|
|
|
|
SocketResult *serverResult = NULL;
|
|
|
|
SocketResult *clientResult = NULL;
|
|
|
|
SocketSet *socketSet = NULL;
|
|
|
|
Socket *connection = NULL;
|
|
|
|
char receiveBuffer[512] = { 0 };
|
2019-04-22 21:03:00 +03:00
|
|
|
|
|
|
|
// Attempt to connect to the network (Either TCP, or UDP)
|
2020-02-20 14:42:37 +03:00
|
|
|
static void NetworkConnect(void)
|
2019-04-22 21:03:00 +03:00
|
|
|
{
|
2019-08-03 12:07:41 +03:00
|
|
|
// If the server is configured as UDP, ignore connection requests
|
2020-02-20 14:42:37 +03:00
|
|
|
if ((serverConfig.type == SOCKET_UDP) && (clientConfig.type == SOCKET_UDP))
|
|
|
|
{
|
|
|
|
ping = true;
|
2019-08-03 12:07:41 +03:00
|
|
|
connected = true;
|
2021-04-22 19:55:24 +03:00
|
|
|
}
|
|
|
|
else
|
2020-02-20 14:42:37 +03:00
|
|
|
{
|
2019-08-03 12:07:41 +03:00
|
|
|
// If the client is connected, run the server code to check for a connection
|
2021-04-22 19:55:24 +03:00
|
|
|
if (clientConnected)
|
2020-02-20 14:42:37 +03:00
|
|
|
{
|
|
|
|
int active = CheckSockets(socketSet, 0);
|
|
|
|
if (active != 0) TraceLog(LOG_INFO, "There are currently %d socket(s) with data to be processed.", active);
|
|
|
|
|
2021-04-22 19:55:24 +03:00
|
|
|
if (active > 0)
|
2020-02-20 14:42:37 +03:00
|
|
|
{
|
|
|
|
if ((connection = SocketAccept(serverResult->socket, &connectionConfig)) != NULL)
|
|
|
|
{
|
|
|
|
AddSocket(socketSet, connection);
|
2019-08-03 12:07:41 +03:00
|
|
|
connected = true;
|
2020-02-20 14:42:37 +03:00
|
|
|
ping = true;
|
2019-08-03 12:07:41 +03:00
|
|
|
}
|
|
|
|
}
|
2021-04-22 19:55:24 +03:00
|
|
|
}
|
2020-02-20 14:42:37 +03:00
|
|
|
else
|
|
|
|
{
|
2019-08-03 12:07:41 +03:00
|
|
|
// Check if we're connected every _delay_ seconds
|
|
|
|
elapsed += GetFrameTime();
|
2020-02-20 14:42:37 +03:00
|
|
|
if (elapsed > delay)
|
|
|
|
{
|
|
|
|
if (IsSocketConnected(clientResult->socket)) clientConnected = true;
|
2021-04-22 19:55:24 +03:00
|
|
|
|
2019-08-03 12:07:41 +03:00
|
|
|
elapsed = 0.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-04-22 21:03:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Once connected to the network, check the sockets for pending information
|
|
|
|
// and when information is ready, send either a Ping or a Pong.
|
2020-02-20 14:42:37 +03:00
|
|
|
static void UpdateNetwork(void)
|
2019-04-22 21:03:00 +03:00
|
|
|
{
|
2020-02-20 14:42:37 +03:00
|
|
|
// CheckSockets, if any of the sockets in the socketSet are pending (received data, or requests)
|
|
|
|
// then mark the socket as being ready. You can check this with IsSocketReady(clientResult->socket)
|
|
|
|
int active = CheckSockets(socketSet, 0);
|
|
|
|
if (active != 0) TraceLog(LOG_DEBUG, "There are currently %d socket(s) with data to be processed.", active);
|
2019-04-22 21:03:00 +03:00
|
|
|
|
2020-02-20 14:42:37 +03:00
|
|
|
// IsSocketReady, if the socket is ready, attempt to receive data from the socket
|
2019-08-03 12:07:41 +03:00
|
|
|
int bytesRecv = 0;
|
2020-02-20 14:42:37 +03:00
|
|
|
if ((serverConfig.type == SOCKET_UDP) && (clientConfig.type == SOCKET_UDP))
|
|
|
|
{
|
|
|
|
if (IsSocketReady(clientResult->socket)) bytesRecv = SocketReceive(clientResult->socket, receiveBuffer, msglen);
|
|
|
|
if (IsSocketReady(serverResult->socket)) bytesRecv = SocketReceive(serverResult->socket, receiveBuffer, msglen);
|
2021-04-22 19:55:24 +03:00
|
|
|
}
|
2020-02-20 14:42:37 +03:00
|
|
|
else if (IsSocketReady(connection)) bytesRecv = SocketReceive(connection, receiveBuffer, msglen);
|
2019-04-22 21:03:00 +03:00
|
|
|
|
2019-08-03 12:07:41 +03:00
|
|
|
// If we received data, was that data a "Ping!" or a "Pong!"
|
2020-02-20 14:42:37 +03:00
|
|
|
if (bytesRecv > 0)
|
|
|
|
{
|
|
|
|
if (strcmp(receiveBuffer, pingmsg) == 0) { pong = true; }
|
|
|
|
if (strcmp(receiveBuffer, pongmsg) == 0) { ping = true; }
|
2019-08-03 12:07:41 +03:00
|
|
|
}
|
2019-04-22 21:03:00 +03:00
|
|
|
|
2019-08-03 12:07:41 +03:00
|
|
|
// After each delay has expired, send a response "Ping!" for a "Pong!" and vice versa
|
|
|
|
elapsed += GetFrameTime();
|
2020-02-20 14:42:37 +03:00
|
|
|
if (elapsed > delay)
|
|
|
|
{
|
|
|
|
if (ping)
|
|
|
|
{
|
2019-08-03 12:07:41 +03:00
|
|
|
ping = false;
|
2020-02-20 14:42:37 +03:00
|
|
|
if (serverConfig.type == SOCKET_UDP && clientConfig.type == SOCKET_UDP) SocketSend(clientResult->socket, pingmsg, msglen);
|
|
|
|
else SocketSend(clientResult->socket, pingmsg, msglen);
|
2021-04-22 19:55:24 +03:00
|
|
|
}
|
2020-02-20 14:42:37 +03:00
|
|
|
else if (pong)
|
|
|
|
{
|
2019-08-03 12:07:41 +03:00
|
|
|
pong = false;
|
2020-02-20 14:42:37 +03:00
|
|
|
if (serverConfig.type == SOCKET_UDP && clientConfig.type == SOCKET_UDP) SocketSend(clientResult->socket, pongmsg, msglen);
|
|
|
|
else SocketSend(clientResult->socket, pongmsg, msglen);
|
2019-08-03 12:07:41 +03:00
|
|
|
}
|
2021-04-22 19:55:24 +03:00
|
|
|
|
2019-08-03 12:07:41 +03:00
|
|
|
elapsed = 0.0f;
|
|
|
|
}
|
2019-04-22 21:03:00 +03:00
|
|
|
}
|
|
|
|
|
2020-02-19 20:16:20 +03:00
|
|
|
int main(void)
|
2019-04-22 21:03:00 +03:00
|
|
|
{
|
2020-02-19 20:16:20 +03:00
|
|
|
// Initialization
|
|
|
|
//--------------------------------------------------------------------------------------
|
|
|
|
const int screenWidth = 800;
|
|
|
|
const int screenHeight = 450;
|
|
|
|
|
2020-02-20 14:42:37 +03:00
|
|
|
InitWindow(screenWidth, screenHeight, "raylib [network] example - ping pong");
|
|
|
|
|
|
|
|
InitNetworkDevice(); // Init network communications
|
|
|
|
|
|
|
|
// Create the server: getaddrinfo + socket + setsockopt + bind + listen
|
2020-02-20 15:46:33 +03:00
|
|
|
serverResult = LoadSocketResult();
|
2021-04-22 19:55:24 +03:00
|
|
|
if (!SocketCreate(&serverConfig, serverResult))
|
2020-02-20 14:42:37 +03:00
|
|
|
{
|
|
|
|
TraceLog(LOG_WARNING, "Failed to open server: status %d, errno %d", serverResult->status, serverResult->socket->status);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!SocketBind(&serverConfig, serverResult))
|
|
|
|
{
|
|
|
|
TraceLog(LOG_WARNING, "Failed to bind server: status %d, errno %d", serverResult->status, serverResult->socket->status);
|
2021-04-22 19:55:24 +03:00
|
|
|
}
|
2020-02-20 14:42:37 +03:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!(serverConfig.type == SOCKET_UDP))
|
|
|
|
{
|
|
|
|
if (!SocketListen(&serverConfig, serverResult))
|
|
|
|
{
|
|
|
|
TraceLog(LOG_WARNING, "Failed to start listen server: status %d, errno %d", serverResult->status, serverResult->socket->status);
|
2019-08-03 12:07:41 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-04-22 21:03:00 +03:00
|
|
|
|
2020-02-20 14:42:37 +03:00
|
|
|
// Create the client: getaddrinfo + socket + setsockopt + connect (TCP only)
|
2020-02-20 15:46:33 +03:00
|
|
|
clientResult = LoadSocketResult();
|
2021-04-22 19:55:24 +03:00
|
|
|
if (!SocketCreate(&clientConfig, clientResult))
|
2020-02-20 14:42:37 +03:00
|
|
|
{
|
|
|
|
TraceLog(LOG_WARNING, "Failed to open client: status %d, errno %d", clientResult->status, clientResult->socket->status);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!(clientConfig.type == SOCKET_UDP))
|
|
|
|
{
|
|
|
|
if (!SocketConnect(&clientConfig, clientResult))
|
|
|
|
{
|
|
|
|
TraceLog(LOG_WARNING, "Failed to connect to server: status %d, errno %d", clientResult->status, clientResult->socket->status);
|
2019-08-03 12:07:41 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-04-22 21:03:00 +03:00
|
|
|
|
2020-02-20 14:42:37 +03:00
|
|
|
// Create and add sockets to the socket set
|
2020-02-20 15:46:33 +03:00
|
|
|
socketSet = LoadSocketSet(3);
|
2020-02-20 14:42:37 +03:00
|
|
|
|
|
|
|
AddSocket(socketSet, serverResult->socket);
|
|
|
|
AddSocket(socketSet, clientResult->socket);
|
2019-04-22 21:03:00 +03:00
|
|
|
|
2020-02-19 20:16:20 +03:00
|
|
|
SetTargetFPS(60); // Set our game to run at 60 frames-per-second
|
|
|
|
//--------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
// Main game loop
|
|
|
|
while (!WindowShouldClose()) // Detect window close button or ESC key
|
|
|
|
{
|
|
|
|
// Update
|
|
|
|
//----------------------------------------------------------------------------------
|
2020-02-20 14:42:37 +03:00
|
|
|
if (connected) UpdateNetwork();
|
|
|
|
//else NetworkConnect();
|
2020-02-19 20:16:20 +03:00
|
|
|
//----------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
// Draw
|
|
|
|
//----------------------------------------------------------------------------------
|
|
|
|
BeginDrawing();
|
|
|
|
|
|
|
|
ClearBackground(RAYWHITE);
|
2021-04-22 19:55:24 +03:00
|
|
|
|
2020-02-20 14:42:37 +03:00
|
|
|
// TODO: Draw relevant connection info
|
2020-02-19 20:16:20 +03:00
|
|
|
|
|
|
|
EndDrawing();
|
|
|
|
//----------------------------------------------------------------------------------
|
|
|
|
}
|
|
|
|
|
|
|
|
// De-Initialization
|
|
|
|
//--------------------------------------------------------------------------------------
|
2020-02-20 14:42:37 +03:00
|
|
|
CloseNetworkDevice(); // Close network communication
|
2021-04-22 19:55:24 +03:00
|
|
|
|
2020-02-20 14:42:37 +03:00
|
|
|
CloseWindow(); // Close window and OpenGL context
|
2020-02-19 20:16:20 +03:00
|
|
|
//--------------------------------------------------------------------------------------
|
2019-04-22 21:03:00 +03:00
|
|
|
|
2019-08-03 12:07:41 +03:00
|
|
|
return 0;
|
2019-04-22 21:03:00 +03:00
|
|
|
}
|