Files
NE_YuR/network/tcpquiclab/quic_server.c
2025-12-25 15:14:14 +08:00

176 lines
6.6 KiB
C

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <quiche.h>
#define LOCAL_CONN_ID_LEN 16
#define MAX_DATAGRAM_SIZE 1350
typedef struct {
int sock;
struct sockaddr_storage peer_addr;
socklen_t peer_addr_len;
quiche_conn *conn;
} Client;
int main(int argc, char *argv[]) {
quiche_config *config = quiche_config_new(QUICHE_PROTOCOL_VERSION);
if (config == NULL) {
fprintf(stderr, "failed to create config\n");
return -1;
}
if (quiche_config_load_cert_chain_from_pem_file(config, "cert.crt") < 0) {
fprintf(stderr, "failed to load certificate chain\n");
return -1;
}
if (quiche_config_load_priv_key_from_pem_file(config, "cert.key") < 0) {
fprintf(stderr, "failed to load private key\n");
return -1;
}
quiche_config_set_application_protos(config, (uint8_t *) "\x0ahq-interop\x05hq-29\x05hq-28\x05hq-27\x08http/0.9", 38);
quiche_config_set_max_idle_timeout(config, 5000);
quiche_config_set_max_recv_udp_payload_size(config, MAX_DATAGRAM_SIZE);
quiche_config_set_max_send_udp_payload_size(config, MAX_DATAGRAM_SIZE);
quiche_config_set_initial_max_data(config, 10000000);
quiche_config_set_initial_max_stream_data_bidi_local(config, 1000000);
quiche_config_set_initial_max_stream_data_bidi_remote(config, 1000000);
quiche_config_set_initial_max_streams_bidi(config, 100);
quiche_config_set_cc_algorithm(config, QUICHE_CC_RENO);
struct sockaddr_in sa;
memset(&sa, 0, sizeof(sa));
sa.sin_family = AF_INET;
sa.sin_port = htons(8888);
sa.sin_addr.s_addr = INADDR_ANY;
int sock = socket(AF_INET, SOCK_DGRAM, 0);
if (sock < 0) {
perror("socket");
return -1;
}
if (bind(sock, (struct sockaddr *)&sa, sizeof(sa)) < 0) {
perror("bind");
return -1;
}
int flags = fcntl(sock, F_GETFL, 0);
fcntl(sock, F_SETFL, flags | O_NONBLOCK);
printf("QUIC Server listening on port 8888\n");
Client *client = NULL;
uint8_t buf[65535];
uint8_t out[MAX_DATAGRAM_SIZE];
while (1) {
struct sockaddr_storage peer_addr;
socklen_t peer_addr_len = sizeof(peer_addr);
ssize_t read_len = recvfrom(sock, buf, sizeof(buf), 0, (struct sockaddr *)&peer_addr, &peer_addr_len);
if (read_len < 0) {
if (errno != EWOULDBLOCK && errno != EAGAIN) {
perror("recvfrom");
break;
}
} else {
uint8_t type;
uint32_t version;
uint8_t scid[QUICHE_MAX_CONN_ID_LEN];
size_t scid_len = sizeof(scid);
uint8_t dcid[QUICHE_MAX_CONN_ID_LEN];
size_t dcid_len = sizeof(dcid);
uint8_t token[256];
size_t token_len = sizeof(token);
int rc = quiche_header_info(buf, read_len, LOCAL_CONN_ID_LEN, &version, &type, scid, &scid_len, dcid, &dcid_len, token, &token_len);
if (rc >= 0) {
if (client == NULL) {
if (!quiche_version_is_supported(version)) {
ssize_t written = quiche_negotiate_version(scid, scid_len, dcid, dcid_len, out, sizeof(out));
if (written > 0) sendto(sock, out, written, 0, (struct sockaddr *)&peer_addr, peer_addr_len);
} else {
// Skip retry and accept connection directly
client = malloc(sizeof(Client));
client->sock = sock;
client->peer_addr = peer_addr;
client->peer_addr_len = peer_addr_len;
uint8_t server_scid[QUICHE_MAX_CONN_ID_LEN];
int rng = open("/dev/urandom", O_RDONLY);
if (rng >= 0) {
read(rng, server_scid, sizeof(server_scid));
close(rng);
}
client->conn = quiche_accept(server_scid, sizeof(server_scid), dcid, dcid_len, (struct sockaddr *)&sa, sizeof(sa), (struct sockaddr *)&peer_addr, peer_addr_len, config);
printf("New connection accepted.\n");
}
}
if (client != NULL) {
quiche_conn_recv(client->conn, buf, read_len, &(quiche_recv_info){
.to = (struct sockaddr *)&sa,
.to_len = sizeof(sa),
.from = (struct sockaddr *)&peer_addr,
.from_len = peer_addr_len,
});
}
}
}
if (client != NULL) {
quiche_conn *conn = client->conn;
if (quiche_conn_is_closed(conn)) {
printf("Connection closed.\n");
quiche_conn_free(conn);
free(client);
client = NULL;
break; // Exit the main loop and terminate
}
if (quiche_conn_is_established(conn)) {
uint64_t s = 0;
quiche_stream_iter *readable = quiche_conn_readable(conn);
while (quiche_stream_iter_next(readable, &s)) {
uint8_t recv_buf[1024];
bool fin = false;
uint64_t err_code = 0;
ssize_t recv_bytes = quiche_conn_stream_recv(conn, s, recv_buf, sizeof(recv_buf), &fin, &err_code);
if (recv_bytes > 0) {
printf("Received %zd bytes on stream %lu: %.*s\n", recv_bytes, s, (int)recv_bytes, recv_buf);
char resp[1200];
snprintf(resp, sizeof(resp), "Server received: %.*s", (int)recv_bytes, recv_buf);
quiche_conn_stream_send(conn, s, (uint8_t*)resp, strlen(resp), true, &err_code);
}
}
quiche_stream_iter_free(readable);
}
while (1) {
quiche_send_info send_info;
ssize_t written = quiche_conn_send(conn, out, sizeof(out), &send_info);
if (written == QUICHE_ERR_DONE) break;
if (written < 0) break;
sendto(sock, out, written, 0, (struct sockaddr *)&send_info.to, send_info.to_len);
}
quiche_conn_on_timeout(conn);
}
usleep(1000);
}
quiche_config_free(config);
return 0;
}