1 #include "iolooper.h" 2 #include "qemu-common.h" 3 4 /* An implementation of iolooper.h based on Unix select() */ 5 #ifdef _WIN32 6 # include <winsock2.h> 7 #else 8 # include <sys/types.h> 9 # include <sys/select.h> 10 #endif 11 #include "sockets.h" 12 13 struct IoLooper { 14 fd_set reads[1]; 15 fd_set writes[1]; 16 fd_set reads_result[1]; 17 fd_set writes_result[1]; 18 int max_fd; 19 int max_fd_valid; 20 }; 21 22 IoLooper* 23 iolooper_new(void) 24 { 25 IoLooper* iol = malloc(sizeof(*iol)); 26 iolooper_reset(iol); 27 return iol; 28 } 29 30 void 31 iolooper_free( IoLooper* iol ) 32 { 33 free(iol); 34 } 35 36 void 37 iolooper_reset( IoLooper* iol ) 38 { 39 FD_ZERO(iol->reads); 40 FD_ZERO(iol->writes); 41 iol->max_fd = -1; 42 iol->max_fd_valid = 1; 43 } 44 45 static void 46 iolooper_add_fd( IoLooper* iol, int fd ) 47 { 48 if (iol->max_fd_valid && fd > iol->max_fd) { 49 iol->max_fd = fd; 50 } 51 } 52 53 static void 54 iolooper_del_fd( IoLooper* iol, int fd ) 55 { 56 if (iol->max_fd_valid && fd == iol->max_fd) 57 iol->max_fd_valid = 0; 58 } 59 60 void 61 iolooper_modify( IoLooper* iol, int fd, int oldflags, int newflags ) 62 { 63 if (fd < 0) 64 return; 65 66 int changed = oldflags ^ newflags; 67 68 if ((changed & IOLOOPER_READ) != 0) { 69 if ((newflags & IOLOOPER_READ) != 0) 70 iolooper_add_read(iol, fd); 71 else 72 iolooper_del_read(iol, fd); 73 } 74 if ((changed & IOLOOPER_WRITE) != 0) { 75 if ((newflags & IOLOOPER_WRITE) != 0) 76 iolooper_add_write(iol, fd); 77 else 78 iolooper_del_write(iol, fd); 79 } 80 } 81 82 83 static int 84 iolooper_fd_count( IoLooper* iol ) 85 { 86 int max_fd = iol->max_fd; 87 int fd; 88 89 if (iol->max_fd_valid) 90 return max_fd + 1; 91 92 /* recompute max fd */ 93 for (fd = 0; fd < FD_SETSIZE; fd++) { 94 if (!FD_ISSET(fd, iol->reads) && !FD_ISSET(fd, iol->writes)) 95 continue; 96 97 max_fd = fd; 98 } 99 iol->max_fd = max_fd; 100 iol->max_fd_valid = 1; 101 102 return max_fd + 1; 103 } 104 105 void 106 iolooper_add_read( IoLooper* iol, int fd ) 107 { 108 if (fd >= 0) { 109 iolooper_add_fd(iol, fd); 110 FD_SET(fd, iol->reads); 111 } 112 } 113 114 void 115 iolooper_add_write( IoLooper* iol, int fd ) 116 { 117 if (fd >= 0) { 118 iolooper_add_fd(iol, fd); 119 FD_SET(fd, iol->writes); 120 } 121 } 122 123 void 124 iolooper_del_read( IoLooper* iol, int fd ) 125 { 126 if (fd >= 0) { 127 iolooper_del_fd(iol, fd); 128 FD_CLR(fd, iol->reads); 129 } 130 } 131 132 void 133 iolooper_del_write( IoLooper* iol, int fd ) 134 { 135 if (fd >= 0) { 136 iolooper_del_fd(iol, fd); 137 FD_CLR(fd, iol->writes); 138 } 139 } 140 141 int 142 iolooper_poll( IoLooper* iol ) 143 { 144 int count = iolooper_fd_count(iol); 145 int ret; 146 fd_set errs; 147 148 if (count == 0) 149 return 0; 150 151 FD_ZERO(&errs); 152 153 do { 154 struct timeval tv; 155 156 tv.tv_sec = tv.tv_usec = 0; 157 158 iol->reads_result[0] = iol->reads[0]; 159 iol->writes_result[0] = iol->writes[0]; 160 161 ret = select( count, iol->reads_result, iol->writes_result, &errs, &tv); 162 } while (ret < 0 && errno == EINTR); 163 164 return ret; 165 } 166 167 int 168 iolooper_wait( IoLooper* iol, int64_t duration ) 169 { 170 int count = iolooper_fd_count(iol); 171 int ret; 172 fd_set errs; 173 struct timeval tm0, *tm = NULL; 174 175 if (count == 0) 176 return 0; 177 178 CLAMP_MAC_TIMEOUT(duration); 179 180 if (duration < 0) 181 tm = NULL; 182 else { 183 tm = &tm0; 184 tm->tv_sec = duration / 1000; 185 tm->tv_usec = (duration - 1000*tm->tv_sec) * 1000; 186 } 187 188 FD_ZERO(&errs); 189 190 do { 191 iol->reads_result[0] = iol->reads[0]; 192 iol->writes_result[0] = iol->writes[0]; 193 194 ret = select( count, iol->reads_result, iol->writes_result, &errs, tm); 195 if (ret == 0) { 196 // Indicates timeout 197 errno = ETIMEDOUT; 198 } 199 } while (ret < 0 && errno == EINTR); 200 201 return ret; 202 } 203 204 205 int 206 iolooper_is_read( IoLooper* iol, int fd ) 207 { 208 return FD_ISSET(fd, iol->reads_result); 209 } 210 211 int 212 iolooper_is_write( IoLooper* iol, int fd ) 213 { 214 return FD_ISSET(fd, iol->writes_result); 215 } 216 217 int 218 iolooper_has_operations( IoLooper* iol ) 219 { 220 return iolooper_fd_count(iol) > 0; 221 } 222 223 int64_t 224 iolooper_now(void) 225 { 226 struct timeval time_now; 227 return gettimeofday(&time_now, NULL) ? -1 : (int64_t)time_now.tv_sec * 1000LL + 228 time_now.tv_usec / 1000; 229 } 230 231 int 232 iolooper_wait_absolute(IoLooper* iol, int64_t deadline) 233 { 234 int64_t timeout = deadline - iolooper_now(); 235 236 /* If the deadline has passed, set the timeout to 0, this allows us 237 * to poll the file descriptor nonetheless */ 238 if (timeout < 0) 239 timeout = 0; 240 241 return iolooper_wait(iol, timeout); 242 } 243