mirror of
				https://github.com/qemu/qemu.git
				synced 2025-10-26 12:03:40 +00:00 
			
		
		
		
	block: add bdrv_co_discard and bdrv_aio_discard support
This similarly adds support for coroutine and asynchronous discard. Signed-off-by: Paolo Bonzini <pbonzini@redhat.com> Signed-off-by: Kevin Wolf <kwolf@redhat.com>
This commit is contained in:
		
							parent
							
								
									6f6dc6565e
								
							
						
					
					
						commit
						4265d620c5
					
				
							
								
								
									
										102
									
								
								block.c
									
									
									
									
									
								
							
							
						
						
									
										102
									
								
								block.c
									
									
									
									
									
								
							| @ -1768,17 +1768,6 @@ int bdrv_has_zero_init(BlockDriverState *bs) | |||||||
|     return 1; |     return 1; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int bdrv_discard(BlockDriverState *bs, int64_t sector_num, int nb_sectors) |  | ||||||
| { |  | ||||||
|     if (!bs->drv) { |  | ||||||
|         return -ENOMEDIUM; |  | ||||||
|     } |  | ||||||
|     if (!bs->drv->bdrv_discard) { |  | ||||||
|         return 0; |  | ||||||
|     } |  | ||||||
|     return bs->drv->bdrv_discard(bs, sector_num, nb_sectors); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /*
 | /*
 | ||||||
|  * Returns true iff the specified sector is present in the disk image. Drivers |  * Returns true iff the specified sector is present in the disk image. Drivers | ||||||
|  * not implementing the functionality are assumed to not support backing files, |  * not implementing the functionality are assumed to not support backing files, | ||||||
| @ -2754,6 +2743,34 @@ BlockDriverAIOCB *bdrv_aio_flush(BlockDriverState *bs, | |||||||
|     return &acb->common; |     return &acb->common; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static void coroutine_fn bdrv_aio_discard_co_entry(void *opaque) | ||||||
|  | { | ||||||
|  |     BlockDriverAIOCBCoroutine *acb = opaque; | ||||||
|  |     BlockDriverState *bs = acb->common.bs; | ||||||
|  | 
 | ||||||
|  |     acb->req.error = bdrv_co_discard(bs, acb->req.sector, acb->req.nb_sectors); | ||||||
|  |     acb->bh = qemu_bh_new(bdrv_co_em_bh, acb); | ||||||
|  |     qemu_bh_schedule(acb->bh); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | BlockDriverAIOCB *bdrv_aio_discard(BlockDriverState *bs, | ||||||
|  |         int64_t sector_num, int nb_sectors, | ||||||
|  |         BlockDriverCompletionFunc *cb, void *opaque) | ||||||
|  | { | ||||||
|  |     Coroutine *co; | ||||||
|  |     BlockDriverAIOCBCoroutine *acb; | ||||||
|  | 
 | ||||||
|  |     trace_bdrv_aio_discard(bs, sector_num, nb_sectors, opaque); | ||||||
|  | 
 | ||||||
|  |     acb = qemu_aio_get(&bdrv_em_co_aio_pool, bs, cb, opaque); | ||||||
|  |     acb->req.sector = sector_num; | ||||||
|  |     acb->req.nb_sectors = nb_sectors; | ||||||
|  |     co = qemu_coroutine_create(bdrv_aio_discard_co_entry); | ||||||
|  |     qemu_coroutine_enter(co, acb); | ||||||
|  | 
 | ||||||
|  |     return &acb->common; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void bdrv_init(void) | void bdrv_init(void) | ||||||
| { | { | ||||||
|     module_call_init(MODULE_INIT_BLOCK); |     module_call_init(MODULE_INIT_BLOCK); | ||||||
| @ -2915,6 +2932,69 @@ int bdrv_flush(BlockDriverState *bs) | |||||||
|     return rwco.ret; |     return rwco.ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static void coroutine_fn bdrv_discard_co_entry(void *opaque) | ||||||
|  | { | ||||||
|  |     RwCo *rwco = opaque; | ||||||
|  | 
 | ||||||
|  |     rwco->ret = bdrv_co_discard(rwco->bs, rwco->sector_num, rwco->nb_sectors); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int coroutine_fn bdrv_co_discard(BlockDriverState *bs, int64_t sector_num, | ||||||
|  |                                  int nb_sectors) | ||||||
|  | { | ||||||
|  |     if (!bs->drv) { | ||||||
|  |         return -ENOMEDIUM; | ||||||
|  |     } else if (bdrv_check_request(bs, sector_num, nb_sectors)) { | ||||||
|  |         return -EIO; | ||||||
|  |     } else if (bs->read_only) { | ||||||
|  |         return -EROFS; | ||||||
|  |     } else if (bs->drv->bdrv_co_discard) { | ||||||
|  |         return bs->drv->bdrv_co_discard(bs, sector_num, nb_sectors); | ||||||
|  |     } else if (bs->drv->bdrv_aio_discard) { | ||||||
|  |         BlockDriverAIOCB *acb; | ||||||
|  |         CoroutineIOCompletion co = { | ||||||
|  |             .coroutine = qemu_coroutine_self(), | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         acb = bs->drv->bdrv_aio_discard(bs, sector_num, nb_sectors, | ||||||
|  |                                         bdrv_co_io_em_complete, &co); | ||||||
|  |         if (acb == NULL) { | ||||||
|  |             return -EIO; | ||||||
|  |         } else { | ||||||
|  |             qemu_coroutine_yield(); | ||||||
|  |             return co.ret; | ||||||
|  |         } | ||||||
|  |     } else if (bs->drv->bdrv_discard) { | ||||||
|  |         return bs->drv->bdrv_discard(bs, sector_num, nb_sectors); | ||||||
|  |     } else { | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int bdrv_discard(BlockDriverState *bs, int64_t sector_num, int nb_sectors) | ||||||
|  | { | ||||||
|  |     Coroutine *co; | ||||||
|  |     RwCo rwco = { | ||||||
|  |         .bs = bs, | ||||||
|  |         .sector_num = sector_num, | ||||||
|  |         .nb_sectors = nb_sectors, | ||||||
|  |         .ret = NOT_DONE, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     if (qemu_in_coroutine()) { | ||||||
|  |         /* Fast-path if already in coroutine context */ | ||||||
|  |         bdrv_discard_co_entry(&rwco); | ||||||
|  |     } else { | ||||||
|  |         co = qemu_coroutine_create(bdrv_discard_co_entry); | ||||||
|  |         qemu_coroutine_enter(co, &rwco); | ||||||
|  |         while (rwco.ret == NOT_DONE) { | ||||||
|  |             qemu_aio_wait(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return rwco.ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /**************************************************************/ | /**************************************************************/ | ||||||
| /* removable device support */ | /* removable device support */ | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										4
									
								
								block.h
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								block.h
									
									
									
									
									
								
							| @ -166,6 +166,9 @@ BlockDriverAIOCB *bdrv_aio_writev(BlockDriverState *bs, int64_t sector_num, | |||||||
|                                   BlockDriverCompletionFunc *cb, void *opaque); |                                   BlockDriverCompletionFunc *cb, void *opaque); | ||||||
| BlockDriverAIOCB *bdrv_aio_flush(BlockDriverState *bs, | BlockDriverAIOCB *bdrv_aio_flush(BlockDriverState *bs, | ||||||
|                                  BlockDriverCompletionFunc *cb, void *opaque); |                                  BlockDriverCompletionFunc *cb, void *opaque); | ||||||
|  | BlockDriverAIOCB *bdrv_aio_discard(BlockDriverState *bs, | ||||||
|  |                                    int64_t sector_num, int nb_sectors, | ||||||
|  |                                    BlockDriverCompletionFunc *cb, void *opaque); | ||||||
| void bdrv_aio_cancel(BlockDriverAIOCB *acb); | void bdrv_aio_cancel(BlockDriverAIOCB *acb); | ||||||
| 
 | 
 | ||||||
| typedef struct BlockRequest { | typedef struct BlockRequest { | ||||||
| @ -196,6 +199,7 @@ void bdrv_flush_all(void); | |||||||
| void bdrv_close_all(void); | void bdrv_close_all(void); | ||||||
| 
 | 
 | ||||||
| int bdrv_discard(BlockDriverState *bs, int64_t sector_num, int nb_sectors); | int bdrv_discard(BlockDriverState *bs, int64_t sector_num, int nb_sectors); | ||||||
|  | int bdrv_co_discard(BlockDriverState *bs, int64_t sector_num, int nb_sectors); | ||||||
| int bdrv_has_zero_init(BlockDriverState *bs); | int bdrv_has_zero_init(BlockDriverState *bs); | ||||||
| int bdrv_is_allocated(BlockDriverState *bs, int64_t sector_num, int nb_sectors, | int bdrv_is_allocated(BlockDriverState *bs, int64_t sector_num, int nb_sectors, | ||||||
|                       int *pnum); |                       int *pnum); | ||||||
|  | |||||||
							
								
								
									
										10
									
								
								block/raw.c
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								block/raw.c
									
									
									
									
									
								
							| @ -45,9 +45,10 @@ static int raw_probe(const uint8_t *buf, int buf_size, const char *filename) | |||||||
|    return 1; /* everything can be opened as raw image */ |    return 1; /* everything can be opened as raw image */ | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int raw_discard(BlockDriverState *bs, int64_t sector_num, int nb_sectors) | static int coroutine_fn raw_co_discard(BlockDriverState *bs, | ||||||
|  |                                        int64_t sector_num, int nb_sectors) | ||||||
| { | { | ||||||
|     return bdrv_discard(bs->file, sector_num, nb_sectors); |     return bdrv_co_discard(bs->file, sector_num, nb_sectors); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int raw_is_inserted(BlockDriverState *bs) | static int raw_is_inserted(BlockDriverState *bs) | ||||||
| @ -109,15 +110,16 @@ static BlockDriver bdrv_raw = { | |||||||
| 
 | 
 | ||||||
|     .bdrv_open          = raw_open, |     .bdrv_open          = raw_open, | ||||||
|     .bdrv_close         = raw_close, |     .bdrv_close         = raw_close, | ||||||
|  | 
 | ||||||
|     .bdrv_co_readv      = raw_co_readv, |     .bdrv_co_readv      = raw_co_readv, | ||||||
|     .bdrv_co_writev     = raw_co_writev, |     .bdrv_co_writev     = raw_co_writev, | ||||||
|     .bdrv_co_flush      = raw_co_flush, |     .bdrv_co_flush      = raw_co_flush, | ||||||
|  |     .bdrv_co_discard    = raw_co_discard, | ||||||
|  | 
 | ||||||
|     .bdrv_probe         = raw_probe, |     .bdrv_probe         = raw_probe, | ||||||
|     .bdrv_getlength     = raw_getlength, |     .bdrv_getlength     = raw_getlength, | ||||||
|     .bdrv_truncate      = raw_truncate, |     .bdrv_truncate      = raw_truncate, | ||||||
| 
 | 
 | ||||||
|     .bdrv_discard       = raw_discard, |  | ||||||
| 
 |  | ||||||
|     .bdrv_is_inserted   = raw_is_inserted, |     .bdrv_is_inserted   = raw_is_inserted, | ||||||
|     .bdrv_media_changed = raw_media_changed, |     .bdrv_media_changed = raw_media_changed, | ||||||
|     .bdrv_eject         = raw_eject, |     .bdrv_eject         = raw_eject, | ||||||
|  | |||||||
| @ -63,6 +63,8 @@ struct BlockDriver { | |||||||
|     void (*bdrv_close)(BlockDriverState *bs); |     void (*bdrv_close)(BlockDriverState *bs); | ||||||
|     int (*bdrv_create)(const char *filename, QEMUOptionParameter *options); |     int (*bdrv_create)(const char *filename, QEMUOptionParameter *options); | ||||||
|     int (*bdrv_flush)(BlockDriverState *bs); |     int (*bdrv_flush)(BlockDriverState *bs); | ||||||
|  |     int (*bdrv_discard)(BlockDriverState *bs, int64_t sector_num, | ||||||
|  |                         int nb_sectors); | ||||||
|     int (*bdrv_is_allocated)(BlockDriverState *bs, int64_t sector_num, |     int (*bdrv_is_allocated)(BlockDriverState *bs, int64_t sector_num, | ||||||
|                              int nb_sectors, int *pnum); |                              int nb_sectors, int *pnum); | ||||||
|     int (*bdrv_set_key)(BlockDriverState *bs, const char *key); |     int (*bdrv_set_key)(BlockDriverState *bs, const char *key); | ||||||
| @ -76,14 +78,17 @@ struct BlockDriver { | |||||||
|         BlockDriverCompletionFunc *cb, void *opaque); |         BlockDriverCompletionFunc *cb, void *opaque); | ||||||
|     BlockDriverAIOCB *(*bdrv_aio_flush)(BlockDriverState *bs, |     BlockDriverAIOCB *(*bdrv_aio_flush)(BlockDriverState *bs, | ||||||
|         BlockDriverCompletionFunc *cb, void *opaque); |         BlockDriverCompletionFunc *cb, void *opaque); | ||||||
|     int (*bdrv_discard)(BlockDriverState *bs, int64_t sector_num, |     BlockDriverAIOCB *(*bdrv_aio_discard)(BlockDriverState *bs, | ||||||
|                         int nb_sectors); |         int64_t sector_num, int nb_sectors, | ||||||
|  |         BlockDriverCompletionFunc *cb, void *opaque); | ||||||
| 
 | 
 | ||||||
|     int coroutine_fn (*bdrv_co_readv)(BlockDriverState *bs, |     int coroutine_fn (*bdrv_co_readv)(BlockDriverState *bs, | ||||||
|         int64_t sector_num, int nb_sectors, QEMUIOVector *qiov); |         int64_t sector_num, int nb_sectors, QEMUIOVector *qiov); | ||||||
|     int coroutine_fn (*bdrv_co_writev)(BlockDriverState *bs, |     int coroutine_fn (*bdrv_co_writev)(BlockDriverState *bs, | ||||||
|         int64_t sector_num, int nb_sectors, QEMUIOVector *qiov); |         int64_t sector_num, int nb_sectors, QEMUIOVector *qiov); | ||||||
|     int coroutine_fn (*bdrv_co_flush)(BlockDriverState *bs); |     int coroutine_fn (*bdrv_co_flush)(BlockDriverState *bs); | ||||||
|  |     int coroutine_fn (*bdrv_co_discard)(BlockDriverState *bs, | ||||||
|  |         int64_t sector_num, int nb_sectors); | ||||||
| 
 | 
 | ||||||
|     int (*bdrv_aio_multiwrite)(BlockDriverState *bs, BlockRequest *reqs, |     int (*bdrv_aio_multiwrite)(BlockDriverState *bs, BlockRequest *reqs, | ||||||
|         int num_reqs); |         int num_reqs); | ||||||
|  | |||||||
| @ -61,6 +61,7 @@ multiwrite_cb(void *mcb, int ret) "mcb %p ret %d" | |||||||
| bdrv_aio_multiwrite(void *mcb, int num_callbacks, int num_reqs) "mcb %p num_callbacks %d num_reqs %d" | bdrv_aio_multiwrite(void *mcb, int num_callbacks, int num_reqs) "mcb %p num_callbacks %d num_reqs %d" | ||||||
| bdrv_aio_multiwrite_earlyfail(void *mcb) "mcb %p" | bdrv_aio_multiwrite_earlyfail(void *mcb) "mcb %p" | ||||||
| bdrv_aio_multiwrite_latefail(void *mcb, int i) "mcb %p i %d" | bdrv_aio_multiwrite_latefail(void *mcb, int i) "mcb %p i %d" | ||||||
|  | bdrv_aio_discard(void *bs, int64_t sector_num, int nb_sectors, void *opaque) "bs %p sector_num %"PRId64" nb_sectors %d opaque %p" | ||||||
| bdrv_aio_flush(void *bs, void *opaque) "bs %p opaque %p" | bdrv_aio_flush(void *bs, void *opaque) "bs %p opaque %p" | ||||||
| bdrv_aio_readv(void *bs, int64_t sector_num, int nb_sectors, void *opaque) "bs %p sector_num %"PRId64" nb_sectors %d opaque %p" | bdrv_aio_readv(void *bs, int64_t sector_num, int nb_sectors, void *opaque) "bs %p sector_num %"PRId64" nb_sectors %d opaque %p" | ||||||
| bdrv_aio_writev(void *bs, int64_t sector_num, int nb_sectors, void *opaque) "bs %p sector_num %"PRId64" nb_sectors %d opaque %p" | bdrv_aio_writev(void *bs, int64_t sector_num, int nb_sectors, void *opaque) "bs %p sector_num %"PRId64" nb_sectors %d opaque %p" | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Paolo Bonzini
						Paolo Bonzini