diff options
Diffstat (limited to 'linden/indra/llcommon/llapp.h')
-rw-r--r-- | linden/indra/llcommon/llapp.h | 279 |
1 files changed, 279 insertions, 0 deletions
diff --git a/linden/indra/llcommon/llapp.h b/linden/indra/llcommon/llapp.h new file mode 100644 index 0000000..aef5cde --- /dev/null +++ b/linden/indra/llcommon/llapp.h | |||
@@ -0,0 +1,279 @@ | |||
1 | /** | ||
2 | * @file llapp.h | ||
3 | * @brief Declaration of the LLApp class. | ||
4 | * | ||
5 | * Copyright (c) 2003-2007, Linden Research, Inc. | ||
6 | * | ||
7 | * The source code in this file ("Source Code") is provided by Linden Lab | ||
8 | * to you under the terms of the GNU General Public License, version 2.0 | ||
9 | * ("GPL"), unless you have obtained a separate licensing agreement | ||
10 | * ("Other License"), formally executed by you and Linden Lab. Terms of | ||
11 | * the GPL can be found in doc/GPL-license.txt in this distribution, or | ||
12 | * online at http://secondlife.com/developers/opensource/gplv2 | ||
13 | * | ||
14 | * There are special exceptions to the terms and conditions of the GPL as | ||
15 | * it is applied to this Source Code. View the full text of the exception | ||
16 | * in the file doc/FLOSS-exception.txt in this software distribution, or | ||
17 | * online at http://secondlife.com/developers/opensource/flossexception | ||
18 | * | ||
19 | * By copying, modifying or distributing this software, you acknowledge | ||
20 | * that you have read and understood your obligations described above, | ||
21 | * and agree to abide by those obligations. | ||
22 | * | ||
23 | * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO | ||
24 | * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY, | ||
25 | * COMPLETENESS OR PERFORMANCE. | ||
26 | */ | ||
27 | |||
28 | #ifndef LL_LLAPP_H | ||
29 | #define LL_LLAPP_H | ||
30 | |||
31 | #include <map> | ||
32 | #include "llapr.h" | ||
33 | #include "llrun.h" | ||
34 | #include "llsd.h" | ||
35 | |||
36 | // Forward declarations | ||
37 | class LLErrorThread; | ||
38 | class LLApp; | ||
39 | |||
40 | |||
41 | typedef void (*LLAppErrorHandler)(); | ||
42 | typedef void (*LLAppChildCallback)(int pid, bool exited, int status); | ||
43 | |||
44 | #if !LL_WINDOWS | ||
45 | extern const S32 LL_SMACKDOWN_SIGNAL; | ||
46 | |||
47 | // Clear all of the signal handlers (which we want to do for the child process when we fork | ||
48 | void clear_signals(); | ||
49 | |||
50 | class LLChildInfo | ||
51 | { | ||
52 | public: | ||
53 | LLChildInfo() : mGotSigChild(FALSE), mCallback(NULL) {} | ||
54 | BOOL mGotSigChild; | ||
55 | LLAppChildCallback mCallback; | ||
56 | }; | ||
57 | #endif | ||
58 | |||
59 | class LLApp | ||
60 | { | ||
61 | friend class LLErrorThread; | ||
62 | public: | ||
63 | typedef enum e_app_status | ||
64 | { | ||
65 | APP_STATUS_RUNNING, // The application is currently running - the default status | ||
66 | APP_STATUS_QUITTING, // The application is currently quitting - threads should listen for this and clean up | ||
67 | APP_STATUS_STOPPED, // The application is no longer running - tells the error thread it can exit | ||
68 | APP_STATUS_ERROR // The application had a fatal error occur - tells the error thread to run | ||
69 | } EAppStatus; | ||
70 | |||
71 | |||
72 | LLApp(); | ||
73 | virtual ~LLApp(); | ||
74 | |||
75 | /** | ||
76 | * @brief Return the static app instance if one was created. | ||
77 | */ | ||
78 | static LLApp* instance(); | ||
79 | |||
80 | /** @name Runtime options */ | ||
81 | //@{ | ||
82 | /** | ||
83 | * @brief Enumeration to specify option priorities in highest to | ||
84 | * lowest order. | ||
85 | */ | ||
86 | enum OptionPriority | ||
87 | { | ||
88 | PRIORITY_RUNTIME_OVERRIDE, | ||
89 | PRIORITY_COMMAND_LINE, | ||
90 | PRIORITY_SPECIFIC_CONFIGURATION, | ||
91 | PRIORITY_GENERAL_CONFIGURATION, | ||
92 | PRIORITY_DEFAULT, | ||
93 | PRIORITY_COUNT | ||
94 | }; | ||
95 | |||
96 | /** | ||
97 | * @brief Get the application option at the highest priority. | ||
98 | * | ||
99 | * If the return value is undefined, the option does not exist. | ||
100 | * @param name The name of the option. | ||
101 | * @return Returns the option data. | ||
102 | */ | ||
103 | LLSD getOption(const std::string& name) const; | ||
104 | |||
105 | /** | ||
106 | * @brief Parse command line options and insert them into | ||
107 | * application command line options. | ||
108 | * | ||
109 | * The name inserted into the option will have leading option | ||
110 | * identifiers (a minus or double minus) stripped. All options | ||
111 | * with values will be stored as a string, while all options | ||
112 | * without values will be stored as true. | ||
113 | * @param argc The argc passed into main(). | ||
114 | * @param argv The argv passed into main(). | ||
115 | * @return Returns true if the parse succeeded. | ||
116 | */ | ||
117 | bool parseCommandOptions(int argc, char** argv); | ||
118 | |||
119 | /** | ||
120 | * @brief Set the options at the specified priority. | ||
121 | * | ||
122 | * This function completely replaces the options at the priority | ||
123 | * level with the data specified. This function will make sure | ||
124 | * level and data might be valid before doing the replace. | ||
125 | * @param level The priority level of the data. | ||
126 | * @param data The data to set. | ||
127 | * @return Returns true if the option was set. | ||
128 | */ | ||
129 | bool setOptionData(OptionPriority level, LLSD data); | ||
130 | |||
131 | /** | ||
132 | * @brief Get the option data at the specified priority. | ||
133 | * | ||
134 | * This method is probably not so useful except when merging | ||
135 | * information. | ||
136 | * @param level The priority level of the data. | ||
137 | * @return Returns The data (if any) at the level priority. | ||
138 | */ | ||
139 | LLSD getOptionData(OptionPriority level); | ||
140 | //@} | ||
141 | |||
142 | |||
143 | |||
144 | // | ||
145 | // Main application logic | ||
146 | // | ||
147 | virtual bool init() = 0; // Override to do application initialization | ||
148 | |||
149 | // | ||
150 | // cleanup() | ||
151 | // | ||
152 | // It's currently assumed that the cleanup() method will only get | ||
153 | // called from the main thread or the error handling thread, as it will | ||
154 | // likely do thread shutdown, among other things. | ||
155 | // | ||
156 | virtual bool cleanup() = 0; // Override to do application cleanup | ||
157 | |||
158 | // | ||
159 | // mainLoop() | ||
160 | // | ||
161 | // Runs the application main loop. It's assumed that when you exit | ||
162 | // this method, the application is in one of the cleanup states, either QUITTING or ERROR | ||
163 | // | ||
164 | virtual bool mainLoop() = 0; // Override for the application main loop. Needs to at least gracefully notice the QUITTING state and exit. | ||
165 | |||
166 | |||
167 | // | ||
168 | // Application status | ||
169 | // | ||
170 | static void setQuitting(); // Set status to QUITTING, the app is now shutting down | ||
171 | static void setStopped(); // Set status to STOPPED, the app is done running and should exit | ||
172 | static void setError(); // Set status to ERROR, the error handler should run | ||
173 | static bool isStopped(); | ||
174 | static bool isRunning(); | ||
175 | static bool isQuitting(); | ||
176 | static bool isError(); | ||
177 | static bool isExiting(); // Either quitting or error (app is exiting, cleanly or not) | ||
178 | #if !LL_WINDOWS | ||
179 | static U32 getSigChildCount(); | ||
180 | static void incSigChildCount(); | ||
181 | #endif | ||
182 | static int getPid(); | ||
183 | |||
184 | // | ||
185 | // Error handling methods | ||
186 | // | ||
187 | void setErrorHandler(LLAppErrorHandler handler); | ||
188 | |||
189 | #if !LL_WINDOWS | ||
190 | // | ||
191 | // Child process handling (Unix only for now) | ||
192 | // | ||
193 | // Set a callback to be run on exit of a child process | ||
194 | // WARNING! This callback is run from the signal handler due to the extreme crappiness of | ||
195 | // Linux threading requiring waitpid() to be called from the thread that spawned the process. | ||
196 | // At some point I will make this more behaved, but I'm not going to fix this right now - djs | ||
197 | void setChildCallback(pid_t pid, LLAppChildCallback callback); | ||
198 | |||
199 | // The child callback to run if no specific handler is set | ||
200 | void setDefaultChildCallback(LLAppChildCallback callback); | ||
201 | |||
202 | // Fork and do the proper signal handling/error handling mojo | ||
203 | // WARNING: You need to make sure your signal handling callback is correct after | ||
204 | // you fork, because not all threads are duplicated when you fork! | ||
205 | pid_t fork(); | ||
206 | #endif | ||
207 | |||
208 | /** | ||
209 | * @brief Get a reference to the application runner | ||
210 | * | ||
211 | * Please use the runner with caution. Since the Runner usage | ||
212 | * pattern is not yet clear, this method just gives access to it | ||
213 | * to add and remove runnables. | ||
214 | * @return Returns the application runner. Do not save the | ||
215 | * pointer past the caller's stack frame. | ||
216 | */ | ||
217 | LLRunner& getRunner() { return mRunner; } | ||
218 | |||
219 | public: | ||
220 | typedef std::map<std::string, std::string> string_map; | ||
221 | string_map mOptionMap; // Contains all command-line options and arguments in a map | ||
222 | |||
223 | protected: | ||
224 | |||
225 | static void setStatus(EAppStatus status); // Use this to change the application status. | ||
226 | static EAppStatus sStatus; // Reflects current application status | ||
227 | static BOOL sErrorThreadRunning; // Set while the error thread is running | ||
228 | |||
229 | #if !LL_WINDOWS | ||
230 | static LLAtomicU32* sSigChildCount; // Number of SIGCHLDs received. | ||
231 | typedef std::map<pid_t, LLChildInfo> child_map; // Map key is a PID | ||
232 | static child_map sChildMap; | ||
233 | static LLAppChildCallback sDefaultChildCallback; | ||
234 | #endif | ||
235 | |||
236 | /** | ||
237 | * @brief This method is called once a frame to do once a frame tasks. | ||
238 | */ | ||
239 | void stepFrame(); | ||
240 | |||
241 | private: | ||
242 | void setupErrorHandling(); // Do platform-specific error-handling setup (signals, structured exceptions) | ||
243 | |||
244 | static void runErrorHandler(); | ||
245 | |||
246 | // *NOTE: On Windows, we need a routine to reset the structured | ||
247 | // exception handler when some evil driver has taken it over for | ||
248 | // their own purposes | ||
249 | typedef int(*signal_handler_func)(int signum); | ||
250 | static LLAppErrorHandler sErrorHandler; | ||
251 | |||
252 | // Default application threads | ||
253 | LLErrorThread* mThreadErrorp; // Waits for app to go to status ERROR, then runs the error callback | ||
254 | |||
255 | // This is the application level runnable scheduler. | ||
256 | LLRunner mRunner; | ||
257 | |||
258 | /** @name Runtime option implementation */ | ||
259 | //@{ | ||
260 | |||
261 | // The application options. | ||
262 | LLSD mOptions; | ||
263 | |||
264 | //@} | ||
265 | |||
266 | private: | ||
267 | // the static application instance if it was created. | ||
268 | static LLApp* sApplication; | ||
269 | |||
270 | |||
271 | #if !LL_WINDOWS | ||
272 | friend void default_unix_signal_handler(int signum, siginfo_t *info, void *); | ||
273 | #endif | ||
274 | |||
275 | public: | ||
276 | static BOOL sLogInSignal; | ||
277 | }; | ||
278 | |||
279 | #endif // LL_LLAPP_H | ||