| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * QEMU System Emulator | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2003-2008 Fabrice Bellard | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a copy | 
					
						
							|  |  |  |  * of this software and associated documentation files (the "Software"), to deal | 
					
						
							|  |  |  |  * in the Software without restriction, including without limitation the rights | 
					
						
							|  |  |  |  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
					
						
							|  |  |  |  * copies of the Software, and to permit persons to whom the Software is | 
					
						
							|  |  |  |  * furnished to do so, subject to the following conditions: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice shall be included in | 
					
						
							|  |  |  |  * all copies or substantial portions of the Software. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | 
					
						
							|  |  |  |  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
					
						
							|  |  |  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
					
						
							|  |  |  |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | 
					
						
							|  |  |  |  * THE SOFTWARE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "qemu-common.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:19:44 +01:00
										 |  |  | #include "block/aio.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:20:00 +01:00
										 |  |  | #include "qemu/main-loop.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  | /***********************************************************/ | 
					
						
							|  |  |  | /* bottom halves (can be seen as timers which expire ASAP) */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct QEMUBH { | 
					
						
							| 
									
										
										
										
											2012-09-24 18:44:14 +02:00
										 |  |  |     AioContext *ctx; | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |     QEMUBHFunc *cb; | 
					
						
							|  |  |  |     void *opaque; | 
					
						
							|  |  |  |     QEMUBH *next; | 
					
						
							| 
									
										
										
										
											2012-04-29 19:08:45 +02:00
										 |  |  |     bool scheduled; | 
					
						
							|  |  |  |     bool idle; | 
					
						
							|  |  |  |     bool deleted; | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-29 23:45:23 +01:00
										 |  |  | QEMUBH *aio_bh_new(AioContext *ctx, QEMUBHFunc *cb, void *opaque) | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     QEMUBH *bh; | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |     bh = g_malloc0(sizeof(QEMUBH)); | 
					
						
							| 
									
										
										
										
											2012-09-24 18:44:14 +02:00
										 |  |  |     bh->ctx = ctx; | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |     bh->cb = cb; | 
					
						
							|  |  |  |     bh->opaque = opaque; | 
					
						
							| 
									
										
										
										
											2012-10-29 23:45:23 +01:00
										 |  |  |     bh->next = ctx->first_bh; | 
					
						
							|  |  |  |     ctx->first_bh = bh; | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |     return bh; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-29 23:45:23 +01:00
										 |  |  | int aio_bh_poll(AioContext *ctx) | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-06-07 17:51:21 +02:00
										 |  |  |     QEMUBH *bh, **bhp, *next; | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |     int ret; | 
					
						
							| 
									
										
										
										
											2011-09-01 16:16:10 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-29 23:45:23 +01:00
										 |  |  |     ctx->walking_bh++; | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ret = 0; | 
					
						
							| 
									
										
										
										
											2012-10-29 23:45:23 +01:00
										 |  |  |     for (bh = ctx->first_bh; bh; bh = next) { | 
					
						
							| 
									
										
										
										
											2011-06-07 17:51:21 +02:00
										 |  |  |         next = bh->next; | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |         if (!bh->deleted && bh->scheduled) { | 
					
						
							|  |  |  |             bh->scheduled = 0; | 
					
						
							|  |  |  |             if (!bh->idle) | 
					
						
							|  |  |  |                 ret = 1; | 
					
						
							|  |  |  |             bh->idle = 0; | 
					
						
							|  |  |  |             bh->cb(bh->opaque); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-29 23:45:23 +01:00
										 |  |  |     ctx->walking_bh--; | 
					
						
							| 
									
										
										
										
											2011-09-01 16:16:10 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |     /* remove deleted bhs */ | 
					
						
							| 
									
										
										
										
											2012-10-29 23:45:23 +01:00
										 |  |  |     if (!ctx->walking_bh) { | 
					
						
							|  |  |  |         bhp = &ctx->first_bh; | 
					
						
							| 
									
										
										
										
											2011-09-01 16:16:10 +02:00
										 |  |  |         while (*bhp) { | 
					
						
							|  |  |  |             bh = *bhp; | 
					
						
							|  |  |  |             if (bh->deleted) { | 
					
						
							|  |  |  |                 *bhp = bh->next; | 
					
						
							|  |  |  |                 g_free(bh); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 bhp = &bh->next; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qemu_bh_schedule_idle(QEMUBH *bh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (bh->scheduled) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     bh->scheduled = 1; | 
					
						
							|  |  |  |     bh->idle = 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qemu_bh_schedule(QEMUBH *bh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (bh->scheduled) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     bh->scheduled = 1; | 
					
						
							|  |  |  |     bh->idle = 0; | 
					
						
							| 
									
										
										
										
											2012-09-24 18:44:14 +02:00
										 |  |  |     aio_notify(bh->ctx); | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qemu_bh_cancel(QEMUBH *bh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bh->scheduled = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qemu_bh_delete(QEMUBH *bh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bh->scheduled = 0; | 
					
						
							|  |  |  |     bh->deleted = 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-24 15:11:48 +02:00
										 |  |  | static gboolean | 
					
						
							|  |  |  | aio_ctx_prepare(GSource *source, gint    *timeout) | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-09-24 15:11:48 +02:00
										 |  |  |     AioContext *ctx = (AioContext *) source; | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |     QEMUBH *bh; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-29 23:45:23 +01:00
										 |  |  |     for (bh = ctx->first_bh; bh; bh = bh->next) { | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |         if (!bh->deleted && bh->scheduled) { | 
					
						
							|  |  |  |             if (bh->idle) { | 
					
						
							|  |  |  |                 /* idle bottom halves will be polled at least
 | 
					
						
							|  |  |  |                  * every 10ms */ | 
					
						
							| 
									
										
										
										
											2012-09-24 15:11:48 +02:00
										 |  |  |                 *timeout = 10; | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 /* non-idle bottom halves will be executed
 | 
					
						
							|  |  |  |                  * immediately */ | 
					
						
							|  |  |  |                 *timeout = 0; | 
					
						
							| 
									
										
										
										
											2012-11-12 13:30:10 +01:00
										 |  |  |                 return true; | 
					
						
							| 
									
										
										
										
											2009-10-22 17:54:37 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-09-24 14:57:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-12 13:30:10 +01:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2012-09-24 14:57:41 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static gboolean | 
					
						
							|  |  |  | aio_ctx_check(GSource *source) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     AioContext *ctx = (AioContext *) source; | 
					
						
							|  |  |  |     QEMUBH *bh; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (bh = ctx->first_bh; bh; bh = bh->next) { | 
					
						
							|  |  |  |         if (!bh->deleted && bh->scheduled) { | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return aio_pending(ctx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static gboolean | 
					
						
							|  |  |  | aio_ctx_dispatch(GSource     *source, | 
					
						
							|  |  |  |                  GSourceFunc  callback, | 
					
						
							|  |  |  |                  gpointer     user_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     AioContext *ctx = (AioContext *) source; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert(callback == NULL); | 
					
						
							|  |  |  |     aio_poll(ctx, false); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-24 18:44:14 +02:00
										 |  |  | static void | 
					
						
							|  |  |  | aio_ctx_finalize(GSource     *source) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     AioContext *ctx = (AioContext *) source; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     aio_set_event_notifier(ctx, &ctx->notifier, NULL, NULL); | 
					
						
							|  |  |  |     event_notifier_cleanup(&ctx->notifier); | 
					
						
							| 
									
										
										
										
											2013-02-20 11:28:32 +01:00
										 |  |  |     g_array_free(ctx->pollfds, TRUE); | 
					
						
							| 
									
										
										
										
											2012-09-24 18:44:14 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-24 14:57:41 +02:00
										 |  |  | static GSourceFuncs aio_source_funcs = { | 
					
						
							|  |  |  |     aio_ctx_prepare, | 
					
						
							|  |  |  |     aio_ctx_check, | 
					
						
							|  |  |  |     aio_ctx_dispatch, | 
					
						
							| 
									
										
										
										
											2012-09-24 18:44:14 +02:00
										 |  |  |     aio_ctx_finalize | 
					
						
							| 
									
										
										
										
											2012-09-24 14:57:41 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GSource *aio_get_g_source(AioContext *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     g_source_ref(&ctx->source); | 
					
						
							|  |  |  |     return &ctx->source; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-09-13 12:28:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-24 18:44:14 +02:00
										 |  |  | void aio_notify(AioContext *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     event_notifier_set(&ctx->notifier); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-29 23:45:23 +01:00
										 |  |  | AioContext *aio_context_new(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-09-24 18:44:14 +02:00
										 |  |  |     AioContext *ctx; | 
					
						
							|  |  |  |     ctx = (AioContext *) g_source_new(&aio_source_funcs, sizeof(AioContext)); | 
					
						
							| 
									
										
										
										
											2013-02-20 11:28:32 +01:00
										 |  |  |     ctx->pollfds = g_array_new(FALSE, FALSE, sizeof(GPollFD)); | 
					
						
							| 
									
										
										
										
											2012-09-24 18:44:14 +02:00
										 |  |  |     event_notifier_init(&ctx->notifier, false); | 
					
						
							|  |  |  |     aio_set_event_notifier(ctx, &ctx->notifier,  | 
					
						
							|  |  |  |                            (EventNotifierHandler *) | 
					
						
							|  |  |  |                            event_notifier_test_and_clear, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ctx; | 
					
						
							| 
									
										
										
										
											2012-09-24 14:57:41 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void aio_context_ref(AioContext *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     g_source_ref(&ctx->source); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void aio_context_unref(AioContext *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     g_source_unref(&ctx->source); | 
					
						
							| 
									
										
										
										
											2012-10-29 23:45:23 +01:00
										 |  |  | } |