summaryrefslogtreecommitdiffstats
path: root/include/block/reqlist.h
blob: 5253497bae35035ac77464b4c2297cee6d231c3d (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
/*
 * 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 */