#include #include #include #include #include #include #include #include #include #include #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; }