made sockets non-blocking
This commit is contained in:
59
app.c
59
app.c
@@ -46,7 +46,7 @@ int djstd_entry(Arena *arena, StringList args) {
|
|||||||
if (isServer) {
|
if (isServer) {
|
||||||
println("Starting server on port %d", port);
|
println("Starting server on port %d", port);
|
||||||
Server myserver = serverInit((ServerInitInfo){
|
Server myserver = serverInit((ServerInitInfo){
|
||||||
.concurrentClients=16,
|
.concurrentClients=2,
|
||||||
.port=port,
|
.port=port,
|
||||||
.memory=Megabytes(64),
|
.memory=Megabytes(64),
|
||||||
});
|
});
|
||||||
@@ -54,48 +54,67 @@ int djstd_entry(Arena *arena, StringList args) {
|
|||||||
server = &myserver;
|
server = &myserver;
|
||||||
|
|
||||||
serverListen(&myserver);
|
serverListen(&myserver);
|
||||||
Socket *client = serverAccept(&myserver);
|
Socket *client1 = serverAccept(&myserver);
|
||||||
|
Socket *client2 = serverAccept(&myserver);
|
||||||
|
|
||||||
|
Forever {
|
||||||
|
string message = s("Hello. You are client 1.\n");
|
||||||
|
socketWrite(client1, message.str, message.length);
|
||||||
|
message = s("Hello. You are client 2.\n");
|
||||||
|
socketWrite(client2, message.str, message.length);
|
||||||
|
|
||||||
forever {
|
|
||||||
string buf = PushStringFill(arena, 256, 0);
|
string buf = PushStringFill(arena, 256, 0);
|
||||||
uint64 bytesRead = socketRead(client, buf.str, buf.length - 1);
|
uint64 bytesRead = socketRead(client1, buf.str, buf.length - 1);
|
||||||
|
|
||||||
if (bytesRead > 0) {
|
if (bytesRead > 0) {
|
||||||
buf.length = bytesRead;
|
buf.length = bytesRead;
|
||||||
println("Client said: %S", strSplit(arena, s("\n"), buf).data[0]);
|
string message = strSplit(arena, s("\n"), buf).data[0];
|
||||||
|
message = strPrintf(arena, "Client 1 said: %S\n", message);
|
||||||
|
println("%S", message);
|
||||||
|
socketWrite(client2, message.str, message.length);
|
||||||
|
|
||||||
println("Saying goodbye");
|
println("Saying goodbye to everyone");
|
||||||
string message = s("Goodbye\n");
|
message = s("Goodbye\n");
|
||||||
socketWrite(client, message.str, message.length);
|
socketWrite(client1, message.str, message.length);
|
||||||
|
socketWrite(client2, message.str, message.length);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
serverClose(&myserver);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
socketClose(client);
|
socketClose(client1);
|
||||||
|
socketClose(client2);
|
||||||
|
serverClose(&myserver);
|
||||||
} else {
|
} else {
|
||||||
println("Connecting to socket at %S on port %d", addr, port);
|
println("Connecting to socket at %S on port %d", addr, port);
|
||||||
Socket sock = socketConnect(arena, (SocketConnectInfo){ .address=addr, .port=port });
|
Socket sock = socketConnect(arena, (SocketConnectInfo){ .address=addr, .port=port });
|
||||||
println("CONNECTED");
|
if (sock.closed) {
|
||||||
|
println("Connection error. Closing.");
|
||||||
|
} else {
|
||||||
string message;
|
string message;
|
||||||
uint64 bytesWritten;
|
uint64 bytesWritten;
|
||||||
|
|
||||||
string buf = PushStringFill(arena, 256, 0);
|
string buf = PushStringFill(arena, 256, 0);
|
||||||
|
|
||||||
forever {
|
|
||||||
message = s("Howdy partner\n");
|
|
||||||
print("Saying: %S", message);
|
|
||||||
bytesWritten = socketWrite(&sock, message.str, message.length);
|
|
||||||
|
|
||||||
socketRead(&sock, buf.str, buf.length);
|
socketRead(&sock, buf.str, buf.length);
|
||||||
message = strSplit(arena, s("\n"), buf).data[0];
|
string messageReceived = strSplit(arena, s("\n"), buf).data[0];
|
||||||
println("Received message: %S", message);
|
println("%S", strPrintf(arena, "Server said: %S", messageReceived));
|
||||||
if (strEql(message, s("Goodbye"))) {
|
|
||||||
|
if (strEql(messageReceived, s("Hello. You are client 1."))) {
|
||||||
|
string broadcast = s("HELLO WORLD!!!!\n");
|
||||||
|
socketWrite(&sock, broadcast.str, broadcast.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
Forever {
|
||||||
|
socketRead(&sock, buf.str, buf.length);
|
||||||
|
messageReceived = strSplit(arena, s("\n"), buf).data[0];
|
||||||
|
println("Server said: %S", messageReceived);
|
||||||
|
if (strEql(messageReceived, s("Goodbye"))) {
|
||||||
println("Quitting");
|
println("Quitting");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
socketClose(&sock);
|
socketClose(&sock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
2
core.h
2
core.h
@@ -16,7 +16,7 @@
|
|||||||
#define function static
|
#define function static
|
||||||
#define global static
|
#define global static
|
||||||
#define local_persist static
|
#define local_persist static
|
||||||
#define forever for (;;)
|
#define Forever for (;;)
|
||||||
|
|
||||||
#define DeferLoop(begin_stmnt, end_stmnt) for(int __defer_i = ((begin_stmnt), 0); __defer_i < 1; (++__defer_i, (end_stmnt)))
|
#define DeferLoop(begin_stmnt, end_stmnt) for(int __defer_i = ((begin_stmnt), 0); __defer_i < 1; (++__defer_i, (end_stmnt)))
|
||||||
|
|
||||||
|
|||||||
4
os.h
4
os.h
@@ -33,6 +33,9 @@ struct Address;
|
|||||||
typedef struct SocketHandle SocketHandle;
|
typedef struct SocketHandle SocketHandle;
|
||||||
struct SocketHandle;
|
struct SocketHandle;
|
||||||
|
|
||||||
|
typedef struct ServerEvents ServerEvents;
|
||||||
|
struct ServerEvents;
|
||||||
|
|
||||||
typedef struct Socket Socket;
|
typedef struct Socket Socket;
|
||||||
struct Socket {
|
struct Socket {
|
||||||
const Address *address;
|
const Address *address;
|
||||||
@@ -49,6 +52,7 @@ struct Server {
|
|||||||
SocketHandle *handle;
|
SocketHandle *handle;
|
||||||
SocketList clients;
|
SocketList clients;
|
||||||
bool listening;
|
bool listening;
|
||||||
|
ServerEvents *events;
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct ServerInitInfo ServerInitInfo;
|
typedef struct ServerInitInfo ServerInitInfo;
|
||||||
|
|||||||
32
os_linux.c
32
os_linux.c
@@ -8,6 +8,8 @@
|
|||||||
#include "unistd.h" // POSIX Standard
|
#include "unistd.h" // POSIX Standard
|
||||||
#include "stdio.h"
|
#include "stdio.h"
|
||||||
#include "pthread.h"
|
#include "pthread.h"
|
||||||
|
#include "fcntl.h"
|
||||||
|
#include "sys/epoll.h"
|
||||||
|
|
||||||
#include "sys/socket.h"
|
#include "sys/socket.h"
|
||||||
#include "arpa/inet.h"
|
#include "arpa/inet.h"
|
||||||
@@ -141,9 +143,17 @@ OS_Thread os_createThread(void *(*entry)(void *), void *ctx) {
|
|||||||
return (OS_Thread){ .id=handle };
|
return (OS_Thread){ .id=handle };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typedef struct EPollServerEvents EPollServerEvents;
|
||||||
|
struct EPollServerEvents {
|
||||||
|
int epollFd;
|
||||||
|
};
|
||||||
|
|
||||||
Server serverInit(ServerInitInfo info) {
|
Server serverInit(ServerInitInfo info) {
|
||||||
Arena *arena = arenaAlloc(info.memory);
|
Arena *arena = arenaAlloc(info.memory);
|
||||||
|
|
||||||
|
EPollServerEvents *events = PushStructZero(arena, EPollServerEvents);
|
||||||
|
events->epollFd = epoll_create1(0);
|
||||||
|
|
||||||
struct sockaddr_in6 *serverAddr = PushStructZero(arena, struct sockaddr_in6);
|
struct sockaddr_in6 *serverAddr = PushStructZero(arena, struct sockaddr_in6);
|
||||||
serverAddr->sin6_family = AF_INET6;
|
serverAddr->sin6_family = AF_INET6;
|
||||||
serverAddr->sin6_port = htons(info.port);
|
serverAddr->sin6_port = htons(info.port);
|
||||||
@@ -156,9 +166,12 @@ Server serverInit(ServerInitInfo info) {
|
|||||||
.listening=false,
|
.listening=false,
|
||||||
.port=info.port,
|
.port=info.port,
|
||||||
.handle=(SocketHandle *)(uint64)socket(AF_INET6, SOCK_STREAM, 0 /* IPPROTO_TCP */),
|
.handle=(SocketHandle *)(uint64)socket(AF_INET6, SOCK_STREAM, 0 /* IPPROTO_TCP */),
|
||||||
|
.events=(ServerEvents *)events,
|
||||||
};
|
};
|
||||||
|
|
||||||
int bindErr = bind((int)(uint64)server.handle, (struct sockaddr *)serverAddr, sizeof(*serverAddr));
|
fcntl((uint64)server.handle, F_SETFL, fcntl((uint64)server.handle, F_GETFL, 0) | O_NONBLOCK);
|
||||||
|
|
||||||
|
int bindErr = bind((uint64)server.handle, (struct sockaddr *)serverAddr, sizeof(*serverAddr));
|
||||||
if (bindErr == -1) {
|
if (bindErr == -1) {
|
||||||
// TODO(dledda): handle err
|
// TODO(dledda): handle err
|
||||||
}
|
}
|
||||||
@@ -169,7 +182,7 @@ Server serverInit(ServerInitInfo info) {
|
|||||||
void serverListen(Server *s) {
|
void serverListen(Server *s) {
|
||||||
int listenErr = listen((uint64)s->handle, s->clients.capacity);
|
int listenErr = listen((uint64)s->handle, s->clients.capacity);
|
||||||
if (listenErr == -1) {
|
if (listenErr == -1) {
|
||||||
// TODO(dledda): handle err
|
// TODO(dledda): handle err ?
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -177,19 +190,22 @@ Socket *serverAccept(Server *s) {
|
|||||||
struct sockaddr_in6 *clientAddr = PushStructZero(s->arena, struct sockaddr_in6);
|
struct sockaddr_in6 *clientAddr = PushStructZero(s->arena, struct sockaddr_in6);
|
||||||
socklen_t clientAddrLen = sizeof(*clientAddr);
|
socklen_t clientAddrLen = sizeof(*clientAddr);
|
||||||
|
|
||||||
uint64 clientSock = accept((int)(uint64)s->handle, (struct sockaddr *)clientAddr, &clientAddrLen);
|
uint64 clientSockHandle = accept((int)(uint64)s->handle, (struct sockaddr *)clientAddr, &clientAddrLen);
|
||||||
if (clientSock == -1) {
|
if (clientSockHandle == -1) {
|
||||||
// TODO(dledda): handle err
|
clientSockHandle = (uint64)NULL;
|
||||||
|
} else {
|
||||||
|
fcntl((uint64)clientSockHandle, F_SETFL, fcntl((uint64)clientSockHandle, F_GETFL, 0) | O_NONBLOCK);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (s->clients.length < s->clients.capacity) {
|
if (s->clients.length < s->clients.capacity) {
|
||||||
AppendList(&s->clients, ((Socket){
|
AppendList(&s->clients, ((Socket){
|
||||||
.handle=(SocketHandle *)(uint64)clientSock,
|
.handle=(SocketHandle *)(uint64)clientSockHandle,
|
||||||
.address=(Address *)clientAddr,
|
.address=(Address *)clientAddr,
|
||||||
}));
|
}));
|
||||||
}
|
|
||||||
|
|
||||||
return &s->clients.data[s->clients.length - 1];
|
return &s->clients.data[s->clients.length - 1];
|
||||||
|
} else {
|
||||||
|
return PushStructZero(s->arena, Socket);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int64 socketRead(Socket *socket, byte *dest, uint64 numBytes) {
|
int64 socketRead(Socket *socket, byte *dest, uint64 numBytes) {
|
||||||
|
|||||||
Reference in New Issue
Block a user