100.00% Lines (12/12)
100.00% Functions (4/4)
| TLA | Baseline | Branch | ||||||
|---|---|---|---|---|---|---|---|---|
| Line | Hits | Code | Line | Hits | Code | |||
| 1 | // | 1 | // | |||||
| 2 | // Copyright (c) 2025 Vinnie Falco (vinnie.falco@gmail.com) | 2 | // Copyright (c) 2025 Vinnie Falco (vinnie.falco@gmail.com) | |||||
| 3 | // | 3 | // | |||||
| 4 | // Distributed under the Boost Software License, Version 1.0. (See accompanying | 4 | // Distributed under the Boost Software License, Version 1.0. (See accompanying | |||||
| 5 | // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) | 5 | // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) | |||||
| 6 | // | 6 | // | |||||
| 7 | // Official repository: https://github.com/cppalliance/capy | 7 | // Official repository: https://github.com/cppalliance/capy | |||||
| 8 | // | 8 | // | |||||
| 9 | 9 | |||||||
| 10 | #ifndef BOOST_CAPY_FRAME_ALLOCATOR_HPP | 10 | #ifndef BOOST_CAPY_FRAME_ALLOCATOR_HPP | |||||
| 11 | #define BOOST_CAPY_FRAME_ALLOCATOR_HPP | 11 | #define BOOST_CAPY_FRAME_ALLOCATOR_HPP | |||||
| 12 | 12 | |||||||
| 13 | #include <boost/capy/detail/config.hpp> | 13 | #include <boost/capy/detail/config.hpp> | |||||
| 14 | 14 | |||||||
| 15 | #include <coroutine> | 15 | #include <coroutine> | |||||
| 16 | #include <memory_resource> | 16 | #include <memory_resource> | |||||
| 17 | 17 | |||||||
| 18 | /* Design rationale (pdimov): | 18 | /* Design rationale (pdimov): | |||||
| 19 | 19 | |||||||
| 20 | This accessor is a thin wrapper over a thread-local pointer. | 20 | This accessor is a thin wrapper over a thread-local pointer. | |||||
| 21 | It returns exactly what was stored, including nullptr. No | 21 | It returns exactly what was stored, including nullptr. No | |||||
| 22 | dynamic initializer on the thread-local; a dynamic TLS | 22 | dynamic initializer on the thread-local; a dynamic TLS | |||||
| 23 | initializer moves you into a costlier implementation bucket | 23 | initializer moves you into a costlier implementation bucket | |||||
| 24 | on some platforms - avoid it. | 24 | on some platforms - avoid it. | |||||
| 25 | 25 | |||||||
| 26 | Null handling is the caller's responsibility (e.g. in | 26 | Null handling is the caller's responsibility (e.g. in | |||||
| 27 | promise_type::operator new). The accessor must not substitute | 27 | promise_type::operator new). The accessor must not substitute | |||||
| 28 | a default, because there are multiple valid choices | 28 | a default, because there are multiple valid choices | |||||
| 29 | (new_delete_resource, the default pmr resource, etc.). If | 29 | (new_delete_resource, the default pmr resource, etc.). If | |||||
| 30 | the allocator is not set, it reports "not set" and the | 30 | the allocator is not set, it reports "not set" and the | |||||
| 31 | caller interprets that however it wants. | 31 | caller interprets that however it wants. | |||||
| 32 | */ | 32 | */ | |||||
| 33 | 33 | |||||||
| 34 | namespace boost { | 34 | namespace boost { | |||||
| 35 | namespace capy { | 35 | namespace capy { | |||||
| 36 | 36 | |||||||
| 37 | namespace detail { | 37 | namespace detail { | |||||
| 38 | 38 | |||||||
| 39 | inline std::pmr::memory_resource*& | 39 | inline std::pmr::memory_resource*& | |||||
| HITCBC | 40 | 31906 | current_frame_allocator_ref() noexcept | 40 | 32017 | current_frame_allocator_ref() noexcept | ||
| 41 | { | 41 | { | |||||
| 42 | static thread_local std::pmr::memory_resource* mr = nullptr; | 42 | static thread_local std::pmr::memory_resource* mr = nullptr; | |||||
| HITCBC | 43 | 31906 | return mr; | 43 | 32017 | return mr; | ||
| 44 | } | 44 | } | |||||
| 45 | 45 | |||||||
| 46 | } // namespace detail | 46 | } // namespace detail | |||||
| 47 | 47 | |||||||
| 48 | /** Return the current frame allocator for this thread. | 48 | /** Return the current frame allocator for this thread. | |||||
| 49 | 49 | |||||||
| 50 | These accessors exist to implement the allocator | 50 | These accessors exist to implement the allocator | |||||
| 51 | propagation portion of the @ref IoAwaitable protocol. | 51 | propagation portion of the @ref IoAwaitable protocol. | |||||
| 52 | Launch functions (`run_async`, `run`) set the | 52 | Launch functions (`run_async`, `run`) set the | |||||
| 53 | thread-local value before invoking a child coroutine; | 53 | thread-local value before invoking a child coroutine; | |||||
| 54 | the child's `promise_type::operator new` reads it to | 54 | the child's `promise_type::operator new` reads it to | |||||
| 55 | allocate the coroutine frame from the correct resource. | 55 | allocate the coroutine frame from the correct resource. | |||||
| 56 | 56 | |||||||
| 57 | The value is only valid during a narrow execution | 57 | The value is only valid during a narrow execution | |||||
| 58 | window. Between a coroutine's resumption | 58 | window. Between a coroutine's resumption | |||||
| 59 | and the next suspension point, the protocol guarantees | 59 | and the next suspension point, the protocol guarantees | |||||
| 60 | that TLS contains the allocator associated with the | 60 | that TLS contains the allocator associated with the | |||||
| 61 | currently running chain. Outside that window the value | 61 | currently running chain. Outside that window the value | |||||
| 62 | is indeterminate. Only code that implements an | 62 | is indeterminate. Only code that implements an | |||||
| 63 | @ref IoAwaitable should call these functions. | 63 | @ref IoAwaitable should call these functions. | |||||
| 64 | 64 | |||||||
| 65 | A return value of `nullptr` means "not specified" - | 65 | A return value of `nullptr` means "not specified" - | |||||
| 66 | no allocator has been established for this chain. | 66 | no allocator has been established for this chain. | |||||
| 67 | The awaitable is free to use whatever allocation | 67 | The awaitable is free to use whatever allocation | |||||
| 68 | strategy makes best sense (e.g. | 68 | strategy makes best sense (e.g. | |||||
| 69 | `std::pmr::new_delete_resource()`). | 69 | `std::pmr::new_delete_resource()`). | |||||
| 70 | 70 | |||||||
| 71 | Use of the frame allocator is optional. An awaitable | 71 | Use of the frame allocator is optional. An awaitable | |||||
| 72 | that does not consult this value to allocate its | 72 | that does not consult this value to allocate its | |||||
| 73 | coroutine frame is never wrong. However, a conforming | 73 | coroutine frame is never wrong. However, a conforming | |||||
| 74 | awaitable must still propagate the allocator faithfully | 74 | awaitable must still propagate the allocator faithfully | |||||
| 75 | so that downstream coroutines can use it. | 75 | so that downstream coroutines can use it. | |||||
| 76 | 76 | |||||||
| 77 | @return The thread-local memory_resource pointer, | 77 | @return The thread-local memory_resource pointer, | |||||
| 78 | or `nullptr` if none has been set. | 78 | or `nullptr` if none has been set. | |||||
| 79 | 79 | |||||||
| 80 | @see set_current_frame_allocator, IoAwaitable | 80 | @see set_current_frame_allocator, IoAwaitable | |||||
| 81 | */ | 81 | */ | |||||
| 82 | inline | 82 | inline | |||||
| 83 | std::pmr::memory_resource* | 83 | std::pmr::memory_resource* | |||||
| HITCBC | 84 | 9933 | get_current_frame_allocator() noexcept | 84 | 9982 | get_current_frame_allocator() noexcept | ||
| 85 | { | 85 | { | |||||
| HITCBC | 86 | 9933 | return detail::current_frame_allocator_ref(); | 86 | 9982 | return detail::current_frame_allocator_ref(); | ||
| 87 | } | 87 | } | |||||
| 88 | 88 | |||||||
| 89 | /** Set the current frame allocator for this thread. | 89 | /** Set the current frame allocator for this thread. | |||||
| 90 | 90 | |||||||
| 91 | Installs @p mr as the frame allocator that will be | 91 | Installs @p mr as the frame allocator that will be | |||||
| 92 | read by the next coroutine's `promise_type::operator | 92 | read by the next coroutine's `promise_type::operator | |||||
| 93 | new` on this thread. Only launch functions and | 93 | new` on this thread. Only launch functions and | |||||
| 94 | @ref IoAwaitable machinery should call this; see | 94 | @ref IoAwaitable machinery should call this; see | |||||
| 95 | @ref get_current_frame_allocator for the full protocol | 95 | @ref get_current_frame_allocator for the full protocol | |||||
| 96 | description. | 96 | description. | |||||
| 97 | 97 | |||||||
| 98 | Passing `nullptr` means "not specified" - no | 98 | Passing `nullptr` means "not specified" - no | |||||
| 99 | particular allocator is established for the chain. | 99 | particular allocator is established for the chain. | |||||
| 100 | 100 | |||||||
| 101 | @param mr The memory_resource to install, or | 101 | @param mr The memory_resource to install, or | |||||
| 102 | `nullptr` to clear. | 102 | `nullptr` to clear. | |||||
| 103 | 103 | |||||||
| 104 | @see get_current_frame_allocator, IoAwaitable | 104 | @see get_current_frame_allocator, IoAwaitable | |||||
| 105 | */ | 105 | */ | |||||
| 106 | inline void | 106 | inline void | |||||
| HITCBC | 107 | 21973 | set_current_frame_allocator( | 107 | 22035 | set_current_frame_allocator( | ||
| 108 | std::pmr::memory_resource* mr) noexcept | 108 | std::pmr::memory_resource* mr) noexcept | |||||
| 109 | { | 109 | { | |||||
| HITCBC | 110 | 21973 | detail::current_frame_allocator_ref() = mr; | 110 | 22035 | detail::current_frame_allocator_ref() = mr; | ||
| HITCBC | 111 | 21973 | } | 111 | 22035 | } | ||
| 112 | 112 | |||||||
| 113 | /** Resume a coroutine handle with frame-allocator TLS protection. | 113 | /** Resume a coroutine handle with frame-allocator TLS protection. | |||||
| 114 | 114 | |||||||
| 115 | Saves the current thread-local frame allocator before | 115 | Saves the current thread-local frame allocator before | |||||
| 116 | calling `h.resume()`, then restores it after the call | 116 | calling `h.resume()`, then restores it after the call | |||||
| 117 | returns. This prevents a resumed coroutine's | 117 | returns. This prevents a resumed coroutine's | |||||
| 118 | `await_resume` from permanently overwriting the caller's | 118 | `await_resume` from permanently overwriting the caller's | |||||
| 119 | allocator value. | 119 | allocator value. | |||||
| 120 | 120 | |||||||
| 121 | Between a coroutine's resumption and its next child | 121 | Between a coroutine's resumption and its next child | |||||
| 122 | invocation, arbitrary user code may run. If that code | 122 | invocation, arbitrary user code may run. If that code | |||||
| 123 | resumes a coroutine from a different chain on this | 123 | resumes a coroutine from a different chain on this | |||||
| 124 | thread, the other coroutine's `await_resume` overwrites | 124 | thread, the other coroutine's `await_resume` overwrites | |||||
| 125 | TLS with its own allocator. Without save/restore, the | 125 | TLS with its own allocator. Without save/restore, the | |||||
| 126 | original coroutine's next child would allocate from | 126 | original coroutine's next child would allocate from | |||||
| 127 | the wrong resource. | 127 | the wrong resource. | |||||
| 128 | 128 | |||||||
| 129 | Event loops, strand dispatch loops, and any code that | 129 | Event loops, strand dispatch loops, and any code that | |||||
| 130 | calls `.resume()` on a coroutine handle should use | 130 | calls `.resume()` on a coroutine handle should use | |||||
| 131 | this function instead of calling `.resume()` directly. | 131 | this function instead of calling `.resume()` directly. | |||||
| 132 | See the @ref Executor concept documentation for details. | 132 | See the @ref Executor concept documentation for details. | |||||
| 133 | 133 | |||||||
| 134 | @param h The coroutine handle to resume. | 134 | @param h The coroutine handle to resume. | |||||
| 135 | 135 | |||||||
| 136 | @see get_current_frame_allocator, set_current_frame_allocator | 136 | @see get_current_frame_allocator, set_current_frame_allocator | |||||
| 137 | */ | 137 | */ | |||||
| 138 | inline void | 138 | inline void | |||||
| HITCBC | 139 | 1364 | safe_resume(std::coroutine_handle<> h) noexcept | 139 | 1393 | safe_resume(std::coroutine_handle<> h) noexcept | ||
| 140 | { | 140 | { | |||||
| HITCBC | 141 | 1364 | auto* saved = get_current_frame_allocator(); | 141 | 1393 | auto* saved = get_current_frame_allocator(); | ||
| HITCBC | 142 | 1364 | h.resume(); | 142 | 1393 | h.resume(); | ||
| HITCBC | 143 | 1364 | set_current_frame_allocator(saved); | 143 | 1393 | set_current_frame_allocator(saved); | ||
| HITCBC | 144 | 1364 | } | 144 | 1393 | } | ||
| 145 | 145 | |||||||
| 146 | } // namespace capy | 146 | } // namespace capy | |||||
| 147 | } // namespace boost | 147 | } // namespace boost | |||||
| 148 | 148 | |||||||
| 149 | #endif | 149 | #endif | |||||