File: FileDescriptor_8h-source.html

package info (click to toggle)
passenger 2.2.11debian-2
  • links: PTS, VCS
  • area: main
  • in suites: squeeze
  • size: 11,576 kB
  • ctags: 28,138
  • sloc: cpp: 66,323; ruby: 9,646; ansic: 2,425; python: 141; sh: 56; makefile: 29
file content (206 lines) | stat: -rw-r--r-- 18,687 bytes parent folder | download
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
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html><head><meta http-equiv="Content-Type" content="text/html;charset=UTF-8">
<title>Passenger: FileDescriptor.h Source File</title>
<link href="tabs.css" rel="stylesheet" type="text/css">
<link href="doxygen.css" rel="stylesheet" type="text/css">
</head><body>
<!-- Generated by Doxygen 1.5.8 -->
<div class="navigation" id="top">
  <div class="tabs">
    <ul>
      <li><a href="main.html"><span>Main&nbsp;Page</span></a></li>
      <li><a href="modules.html"><span>Modules</span></a></li>
      <li><a href="namespaces.html"><span>Namespaces</span></a></li>
      <li><a href="annotated.html"><span>Classes</span></a></li>
      <li class="current"><a href="files.html"><span>Files</span></a></li>
    </ul>
  </div>
  <div class="tabs">
    <ul>
      <li><a href="files.html"><span>File&nbsp;List</span></a></li>
    </ul>
  </div>
<h1>FileDescriptor.h</h1><div class="fragment"><pre class="fragment"><a name="l00001"></a>00001 <span class="comment">/*</span>
<a name="l00002"></a>00002 <span class="comment"> *  Phusion Passenger - http://www.modrails.com/</span>
<a name="l00003"></a>00003 <span class="comment"> *  Copyright (c) 2010 Phusion</span>
<a name="l00004"></a>00004 <span class="comment"> *</span>
<a name="l00005"></a>00005 <span class="comment"> *  "Phusion Passenger" is a trademark of Hongli Lai &amp; Ninh Bui.</span>
<a name="l00006"></a>00006 <span class="comment"> *</span>
<a name="l00007"></a>00007 <span class="comment"> *  Permission is hereby granted, free of charge, to any person obtaining a copy</span>
<a name="l00008"></a>00008 <span class="comment"> *  of this software and associated documentation files (the "Software"), to deal</span>
<a name="l00009"></a>00009 <span class="comment"> *  in the Software without restriction, including without limitation the rights</span>
<a name="l00010"></a>00010 <span class="comment"> *  to use, copy, modify, merge, publish, distribute, sublicense, and/or sell</span>
<a name="l00011"></a>00011 <span class="comment"> *  copies of the Software, and to permit persons to whom the Software is</span>
<a name="l00012"></a>00012 <span class="comment"> *  furnished to do so, subject to the following conditions:</span>
<a name="l00013"></a>00013 <span class="comment"> *</span>
<a name="l00014"></a>00014 <span class="comment"> *  The above copyright notice and this permission notice shall be included in</span>
<a name="l00015"></a>00015 <span class="comment"> *  all copies or substantial portions of the Software.</span>
<a name="l00016"></a>00016 <span class="comment"> *</span>
<a name="l00017"></a>00017 <span class="comment"> *  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR</span>
<a name="l00018"></a>00018 <span class="comment"> *  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,</span>
<a name="l00019"></a>00019 <span class="comment"> *  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE</span>
<a name="l00020"></a>00020 <span class="comment"> *  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER</span>
<a name="l00021"></a>00021 <span class="comment"> *  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,</span>
<a name="l00022"></a>00022 <span class="comment"> *  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN</span>
<a name="l00023"></a>00023 <span class="comment"> *  THE SOFTWARE.</span>
<a name="l00024"></a>00024 <span class="comment"> */</span>
<a name="l00025"></a>00025 <span class="preprocessor">#ifndef _PASSENGER_FILE_DESCRIPTOR_H_</span>
<a name="l00026"></a>00026 <span class="preprocessor"></span><span class="preprocessor">#define _PASSENGER_FILE_DESCRIPTOR_H_</span>
<a name="l00027"></a>00027 <span class="preprocessor"></span>
<a name="l00028"></a>00028 <span class="preprocessor">#include &lt;boost/shared_ptr.hpp&gt;</span>
<a name="l00029"></a>00029 <span class="preprocessor">#include &lt;oxt/system_calls.hpp&gt;</span>
<a name="l00030"></a>00030 
<a name="l00031"></a>00031 <span class="preprocessor">#include &lt;unistd.h&gt;</span>
<a name="l00032"></a>00032 <span class="preprocessor">#include &lt;cerrno&gt;</span>
<a name="l00033"></a>00033 
<a name="l00034"></a>00034 <span class="preprocessor">#include "MessageChannel.h"</span>
<a name="l00035"></a>00035 <span class="preprocessor">#include "Exceptions.h"</span>
<a name="l00036"></a>00036 
<a name="l00037"></a>00037 <span class="keyword">namespace </span>Passenger {
<a name="l00038"></a>00038 
<a name="l00039"></a>00039 <span class="keyword">using namespace </span>boost;
<a name="l00040"></a>00040 <span class="keyword">using namespace </span>oxt;
<a name="l00041"></a>00041 
<a name="l00042"></a>00042 <span class="comment"></span>
<a name="l00043"></a>00043 <span class="comment">/**</span>
<a name="l00044"></a>00044 <span class="comment"> * Wrapper class around a file descriptor integer, for RAII behavior.</span>
<a name="l00045"></a>00045 <span class="comment"> *</span>
<a name="l00046"></a>00046 <span class="comment"> * A FileDescriptor object behaves just like an int, so that you can pass it to</span>
<a name="l00047"></a>00047 <span class="comment"> * system calls such as read(). It performs reference counting. When the last</span>
<a name="l00048"></a>00048 <span class="comment"> * copy of a FileDescriptor has been destroyed, the underlying file descriptor</span>
<a name="l00049"></a>00049 <span class="comment"> * will be automatically closed. In this case, any close() system call errors</span>
<a name="l00050"></a>00050 <span class="comment"> * are silently ignored. If you are interested in whether the close() system</span>
<a name="l00051"></a>00051 <span class="comment"> * call succeeded, then you should call FileDescriptor::close().</span>
<a name="l00052"></a>00052 <span class="comment"> *</span>
<a name="l00053"></a>00053 <span class="comment"> * This class is *not* thread-safe. It is safe to call system calls on the</span>
<a name="l00054"></a>00054 <span class="comment"> * underlying file descriptor from multiple threads, but it's not safe to</span>
<a name="l00055"></a>00055 <span class="comment"> * call FileDescriptor::close() from multiple threads if all those</span>
<a name="l00056"></a>00056 <span class="comment"> * FileDescriptor objects point to the same underlying file descriptor.</span>
<a name="l00057"></a>00057 <span class="comment"> */</span>
<a name="l00058"></a><a class="code" href="classPassenger_1_1FileDescriptor.html">00058</a> <span class="keyword">class </span><a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> {
<a name="l00059"></a>00059 <span class="keyword">private</span>:
<a name="l00060"></a>00060         <span class="keyword">struct </span>SharedData {
<a name="l00061"></a>00061                 <span class="keywordtype">int</span> fd;
<a name="l00062"></a>00062                 
<a name="l00063"></a>00063                 SharedData(<span class="keywordtype">int</span> fd) {
<a name="l00064"></a>00064                         this-&gt;fd = fd;
<a name="l00065"></a>00065                 }
<a name="l00066"></a>00066                 
<a name="l00067"></a>00067                 ~SharedData() {
<a name="l00068"></a>00068                         <span class="keywordflow">if</span> (fd != -1) {
<a name="l00069"></a>00069                                 this_thread::disable_syscall_interruption dsi;
<a name="l00070"></a>00070                                 <a class="code" href="classPassenger_1_1FileDescriptor.html#d7c1a99531181878cbab74a7400c5432" title="Close the underlying file descriptor.">syscalls::close</a>(fd);
<a name="l00071"></a>00071                         }
<a name="l00072"></a>00072                 }
<a name="l00073"></a>00073                 
<a name="l00074"></a>00074                 <span class="keywordtype">void</span> <a class="code" href="classPassenger_1_1FileDescriptor.html#d7c1a99531181878cbab74a7400c5432" title="Close the underlying file descriptor.">close</a>() {
<a name="l00075"></a>00075                         <span class="keywordflow">if</span> (fd != -1) {
<a name="l00076"></a>00076                                 this_thread::disable_syscall_interruption dsi;
<a name="l00077"></a>00077                                 <span class="keywordtype">int</span> theFd = fd;
<a name="l00078"></a>00078                                 fd = -1;
<a name="l00079"></a>00079                                 <span class="keywordflow">if</span> (syscalls::close(theFd) == -1 &amp;&amp; errno != ENOTCONN) {
<a name="l00080"></a>00080                                         <span class="keywordtype">int</span> e = errno;
<a name="l00081"></a>00081                                         <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a>(<span class="stringliteral">"Cannot close file descriptor"</span>, e);
<a name="l00082"></a>00082                                 }
<a name="l00083"></a>00083                         }
<a name="l00084"></a>00084                 }
<a name="l00085"></a>00085         };
<a name="l00086"></a>00086 <span class="comment"></span>
<a name="l00087"></a>00087 <span class="comment">        /** Shared pointer for reference counting on this file descriptor */</span>
<a name="l00088"></a>00088         shared_ptr&lt;SharedData&gt; data;
<a name="l00089"></a>00089 
<a name="l00090"></a>00090 <span class="keyword">public</span>:<span class="comment"></span>
<a name="l00091"></a>00091 <span class="comment">        /** </span>
<a name="l00092"></a>00092 <span class="comment">         * Creates a new empty FileDescriptor instance that has no underlying</span>
<a name="l00093"></a>00093 <span class="comment">         * file descriptor.</span>
<a name="l00094"></a>00094 <span class="comment">         *</span>
<a name="l00095"></a>00095 <span class="comment">         * @post *this == -1</span>
<a name="l00096"></a>00096 <span class="comment">         */</span>
<a name="l00097"></a><a class="code" href="classPassenger_1_1FileDescriptor.html#05d83ad9cc77c493a3ce4f2b51d44559">00097</a>         <a class="code" href="classPassenger_1_1FileDescriptor.html#05d83ad9cc77c493a3ce4f2b51d44559" title="Creates a new empty FileDescriptor instance that has no underlying file descriptor...">FileDescriptor</a>() { }
<a name="l00098"></a>00098         <span class="comment"></span>
<a name="l00099"></a>00099 <span class="comment">        /**</span>
<a name="l00100"></a>00100 <span class="comment">         * Creates a new FileDescriptor instance with the given fd as a handle.</span>
<a name="l00101"></a>00101 <span class="comment">         *</span>
<a name="l00102"></a>00102 <span class="comment">         * @post *this == fd</span>
<a name="l00103"></a>00103 <span class="comment">         */</span>
<a name="l00104"></a><a class="code" href="classPassenger_1_1FileDescriptor.html#62c05b925cb1cf964169cef7635650ee">00104</a>         <a class="code" href="classPassenger_1_1FileDescriptor.html#62c05b925cb1cf964169cef7635650ee" title="Creates a new FileDescriptor instance with the given fd as a handle.">FileDescriptor</a>(<span class="keywordtype">int</span> fd) {
<a name="l00105"></a>00105                 data.reset(<span class="keyword">new</span> SharedData(fd));
<a name="l00106"></a>00106         }
<a name="l00107"></a>00107         <span class="comment"></span>
<a name="l00108"></a>00108 <span class="comment">        /**</span>
<a name="l00109"></a>00109 <span class="comment">         * Close the underlying file descriptor. If it was already closed, then</span>
<a name="l00110"></a>00110 <span class="comment">         * nothing will happen.</span>
<a name="l00111"></a>00111 <span class="comment">         *</span>
<a name="l00112"></a>00112 <span class="comment">         * @throws SystemException Something went wrong while closing</span>
<a name="l00113"></a>00113 <span class="comment">         *                         the file descriptor.</span>
<a name="l00114"></a>00114 <span class="comment">         * @post *this == -1</span>
<a name="l00115"></a>00115 <span class="comment">         */</span>
<a name="l00116"></a><a class="code" href="classPassenger_1_1FileDescriptor.html#d7c1a99531181878cbab74a7400c5432">00116</a>         <span class="keywordtype">void</span> <a class="code" href="classPassenger_1_1FileDescriptor.html#d7c1a99531181878cbab74a7400c5432" title="Close the underlying file descriptor.">close</a>() {
<a name="l00117"></a>00117                 <span class="keywordflow">if</span> (data != NULL) {
<a name="l00118"></a>00118                         data-&gt;close();
<a name="l00119"></a>00119                 }
<a name="l00120"></a>00120         }
<a name="l00121"></a>00121         <span class="comment"></span>
<a name="l00122"></a>00122 <span class="comment">        /**</span>
<a name="l00123"></a>00123 <span class="comment">         * Overloads the integer cast operator so that it will return the underlying</span>
<a name="l00124"></a>00124 <span class="comment">         * file descriptor handle as an integer.</span>
<a name="l00125"></a>00125 <span class="comment">         *</span>
<a name="l00126"></a>00126 <span class="comment">         * Returns -1 if FileDescriptor::close() was called.</span>
<a name="l00127"></a>00127 <span class="comment">         */</span>
<a name="l00128"></a><a class="code" href="classPassenger_1_1FileDescriptor.html#c335c2a97f3a6c8e3920f573b0d04081">00128</a>         <a class="code" href="classPassenger_1_1FileDescriptor.html#c335c2a97f3a6c8e3920f573b0d04081" title="Overloads the integer cast operator so that it will return the underlying file descriptor...">operator int </a>()<span class="keyword"> const </span>{
<a name="l00129"></a>00129                 <span class="keywordflow">if</span> (data == NULL) {
<a name="l00130"></a>00130                         <span class="keywordflow">return</span> -1;
<a name="l00131"></a>00131                 } <span class="keywordflow">else</span> {
<a name="l00132"></a>00132                         <span class="keywordflow">return</span> data-&gt;fd;
<a name="l00133"></a>00133                 }
<a name="l00134"></a>00134         }
<a name="l00135"></a>00135 };
<a name="l00136"></a>00136 <span class="comment"></span>
<a name="l00137"></a>00137 <span class="comment">/**</span>
<a name="l00138"></a>00138 <span class="comment"> * A synchronization mechanism that's implemented with file descriptors,</span>
<a name="l00139"></a>00139 <span class="comment"> * and as such can be used in combination with select() and friends.</span>
<a name="l00140"></a>00140 <span class="comment"> *</span>
<a name="l00141"></a>00141 <span class="comment"> * One can wait for an event on an EventFd by select()ing it on read events.</span>
<a name="l00142"></a>00142 <span class="comment"> * Another thread can signal the EventFd by calling notify().</span>
<a name="l00143"></a>00143 <span class="comment"> */</span>
<a name="l00144"></a><a class="code" href="classPassenger_1_1EventFd.html">00144</a> <span class="keyword">class </span><a class="code" href="classPassenger_1_1EventFd.html" title="A synchronization mechanism that&amp;#39;s implemented with file descriptors, and as...">EventFd</a> {
<a name="l00145"></a>00145 <span class="keyword">private</span>:
<a name="l00146"></a>00146         <span class="keywordtype">int</span> reader;
<a name="l00147"></a>00147         <span class="keywordtype">int</span> writer;
<a name="l00148"></a>00148         
<a name="l00149"></a>00149 <span class="keyword">public</span>:
<a name="l00150"></a>00150         <a class="code" href="classPassenger_1_1EventFd.html" title="A synchronization mechanism that&amp;#39;s implemented with file descriptors, and as...">EventFd</a>() {
<a name="l00151"></a>00151                 <span class="keywordtype">int</span> fds[2];
<a name="l00152"></a>00152                 
<a name="l00153"></a>00153                 <span class="keywordflow">if</span> (syscalls::pipe(fds) == -1) {
<a name="l00154"></a>00154                         <span class="keywordtype">int</span> e = errno;
<a name="l00155"></a>00155                         <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a>(<span class="stringliteral">"Cannot create a pipe"</span>, e);
<a name="l00156"></a>00156                 }
<a name="l00157"></a>00157                 reader = fds[0];
<a name="l00158"></a>00158                 writer = fds[1];
<a name="l00159"></a>00159         }
<a name="l00160"></a>00160         
<a name="l00161"></a>00161         ~<a class="code" href="classPassenger_1_1EventFd.html" title="A synchronization mechanism that&amp;#39;s implemented with file descriptors, and as...">EventFd</a>() {
<a name="l00162"></a>00162                 this_thread::disable_syscall_interruption dsi;
<a name="l00163"></a>00163                 syscalls::close(reader);
<a name="l00164"></a>00164                 syscalls::close(writer);
<a name="l00165"></a>00165         }
<a name="l00166"></a>00166         
<a name="l00167"></a>00167         <span class="keywordtype">void</span> notify() {
<a name="l00168"></a>00168                 <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a>(writer).writeRaw(<span class="stringliteral">"x"</span>, 1);
<a name="l00169"></a>00169         }
<a name="l00170"></a>00170         
<a name="l00171"></a>00171         <span class="keywordtype">int</span> fd()<span class="keyword"> const </span>{
<a name="l00172"></a>00172                 <span class="keywordflow">return</span> reader;
<a name="l00173"></a>00173         }
<a name="l00174"></a>00174 };
<a name="l00175"></a>00175 
<a name="l00176"></a>00176 } <span class="comment">// namespace Passenger</span>
<a name="l00177"></a>00177 
<a name="l00178"></a>00178 <span class="preprocessor">#endif </span><span class="comment">/* _PASSENGER_FILE_DESCRIPTOR_H_ */</span>
</pre></div></div>
<hr size="1"><address style="text-align: right;"><small>Generated on Sun Feb 21 12:22:45 2010 for Passenger by&nbsp;
<a href="http://www.doxygen.org/index.html">
<img src="doxygen.png" alt="doxygen" align="middle" border="0"></a> 1.5.8 </small></address>
</body>
</html>