#include #include #include #include #include #include #include #include "common.h" typedef struct { int sock; uint8_t *recv_seq; struct sockaddr_in6 *peer; } recv_args; void *receiver(void *arg) { recv_args *a = (recv_args *)arg; while (1) { fd_set fds; FD_ZERO(&fds); FD_SET(a->sock, &fds); struct timeval tv = {0, 100000}; if (select(a->sock + 1, &fds, NULL, NULL, &tv) > 0) { struct packet rcv; recvfrom(a->sock, &rcv, sizeof(rcv), 0, NULL, NULL); rcv.payload[sizeof(rcv.payload) - 1] = '\0'; if (rcv.type == TYPE_DATA) { if (rcv.seq == *a->recv_seq) { printf("\npeer: %s\n", rcv.payload); *a->recv_seq ^= 1; } rcv.type = TYPE_ACK; sendto(a->sock, &rcv, 2, 0, (struct sockaddr *)a->peer, sizeof(*a->peer)); } } } return NULL; } /* 停等协议发送(带超时重传) */ void send_with_stopwait(int sock, struct sockaddr_in6 *peer, struct packet *pkt, int pkt_len) { socklen_t len = sizeof(*peer); while (1) { sendto(sock, pkt, pkt_len, 0, (struct sockaddr *)peer, len); fd_set fds; FD_ZERO(&fds); FD_SET(sock, &fds); struct timeval tv; tv.tv_sec = TIMEOUT_SEC; tv.tv_usec = 0; int ret = select(sock + 1, &fds, NULL, NULL, &tv); if (ret > 0) { struct packet ack; recvfrom(sock, &ack, sizeof(ack), 0, NULL, NULL); if (ack.type == TYPE_ACK && ack.seq == pkt->seq) { return; // 正确 ACK } } printf("[timeout] retransmit\n"); } } int main(int argc, char *argv[]) { if (argc != 2) { printf("用法: %s <对方IPv6地址>\n", argv[0]); return 1; } int sock = socket(AF_INET6, SOCK_DGRAM, 0); /* 绑定本地 IPv6 端口 */ struct sockaddr_in6 local; memset(&local, 0, sizeof(local)); local.sin6_family = AF_INET6; local.sin6_addr = in6addr_any; local.sin6_port = htons(PORT); bind(sock, (struct sockaddr *)&local, sizeof(local)); /* 对端地址 */ struct sockaddr_in6 peer; memset(&peer, 0, sizeof(peer)); peer.sin6_family = AF_INET6; peer.sin6_port = htons(PORT); if (inet_pton(AF_INET6, argv[1], &peer.sin6_addr) != 1) { printf("无效的 IPv6 地址\n"); return 1; } uint8_t send_seq = 0; uint8_t recv_seq = 0; struct packet pkt; memset(&pkt, 0, sizeof(pkt)); printf("Linux UDP IPv6 Stop-Wait Chat\n"); // 启动接收线程 recv_args args = {sock, &recv_seq, &peer}; pthread_t recv_thread; pthread_create(&recv_thread, NULL, receiver, &args); while (1) { printf(">> "); if (!fgets(pkt.payload, sizeof(pkt.payload), stdin)) break; size_t len = strlen(pkt.payload); if (len > 0 && pkt.payload[len - 1] == '\n') pkt.payload[len - 1] = '\0'; pkt.seq = send_seq; pkt.type = TYPE_DATA; int pkt_len = 2 + strlen(pkt.payload) + 1; send_with_stopwait(sock, &peer, &pkt, pkt_len); send_seq ^= 1; } close(sock); return 0; }