aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/linden/indra/llmessage/lliopipe.h
diff options
context:
space:
mode:
Diffstat (limited to 'linden/indra/llmessage/lliopipe.h')
-rw-r--r--linden/indra/llmessage/lliopipe.h310
1 files changed, 310 insertions, 0 deletions
diff --git a/linden/indra/llmessage/lliopipe.h b/linden/indra/llmessage/lliopipe.h
new file mode 100644
index 0000000..f419703
--- /dev/null
+++ b/linden/indra/llmessage/lliopipe.h
@@ -0,0 +1,310 @@
1/**
2 * @file lliopipe.h
3 * @author Phoenix
4 * @date 2004-11-18
5 * @brief Declaration of base IO class
6 *
7 * Copyright (c) 2004-2007, Linden Research, Inc.
8 *
9 * The source code in this file ("Source Code") is provided by Linden Lab
10 * to you under the terms of the GNU General Public License, version 2.0
11 * ("GPL"), unless you have obtained a separate licensing agreement
12 * ("Other License"), formally executed by you and Linden Lab. Terms of
13 * the GPL can be found in doc/GPL-license.txt in this distribution, or
14 * online at http://secondlife.com/developers/opensource/gplv2
15 *
16 * There are special exceptions to the terms and conditions of the GPL as
17 * it is applied to this Source Code. View the full text of the exception
18 * in the file doc/FLOSS-exception.txt in this software distribution, or
19 * online at http://secondlife.com/developers/opensource/flossexception
20 *
21 * By copying, modifying or distributing this software, you acknowledge
22 * that you have read and understood your obligations described above,
23 * and agree to abide by those obligations.
24 *
25 * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO
26 * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY,
27 * COMPLETENESS OR PERFORMANCE.
28 */
29
30#ifndef LL_LLIOPIPE_H
31#define LL_LLIOPIPE_H
32
33#include <boost/intrusive_ptr.hpp>
34#include <boost/shared_ptr.hpp>
35#include "apr-1/apr_poll.h"
36
37#include "llsd.h"
38
39class LLIOPipe;
40class LLPumpIO;
41class LLBufferArray;
42class LLChannelDescriptors;
43
44// Debugging schmutz for deadlocks
45#define LL_DEBUG_PUMPS
46#ifdef LL_DEBUG_PUMPS
47void pump_debug(const char *file, S32 line);
48#define PUMP_DEBUG pump_debug(__FILE__, __LINE__);
49#define END_PUMP_DEBUG pump_debug("none", 0);
50#endif
51
52
53/**
54 * intrusive pointer support
55 */
56namespace boost
57{
58 void intrusive_ptr_add_ref(LLIOPipe* p);
59 void intrusive_ptr_release(LLIOPipe* p);
60};
61
62/**
63 * @class LLIOPipe
64 * @brief This class is an abstract base class for data processing units
65 * @see LLPumpIO
66 *
67 * The LLIOPipe is a base class for implementing the basic non-blocking
68 * processing of data subsystem in our system.
69 *
70 * Implementations of this class should behave like a stateful or
71 * stateless signal processor. Each call to <code>process()</code>
72 * hands the pipe implementation a buffer and a set of channels in the
73 * buffer to process, and the pipe returns the status of the
74 * operation. This is an abstract base class and developer created
75 * concrete implementations provide block or stream based processing
76 * of data to implement a particular protocol.
77 */
78class LLIOPipe
79{
80public:
81 /**
82 * @brief I have decided that IO objects should have a reference
83 * count. In general, you can pass bald LLIOPipe pointers around
84 * as you need, but if you need to maintain a reference to one,
85 * you need to hold a ptr_t.
86 */
87 typedef boost::intrusive_ptr<LLIOPipe> ptr_t;
88
89 /**
90 * @brief Scattered memory container.
91 */
92 typedef boost::shared_ptr<LLBufferArray> buffer_ptr_t;
93
94 /**
95 * @brief Enumeration for IO return codes
96 *
97 * A status code a positive integer value is considered a success,
98 * but may indicate special handling for future calls, for
99 * example, issuing a STATUS_STOP to an LLIOSocketReader instance
100 * will tell the instance to stop reading the socket. A status
101 * code with a negative value means that a problem has been
102 * encountered which will require further action on the caller or
103 * a developer to correct. Some mechanisms, such as the LLPumpIO
104 * may depend on this definition of success and failure.
105 */
106 enum EStatus
107 {
108 // Processing occurred normally, future calls will be accepted.
109 STATUS_OK = 0,
110
111 // Processing occured normally, but stop unsolicited calls to
112 // process.
113 STATUS_STOP = 1,
114
115 // This pipe is done with the processing. Future calls to
116 // process will be accepted as long as new data is available.
117 STATUS_DONE = 2,
118
119 // This pipe is requesting that it become the head in a process.
120 STATUS_BREAK = 3,
121
122 // This pipe is requesting that it become the head in a process.
123 STATUS_NEED_PROCESS = 4,
124
125 // Keep track of the highest number of success codes here.
126 STATUS_SUCCESS_COUNT = 5,
127
128 // A generic error code.
129 STATUS_ERROR = -1,
130
131 // This method has not yet been implemented. This usually
132 // indicates the programmer working on the pipe is not yet
133 // done.
134 STATUS_NOT_IMPLEMENTED = -2,
135
136 // This indicates that a pipe precondition was not met. For
137 // example, many pipes require an element to appear after them
138 // in a chain (ie, mNext is not null) and will return this in
139 // response to method calls. To recover from this, it will
140 // require the caller to adjust the pipe state or may require
141 // a dev to adjust the code to satisfy the preconditions.
142 STATUS_PRECONDITION_NOT_MET = -3,
143
144 // This means we could not connect to a remote host.
145 STATUS_NO_CONNECTION = -4,
146
147 // This means we could not connect to a remote host.
148 STATUS_EXPIRED = -5,
149
150 // Keep track of the count of codes here.
151 STATUS_ERROR_COUNT = 5,
152 };
153
154 /**
155 * @brief Helper function to check status.
156 *
157 * When writing code to check status codes, if you do not
158 * specifically check a particular value, use this method for
159 * checking an error condition.
160 * @param status The status to check.
161 * @return Returns true if the code indicates an error occurred.
162 */
163 inline static bool isError(EStatus status)
164 {
165 return ((S32)status < 0);
166 }
167
168 /**
169 * @brief Helper function to check status.
170 *
171 * When writing code to check status codes, if you do not
172 * specifically check a particular value, use this method for
173 * checking an error condition.
174 * @param status The status to check.
175 * @return Returns true if the code indicates no error was generated.
176 */
177 inline static bool isSuccess(EStatus status)
178 {
179 return ((S32)status >= 0);
180 }
181
182 /**
183 * @brief Helper function to turn status into a string.
184 *
185 * @param status The status to check.
186 * @return Returns the name of the status code or empty string on failure.
187 */
188 static std::string lookupStatusString(EStatus status);
189
190 /**
191 * @brief Process the data in buffer.
192 *
193 * @param data The data processed
194 * @param eos True if this function call is the last because end of stream.
195 * @param pump The pump which is calling process. May be NULL.
196 * @param context Shared meta-data for the process.
197 * @return Returns a status code from the operation.
198 */
199 EStatus process(
200 const LLChannelDescriptors& channels,
201 buffer_ptr_t& buffer,
202 bool& eos,
203 LLSD& context,
204 LLPumpIO* pump);
205
206 /**
207 * @brief Give this pipe a chance to handle a generated error
208 *
209 * If this pipe is in a chain being processed by a pump, and one
210 * of the pipes generates an error, the pump will rewind through
211 * the chain to see if any of the links can handle the error. For
212 * example, if a connection is refused in a socket connection, the
213 * socket client can try to find a new destination host. Return an
214 * error code if this pipe does not handle the error passed in.
215 * @param status The status code for the error
216 * @param pump The pump which was calling process before the error
217 * was generated.
218 * @return Returns a status code from the operation. Returns an
219 * error code if the error passed in was not handled. Returns
220 * STATUS_OK to indicate the error has been handled.
221 */
222 virtual EStatus handleError(EStatus status, LLPumpIO* pump);
223
224 /**
225 * @brief Base Destructor - do not call <code>delete</code> directly.
226 */
227 virtual ~LLIOPipe();
228
229protected:
230 /**
231 * @brief Base Constructor.
232 */
233 LLIOPipe();
234
235 /**
236 * @brief Process the data in buffer
237 */
238 virtual EStatus process_impl(
239 const LLChannelDescriptors& channels,
240 buffer_ptr_t& buffer,
241 bool& eos,
242 LLSD& context,
243 LLPumpIO* pump) = 0;
244
245private:
246 friend void boost::intrusive_ptr_add_ref(LLIOPipe* p);
247 friend void boost::intrusive_ptr_release(LLIOPipe* p);
248 U32 mReferenceCount;
249};
250
251namespace boost
252{
253 inline void intrusive_ptr_add_ref(LLIOPipe* p)
254 {
255 ++p->mReferenceCount;
256 }
257 inline void intrusive_ptr_release(LLIOPipe* p)
258 {
259 if(0 == --p->mReferenceCount)
260 {
261 delete p;
262 }
263 }
264};
265
266
267#if 0
268/**
269 * @class LLIOBoiler
270 * @brief This class helps construct new LLIOPipe specializations
271 * @see LLIOPipe
272 *
273 * THOROUGH_DESCRIPTION
274 */
275class LLIOBoiler : public LLIOPipe
276{
277public:
278 LLIOBoiler();
279 virtual ~LLIOBoiler();
280
281protected:
282 /* @name LLIOPipe virtual implementations
283 */
284 //@{
285 /**
286 * @brief Process the data in buffer
287 */
288 virtual EStatus process_impl(
289 const LLChannelDescriptors& channels,
290 buffer_ptr_t& buffer,
291 bool& eos,
292 LLSD& context,
293 LLPumpIO* pump);
294 //@}
295};
296
297// virtual
298LLIOPipe::EStatus process_impl(
299 const LLChannelDescriptors& channels,
300 buffer_ptr_t& buffer,
301 bool& eos,
302 LLSD& context,
303 LLPumpIO* pump)
304{
305 return STATUS_NOT_IMPLEMENTED;
306}
307
308#endif // #if 0 - use this block as a boilerplate
309
310#endif // LL_LLIOPIPE_H