1 /* Copyright (C) 2007-2008 The Android Open Source Project
2 **
3 ** This software is licensed under the terms of the GNU General Public
4 ** License version 2, as published by the Free Software Foundation, and
5 ** may be copied, distributed, and modified under those terms.
6 **
7 ** This program is distributed in the hope that it will be useful,
8 ** but WITHOUT ANY WARRANTY; without even the implied warranty of
9 ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 ** GNU General Public License for more details.
11 */
12 #include "sockets.h"
13 #include "sysdeps.h"
14 #include "qemu-common.h"
15 #include "qemu-timer.h"
16 #include "qemu-char.h"
17 #ifdef _WIN32
18 #include <winsock2.h>
19 #else
20 #include <sys/socket.h>
21 #include <sys/select.h>
22 #include <sys/types.h>
23 #include <netinet/in.h>
24 #include <netinet/tcp.h>
25 #include <netdb.h>
26 #endif
27
28 #define DEBUG 0
29
30 #define D_ACTIVE DEBUG
31
32 #if DEBUG
33 #define D(...) do { if (D_ACTIVE) fprintf(stderr, __VA_ARGS__); } while (0)
34 #else
35 #define D(...) ((void)0)
36 #endif
37
38 /** TIME
39 **/
40
41 SysTime
sys_time_ms(void)42 sys_time_ms( void )
43 {
44 return qemu_get_clock_ms(rt_clock);
45 }
46
47 /** TIMERS
48 **/
49
50 typedef struct SysTimerRec_ {
51 QEMUTimer* timer;
52 QEMUTimerCB* callback;
53 void* opaque;
54 SysTimer next;
55 } SysTimerRec;
56
57 #define MAX_TIMERS 32
58
59 static SysTimerRec _s_timers0[ MAX_TIMERS ];
60 static SysTimer _s_free_timers;
61
62 static void
sys_init_timers(void)63 sys_init_timers( void )
64 {
65 int nn;
66 for (nn = 0; nn < MAX_TIMERS-1; nn++)
67 _s_timers0[nn].next = _s_timers0 + (nn+1);
68
69 _s_free_timers = _s_timers0;
70 }
71
72 static SysTimer
sys_timer_alloc(void)73 sys_timer_alloc( void )
74 {
75 SysTimer timer = _s_free_timers;
76
77 if (timer != NULL) {
78 _s_free_timers = timer->next;
79 timer->next = NULL;
80 timer->timer = NULL;
81 }
82 return timer;
83 }
84
85
86 static void
sys_timer_free(SysTimer timer)87 sys_timer_free( SysTimer timer )
88 {
89 if (timer->timer) {
90 qemu_del_timer( timer->timer );
91 qemu_free_timer( timer->timer );
92 timer->timer = NULL;
93 }
94 timer->next = _s_free_timers;
95 _s_free_timers = timer;
96 }
97
98
sys_timer_create(void)99 SysTimer sys_timer_create( void )
100 {
101 SysTimer timer = sys_timer_alloc();
102 return timer;
103 }
104
105 void
sys_timer_set(SysTimer timer,SysTime when,SysCallback _callback,void * opaque)106 sys_timer_set( SysTimer timer, SysTime when, SysCallback _callback, void* opaque )
107 {
108 QEMUTimerCB* callback = (QEMUTimerCB*)_callback;
109
110 if (callback == NULL) { /* unsetting the timer */
111 if (timer->timer) {
112 qemu_del_timer( timer->timer );
113 qemu_free_timer( timer->timer );
114 timer->timer = NULL;
115 }
116 timer->callback = callback;
117 timer->opaque = NULL;
118 return;
119 }
120
121 if ( timer->timer ) {
122 if ( timer->callback == callback && timer->opaque == opaque )
123 goto ReuseTimer;
124
125 /* need to replace the timer */
126 qemu_free_timer( timer->timer );
127 }
128
129 timer->timer = qemu_new_timer_ms( rt_clock, callback, opaque );
130 timer->callback = callback;
131 timer->opaque = opaque;
132
133 ReuseTimer:
134 qemu_mod_timer( timer->timer, when );
135 }
136
137 void
sys_timer_unset(SysTimer timer)138 sys_timer_unset( SysTimer timer )
139 {
140 if (timer->timer) {
141 qemu_del_timer( timer->timer );
142 }
143 }
144
145 void
sys_timer_destroy(SysTimer timer)146 sys_timer_destroy( SysTimer timer )
147 {
148 sys_timer_free( timer );
149 }
150
151
152 /** CHANNELS
153 **/
154
155 typedef struct SysChannelRec_ {
156 int fd;
157 SysChannelCallback callback;
158 void* opaque;
159 SysChannel next;
160 } SysChannelRec;
161
162 #define MAX_CHANNELS 16
163
164 static SysChannelRec _s_channels0[ MAX_CHANNELS ];
165 static SysChannel _s_free_channels;
166
167 static void
sys_init_channels(void)168 sys_init_channels( void )
169 {
170 int nn;
171
172 for ( nn = 0; nn < MAX_CHANNELS-1; nn++ ) {
173 _s_channels0[nn].next = _s_channels0 + (nn+1);
174 }
175 _s_free_channels = _s_channels0;
176 }
177
178 static SysChannel
sys_channel_alloc()179 sys_channel_alloc( )
180 {
181 SysChannel channel = _s_free_channels;
182 if (channel != NULL) {
183 _s_free_channels = channel->next;
184 channel->next = NULL;
185 channel->fd = -1;
186 channel->callback = NULL;
187 channel->opaque = NULL;
188 }
189 return channel;
190 }
191
192 static void
sys_channel_free(SysChannel channel)193 sys_channel_free( SysChannel channel )
194 {
195 if (channel->fd >= 0) {
196 socket_close( channel->fd );
197 channel->fd = -1;
198 }
199 channel->next = _s_free_channels;
200 _s_free_channels = channel;
201 }
202
203
204 static void
sys_channel_read_handler(void * _channel)205 sys_channel_read_handler( void* _channel )
206 {
207 SysChannel channel = _channel;
208 D( "%s: read event for channel %p:%d\n", __FUNCTION__,
209 channel, channel->fd );
210 channel->callback( channel->opaque, SYS_EVENT_READ );
211 }
212
213 static void
sys_channel_write_handler(void * _channel)214 sys_channel_write_handler( void* _channel )
215 {
216 SysChannel channel = _channel;
217 D( "%s: write event for channel %p:%d\n", __FUNCTION__, channel, channel->fd );
218 channel->callback( channel->opaque, SYS_EVENT_WRITE );
219 }
220
221 void
sys_channel_on(SysChannel channel,int events,SysChannelCallback event_callback,void * event_opaque)222 sys_channel_on( SysChannel channel,
223 int events,
224 SysChannelCallback event_callback,
225 void* event_opaque )
226 {
227 IOHandler* read_handler = NULL;
228 IOHandler* write_handler = NULL;
229
230 if (events & SYS_EVENT_READ) {
231 read_handler = sys_channel_read_handler;
232 }
233 if (events & SYS_EVENT_WRITE) {
234 write_handler = sys_channel_write_handler;
235 }
236 channel->callback = event_callback;
237 channel->opaque = event_opaque;
238 qemu_set_fd_handler( channel->fd, read_handler, write_handler, channel );
239 }
240
241 int
sys_channel_read(SysChannel channel,void * buffer,int size)242 sys_channel_read( SysChannel channel, void* buffer, int size )
243 {
244 int len = size;
245 char* buf = (char*) buffer;
246
247 while (len > 0) {
248 int ret = socket_recv(channel->fd, buf, len);
249 if (ret < 0) {
250 if (errno == EINTR)
251 continue;
252 if (errno == EWOULDBLOCK || errno == EAGAIN)
253 break;
254 D( "%s: after reading %d bytes, recv() returned error %d: %s\n",
255 __FUNCTION__, size - len, errno, errno_str);
256 return -1;
257 } else if (ret == 0) {
258 break;
259 } else {
260 buf += ret;
261 len -= ret;
262 }
263 }
264 return size - len;
265 }
266
267
268 int
sys_channel_write(SysChannel channel,const void * buffer,int size)269 sys_channel_write( SysChannel channel, const void* buffer, int size )
270 {
271 int len = size;
272 const char* buf = (const char*) buffer;
273
274 while (len > 0) {
275 int ret = socket_send(channel->fd, buf, len);
276 if (ret < 0) {
277 if (errno == EINTR)
278 continue;
279 if (errno == EWOULDBLOCK || errno == EAGAIN)
280 break;
281 D( "%s: send() returned error %d: %s\n",
282 __FUNCTION__, errno, errno_str);
283 return -1;
284 } else if (ret == 0) {
285 break;
286 } else {
287 buf += ret;
288 len -= ret;
289 }
290 }
291 return size - len;
292 }
293
sys_channel_close(SysChannel channel)294 void sys_channel_close( SysChannel channel )
295 {
296 qemu_set_fd_handler( channel->fd, NULL, NULL, NULL );
297 sys_channel_free( channel );
298 }
299
sys_main_init(void)300 void sys_main_init( void )
301 {
302 sys_init_channels();
303 sys_init_timers();
304 }
305
306
sys_main_loop(void)307 int sys_main_loop( void )
308 {
309 /* no looping, qemu has its own event loop */
310 return 0;
311 }
312
313
314
315
316 SysChannel
sys_channel_create_tcp_server(int port)317 sys_channel_create_tcp_server( int port )
318 {
319 SysChannel channel = sys_channel_alloc();
320
321 channel->fd = socket_anyaddr_server( port, SOCKET_STREAM );
322 if (channel->fd < 0) {
323 D( "%s: failed to created network socket on TCP:%d\n",
324 __FUNCTION__, port );
325 sys_channel_free( channel );
326 return NULL;
327 }
328
329 D( "%s: server channel %p:%d now listening on port %d\n",
330 __FUNCTION__, channel, channel->fd, port );
331
332 return channel;
333 }
334
335
336 SysChannel
sys_channel_create_tcp_handler(SysChannel server_channel)337 sys_channel_create_tcp_handler( SysChannel server_channel )
338 {
339 SysChannel channel = sys_channel_alloc();
340
341 D( "%s: creating handler from server channel %p:%d\n", __FUNCTION__,
342 server_channel, server_channel->fd );
343
344 channel->fd = socket_accept_any( server_channel->fd );
345 if (channel->fd < 0) {
346 perror( "accept" );
347 sys_channel_free( channel );
348 return NULL;
349 }
350
351 /* disable Nagle algorithm */
352 socket_set_nodelay( channel->fd );
353
354 D( "%s: handler %p:%d created from server %p:%d\n", __FUNCTION__,
355 server_channel, server_channel->fd, channel, channel->fd );
356
357 return channel;
358 }
359
360
361 SysChannel
sys_channel_create_tcp_client(const char * hostname,int port)362 sys_channel_create_tcp_client( const char* hostname, int port )
363 {
364 SysChannel channel = sys_channel_alloc();
365
366 channel->fd = socket_network_client( hostname, port, SOCKET_STREAM );
367 if (channel->fd < 0) {
368 sys_channel_free(channel);
369 return NULL;
370 };
371
372 /* set to non-blocking and disable Nagle algorithm */
373 socket_set_nonblock( channel->fd );
374 socket_set_nodelay( channel->fd );
375
376 return channel;
377 }
378
379