• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
2<html><meta http-equiv="Content-Type" content="text/html; charset=utf-8">
3<title>Custom exception ptr - Boost.Outcome documentation</title>
4<link rel="stylesheet" href="../../../css/boost.css" type="text/css">
5<meta name="generator" content="Hugo 0.52 with Boostdoc theme">
6<meta name="viewport" content="width=device-width,initial-scale=1.0"/>
7
8<link rel="icon" href="../../../images/favicon.ico" type="image/ico"/>
9<body><div class="spirit-nav">
10<a accesskey="p" href="../../../tutorial/advanced/hooks/hook_result.html"><img src="../../../images/prev.png" alt="Prev"></a>
11    <a accesskey="u" href="../../../tutorial/advanced/hooks.html"><img src="../../../images/up.png" alt="Up"></a>
12    <a accesskey="h" href="../../../index.html"><img src="../../../images/home.png" alt="Home"></a><a accesskey="n" href="../../../tutorial/advanced/hooks/hook_outcome.html"><img src="../../../images/next.png" alt="Next"></a></div><div id="content">
13  <div class="titlepage"><div><div><h1 style="clear: both">Custom exception ptr</h1></div></div></div>
14  <p>If you merely want <code>result</code> to capture stack backtraces without calling a memory allocator
15and retaining any triviality of copy which is important for optimisation,
16you already have everything you need.</p>
17
18<p>But let&rsquo;s keep going by intercepting any
19construction of our localised <code>outcome</code> from our localised <code>result</code>, retrieving any
20stored backtrace and using it to synthesise an exception ptr with a message text
21including the backtrace. Firstly let us look at the function which synthesises
22the exception ptr:</p>
23
24<div class="code-snippet"><div class="highlight"><pre class="chroma"><code class="language-c++" data-lang="c++"><span class="k">namespace</span> <span class="n">error_code_extended</span>
25<span class="p">{</span>
26  <span class="c1">// Synthesise a custom exception_ptr from the TLS slot and write it into the outcome
27</span><span class="c1"></span>  <span class="k">template</span> <span class="o">&lt;</span><span class="k">class</span><span class="err"> </span><span class="nc">R</span><span class="o">&gt;</span> <span class="kr">inline</span> <span class="kt">void</span> <span class="n">poke_exception</span><span class="p">(</span><span class="n">outcome</span><span class="o">&lt;</span><span class="n">R</span><span class="o">&gt;</span> <span class="o">*</span><span class="n">o</span><span class="p">)</span>
28  <span class="p">{</span>
29    <span class="k">if</span><span class="p">(</span><span class="n">o</span><span class="o">-&gt;</span><span class="n">has_error</span><span class="p">())</span>
30    <span class="p">{</span>
31      <span class="n">extended_error_info</span> <span class="o">*</span><span class="n">eei</span> <span class="o">=</span> <span class="n">mythreadlocaldata</span><span class="p">().</span><span class="n">get</span><span class="p">(</span><span class="n">BOOST_OUTCOME_V2_NAMESPACE</span><span class="o">::</span><span class="n">hooks</span><span class="o">::</span><span class="n">spare_storage</span><span class="p">(</span><span class="n">o</span><span class="p">));</span>
32      <span class="k">if</span><span class="p">(</span><span class="n">eei</span> <span class="o">!=</span> <span class="k">nullptr</span><span class="p">)</span>
33      <span class="p">{</span>
34        <span class="c1">// Make a custom string for the exception
35</span><span class="c1"></span>        <span class="n">std</span><span class="o">::</span><span class="n">string</span> <span class="n">str</span><span class="p">(</span><span class="n">o</span><span class="o">-&gt;</span><span class="n">error</span><span class="p">().</span><span class="n">message</span><span class="p">());</span>
36        <span class="n">str</span><span class="p">.</span><span class="n">append</span><span class="p">(</span><span class="s">&#34; [&#34;</span><span class="p">);</span>
37        <span class="k">struct</span> <span class="n">unsymbols</span>  <span class="c1">// RAII cleaner for symbols
38</span><span class="c1"></span>        <span class="p">{</span>
39          <span class="kt">char</span> <span class="o">**</span><span class="n">_</span><span class="p">{</span><span class="k">nullptr</span><span class="p">};</span>
40          <span class="o">~</span><span class="n">unsymbols</span><span class="p">()</span> <span class="p">{</span> <span class="o">::</span><span class="n">free</span><span class="p">(</span><span class="n">_</span><span class="p">);</span> <span class="p">}</span>
41        <span class="p">}</span> <span class="n">symbols</span><span class="p">{</span><span class="o">::</span><span class="n">backtrace_symbols</span><span class="p">(</span><span class="n">eei</span><span class="o">-&gt;</span><span class="n">backtrace</span><span class="p">.</span><span class="n">data</span><span class="p">(),</span> <span class="n">eei</span><span class="o">-&gt;</span><span class="n">items</span><span class="p">)};</span>
42        <span class="k">if</span><span class="p">(</span><span class="n">symbols</span><span class="p">.</span><span class="n">_</span> <span class="o">!=</span> <span class="k">nullptr</span><span class="p">)</span>
43        <span class="p">{</span>
44          <span class="k">for</span><span class="p">(</span><span class="n">size_t</span> <span class="n">n</span> <span class="o">=</span> <span class="mi">0</span><span class="p">;</span> <span class="n">n</span> <span class="o">&lt;</span> <span class="n">eei</span><span class="o">-&gt;</span><span class="n">items</span><span class="p">;</span> <span class="n">n</span><span class="o">++</span><span class="p">)</span>
45          <span class="p">{</span>
46            <span class="k">if</span><span class="p">(</span><span class="n">n</span> <span class="o">&gt;</span> <span class="mi">0</span><span class="p">)</span>
47            <span class="p">{</span>
48              <span class="n">str</span><span class="p">.</span><span class="n">append</span><span class="p">(</span><span class="s">&#34;; &#34;</span><span class="p">);</span>
49            <span class="p">}</span>
50            <span class="n">str</span><span class="p">.</span><span class="n">append</span><span class="p">(</span><span class="n">symbols</span><span class="p">.</span><span class="n">_</span><span class="p">[</span><span class="n">n</span><span class="p">]);</span>
51          <span class="p">}</span>
52        <span class="p">}</span>
53        <span class="n">str</span><span class="p">.</span><span class="n">append</span><span class="p">(</span><span class="s">&#34;]&#34;</span><span class="p">);</span>
54
55        <span class="c1">// Override the payload/exception member in the outcome with our synthesised exception ptr
56</span><span class="c1"></span>        <span class="n">BOOST_OUTCOME_V2_NAMESPACE</span><span class="o">::</span><span class="n">hooks</span><span class="o">::</span><span class="n">override_outcome_exception</span><span class="p">(</span><span class="n">o</span><span class="p">,</span> <span class="n">std</span><span class="o">::</span><span class="n">make_exception_ptr</span><span class="p">(</span><span class="n">std</span><span class="o">::</span><span class="n">runtime_error</span><span class="p">(</span><span class="n">str</span><span class="p">)));</span>
57      <span class="p">}</span>
58    <span class="p">}</span>
59  <span class="p">}</span>
60<span class="p">}</span>
61</code></pre></div><a href="https://github.com/boostorg/outcome/tree/master/doc/src/snippets/error_code_extended.cpp#L142" class="code-snippet-url" target="_blank">View this code on Github</a></div>
62
63
64<p>If the localised <code>outcome</code> being constructed is errored, try fetching the TLS slot
65for the unique 16-bit value in its spare storage. If that is valid, symbolise the
66stack backtrace into a string and make an exception ptr with a runtime error with
67that string. Finally, override the payload/exception member in our just-copy-constructed
68localised <code>outcome</code> with the new exception ptr.</p>
69
70<hr>
71
72<p>As the reference documentation for <a href="../../../reference/functions/hooks/override_outcome_exception.html" class="api-reference"><code>void override_outcome_exception(basic_outcome&lt;T, EC, EP, NoValuePolicy&gt; *, U &amp;&amp;) noexcept</code></a>
73
74points out, you <em>almost certainly</em> never want to use this function if there is any
75other alternative. It is worth explaining what is meant by this.</p>
76
77<p>In this section, we <em>always</em> synthesise an exception ptr from the stored state and
78error code at the exact point of transition from <code>result</code> based APIs to <code>outcome</code>
79based APIs. This is acceptable only because we know that our code enforces that
80discipline.</p>
81
82<p>If one were designing a library facility, one could not assume such discipline in the
83library user. One would probably be better off making the exception ptr synthesis
84<em>lazy</em> via a custom no-value policy which generates the stacktrace-containing error
85message only on demand e.g. <code>.exception()</code> observation, or a <code>.value()</code> observation
86where no value is available.</p>
87
88<p>Such a design is however more indeterminate than the design presented in this section,
89because the indeterminacy is less predictable than in this design. Ultimately which
90strategy you adopt depends on how important absolute determinism is to your Outcome-based
91application.</p>
92
93
94        </div><p><small>Last revised: February 08, 2019 at 22:18:08 UTC</small></p>
95<hr>
96<div class="spirit-nav">
97<a accesskey="p" href="../../../tutorial/advanced/hooks/hook_result.html"><img src="../../../images/prev.png" alt="Prev"></a>
98    <a accesskey="u" href="../../../tutorial/advanced/hooks.html"><img src="../../../images/up.png" alt="Up"></a>
99    <a accesskey="h" href="../../../index.html"><img src="../../../images/home.png" alt="Home"></a><a accesskey="n" href="../../../tutorial/advanced/hooks/hook_outcome.html"><img src="../../../images/next.png" alt="Next"></a></div></body>
100</html>
101