source: example/io/simple/server-nonblk.c@ a97b9ed

Last change on this file since a97b9ed was 0640189e, checked in by Thierry Delisle <tdelisle@…>, 4 years ago

Added example of a server using o_NONBLOCK, which does NOT work with io_uring

  • Property mode set to 100644
File size: 3.2 KB
RevLine 
[0640189e]1/*
2This is a simple server that users io_uring in blocking mode.
3It demonstrates the bare minimum needed to use io_uring.
4It uses liburing for simplicity.
5*/
6
7
8#include <assert.h>
9#include <stdio.h>
10#include <stdlib.h>
11#include <string.h>
12
13#include <unistd.h>
14
15#include <sys/types.h>
16#include <sys/socket.h>
17#include <netinet/in.h>
18
19#include <liburing.h>
20
21extern int accept4(int sockfd, struct sockaddr *addr, socklen_t *addrlen, int flags);
22
23struct io_uring ring;
24
25char data[256];
26struct iovec iov = { data, 256 };
27struct msghdr msg = { (void *)"", 0, &iov, 1, NULL, 0, 0 };
28static int async_read(int sock) {
29 int ret;
30
31 /* get an sqe and fill in a READ operation */
32 struct io_uring_sqe * sqe = io_uring_get_sqe(&ring);
33 io_uring_prep_recvmsg(sqe, sock, &msg, 0);
34 sqe->user_data = 0;
35
36 /* tell the kernel we have an sqe ready for consumption */
37 ret = io_uring_submit(&ring);
38 assert(ret == 1);
39
40 struct io_uring_cqe * cqe;
41 ret = io_uring_wait_cqe( &ring, &cqe );
42
43 if( ret < 0 ) {
44 printf( "Main Loop Error : %s\n", strerror(-ret) );
45 close( sock );
46 exit( EXIT_FAILURE );
47 }
48
49 // Problem ?
50 if(cqe->user_data != 0) {
51 printf("Unexpected user data : %llu", cqe->user_data);
52 exit( EXIT_FAILURE );
53 }
54
55 int res = cqe->res;
56 io_uring_cqe_seen( &ring, cqe );
57
58 if(res < 0 && (errno == EAGAIN || errno == EWOULDBLOCK)) {
59 printf("io_uring returned EAGAIN\n");
60 }
61
62 return res;
63}
64
65static int do_read(int sock) {
66 int res = recvmsg(sock, &msg, 0);
67
68 if(res < 0 && (errno == EAGAIN || errno == EWOULDBLOCK)) {
69 printf("Had EAGAIN\n");
70 return async_read(sock);
71 }
72
73 return res;
74}
75
76int main(int argc, char *argv[]) {
77 if(argc != 2) {
78 printf("usage: %s portnumber\n", argv[0]);
79 exit( EXIT_FAILURE );
80 }
81 int port = atoi(argv[1]);
82 if(port < 1) {
83 printf("Invalid port : %d (from %s)\n", port, argv[1]);
84 exit( EXIT_FAILURE );
85 }
86
87 int sock = socket(AF_INET, SOCK_STREAM, 0);
88 if(sock < 0) {
89 perror( "socket" );
90 exit( EXIT_FAILURE );
91 }
92
93 struct sockaddr_in serv_addr;
94 memset(&serv_addr, 0, sizeof(serv_addr));
95 serv_addr.sin_family = AF_INET;
96 serv_addr.sin_addr.s_addr = INADDR_ANY;
97 serv_addr.sin_port = htons(port);
98
99 int ret = bind(sock, (struct sockaddr *) &serv_addr, sizeof(serv_addr));
100 if(ret < 0) {
101 perror( "bind" );
102 exit( EXIT_FAILURE );
103 }
104
105 listen(sock,1);
106
107 struct sockaddr_in cli_addr;
108 __socklen_t clilen = sizeof(cli_addr);
109 int newsock = accept4(sock, (struct sockaddr *) &cli_addr, &clilen, SOCK_NONBLOCK);
110 if (newsock < 0) {
111 perror( "accept" );
112 exit( EXIT_FAILURE );
113 }
114
115 io_uring_queue_init( 16, &ring, 0 );
116
117 while(1) {
118 int res = do_read(newsock);
119
120 // Did we get an error
121 if( res < 0 ) {
122 perror( "Main Loop Error" );
123 close( sock );
124 exit( EXIT_FAILURE );
125 }
126
127 // If it is the end of file we are done
128 if( res == 0 ) {
129 goto END;
130 }
131
132 // just echo the data
133 printf("'%.*s'\n", res - 1, data);
134 }
135END:
136
137 io_uring_queue_exit( &ring );
138
139 ret = close(newsock);
140 if(ret < 0) {
141 perror( "close new" );
142 exit( EXIT_FAILURE );
143 }
144
145 ret = close(sock);
146 if(ret < 0) {
147 perror( "close old" );
148 exit( EXIT_FAILURE );
149 }
150
151 return 0;
152}
Note: See TracBrowser for help on using the repository browser.