root/library/doc/html/memory_management.html @ 593

Revision 591, 9.0 kB (checked in by smidl, 15 years ago)

doc

Line 
1<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
2<html xmlns="http://www.w3.org/1999/xhtml">
3<head>
4<meta http-equiv="Content-Type" content="text/xhtml;charset=UTF-8"/>
5<title>mixpp: Memory Management in BDM</title>
6<link href="tabs.css" rel="stylesheet" type="text/css"/>
7<link href="doxygen.css" rel="stylesheet" type="text/css"/>
8</head>
9<body>
10<!-- Generated by Doxygen 1.6.1 -->
11<script type="text/javascript">
12<!--
13function changeDisplayState (e){
14  var num=this.id.replace(/[^[0-9]/g,'');
15  var button=this.firstChild;
16  var sectionDiv=document.getElementById('dynsection'+num);
17  if (sectionDiv.style.display=='none'||sectionDiv.style.display==''){
18    sectionDiv.style.display='block';
19    button.src='open.gif';
20  }else{
21    sectionDiv.style.display='none';
22    button.src='closed.gif';
23  }
24}
25function initDynSections(){
26  var divs=document.getElementsByTagName('div');
27  var sectionCounter=1;
28  for(var i=0;i<divs.length-1;i++){
29    if(divs[i].className=='dynheader'&&divs[i+1].className=='dynsection'){
30      var header=divs[i];
31      var section=divs[i+1];
32      var button=header.firstChild;
33      if (button!='IMG'){
34        divs[i].insertBefore(document.createTextNode(' '),divs[i].firstChild);
35        button=document.createElement('img');
36        divs[i].insertBefore(button,divs[i].firstChild);
37      }
38      header.style.cursor='pointer';
39      header.onclick=changeDisplayState;
40      header.id='dynheader'+sectionCounter;
41      button.src='closed.gif';
42      section.id='dynsection'+sectionCounter;
43      section.style.display='none';
44      section.style.marginLeft='14px';
45      sectionCounter++;
46    }
47  }
48}
49window.onload = initDynSections;
50-->
51</script>
52<div class="navigation" id="top">
53  <div class="tabs">
54    <ul>
55      <li><a href="main.html"><span>Main&nbsp;Page</span></a></li>
56      <li class="current"><a href="pages.html"><span>Related&nbsp;Pages</span></a></li>
57      <li><a href="annotated.html"><span>Classes</span></a></li>
58      <li><a href="files.html"><span>Files</span></a></li>
59    </ul>
60  </div>
61</div>
62<div class="contents">
63
64
65<h1><a class="anchor" id="memory_management">Memory Management in BDM </a></h1><p>C++ memory management is notoriously flexible, allowing a wide range of efficient and dangerous techniques. BDM uses conventions which allow high implementation efficiency (not absolutely maximal, but within a measurement error of the most efficient way) while substantially reducing the danger of memory errors. These conventions are described below.</p>
66<h2><a class="anchor" id="Constructors">
67Constructors</a></h2>
68<ul>
69<li>Each configurable class must have public default constructor (that is, a constructor which takes no parameters). This is required by the configuration framework. Other constructors may also be defined when convenient.</li>
70</ul>
71<ul>
72<li>Each constructor must initialize all fields of the constructed object, to prevent unpredictable behavior.</li>
73</ul>
74<p>One consequence of the points above is the case when a default constructor doesn't have the data with which to initialize a new object - in that case it can simply call the default constructors of all its object fields (and base classes), but must explicitly initialize all numeric fields and raw pointers to 0. Such an object isn't valid as constructed and must have some additional initialization methods (typically from_settings, for reading its configured state), but it can at least be destroyed.</p>
75<h2><a class="anchor" id="Exceptions">
76Exceptions</a></h2>
77<p>BDM uses exceptions to signal runtime and some logic errors. The library aims to provide the minimal exception safety (that is, throwing an exception doesn't crash and doesn't leak any resources) for all thrown exceptions <b>except</b> memory errors - when a program using BDM exhausts memory, it should be terminated as soon as possible (and in most cases it has probably already terminated by itself). Specific exceptions may provide stronger guarantees, as documented for specific cases. All exceptions thrown out of the library are descendants of std::exception. Since they're organized into a class hierarchy, they should be caught by reference:</p>
78<div class="fragment"><pre class="fragment"><span class="keywordflow">try</span> {
79        mpdf* mtmp = UI::build&lt;mpdf&gt; (_Sources, i);
80        Sources (i) = mtmp;
81} <span class="keywordflow">catch</span> (UIException &amp;exc) {
82</pre></div><p>This saves one call to copy constructor and prevents slicing.</p>
83<h2><a class="anchor" id="Pointers">
84Pointers</a></h2>
85<p>Pointers are used extensively (for efficiency), but usage of raw pointers should be minimized.</p>
86<p>Objects allocated by operator new should be assigned to a smart pointer instance immediately upon their construction, so that they can be automatically deleted after use. BDM implements its own reference-counted smart pointer template, <a class="el" href="classbdm_1_1shared__ptr.html" title="A naive implementation of roughly a subset of the std::tr1::shared_ptr spec.">bdm::shared_ptr</a>, whose interface and semantics are close to the proposed standard std::tr1:shared_ptr (which is planned to replace <a class="el" href="classbdm_1_1shared__ptr.html" title="A naive implementation of roughly a subset of the std::tr1::shared_ptr spec.">bdm::shared_ptr</a> once it becomes widely available). Note that objects allocated on the stack <b>must</b> <b>not</b> have their addresses passed to shared_ptr - that is a bug leading to intermittent runtime errors.</p>
87<p>Non-null heap pointers may also be kept in instances of object_ptr, which is convertible to (and from) shared_ptr. The SHAREDPTR macro (or SHAREDPTR2, for templated types) defines standartized names for these instances, simplifying library usage:</p>
88<div class="fragment"><pre class="fragment"><span class="comment">/*</span>
89<span class="comment"> egamma_ptr is typedef for object_ptr&lt;egamma&gt;, whose default</span>
90<span class="comment"> constructor calls new egamma()</span>
91<span class="comment">*/</span>
92egamma_ptr eG;
93eG-&gt;set_parameters ( a, b );
94
95epdf_array Coms ( 2 ); <span class="comment">// epdf_array is typedef for Array&lt;shared_ptr&lt;epdf&gt; &gt;</span>
96Coms ( 0 ) = eG; <span class="comment">// object_ptr&lt;T&gt; is derived from shared_ptr&lt;T&gt;</span>
97
98<span class="comment">/*</span>
99<span class="comment"> The egamma instance doesn&apos;t leak: if the shared_ptr instance which</span>
100<span class="comment"> wraps it isn&apos;t assigned to anything else, the pointer is deleted by</span>
101<span class="comment"> the destructor of either object_ptr, or Array, whichever runs last.</span>
102<span class="comment">*/</span>
103</pre></div><p>Pointers kept in object fields should be wrapped in a shared_ptr instance, which will automatically keep them valid (at least) for the lifetime of the containing object. When that isn't possible, it should be documented why their containing class doesn't delete them and who does.</p>
104<p>Pointers passed as arguments into functions (and methods) are generally not expected to stay valid after the function returns - when that is required, the parameter's documentation should specify the required scope:</p>
105<div class="fragment"><pre class="fragment"><span class="comment">/* the pointer must stay valid for the lifetime of the object */</span>
106CurrentContext ( <span class="keyword">const</span> <span class="keywordtype">char</span> *name, <span class="keywordtype">int</span> idx );
107</pre></div><p>A simpler alternative is just to pass a shared pointer:</p>
108<div class="fragment"><pre class="fragment"><span class="keyword">class </span>mepdf : <span class="keyword">public</span> mpdf
109{
110        shared_ptr&lt;epdf&gt; iepdf;
111<span class="keyword">public</span>:
112        mepdf (shared_ptr&lt;epdf&gt; em) {
113                iepdf = em;
114</pre></div><p>In the case above, passing a (constant) reference to <a class="el" href="classbdm_1_1shared__ptr.html">shared_ptr&lt;epdf&gt;</a> might be more efficient, but no measurements have been performed.</p>
115<p>Functions returning raw pointers should document the scope of their validity:</p>
116<div class="fragment"><pre class="fragment"><span class="comment">/*</span>
117<span class="comment"> Returns the stored pointer (which remains owned by this</span>
118<span class="comment"> instance).</span>
119<span class="comment">*/</span>
120T *<span class="keyword">get</span>();
121</pre></div><p>Functions generally shouldn't return raw pointers allocated by operator new - such pointers should be wrapped in an instance of shared_ptr, so that the pointer's unlimited life expectancy is encoded in the function signature:</p>
122<div class="fragment"><pre class="fragment"><span class="keyword">virtual</span> shared_ptr&lt;epdf&gt; marginal (<span class="keyword">const</span> RV &amp;rv) <span class="keyword">const</span>;
123</pre></div> </div>
124<hr size="1"/><address style="text-align: right;"><small>Generated on Sun Aug 30 22:10:49 2009 for mixpp by&nbsp;
125<a href="http://www.doxygen.org/index.html">
126<img class="footer" src="doxygen.png" alt="doxygen"/></a> 1.6.1 </small></address>
127</body>
128</html>
Note: See TracBrowser for help on using the browser.