76 lines
2.3 KiB
C
76 lines
2.3 KiB
C
|
/*
|
||
|
* reqlist API
|
||
|
*
|
||
|
* Copyright (C) 2013 Proxmox Server Solutions
|
||
|
* Copyright (c) 2021 Virtuozzo International GmbH.
|
||
|
*
|
||
|
* Authors:
|
||
|
* Dietmar Maurer (dietmar@proxmox.com)
|
||
|
* Vladimir Sementsov-Ogievskiy <vsementsov@virtuozzo.com>
|
||
|
*
|
||
|
* This work is licensed under the terms of the GNU GPL, version 2 or later.
|
||
|
* See the COPYING file in the top-level directory.
|
||
|
*/
|
||
|
|
||
|
#ifndef REQLIST_H
|
||
|
#define REQLIST_H
|
||
|
|
||
|
#include "qemu/coroutine.h"
|
||
|
|
||
|
/*
|
||
|
* The API is not thread-safe and shouldn't be. The struct is public to be part
|
||
|
* of other structures and protected by third-party locks, see
|
||
|
* block/block-copy.c for example.
|
||
|
*/
|
||
|
|
||
|
typedef struct BlockReq {
|
||
|
int64_t offset;
|
||
|
int64_t bytes;
|
||
|
|
||
|
CoQueue wait_queue; /* coroutines blocked on this req */
|
||
|
QLIST_ENTRY(BlockReq) list;
|
||
|
} BlockReq;
|
||
|
|
||
|
typedef QLIST_HEAD(, BlockReq) BlockReqList;
|
||
|
|
||
|
/*
|
||
|
* Initialize new request and add it to the list. Caller must be sure that
|
||
|
* there are no conflicting requests in the list.
|
||
|
*/
|
||
|
void reqlist_init_req(BlockReqList *reqs, BlockReq *req, int64_t offset,
|
||
|
int64_t bytes);
|
||
|
/* Search for request in the list intersecting with @offset/@bytes area. */
|
||
|
BlockReq *reqlist_find_conflict(BlockReqList *reqs, int64_t offset,
|
||
|
int64_t bytes);
|
||
|
|
||
|
/*
|
||
|
* If there are no intersecting requests return false. Otherwise, wait for the
|
||
|
* first found intersecting request to finish and return true.
|
||
|
*
|
||
|
* @lock is passed to qemu_co_queue_wait()
|
||
|
* False return value proves that lock was released at no point.
|
||
|
*/
|
||
|
bool coroutine_fn reqlist_wait_one(BlockReqList *reqs, int64_t offset,
|
||
|
int64_t bytes, CoMutex *lock);
|
||
|
|
||
|
/*
|
||
|
* Wait for all intersecting requests. It just calls reqlist_wait_one() in a
|
||
|
* loop, caller is responsible to stop producing new requests in this region
|
||
|
* in parallel, otherwise reqlist_wait_all() may never return.
|
||
|
*/
|
||
|
void coroutine_fn reqlist_wait_all(BlockReqList *reqs, int64_t offset,
|
||
|
int64_t bytes, CoMutex *lock);
|
||
|
|
||
|
/*
|
||
|
* Shrink request and wake all waiting coroutines (maybe some of them are not
|
||
|
* intersecting with shrunk request).
|
||
|
*/
|
||
|
void coroutine_fn reqlist_shrink_req(BlockReq *req, int64_t new_bytes);
|
||
|
|
||
|
/*
|
||
|
* Remove request and wake all waiting coroutines. Do not release any memory.
|
||
|
*/
|
||
|
void coroutine_fn reqlist_remove_req(BlockReq *req);
|
||
|
|
||
|
#endif /* REQLIST_H */
|