127 lines
3.2 KiB
C
127 lines
3.2 KiB
C
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include <sys/socket.h>
|
|
#include <sys/types.h> // BSD compatible
|
|
#include <arpa/inet.h>
|
|
#include <unistd.h>
|
|
|
|
#include <errno.h>
|
|
#include <netdb.h>
|
|
|
|
#include "network.h"
|
|
|
|
const int DGRAM_MSG_START = 0;
|
|
const int DGRAM_PAYLOAD_START = DGRAM_MSG_START + DGRAM_MSG_LENGTH + DGRAM_NULL_LENGTH;
|
|
|
|
int sendToXPlane(int sock, char *destAddr, int destPort, char *msg, void *payload, int lengthOfPayload)
|
|
{
|
|
char bytesToSend[SEND_BUFFER];
|
|
struct sockaddr_in6 xPlaneAddress;
|
|
memset(bytesToSend, 0, SEND_BUFFER);
|
|
|
|
// msg is always 4 bytes long
|
|
if (strlen(msg) != DGRAM_MSG_LENGTH)
|
|
{
|
|
return -10;
|
|
}
|
|
|
|
// payload shouldn't exceed buffer
|
|
if (lengthOfPayload > (SEND_BUFFER - DGRAM_MSG_LENGTH - DGRAM_NULL_LENGTH))
|
|
{
|
|
return -11;
|
|
}
|
|
|
|
strcpy(&bytesToSend[DGRAM_MSG_START], msg);
|
|
memcpy(&bytesToSend[DGRAM_PAYLOAD_START], payload, lengthOfPayload);
|
|
|
|
// prepare destination address struct
|
|
xPlaneAddress.sin6_family = AF_INET6;
|
|
xPlaneAddress.sin6_port = htons(destPort);
|
|
inet_pton(AF_INET6, destAddr, &xPlaneAddress.sin6_addr.s6_addr);
|
|
|
|
int numBytesSend = sendto(sock, bytesToSend, DGRAM_MSG_LENGTH + DGRAM_NULL_LENGTH + lengthOfPayload, 0, (struct sockaddr *)&xPlaneAddress, sizeof(xPlaneAddress));
|
|
if (numBytesSend >= 0)
|
|
{
|
|
return numBytesSend;
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "Failed to send message (%d): %s\n", errno, strerror(errno));
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
int recvFromXPlane(int sock, char msg[DGRAM_MSG_LENGTH], void *payload, int maxLength)
|
|
{
|
|
char buf[maxLength + DGRAM_MSG_LENGTH];
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
strcpy(msg, "");
|
|
|
|
int recvBytes = recv(sock, buf, maxLength + DGRAM_MSG_LENGTH, MSG_DONTWAIT);
|
|
if (recvBytes >= 0)
|
|
{
|
|
if (recvBytes >= DGRAM_MSG_LENGTH)
|
|
{
|
|
memcpy(msg, &buf[0], DGRAM_MSG_LENGTH);
|
|
memcpy(payload, &buf[DGRAM_MSG_LENGTH], maxLength);
|
|
return recvBytes - DGRAM_MSG_LENGTH;
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "Received data without message header?!\n");
|
|
return -11;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (errno != EAGAIN)
|
|
{
|
|
fprintf(stderr, "Error receiving data (%d): %s\n", errno, strerror(errno));
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
int createSocket(int srcPort)
|
|
{
|
|
int xPlaneSocket;
|
|
struct sockaddr_in6 localAddress;
|
|
|
|
int err;
|
|
|
|
// create socket
|
|
xPlaneSocket = socket(AF_INET6, SOCK_DGRAM, 0);
|
|
if (xPlaneSocket == -1)
|
|
{
|
|
fprintf(stderr, "Error creating socket (%d): %s\n", errno, strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
// prepare source address struct
|
|
localAddress.sin6_family = AF_INET6;
|
|
localAddress.sin6_port = htons(srcPort);
|
|
localAddress.sin6_addr = in6addr_any;
|
|
|
|
// bind to local port
|
|
err = bind(xPlaneSocket, (struct sockaddr *)&localAddress, sizeof(localAddress));
|
|
if (err)
|
|
{
|
|
fprintf(stderr, "Error binding [::]:%d (%d): %s\n", srcPort, errno, gai_strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
return xPlaneSocket;
|
|
}
|
|
|
|
void closeSocket(int sock)
|
|
{
|
|
close(sock);
|
|
}
|