aboutsummaryrefslogtreecommitdiff
path: root/docs/devel/block-coroutine-wrapper.rst
blob: 412851986b1a045453ab864afd5a84d647cac944 (plain)
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
=======================
block-coroutine-wrapper
=======================

A lot of functions in QEMU block layer (see ``block/*``) can only be
called in coroutine context. Such functions are normally marked by the
coroutine_fn specifier. Still, sometimes we need to call them from
non-coroutine context; for this we need to start a coroutine, run the
needed function from it and wait for the coroutine to finish in a
BDRV_POLL_WHILE() loop. To run a coroutine we need a function with one
void* argument. So for each coroutine_fn function which needs a
non-coroutine interface, we should define a structure to pack the
parameters, define a separate function to unpack the parameters and
call the original function and finally define a new interface function
with same list of arguments as original one, which will pack the
parameters into a struct, create a coroutine, run it and wait in
BDRV_POLL_WHILE() loop. It's boring to create such wrappers by hand,
so we have a script to generate them.

Usage
=====

Assume we have defined the ``coroutine_fn`` function
``bdrv_co_foo(<some args>)`` and need a non-coroutine interface for it,
called ``bdrv_foo(<same args>)``. In this case the script can help. To
trigger the generation:

1. You need ``bdrv_foo`` declaration somewhere (for example, in
   ``block/coroutines.h``) with the ``generated_co_wrapper`` mark,
   like this:

.. code-block:: c

    int generated_co_wrapper bdrv_foo(<some args>);

2. You need to feed this declaration to block-coroutine-wrapper script.
   For this, add the .h (or .c) file with the declaration to the
   ``input: files(...)`` list of ``block_gen_c`` target declaration in
   ``block/meson.build``

You are done. During the build, coroutine wrappers will be generated in
``<BUILD_DIR>/block/block-gen.c``.

Links
=====

1. The script location is ``scripts/block-coroutine-wrapper.py``.

2. Generic place for private ``generated_co_wrapper`` declarations is
   ``block/coroutines.h``, for public declarations:
   ``include/block/block.h``

3. The core API of generated coroutine wrappers is placed in
   (not generated) ``block/block-gen.h``